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.
121 lines
3.5 KiB
121 lines
3.5 KiB
5 years ago
|
---
|
||
|
应用进程的启动
|
||
|
---
|
||
|
|
||
|
1. 了解 Linux 下进程启动的方式
|
||
|
2. 熟悉应用进程启动的基本流程
|
||
|
3. 深入理解应用进程启动的原理
|
||
|
|
||
|
#### 进程启动方式
|
||
|
|
||
|
```c
|
||
|
if((pid = fork()) < 0){
|
||
|
// error
|
||
|
}else if(pid == 0){
|
||
|
// child process
|
||
|
// 子进程共享父进程的资源
|
||
|
}else {
|
||
|
// parent process
|
||
|
}
|
||
|
```
|
||
|
|
||
|
```c
|
||
|
if((pid = fork()) < 0){
|
||
|
// error
|
||
|
}else if(pid == 0){
|
||
|
// child process
|
||
|
// 子进程资源完全被 path 替换掉了,这个 path 是个二进程程序的路径
|
||
|
execve(path,argv,env);
|
||
|
}else {
|
||
|
// parent process
|
||
|
}
|
||
|
```
|
||
|
|
||
|
#### 应用进程启动原理
|
||
|
|
||
|
1. 什么时候触发的进程启动?谁发起的?
|
||
|
2. 进程是谁启动的?怎么启动的?
|
||
|
|
||
|
**触发进程启动**
|
||
|
|
||
|
应用进程的启动是被动的,即在启动组件的时候,如果发现该组件所在的进程未被启动,那么这时候就会去启动进程。该操作是 Framework 层去做的:
|
||
|
|
||
|
```java
|
||
|
ProcessRecord app = getProcessRecordLocked(...) {
|
||
|
// app.thread 是 IApplicationThread,是应用层的 binder 句柄
|
||
|
// 为了是 AMS 和应用进行双向调用
|
||
|
if(app != null && app.thread != null) {
|
||
|
// 进程已经启动了,这里就可以启动组件了
|
||
|
return;
|
||
|
}
|
||
|
startProcessLocked(r.processName);
|
||
|
}
|
||
|
```
|
||
|
|
||
|
应用进程启动后的入口函数:
|
||
|
|
||
|
```java
|
||
|
public static void main(String[] args) {
|
||
|
Looper.prepareMainLooper();
|
||
|
|
||
|
ActivityThread thread = new ActivityThread();
|
||
|
thread.attach(false);
|
||
|
|
||
|
Looper.loop();
|
||
|
throw new RuntimeException("...");
|
||
|
}
|
||
|
```
|
||
|
|
||
|
```java
|
||
|
// 获取 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 函数:
|
||
|
|
||
|
```java
|
||
|
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 才能继续启用应用的各个组件
|