一区二区三区日韩精品-日韩经典一区二区三区-五月激情综合丁香婷婷-欧美精品中文字幕专区

分享

Linux Kernel 2.6.37 啟動過程:startup

 Liucw2012 2011-12-08
一路走來真是內(nèi)牛滿面,現(xiàn)在終于來到了kernel的32位入口了。這就是startup_32。這個入口在arch/x86/boot/compressed/header_32.S里面。至此,請記住,bootloader對于kernel來說意義只有boot_params結(jié)構(gòu)了,其他的一切的一切已經(jīng)都是浮云了。而boot_params當(dāng)前地址在esi中。
startup_32缺省在內(nèi)存的絕對地址0x10000。讓我們慢慢解析startup_32。
  1. #include <linux/init.h>
  2. #include <linux/linkage.h>
  3. #include <asm/segment.h>
  4. #include <asm/page_types.h>
  5. #include <asm/boot.h>
  6. #include <asm/asm-offsets.h>
  7. __HEAD
  8. ENTRY(startup_32)
  9. cld
  10. /*
  11. * Test KEEP_SEGMENTS flag to see if the bootloader is asking
  12. * us to not reload segments. BP_loadflags(%esi)即指向boot_params.loadflags.這一位應(yīng)該是在設(shè)置了code32_start Hook的時(shí)候使用,因?yàn)樵趐rotected_mode_jump的最后已經(jīng)將所有的段都置為_BOOT_DS了。如果bootloader hook了code32_start,返回kernel的時(shí)候顯然kernel需要去恢復(fù)所有的段。
  13. */
  14. testb $(1<<6), BP_loadflags(%esi)
  15. jnz 1f
  16. cli
  17. movl $__BOOT_DS, %eax
  18. movl %eax, %ds
  19. movl %eax, %es
  20. movl %eax, %fs
  21. movl %eax, %gs
  22. movl %eax, %ss
  23. 1:
  24. /*
  25. * Calculate the delta between where we were compiled to run
  26. * at and where we were actually loaded at. This can only be done
  27. * with a short local call on x86. Nothing else will tell us what
  28. * address we are running at. The reserved chunk of the real-mode
  29. * data at 0x1e4 (defined as a scratch field) are used as the stack
  30. * for this calculation. Only 4 bytes are needed.
  31. */
  32. //仔細(xì)閱讀了以上的這段英文,不能不說代碼構(gòu)思的巧妙。由于不知道代碼是否被加載到0x100000的地址,通過以下的代碼就能計(jì)算出實(shí)際加載的地址和預(yù)期地址的差異,也就是說是實(shí)際的startup_32的位置。
  33. leal (BP_scratch+4)(%esi), %esp //boot_params.scratch的地址設(shè)置成為堆棧頂。
  34. call 1f //boot_params.scratch里面就是1:的實(shí)際地址
  35. 1: popl %ebp //ebp就是1:的實(shí)際地址
  36. subl $1b, %ebp //ebp-1:就是實(shí)際與預(yù)期的差異, 也就是說是實(shí)際的startup_32的位置。
  37. /*
  38. * %ebp contains the address we are loaded at by the boot loader and %ebx
  39. * contains the address where we should move the kernel image temporarily
  40. * for safe in-place decompression.
  41. */
  42. #ifdef CONFIG_RELOCATABLE
  43. movl %ebp, %ebx
  44. //kernel_alignment里面是kernel地址對齊所需要移動的位移量,這是有bootloader填入的,因?yàn)閎ootloader可能將startup_32裝載在非對齊的地址。那么就需要增加移動的位移量來保證對齊而達(dá)到更好的性能。下面的代碼就是要調(diào)整地址的位移而保證對齊。
  45. movl BP_kernel_alignment(%esi), %eax
  46. decl %eax
  47. addl %eax, %ebx
  48. notl %eax
  49. andl %eax, %ebx
  50. #else
  51. movl $LOAD_PHYSICAL_ADDR, %ebx //LOAD_PHYSICAL_ADDR在 arch/x86/include/asm/boot.h里.實(shí)際上應(yīng)該是0x100000
  52. #endif
  53. /* Target address to relocate to for decompression */
  54. addl $z_extract_offset, %ebx //z_extract_offset由MKpiggy.c 在編譯時(shí)產(chǎn)生的piggy.S里面定義。在我編譯kernel時(shí),z_extract_offset是0x4a0000,現(xiàn)在ebx的值在不考慮reloc的情況下是0x5a0000
  55. /* Set up the stack */
  56. leal boot_stack_end(%ebx), %esp //在0x5a0000+boot_stack_end的位置建立棧。
  57. /* Zero EFLAGS */
  58. pushl $0
  59. popfl
  60. /*
  61. * Copy the compressed kernel to the end of our buffer
  62. * where decompression in place becomes safe.
  63. */
  64. pushl %esi
  65. leal (_bss-4)(%ebp), %esi //esi指向源,即ebp+_bss-4的地址,是當(dāng)前bootloader加載32位kernel的地址空間
  66. leal (_bss-4)(%ebx), %edi //edi指向目的地址,即ebx+_bss-4的地址,如果kernel不要reloc,就是0x5a0000+_bss-4
  67. movl $(_bss - startup_32), %ecx //從startup_32d到_bss有多少個字節(jié)?
  68. shrl $2, %ecx //實(shí)際我們移動每次4個字節(jié),所以ecx需要除4.
  69. std
  70. rep movsl //走咯,我們把自己移動上去
  71. cld
  72. popl %esi
  73. /*
  74. * Jump to the relocated address.
  75. */
  76. leal relocated(%ebx), %eax
  77. jmp *%eax //跳轉(zhuǎn)到relocated上去即ebx+relocated,即0x5a0000+relocated.
  78. ENDPROC(startup_32)
  79. .text
  80. relocated:
  81. /*
  82. * Clear BSS (stack is currently empty)
  83. */
  84. xorl %eax, %eax
  85. leal _bss(%ebx), %edi
  86. leal _ebss(%ebx), %ecx
  87. subl %edi, %ecx
  88. shrl $2, %ecx
  89. rep stosl
  90. /*
  91. * Adjust our own GOT GOT是什么?難道是Global Object Table?為什么GOT里面的每一個項(xiàng)都加上了ebx(0x5a0000)?難道里面是一堆指針需要調(diào)整所以加上ebx?
  92. */
  93. leal _got(%ebx), %edx
  94. leal _egot(%ebx), %ecx
  95. 1:
  96. cmpl %ecx, %edx
  97. jae 2f
  98. addl %ebx, (%edx)
  99. addl $4, %edx
  100. jmp 1b
  101. 2:
  102. /*
  103. * Do the decompression, and jump to the new kernel..
  104. */
  105. leal z_extract_offset_negative(%ebx), %ebp //ebp=ebx-0x4a0000=0x100000
  106. /* push arguments for decompress_kernel: */
  107. pushl %ebp /* output address */ //將Kernel解壓縮到0x100000
  108. pushl $z_input_len /* input_len */ //壓縮過的kernel大小
  109. leal input_data(%ebx), %eax //壓縮kernel開始地址
  110. pushl %eax /* input_data */
  111. leal boot_heap(%ebx), %eax //工作的堆
  112. pushl %eax /* heap area */
  113. pushl %esi /* real mode pointer */ //esi是boot_params
  114. call decompress_kernel //我不準(zhǔn)備去看怎么解壓,只要知道它解壓了好了
  115. addl $20, %esp //看來不需要恢復(fù)寄存器
  116. #if CONFIG_RELOCATABLE
  117. /*
  118. * Find the address of the relocations.
  119. */
  120. leal z_output_len(%ebp), %edi
  121. /*
  122. * Calculate the delta between where vmlinux was compiled to run
  123. * and where it was actually loaded.
  124. */
  125. movl %ebp, %ebx
  126. subl $LOAD_PHYSICAL_ADDR, %ebx
  127. jz 2f /* Nothing to be done if loaded at compiled addr. */ //如果ebx=0x100000,則不許要reloc
  128. /*
  129. * Process relocations. //這段沒懂,但應(yīng)該不影響理解
  130. */
  131. 1: subl $4, %edi
  132. movl (%edi), %ecx
  133. testl %ecx, %ecx
  134. jz 2f
  135. addl %ebx, -__PAGE_OFFSET(%ebx, %ecx)
  136. jmp 1b
  137. 2:
  138. #endif
  139. /*
  140. * Jump to the decompressed kernel.
  141. */
  142. xorl %ebx, %ebx
  143. jmp *%ebp //重新跳到0x100000開始。
  144. /*
  145. * Stack and heap for uncompression
  146. */
  147. .bss
  148. .balign 4
  149. boot_heap:
  150. .fill BOOT_HEAP_SIZE, 1, 0
  151. boot_stack:
  152. .fill BOOT_STACK_SIZE, 1, 0
  153. boot_stack_end:
