安卓系统中的系统属性值包括两个部分:文件保存的持久属性和每次开机导入的cache属性。可以在build.prop文件中看到,即out/target/product/~/system/build.prop(以ro.开头的系统属性值具有只读属性)。如下:

Android 只读属性 国家码 安卓只读属性怎么设置_java



1.系统属性值类型及特点

①以“ro.”开头,具有只读属性。一旦设置,属性值不能再改变。

②以“persist.”开头。当设置这个属性值时,其值将写入/data/property。

③以“net.”开头。当设置这个属性值时,“net.change”属性将会自动设置,以加入到最后修改的属性名

④属性“ctrl.start”和“ctrl.stop”是用来启动和停止服务的。



2.使用adb 操作系统属性值

adb shell prop                                                                                 可以查看安卓系统的系统属性

adb shell getprop 属性名                获取单个系统属性值

adb shell setprop 属性名 新的属性值             设置新的属性值(对于非只读属性)

 

3.build.prop文件生成过程

build/core/Makefile相关变量定义:


intermediate_system_build_prop := $(call intermediates-dir-for,ETC, system_build_prop)/build.prop
	   //  obj下的路径out/target/product/~/obj/ETC/system_build_prop_intermediates/build.prop



INSTALLED_BUILD_PROP_TARGET := $(TARGET_OUT)/build.prop 
                                      // system下的路径,即out/target/product/~/system/builg.prop



BUILDINFO_SH := build/tools/buildinfo.sh

build.prop文件中的系统属性值主要包括三个部分—buildinfo.sh文件中的值,systen.prop文件中的值以及ADDITIONAL_BUILD_PROPERTIES:

相关Makfile中的target:

(1) system/build.prop

$(INSTALLED_BUILD_PROP_TARGET): $(intermediate_system_build_prop)
$(INSTALLED_RECOVERYIMAGE_TARGET)
@echo "Target build info: $@"
$(hide) cat $(intermediate_system_build_prop) > $@
								// 将etc下build.prop中的值写入system下的build.prop文件中
ifdef INSTALLED_RECOVERYIMAGE_TARGET
$(hide) echo ro.expect.recovery_id=`cat $(RECOVERYIMAGE_ID_FILE)`

(2) etc/system_build_prop_intermediates/build.prop

$(if $(ADDITIONAL_BUILD_PROPERTIES), \
	$(hide) echo >> $@; \
		    echo "#" >> $@; \
		    echo "# ADDITIONAL_BUILD_PROPERTIES" >> $@; \
		    echo "#" >> $@; )
$(hide) $(foreach line,$(ADDITIONAL_BUILD_PROPERTIES), \
echo "$(line)" >> $@;)
$(hide) cat $(INSTALLED_ANDROID_INFO_TXT_TARGET) | grep 'require version-' | sed -e 's/require version-/ro.build.expect./g' >> $@
$(hide) build/tools/post_process_props.py $@ 
$(PRODUCTS.$(INTERNAL_PRODUCT).PRODUCT_SYSTEM_PROPERTY_BLACKLIST)

(3)build/tools/buildinfo.sh

buildinfo.sh中主要是echo一些属性:

echo "ro.build.id=$BUILD_ID"
echo "ro.build.display.id=$BUILD_DISPLAY_ID"
echo "ro.build.version.incremental=$BUILD_NUMBER"
echo "ro.build.version.sdk=$PLATFORM_SDK_VERSION"
echo "ro.build.version.preview_sdk=$PLATFORM_PREVIEW_SDK_VERSION"
echo "ro.build.version.codename=$PLATFORM_VERSION_CODENAME"
echo "ro.build.version.all_codenames=$PLATFORM_VERSION_ALL_CODENAMES"
........
echo "ro.product.model=$PRODUCT_MODEL"
echo "ro.product.brand=$PRODUCT_BRAND"
echo "ro.product.name=$PRODUCT_NAME"
echo "ro.product.device=$TARGET_DEVICE"
echo "ro.product.board=$TARGET_BOOTLOADER_BOARD_NAME"



4.系统属性值的使用


①Java文件中

