这篇文章记录下如何使用交叉编译,编译出Android平台下的可执行程序。
首先来看看Linux平台下如何使用gcc编译出当下平台的可执行程序。
我们先来准备一个main.c文件
#include<stdio.h>
int main(){
printf("Hello Gcc!");
return 0;
}
上面是一个c语言写的很简单的一段代码,目的是要输出“Hello Gcc!”。
Linux平台下把这个c文件编译成可执行程序,很简单,只需要执行如下命令:
gcc -o main main.c
然后我们执行这个程序,就会看到输出了“Hello Gcc!”
./main
但是这个可执行程序,现在不能再其它平台运行,如果我们想要它在Android平台下运行,该如何做呢?这个时候就需要进行交叉编译了,不能使用Linux平台的编译器了,而要使用Android的编译器,这个编译器我们可以从NDK中获取。
生成Android平台下的可执行文件,我们需要使用NDK,我这里使用的是NDK17,这里提供了NDK下载地址,选择最新的版本或者其他版本都可以,不过各个版本的结构目录稍微不太一样。
下载NDK17:
wget https://dl.google.com/android/repository/android-ndk-r17c-linux-x86_64.zip
完成后解压,在当前目录下生成android-ndk-r17c 文件。
接下来我们设置下编译Android平台可执行程序的编译器的环境变量,这里设置了一个临时变量:
export CC=/root/android-ndk-r17c/toolchains/arm-linux-androideabi-4.9/prebuilt/linux-x86_64/bin/arm-linux-androideabi-gcc
$CC main.c
上面的意思可以理解为在编译Linux平台使用的gcc。
这里我们遇到了一个错误:
main.c:1:19: fatal error: stdio.h: No such file or directory
#include <stdio.h>
^
compilation terminated.
提示我们没有找到这个头文件,我们明明已经添加了头文件,为什么还会报这个错误呢?
其实这个编译的时候还是会去Linux系统中默认的路径去寻找这个头文件,而不是使用的我们NDK中的头文件,那接下来我们指定使用NDK中的头文件。
export AAA="--sysroot=/root/android-ndk-r17c/platforms/android-21/arch-arm -isystem /root/android-ndk-r17c/sysroot/usr/include -isystem /root/android-ndk-r17c/sysroot/usr/include/arm-linux-androideabi"
$CC $AAA -pie main.c -o main
上面的意思是把AAA当做参数传给CC,就是传递给编译器的参数。
注意:-pie不要忘了,没有的话在android中会报pie的错误,编译可执行文件要加上-pie。
我们把这个main文件导出,在shell中查看:
root@x86:/ # cd data/
root@x86:/data # cd local/
root@x86:/data/local # cd tmp/
root@x86:/data/local/tmp # chmod 777 main
root@x86:/data/local/tmp # ./main
Hello Gcc!root@x86:/data/local/tmp #
可以看到成功了在Android平台上输出了“Hello Gcc!”,说明我们使用交叉编译在Linux平台上编译出了Android可用的执行程序!
最后,还有几点需要了解NDK中也提供的交叉编译链的位置。
E:\androidstudio3.4\SDK\ndk-bundle\toolchains
这个toolchains中包含了交叉编译的工具。