把以上的代碼總結(jié)一下其實(shí)很簡單,startup_32將壓縮過的kernel和本身移動到0x100000(或bootloader裝載startup_32的地址)+0x4a0000的位置,然后解壓縮kernel回到0x100000(或bootloader裝載startup_32的地址),然后將控制權(quán)交回到0x100000((或bootloader裝載startup_32的地址)).

啟動代碼終于結(jié)束了,明天就要進(jìn)入真正的kernel了。寫得有些亂,就是對代碼進(jìn)行注釋,暫時(shí)找不到更好的方法來對代碼進(jìn)行解釋,可能使大家看起來有點(diǎn)累。希望對大家有幫助

    本站是提供個人知識管理的網(wǎng)絡(luò)存儲空間,所有內(nèi)容均由用戶發(fā)布,不代表本站觀點(diǎn)。請注意甄別內(nèi)容中的聯(lián)系方式、誘導(dǎo)購買等信息,謹(jǐn)防詐騙。如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,請點(diǎn)擊一鍵舉報(bào)。
    轉(zhuǎn)藏 分享 獻(xiàn)花(0

    0條評論

    發(fā)表

    請遵守用戶 評論公約

    類似文章 更多

    果冻传媒精选麻豆白晶晶| 日韩欧美综合在线播放| 欧美三级大黄片免费看| 91人妻人澡人人爽人人精品| 国产精品欧美一级免费| 五月婷婷六月丁香在线观看| 久久青青草原中文字幕| 99久久精品一区二区国产| 日本午夜福利视频免费观看| 欧美字幕一区二区三区| 少妇熟女精品一区二区三区| 国产盗摄精品一区二区视频 | 欧美日韩最近中国黄片| 免费观看一级欧美大片| 成人三级视频在线观看不卡 | 高清不卡视频在线观看| 日本不卡在线视频中文国产| 亚洲熟妇av一区二区三区色堂| 精品人妻一区二区三区免费看| 在线免费看国产精品黄片| 91麻豆精品欧美视频| 亚洲国产av国产av| 日韩日韩日韩日韩在线| 欧美日韩视频中文字幕| 一区二区三区四区亚洲另类| 日本午夜免费福利视频| 欧美日韩中黄片免费看| 好吊色欧美一区二区三区顽频 | 亚洲中文字幕在线综合视频| 精品少妇一区二区三区四区| 国产日韩欧美在线播放| 亚洲内射人妻一区二区| 国产欧美日本在线播放| 亚洲一区二区三区四区性色av | 久热这里只有精品九九| 亚洲精品国产精品日韩| 东京热加勒比一区二区| 中国黄色色片色哟哟哟哟哟哟| 91亚洲精品综合久久| 少妇一区二区三区精品| 色偷偷亚洲女人天堂观看|