Android FrameWork--Zygote启动

1、Zygote介绍

Zygote是一个进程,当开机引导内核启动之后,首先用户空间的第一个进程Init进程被启动,接着Init进程会启动Zygote进程。Zygote进程作为一个孵化器,主要用于fork新的进程,比如用于系统服务的SystemServer进程,我们的APP进程等。而Zygote在启动的时候会创建启动我们的Java虚拟机,而Zygote通过fork出的进程也会拥有父进程一样的功能,也就拥有我们的Java虚拟机环境。所以我们常说的在安卓系统中每个进程拥有一个虚拟机的缘由是这样来的,当然除了非Zygote fork的进程。

2、Zygote的启动脚本

脚本路径:system/core/rootdir/init.zygote32.rc
脚本内容:

    service zygote /system/bin/app_process -Xzygote /system/bin --zygote --start-system-server
    class main
    priority -20
    user root
    group root readproc
    socket zygote stream 660 root system
    onrestart write /sys/android_power/request_state wake
    onrestart write /sys/power/state on
    onrestart restart audioserver
    onrestart restart cameraserver
    onrestart restart media
    onrestart restart netd
    onrestart restart wificond
    writepid /dev/cpuset/foreground/tasks

脚本的第一行大概可以看出,Zygote的进程名为 zygote ,执行程序为app_process ,class name为main,我们去找到对应的app_process 。
路径:frameworks/base/cmds/app_process
在app_process的路径下有一个app_main.cpp的文件:frameworks/base/cmds/app_process/app_main.cpp
由此可以猜到这里可以作为zygote 进程启动的代码入口。

3、app_process app_main.cpp源码分析

frameworks/base/cmds/app_process/app_main.cpp
我们找到app_main.cpp其main函数

int main(int argc, char* const argv[])
{
    ....
   // 1、创建了AppRuntime 对象
    AppRuntime runtime(argv[0], computeArgBlockSize(argc, argv));
    // Process command line arguments
    // ignore argv[0]
    argc--;
    argv++;

  
    // Parse runtime arguments.  Stop at first unrecognized option.
    bool zygote = false;
    bool startSystemServer = false;
    bool application = false;
    String8 niceName;
    String8 className;

    ++i;  // Skip unused "parent dir" argument.
    while (i < argc) {
        const char* arg = argv[i++];
        if (strcmp(arg, "--zygote") == 0) {
            zygote = true;
            niceName = ZYGOTE_NICE_NAME;
        } else if (strcmp(arg, "--start-system-server") == 0) {
            startSystemServer = true;
        } else if (strcmp(arg, "--application") == 0) {
            application = true;
        } else if (strncmp(arg, "--nice-name=", 12) == 0) {
            niceName.setTo(arg + 12);
        } else if (strncmp(arg, "--", 2) != 0) {
            className.setTo(arg);
            break;
        } else {
            --i;
            break;
        }
    }

 ......
 ......

    if (zygote) {
      //2、调用runtime的start函数
        runtime.start("com.android.internal.os.ZygoteInit", args, zygote);
    } else if (className) {
        runtime.start("com.android.internal.os.RuntimeInit", args, zygote);
    } else {
        fprintf(stderr, "Error: no class name or --zygote supplied.\n");
        app_usage();
        LOG_ALWAYS_FATAL("app_process: no class name or --zygote supplied.");
    }
}

main函数的核心功能主要是创建了AppRuntime runtime对象,并调用start方法,这里传入的className参数是com.android.internal.os.ZygoteInit(全类名)
AppRuntime 继承了AndroidRuntime,我们看AndroidRuntime中的start函数。

