资讯专栏INFORMATION COLUMN

Linux启动分析——init进程与app启动

lidashuang / 2097人阅读

摘要:概述本文通过简要分析进程源码,梳理其处理流程,重点关注进程如何启动应用程序,总结启动脚本文件的编写思路进程源码分析进程是内核启动的第一个进程,怎么知道的从内核源码代码的函数分析,可以发现,内核会根据传入的参数来启动第一个进程,一般都是怎么

概述

本文通过简要分析init进程源码,梳理其处理流程,重点关注init进程如何启动应用程序,总结启动脚本文件的编写思路

init进程源码分析

init进程是linux内核启动的第一个进程,怎么知道的?从内核源码linux-2.6.xxx/init/main.c代码的kernel_init()函数分析,可以发现,内核会根据uboot传入的参数来启动第一个进程,一般都是init

怎么启动的呢,调用kernel_execve()函数完成的,猜测是从根文件系统的/sbin/init来启动的,linux的任何应用程序都是基于文件系统的,启动应用程序前提是根文件系统已经挂载好了。好,那么根文件系统又是从哪里来的呢,是由busybox这个工具配置编译生成的,所以要分析init源码,要去busybox里找init的源码

源码位置:/busybox/init/init.c,在其中查找main()函数,发现只有init_main(),没有main(),可以猜测busybox是通过一些方法将init进程的入口修改为init_main(),实际上所有busybox的命令工具都是一个到busybox程序的链接,

cd /sbin
ls -l init
lrwxrwxrwx 1 root 0 14 Nov 16 2016 init -> ../bin/busybox

可以看到,init进程其实是到busybox的链接,不用管它,知道init进程的入口是init_main()函数就行了

#if DEBUG_SEGV_HANDLER
    {
        struct sigaction sa;
        memset(&sa, 0, sizeof(sa));
        sa.sa_sigaction = handle_sigsegv;
        sa.sa_flags = SA_SIGINFO;
        sigaction(SIGSEGV, &sa, NULL);
        ......
    }
#endif
......
console_init();
set_sane_term();
......
/* Make sure environs is set to something sane */
putenv((char *) "HOME=/");
putenv((char *) bb_PATH_root_path);
putenv((char *) "SHELL=/bin/sh");
putenv((char *) "USER=root"); /* needed? why? */

这一段是init进程最开始要做的事情,设置一些信号相关的东西,初始化console,然后设置环境变量,跟启动app似乎没有什么关系,不用管,继续往下看

/* Check if we are supposed to be in single user mode */
if (argv[1]
 && (strcmp(argv[1], "single") == 0 || strcmp(argv[1], "-s") == 0 || LONE_CHAR(argv[1], "1"))
) {
    /* ??? shouldn"t we set RUNLEVEL="b" here? */
    /* Start a shell on console */
    new_init_action(RESPAWN, bb_default_login_shell, "");
} else {
    /* Not in single user mode - see what inittab says */

    /* NOTE that if CONFIG_FEATURE_USE_INITTAB is NOT defined,
     * then parse_inittab() simply adds in some default
     * actions (i.e., INIT_SCRIPT and a pair
     * of "askfirst" shells) */
    parse_inittab();
}

这一段代码是一个if判断,注释说如果是single user mode,则走上半段代码,如果不是single user mode,则调用parse_inittab() 函数,因为内核启动init进程没有传入附加参数,所以argv[1]不存在,程序走parse_inittab()

注释还说如果没有定义CONFIG_FEATURE_USE_INITTAB 这个宏,程序会执行一些默认的action,那怎么知道这个宏定义了没有呢,猜测这个宏应该是对busybox配置时的选项,好,怎么查看busybox配置呢,和linux内核配置一样的道理,结合make menuconfig和各级config文件来看

是否定义了宏CONFIG_FEATURE_USE_INITTAB?

在busybox中执行make meunconfig,进入熟悉的配置界面

大概浏览一下,和init有关系的好像有个Init Utilities项,进去

这里面有一项“Support reading an inittab file”,这个配置项是选中的,描述中有“inittab”这个单词,和init源码中说到的parse_inittab()很相似,好,make menuconfig先放到一边,来看看配置文件,打开顶层目录的Config.in,全局搜一下"init",发现只有最下面有:

source init/Config.in

进入init文件夹,打开其中的Config.in文件,发现配置项

config FEATURE_USE_INITTAB
    bool "Support reading an inittab file"
    default y
    depends on INIT
    help
      Allow init to read an inittab file when the system boot.

猜测没错,CONFIG_FEATURE_USE_INITTAB这个宏确实定义了,回到init源码分析,进入parse_inittab()函数。首先看到这个函数前有一大段注释,看看它说什么

/* NOTE that if CONFIG_FEATURE_USE_INITTAB is NOT defined,
 * then parse_inittab() simply adds in some default
 * actions (i.e., runs INIT_SCRIPT and then starts a pair
 * of "askfirst" shells).  If CONFIG_FEATURE_USE_INITTAB
 * _is_ defined, but /etc/inittab is missing, this
 * results in the same set of default behaviors.
 */

