Android FrameWork之SytemServer進(jìn)程fork示例
1、Linux的fork
在Linux平臺(tái)我們可以通過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 */ //在這里請(qǐng)求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ù)請(qǐng)求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對(duì)象。
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對(duì)象,并執(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的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Android使用xUtils3.0實(shí)現(xiàn)文件上傳
這篇文章主要為大家詳細(xì)介紹了Android使用xUtils3.0實(shí)現(xiàn)文件上傳的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-11-11Android開發(fā)之開發(fā)者頭條(二)實(shí)現(xiàn)左滑菜單
本文給大家介紹Android開發(fā)之開發(fā)者頭條(二)實(shí)現(xiàn)左滑菜單,主要用android自帶的DrawerLayout控件實(shí)現(xiàn)的此功能,具體實(shí)現(xiàn)過程請(qǐng)參考下本文2016-04-04Android平臺(tái)預(yù)置GMS包后關(guān)機(jī)鬧鐘失效問題及解決方法
這篇文章主要介紹了Android平臺(tái)預(yù)置GMS包后,關(guān)機(jī)鬧鐘失效,本文給大家分享問題原因及解決方法,對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-09-09Android使用BroadcastReceiver實(shí)現(xiàn)手機(jī)開機(jī)之后顯示畫面的功能
這篇文章主要介紹了Android使用BroadcastReceiver實(shí)現(xiàn)手機(jī)開機(jī)之后顯示畫面的功能,結(jié)合實(shí)例形式分析了BroadcastReceiver的具體使用技巧及實(shí)現(xiàn)開機(jī)畫面的相關(guān)功能代碼,需要的朋友可以參考下2016-01-01Android實(shí)現(xiàn)登錄郵箱的自動(dòng)補(bǔ)全功能
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)登錄郵箱的自動(dòng)補(bǔ)全功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-04-04Flutter實(shí)現(xiàn)webview與原生組件組合滑動(dòng)的示例代碼
這篇文章主要介紹了Flutter實(shí)現(xiàn)webview與原生組件組合滑動(dòng)的示例代碼,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-03-03Flutter模仿實(shí)現(xiàn)微信底部導(dǎo)航欄流程詳解
這篇文章主要介紹了Flutter模仿實(shí)現(xiàn)微信底部導(dǎo)航欄流程,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)吧2023-05-05Android 給圖片加上水印的示例代碼(支持logo+文字)
本篇文章主要介紹了Android 給圖片加上水印的示例代碼(支持logo+文字),具有一定的參考價(jià)值,有興趣的可以了解一下2017-08-08