我们经常需将一些C/C++源码编译成本地二进制,直接在android的linux内核上运行,这是就需要进行交叉编译。由于Android的运行环境核普通Linux又区别,所以常规方式针对ARM进行交叉编译得到的二进制,无法直接运行在Android环境上。我们需要机遇Android环境进行交叉编译,这时有两种方式:一种是编写Android.mk,在安卓源码结构下依靠安卓的编译系统搞定;另一种方式是脱离android源码体系,利用独立的android交叉编译工具链,并适当修改目标代码的Makefile(必要时还要修改源码)完成交叉编译。第一种方式适用于编译目标源码结构简单的情况;当目标源码的Makefile组织非常复杂的时候,第二种方式则是更好的选择。本文介绍第二种方式,即搭建Android独立交叉编译环境,编译出可以在Android(默认基于ARM64)上运行的二进制(ELF)程序。
1、NDK下载
https://developer.android.google.cn/ndk/downloads/index.html
本文以宿主机系统MacOS为例,所以下载MacOSX版本的ndk包
2、利用NDK安装android交叉编译工具链
解压NDK后,进入buildtools目录
NickPro:android-ndk-r14b nickyang$ cd build/tools/
NickPro:tools nickyang$ pwd
/Users/nickyang/android-ndk-r14b/build/tools
NDK提供了两个脚步用于一键化构建gcc工具链:
make-standalone-toolchain.sh
make_standalone_toolchain.py
我们使用py版本的,按照官方说明,sh版本将被淘汰
./make_standalone_toolchain.py --arch arm64 --api 24 --unified-headers --install-dir ~/android-build
参数解释:
—arch : 交叉编译目标平台
--unified-header: 使用libc头文件=
—api: Android系统版本
--install-dir:生成的交叉编译工具链输出位置,可自行设置
3、编译方法
通常类Unix系统下,C/C++源码都是通过Makefile组织的(或者通过一个configure脚本生成Makefile),所以我们要阅读其Makefile(或configure脚步),将其编译器指定为我们的android交叉编译器,然后在编译过程中会遇到一下函数为定义等错误,这是因为andorid的bionic C库和纯Linux的C库差别导致的,比如某些函数在bionic中不支持,对这些函数,将其干掉或者替换,或者自己重写,等等。下面总结下通常要做的几点:
1)查看目标源码的Makefile文件,将交叉编译器指定为咱们安装的交叉编译器
通常是修改CC环境变量
CC= /path_to_your_cross_compiler/you_cross_compiler_xxx_gcc
2)CFLAGS加上“-pie -fPIE”选项,否则报错:
error: only position independent executables (PIE) are supported
3) 一下编译参数不支持,要去掉
#PTHREAD_LIBS= -lpthread -lrt
4、小试牛刀
新建一个hello.c文件:
#include <stdio.h>
int main(void)
{
printf("Hello, this is a program compiled by ndk-cross-compiler!\n");
}
编译:
/Users/nickyang/android-build/bin/aarch64-linux-android-gcc -pie -fPIE hello.c -o hello
推送到android设备并执行(默认是ARM64架构):
NickPro:tmp nickyang$ adb push hello /data
hello: 1 file pushed. 0.1 MB/s (7808 bytes in 0.112s)
NickPro:tmp nickyang$ adb shell
generic_arm64:/ # cd data
generic_arm64:/data # ls -l | grep hello
-rwxrwxrwx 1 root root 7808 2018-02-18 16:20 hello
generic_arm64:/data # ./hello
Hello, this is a program compiled by ndk-cross-compiler!
10|generic_arm64:/data #
执行成功,顺利输出打印!