在Android上運行本地C語言代碼(Hello C on Android)

編譯可以在Android上運行的native(C/C++)程序 Hello C on Android

JNI編程都會,寫出一個共享庫(.so),然後Java代碼來調用

現在來編譯一個可執行的native程序直接在Android Device上運行
這裡以C語言來測試

一樣按照通常的編程步驟來進行,編寫源文件,用編譯器編譯,鏈接器鏈接,然後運行

編寫源文件和運行的環境都比較好弄,編譯器和鏈接器這裡就稍稍有點不同。
因為Android所采用的kernel不是標準的Linux Kernel,C庫用的是Bionic(http://en.wikipedia.org/wiki/Bionic_(software))
所以用普通的一套GNU工具編譯出來的是不能在Android上面運行的,還好Android給我們提供瞭一些工具來完成這些。

有兩種方式可以達到這個目的:
1、編譯Android ROM的時候會生成這些相關的工具,我們可以利用這些工具來編譯生成我們的程序。

 但是我測試瞭下,沒有成功,時間比較緊就沒有來具體去查原因,有空再來看看。
下面是我用這種方式所產生的錯誤,如果你知道原因,歡迎能告訴我

1 $ agcc.pl -o hello hello.c
/home/guohai/src/ics/prebuilt/linux-x86/toolchain/arm-eabi-4.3.1/bin/../lib/gcc/arm-eabi/4.3.1/http://www.cnblogs.com/http://www.cnblogs.com/arm-eabi/bin/ld: warning: /tmp/cc8eCaMQ.o uses variable-size enums yet the output is to use 32-bit enums; use of enum values across objects may fail

 2、通過Android NDK來編譯生成,這個比較簡單
我是參照http://ideone.com/lt6BW修改過來的
為瞭簡單起見,腳本裡面的路徑都是hard code, tools/blob/master/agcc。
對比下應該很好改出自己的版本,使用的時候要記得給這個腳本可執行權限,並且最好把該腳本增加到PATH當中,方便後面的使用。

但是直接運行agcc會報錯,如下

1 $ agcc
GCC:/home/guohai/dev/android-ndk-r7/toolchains/arm-linux-androideabi-4.4.3/prebuilt/linux-x86/bin/arm-linux-androideabi-gcc LIB:/home/guohai/vocume/dev/android-ndk-r7/platforms/android-14/arch-arm/usr/lib LINKER:/system/bin/linker PARAMS:
/home/guohai/dev/android-ndk-r7/platforms/android-14/arch-arm/usr/lib/crtbegin_dynamic.o: In function `_start’:
(.text+0×14): undefined reference to `main’
collect2: ld returned 1 exit status
如果你發現是這個錯誤,不要擔心,可能是這個原因 隻要你輸入的命令是完整的,比如agcc hello.c -o hello,應該就不會出錯。

假設後面該腳本的使用都是滿足上面這些條件的。

現在基本環境都搭建好瞭,我們還沒有源程序。

先在Host Machine上編寫一個hello.c文件

1 #include

3 main()
4 {
5     printf ("Hello C on Android!\n");
6 }   www.aiwalls.com
然後運行agcc hello.c -o hello,如果沒有任何錯誤的話,就應該看到有如下的返回,並且生成瞭一個hello文件。

1 $ agcc hello.c -o hello
2 GCC:/home/guohai/dev/android-ndk-r7/toolchains/arm-linux-androideabi-4.4.3/prebuilt/linux-x86/bin/arm-linux-androideabi-gcc LIB:/home/guohai/dev/android-ndk-r7/platforms/android-14/arch-arm/usr/lib LINKER:/system/bin/linker PARAMS:hello.c -o hello
執行

1 $ file hello
2 hello: ELF 32-bit LSB executable, ARM, version 1 (SYSV), dynamically linked (uses shared libs), not stripped
返回信息應該是類似上面這樣的。

現在就隻差最後一步瞭,把程序放到Android上去運行

1 $ adb remount
2 $ adb shell mkdir /data/hello_c
3 $ adb push hello /data/hello_c
4 $ adb shell
1 # cd /data/hello_c/
2 # ./hello
應該在ADB Shell上會出現

1 Hello C on Android!
這樣這個實驗就算完成瞭。後面會有實際的應用。

PS.
如果你直接用gcc編譯出來,放到Android上面去運行會出現
/system/bin/sh: ./android-smemcap: not executable: magic 7F45

This entry was posted in Android and tagged NDK. Bookmark the permalink.

You May Also Like