【Android】圖文解密Android OverlayManagerService

1、簡介

Android OverlayManagerService(OMS)是一個有意思的模塊,可以在運行時動態替換res,如圖片。簡單來說就是一個普通apk有若干res,另一個相關的overlay apk有相同的res,關閉overlay apk時,普通apk運行時加載的是自己本身的res,而打開overlay apk時,普通apk運行時加載的則是overlay apk中的res。

2、用法舉例

普通apk爲fruit.apk,package爲com.example.fruit,res爲/res/drawable/fruit.jpeg,這個圖片可以在xml或java中使用。目錄結構大致如下:

├── AndroidManifest.xml
├── Android.mk
├── java
│   └── com
│       └── example
│           └── fruit
│               ├── App.java
│               └── MainActivity.java
└── res
    ├── drawable
    │   ├── fruit.jpeg
    │   └── ic_launcher_background.xml
    ├── drawable-v24
    │   └── ic_launcher_foreground.xml
    ├── layout
    │   └── activity_main.xml
    ├── mipmap-anydpi-v26
    │   ├── ic_launcher_round.xml
    │   └── ic_launcher.xml
    ├── mipmap-hdpi
    │   ├── ic_launcher.png
    │   └── ic_launcher_round.png
    ├── mipmap-mdpi
    │   ├── ic_launcher.png
    │   └── ic_launcher_round.png
    ├── mipmap-xhdpi
    │   ├── ic_launcher.png
    │   └── ic_launcher_round.png
    ├── mipmap-xxhdpi
    │   ├── ic_launcher.png
    │   └── ic_launcher_round.png
    ├── mipmap-xxxhdpi
    │   ├── ic_launcher.png
    │   └── ic_launcher_round.png
    └── values
        ├── colors.xml
        ├── strings.xml
        └── styles.xml

Androd.mk使用LOCAL_CERTIFICATE := platform進行簽名,LOCAL_AAPT_FLAGS := --auto-add-overlay的目的是支持overlay apk的資源可以和普通apk的資源不同,代碼如下:

LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_SRC_FILES := $(call all-java-files-under, java)
LOCAL_RESOURCE_DIR := $(LOCAL_PATH)/res
LOCAL_PACKAGE_NAME := myapp
LOCAL_AAPT_FLAGS := --auto-add-overlay
LOCAL_CERTIFICATE := platform
include $(BUILD_PACKAGE)

AndroidManifest.xm通過android:sharedUserId="android.uid.system"使用系統權限,並添加overlay相應的權限,代碼如下:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.fruit"
    android:sharedUserId="android.uid.system">
    <uses-permission android:name="android.permission.MODIFY_THEME_OVERLAY" />
    <uses-permission android:name="android.permission.CHANGE_OVERLAY_PACKAGES" />
    <uses-permission android:name="android.permission.INTERACT_ACROSS_USERS" />
    <uses-permission android:name="android.permission.INTERACT_ACROSS_USERS_FULL" />
    <application
        android:name=".App"
        android:allowBackup="true"
        android:hardwareAccelerated="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:theme="@style/AppTheme">
        <activity android:name=".MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>
</manifest>

然後是對應的overlay apk,即fruitOverlay.apk,package爲com.example.fruit.overlay,有同樣的res即res/drawable/fruit.jpeg。目錄結構如下:

├── AndroidManifest.xml
├── Android.mk
└── res
    └── drawable
        └── fruit.jpeg

Android.mk中LOCAL_RRO_THEME = fruit指定fruitOverlay apk編譯後在vendor/overlay/fruit目錄下,include $(BUILD_RRO_PACKAGE)表示編譯overlay apk,RRO即Runtime Resource Overlay,對應的還有個SRO,表示Static Resource Overlay,SRO發生在編譯時,這裏不作介紹。

LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_RESOURCE_DIR := $(LOCAL_PATH)/res
LOCAL_PACKAGE_NAME := fruitOverlay
LOCAL_RRO_THEME = fruit
include $(BUILD_RRO_PACKAGE)

AndroidManifest.xml的overlay標籤是個重要的內容,targetPackage表示作用的普通apk的package,priority表示優先級,isStatic爲true時表示overlay apk的狀態不可動態切換,代碼如下:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.fruit.overlay "
    android:versionCode="1"
    android:versionName="1.0">
    <overlay android:targetPackage="com.example.fruit"
    	android:priority="1"
    	android:isStatic="false" />
</manifest>

最後,可以進行測試了,可以在命令行進行測試,也可以使用OverlayManagerService的接口。命令行“cmd overlay enable com.example.fruit.overlay”表示打開overlay, “cmd overlay disable com.example.fruit.overlay”表示關閉overlay,其它命令可通過“cmd overlay help”進行查看。OverlayManagerService的接口可參照“IOverlayManger.aidl”。

3、相關代碼

frameworks/base/core/java/android/content/om/
├── IOverlayManager.aidl
├── OverlayInfo.aidl
└── OverlayInfo.java

frameworks/base/services/core/java/com/android/server/om
├── IdmapManager.java
├── OverlayManagerServiceImpl.java
├── OverlayManagerService.java
├── OverlayManagerSettings.java
└── OverlayManagerShellCommand.java

IOverlayManager:aidl,提供了管理overlay package的接口。
OverlayInfo:封裝了ovelay package相關的所有信息。
IdmapManager:overlay package資源管理。
OverlayManagerService/OverlayManagerServiceImpl:OMS管理、實現類。
OverlayManagerSettings:專門用於設置overlay 的各個狀態。
OverlayManagerShellCommand:支持overlay命令行。

4、啓動流程

OMS啓動流程如下。

在這裏插入圖片描述

5、使用一個overlay

前面介紹了使用overlay的方法,overlay狀態發生變化時,對應的package的Activity會重啓,下面介紹下enable一個overlay的大致流程。

在這裏插入圖片描述

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