Android FrameWork之SytemServer進(jìn)程fork示例
1、Linux的fork
在Linux平臺我們可以通過fork系統(tǒng)調(diào)用來創(chuàng)建一個(gè)新的進(jìn)程,這個(gè)新的進(jìn)程將會(huì)擁有原始進(jìn)程的一份副本,包括代碼、數(shù)據(jù)、內(nèi)存等等。唯一的區(qū)別是新的進(jìn)程擁有一個(gè)新的ID,使得它成為一個(gè)獨(dú)立的進(jìn)程,運(yùn)行自己的代碼。
fork()系統(tǒng)調(diào)用會(huì)返回兩次,在原始進(jìn)程中會(huì)返回進(jìn)程ID,在新的進(jìn)程中會(huì)返回0。兩個(gè)進(jìn)程可以執(zhí)行相同的任務(wù),也可以按照需要執(zhí)行不同的代碼。
fork的例子
#include <stdio.h>
#include <unistd.h>
int main(){
printf("main start \n");
int result = fork();
pid_t pid = getpid();
printf("pid is %d and result is %d\n",pid,result );
return 0;
}main start
pid is 11647 and result is 11648
pid is 11648 and result is 0
我們看到main start只打印了一次,子進(jìn)程從fork之后開始分叉執(zhí)行。
第二行當(dāng)前pid是11647 ,fork的結(jié)果是11648,表示當(dāng)前在父進(jìn)程執(zhí)行
第三行當(dāng)前pid是11648,恰好是父進(jìn)程fork返回的進(jìn)程ID,result的結(jié)果是0,也說明了本次的打印是在子進(jìn)程中執(zhí)行的。
2、SystemServer進(jìn)程的fork
接著分析上次的ZygoteInit的源碼。
frameworks/base/core/java/com/android/internal/os/ZygoteInit.java
if (startSystemServer) {
Runnable r = forkSystemServer(abiList, socketName, zygoteServer);
// {@code r == null} in the parent (zygote) process, and {@code r != null} in the
// child (system_server) process.
//如果是在system_server進(jìn)程中執(zhí)行,則返回Runnable ,并執(zhí)行r.run函數(shù)
if (r != null) {
r.run();
return;
}
}
/**
* Prepare the arguments and forks for the system server process.
*
* Returns an {@code Runnable} that provides an entrypoint into system_server code in the
* child process, and {@code null} in the parent.
*/
private static Runnable forkSystemServer(String abiList, String socketName,
ZygoteServer zygoteServer) {
int pid;
try {
parsedArgs = new ZygoteConnection.Arguments(args);
ZygoteConnection.applyDebuggerSystemProperty(parsedArgs);
ZygoteConnection.applyInvokeWithSystemProperty(parsedArgs);
/* Request to fork the system server process */
//在這里請求fork system server 進(jìn)程
pid = Zygote.forkSystemServer(
parsedArgs.uid, parsedArgs.gid,
parsedArgs.gids,
parsedArgs.debugFlags,
null,
parsedArgs.permittedCapabilities,
parsedArgs.effectiveCapabilities);
} catch (IllegalArgumentException ex) {
throw new RuntimeException(ex);
}
/* For child process */
//如果pid為0,說明是在子進(jìn)程中執(zhí)行,因此關(guān)掉zygote進(jìn)程繼承下來的Socket
if (pid == 0) {
if (hasSecondZygote(abiList)) {
waitForSecondaryZygote(socketName);
}
zygoteServer.closeServerSocket();
return handleSystemServerProcess(parsedArgs);
}
調(diào)用Zygote類的forkSystemServe函數(shù)請求fork子進(jìn)程,如果pid為0,說明是在子進(jìn)程中執(zhí)行,因此關(guān)掉zygote進(jìn)程繼承下來的Socket。
forkSystemServer的代碼
public static int forkSystemServer(int uid, int gid, int[] gids, int debugFlags,
int[][] rlimits, long permittedCapabilities, long effectiveCapabilities) {
......
int pid = nativeForkSystemServer(
uid, gid, gids, debugFlags, rlimits, permittedCapabilities, effectiveCapabilities);
.......
return pid;
}
native private static int nativeForkSystemServer(int uid, int gid, int[] gids, int debugFlags,
int[][] rlimits, long permittedCapabilities, long effectiveCapabilities);
在forkSystemServer函數(shù)中又調(diào)用了nativeForkSystemServer方法,這里要通過JNI調(diào)用Native函數(shù)
frameworks/base/core/jni/com_android_internal_os_Zygote.cpp
jint com_android_internal_os_Zygote_nativeForkSystemServer
ForkAndSpecializeCommon
static jint com_android_internal_os_Zygote_nativeForkSystemServer(
JNIEnv* env, jclass, uid_t uid, gid_t gid, jintArray gids,
jint debug_flags, jobjectArray rlimits, jlong permittedCapabilities,
jlong effectiveCapabilities) {
pid_t pid = ForkAndSpecializeCommon(env, uid, gid, gids,
debug_flags, rlimits,
permittedCapabilities, effectiveCapabilities,
MOUNT_EXTERNAL_DEFAULT, NULL, NULL, true, NULL,
NULL, NULL, NULL);
if (pid > 0) {
.....
}
return pid;
}// Utility routine to fork zygote and specialize the child process.
static pid_t ForkAndSpecializeCommon(JNIEnv* env, uid_t uid, gid_t gid, jintArray javaGids,
jint debug_flags, jobjectArray javaRlimits,
jlong permittedCapabilities, jlong effectiveCapabilities,
jint mount_external,
jstring java_se_info, jstring java_se_name,
bool is_system_server, jintArray fdsToClose,
jintArray fdsToIgnore,
jstring instructionSet, jstring dataDir) {
...........
pid_t pid = fork();
.....
}在ForkAndSpecializeCommon函數(shù)中我們看到了fork()的系統(tǒng)調(diào)用,這樣system server進(jìn)程就會(huì)被fork出來了。
而在最開始的代碼我們看到最終調(diào)用forkSystemServer之后返回了一個(gè)Runnable r對象。
Runnable r = forkSystemServer(abiList, socketName, zygoteServer);
// {@code r == null} in the parent (zygote) process, and {@code r != null} in the
// child (system_server) process.
//如果是在system_server進(jìn)程中執(zhí)行,則返回Runnable ,并執(zhí)行r.run函數(shù)
if (r != null) {
r.run();
return;
}
如果是在system_server進(jìn)程中執(zhí)行的話會(huì)返回一個(gè)Runnable r對象,并執(zhí)行r.run()函數(shù),也就是之后system_server進(jìn)程的執(zhí)行主要流程就在Runnable 的run函數(shù)中
以上就是Android FrameWork之SytemServer進(jìn)程fork示例的詳細(xì)內(nèi)容,更多關(guān)于Android FrameWork SytemServer的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Android使用xUtils3.0實(shí)現(xiàn)文件上傳
這篇文章主要為大家詳細(xì)介紹了Android使用xUtils3.0實(shí)現(xiàn)文件上傳的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-11-11
Android開發(fā)之開發(fā)者頭條(二)實(shí)現(xiàn)左滑菜單
本文給大家介紹Android開發(fā)之開發(fā)者頭條(二)實(shí)現(xiàn)左滑菜單,主要用android自帶的DrawerLayout控件實(shí)現(xiàn)的此功能,具體實(shí)現(xiàn)過程請參考下本文2016-04-04
Android平臺預(yù)置GMS包后關(guān)機(jī)鬧鐘失效問題及解決方法
這篇文章主要介紹了Android平臺預(yù)置GMS包后,關(guān)機(jī)鬧鐘失效,本文給大家分享問題原因及解決方法,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-09-09
Android使用BroadcastReceiver實(shí)現(xiàn)手機(jī)開機(jī)之后顯示畫面的功能
這篇文章主要介紹了Android使用BroadcastReceiver實(shí)現(xiàn)手機(jī)開機(jī)之后顯示畫面的功能,結(jié)合實(shí)例形式分析了BroadcastReceiver的具體使用技巧及實(shí)現(xiàn)開機(jī)畫面的相關(guān)功能代碼,需要的朋友可以參考下2016-01-01
Android實(shí)現(xiàn)登錄郵箱的自動(dòng)補(bǔ)全功能
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)登錄郵箱的自動(dòng)補(bǔ)全功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-04-04
Flutter實(shí)現(xiàn)webview與原生組件組合滑動(dòng)的示例代碼
這篇文章主要介紹了Flutter實(shí)現(xiàn)webview與原生組件組合滑動(dòng)的示例代碼,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-03-03
Flutter模仿實(shí)現(xiàn)微信底部導(dǎo)航欄流程詳解
這篇文章主要介紹了Flutter模仿實(shí)現(xiàn)微信底部導(dǎo)航欄流程,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)吧2023-05-05
Android 給圖片加上水印的示例代碼(支持logo+文字)
本篇文章主要介紹了Android 給圖片加上水印的示例代碼(支持logo+文字),具有一定的參考價(jià)值,有興趣的可以了解一下2017-08-08

