android 5.0預製APK方法,android5.0預製apk

最近在集成三方APK到system/app下,而且三方apk含有so庫,查了很多資料,發現是so庫問題導致的,

現記錄下來,希望能幫助到遇到同樣問題的人

第一情況、如何將帶源碼的APK預置進系統?
1) 在 packages/apps 下面以需要預置的 APK的 名字創建一個新文件夾,以預置一個名爲Test的APK 爲例
2) 將 Test APK的Source code 拷貝到 Test 文件夾下,刪除 /bin 和 /gen 目錄
3) 在 Test 目錄下創建一個名爲 Android.mk的文件,內容如下:
LOCAL_PATH:= $(call my-dir)
include $(CLEAR_VARS)

LOCAL_MODULE_TAGS := optional
LOCAL_SRC_FILES := $(call all-subdir-java-files)

LOCAL_PACKAGE_NAME := Test
include $(BUILD_PACKAGE)
4) 打開文件 device\mediatek\common\device.mk
將 Test 添加到 PRODUCT_PACKAGES 裏面。
PRODUCT_PACKAGES += Test
5) 重新 build 整個工程

第二情況、如何將無源碼的 APK 預置進系統? 
1) 在 packages/apps 下面以需要預置的 APK 名字創建文件夾,以預置一個名爲Test的APK爲例
2) 將 Test.apk 放到 packages/apps/Test 下面
3) 在 packages/apps/Test 下面創建文件 Android.mk,文件內容如下:
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
# Module name should match apk name to be installed
LOCAL_MODULE := Test
LOCAL_MODULE_TAGS := optional

LOCAL_SRC_FILES := $(LOCAL_MODULE).apk
LOCAL_MODULE_CLASS := APPS
LOCAL_MODULE_SUFFIX := $(COMMON_ANDROID_PACKAGE_SUFFIX)

LOCAL_PREBUILT_JNI_LIBS:= \
@lib/armeabi/libtest.so \
@lib/armeabi/libtest2.so 

LOCAL_CERTIFICATE := PRESIGNED
include $(BUILD_PREBUILT)

若無so,刪除LOCAL_PREBUILT_JNI_LIBS
若有so,使用LOCAL_PREBUILT_JNI_LIBS列出所有so的路徑,不要忘記使用@。@標識符會將apk中的so抽離出來build進apk同級目錄下的lib文件夾中

若apk支持不同cpu類型的so,針對so的部分的處理:
Ifeq ($(TARGET_ARCH),arm)
LOCAL_PREBUILT_JNI_LIBS := \
@lib/armeabi-v7a/xxx.so\
@ lib/armeabi-v7a/xxxx.so
else ifeq ($(TARGET_ARCH),x86)
LOCAL_PREBUILT_JNI_LIBS := \
@lib/x86/xxx.so
else ifeq ($(TARGET_ARCH),arm64)
LOCAL_PREBUILT_JNI_LIBS := \
@lib/armeabi-v8a/xxx.so

即將和TARGET_ARCH對應的so抽離出來

4) 打開文件 device\mediatek\common\device.mk
將 Test 添加到 PRODUCT_PACKAGES 裏面。
PRODUCT_PACKAGES += Test
5) 重新 build 整個工程

注:如果App使用System Level的permission,需要預置到/system/priv-app底下 (原在/system/app)。
修改Android.mk,增加LOCAL_PRIVILEGED_MODULE := true,以聲明app需要放在/system/priv-app下。

第三情況、如何預置APK使得用戶可以卸載,恢復出廠設置時不能恢復?
1) 在 packages/apps 下面以需要預置的 APK 名字創建文件夾,以預置一個名爲Test的APK爲例
2) 將 Test.apk 放到 packages/apps/Test 下面
3) 在 packages/apps/Test 下面創建文件 Android.mk,文件內容如下:

LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)

# Module name should match apk name to be installed
LOCAL_MODULE := Test
LOCAL_MODULE_TAGS := optional

LOCAL_SRC_FILES := $(LOCAL_MODULE).apk
LOCAL_MODULE_CLASS := APPS
LOCAL_MODULE_SUFFIX := $(COMMON_ANDROID_PACKAGE_SUFFIX)
# LOCAL_PRIVILEGED_MODULE := true
LOCAL_MODULE_PATH := $(TARGET_OUT_DATA_APPS)

LOCAL_CERTIFICATE := PRESIGNED
include $(BUILD_PREBUILT)

4) 打開文件 device\mediatek\common\device.mk
將 Test 添加到 PRODUCT_PACKAGES 裏面。
PRODUCT_PACKAGES += Test
5) 重新 build 整個工程
注意:這個比不能卸載的多了一句
LOCAL_MODULE_PATH := $(TARGET_OUT_DATA_APPS)

