Android 启动过程详解

本来不想再转载什么其他人的文章,但是发现昨天搜到的好东东,今天就找不到了,所以还是在这里把好文章Mark一下吧。

同时转一个System/Core/Init/Readme.txt的翻译

====================================================================================================================

来源:http://blog.csdn.net/maxleng/article/details/5508372


Android 启动过程详解

AndroidLinux系统启动有4个步骤;

(1) init进程启动

(2) Native服务启动

(3) System ServerAndroid服务启动

(4) Home启动

总体启动框架图如:

image

 

第一步:initial进程(system/core/init)

     init进程,它是一个由内核启动的用户级进程。内核自行启动(已经被载入内存,开始运行,并已初始化所有的设备驱动程序和数据结构等)之后,就通过启动一个用户级程序init的方式,完成引导进程。init始终是第一个进程.

Init.rc

Init.marvell.rc


Init进程一起来就根据init.rcinit.xxx.rc脚本文件建立了几个基本的服务:

  •  servicemanamger
  •  zygote

。。。

最后Init并不退出,而是担当起property service的功能。

1.1脚本文件

init@System/Core/Init

Init.c: parse_config_file(Init.rc)

  @parse_config_file(Init.marvel.rc)

解析脚本文件:Init.rcInit.xxxx.rc(硬件平台相关)

Init.rcAndroid自己规定的初始化脚本(Android Init Language, System/Core/Init/readme.txt)

该脚本包含四个类型的声明:

  • Actions
  • Commands
  • Services
  • Options.
1.2 服务启动机制

我们来看看Init是这样解析.rc文件开启服务的。

1)打开.rc文件,解析文件内容@ system/core/init/init.c

service信息放置到service_list中。@ system/core/init parser.c

2restart_service()@ system/core/init/init.c

service_start

execve().建立service进程。

第二步 Zygote

Servicemanagerzygote进程就奠定了Android的基础。Zygote这个进程起来才会建立起真正的Android运行空间,初始化建立的Service都是Navtive service..rc脚本文件中zygote的描述:

service zygote /system/bin/app_process -Xzygote /system/bin --zygote --start-system-server

所以Zygotemain()@frameworks/base/cmds/app_main.cpp开始。

(1) main()@frameworks/base/cmds/app_main.cpp

  • 建立Java Runtime
  • runtime.start("com.android.internal.os.ZygoteInit", startSystemServer);

(2) runtime.start@AndroidRuntime.cpp

  • 建立虚拟机
  • 运行:com.android.internal.os.ZygoteInitmain函数。

(3)main()@com.android.internal.os.ZygoteInit//正真的Zygote

  • registerZygoteSocket();//登记Listen端口
  • startSystemServer();
  • 进入Zygote服务框架。

经过这几个步骤,Zygote就建立好了,利用Socket通讯,接收ActivityManangerService的请求,Fork应用程序。

第三步 System Server

