Zygote进程【3】——SystemServer的诞生

简介:

在ZygoteInit的main()方法中做了几件大事,其中一件便是启动Systemserver进程,代码如下:

@/frameworks/base/core/Java/com/Android/internal/os/ZygoteInit.java

[java]  view plain  copy print?在CODE上查看代码片派生到我的代码片
  1. public static void main(String argv[]) {  
  2.     try {  
  3.         ......  
  4.         if (argv[1].equals("start-system-server")) {  
  5.             startSystemServer();//启动system_server进程  
  6.         } else if (!argv[1].equals("")) {  
  7.             throw new RuntimeException(argv[0] + USAGE_STRING);  
  8.         }  
  9.         ......  
  10. }  

startSystemServer方法的实现如下:

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

[java]  view plain  copy print?在CODE上查看代码片派生到我的代码片
  1. /** 
  2.  * Prepare the arguments and fork for the system server process. 
  3.  */  
  4. private static boolean startSystemServer()  
  5.         throws MethodAndArgsCaller, RuntimeException {  
  6.     long capabilities = posixCapabilitiesAsBits(  
  7.         OsConstants.CAP_KILL,  
  8.         OsConstants.CAP_NET_ADMIN,  
  9.         OsConstants.CAP_NET_BIND_SERVICE,  
  10.         OsConstants.CAP_NET_BROADCAST,  
  11.         OsConstants.CAP_NET_RAW,  
  12.         OsConstants.CAP_SYS_MODULE,  
  13.         OsConstants.CAP_SYS_NICE,  
  14.         OsConstants.CAP_SYS_RESOURCE,  
  15.         OsConstants.CAP_SYS_TIME,  
  16.         OsConstants.CAP_SYS_TTY_CONFIG  
  17.     );  
  18.     /* Hardcoded command line to start the system server */  
  19.     String args[] = {  
  20.         "--setuid=1000",  
  21.         "--setgid=1000",  
  22.         "--setgroups=1001,1002,1003,1004,1005,1006,1007,1008,1009,1010,1018,1032,3001,3002,3003,3006,3007",  
  23.         "--capabilities=" + capabilities + "," + capabilities,  
  24.         "--runtime-init",  
  25.         "--nice-name=system_server",  
  26.         "com.android.server.SystemServer",  
  27.     };  
  28.     ZygoteConnection.Arguments parsedArgs = null;  
  29.   
  30.     int pid;  
  31.   
  32.     try {  
  33.         parsedArgs = new ZygoteConnection.Arguments(args);  
  34.         ZygoteConnection.applyDebuggerSystemProperty(parsedArgs);  
  35.         ZygoteConnection.applyInvokeWithSystemProperty(parsedArgs);  
  36.   
  37.         /* Request to fork the system server process */  
  38.         pid = Zygote.forkSystemServer(//以fork的方式创建system_server进程  
  39.                 parsedArgs.uid, parsedArgs.gid,  
  40.                 parsedArgs.gids,  
  41.                 parsedArgs.debugFlags,  
  42.                 null,  
  43.                 parsedArgs.permittedCapabilities,  
  44.                 parsedArgs.effectiveCapabilities);  
  45.     } catch (IllegalArgumentException ex) {  
  46.         throw new RuntimeException(ex);  
  47.     }  
  48.   
  49.     /* For child process */  
  50.     if (pid == 0) {//pid==0说明在子进程中,父进程为Zygote  
  51.         handleSystemServerProcess(parsedArgs);  
  52.     }  
  53.   
  54.     return true;  
  55. }  

在startSystemServer中先设置了fork SystemServer所需的参数,然后通过forkSystemServer方法fork出SystemServer进程,最后通过handleSystemServerProcess处理新进程中的善后事宜。

首先看一下参数:

1、setuid=1000,这里1000代表SYSTEM_UID,即系统进程,关于进程ID的说明可以参见:/frameworks/base/core/java/android/os/Process.java。

2、nice-name=system_server表示制定进程的名字为“system_server”

3、com.android.server.SystemServer表示SystemServer类的位置。

接下来看一下forkSystemServer的实现:

@/libcore/dalvik/src/main/java/dalvik/system/Zygote.java

[java]  view plain  copy print?在CODE上查看代码片派生到我的代码片
  1. /** 
  2.  * Special method to start the system server process. In addition to the 
  3.  * common actions performed in forkAndSpecialize, the pid of the child 
  4.  * process is recorded such that the death of the child process will cause 
  5.  * zygote to exit. 
  6.  * 
  7.  * @param uid the UNIX uid that the new process should setuid() to after 
  8.  * fork()ing and and before spawning any threads. 
  9.  * @param gid the UNIX gid that the new process should setgid() to after 
  10.  * fork()ing and and before spawning any threads. 
  11.  * @param gids null-ok; a list of UNIX gids that the new process should 
  12.  * setgroups() to after fork and before spawning any threads. 
  13.  * @param debugFlags bit flags that enable debugging features. 
  14.  * @param rlimits null-ok an array of rlimit tuples, with the second 
  15.  * dimension having a length of 3 and representing 
  16.  * (resource, rlim_cur, rlim_max). These are set via the posix 
  17.  * setrlimit(2) call. 
  18.  * @param permittedCapabilities argument for setcap() 
  19.  * @param effectiveCapabilities argument for setcap() 
  20.  * 
  21.  * @return 0 if this is the child, pid of the child 
  22.  * if this is the parent, or -1 on error. 
  23.  */  
  24. public static int forkSystemServer(int uid, int gid, int[] gids, int debugFlags,  
  25.         int[][] rlimits, long permittedCapabilities, long effectiveCapabilities) {  
  26.     preFork();  
  27.     int pid = nativeForkSystemServer(  
  28.             uid, gid, gids, debugFlags, rlimits, permittedCapabilities, effectiveCapabilities);  
  29.     postFork();  
  30.     return pid;  
  31. }  
  32.   
  33. native public static int nativeForkSystemServer(int uid, int gid, int[] gids, int debugFlags,  
  34.         int[][] rlimits, long permittedCapabilities, long effectiveCapabilities);  

nativeForkSystemServer最终通过JNI实现,代码为:

@/dalvik/vm/native/dalvik_system_Zygote.cpp

[java]  view plain  copy print?在CODE上查看代码片派生到我的代码片
  1. /* 
  2.  * native public static int nativeForkSystemServer(int uid, int gid, 
  3.  *     int[] gids, int debugFlags, int[][] rlimits, 
  4.  *     long permittedCapabilities, long effectiveCapabilities); 
  5.  */  
  6. static void Dalvik_dalvik_system_Zygote_forkSystemServer(  
  7.         const u4* args, JValue* pResult)  
  8. {  
  9.     pid_t pid;  
  10.     pid = forkAndSpecializeCommon(args, true);  
  11.   
  12.   
  13.     /* The zygote process checks whether the child process has died or not. */  
  14.     if (pid > 0) {//pid大于0,说明是在父进程中  
  15.         int status;  
  16.   
  17.   
  18.         ALOGI("System server process %d has been created", pid);  
  19.         gDvm.systemServerPid = pid;  
  20.         /* There is a slight window that the system server process has crashed 
  21.          * but it went unnoticed because we haven't published its pid yet. So 
  22.          * we recheck here just to make sure that all is well. 
  23.          */  
  24.         if (waitpid(pid, &status, WNOHANG) == pid) {//堵塞,等待system_server进程  
  25.             ALOGE("System server process %d has died. Restarting Zygote!", pid);  
  26.             kill(getpid(), SIGKILL);//一旦上面的等待返回,说明进程pid(system_server)已终止,此时Zygote杀死自己  
  27.         }  
  28.     }  
  29.     RETURN_INT(pid);  
  30. }  

可以看出Dalvik_dalvik_system_Zygote_forkSystemServer会调用forkAndSpecializeCommon来fork出system_server进程。这里要注意最后几句,在fork出system_server以后,Zygote会调用waitpid等待system_server的终止,一旦发现system_server终止,Zygote则马上自杀。

接下来看一下handleSystemServerProcess的实现:

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

[java]  view plain  copy print?在CODE上查看代码片派生到我的代码片
  1.      /** 
  2.      * Finish remaining work for the newly forked system server process. 
  3.      */  
  4.     private static void handleSystemServerProcess(  
  5.             ZygoteConnection.Arguments parsedArgs)  
  6.             throws ZygoteInit.MethodAndArgsCaller {  
  7.   
  8.   
  9.         closeServerSocket();//关闭从Zygote复制过来的socket  
  10.   
  11.   
  12.         // set umask to 0077 so new files and directories will default to owner-only permissions.  
  13.         Libcore.os.umask(S_IRWXG | S_IRWXO);//设置文件的默认权限,去除所有者之外的权限  
  14.   
  15.   
  16.         if (parsedArgs.niceName != null) {  
  17.             Process.setArgV0(parsedArgs.niceName);//system_server  
  18.         }  
  19.   
  20.   
  21.         if (parsedArgs.invokeWith != null) {  
  22.             WrapperInit.execApplication(parsedArgs.invokeWith,  
  23.                     parsedArgs.niceName, parsedArgs.targetSdkVersion,  
  24.                     null, parsedArgs.remainingArgs);  
  25.         } else {  
  26.             /* 
  27.              * Pass the remaining arguments to SystemServer. 
  28.              */  
  29.             RuntimeInit.zygoteInit(parsedArgs.targetSdkVersion, parsedArgs.remainingArgs);  
  30.         }  
  31.   
  32.   
  33.         /* should never reach here */  
  34.     }  

上面的代码中用到了Linux中的umask这个函数,不明白的读者可以参考:http://hi.baidu.com/fengyun409/item/82cd158ffe7f67c8b17154e7

下面继续看RuntimeInit.zygoteInit方法的实现:

@/frameworks/base/core/java/com/android/internel/os/RuntimeInit.java

[java]  view plain  copy print?在CODE上查看代码片派生到我的代码片
  1. /** 
  2.  * The main function called when started through the zygote process. This 
  3.  * could be unified with main(), if the native code in nativeFinishInit() 
  4.  * were rationalized with Zygote startup.<p> 
  5.  * 
  6.  * Current recognized args: 
  7.  * <ul> 
  8.  *   <li> <code> [--] <start class name>  <args> 
  9.  * </ul> 
  10.  * 
  11.  * @param targetSdkVersion target SDK version 
  12.  * @param argv arg strings 
  13.  */  
  14. public static final void zygoteInit(int targetSdkVersion, String[] argv)  
  15.         throws ZygoteInit.MethodAndArgsCaller {  
  16.     if (DEBUG) Slog.d(TAG, "RuntimeInit: Starting application from zygote");  
  17.   
  18.     redirectLogStreams();//将System.out 和 System.err 输出重定向到Android 的Log系统  
  19.     /* 
  20.      * 初始化了一些系统属性,其中最重要的一点就是设置了一个未捕捉异常的handler, 
  21.      * 当代码有任何未知异常,就会执行它, 
  22.      * 调试过Android代码的同学经常看到的"*** FATAL EXCEPTION IN SYSTEM PROCESS" 打印就出自这里 
  23.      */  
  24.     commonInit();  
  25.     /* 
  26.      * 最终会调用app_main的onZygoteInit函数 
  27.      * 这里的作用是在新进程中引入Binder,也就说通过nativeZygoteInit以后,新的进程就可以使用Binder进程通信了 
  28.      */  
  29.     nativeZygoteInit();  
  30.   
  31.     applicationInit(targetSdkVersion, argv);//应用初始化  
  32. }  

这个函数是不是有些面熟?没错在《Zygote进程【2】——Zygote的分裂》一文中我们见过,Zygote进程在接收到ActivityManagerService请求创建进程的请求时就调用的该方法来处理创建子进程的后续工作。

[java]  view plain  copy print?在CODE上查看代码片派生到我的代码片
  1. private static void applicationInit(int targetSdkVersion, String[] argv)  
  2.         throws ZygoteInit.MethodAndArgsCaller {  
  3.     // If the application calls System.exit(), terminate the process  
  4.     // immediately without running any shutdown hooks.  It is not possible to  
  5.     // shutdown an Android application gracefully.  Among other things, the  
  6.     // Android runtime shutdown hooks close the Binder driver, which can cause  
  7.     // leftover running threads to crash before the process actually exits.  
  8.     nativeSetExitWithoutCleanup(true);  
  9.   
  10.     // We want to be fairly aggressive about heap utilization, to avoid  
  11.     // holding on to a lot of memory that isn't needed.  
  12.     VMRuntime.getRuntime().setTargetHeapUtilization(0.75f);  
  13.     VMRuntime.getRuntime().setTargetSdkVersion(targetSdkVersion);  
  14.   
  15.     final Arguments args;  
  16.     try {  
  17.         args = new Arguments(argv);  
  18.     } catch (IllegalArgumentException ex) {  
  19.         Slog.e(TAG, ex.getMessage());  
  20.         // let the process exit  
  21.         return;  
  22.     }  
  23.   
  24.     // Remaining arguments are passed to the start class's static main  
  25.     invokeStaticMain(args.startClass, args.startArgs);  
  26. }  

所以,这里与Zygote分裂时不同的是:这里的args.startClass的值为com.android.server.SystemServer。接下来大家都知道了,SystemServer类的main函数将会被调用。

@/frameworks/base/services/java/com/android/server/SystemServer.java

[java]  view plain  copy print?在CODE上查看代码片派生到我的代码片
  1. public static void main(String[] args) {  
  2.     if (System.currentTimeMillis() < EARLIEST_SUPPORTED_TIME) {  
  3.         // If a device's clock is before 1970 (before 0), a lot of  
  4.         // APIs crash dealing with negative numbers, notably  
  5.         // java.io.File#setLastModified, so instead we fake it and  
  6.         // hope that time from cell towers or NTP fixes it  
  7.         // shortly.  
  8.         Slog.w(TAG, "System clock is before 1970; setting to 1970.");  
  9.         SystemClock.setCurrentTimeMillis(EARLIEST_SUPPORTED_TIME);//初始化系统时间  
  10.     }  
  11.   
  12.     if (SamplingProfilerIntegration.isEnabled()) {  
  13.         SamplingProfilerIntegration.start();  
  14.         timer = new Timer();  
  15.         timer.schedule(new TimerTask() {  
  16.             @Override  
  17.             public void run() {  
  18.                 SamplingProfilerIntegration.writeSnapshot("system_server", null);  
  19.             }  
  20.         }, SNAPSHOT_INTERVAL, SNAPSHOT_INTERVAL);  
  21.     }  
  22.   
  23.     // Mmmmmm... more memory!  
  24.     dalvik.system.VMRuntime.getRuntime().clearGrowthLimit();  
  25.   
  26.     // The system server has to run all of the time, so it needs to be  
  27.     // as efficient as possible with its memory usage.  
  28.     VMRuntime.getRuntime().setTargetHeapUtilization(0.8f);  
  29.   
  30.     Environment.setUserRequired(true);  
  31.   
  32.     System.loadLibrary("android_servers");//加载android_servers库  
  33.   
  34.     Slog.i(TAG, "Entered the Android system server!");  
  35.   
  36.     // Initialize native services.  
  37.     nativeInit();//初始化native service  
  38.   
  39.     // This used to be its own separate thread, but now it is  
  40.     // just the loop we run on the main thread.  
  41.     ServerThread thr = new ServerThread();  
  42.     thr.initAndLoop();  
  43. }  

在main中会加载libandroid_servers.so库,然后调用nativeInit初始化native层的Service。

[java]  view plain  copy print?在CODE上查看代码片派生到我的代码片
  1. /** 
  2.  * Called to initialize native system services. 
  3.  */  
  4. private static native void nativeInit();  

@/frameworks/base/services/jni/com_android_server_SystemServer.cpp

[java]  view plain  copy print?在CODE上查看代码片派生到我的代码片
  1. static void android_server_SystemServer_nativeInit(JNIEnv* env, jobject clazz) {  
  2.     char propBuf[PROPERTY_VALUE_MAX];  
  3.     property_get("system_init.startsensorservice", propBuf, "1");  
  4.     if (strcmp(propBuf, "1") == 0) {  
  5.         // Start the sensor service  
  6.         SensorService::instantiate();  
  7.     }  
  8. }  

可以看出这里只初始化了传感器service,这与之前的代码有所不同,在比较早的Android版本中,服务的初始化分为init1和init2两个过程,其中init主要负责native层service的初始化(SurfaceFlinger、AudioFlinger等),init2负责java层service的初始化。

在main方法最后会调用ServerThread类的initAndLoop来初始化系统服务,这个函数比较长,这里就不复制代码了。


好了,到这里SystemServer的诞生过程就完了




    本文转自 一点点征服   博客园博客,原文链接:http://www.cnblogs.com/ldq2016/p/6232336.html,如需转载请自行联系原作者



相关文章
|
8月前
|
存储 Java Android开发
Zygote进程启动过程
Zygote进程启动过程
55 1
|
Java Android开发
|
缓存 监控 网络协议
|
Java 调度 Android开发
android体系课-系统启动流程-之zygote进程启动过程源码分析
笔者刚开始学习Android的时候也和大部分同学一样,只会使用一些应用层面的知识,对于一些比较常见的开源框架如<mark>RxJava</mark>,<mark>OkHttp</mark>,<mark>Retrofit</mark>,以及后来谷歌推出的<mark>协程</mark>等,都只在使用层面,对于他们<mark>内部原理</mark>,基本没有去了解觉得够用就可以了,又比如Activity,Service等四大组件的使用原理,系统开机过程,Launcher启动过程等知之甚少,知其然而不知其所以然,结果就是出现某些问题,不知道从哪里找原因,只能依赖万能的百度,但是百度看多了,你会发现自己
|
7月前
|
监控 Linux 应用服务中间件
探索Linux中的`ps`命令:进程监控与分析的利器
探索Linux中的`ps`命令:进程监控与分析的利器
146 13
|
6月前
|
运维 关系型数据库 MySQL
掌握taskset:优化你的Linux进程,提升系统性能
在多核处理器成为现代计算标准的今天,运维人员和性能调优人员面临着如何有效利用这些处理能力的挑战。优化进程运行的位置不仅可以提高性能,还能更好地管理和分配系统资源。 其中,taskset命令是一个强大的工具,它允许管理员将进程绑定到特定的CPU核心,减少上下文切换的开销,从而提升整体效率。
掌握taskset:优化你的Linux进程,提升系统性能
|
6月前
|
弹性计算 Linux 区块链
Linux系统CPU异常占用(minerd 、tplink等挖矿进程)
Linux系统CPU异常占用(minerd 、tplink等挖矿进程)
203 4
Linux系统CPU异常占用(minerd 、tplink等挖矿进程)
|
5月前
|
算法 Linux 调度
探索进程调度:Linux内核中的完全公平调度器
【8月更文挑战第2天】在操作系统的心脏——内核中,进程调度算法扮演着至关重要的角色。本文将深入探讨Linux内核中的完全公平调度器(Completely Fair Scheduler, CFS),一个旨在提供公平时间分配给所有进程的调度器。我们将通过代码示例,理解CFS如何管理运行队列、选择下一个运行进程以及如何对实时负载进行响应。文章将揭示CFS的设计哲学,并展示其如何在现代多任务计算环境中实现高效的资源分配。