Android 系统的启动过程

最近在网上找到了一个学习 Android framework 源码的地址,着实开心。因为在虚拟机ubuntu下载源码实在是太难等了,而且自己也不是要移植系统开发学习,只是为了学习里面的代码。所以这个地址真是太适合了。下来来介绍下Android系统的启动过程

Android ROM

想了解Android系统的启动过程,就不得不了解下ROM包的组成,现在国内市场上很多ROM包,小米,魅族,一加等等。那么ROM包都是由什么构成的呢。这里以红米2A的某个版本解压来看看,可以看到images目录如下图

images.png
misc.png

其中这里面包含了Android很多的分区

  • boot.img boot分区,包括内核文件和虚拟内存盘Ramdisk,负责手机开机,可在recovery模式进行擦除,重新安装带有boot分区的新系统
  • system.img system分区,包含Android系统的用户界面以及设置上的一些预装系统应用
  • recovery.img recovery分区,替代启动分区,执行恢复和维护系统的一些操作
  • userdata.img data分区,用于保存用户的数据,例如联系人,短信,设置偏好和应用程序存放的地方
  • cache.img cache分区,用于放置系统频繁访问的数据和应用程序组件的分区
  • misc misc分区存放一些系统设置,包括USB配置和某些硬件设置,如果损坏,一些设备功能将无法启动
  • persist.img persist分区包含了设备的传感器和信号部分的驱动程序,例如wifi,蓝牙连接都有关系
  • splash.img 开机画面文件
  • NON-HLOS.bin是负责处理通讯协议相关的基带镜像
  • 8916_msimage.mbn 平台镜像,包含一些必要的启动分区和分区表,例如里面的sbl1.mbn、tz.mbn、rpm.mbn等等
  • sbl1.mbn 硬件的初始化,并且保存加载其他模块信息的顺序
  • tz.mbn 它和其他模块代码运行在相互隔离的区域,主要实现底层很多安全性特性
  • rpm.mbn 负责资源电源管理
  • emmc_appsboot.mbn bootloader文件,进入fastboot模式相关文件
  • gpt_backup0.bin 备份gpt头和分区表
  • gpt_both0.bin 对应EMMC的分区表
  • gpt_main0.bin GPT头和分区表
  • patch0.xml 服务端返回的具体磁盘大小打上最后一个分区的补丁、完成分区表头校验的配置文
  • 其他都是一些烧录程序的文件,如prog_emmc_firehose_8916.mbn 、rawprogram0.xml 、sec.bat

Android BootLoader

Linux软件系统中,有一个叫引导加载程序的部分,它主要是由固化代码程序和BootLoader程序组成,其中固化程序已经烧入到了生产的芯片中,而BootLoader程序通过它的启动模式去引导系统CPU和部分外设的初始化。为把Linux内核加载到RAM内存中做准备。例如,在上文的ROM中,系统首先运行的就是emmc_appsboot.mbn文件中的bootloader程序。详细程序根据不同厂商芯片而不同,如要完全搞懂,得深入ARM开发相关知识。

Linux kernel

BootLoader程序执行完后,标志着CPU和部分外设初始化完成,接下来要做的就是把Linux内核加载到内存中,主要流程如下

Linux内核.png

说到Linux内核,先来了解下Linux内核的结构

Linux内核.png
  • SCI供用户空间调用的一些API函数,API函数的具体表现依赖于底部的体系结构
  • PM 进程管理,接收SCI创建的进程,然后实现进程之间的通信和同步管理
  • MM 内存管理,通过内存页的方式进行管理,主要管理缓存区的内存的分配
  • VFS 虚拟文件系统,在内核所支持的文件系统和SCI之间提供了一个交换层
  • NS 网络堆栈,主要是一些TCP/IPSocket相关的网络协议
  • DD 设备驱动,驱动程序,驱动一些硬件设备的功能
  • Arch 依赖体系结构的代码

上文说的bootloader程序执行完成之后,就会启动Linux内核,Linux内核启动调用的第一个函数就是start_kernel,start_kernel主要会执行

  1. 调用setup_arch初始化化一些硬件相关的体系结构依赖代码
  2. 调用paging_init()初始化内核页表
  3. 调用mem_init()函数初始化页描述符
  4. 调用trap_init()init_IRQ初始化中断向量表IDT
  5. 调用kmem_cache_init()kmem_cache_sizes_init()函数初始化slab分配器
  6. 调用time_init()函数初始化系统日期和时间
  7. 调用kernel_thread()函数为进程1创建内核线,同时执行init程序
  8. 调用fork_init()函数初始化进程创建机制
  9. 调用vfs_caches_init()初始化虚拟文件系统
    start_kernel执行完后,接下来会进入rest_init函数的执行
    它会生成两个线程kernel_initkthreadd。其中kernel_init也叫init进程,主要是初始化和执行进程,kthreadd是用来管理调度其他应用进程。rest_init主要会去调用cpu_idle函数,该函数有一个死循环,当有命令过来的时候则启动执行,反之进入空闲状态。

