最近在基于Android做IPV6方面的开发和测试,需要使用busybox中提供的ping, ping6, nslookup, traceroute6等工具。经过测试发现:网络上介绍交叉编译busybox并放到Android中运行的文章很多,交叉编译好的busybox也可以直接下载到。但是,由于它们都是静态编译busybox并链接在标准的glibc上,导致一些功能在Android中不能正常运行,例如:#busybox ping www.baidu.com
Android并没有采用glibc作为C库,而是采用了Google自己开发的Bionic Libc,它的官方Toolchain也是基于Bionic Libc而并非glibc的。与glibc相比,Bionic Libc有如下一些特点:
我这里要做的就是基于bionic libc来编译busybox,以更好的提供对网络相关工具的支持。上网搜索发现,已经有一些人对busybox的移植做了许多工作,并且工作了自己的成果,比如:
对他们的文章和源码进行了研究与尝试,可以基于cyanogenmod的源码方便的移植busybox到Android,打开ipv6相关功能,并且不需要替换libc.so。下面是我的移植步骤和经验总结:
1. 下载cyanogenmod修改过的busybox源码,并解压到<src>/external/busybox目录
下载地址:http://github.com/CyanogenMod/android_external_busybox
2. 编译busybox并修正编译错误
在Andorid源码根目录中执行: # make -j4 busybox
可能遇到的第一类编译错误如下:
In file included from external/busybox/coreutils/df.c:25: bionic/libc/include/mntent.h:48: error: expected '=', ',', ';', 'asm' or '__attribute__' before 'struct'
解决方法是在出错的文件中,包含头文件stdio.h。例如在df.c中增加
#include <stdio.h>
类似的编译出错的文件可能会有:
coreutile/df.c util-linux/mount.c util-linux/umount.c
可能遇到的第二类编译错误如下:
external/busybox/shell/ash.c:12714: error: 'rlim_t' undeclared (first use in this function)
出错原因是你使用的是早期的Android版本,其中rlim_t没有定义。解决方法是在<src>/bionic/libc/include/sys/resource.h中增加
typedef unsigned long rlim_t;
可以参考Froyo中<src>/bionic/libc/include/sys/resource.h文件。
可能遇到的第三类编译错误如下:
busybox/networking/ntpd.c:1458: undefined reference to `adjtimex'
出错原因是bionic中没有定义adjtimex,stime,swapon,swapoff,sysinfo,getsid等函数。解决方法是修改Android.mk,增加
CYANOGEN_BIONIC:=true
3. 将编译好的busybox拷贝到手机/system/xbin/目录下测试
#adb push busybox /system/xbin/
附:编译好的busybox二进制下载地址 (支持ipv6)