前面的话和之前的if判断意思差不多,如果定义了XXX这个宏,但是/etc/inittab这个文件没有,也会走默认的action,好,大概猜想一下,parse_inttab()函数好像和要分析的app启动有点关系了,如果定义了XXX宏,就去解析/etc/inittab这个文件,执行里面的东西,如果没有定义XXX宏或者/etc/inittab文件不存在,执行一些默认的东西

好,搞清楚一件事,/etc/inittab这个文件很重要,可能需要自己来创建这个文件,往里面写东西,但是写什么内容呢?目前还不知道。那如果不走/etc/inittab这一条路呢,默认会执行的action又是什么意思?来分析一下parse_inittab()这个函数

static void parse_inittab(void)
{
#if ENABLE_FEATURE_USE_INITTAB
    char *token[4];
    parser_t *parser = config_open2("/etc/inittab", fopen_for_read);

    if (parser == NULL)
#endif
    {
        /* No inittab file - set up some default behavior */
        /* Sysinit */
        new_init_action(SYSINIT, INIT_SCRIPT, "");
        /* Askfirst shell on tty1-4 */
        new_init_action(ASKFIRST, bb_default_login_shell, "");
//TODO: VC_1 instead of ""? "" is console -> ctty problems -> angry users
        new_init_action(ASKFIRST, bb_default_login_shell, VC_2);
        new_init_action(ASKFIRST, bb_default_login_shell, VC_3);
        new_init_action(ASKFIRST, bb_default_login_shell, VC_4);
        /* Reboot on Ctrl-Alt-Del */
        new_init_action(CTRLALTDEL, "reboot", "");
        /* Umount all filesystems on halt/reboot */
        new_init_action(SHUTDOWN, "umount -a -r", "");
        /* Swapoff on halt/reboot */
        new_init_action(SHUTDOWN, "swapoff -a", "");
        /* Restart init when a QUIT is received */
        new_init_action(RESTART, "init", "");
        return;
    }

#if ENABLE_FEATURE_USE_INITTAB
    /* optional_tty:ignored_runlevel:action:command
     * Delims are not to be collapsed and need exactly 4 tokens
     */
    while (config_read(parser, token, 4, 0, "#:",
                PARSE_NORMAL & ~(PARSE_TRIM | PARSE_COLLAPSE))) {
        /* order must correspond to SYSINIT..RESTART constants */
        static const char actions[] ALIGN1 =
            "sysinit""wait""once""respawn""askfirst"
            "ctrlaltdel""shutdown""restart";
        int action;
        char *tty = token[0];

        if (!token[3]) /* less than 4 tokens */
            goto bad_entry;
        action = index_in_strings(actions, token[2]);
        if (action < 0 || !token[3][0]) /* token[3]: command */
            goto bad_entry;
        /* turn .*TTY -> /dev/TTY */
        if (tty[0]) {
            tty = concat_path_file("/dev/", skip_dev_pfx(tty));
        }
        new_init_action(1 << action, token[3], tty);
        if (tty[0])
            free(tty);
        continue;
 bad_entry:
        message(L_LOG | L_CONSOLE, "Bad inittab entry at line %d",
                parser->lineno);
    }
    config_close(parser);
#endif
}

首先去读了/etc/inittab这个文件,如果不存在,执行了很多new_init_action() ,如果存在,就走了一个while()循环,猜测应该是解析/etc/inittab文件的内容,根据文件的内容执行new_init_action() 。好,那么inittab文件到底写什么格式,什么东西呢,while()循环里面有一个static const char actions[]数组看起来像是和inittab的内容有关系,里面有“sysinit”等字符串,但是还是没办法搞清楚怎么写inittab文件

inittab文件怎么写

/busybox/examples/下面找到一个inittab脚本的例子,打开,看到一个类似格式说明的句子:

Format for each entry: :::

猜测inittab文件里可以有多条entry,每条entry格式中有id、runlevels、action和process这四项内容,这里也出现了action,和代码里的action数组很像。文件里又说id和runlevels无关紧要,好,要搞清楚inittab怎么写,关键在于理解action和process,继续看说明

action

action包括:sysinit、respawn、askfirst、wait、once、restart、ctrlaltdel、和shutdown共八种,

process

指定要运行的程序和它的参数

然后还说了如果没有inittab文件,则运行以下内容

::sysinit:/etc/init.d/rcS
::askfirst:/bin/sh
::ctrlaltdel:/sbin/reboot
::shutdown:/sbin/swapoff -a
::shutdown:/bin/umount -a -r
::restart:/sbin/init
tty2::askfirst:/bin/sh        
tty3::askfirst:/bin/sh
tty4::askfirst:/bin/sh

这应该就是代码中如果读不到inittab文件,则执行的一系列net_init_action的内容

再往下看,出现的第一条示例entry

::sysinit:/etc/init.d/rcS

是不是很熟悉,linux系统嵌入式设备里通常会有/etc/init.d/rcS这个文件,它是一个shell脚本,根据前面的格式,分析一下,id和runlevel为空,action为sysinit,process为/etc/init.d/rcS,所以第一件要干的事情是去执行rcS脚本,而rcS脚本里可以做自己想做的任何事情了

