You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 

3.5 KiB

应用进程的启动
  1. 了解 Linux 下进程启动的方式
  2. 熟悉应用进程启动的基本流程
  3. 深入理解应用进程启动的原理

进程启动方式

if((pid = fork()) < 0){
	// error
}else if(pid == 0){
	// child process
    // 子进程共享父进程的资源
}else {
	// parent process
}
if((pid = fork()) < 0){
	// error
}else if(pid == 0){
	// child process
    // 子进程资源完全被 path 替换掉了,这个 path 是个二进程程序的路径
	execve(path,argv,env);
}else {
	// parent process
}

应用进程启动原理

  1. 什么时候触发的进程启动?谁发起的?
  2. 进程是谁启动的?怎么启动的?

触发进程启动

应用进程的启动是被动的,即在启动组件的时候,如果发现该组件所在的进程未被启动,那么这时候就会去启动进程。该操作是 Framework 层去做的:

ProcessRecord app = getProcessRecordLocked(...) {
    // app.thread 是 IApplicationThread,是应用层的 binder 句柄
    // 为了是 AMS 和应用进行双向调用
	if(app != null && app.thread != null) {
		// 进程已经启动了,这里就可以启动组件了
		return;
	}
	startProcessLocked(r.processName);
}

应用进程启动后的入口函数:

public static void main(String[] args) {
	Looper.prepareMainLooper();
	
	ActivityThread thread = new ActivityThread();
	thread.attach(false);
	
	Looper.loop();
	throw new RuntimeException("...");
}
// 获取 AMS 的 binder 对象
final IActivityManager mgr = ActivityManagerNative.getDefault();
mgr.attachApplication(mAppThread);

对于 AMS 来说,一个应用进程怎么样才算是启动完成呢?

  1. AMS 向 Zygote 发起启动应用进程的请求,然后 Zygote 启动并返回 pid 给 AMS
  2. 应用进程启动完成之后,告诉 AMS 注册 ApplicationThread

进程是怎么启动的?

启动进程是在 AMS 里面的 startProcessLocked 函数,这个函数不会直接去启动进程,而是只是发起启动进程:

  1. 打开本地 socket,openZygoteSocketIfNeeded 函数
  2. 通过 Socket 发送参数列表,参数包括应用进程启动之后要执行的类名,zygoteSendArgsAndGetResult 函数
  3. 发送完参数列表之后等待返回创建的进程的 pid 即可

那么 Zygote 里面是如何处理 Socket 请求的?

Zygote 会在一个 for 循环里面不断看有没有消息过来,如果有消息过来就执行 runOnce 函数:

boolean runOnce() {
    // 读取 Zygote 发送过来的参数列表
	String[] args = readArgumentList();
    // 创建应用进程
	int pid = Zygote.forkAndSpecialize(...);
	if(pid == 0) {
        // 子进程,执行 ActivityThread#main 函数
		handleChildProc(...);
		return true;
	} else {
        // 父进程把 pid 通过 Socket 在写回去
		return handlePArentProc(pid, ...);
	}
}

课堂作业

  1. 为什么是 zygote 来创建进程,而不是 systemserver?
  2. AMS 和 zygote 为什么不用 binder 通信?

总结

应用进程是怎么启动的呢?

  1. 应用进程是什么时候起的?
  2. 由 ASM 向 zygote 发起的,通过 socket 通信
  3. zygote fork 出应用进程,执行 ActivityThread#main 入口函数,这个入口函数的类名是 AMS 通过 Socket 发给 zygote 的
  4. 进程启动之后要向 AMS 报告,注册 ApplicationThread,这样整个启动才算结束,因为 AMS 有了该报告,应用进程才算是可用,AMS 才能继续启用应用的各个组件