安卓RRO机制

年前和组内的小伙伴讨论过一个需求的RRO实现方案。我其实之前对RRO也只是处于大概了解的程度,并没有实际去操作过,趁着过年这段时间有空也写了个demo实际验证了下。

由于官方文档实际上对整个RRO机制已经讲的比较清楚了,我这里只做一些补充。

RRO的默认状态

根据官方文档的介绍普通的RRO默认是停用状态的,类似国内的系统主题包其实下载安装之后是没有启用的,需要程序用OverlayManager.setEnable去启动它。

但我们是需要预装默认启用的,有下面两种方式可以实现

静态 RRO

第一种是使用静态 RRO,即将manifest里面的android:isStatic设置成true。

这种方式预装的overlay包会默认启用,而且不能用命令或者程序去禁用它(如果是用pm install去安装的话不会生效,还是默认停用需要用命令或者程序去启用)

这种方式比较适合运行时不会改变的客制化需求(俗称换皮)。

OverlayConfig

第二种是使用OverlayConfig,在机器的/{partition}/overlay/config/config.xml里面添加配置:

<config>
    <merge path="OEM-common-rros-config.xml" />
    <overlay package="com.oem.overlay.device" mutable="false" enabled="true" />
    <overlay package="com.oem.green.theme" enabled="true" />
</config>

这里的enabled配置默认是否启用(默认为false),mutable配置运行时是否可修改(默认为true),如果配置成mutable="false" enabled="true"则和android:isStatic设置成true使用静态 RRO效果一样。

然后需要将overlay apk预装到/{partition}/overlay/目录下,由于OverlayConfigParser是在xml上级的overlay里面扫描overlay apk,如果不预装在这个路径的话就会报错:

E Zygote  : System zygote died with fatal exception
E Zygote  : java.lang.ExceptionInInitializerError
E Zygote  :      at java.lang.Class.classForName(Native Method)
E Zygote  :      at java.lang.Class.forName(Class.java:454)
E Zygote  :      at com.android.internal.os.ZygoteInit.preloadClasses(ZygoteInit.java:301)
E Zygote  :      at com.android.internal.os.ZygoteInit.preload(ZygoteInit.java:140)
E Zygote  :      at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:889)
E Zygote  : Caused by: java.lang.IllegalStateException: overlay me.linjw.demo.overlay.overlay not present in partition /vendor/overlay in /vendor/overlay/config/config.xml at START_TAG (empty) <overlay package='me.linjw.demo.overlay.overlay' enabled='true'>@2:66 in java.io.FileReader@583e450
E Zygote  :      at com.android.internal.content.om.OverlayConfigParser.parseOverlay(OverlayConfigParser.java:372)
E Zygote  :      at com.android.internal.content.om.OverlayConfigParser.readConfigFile(OverlayConfigParser.java:249)
E Zygote  :      at com.android.internal.content.om.OverlayConfigParser.getConfigurations(OverlayConfigParser.java:220)
E Zygote  :      at com.android.internal.content.om.OverlayConfig.<init>(OverlayConfig.java:152)
E Zygote  :      at com.android.internal.content.om.OverlayConfig.getZygoteInstance(OverlayConfig.java:218)
E Zygote  :      at android.content.res.AssetManager.createSystemAssetsInZygoteLocked(AssetManager.java:252)
E Zygote  :      at android.content.res.AssetManager.getSystem(AssetManager.java:276)
E Zygote  :      at android.content.res.Resources.<init>(Resources.java:347)
E Zygote  :      at android.content.res.Resources.getSystem(Resources.java:236)
E Zygote  :      at com.android.internal.telephony.GsmAlphabet.enableCountrySpecificEncodings(GsmAlphabet.java:1090)
E Zygote  :      at com.android.internal.telephony.GsmAlphabet.<clinit>(GsmAlphabet.java:1495)
E Zygote  :      ... 5 more

partition可以是下面的值,如果有多个overlay apk对同一个资源做overlay,优先级从低到高:

  • system
  • vendor
  • odm
  • oem
  • product
  • system_ext

我这里的和官方文档里面写的优先级odm、oem反了,这是因为我看安卓13的实现代码里面定义是这样的:

// https://cs.android.com/android/platform/superproject/+/android-13.0.0_r74:frameworks/base/core/java/com/android/internal/content/om/OverlayConfig.java;l=132
partitions = new ArrayList<>(
                      PackagePartitions.getOrderedPartitions(OverlayPartition::new));

// https://cs.android.com/android/platform/superproject/+/android-13.0.0_r74:frameworks/base/core/java/android/content/pm/PackagePartitions.java
/**
 * The list of all system partitions that may contain packages in ascending order of
 * specificity (the more generic, the earlier in the list a partition appears).
 */
