成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Linux啟動分析——init進程與app啟動

系統(tǒng) Linux
本文通過簡要分析init進程源碼,梳理其處理流程,重點關(guān)注init進程如何啟動應(yīng)用程序,總結(jié)啟動腳本文件的編寫思路。

概述

本文通過簡要分析init進程源碼,梳理其處理流程,重點關(guān)注init進程如何啟動應(yīng)用程序,總結(jié)啟動腳本文件的編寫思路

init進程源碼分析

init進程是linux內(nèi)核啟動的第一個進程,怎么知道的?從內(nèi)核源碼linux-2.6.xxx/init/main.c代碼的kernel_init()函數(shù)分析,可以發(fā)現(xiàn),內(nèi)核會根據(jù)uboot傳入的參數(shù)來啟動第一個進程,一般都是init

怎么啟動的呢,調(diào)用kernel_execve()函數(shù)完成的,猜測是從根文件系統(tǒng)的/sbin/init來啟動的,linux的任何應(yīng)用程序都是基于文件系統(tǒng)的,啟動應(yīng)用程序前提是根文件系統(tǒng)已經(jīng)掛載好了。好,那么根文件系統(tǒng)又是從哪里來的呢,是由busybox這個工具配置編譯生成的,所以要分析init源碼,要去busybox里找init的源碼

源碼位置:/busybox/init/init.c,在其中查找main()函數(shù),發(fā)現(xiàn)只有init_main(),沒有main(),可以猜測busybox是通過一些方法將init進程的入口修改為init_main(),實際上所有busybox的命令工具都是一個到busybox程序的鏈接,

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

可以看到,init進程其實是到busybox的鏈接,不用管它,知道init進程的入口是init_main()函數(shù)就行了

  1. #if DEBUG_SEGV_HANDLER 
  2.     { 
  3.         struct sigaction sa; 
  4.         memset(&sa, 0, sizeof(sa)); 
  5.         sa.sa_sigaction = handle_sigsegv; 
  6.         sa.sa_flags = SA_SIGINFO; 
  7.         sigaction(SIGSEGV, &sa, NULL); 
  8.         ...... 
  9.     } 
  10. #endif 
  11. ...... 
  12. console_init(); 
  13. set_sane_term(); 
  14. ...... 
  15. /* Make sure environs is set to something sane */ 
  16. putenv((char *) "HOME=/"); 
  17. putenv((char *) bb_PATH_root_path); 
  18. putenv((char *) "SHELL=/bin/sh"); 
  19. putenv((char *) "USER=root"); /* needed? why? */  

