Android 13 sysprop_library新增属性

前提

我们在androidP及之前的版本,平台侧及应用层开发习惯于通过调用(或者反射)SystemProperties系统API的方式进行系统属性的读写。Android R以后,平台侧代码采用了一种将系统属性封装成类方法的形式供开发者调用。

Android R以前读写属性

1:mk文件或者system.prop文件新增属性
mk文件:
PRODUCT_PROPERTY_OVERRIDES += persist.vendor.usb.config=adb
*.prop文件
persist.vendor.usb.config=true
2:给属性配置上下文
persist.vendor.usb.config u:object_r:usb_prop:s0
3:通过SystemProperties进行读写
SystemProperties.set("persist.vendor.usb.config", "adb,mtp");
String value = SystemProperties.get("persist.vendor.usb.config");

Android R以后读写属性:

例如TelephonyManager.java为例,读写属性:gsm.sim.operator.iso-country

1:设置sim卡国家码
@UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023)
public void setSimCountryIsoForPhone(int phoneId, String iso) {
    if (SubscriptionManager.isValidPhoneId(phoneId)) {
        List<String> newList = updateTelephonyProperty(
                TelephonyProperties.icc_operator_iso_country(), phoneId, iso);
        TelephonyProperties.icc_operator_iso_country(newList);
    }
}
2:读取国家码
@UnsupportedAppUsage
public static String getSimCountryIsoForPhone(int phoneId) {
    return getTelephonyProperty(phoneId, TelephonyProperties.icc_operator_iso_country(), "");
}

以上的属性定义在TelephonyProperties

//QSSI.13\frameworks\base\telephony\java\com\android\internal\telephony\TelephonyProperties.java
public interface TelephonyProperties
{
....
/** ISO country code equivalent for the SIM provider's country code*/
static String PROPERTY_ICC_OPERATOR_ISO_COUNTRY = "gsm.sim.operator.iso-country";
....
}

我们看到TelephonyProperties为接口,并且接口类中未定义实现icc_operator_iso_country(),只定义了属性PROPERTY_ICC_OPERATOR_ISO_COUNTRY,其实icc_operator_iso_country定义在如下文件

//QSSI.13/system/libsysprop/srcs/android/sysprop/TelephonyProperties.sysprop
# ISO country code equivalent for the SIM provider's country code.
# Indexed by phone ID
prop {
    api_name: "icc_operator_iso_country"
    type: StringList
    scope: Internal
    access: ReadWrite
    prop_name: "gsm.sim.operator.iso-country"
}

/system/libsysprop/srcs/android/sysprop/,发现这里面有很多文件,文件后缀均为sysprop。

我们可以先看下bp文件

sysprop_library {
    name: "PlatformProperties",
    srcs: ["**/*.sysprop"],
    property_owner: "Platform",
    api_packages: ["android.sysprop"],

    apex_available: [
        "//apex_available:platform",
        "com.android.art",
        "com.android.art.debug",
        "com.android.bluetooth",
        "com.android.tethering",
    ],
    cpp: {
        min_sdk_version: "S",
    },
    vendor_available: true,
}

可以看到该模块会被编译为jar包为:PlatformProperties.jar,该模块对应的jar包位置为:

QSSI.13\out\soong\.intermediates\system\libsysprop\srcs\PlatformProperties\android_common\javac\PlatformProperties.jar

我们可以把该jar包放到AndroidStudio中引用看下,可以到TelephonyProperties.sysprop会被转换为TelephonyProperties.java

我们看下代码就可以看到原来也是调用SystemProperties

这个用法类似于aidl,说明编译脚本会将我们的sysprop文件转换为java文件,并生成对应的方法。此工作是在以下路径的脚本文件处理

//QSSI.13/build/soong/sysprop/sysprop_library.go
...
func (g *syspropJavaGenRule) GenerateAndroidBuildActions(ctx android.ModuleContext) {
    var checkApiFileTimeStamp android.WritablePath

    ctx.VisitDirectDeps(func(dep android.Module) {
        if m, ok := dep.(*syspropLibrary); ok {
            checkApiFileTimeStamp = m.checkApiFileTimeStamp
        }
    })

    for _, syspropFile := range android.PathsForModuleSrc(ctx, g.properties.Srcs) {
        srcJarFile := android.GenPathWithExt(ctx, "sysprop", syspropFile, "srcjar")

        ctx.Build(pctx, android.BuildParams{
            Rule:        syspropJava,
            Description: "sysprop_java " + syspropFile.Rel(),
            Output:      srcJarFile,
            Input:       syspropFile,
            Implicit:    checkApiFileTimeStamp,
            Args: map[string]string{
                "scope": g.properties.Scope,
            },
        })

        g.genSrcjars = append(g.genSrcjars, srcJarFile)
    }
}
...

从上面的编译逻辑中可以窥知,编译过程中会输出srcJarFile类型的文件。例如TelephonyProperties.sysprop文件会被转换为TelephonyProperties.srcjar java文件

