1. 程式人生 > >Android編譯自己的程式到/system/bin

Android編譯自己的程式到/system/bin

背景

有時候我們想建立一個程式,放在系統中,供其他APP執行。我們知道,在生成system.img的時候,編譯系統會將out/target/product/[product]/system/bin目錄打包進去。所以,我們想辦法讓編譯系統在編譯的過程中,把我們的程式編譯了,並把編譯生成的二進位制檔案自動放到out/target/product/[product]/system/bin。

做法

假如我們要建立一個mytest的程式
1.在external目錄下建立名為mytest的資料夾,這個資料夾用於存放我們的程式程式碼和makefile檔案以及這個程式的依賴庫。
2.建立我們的程式程式碼和makefile檔案,這裡主要看看makefile檔案怎麼寫,例子如下:

LOCAL_PATH := $(call my-dir)

common_cflags := \
    -std=c99 \
    -Os \
    -Wall \
    -Wextra \
    -Wno-char-subscripts \
    -Wno-sign-compare \
    -Wno-string-plus-int \
    -Wno-uninitialized \
    -Wno-unused-parameter \
    -funsigned-char \
    -ffunction-sections -fdata-sections \
    -fno-asynchronous-unwind-tables \

# static executable for use in limited environments
include $(CLEAR_VARS) LOCAL_SRC_FILES := mytest.c LOCAL_CFLAGS := $(common_cflags) LOCAL_CXX_STL := none LOCAL_CLANG := true # LOCAL_MODULE_PATH and LOCAL_UNSTRIPPED_PATH do not equal LOCAL_UNSTRIPPED_PATH := $(PRODUCT_OUT)/symbols/utilities LOCAL_MODULE := mytest_static LOCAL_MODULE_CLASS := UTILITY_EXECUTABLES LOCAL_MODULE_PATH
:= $(PRODUCT_OUT)/system/bin LOCAL_MODULE_TAGS := optional LOCAL_MODULE_STEM := mytest LOCAL_PACK_MODULE_RELOCATIONS := false LOCAL_STATIC_LIBRARIES := libc libcurl libz libcrypto_static LOCAL_SHARED_LIBRARIES := libssl LOCAL_FORCE_STATIC_EXECUTABLE := true include $(BUILD_EXECUTABLE)

上面的程式碼中,值得關注的幾個內容:

LOCAL_SRC_FILES:要編譯的原始檔
LOCAL_CFLAGS:編譯時所要使用的編譯選項
LOCAL_UNSTRIPPED_PATH:生成的二進位制檔案臨時存放目錄。$(PRODUCT_OUT)是產品生成目錄,等於上文提到的out/target/product/[product]
LOCAL_MODULE:模組的名稱。這裡我們生成的是靜態可執行檔案,所以後面加上'_static'
LOCAL_MODULE_PATH:模組生成後的存放的目錄。$(PRODUCT_OUT)是產品生成目錄,等於上文提到的out/target/product/[product]
LOCAL_MODULE_TAGS:設定模組標籤,這影響在什麼版本下編譯該模組。
LOCAL_STATIC_LIBRARIES:依賴的靜態庫
LOCAL_SHARED_LIBRARIES:依賴的動態庫
LOCAL_FORCE_STATIC_EXECUTABLE:是否強制生成靜態可執行檔案
include $(BUILD_EXECUTABLE):生成可執行檔案

注意:建立可執行檔案,原始碼要有main函式

3.讓編譯系統執行後編譯我們的模組
(1)在/build/target/product目錄下,建立一個makefile檔案,這裡就取為mytest.mk
(2)在裡面輸入:

PRODUCT_PACKAGES := mytest_static

PRODUCT_PACKAGES是編譯系統定義的變數,在後面寫上我們模組名,這樣我們的模組就會被編譯。
(3)讓編譯系統找到我們的mytest.mk,在build/target/product/core.mk檔案末尾追加一行:
$(call inherit-product, $(SRC_TARGET_DIR)/product/mytest.mk)

測試模組

生成我們自己的模組後,有兩種方式測試。
1.只編譯模組,然後把它push到我們的裝置。這種方式是最方便的。在原始碼目錄下執行下面兩個命令:
(1)編譯模組

mmm ./external/mytest

(2)將模組push到手機

adb push $OUT/system/bin/mytest /system/bin

注:如果提示couldn't create file: Read-only file system,先執行adb remount
(3)執行我們的編譯好的模組

adb shell mytest

2.編譯整個system.img,模組會打包進映象中
(1)編譯system.img

make systemimage

(2)刷入system.img

fastboot flash system $OUT/system.img

(3)執行我們的編譯好的模組

adb shell mytest