這一段是init進程最開始要做的事情,設(shè)置一些信號相關(guān)的東西,初始化console,然后設(shè)置環(huán)境變量,跟啟動app似乎沒有什么關(guān)系,不用管,繼續(xù)往下看

  1. /* Check if we are supposed to be in single user mode */ 
  2. if (argv[1] 
  3.  && (strcmp(argv[1], "single") == 0 || strcmp(argv[1], "-s") == 0 || LONE_CHAR(argv[1], '1')) 
  4. ) { 
  5.     /* ??? shouldn't we set RUNLEVEL="b" here? */ 
  6.     /* Start a shell on console */ 
  7.     new_init_action(RESPAWN, bb_default_login_shell, ""); 
  8. else { 
  9.     /* Not in single user mode - see what inittab says */ 
  10.  
  11.     /* NOTE that if CONFIG_FEATURE_USE_INITTAB is NOT defined, 
  12.      * then parse_inittab() simply adds in some default 
  13.      * actions (i.e., INIT_SCRIPT and a pair 
  14.      * of "askfirst" shells) */ 
  15.     parse_inittab(); 
  16.  

這一段代碼是一個if判斷,注釋說如果是single user mode,則走上半段代碼,如果不是single user mode,則調(diào)用parse_inittab() 函數(shù),因為內(nèi)核啟動init進程沒有傳入附加參數(shù),所以argv[1]不存在,程序走parse_inittab()

注釋還說如果沒有定義CONFIG_FEATURE_USE_INITTAB 這個宏,程序會執(zhí)行一些默認(rèn)的action,那怎么知道這個宏定義了沒有呢,猜測這個宏應(yīng)該是對busybox配置時的選項,好,怎么查看busybox配置呢,和linux內(nèi)核配置一樣的道理,結(jié)合make menuconfig和各級config文件來看

是否定義了宏CONFIG_FEATURE_USE_INITTAB?

在busybox中執(zhí)行make meunconfig,進入熟悉的配置界面

 

大概瀏覽一下,和init有關(guān)系的好像有個Init Utilities項,進去

 

這里面有一項“Support reading an inittab file”,這個配置項是選中的,描述中有“inittab”這個單詞,和init源碼中說到的parse_inittab()很相似,好,make menuconfig先放到一邊,來看看配置文件,打開頂層目錄的Config.in,全局搜一下"init",發(fā)現(xiàn)只有最下面有:

  1. source init/Config.in 

進入init文件夾,打開其中的Config.in文件,發(fā)現(xiàn)配置項

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

猜測沒錯,CONFIG_FEATURE_USE_INITTAB這個宏確實定義了,回到init源碼分析,進入parse_inittab()函數(shù)。首先看到這個函數(shù)前有一大段注釋,看看它說什么

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

前面的話和之前的if判斷意思差不多,如果定義了XXX這個宏,但是/etc/inittab這個文件沒有,也會走默認(rèn)的action,好,大概猜想一下,parse_inttab()函數(shù)好像和要分析的app啟動有點關(guān)系了,如果定義了XXX宏,就去解析/etc/inittab這個文件,執(zhí)行里面的東西,如果沒有定義XXX宏或者/etc/inittab文件不存在,執(zhí)行一些默認(rèn)的東西

好,搞清楚一件事,/etc/inittab這個文件很重要,可能需要自己來創(chuàng)建這個文件,往里面寫東西,但是寫什么內(nèi)容呢?目前還不知道。那如果不走/etc/inittab這一條路呢,默認(rèn)會執(zhí)行的action又是什么意思?來分析一下parse_inittab()這個函數(shù)

  1. static void parse_inittab(void) 
  2. #if ENABLE_FEATURE_USE_INITTAB 
  3.     char *token[4]; 
  4.     parser_t *parser = config_open2("/etc/inittab", fopen_for_read); 
  5.  
  6.     if (parser == NULL
  7. #endif 
  8.     { 
  9.         /* No inittab file - set up some default behavior */ 
  10.         /* Sysinit */ 
  11.         new_init_action(SYSINIT, INIT_SCRIPT, ""); 
  12.         /* Askfirst shell on tty1-4 */ 
  13.         new_init_action(ASKFIRST, bb_default_login_shell, ""); 
  14. //TODO: VC_1 instead of """" is console -> ctty problems -> angry users 
  15.         new_init_action(ASKFIRST, bb_default_login_shell, VC_2); 
  16.         new_init_action(ASKFIRST, bb_default_login_shell, VC_3); 
  17.         new_init_action(ASKFIRST, bb_default_login_shell, VC_4); 
  18.         /* Reboot on Ctrl-Alt-Del */ 
  19.         new_init_action(CTRLALTDEL, "reboot"""); 
  20.         /* Umount all filesystems on halt/reboot */ 
  21.         new_init_action(SHUTDOWN, "umount -a -r"""); 
  22.         /* Swapoff on halt/reboot */ 
  23.         new_init_action(SHUTDOWN, "swapoff -a"""); 
  24.         /* Restart init when a QUIT is received */ 
  25.         new_init_action(RESTART, "init"""); 
  26.         return
  27.     } 
  28.  
  29. #if ENABLE_FEATURE_USE_INITTAB 
  30.     /* optional_tty:ignored_runlevel:action:command 
  31.      * Delims are not to be collapsed and need exactly 4 tokens 
  32.      */ 
  33.     while (config_read(parser, token, 4, 0, "#:"
  34.                 PARSE_NORMAL & ~(PARSE_TRIM | PARSE_COLLAPSE))) { 
  35.         /* order must correspond to SYSINIT..RESTART constants */ 
  36.         static const char actions[] ALIGN1 = 
  37.             "sysinit\0""wait\0""once\0""respawn\0""askfirst\0" 
  38.             "ctrlaltdel\0""shutdown\0""restart\0"
  39.         int action
  40.         char *tty = token[0]; 
  41.  
  42.         if (!token[3]) /* less than 4 tokens */ 
  43.             goto bad_entry; 
  44.         action = index_in_strings(actions, token[2]); 
  45.         if (action < 0 || !token[3][0]) /* token[3]: command */ 
  46.             goto bad_entry; 
  47.         /* turn .*TTY -> /dev/TTY */ 
  48.         if (tty[0]) { 
  49.             tty = concat_path_file("/dev/", skip_dev_pfx(tty)); 
  50.         } 
  51.         new_init_action(1 << action, token[3], tty); 
  52.         if (tty[0]) 
  53.             free(tty); 
  54.         continue
  55.  bad_entry: 
  56.         message(L_LOG | L_CONSOLE, "Bad inittab entry at line %d"
  57.                 parser->lineno); 
  58.     } 
  59.     config_close(parser); 
  60. #endif 
  61.  

首先去讀了/etc/inittab這個文件,如果不存在,執(zhí)行了很多new_init_action() ,如果存在,就走了一個while()循環(huán),猜測應(yīng)該是解析/etc/inittab文件的內(nèi)容,根據(jù)文件的內(nèi)容執(zhí)行new_init_action() 。好,那么inittab文件到底寫什么格式,什么東西呢,while()循環(huán)里面有一個static const char actions[]數(shù)組看起來像是和inittab的內(nèi)容有關(guān)系,里面有“sysinit”等字符串,但是還是沒辦法搞清楚怎么寫inittab文件

inittab文件怎么寫

/busybox/examples/下面找到一個inittab腳本的例子,打開,看到一個類似格式說明的句子:

  1. Format for each entry: <id>:<runlevels>:<action>:<process> 

猜測inittab文件里可以有多條entry,每條entry格式中有id、runlevels、action和process這四項內(nèi)容,這里也出現(xiàn)了action,和代碼里的action數(shù)組很像。文件里又說id和runlevels無關(guān)緊要,好,要搞清楚inittab怎么寫,關(guān)鍵在于理解action和process,繼續(xù)看說明

action

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

process

指定要運行的程序和它的參數(shù)

然后還說了如果沒有inittab文件,則運行以下內(nèi)容

  1. ::sysinit:/etc/init.d/rcS 
  2. ::askfirst:/bin/sh 
  3. ::ctrlaltdel:/sbin/reboot 
  4. ::shutdown:/sbin/swapoff -a 
  5. ::shutdown:/bin/umount -a -r 
  6. ::restart:/sbin/init 
  7. tty2::askfirst:/bin/sh         
  8. tty3::askfirst:/bin/sh 
  9. tty4::askfirst:/bin/sh  

這應(yīng)該就是代碼中如果讀不到inittab文件,則執(zhí)行的一系列net_init_action的內(nèi)容

再往下看,出現(xiàn)的第一條示例entry

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

是不是很熟悉,linux系統(tǒng)嵌入式設(shè)備里通常會有/etc/init.d/rcS這個文件,它是一個shell腳本,根據(jù)前面的格式,分析一下,id和runlevel為空,action為sysinit,process為/etc/init.d/rcS,所以第一件要干的事情是去執(zhí)行rcS腳本,而rcS腳本里可以做自己想做的任何事情了

下一條示例是

  1. ::askfirst:-/bin/sh 

注釋說的是啟動shell到console口,不管,繼續(xù)看

  1. tty4::respawn:/sbin/getty 38400 tty5 
  2.  
  3. tty5::respawn:/sbin/getty 38400 tty6  

開啟getty

  1. ::restart:/sbin/init 

指定init進程的重啟位置

  1. ::ctrlaltdel:/sbin/reboot 
  2.  
  3. ::shutdown:/bin/umount -a -r 
  4.  
  5. ::shutdown:/sbin/swapoff -a  

在重啟之前要做的事情

再回到代碼上,這個while()循環(huán)遍歷了inittab文件的每一個entry,解析出entry的四個部分:id、runlevel、action和process,放到一個指針數(shù)組char *token[4]中,則token[2]和token[3]代表action和process,程序里調(diào)用index_in_strings()函數(shù)將token[2]轉(zhuǎn)成字符串,即“sysinit”等值,再調(diào)用net_init_action(),分析net_init_action()源碼可以看出,其實只是把這些action和process添加到一個鏈表中,并沒有做實際的處理,真正的處理在后續(xù)的代碼中,parse_inittab()函數(shù)返回,

  1. ...... 
  2.    /* Now run everything that needs to be run */ 
  3.    /* First run the sysinit command */ 
  4.    run_actions(SYSINIT); 
  5.    check_delayed_sigs(); 
  6.    /* Next run anything that wants to block */ 
  7.    run_actions(WAIT); 
  8.    check_delayed_sigs(); 
  9.    /* Next run anything to be run only once */ 
  10.    run_actions(ONCE); 
  11.  
  12.    /* Now run the looping stuff for the rest of forever. 
  13.     */ 
  14.    while (1) { 
  15.        ......  

這里調(diào)用run_action()運行鏈表中每一個entry,并且首先運行的是action為sysinit的動作

總結(jié)

到這里,大致搞清楚了init進程是怎么啟動app的了,上流程圖 

 

簡單來說,init進程首先分析/etc/inittab文件,當(dāng)然,可以自己修改busybox源碼,讓它從任意文件開始分析,如果不存在inittab文件,則執(zhí)行默認(rèn)的action;如果inittab文件存在,則根據(jù)inittab文件中的條目執(zhí)行,通常是去/etc/init.d/rcS文件中執(zhí)行腳本命令,當(dāng)然,修改源碼,你也可以讓它執(zhí)行別的腳本

rcS腳本是以shell腳本語言編寫,一般的套路是

  • 加載驅(qū)動模塊
  • 配置網(wǎng)絡(luò),建橋、配網(wǎng)卡地址
  • 啟動app 
責(zé)任編輯:龐桂玉 來源: segmentfault
相關(guān)推薦

2010-03-02 16:37:53

Linux Quagg

2022-04-12 12:35:02

Linux啟動性能systemd

2021-10-20 10:04:47

鴻蒙HarmonyOS應(yīng)用

2010-01-05 13:11:04

Ubuntu進程管理

2021-11-23 10:25:35

性能優(yōu)化iOS App 啟動優(yōu)化

2011-06-28 13:27:13

ARM Linux

2010-03-03 09:16:17

2009-08-28 16:41:12

啟動C# AutoCA

2011-02-13 13:44:08

Linux關(guān)機重新啟動

2018-03-13 13:00:03

Linux運維啟動分析

2017-01-23 21:05:00

AndroidApp啟動優(yōu)化

2021-09-18 14:26:49

Linux Linux 啟動流程Linux 系統(tǒng)

2010-07-19 14:37:01

Perl進程啟動函數(shù)

2010-08-03 11:23:30

路由器

2017-02-15 09:40:38

JavaScript分析解決

2009-06-18 13:18:32

軟件測試需求分析

2020-09-09 10:00:41

JavaScript前端瓶頸

2009-10-27 15:06:15

Linux內(nèi)核啟動

2015-01-22 14:12:40

Android源碼APP啟動引導(dǎo)

2011-03-21 10:19:35

啟動停止Nagios
點贊
收藏

51CTO技術(shù)棧公眾號

主站蜘蛛池模板: 91毛片在线观看 | 91精品一区| 国产精品久久777777 | 欧美精品一二区 | 国产精品一区二区免费 | 欧美精品中文 | 中文字幕精品一区二区三区精品 | 91最新在线视频 | 一区二区三区国产精品 | 成人av一区 | 欧美福利 | 一区二区三区免费观看 | 日韩视频在线免费观看 | 精品国产乱码久久久久久闺蜜 | 国产午夜精品一区二区三区四区 | 久久成人一区 | 操视频网站| av在线免费观看网站 | 91色视频在线观看 | 一级片免费观看 | 在线观看视频亚洲 | 亚洲欧美久久 | 黄色av网站在线观看 | www.久| 国产精品久久久久久婷婷天堂 | 911精品美国片911久久久 | 国产免费一区二区三区 | 国产这里只有精品 | 国产一区二区不卡 | av片免费| 日韩在线精品 | 日韩一级精品视频在线观看 | 久久久久久国产 | 国产视频线观看永久免费 | 久久不卡区 | 国产91在线 | 亚洲 | 日韩视频一区二区三区 | 国产精品美女久久久久aⅴ国产馆 | 97免费视频在线观看 | 亚洲欧美一区二区三区情侣bbw | 国产在线精品免费 |