out\soong.intermediates\system\libsysprop\srcs\PlatformProperties_java_gen\gen\sysprop\system\libsysprop\srcs\android\sysprop\TelephonyProperties.srcjar

至此我们知道编译脚本会对.sysprop文件 转换,我们只需要按照.sysprop文件格式定义对应的api即可

下面演示下如何新增一个prop

1:在TelephonyProperties.sysprop 中定义属性
//QSSI.13/system/libsysprop/srcs/android/sysprop/TelephonyProperties.sysprop
# set default dns list
prop {
    api_name: "sim_default_dns"
    type: StringList
    scope: Public
    access: ReadWrite
    prop_name: "sim.default.dns"
}
scope为public 可被外部读写,scope: Internal 则只可被同包下的读写,并且public需要更新PlatformProperties-current.txt Internal 需要更新PlatformProperties-latest.txt文件
2:更新api接口
我们新增了属性,需要执行如下指令,更新api文件,具体的详见脚本文件:/build/soong/scripts/freeze-sysprop-api-files.sh
m PlatformProperties-dump-api && rm -rf system/libsysprop/srcs/api/PlatformProperties-current.txt && cp -f out/soong/.intermediates/system/libsysprop/srcs/PlatformProperties_sysprop_library/api-dump.txt system/libsysprop/srcs/api/PlatformProperties-current.txt
执行上述指令以后会更新如下文件
//QSSI.13/system/libsysprop/srcs/api/PlatformProperties-current.txt
prop {
    api_name: "sim_default_dns"
    type: StringList
    access: ReadWrite
    prop_name: "sim.default.dns"
  }
3:我们新增了属性为sim.default.dns,我们还需要增加selinux权限,即为此属性配置上下文,可直接使用之前系统定义好的上下文
//QSSI.13/system/sepolicy/private/property_contexts
sim.default.dns                u:object_r:telephony_status_prop:s0 exact string
//QSSI.13/system/sepolicy/prebuilts/api/33.0/private/property_contexts
sim.default.dns                u:object_r:telephony_status_prop:s0 exact string
4:在如下接口中新增属性
QSSI.13\frameworks\base\telephony\java\com\android\internal\telephony\TelephonyProperties.java
/**
 * PROPERTY_SIM_DEFAULT_DNS is to set the default DNS
 */
static String PROPERTY_SIM_DEFAULT_DNS = "sim.default.dns";
5:如上以后我们可以重新编译PlatformProperties
make PlatformProperties

然后把:out\soong.intermediates\system\libsysprop\srcs\PlatformProperties_java_gen\gen\sysprop\system\libsysprop\srcs\android\sysprop\TelephonyProperties.srcjar

路径下的jar包拖到AS中,可以看到系统为我们生成了对应的方法

然后我们就可以直接通过如下代码去读写了

 /**
     * Set TelephonyProperties.default_dns for dns.
     *
     * @hide
     */
    public void setSimDnsServers(int phoneId, String dnsStr) {
        if (SubscriptionManager.isValidPhoneId(phoneId)) {
            List<String> newList = new ArrayList();
            if (!TextUtils.isEmpty(dnsStr)) {
                List<String> tList = updateTelephonyProperty(TelephonyProperties.sim_default_dns(), phoneId, dnsStr);
                newList.addAll(tList);
            }
            for (int i = 0; i < newList.size(); i++) {
                Log.e(TAG, "setSimDns...." + newList.get(i));
            }
            Log.e(TAG, "setSimDnsList...." + dnsStr + ":" + newList);
            TelephonyProperties.sim_default_dns(newList);
        }
    }

    /**
     * get TelephonyProperties.default_dns
     *
     * @hide
     */
    public String getSimDnsServers(int phoneId) {
        return getTelephonyProperty(phoneId, TelephonyProperties.sim_default_dns(), "");
    }
相关推荐
码上一元2 小时前
SpringBoot自动装配原理解析
java·spring boot·后端
计算机-秋大田2 小时前
基于微信小程序的养老院管理系统的设计与实现,LW+源码+讲解
java·spring boot·微信小程序·小程序·vue
大耳猫3 小时前
主动测量View的宽高
android·ui
魔道不误砍柴功4 小时前
简单叙述 Spring Boot 启动过程
java·数据库·spring boot
失落的香蕉4 小时前
C语言串讲-2之指针和结构体
java·c语言·开发语言
枫叶_v4 小时前
【SpringBoot】22 Txt、Csv文件的读取和写入
java·spring boot·后端
wclass-zhengge4 小时前
SpringCloud篇(配置中心 - Nacos)
java·spring·spring cloud
路在脚下@4 小时前
Springboot 的Servlet Web 应用、响应式 Web 应用(Reactive)以及非 Web 应用(None)的特点和适用场景
java·spring boot·servlet
黑马师兄4 小时前
SpringBoot
java·spring
数据小小爬虫4 小时前
如何用Java爬虫“偷窥”淘宝商品类目API的返回值
java·爬虫·php