美文网首页Android
Android之zygote源码剖析(一)

Android之zygote源码剖析(一)

作者: Lee_5566 | 来源:发表于2021-03-04 20:10 被阅读0次
    image.png

    目录

    Android之zygote源码剖析(一)
    Android之zygote源码剖析(二)
    Android之zygote源码剖析(三)
    Android之SystemServer介绍(一)
    Android之SystemServer介绍(二)
    Android之Launcher介绍(一))

    zygote源码剖析

    源码路径:frameworks/base/cmds/app_process/
    在Android.mk文件中可以看到:

    ……
    LOCAL_WHOLE_STATIC_LIBRARIES := $(app_process_common_static_libs)
    
    LOCAL_MODULE:= app_process
    LOCAL_MULTILIB := both
    LOCAL_MODULE_STEM_32 := app_process32
    LOCAL_MODULE_STEM_64 := app_process64
    ……
    

    编译出来的程序名字为app_process。

    它的启动在init.rc中。

    init.rc文件是系统启动时候init进程解析的文件,会启动一系列的服务。

    ……
    import /init.environ.rc
    import /init.usb.rc
    import /init.${ro.hardware}.rc
    import /vendor/etc/init/hw/init.${ro.hardware}.rc
    import /init.usb.configfs.rc
    import /init.${ro.zygote}.rc
    ……
    

    其中的import /init.${ro.zygote}.rc就是来导入init.zygote32.rc文件的这里的ro.zygote是个变量,会根据编译的不同选择不同的rc文件。

    zygote的rc文件大约有以下几个:


    image.png

    我们主要来看下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,对应的程序是/system/bin/app_process,后面的便是传入的参数。

    下面来分析下源码(源码中有注释):

    ……
    int main(int argc, char* const argv[])
    {
        // 创建runtime
        // argv[0]就是程序本身
        AppRuntime runtime(argv[0], computeArgBlockSize(argc, argv));
    
        //忽略argv[0]
        argc--;
        argv++;
    
        const char* spaced_commands[] = { "-cp", "-classpath" };
        bool known_command = false;
    
        int i;
        // 执行参数(-Xzygote /system/bin --zygote --start-system-server)
        for (i = 0; i < argc; i++) {
            if (known_command == true) {
              runtime.addOption(strdup(argv[i]));
              ALOGV("app_process main add known option '%s'", argv[i]);
              known_command = false;
              continue;
            }
    
            // 对比参数中是否包含"-cp"和"-classpath"
            // 从上文中看到是没有包含的
            for (int j = 0;
                 j < static_cast<int>(sizeof(spaced_commands) / sizeof(spaced_commands[0]));
                 ++j) {
              if (strcmp(argv[i], spaced_commands[j]) == 0) {
                known_command = true;
                ALOGV("app_process main found known command '%s'", argv[i]);
              }
            }
    
    
            if (argv[i][0] != '-') {
                break;
            }
    
            if (argv[i][1] == '-' && argv[i][2] == 0) {
                ++i; // Skip --.
                break;
            }
    
            // 为runtime添加参数
            // 所以执行参数(-Xzygote /system/bin --zygote --start-system-server)时候
            // -Xzygote加入runtime后执行到/system/bin跳出循环
            runtime.addOption(strdup(argv[i]));
            ALOGV("app_process main add option '%s'", argv[i]);
        }
    
        bool zygote = false;
        bool startSystemServer = false;
        bool application = false;
        String8 niceName;
        String8 className;
    
        // 跳过/system/bin
        ++i; 
        while (i < argc) {
            // 第一次执行时候 arg == --zygote
            // 第二次执行时候 arg == --start-system-server
            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;
            }
        }
    
        // 重新构建参数
        Vector<String8> args;
        // 之前代码可以看出并未赋值给className,所以className为空
        if (!className.isEmpty()) {
            // We're not in zygote mode, the only argument we need to pass
            // to RuntimeInit is the application argument.
            //
            // The Remainder of args get passed to startup class main(). Make
            // copies of them before we overwrite them with the process name.
            args.add(application ? String8("application") : String8("tool"));
            runtime.setClassNameAndArgs(className, argc - i, argv + i);
    
            if (!LOG_NDEBUG) {
              String8 restOfArgs;
              char* const* argv_new = argv + i;
              int argc_new = argc - i;
              for (int k = 0; k < argc_new; ++k) {
                restOfArgs.append("\"");
                restOfArgs.append(argv_new[k]);
                restOfArgs.append("\" ");
              }
              ALOGV("Class name = %s, args = %s", className.string(), restOfArgs.string());
            }
        } else {
            // 执行进入
            maybeCreateDalvikCache();
    
            // 参数添加start-system-server
            if (startSystemServer) {
                args.add(String8("start-system-server"));
            }
    
            char prop[PROP_VALUE_MAX];
            if (property_get(ABI_LIST_PROPERTY, prop, NULL) == 0) {
                LOG_ALWAYS_FATAL("app_process: Unable to determine ABI list from property %s.",
                    ABI_LIST_PROPERTY);
                return 11;
            }
    
            String8 abiFlag("--abi-list=");
            abiFlag.append(prop);
            args.add(abiFlag);
    
            // In zygote mode, pass all remaining arguments to the zygote
            // main() method.
            for (; i < argc; ++i) {
                args.add(String8(argv[i]));
            }
        }
    
        if (!niceName.isEmpty()) {
            runtime.setArgv0(niceName.string(), true /* setProcName */);
        }
    
        if (zygote) {
            // 开始执行ZygoteInit
            runtime.start("com.android.internal.os.ZygoteInit", args, zygote);
        } else if (className) {
            // 执行RuntimeInit
            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.");
        }
    }
    ……
    
    image.png

    相关文章

      网友评论

        本文标题:Android之zygote源码剖析(一)

        本文链接:https://www.haomeiwen.com/subject/mhbxqltx.html