问题场景:
之前项目需求要加入抓取系统内核打印的功能。同时也需要将内核打印级别调到最大级别,但在Android的应用层或Frameworks层代码是无法执行下面这条命令的。
echo 7 > /proc/sys/kernel/printk
查阅的相关资料说是可以使用Runtime.getRuntime().exec去执行,但是实际从命令后会发现报错无法没有root权限。而我们这时恰恰可以通过使用service的方式去启动。
什么是Service?
Android6.1以后增加了Selinux,对安全性的管理更加严格了,但是有时候项目中需要需要执行su操作,来获取系统的权限。但在android原生系统中,只有root权限和shell权限下才可以使用su命令。既然Android上层无法执行这种命令,那我们能不能换个地方去处理呢?答案是可以的,底层是就可以执行su权限的命令。通过这种思路,我们可以通过底层放一个由init启动的service,然后使用on property监听系统属性的变化去执行这个service。听到系统属性是不是很熟悉,没错就是我们代码中使用到的SystemProperties.setXXX这种形式去设置的值。
service其实是一个后台的程序,这个程序在init进程中被启动,如果退出了可以由系统重新启动(可选择)。
服务的格式如下:
service <name> <pathname> [<argument> ]*
<option>
<option>
...
知悉了这种方法,我们就可以在底层上添加我们需要执行的脚本。在脚本中我们可以执行一些需要root权限才可以执行的命令。
解决方案:
针对这个需要修改内核打印级别的命令,我们先编写一个set_loglevelmax文件,以下命令就是我们需要的修改内核打印代码。
/system/bin/echo 7 > /proc/sys/kernel/printk
然后其预置到system/bin目录下。然后在对应的系统软件所使用的rc文件,在其中添加以下代码。
on property:vendor.logmode.enable=1
start service_setloglevel
on property:vendor.logmode.enable=0
stop service_setloglevel
service service_setloglevel /system/bin/sh /system/bin/set_loglevelmax.sh
class main
user root
group root
console
oneshot
seclabel u:r:shell:s0
在应用层我们通过修改vendor.logmode.enable这个属性来控制服务的启动与关闭,当这个服务启动的时候也就会执行我们的shell脚本。
总结:
本文针对的AOSP开发,通过本文介绍的方法,我们可以通过在rc文件添加service去执行某些我们在应用层操作不到的权限或执行某些通过底层操作更有效,便捷的操作。