获取系统属性值,例如:

String newBuildId = SystemProperties.get("ro.build.id", null);

修改系统属性值,例如:

private static final String PROPERTY_BOOT_SOUNDS = "persist.sys.bootanim.play_sound";
SystemProperties.set(PROPERTY_BOOT_SOUNDS, mBootSounds.isChecked() ? "1" : "0");

②Native代码

获取系统属性值,例如:

property_get("bluetooth.mock_stack", value, "");

设置系统属性值:

property_set("ctl.start", "surfaceflinger");
property_set("persist.sys.actionsafe.width", actionsafeWidth);

5.系统属性值设置分析


开机导入的cache属性主要是通过frameworks/base/core/java/android/os/SystemProperties.java的接口定义


private static native void native_set(String key, String def);

    public static void set(String key, String val) {
        if (key.length() > PROP_NAME_MAX) {
            throw new IllegalArgumentException("key.length > " + PROP_NAME_MAX);
        }
        if (val != null && val.length() > PROP_VALUE_MAX) {
            throw new IllegalArgumentException("val.length > " +
                PROP_VALUE_MAX);
        }
        native_set(key, val);
    }


native_set()实际操作通过JNI调用的是cpp文件对应的接口:

frameworks/base/core/jni/android_os_SystemProperties.cpp

static const JNINativeMethod method_table[] = {
{ 
	……
"native_set", "(Ljava/lang/String;Ljava/lang/String;)V",
(void*) SystemProperties_set },
	 ……
}

int register_android_os_SystemProperties(JNIEnv *env)
{
    return RegisterMethodsOrDie(env, "android/os/SystemProperties", method_table,
                                NELEM(method_table));
}
static void SystemProperties_set(JNIEnv *env, jobject clazz,
                                      jstring keyJ, jstring valJ)
{
    int err;
    const char* key;
    const char* val;

    if (keyJ == NULL) {
        jniThrowNullPointerException(env, "key must not be null.");
        return ;
    }
    key = env->GetStringUTFChars(keyJ, NULL);

    if (valJ == NULL) {
        val = "";       /* NULL pointer not allowed here */
    } else {
        val = env->GetStringUTFChars(valJ, NULL);
    }

    err = property_set(key, val);
    env->ReleaseStringUTFChars(keyJ, key);
    if (valJ != NULL) {
        env->ReleaseStringUTFChars(valJ, val);
    }

    if (err < 0) {
        jniThrowException(env, "java/lang/RuntimeException",
                          "failed to set system property");
    }
}



system/core/init/property_service.cpp

int property_set(const char* name, const char* value) {
    int rc = property_set_impl(name, value);
    if (rc == -1) {
        ERROR("property_set(\"%s\", \"%s\") failed\n", name, value);
    }
    return rc;
}



设置key的value时,需要鉴定属性值的权限,Android5.0以后的系统中使用SELinux进行权限管理。

system/core/init/property_service.cpp

static int check_mac_perms(const char *name, char *sctx, struct ucred *cr)
{                          									      // 判断请求进程是否有权限修改属性值
    char *tctx = NULL;
    int result = 0;
    property_audit_data audit_data;

    ……

    audit_data.name = name;
    audit_data.cr = cr;

    if (selinux_check_access(sctx, tctx, "property_service", "set", reinterpret_cast<void*>(&audit_data)) == 0)
	              // 检查property_contexts中是否定义了目标sctx,将源sctx和目标tctx进行比较,判断是否具有相关权限,sctx是调用set操作进程的sContext
        result = 1;

    freecon(tctx);
 err:
    return result;
}



external/sepolicy/ property_contexts

##########################
# property service keys
net.rmnet0              u:object_r:radio_prop:s0
net.gprs                u:object_r:radio_prop:s0
net.ppp                 u:object_r:radio_prop:s0
		  								 // 表示只有有权限访问Type为radio_prop的资源的进程可以访问
……
gsm.                    u:object_r:radio_prop:s0
net.                    u:object_r:system_prop:s0
dev.                    u:object_r:system_prop:s0
										// 表示只有有权限访问Type为system_prop的资源的进程可以访问