一、为什么使用 init.rc 机制
我们知道其他进程都是一个二进制文件,可以直接通过 exec 的命令方式来启动,但是 Android 系统有那么多的 Native 进程,如果都是通过传 exec 在代码中一个一个的启动进程,无疑是一个灾难性的设计。因此在这个基础上 Android 推出了一个 init.rc 的机制,类似通过读取配置文件的方式,来启动不同的进程。
二、init.rc 文件的一般形式
三、init.rc 语句类型
init.rc 是一个配置文件,是由Android初始化语言编写的脚本,主要包含五种类型语句:Action、Command、Service、Option、Import ,接下来就上述语句做分别做叙述。
3.1 Action
动作由一组 命令(Commands)组成,动作还包含了一个 触发器 ,决定了运行这个动作的时机。
Action:通过触发器 trigger ,即通过以 on 开头的语句来决定执行相应的 service 的时机,具体有如下时机:
- on early-init; 在初始化早期阶段触发;
- on init; 在初始化阶段触发;
- on late-init; 在初始化晚期阶段触发;
- on boot/charger: 当系统启动/充电时触发;
- on property:<key>=<value>: 当属性值满足条件时触发;
3.2 Command
Command 是 action 的命令列表中命令,或者是 service 中的选项 onrestart 的参数命令,命令将在所属事件发生时被一个一个的执行。常用命令如下:
- class_start <service_class_name>: 启动属于同一个class的所有服务;
- class_stop <service_class_name> : 停止指定类的服务
- start <service_name>: 启动指定的服务,若已启动则跳过;
- stop <service_name>: 停止正在运行的服务
- setprop <name> <value>:设置属性值
- mkdir <path>:创建指定目录
- symlink <target> <sym_link>: 创建连接到<target>的<sym_link>符号链接;
- write <path> <string>: 向文件path中写入字符串;
- exec: fork并执行,会阻塞init进程直到程序完毕;
- exprot <name> <name>:设定环境变量;
- loglevel <level>:设置log级别
- hostname <name> : 设置主机名
- import <filename> :导入一个额外的init配置文件
3.3 Service
服务 Service ,以 service 开头,由 init 进程启动,一般运行在 init 的一个子进程中,所以启动 service 前需要判断对应的可执行文件是否存在。
命令: service <name><pathname> [ <argument> ]* <option> <option>
- <name>:表示 service 的名称
- <pathname>:表示此 service 所在的路径( service 为可执行文件,所以存在存储路径)
- <argument>:表示启动 service 所带的参数
- <option>:表示对此 service 的约束选项
init 生成的子进程,定义在rc文件,其中每一个service在启动时会通过fork方式生成子进程。
3.4 Option
Options是Service的可选项,与service配合使用
- disabled: 不随class自动启动,只有根据service名才启动;
- oneshot: service退出后不再重启;
- user/group: 设置执行服务的用户/用户组,默认都是root;
- class:设置所属的类名,当所属类启动/退出时,服务也启动/停止,默认为default;
- onrestart:当服务重启时执行相应命令;
- socket: 创建名为/decd
- critical: 在规定时间内该service不断重启,则系统会重启并进入恢复模式
default: 意味着 disabled=false,oneshot=false,critical=false
3.5 Import
用来导入其他的 rc 文件
命令: import <filename>
四、 init.rc 实例
从 Android 5.0 的版本开始,Android支持 64 位的编译,因此 Zygote 本身也支持 32 位和 64 位。通过属性 ro.zygote 来控制不同版本的 zygote 进程启动。
init.rc 位于 "/system/core/rootdir" 目录下, 在这个路径下还包括四个关于 zygote 的 rc 文件。分别是 init.zygote32.rc,init.zygote32_64.rc,init.zygote64.rc,init.zygote64_32.rc,由硬件决定调用哪个文件。
import /init.environ.rc
import /init.usb.rc
import /init.${ro.hardware}.rc
import /vendor/etc/init/hw/init.${ro.hardware}.rc
import /init.usb.configfs.rc
import /init.${ro.zygote}.rc
on early-init
# Set init and its forked children's oom_adj.
write /proc/1/oom_score_adj -1000
......