void AndroidRuntime::start(const char* className, const Vector<String8>& options, bool zygote)
{
    ....
   ....
   ....
    /* start the virtual machine 3、启动Java虚拟机*/
    JniInvocation jni_invocation;
    jni_invocation.Init(NULL);
    JNIEnv* env;
    if (startVm(&mJavaVM, &env, zygote) != 0) {
        return;
    }
    onVmCreated(env);
    .....
   /*
     * Register android functions.
      *Register android native functions with the VM.
     * 4、向虚拟机注册android本地函数。
     */
    if (startReg(env) < 0) {
        ALOGE("Unable to register all android natives\n");
        return;
    }
  .....
   .....
.....

    /*
     * Start VM.  This thread becomes the main thread of the VM, and will
     * not return until the VM exits.
     */
    char* slashClassName = toSlashClassName(className != NULL ? className : "");
    jclass startClass = env->FindClass(slashClassName);
    if (startClass == NULL) {
        ALOGE("JavaVM unable to locate class '%s'\n", slashClassName);
        /* keep going */
    } else {
        jmethodID startMeth = env->GetStaticMethodID(startClass, "main",
            "([Ljava/lang/String;)V");
        if (startMeth == NULL) {
            ALOGE("JavaVM unable to find main() in '%s'\n", className);
            /* keep going */
        } else {
        //5、使用env对象,通过JNI调用startClass这个类的main函数,
        //而startClass这个类就是之前传起来的 
        //com.android.internal.os.ZygoteInit
            env->CallStaticVoidMethod(startClass, startMeth, strArray);
    }

以上的流程主要就是:

  • 创建AppRuntime对象,调用其start函数。
  • 在start函数中首先调用startVm函数启动Java虚拟机
  • 启动Java虚拟机之后调用startReg函数,注册JNI
  • 通过JNI调用ZygoteInit类的main函数
    之后就从Native进入了Java是世界

4、ZygoteInit.java源码分析

frameworks/base/core/java/com/android/internal/os/ZygoteInit.java

public static void main(String argv[]) {
      //1、创建一个ZygoteServer对象
        ZygoteServer zygoteServer = new ZygoteServer();
        final Runnable caller;
        try {
            ....
            ....

            boolean startSystemServer = false;
            String socketName = "zygote";
            String abiList = null;
            boolean enableLazyPreload = false;
            for (int i = 1; i < argv.length; i++) {
                if ("start-system-server".equals(argv[i])) {
                    startSystemServer = true;
                } else if ("--enable-lazy-preload".equals(argv[i])) {
                    enableLazyPreload = true;
                } else if (argv[i].startsWith(ABI_LIST_ARG)) {
                    abiList = argv[i].substring(ABI_LIST_ARG.length());
                } else if (argv[i].startsWith(SOCKET_NAME_ARG)) {
                    socketName = argv[i].substring(SOCKET_NAME_ARG.length());
                } else {
                    throw new RuntimeException("Unknown command line argument: " + argv[i]);
                }
            }
           //2、注册一个本地ServerSocket,名为 "zygote"
            zygoteServer.registerServerSocket(socketName);


                if (!enableLazyPreload) {
                bootTimingsTraceLog.traceBegin("ZygotePreload");
                EventLog.writeEvent(LOG_BOOT_PROGRESS_PRELOAD_START,
                    SystemClock.uptimeMillis());
              //preload执行预加载,在preload函数中主要调用了 
              //preloadClasses和 preloadResources();加载一些系统资源和系统类,
                preload(bootTimingsTraceLog);
                EventLog.writeEvent(LOG_BOOT_PROGRESS_PRELOAD_END,
                    SystemClock.uptimeMillis());
                bootTimingsTraceLog.traceEnd(); // ZygotePreload
            } else {
                Zygote.resetNicePriority();
            }

     
            if (startSystemServer) {
              //调用fork函数,启创建启动SystemServer进程
                Runnable r = forkSystemServer(abiList, socketName, zygoteServer);

                // {@code r == null} in the parent (zygote) process, and {@code r != null} in the
                // child (system_server) process.
              //这句很关键如果在父进程zygote,则返回null,如果在system_server进程则不会null,
              //所以当在zygote进程的之后,该条件不满足
                if (r != null) {
                    r.run();
                    return;
                }
            }

            Log.i(TAG, "Accepting command socket connections");

            // The select loop returns early in the child process after a fork and
            // loops forever in the zygote.
          //在zygote进程中,调用runSelectLoop进入无线循环,如果是子进程则返回caller 对象
            caller = zygoteServer.runSelectLoop(abiList);
        } catch (Throwable ex) {
            Log.e(TAG, "System zygote died with exception", ex);
            throw ex;
        } finally {
          //如果不是在zygote进程则退出循环进入到这里,关闭socket server
            zygoteServer.closeServerSocket();
        }

        // We're in the child process and have exited the select loop. Proceed to execute the
      
        // command.
      //如果是在子进程中,通过返回的caller 对象,调用caller 的run函数。
        if (caller != null) {
            caller.run();
        }
    }

在ZygoteInit的main函数中主要做了以下工作:

  • 创建并注册一个名为"zygote"本地SocketServer,也就是此时Zygote进程作为LocalSocket的服务端
  • 在preload函数中的调用preloadClasses和preloadResources函数预加载系统类和一些系统资源
  • 调用forkSystemServer创建启动SystemServer进程并返回了一个Runnable r对象,此时就同时拥有了Zygote和SystemServer进程在运行,不通进程它们分别执行不同的代码分支
  • 如果是在SystemServer进程则直接执行 r.run函数并return,在Zygote进程则往下走,调用了zygoteServer中的runSelectLoop函数,进入无限循环,注释也描述了如果是在其他子进程的话,则该循环会退出,返回一个Runnable caller对象。并且往下走调用caller.run函数。

5、小结

Zygote进程的启动主要做了以下一些工作
Native层

  • 创建启动Java虚拟机
  • 注册JNI
  • 通过JNI调用ZygoteInit.java的main函数
    Java层
  • 创建并注册名为"zygote"本地SocketServer,用于进程之间通信,Zygote进程作为服务端,其实目的就是为了和之后的SystemServer进程通信
  • 预加载一些类和一些资源
  • fork子进程SystemServer
  • runSelectLoop进入无线循环
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,242评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,769评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,484评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,133评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,007评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,080评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,496评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,190评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,464评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,549评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,330评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,205评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,567评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,889评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,160评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,475评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,650评论 2 335

推荐阅读更多精彩内容