目的
最近打算回顾一下以前研究Framework时遇到的问题。
自己发现Android演进到8.0后,许多流程又都发生了改变,
于是打算在之前博客的基础上,结合新的代码重新梳理一遍。
本篇博客主要记录一下Android 8.0中的init流程。
背景
当linux内核启动之后,运行的第一个进程是init。
这个进程是一个守护进程,它的生命周期贯穿整个linux 内核运行的始终,
linux中所有其它的进程的共同始祖均为init进程。
Android系统是运作在linux内核上的,为了启动并运行整个android系统,
google实现了android系统的init进程。
Android init进程的入口文件在system/core/init/init.cpp中。
在main函数中,按顺序主要进行了以下工作:
一、判断及增加环境变量
int main(int argc, char** argv) {
//根据参数,判断是否需要启动的ueventd和watchdogd
if (!strcmp(basename(argv[0]), "ueventd")) {
return ueventd_main(argc, argv);
}
if (!strcmp(basename(argv[0]), "watchdogd")) {
return watchdogd_main(argc, argv);
}
//若紧急重启,则安装对应的消息处理器
if (REBOOT_BOOTLOADER_ON_PANIC) {
install_reboot_signal_handlers();
}
//增加环境变量
add_environment("PATH", _PATH_DEFPATH);
二、创建文件系统目录并挂载相关的文件系统
//第一次进入时,is_first_stage为true,对应于初始化的第一阶段
bool is_first_stage = (getenv("INIT_SECOND_STAGE") == nullptr);
if (is_first_stage) {
//用于记录启动时间
boot_clock::time_point start_time = boot_clock::now();
//清除屏蔽字(file mode creation mask),保证新建的目录的访问权限不受屏蔽字影响。
umask(0);
// Get the basic filesystem setup we need put together in the initramdisk
// on / and then we'll let the rc file figure out the rest.
mount("tmpfs", "/dev", "tmpfs", MS_NOSUID, "mode=0755");
mkdir("/dev/pts", 0755);
mkdir("/dev/socket", 0755);
mount("devpts", "/dev/pts", "devpts", 0, NULL);
#define MAKE_STR(x) __STRING(x)
mount("proc", "/proc", "proc", 0, "hidepid=2,gid=" MAKE_STR(AID_READPROC));
//这个是8.0新增的, 收紧了cmdline目录的权限
// Don't expose the raw commandline to unprivileged processes.
chmod("/proc/cmdline", 0440);
//8.0增加了个用户组
gid_t groups[] = { AID_READPROC };
setgroups(arraysize(groups), groups);
mount("sysfs", "/sys", "sysfs", 0, NULL);
//以下这部分也是8.0新增的
mount("selinuxfs", "/sys/fs/selinux"