序言:Linux的启动代码真的挺大,从汇编到C,从Makefile到LDS文件,须要理解的东西好多。虽然Linux内核是由好多人,耗费了巨大的时间和精力写下来的。并且直至现今,这个世界上依然有成千上万的程序员在不断建立Linux内核的代码。
嵌入式进阶教程分门别类整理好了linux内核源代码情景分析(下册),看的时侯非常便捷,因为内容较多,这儿就截取一部份图吧。
须要的同事私信【内核】即可申领。
内核学习地址:Linux内核源码/显存调优/文件系统/进程管理/设备驱动/网路合同栈-学习视频教程-腾讯课堂
Linux内核启动及文件系统加载过程
当u-boot开始执行bootcmd命令,就步入Linux内核启动阶段,与u-boot类似,普通Linux内核的启动过程也可以分为两个阶段,但针对压缩了的内核如uImage就要包括内核自解压过程了。本文以linux-2.6.37版源码为例分三个阶段来描述内核启动全过程。第一阶段为内核自解压过程,第二阶段主要工作是设置ARM处理器工作模式、使能MMU、设置一级页表等,而第三阶段则主要为C代码,包括内核初始化的全部工作。
Linux内核启动流程
arch/arm/kernel/head-armv.S该文件是内核最先执行的一个文件,包括内核入口ENTRY(stext)到start_kernel间的初始化代码,主要作用是检测CPUID,ArchitectureType,初始化BSS等操作,并跳到start_kernel函数。
在执行前,处理器应满足以下状态:
r0 - should be 0
r1 - unique architecture number
MMU - off
I-cache - on or off
D-cache – off
/*部份源代码剖析*//*内核入口点*/ENTRY(stext)/*程序状态,严禁FIQ、IRQ,设定SVC模式*/movr0,#F_BIT|I_BIT|MODE_SVC@makesuresvcmode/*当前程序状态寄存器*/msrcpsr_c红旗linux安装,r0@andallirqsdisabled/*判定CPU类型,查找运行的CPUID值与Linux编译支持的ID值是否支持*/bl__lookup_processor_type/*跳到__error*/teqr10,#0@invalidprocessor?moveqr0,#‘p’@yes,error‘p’beq__error/*判定体系类型,查看R1寄存器的ArchitectureType值是否支持*/bl__lookup_architecture_type/*不支持,跳到出错*/teqr7,#0@invalidarchitecture?moveqr0,#‘a’@yes,error‘a’beq__error/*创建核心页表*/bl__create_page_tablesadrlr,__ret@returnaddressaddpc,r10,#12@iniTIaliseprocessor/*跳转到start_kernel函数*/bstart_kernelLinux内核启动第一阶段stage1
这儿所以说的第一阶段stage1就是内核解压完成并出现UncompressingLinux.。.done,booTIngthekernel.以后的阶段。该部份代码实现在arch/arm/kernel的head.S中linux vi命令,该文件中的汇编代码通过查找处理器内核类型和机器码类型调用相应的初始化函数,再建立页表,最后跳转到start_kernel()函数开始内核的初始化工作。
检查处理器类型是在汇编子函数__lookup_processor_type中完成的,通过以下代码可实现对它的调用:bl__lookup_processor_type(在文件head-commom.S实现)。__lookup_processor_type调用结束返回原程序时,会将返回结果保存到寄存器中。其中r5寄存器返回一个拿来描述处理器的结构体地址,并对r5进行判别,假如r5的值为0则说明不支持这些处理器,将步入__error_p。r8保存了页表的标志位,r9保存了处理器的ID号,r10保存了与处理器相关的structproc_info_list结构地址。
Head.S核心代码如下:
Linux内核启动第二阶段stage2
从start_kernel函数开始
Linux内核启动的第二阶段从start_kernel函数开始。start_kernel是所有Linux平台步入系统内核初始化后的入口函数,它主要完成剩余的与硬件平台相关的初始化工作,在进行一系列与内核相关的初始化后,调用第一个用户进程-init进程并等待用户进程的执行,这样整个Linux内核便启动完毕。该函数坐落init/main.c文件中,主要工作流程如图3所示:
该函数所做的具体工作有:
ARM-Linux在初始化过程中通常还会初始化一个并口做为内核的控制台,而并口Uart驱动却把并口设备名写死了,如本例中linux2.6.37并口设备名为ttyO0,而不是常用的ttyS0。有了控制台内核在启动过程中就可以通过并口输出信息便于开发者或用户了解系统的启动进程。
创建和初始化系统cache,为各类显存调用机制提供缓存,包括;动态显存分配,虚拟文件系统(VirtualFileSystem)及页缓存。
初始化显存管理,检查显存大小及被内核占用的显存情况;
初始化系统的进程间通讯机制(IPC);当以上所有的初始化工作结束后,start_kernel()函数会调用rest_init()函数来进行最后的初始化,包括创建系统的第一个进程-init进程来结束内核的启动。
挂载根文件系统并启动init
Linux内核启动的下一过程是启动第一个进程init,但必须以根文件系统为载体linux内核源代码情景分析(下册),所以在启动init之前,还要挂载根文件系统。