private static final ArrayList<SystemPartition> SYSTEM_PARTITIONS =
        new ArrayList<>(Arrays.asList(
                new SystemPartition(Environment.getRootDirectory(),
                        PARTITION_SYSTEM, Partition.PARTITION_NAME_SYSTEM,
                        true /* containsPrivApp */, false /* containsOverlay */),
                new SystemPartition(Environment.getVendorDirectory(),
                        PARTITION_VENDOR, Partition.PARTITION_NAME_VENDOR,
                        true /* containsPrivApp */, true /* containsOverlay */),
                new SystemPartition(Environment.getOdmDirectory(),
                        PARTITION_ODM, Partition.PARTITION_NAME_ODM,
                        true /* containsPrivApp */, true /* containsOverlay */),
                new SystemPartition(Environment.getOemDirectory(),
                        PARTITION_OEM, Partition.PARTITION_NAME_OEM,
                        false /* containsPrivApp */, true /* containsOverlay */),
                new SystemPartition(Environment.getProductDirectory(),
                        PARTITION_PRODUCT, Partition.PARTITION_NAME_PRODUCT,
                        true /* containsPrivApp */, true /* containsOverlay */),
                new SystemPartition(Environment.getSystemExtDirectory(),
                        PARTITION_SYSTEM_EXT, Partition.PARTITION_NAME_SYSTEM_EXT,
                        true /* containsPrivApp */, true /* containsOverlay */)));

/**
 * Returns a list in which the elements are products of the specified function applied to the
 * list of {@link #SYSTEM_PARTITIONS} in increasing specificity order.
 */
public static <T> ArrayList<T> getOrderedPartitions(
        @NonNull Function<SystemPartition, T> producer) {
    final ArrayList<T> out = new ArrayList<>();
    for (int i = 0, n = SYSTEM_PARTITIONS.size(); i < n; i++) {
        final T v = producer.apply(SYSTEM_PARTITIONS.get(i));
        if (v != null)  {
            out.add(v);
        }
    }
    return out;
}

overlay命令

根据官方文档在调试阶段可以用shell命令去调试overlay。有时候对具体命令的使用有疑问的话可以直接看它的实现代码:

// https://cs.android.com/android/platform/superproject/+/android-13.0.0_r74:frameworks/base/services/core/java/com/android/server/om/OverlayManagerShellCommand.java;l=61

public int onCommand(@Nullable final String cmd) {
    if (cmd == null) {
        return handleDefaultCommands(cmd);
    }
    final PrintWriter err = getErrPrintWriter();
    try {
        switch (cmd) {
            case "list":
                return runList();
            case "enable":
                return runEnableDisable(true);
            case "disable":
                return runEnableDisable(false);
            case "enable-exclusive":
                return runEnableExclusive();
            case "set-priority":
                return runSetPriority();
            case "lookup":
                return runLookup();
            case "fabricate":
                return runFabricate();
            default:
                return handleDefaultCommands(cmd);
        }
    } catch (IllegalArgumentException e) {
        err.println("Error: " + e.getMessage());
    } catch (RemoteException e) {
        err.println("Remote exception: " + e);
    }
    return -1;
}

例如cmd overlay list命令列出了各个overlay apk的启用状态,[x]是已经启用,[ ]是停用:

me.linjw.demo.overlay1.app
[ ] me.linjw.demo.overlay1.overlay

me.linjw.demo.overlay2.app
[x] me.linjw.demo.overlay2.overlay

com.android.connectivity.resources
--- com.rockchip.networkstack.tethering.nokeepalive.overlay

---则是代表这个overlay包处于不可用状态无法使用cmd overlay enable {package}或者cmd overlay disable {package}去启用或者停用它。这种情况会出现在目标包未预装、没有找到需要overlay的资源等情况:

// https://cs.android.com/android/platform/superproject/+/android-13.0.0_r74:frameworks/base/services/core/java/com/android/server/om/OverlayManagerShellCommand.java;l=193
private void printListOverlay(PrintWriter out, OverlayInfo oi) {
    String status;
    switch (oi.state) {
        case OverlayInfo.STATE_ENABLED_IMMUTABLE:
        case OverlayInfo.STATE_ENABLED:
            status = "[x]";
            break;
        case OverlayInfo.STATE_DISABLED:
            status = "[ ]";
            break;
        default:
            status = "---";
            break;
    }
    out.println(String.format("%s %s", status, oi.getOverlayIdentifier()));
}

