`
quanminchaoren
  • 浏览: 926802 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

Android 根文件系统启动过程解析

阅读更多

在Android系统启动时,内核引导参数上一般都会设置“init=/init”, 这样的话,如果内核成功挂载了这个文件系统之后,首先运行的就是这个根目录下的init程序。这个程序所了什么呢? 我们只有RFSC(Read the Fucking Source code)!!
init程序源码在Android官方源码的system/core/init中,main在init.c里。我们的分析就从main开始。

init:
(1)安装SIGCHLD信号。(如果父进程不等待子进程结束,子进程将成为僵尸进程(zombie)从而占用系统资源。因此需要对SIGCHLD信号做出处理,回收僵尸进程的资源,避免造成不必要的资源浪费。)
(2)对umask进行清零。
    何为umask,请看关于Linux下的umask

(3)为rootfs建立必要的文件夹,并挂载适当的分区。
    /dev (tmpfs)
        /dev/pts (devpts)
        /dev/socket
    /proc (proc)
    /sys  (sysfs)
 (4)创建/dev/null和/dev/kmsg节点。
 (5)解析/init.rc,将所有服务和操作信息加入链表。
 (6)从/proc/cmdline中提取信息内核启动参数,并保存到全局变量。
(7)先从上一步获得的全局变量中获取信息硬件信息和版本号,如果没有则从/proc/cpuinfo中提取,并保存到全局变量。
(8)根据硬件信息选择一个/init.(硬件).rc,并解析,将服务和操作信息加入链表。
    在G1的ramdisk根目录下有两个/init.(硬件).rc:init.goldfish.rc和init.trout.rc,init程序会根据上一步获得的硬件信息选择一个解析。
(9)执行链表中带有“early-init”触发的的命令。
(10)遍历/sys文件夹,是内核产生设备添加事件(为了自动产生设备节点)。
(11)初始化属性系统,并导入初始化属性文件。
(12)从属性系统中得到ro.debuggable,若为1,則初始化keychord監聽。
(13)打開console,如果cmdline中沒有指定console則打開默認的/dev/console
(14)讀取/initlogo.rle(一張565 rle 壓縮的位圖),如果成功則在/dev/graphics/fb0顯示Logo,如果失敗則將/dev/tty0設為TEXT模式并打開/dev/tty0,輸出文本“ANDROID”字樣。
(15)判斷cmdline 中的參數,并设置属性系统中的参数:
         1、 如果 bootmode為
         - factory,設置ro.factorytest值為1
         - factory2,設置ro.factorytest值為2
         - 其他的設ro.factorytest值為0
       2、如果有serialno参数,則設置ro.serialno,否則為""
       3、如果有bootmod参数,則設置ro.bootmod,否則為"unknown"
       4、如果有baseband参数,則設置ro.baseband,否則為"unknown"
       5、如果有carrier参数,則設置ro.carrier,否則為"unknown"
       6、如果有bootloader参数,則設置ro.bootloader,否則為"unknown"
       7、通过全局变量(前面从/proc/cpuinfo中提取的)設置ro.hardware和ro.version。
(16)執行所有触发标识为init的action。
(17)開始property服務,讀取一些property文件,這一動作必須在前面那些ro.foo設置后做,以便/data/local.prop不能干預到他們。
      - /system/build.prop
      - /system/default.prop
      - /data/local.prop
      - 在讀取默認的property后讀取presistent propertie,在/data/property中
(18)為sigchld handler創建信號機制
(19)確認所有初始化工作完成:
    device_fd(device init 完成)
    property_set_fd(property server start 完成)
    signal_recv_fd (信號機制建立)
(20) 執行所有触发标识为early-boot的action
(21) 執行所有触发标识为boot的action
(22)基于當前property狀態,執行所有触发标识为property的action
(23)注冊輪詢事件:
      - device_fd
      - property_set_fd
      -signal_recv_fd
      -如果有keychord,則注冊keychord_fd
(24)如果支持BOOTCHART,則初始化BOOTCHART
(25)進入主進程循環:
      - 重置輪詢事件的接受狀態,revents為0
      - 查詢action隊列,并执行。
      - 重啟需要重啟的服务
      - 輪詢注冊的事件
          - 如果signal_recv_fd的revents為POLLIN,則得到一個信號,獲取并處理
          - 如果device_fd的revents為POLLIN,調用handle_device_fd
          - 如果property_fd的revents為POLLIN,調用handle_property_set_fd
          - 如果keychord_fd的revents為POLLIN,調用handle_keychord

由于对Android 的属性系统不是很了解,所以翻译了这篇文章《Android Property System》 。

 

    每个属性都有一个名称和值,他们都是字符串格式。属性被大量使用在Android系统中,用来记录系统设置或进程之间的信息交换。属性是在整个系统中全局可见的。每个进程可以get/set属性。
    在系统初始化时,Android将分配一个共享内存区来存储的属性。这些是由“init”守护进程完成的,其源代码位于: device/system/init。“init”守护进程将启动一个属性服务。属性服务在“init”守护进程中运行。每一个客户端想要设置属性时, 必须连接属性服务,再向其发送信息。属性服务将会在共享内存区中修改和创建属性。任何客户端想获得属性信息,可以从共享内存直接读取。这提高了读取性能。
    客户端应用程序可以调用libcutils中的API函数以GET/SET属性信息。libcutils的源代码位于:device/libs/cutils。API函数是:
    int property_get(const char *key, char *value, const char *default_value);
    int property_set(const char *key, const char *value);
    而libcutils又调用libc中的 __system_property_xxx 函数获得共享内存中的属性。libc的源代码位于:device/system/bionic。
    属性服务调用libc中的__system_property_init函数来初始化属性系统的共享内存。当启动属性服务时,将从以下文件中加载默认属性:
    / default.prop
    /system/build.prop
    /system/default.prop
    /data/local.prop
    属性将会以上述顺序加载。后加载的属性将覆盖原先的值。这些属性加载之后,最后加载的属性会被保持在/data/property中。

     特别属性
    如果属性名称以“ro.”开头,那么这个属性被视为只读属性。一旦设置,属性值不能改变。
    如果属性名称以“persist.”开头,当设置这个属性时,其值也将写入/data/property。
    如果属性名称以“net.”开头,当设置这个属性时,“net.change”属性将会自动设置,以加入到最后修改的属性名。(这是很巧妙的。 netresolve模块的使用这个属性来追踪在net.*属性上的任何变化。)
    属性“ ctrl.start ”和“ ctrl.stop ”是用来启动和停止服务。每一项服务必须在/init.rc中定义.系统启动时,与init守护进程将解析init.rc和启动属性服务。一旦收到设置“ ctrl.start ”属性的请求,属性服务将使用该属性值作为服务名找到该服务,启动该服务。这项服务的启动结果将会放入“ init.svc.<服务名>“属性中 。客户端应用程序可以轮询那个属性值,以确定结果。

    Android toolbox程序
    Android toolbox程序提供了两个工具: setprop和getprop获取和设置属性。其使用方法:
    getprop <属性名>
    setprop <属性名><<属性值>

    Java
    在Java应用程序可以使用System.getProperty()和System.setProperty()函数获取和设置属性。

    Action
    默认情况下,设置属性只会使"init"守护程序写入共享内存,它不会执行任何脚本或二进制程序。但是,您可以将您的想要的实现的操作与init.rc中某个属性的变化相关联.例如,在默认的init.rc中有:

        # adbd on at boot in emulator
        on property:ro.kernel.qemu=1
           start adbd
        on property:persist.service.adb.enable=1
           start adbd
        on property:persist.service.adb.enable=0
           stop adbd

    这样,如果你设置persist.service.adb.enable为1 ,"init"守护程序就知道需要采取行动:开启adbd服务。

文章中提到的共享内存就是Android特有的共享方式:ashmen

    Ashmem是一个匿名共享内存(Anonymous SHared MEMory)系统,该系统增加了接口因此进程间可以共享具名内存块。举一个例子,系统可以利用Ashmem存储图标,当绘制用户界面的时候多个进程也可 以访问。Ashmem优于传统Linux共享内存表现在当共享内存块不再被用的时候,它为Kernel提供一种回收这些共享内存块的手段。如果一个程序尝 试访问Kernel释放的一个共享内存块,它将会收到一个错误提示,然后重新分配内存并重载数据。

分享到:
评论

相关推荐

    linux 内核启动过程以及挂载android 根文件系统的过程

    ### Linux内核启动过程及Android根文件系统的挂载 #### Linux内核启动流程概览 Linux内核的启动过程是一个复杂而有序的过程,涉及到多个关键步骤。为了更好地理解这一过程,我们将从架构特定的启动代码开始,逐步...

    Android 根文件系统启动过程

    Android系统的启动过程中,根文件系统起着至关重要的作用。整个启动流程从内核引导开始,内核会在加载完根文件系统后执行`/init`程序,这是整个Android启动流程的起点。`/init`程序位于Android官方源码的`system/...

    android系统启动全程解析

    Android系统启动全程解析 Android系统启动是一个复杂而有序的过程,涉及多个层次的协同工作,从硬件初始化到用户界面呈现。本文将深入探讨Android系统启动的三个主要阶段:Linux系统层、Android系统服务层以及...

    android根文件系统分析总结

    #### 二、Linux内核启动与挂载Android根文件系统过程分析 Linux内核启动和挂载Android根文件系统是一个复杂但关键的过程。简单来说,这个过程可以分为以下几个步骤: 1. **内核初始化**:内核加载并执行初始的硬件...

    Android系统启动过程及举例.docx

    Init.rc 文件是 Android 系统启动过程中的一个关键文件。Init.rc 文件中定义了系统初始化要触发的动作和要启动的服务及其各自属性。 Service 类 Service 类是 Android 系统中的一种重要组件。Service 类用于定义...

    Android解析midi文件

    本教程将深入探讨如何在Android中解析MIDI文件,通过Java编程语言实现。 首先,我们需要了解MIDI文件的基本结构。MIDI文件由一系列事件组成,包括音符开始、音符结束、控制器变化等。这些事件被组织在称为“轨道”...

    Android 启动过程详解

    本文将对每个阶段进行详细解析,帮助读者更好地理解Android系统的启动流程。 #### 二、init进程启动 **init进程**是Android启动的第一个用户级进程,由内核直接启动。它的主要职责是读取配置文件`init.rc`及平台...

    展锐Android11.0开机启动流程介绍V1.0-nowatermark

    init 进程的启动过程主要包括以下几个步骤:kernel 启动 init 进程、init 进程解析配置文件、init 进程启动系统服务、init 进程挂载文件系统等。其中,init.rc 文件是 init 进程的配置文件,用于配置系统服务的启动...

    Android-init.rc文件解析过程详解.docx

    Android系统的初始化过程中,`init.rc`文件起...总之,`init.rc`文件是Android系统启动的核心配置文件,它的解析和执行过程是系统初始化的关键步骤。理解这一过程有助于开发者更好地理解和定制Android系统的启动行为。

    android 解析 excel 文件

    在Android平台上,解析Excel文件是一项常见的任务,尤其在数据处理、报表展示或者用户交互中。本文将详细讲解如何在Android应用中实现Excel文件的解析,主要聚焦于使用JXL库进行处理。 首先,JXL是一个Java库,专门...

    Android Framework精编内核解析

    《Android Framework精编内核解析》是一本深入探讨Android系统核心框架层的专著,它主要涵盖了Android系统架构的关键组成部分,旨在帮助读者理解和掌握Android应用开发背后的机制。在这个压缩包中,包含了一份名为...

    Android系统进程Zygote启动过程的源代码分析.pdf

    "Android系统进程Zygote启动过程的源代码分析" 在 Android 系统中,所有的应用程序进程以及系统服务进程SystemServer都是由 Zygote 进程孕育(fork )出来的,这也许就是为什么要把它称为Zygote (受精卵)的原因吧...

    android xml文件解析 SAX

    `parseXML`方法则负责创建解析器并启动解析过程。 通过了解和实践SAX解析,开发者可以更有效地处理XML数据,尤其是在处理大量XML数据时,SAX解析的性能优势尤为明显。同时,理解SAX解析机制有助于优化XML处理逻辑,...

    Android12系统资源目录分析树状图pdf(使用xmind思维导图软件绘画,导出pdf文件,此是压缩包文件,需解压)

    自己整理的Android12版本的系统资源目录分析,包括各个目录对应的分支 art: Android Runtime,一种App运行模式,区别于传统的Dalvik虚拟机,旨在提高Android系统的流畅性.PDF bionic:系统C库源代码,Android改造的C...

    文件管理中打开文件,关联并自己的APP启动

    当我们谈论“文件管理中打开文件,关联并自己的APP启动”这个主题时,我们实际上是在讨论一种功能,即如何让一个应用程序能够处理特定类型的文件,并且在用户尝试打开这类文件时,使该应用出现在系统的打开方式列表中...

    android XML文件解析

    4. 解析XML文件:最后,使用`parse()`方法启动解析过程,传入XML文件的输入流和事件处理器。 三、具体示例代码 ```java import org.xml.sax.XMLReader; import org.xml.sax.helpers.DefaultHandler; public class ...

    android开机启动流程

    通过上述分析,我们可以看出Android系统的启动流程是一个复杂而有序的过程,涉及到多个关键组件和服务的协同工作。从Linux内核的启动到用户界面的呈现,每一个步骤都至关重要。了解这些基础知识有助于开发者更好地...

    Android 音频开发-如何存储和解析wav文件

    在Android中,可以通过`Intent.ACTION_OPEN_DOCUMENT`启动文件选择器,让用户从文件浏览器中选择WAV文件。获取到文件 Uri 后,使用`ContentResolver`读取文件: ```java ContentResolver resolver = ...

Global site tag (gtag.js) - Google Analytics