第四情況如何預置APK使得用戶可以卸載,並且恢復出廠設置時能夠恢復?
1在 vendor\mediatek\proprietary\binary\3rd-party\free下面以需要預置的 APK 名字創建文件夾,以預置一個名爲Test的APK爲例
2 將Test.apk 放入vendor\mediatek\proprietary\binary\3rd-party\free\Test下面
3 在vendor\mediatek\proprietary\binary\3rd-party\free\Test 下面創建文件 Android.mk,文件內容如下

LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)

# Module name should match apk name to be installed
LOCAL_MODULE := Test
LOCAL_MODULE_TAGS := optional
LOCAL_SRC_FILES := $(LOCAL_MODULE).apk

LOCAL_MODULE_CLASS := APPS
LOCAL_MODULE_SUFFIX := $(COMMON_ANDROID_PACKAGE_SUFFIX)
LOCAL_CERTIFICATE := PRESIGNED

LOCAL_MODULE_PATH := $(TARGET_OUT)/vendor/operator/app
include $(BUILD_PREBUILT)

2 打開文件device\mediatek\common\device.mk
將 Test 添加到 PRODUCT_PACKAGES 裏面。
PRODUCT_PACKAGES += Test
3 然後重新build整個工程


若需要apk作爲32bit的apk運行,則需要在Android.mk中定義

LOCAL_MULTILIB :=32



對於第三方apk預裝入系統時如果將apk裝入system/app目錄下,由於第三方apk帶有自己的.so文件,此時會出現apk無法打開的情況。
解決方法:
1:解壓apk,將裏面的.so文件放入device某個文件夾下,
  例如我們4.2box的平臺:\rk3066box_4.2.2_r1\device\rockchip\rk30sdk\apk目錄下新建一個目錄realtoplib,在此目錄下放入所有需要的.so文件。
         然後在\rk3066box_4.2.2_r1\device\rockchip\rk30sdk\devices.mk中加入
#########################################################
# Copy proprietary so
#########################################################
  define all-so-files-under
  $(patsubst ./%,%, \
        $(shell cd $(LOCAL_PATH)/$(1) ; \
          find ./ -maxdepth 1  -name "*.apk" -and -not -name ".*") \
    )
endef


COPY_APK_TARGET := $(call all-so-files-under,apk/realtoplib)
PRODUCT_COPY_FILES += $(foreach soName, $(COPY_APK_TARGET), \
$(addprefix $(LOCAL_PATH)/apk/realtoplib/, $(soName)):$(addprefix system/lib/, $(soName)))
此去腳本的的意思是遍歷目錄下的.so文件copy至system/lib下。
這樣就可以正常啓動apk 了。


2:將apk裝入data/app目錄下,此目錄下的apk 相對system/app來說,用戶是可以刪除的。
   例如我們4.2box的平臺:\rk3066box_4.2.2_r1\device\rockchip\rk30sdk目錄下建一個realtopapk目錄
   將需要放入data/app目錄下的(也就是可供用戶卸載的apk )放入realtopapk目錄下
   同樣修改devices.mk
#########################################################
# Copy proprietary apk to system/usr/app
#########################################################
define all-data-files-under
$(patsubst ./%,%, \
  $(shell cd $(LOCAL_PATH)/$(1) ; \
          find ./ -maxdepth 1  -name "*.apk" -and -not -name ".*") \
 )
endef


COPY_APK_TARGET := $(call all-data-files-under,realtopapk)
PRODUCT_COPY_FILES += $(foreach apkName, $(COPY_APK_TARGET), \
$(addprefix $(LOCAL_PATH)/realtopapk/, $(apkName)):$(addprefix system/usr/app/, $(apkName)))
此去腳本的的意思是遍歷目錄下的.apk文件copy至system/usr/app/下。
然後在\rk3066box_4.2.2_r1\device\rockchip\rk30sdk目錄下添加一個腳本文件
例如run_apk.sh
#!/system/bin/sh
  busybox cp /system/usr/app/AngryBirds.apk data/app
  busybox cp /system/usr/app/Skype.apk data/app
  busybox cp /system/usr/app/Twitter.apk data/app
  chmod 777 data/app/AngryBirds.apk
  chmod 777 data/app/Skype.apk
  chmod 777 data/app/Twitter.apk
修改devices.mk將此腳本複製到system/bin下
PRODUCT_COPY_FILES += \
device/rockchip/$(TARGET_PRODUCT)/run_apk.sh:system/bin/run_apk.sh
同時修改init.rk30board.rc文件添加啓動服務
service run_apk /system/bin/run_apk.sh
    class main
    oneshot
重新編譯系統,系統啓動後就會安裝data/app目錄下的apk,用戶可自行卸載。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章