Android之Zygote(下)--- SystemServer详解

Android之Zygote(下)--- SystemServer详解_第1张图片

SystemServer是由zygote.forkSystemServer函数fork出来的子进程,forkSystemServer是一个native函数,

/dalvik/vm/native/dalvik_system_Zygote.c

static void Dalvik_dalvik_system_Zygote_forkSystemServer(
        const u4* args, JValue* pResult)
{
    pid_t pid;
    pid = forkAndSpecializeCommon(args, true);//fork一个子进程
    if (pid > 0) {
        int status;

        LOGI("System server process %d has been created", pid);
        gDvm.systemServerPid = pid; //保存了system_server进程的id
        if (waitpid(pid, &status, WNOHANG) == pid) { //检查刚才创建的子进程是否退出了
            LOGE("System server process %d has died. Restarting Zygote!", pid);
            kill(getpid(), SIGKILL); //system_server推出了
        }
    }
RETURN_INT(pid);

static pid_t forkAndSpecializeCommon(const u4* args, bool isSystemServer)
{
 setSignalHandler();   //设置信号处理函数
}
static void setSignalHandler() 
{
     sa.sa_handler = sigchldHandler;
    err = sigaction (SIGCHLD, &sa, NULL); //子进程死亡的信号
}
static void sigchldHandler(int s)
{
   if (pid == gDvm.systemServerPid) {
            LOG(LOG_INFO, ZYGOTE_LOG_TAG,
                "Exit zygote because system server (%d) has terminated\n", 
                (int) pid);
            kill(getpid(), SIGKILL); //死去的进程是SS zygote直接自杀
        }
}
}

SystemServer的使命(以下简称SS)

Android之Zygote(下)--- SystemServer详解_第2张图片

 可以看到在创建SS后,SS便会调用该方法

    private static void handleSystemServerProcess(
            ZygoteConnection.Arguments parsedArgs)
            throws ZygoteInit.MethodAndArgsCaller {
        //关闭从zygote继承下来的socket
        closeServerSocket();
        RuntimeInit.zygoteInit(parsedArgs.remainingArgs); //调用此函数
    }

Framworks\base\core\java\com.android.internal.os\RuntimeInit.java

zygoteInitNative在AndroidRuntime.cpp中,AndroidRuntime的构造函数getCurRuntime调用了onZygoteInit函数(/frameworks/base/cmds/app_process/app_main.cpp)

public static final void zygoteInit(String[] argv)
            throws ZygoteInit.MethodAndArgsCaller {
   zygoteInitNative();//①native层的初始化
   invokeStaticMain(startClass, startArgs); //②调用systemServer的main函数
}
  • invokeStaticMain,在RunTimeInit中,
    virtual void onZygoteInit()
        {
            sp proc = ProcessState::self();
            if (proc->supportsProcesses()) {
                LOGV("App process: starting thread pool.\n");
                proc->startThreadPool();//启动一个线程 用于binder通信
            }       
        }
    
     private static void invokeStaticMain(String className, String[] argv)
                throws ZygoteInit.MethodAndArgsCaller { //classname是SS
            try { //找到SS的main函数
                m = cl.getMethod("main", new Class[] { String[].class });
            } 
         //最后抛出了一个异常    
         throw new ZygoteInit.MethodAndArgsCaller(m, argv);
    }

    在zygoteInit函数中,捕获到了这个函数,然后直接调用caller.run函数

Android之Zygote(下)--- SystemServer详解_第3张图片

 

在run函数中调用了SS的main函数

但是为什么要抛出异常后再调用??

这是因为在zygoteInit.main中调用,相当于native的main函数,即入口函数,位于堆栈的顶层,如果不采用抛出异常的方式,就会直接在 invokeStaticMain里调用,就会浪费之前函数调用所占用的一些调用堆栈

Zygote的分裂就是为了调用SS的main函数

public static void main(String[] args) {
System.loadLibrary("android_servers");//加载libandroid_server.so库
      init1(args);
}

init1是native函数,调用了system_init方法,

 frameworks/base/cmds/system_server/library

