盒子
盒子
文章目录
  1. RRO的默认状态
    1. 静态 RRO
    2. OverlayConfig
  • overlay命令
  • idmap
  • 新增资源
  • 完整Demo
  • 安卓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里面添加配置:

    1
    2
    3
    4
    5
    <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,如果不预装在这个路径的话就会报错:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    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的实现代码里面定义是这样的:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    // 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。有时候对具体命令的使用有疑问的话可以直接看它的实现代码:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    // 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]是已经启用,[ ]是停用:

    1
    2
    3
    4
    5
    6
    7
    8
    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的资源等情况:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    62
    63
    64
    65
    66
    67
    68
    69
    70
    71
    72
    73
    74
    75
    76
    77
    78
    79
    80
    81
    82
    83
    84
    85
    86
    87
    88
    89
    90
    91
    92
    93
    94
    95
    96
    97
    98
    99
    100
    101
    102
    103
    // 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包的情况下最终读取到的资源值是什么:

    1
    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}命令去打印映射关系:

    1
    2
    3
    4
    5
    6
    7
    8
    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然后读取,实现新增资源的目的:

    1
    2
    3
    4
    5
    6
    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