// https://cs.android.com/android/platform/superproject/+/android-13.0.0_r74:frameworks/base/core/java/android/content/om/OverlayInfo.java;l=45
@IntDef(prefix = "STATE_", value = {
        STATE_UNKNOWN,
        STATE_MISSING_TARGET,
        STATE_NO_IDMAP,
        STATE_DISABLED,
        STATE_ENABLED,
        STATE_ENABLED_IMMUTABLE,
        // @Deprecated STATE_TARGET_IS_BEING_REPLACED,
        STATE_OVERLAY_IS_BEING_REPLACED,
})
/** @hide */
@Retention(RetentionPolicy.SOURCE)
public @interface State {}

/**
 * An internal state used as the initial state of an overlay. OverlayInfo
 * objects exposed outside the {@link
 * com.android.server.om.OverlayManagerService} should never have this
 * state.
 *
 * @hide
 */
public static final int STATE_UNKNOWN = -1;

/**
 * The target package of the overlay is not installed. The overlay cannot be enabled.
 *
 * @hide
 */
public static final int STATE_MISSING_TARGET = 0;

/**
 * Creation of idmap file failed (e.g. no matching resources). The overlay
 * cannot be enabled.
 *
 * @hide
 */
public static final int STATE_NO_IDMAP = 1;

/**
 * The overlay is currently disabled. It can be enabled.
 *
 * @see IOverlayManager#setEnabled
 * @hide
 */
public static final int STATE_DISABLED = 2;

/**
 * The overlay is currently enabled. It can be disabled.
 *
 * @see IOverlayManager#setEnabled
 * @hide
 */
public static final int STATE_ENABLED = 3;

/**
 * The target package is currently being upgraded or downgraded; the state
 * will change once the package installation has finished.
 * @hide
 *
 * @deprecated No longer used. Caused invalid transitions from enabled -> upgrading -> enabled,
 * where an update is propagated when nothing has changed. Can occur during --dont-kill
 * installs when code and resources are hot swapped and the Activity should not be relaunched.
 * In all other cases, the process and therefore Activity is killed, so the state loop is
 * irrelevant.
 */
@Deprecated
public static final int STATE_TARGET_IS_BEING_REPLACED = 4;

/**
 * The overlay package is currently being upgraded or downgraded; the state
 * will change once the package installation has finished.
 * @hide
 */
public static final int STATE_OVERLAY_IS_BEING_REPLACED = 5;

/**
 * The overlay package is currently enabled because it is marked as
 * 'immutable'. It cannot be disabled but will change state if for instance
 * its target is uninstalled.
 * @hide
 */
@Deprecated
public static final int STATE_ENABLED_IMMUTABLE = 6;

然后可以用lookup命令查看启用、停用overlay包的情况下最终读取到的资源值是什么:

cmd overlay lookup me.linjw.demo.overlay.app me.linjw.demo.overlay.app:string/app_name

idmap

之前我写过一系列博客探索过安卓的资源机制,实际上安卓是通过一个int的资源id去resources.arsc里面查询的资源,如果Overlay apk里面的id和目标apk的资源id不一致要怎么处理?

其实安卓是通过idmap机制实现的,在/data/resource-cache/目录下保存有各个overlay应用的idmap映射文件,可以用idmap2 dump --idmap-path {file}命令去打印映射关系:

idmap2 dump --idmap-path /data/resource-cache/system_ext@overlay@OverlayDemo.apk@idmap
Paths:
    target path  : /system_ext/app/OverlayDemo/OverlayDemo.apk
    overlay path : /system_ext/overlay/OverlayDemoOverlay.apk
Debug info:
    W failed to find resource 'string/app_name2'
Mapping:
    0x7f030000 -> 0x7f010000 (string/app_name -> string/app_name)

例如上面的打印指的是在OverlayDemo.apk里面用0x7f030000这个id去搜索资源的时候会映射到0x7f010000这个id去它的overlay apk(OverlayDemoOverlay.apk)里面搜索。

新增资源

安卓的RRO机制应该是不支持新增资源的,所以我在overlay apk里面加入了目标apk没有的app_name2字符串之后dump的时候打印W failed to find resource 'string/app_name2'

但我们可以通过代码在指定的overlay包里面搜索资源id然后读取,实现新增资源的目的:

id = getResources().getIdentifier("app_name2", "string", "me.linjw.demo.overlay.overlay");
if (id != 0) {
    sb.append("app_name2=" + getResources().getString(id));
}else {
    sb.append("app_name2=?");
}

完整Demo

完整的Demo已经上传到Github

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,839评论 6 482
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,543评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 153,116评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,371评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,384评论 5 374
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,111评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,416评论 3 400
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,053评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,558评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,007评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,117评论 1 334
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,756评论 4 324
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,324评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,315评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,539评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,578评论 2 355
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,877评论 2 345

推荐阅读更多精彩内容