零.
Continue my previous avahi article, here, i would port the same example code to Android platform.
Download and unravel Android NDK
for me, the untarred NDK root locates in ~/Android/AndroidNDK/android-ndk-r10e
And install the android tools in the linux system. (yum for fedora, apt-get for ubunu/debian)
sudo apt-get install gcc-arm-linux-androideabi
sudo apt-get install android-tools-adb
I strongly suggest you DO NOT install the others android-tool, headers and libraries via apt-get/yum.
If you need those, just download from Android website and decompress them.
一.
Download and untar the avahi libraries, in here I use version 0.6.31.
Download 2 patches and place them in the untarred avahi folder:
0001-Add-Android-support.patch
0002-Add-uninstalled.pc.in-files.patch
apply the patches.
patch -p1 0001-Add-Android-support.patch
You would encounter a warning here, just ignore it. it is for there is no .git folder inside the avahi folder.
patch -p1 0002-Add-uninstalled.pc.in-files.patch
For the old configuration tools does not support androideabi, you need to upgrade the config.guess and config.sub from here. it is, to overwrite these 2 files inside the avahi root folder.
二.
The struct in6_pktinfo has been added in Android 4.0(2011/10). When the avahi 0.6.31 was released (2012/2), the BioniC (the position of glibC for Android) libraries on most populace's Android phone was not implemented the whole IVP6. Therefore, the structure ha been declare inside avahi. In present(2015/12), I thought there is no one set build-target version before 4.0. It is requisite to modify code to remove the obsolete lines in avahi-core/socket.c , about line 67:
#endif #if(0) #ifdef __BIONIC__ struct in6_pktinfo { struct in6_addr ipi6_addr; int ipi6_ifindex; }; #endif #endif static void mdns_mcast_group_ipv4(struct sockaddr_in *ret_sa) {
The in6_pktinfo has been declared in NDKfolder/platforms/android-version/arch/usr/include/linux/ipv6.h, to comment out it to avoid compilation error.
Optional:
if your application would be integrated web-browser and avahi disovering functions, you should modify the code avahi-core/netlink.c, about line 160:
memset(&addr, 0, sizeof(addr)); addr.nl_family = AF_NETLINK; addr.nl_groups = groups; #if(1) addr.nl_pid = 0; #else addr.nl_pid = getpid(); #endif if (bind(fd, (struct sockaddr *) &addr, sizeof(addr)) < 0) {
It is, the add.nl_pid value would not be conflicted with browser's. More detail of this change is in here.
三.
The Avahi configure arguments for me be:browser
./configure --prefix=$PWD/built --disable-dbus --disable-mono --disable-monodoc --disable-qt4 --disable-qt3 --disable-glib --disable-gobject --disable-gtk --disable-gtk3 --disable-gdbm --disable-dbm --disable-python --disable-pygtk -disable-libdaemon --disable-nls --disable-xmltoman --with-xml=none --disable-core-docs --disable-doxygen-doc CPPFLAGS="-I/home/gaiger/Android/AndroidNDK/android-ndk-r10e/platforms/android-14/arch-arm/usr/include -mcpu=arm926ej-s -mfloat-abi=soft" LDFLAGS=-L/home/gaiger/Android/AndroidNDK/android-ndk-r10e/platforms/android-14/arch-arm/usr/lib --host=arm-linux-androideabi --disable-stack-protector --with-distro=debian
NOTE :
Zero:
I disable most functions of Avahi. Because those are useless for the goal of discovering mDNS service and dependent on dbus, gtk, qt and so forth libraries. To building those depended libraries are difficult for Android.
One:
The path, /home/gaiger/Android/AndroidNDK/android-ndk-r10e/platforms/android-14/arch-arm/usr, is where the Android system libraries locates, those are being part of Android NDK.
/home/gaiger/Android/AndroidNDK/android-ndk-r10e : where AndroidNDK root locates.
platforms/android-14/arch-arm : I chose to android 4.0 with arm-architecture as my target system.
it is very natural to add include path (-Iusr_path/include) and linking path(-Iusr_path/lib) as arguments in cross-compilation.
TWO:
-mcpu=arm926ej-s -mfloat-abi=softfp : To avoid crash in some (mostly those are not existed ) cheap Android device CPU without hardware floating computing. Besides, Avahi do not involve in heavy floating point computing, So adapt pure soft-float computing would not lead mDNS discovering sluggish.
arm926ej-s means, assuming the CPU does not support hardware floating but supports Jazelle ( for java VM).
More detail of CPU architecture, you could refer to this article.
THREE :
--with-distro=debian : for me , I use ubuntu. If you use fedora/CentOS, you use set it as fedora.
FOUR:
--disable-stack-protector : there is no libssp (stack smash protector) in current BioniC defaultly. To advert configuring error, one should disable this function. The detail about libssp is in here.
四.
After configuring done, it is very instinctive to press make and make install.
To this step, the avahi libraries for Android has been built, in the avahi_root/built
五.
Create a folder and a sub-folder in avahi root: sandyboxcode/jni
Copy the code sandbox-discover-standalone.c of my last article in folder sandyboxcode/jni.
And Create two files, Application.mk and Android.mk in the same folder ( sandyboxcode/jni).
The content of file Application.mk be:
#APP_ABI := armeabi-v7a APP_ABI := armeabi
The Android.mk file be:
LOCAL_PATH := $(my-dir) ifneq ($(DYNAMIC_LINK),true) DYNAMIC_LINK = false include $(CLEAR_VARS) LOCAL_MODULE := libavahi-core LOCAL_SRC_FILES := ../../built/lib/libavahi-core.a include $(PREBUILT_STATIC_LIBRARY) include $(CLEAR_VARS) LOCAL_MODULE := libavahi-common LOCAL_SRC_FILES := ../../built/lib/libavahi-common.a include $(PREBUILT_STATIC_LIBRARY) else include $(CLEAR_VARS) LOCAL_MODULE := libavahi-core LOCAL_SRC_FILES := ../../built/lib/libavahi-core.so include $(PREBUILT_SHARED_LIBRARY) include $(CLEAR_VARS) LOCAL_MODULE := libavahi-common LOCAL_SRC_FILES := ../../built/lib/libavahi-common.so include $(PREBUILT_SHARED_LIBRARY) endif #if $DYNAMIC_LINK .neq. true include $(CLEAR_VARS) LOCAL_MODULE := sandbox-discover-standalone LOCAL_SRC_FILES := sandbox-discover-standalone.c LOCAL_CFLAGS := -O2 -mcpu=arm926ej-s -mfloat-abi=softfp LOCAL_C_INCLUDES := $(LOCAL_PATH)/../../built/include ifeq ($(DYNAMIC_LINK),false) LOCAL_STATIC_LIBRARIES := libavahi-core libavahi-common else LOCAL_SHARED_LIBRARIES += libavahi-common libavahi-core endif #if $(DYNAMIC_LINK) eq false LOCAL_LDLIBS += -llog include $(BUILD_EXECUTABLE)
Then to the sandbox folder, type :
$your_ndk_path/ndk-build -B V=1
Now you have gotten binary of sandbox-discover-standalone for Android, under the folder libs/armeabi .
六.
Push the built binary to a android device, which might need to be rooted to execute a binary in shell.
For me, I use the development board tiny210 on Android 4.0, which was rooted natively.
check if the device connected to working computer indeed:
$ adb devices List of devices attached 0123456789ABCDEF device
push the binary to the android :
$ adb push libs/armeabi/sandbox-discover-standalone sandbox-discover-standalone 2291 KB/s (202300 bytes in 0.086s)
execute it in the shell:
$ adb shell / # chmod 777 sandbox-discover-standalone / # ./sandbox-discover-standalone 1 services have been found Service Type: _http._tcp Service Name: GAIGER NETEYE CS2230 Domain Name: local Interface: eth0 IPv4 Address: 192.168.13.107/NETEYECS2230-001A97018882.local:80 TEXT Data: NULL discover cast 1002 msec
It is successful totally.
Known insufficiency :
if you encounter Segmentation fault when you execute the binary in Android,
you should correct the code in sandbox-discover-standalone.c, of function ServiceResolverCallback:
pServiceInfo->pServiceName = strndup(pServiceName, MAX_STR_LEN); pServiceInfo->pDomainName = strndup(pDomain, MAX_STR_LEN); #ifndef __ANDROID__ pServiceInfo->pInterface = strndup(if_indextoname(interface, (char*)pServiceName), MAX_STR_LEN); #endif pServiceInfo->pProtocolNumber = strndup(avahi_proto_to_string(protocol), MAX_STR_LEN);
It could avoid the crashing:
Extra:
It is passible to write a stand Makefile for Android ndk build :
i assure the Makefile be in sandyboxcode/Makefile (NOT inside the jni folder), for the sandbox-discover-sandalone.c be :
CC := arm-linux-androideabi-gcc CXX := arm-linux-androideabi-g++ INC := -I~/Android/AndroidNDK/android-ndk-r10e/platforms/android-14/arch-arm/usr/include INC += -I../built/include -Ijni AVAHI_LIB_PATH := ../built/lib LIB := ~/Android/AndroidNDK/android-ndk-r10e/platforms/android-14/arch-arm/usr/lib/liblog.so LIB += $(AVAHI_LIB_PATH)/libavahi-core.a LIB += $(AVAHI_LIB_PATH)/libavahi-common.a CFLAGS := -g all : $(CC) $(CFLAGS) $(INC) jni/sandbox-discover-standalone.c $(LIB) -o sandbox-discover-standalone clean : rm sandbox-discover-standalone -rf
Note : the shared libraries liblog.so be linking directly!.
Hello,
回覆刪除do you still have the file "0002-Add-uninstalled.pc.in-files.patch" available and would be so kind as to send it to me?
My address is: andrea.sonnleitner1993@gmail.com
I do not know if this message reaches you but I hope for the best. Thank you very much.
Andrea Sonnleitner
Thank you very much that you posted the files! They helped a lot. Unfortunately, I have problems running your configure script. Would you be so kind and let me explain my project over a different channel like my E-Mail address (andrea.sonnleitner1993@gmail.com) or stackoverflow?
回覆刪除Thank you for your time,
Andrea S.
Thank you very much that you posted the files! They helped a lot. Unfortunately, I have problems running your configure script. Would you be so kind and let me explain my project over a different channel like my E-Mail address (andrea.sonnleitner1993@gmail.com) or stackoverflow?
回覆刪除Thank you for your time,
Andrea S.
I try to run the configure script as you mention above but receive the error: missing posix threads support. Have you encountered this error before?
回覆刪除Hi Chen, Can i ask if we install avahi into android TV aosp, if I turn on the Wi-Fi tethering (192.168.43.1), I can discover the chromecast device on the LAN IP range? Appreciate if you can reply me. Vincent Khoo
回覆刪除I have no experience in using Android TV, sorry.
刪除