Android init

上文kernel_init初始化之后,则会进入Android init进程,执行init.c里面的main函数。流程如下:

流程.png

其中AndroidMain()函数位于<Android源代码本目录>/system/core/init/,在init.rc它会去读取init.rc配置文件

// android-7.1.2_r36/init/init.cpp
//.... 
    Parser& parser = Parser::GetInstance();
    parser.AddSectionParser("service",std::make_unique<ServiceParser>());
    parser.AddSectionParser("on", std::make_unique<ActionParser>());
    parser.AddSectionParser("import", std::make_unique<ImportParser>());
    parser.ParseConfig("/init.rc");
//....

init.rc使用的是AIL语言,代码位于/system/core/rootdir/,在init.rc里面会创建一个Zygote进程

// android-7.1.2_r36/rootdir/init.rc
//.... 
on nonencrypted
    # A/B update verifier that marks a successful boot.
    exec - root cache -- /system/bin/update_verifier nonencrypted
    class_start main
    class_start late_start
//....

上面的class main即为创建Zygote进程,创建具体代码在rootdir/目录下的init.zygote64.rc

// android-7.1.2_r36/rootdir/init.zygote64.rc
service zygote /system/bin/app_process64 -Xzygote /system/bin --zygote --start-system-server
    class main
    socket zygote stream 660 root system
//...
  • service AIL语言语法,启动一个系统程序
  • zygote 启动程序名称,这里指的是Zygote进程
  • /system/bin/app_process64 需要操作对象的路径pathname,这里指的是/base/cmds/app_process/app_main.cpp
  • -Xzygote /system/bin 指定参数传到/base/cmds/app_process/app_main.cpp里面
  • --zygote --start-system-server传的具体参数值。这里指的参数为--zygote--start-system-server

Android app_main.cpp

上面传的参数可以在main函数里面得到

// android-7.1.2_r36/cmds/app_process/app_main.cpp
//.... 
 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;
        }
      //.... 
    }
//.... 

然后继续往下

// android-7.1.2_r36/cmds/app_process/app_main.cpp
//.... 
if (zygote) {
        runtime.start("com.android.internal.os.ZygoteInit", args, zygote);
    } else if (className) {
        runtime.start("com.android.internal.os.RuntimeInit", args, zygote);
    }
//.... 

如果传了--zygote参数,则通过runtime启动ZygoteInit,同时传入相关的参数,反之启动RuntimeInit
ZygoteInit源码位于framework/base/core/java/com/android/internal/os/ZygoteInit.java
RuntimeInit源码位于framework/base/core/java/com/android/internal/os/RuntimeInit.java
在上面的代码中runtime指的是在同文件定义的AppRuntimeAppRuntime继承于AppRuntimeAndroid的虚拟机ART,源代码位于frameworks/base/core/jni/AndroidRuntime.cpp

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,542评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,596评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,021评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,682评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,792评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,985评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,107评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,845评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,299评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,612评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,747评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,441评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,072评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,828评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,069评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,545评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,658评论 2 350

推荐阅读更多精彩内容

  • 1:InputChannel提供函数创建底层的Pipe对象 2: 1)客户端需要新建窗口 2)new ViewRo...
    自由人是工程师阅读 5,281评论 0 18
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,846评论 25 707
  • 美好的邂逅总需要漫长的等待,也许这就是人们常说的好事多磨吧,对于默潇来说,即使这场难捱的等待只有一个星期,但也拉长...
    GZ徐阅读 255评论 0 1
  • 今天在妈妈住的小区门口,因为我在和门口的保安登记住址,拿停车凭证,并没有注意到右边闪出来一个保安大爷。当我...
    焜娃阅读 156评论 0 0
  • 周六,我陪朋友去天津看房,并最终定下一套合适的小户型商住两用房。 回想这几年与房子的交集,我收获不少。 第一次都购...
    后知后觉一只猪阅读 308评论 0 0