extern "C" status_t system_init()
{
sp proc(ProcessState::self());
    sp sm = defaultServiceManager();    
    sm->asBinder()->linkToDeath(grim, grim.get(), 0);//binder通信

AndroidRuntime* runtime = AndroidRuntime::getRuntime();
runtime->callStatic("com/android/server/SystemServer", "init2");//init2
}

Init2在SS.java中,

Android之Zygote(下)--- SystemServer详解_第4张图片

class ServerThread extends Thread {
//启动Entropy Service
ServiceManager.addService("entropy", new EntropyService());
//启动电源管理服务
power = new PowerManagerService();
ServiceManager.addService(Context.POWER_SERVICE, power);
//启动电池管理服务
battery = new BatteryService(context);
ServiceManager.addService("battery", battery);
//启动windowsManger服务
 wm = WindowManagerService.main(context, power,
                    factoryTest != SystemServer.FACTORY_TEST_LOW_LEVEL);
            ServiceManager.addService(Context.WINDOW_SERVICE, wm);
//启动activityManger服务等 Java核心service
...
//进行消息循环
Looper.loop();
}

Zygote的分裂

Zygote第四步,在分裂出SS后,就通过runSelectLoopMode等待并处理来自客户的消息了,谁会给zygote发消息呢?

ActivityManagerService(以下简称AMS)

AMS是由SS创建的,以启动一个Activity为例,而这个Activity属于一个还未启动的进程中,AMS.java中代码量很多,看其中一个方法startProcessLocked。

private final void startProcessLocked(ProcessRecord app,
            String hostingType, String hostingNameStr) {
     int pid = Process.start("android.app.ActivityThread",
                    mSimpleProcessManagement ? app.processName : null, uid, uid,
                    gids, debugFlags, null);

}

 调用了framework/base/core/java/android/os/Process.java里的start方法。

public static final int start(final String processClass,final String niceName,int uid, int gid, int[] gids,int debugFlags,String[] zygoteArgs){
if (supportsProcesses()) {
 return startViaZygote(processClass, niceName, uid, gid, gids,
                        debugFlags, zygoteArgs);
}
}
private static int startViaZygote(final String processClass, final String niceName, final int uid, final int gid,final int[] gids,int debugFlags,String[] extraArgs) throws ZygoteStartFailedEx {
    argsForZygote.add("--runtime-init");//做一些参数处理
pid = zygoteSendArgsAndGetPid(argsForZygote); //调用如下函数
return pid;
}

private static int zygoteSendArgsAndGetPid(ArrayList args) throws ZygoteStartFailedEx {
   openZygoteSocketIfNeeded();//打开和zygote通信的socket
    sZygoteWriter.write(Integer.toString(args.size()));//把要请求的参数发到zygote
sZygoteWriter.newLine();
sZygoteWriter.write(arg);
sZygoteWriter.newLine();
pid = sZygoteInputStream.readInt();//读取zygote处理完的结果便可知是某个进程的pid
}

根绝zygote(上)的介绍中可以ZygoteInit中是通过ZygoteConnection的runOnce函数进行处理,

boolean runOnce() throws ZygoteInit.MethodAndArgsCaller {
   args = readArgumentList();//读取SS发过来的参数
   pid = Zygote.forkAndSpecialize(parsedArgs.uid, parsedArgs.gid,//分裂出一个子进程

   if (pid == 0) {
            handleChildProc(parsedArgs, descriptors, newStderr); //处理子进程
            return true;
        } else {
            return handleParentProc(pid, descriptors, parsedArgs); //zygote进程
        }
}
private void handleChildProc(Arguments parsedArgs,FileDescriptor[] descriptors, PrintStream newStderr)throws ZygoteInit.MethodAndArgsCaller {
if (parsedArgs.runtimeInit) {//之前SS传入的参数有runtimeInit 为true
            RuntimeInit.zygoteInit(parsedArgs.remainingArgs);
        } 
}

最后在RuntimeInit.zygoteInit中还是调用了invokeStaticMain 函数。这个函数上文讲过

 

你可能感兴趣的:(android,zygote,android,studio)