startSystemServer@com.android.internal.os.ZygoteInitZygotefork了一个进程: com.android.server.SystemServer.于是SystemServer@(SystemServer.java就建立了。Android的所有服务循环框架都是建立SystemServer@(SystemServer.java)上。在SystemServer.java中看不到循环结构,只是可以看到建立了init2的实现函数,建立了一大堆服务,并AddService到service Manager。

main() @ com/android/server/SystemServer

{

init1();

}

Init1()是在Native空间实现的(com_andoird_server_systemServer.cpp)。我们一看这个函数就知道了,init1->system_init() @System_init.cpp

system_init()我们看到了循环闭合管理框架。

{

Call "com/android/server/SystemServer", "init2"

…..

ProcessState::self()->startThreadPool();

    IPCThreadState::self()->joinThreadPool();

}


init2()@SystemServer.java中建立了Android中所有要用到的服务。

这个init2()建立了一个线程,来New ServiceAddService来建立服务


第三步 Home启动

ServerThread@SystemServer.java后半段,我们可以看到系统在启动完所有的Android服务后,做了这样一些动作:

(1) 使用xxx.systemReady()通知各个服务,系统已经就绪。

(2)  特别对于ActivityManagerService.systemReady(回调)

Widget.wallpaper,imm(输入法)ready通知。

Home就是在ActivityManagerService.systemReady()通知的过程中建立的。下面是ActivityManagerService.systemReady()的伪代码:

systemReady()@ActivityManagerService.java

resumeTopActivityLocked()

startHomeActivityLocked();//如果是第一个则启动HomeActivity

startActivityLocked(。。。)CATEGORY_HOME



翻译Android初始化语言(init.*.rc、init.conf文件格式)

============================================================================================================================

Android初始化语言包含了四种类型的声明:Actions(行动)、Commands(命令)、Services(服务)和Options(选项)。

所有这些都是以行为单位的,各种记号由空格来隔开。C语言风格的反斜杠号可用于在记号间插入空格。双引号也可用于防止字符串被空格分割成多个记号。行末的反斜杠用于折行。
注释行以井号(#)开头(允许以空格开头)。
Actions和Services声明一个新的分组。所有的命令或选项都属于最近申明的分组。位于第一个分组之前的命令或选项将会被忽略。
Actions和Services有唯一的名字。如果有重名的情况,第二个申明的将会被作为错误忽略。(???我们是否应该以覆盖来代替忽略)

Actions(行动)

    Actions其实就是一序列的Commands(命令)。Actions都有一个trigger(触发器),它被用于决定action的执行时间。当一个符合action触发条件的事件发生时,action会被加入到执行队列的末尾,除非它已经在队列里了。
    队列中的每一个action都被依次提取出,而这个action中的每个command(命令)都将被依次执行。Init在这些命令的执行期间还控制着其他的活动(设备节点的创建和注销、属性的设置、进程的重启)。

Actions的形式如下:

        on <trigger>
           <command>
           <command>
           <command>
Services(服务)

    Services(服务)是一个程序,他在初始化时启动,并在退出时重启(可选)。Services(服务)的形式如下:
        service <name> <pathname> [ <argument> ]*
           <option>
           <option>
           ...
Options(选项)

    Options(选项)是一个Services(服务)的修正者。他们影响Services(服务)在何时,并以何种方式运行。

     critical(关键)
            说明这是一个对于设备关键的服务。如果他四分钟内退出大于四次,系统将会重启并进入recovery(恢复)模式。

     disabled(失效)
            说明这个服务不会同与他同trigger(触发器)下的服务自动启动。他必须被明确的按名启动。

      setenv <name> <value> (设置环境变量)
            在进程启动时将环境变量<name>设置为<value>。

       socket <name> <type> <perm> [ <user> [ <group> ] ]
            创建一个Uinx域的名为/dev/socket/<name> 的套接字,并传递它的文件描述符给已启动的进程。<type> 必须是 "dgram"或"stream"。User 和 group默认为0。

        user <username>
            在启动这个服务前改变该服务的用户名。此时默认为root。(???有可能的话应该默认为nobody)。当前,如果你的进程要求Linux capabilities(能力),你无法使用这个命令。即使你是root,你也必须在程序中请求capabilities(能力)。然后降到你想要的 uid。

        group <groupname> [ <groupname> ]*
            在启动这个服务前改变该服务的组名。除了(必需的)第一个组名,附加的组名通常被用于设置进程的补充组(通过setgroups())。此时默认为root。(???有可能的话应该默认为nobody)。
   
        oneshot
            服务退出时不重启。

        class <name>
            指定一个服务类。所有同一类的服务可以同时启动和停止。如果不通过class选项指定一个类,则默认为"default"类服务。

        onrestart
            当服务重启,执行一个命令(下详)。

Triggers(触发器)

    Triggers(触发器)是一个用于匹配特定事件类型的字符串,用于使Actions(行动)发生。
   
        boot
            这是init执行后的第一个被触发的Triggers(触发器)。(在 /init.conf (启动配置文件)被装载之后)

        <name>=<value>
            这种形式的Triggers(触发器)会在属性<name>被设置为指定的<value>时被触发。

        device-added-<path>
        device-removed-<path>
            这种形式的Triggers(触发器)会在一个设备节点文件被增删时触发。

        service-exited-<name>
            这种形式的Triggers(触发器)会在一个特定的服务退出时触发。


Commands(命令)

    exec <path> [ <argument> ]*
         创建和执行一个程序(<path>)。在程序完全执行前,init将会阻塞。由于它不是内置命令,应尽量避免使用exec,它可能会引起init卡死。(??? 是否需要一个超时设置?)

    export <name> <value>
        在全局环境变量中设在环境变量 <name>为<value>。(这将会被所有在这命令之后运行的进程所继承)
      
    ifup <interface>
        启动网络接口<interface>

    import <filename>
           解析一个init配置文件,扩展当前配置。

    hostname <name>
           设置主机名。

    chmod <octal-mode> <path>
           更改文件访问权限。

    chown <owner> <group> <path>
           更改文件的所有者和组。

    class_start <serviceclass>
           启动所有指定服务类下的未运行服务。

    class_stop <serviceclass>
        停止指定服务类下的所有已运行的服务。

    domainname <name>
           设置域名。

    insmod <path>
           加载<path>中的模块。

    mkdir <path> [mode] [owner] [group]
           创建一个目录<path>,可以选择性地指定mode、owner以及group。如果没有指定,默认的权限为755,并属于root用户和root组。

    mount <type> <device> <dir> [ <mountoption> ]*
        试图在目录<dir>挂载指定的设备。<device> 可以是以 mtd@name 的形式指定一个mtd块设备。<mountoption>包括 "ro"、"rw"、"remount"、"noatime"、 ...

    setkey
           待完成......(暂时不可用)

    setprop <name> <value>
           设置系统属性 <name> 为 <value>值.

    setrlimit <resource> <cur> <max>
        设置<resource>的rlimit(资源限制)。

    start <service>
        启动指定服务(如果此服务还未运行)。

    stop <service>
        停止指定服务(如果此服务在运行中)。

    symlink <target> <path>
        创建一个指向<path>的软连接<target>。

    sysclktz <mins_west_of_gmt>
        设置系统时钟基准(0代表时钟滴答以格林威治平均时(GMT)为准)

    trigger <event>
           触发一个事件。用于将一个action与另一个 action排列。(?????)

    write <path> <string> [ <string> ]*
           打开路径为<path>的一个文件,并写入一个或多个字符串。


Properties(属性)

    Init更新一些系统属性以提供对正在发生的事件的监控能力:

        init.action
               此属性值为正在被执行的action的名字,如果没有则为""。

        init.command
               此属性值为正在被执行的command的名字,如果没有则为""。

        init.svc.<name>
               名为<name>的service的状态("stopped"(停止), "running"(运行), "restarting"(重启))               


init.conf实例
# not complete -- just providing some examples of usage
    #
    on boot
       export PATH /sbin:/system/sbin:/system/bin
       export LD_LIBRARY_PATH /system/lib

       mkdir /dev
       mkdir /proc
       mkdir /sys

       mount tmpfs tmpfs /dev
       mkdir /dev/pts
       mkdir /dev/socket
       mount devpts devpts /dev/pts
       mount proc proc /proc
       mount sysfs sysfs /sys

       write /proc/cpu/alignment 4

       ifup lo

       hostname localhost
       domainname localhost

       mount yaffs2 mtd@system /system
       mount yaffs2 mtd@userdata /data

       import /system/etc/init.conf

       class_start default

    service adbd /sbin/adbd
       user adb
       group adb

    service usbd /system/bin/usbd -r
       user usbd
       group usbd
       socket usbd 666

    service zygote /system/bin/app_process -Xzygote /system/bin --zygote
       socket zygote 666

    service runtime /system/bin/runtime
       user system
       group system

    on device-added-/dev/compass
       start akmd

    on device-removed-/dev/compass
       stop akmd

    service akmd /sbin/akmd
       disabled
       user akmd
       group akmd
调试记录

在默认情况下,程序在被init执行时会将标准输出和标准错误都重定向到/dev/null(丢弃)。若你想要获得调试信息,你可以通过Andoird系统中的logwrapper程序执行你的程序。它会将标准输出/标准错误都重定向到Android日志系统(通过logcat访问)。

例如:

    service akmd /system/bin/logwrapper /sbin/akmd


来源:http://www.hiapk.com/thread-6395-1-1.html

==================================================================================================================

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值