下一条示例是

::askfirst:-/bin/sh

注释说的是启动shell到console口,不管,继续看

tty4::respawn:/sbin/getty 38400 tty5
tty5::respawn:/sbin/getty 38400 tty6

开启getty

::restart:/sbin/init

指定init进程的重启位置

::ctrlaltdel:/sbin/reboot
::shutdown:/bin/umount -a -r
::shutdown:/sbin/swapoff -a

在重启之前要做的事情

再回到代码上,这个while()循环遍历了inittab文件的每一个entry,解析出entry的四个部分:id、runlevel、action和process,放到一个指针数组char *token[4]中,则token[2]token[3]代表action和process,程序里调用index_in_strings()函数将token[2]转成字符串,即“sysinit”等值,再调用net_init_action(),分析net_init_action()源码可以看出,其实只是把这些action和process添加到一个链表中,并没有做实际的处理,真正的处理在后续的代码中,parse_inittab()函数返回,

    ......
    /* Now run everything that needs to be run */
    /* First run the sysinit command */
    run_actions(SYSINIT);
    check_delayed_sigs();
    /* Next run anything that wants to block */
    run_actions(WAIT);
    check_delayed_sigs();
    /* Next run anything to be run only once */
    run_actions(ONCE);

    /* Now run the looping stuff for the rest of forever.
     */
    while (1) {
        ......

这里调用run_action()运行链表中每一个entry,并且首先运行的是action为sysinit的动作

总结

到这里,大致搞清楚了init进程是怎么启动app的了,上流程图

简单来说,init进程首先分析/etc/inittab文件,当然,可以自己修改busybox源码,让它从任意文件开始分析,如果不存在inittab文件,则执行默认的action;如果inittab文件存在,则根据inittab文件中的条目执行,通常是去/etc/init.d/rcS文件中执行脚本命令,当然,修改源码,你也可以让它执行别的脚本

rcS脚本是以shell脚本语言编写,一般的套路是

加载驱动模块

配置网络,建桥、配网卡地址

启动app

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/10275.html

相关文章

  • Android系统的创世之初以及Activity的生命周期

    摘要:在系统启动完成完成后,将变为守护进程监视系统其他进程。由上可知进程是系统中所有其他用户进程的祖先进程。进程负责创建系统中的几个关键进程,其中之一的,是世界的开创者。 该文章是一个系列文章,是本人在Android开发的漫漫长途上的一点感想和记录,如果能给各位看官带来一丝启发或者帮助,那真是极好的。 前言 先来个最简单的HelloWord代码,用Android Studio 3.0新建项...

    AlienZHOU 评论0 收藏0
  • 当我们按下电源键,Android 究竟做了些什么?

    摘要:根据组的特性,这些进程会同时启动或停止。另外,配置含有属性,这意味着它是系统关键进程如果进程不幸在分钟内异常退出超过次,设备将重启并进入还原模式。当每次重启时,其他关键进程等也会被。欢迎大家前往腾讯云+社区,获取更多腾讯海量技术实践干货哦~ 本文由goo发表于云+社区专栏 相信我们对Android系统都不陌生,而Android系统博大精深,被各种各样的智能设备承载的同时,我们会否好奇过...

    番茄西红柿 评论0 收藏0
  • Android后台杀死系列之三:LowMemoryKiller原理(4.3-6.0)

    摘要:本篇是后台杀死系列的第三篇,前面两篇已经对后台杀死注意事项,杀死恢复机制做了分析,本篇主要讲解的是后台杀死原理。有关保存和恢复状态或者异常杀死恢复可以参考前两篇文章。 本篇是Android后台杀死系列的第三篇,前面两篇已经对后台杀死注意事项,杀死恢复机制做了分析,本篇主要讲解的是Android后台杀死原理。相对于后台杀死恢复,LowMemoryKiller原理相对简单,并且在网上还是能...

    jhhfft 评论0 收藏0
  • App启动流程

    摘要:孵化进程相当于是系统的根进程,后面所有的进程都是通过这个进程出来的虽然进程相当于系统的根进程,但是事实上它也是由系统的进程启动的。 目录介绍 1.什么是Zygote进程 1.1 简单介绍 1.2 各个进程的先后顺序 1.3 进程作用说明 2.Zygote进程的启动流程 2.1 源码位置 2.2 ZygoteInit类的main方法 2.3 registerZygoteSo...

    geekidentity 评论0 收藏0
  • android源码分析—Zygote和SystemServer启动

    摘要:注此次分析以源码为例。孵化器受精卵名字是受精卵,其实就是帮助或其他进程启动的一个玩意儿。启动系统服务是系统的大核心之一,和一并重要,专管所有的系统服务。每个进程都走这一步这个分支到此先不往下跟踪了,和启动的过程关系不大了。 注:此次分析以6.0源码为例。 android系统是从linux改过来的,因此这里从init进程开始进行分析。 init初始化过程 让我们进入init.cpp来看看...

    RyanHoo 评论0 收藏0

发表评论

0条评论

lidashuang

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<