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