linux内核通俗理解,简洁明了!高手带你理解ARM-Linux的启动过程

原标题:简洁明了!高手带你理解ARM-Linux的启动过程

160578584_777180

99e0d5ac6c545ce0712eab092d8aa097.png

86bd44411ab5663689a9a6abef2433c2.png

1. kernel运行的史前时期和内存布局

b949c4d7c81b860ffbfc76ef27b6f857.png

e2aea192edbfc559faf642c3a7aedc81.png

在arm平台下,zImage.bin压缩镜像是由bootloader加载到物理内存,然后跳到zImage.bin里一段程序,它专门于将被压缩的kernel解压缩到KERNEL_RAM_PADDR开始的一段内存中,接着跳进真正的kernel去执行。该kernel的执行起点是stext函数,定义于arch/arm/kernel/head.S。

在分析stext函数前,先介绍此时内存的布局如下图所示

74b40a22cd05e2325a70c154d9602b9b.png

在开发板3c2440中,SDRAM连接到内存控制器的Bank6中,它的开始内存地址是0x30000000,大小为64M,即0x20000000。 ARM Linux kernel将SDRAM的开始地址定义为PHYS_OFFSET。经bootloader加载kernel并由自解压部分代码运行后,最终kernel被放置到KERNEL_RAM_PADDR(=PHYS_OFFSET + TEXT_OFFSET,即0x30008000)地址上的一段内存,经此放置后,kernel代码以后均不会被移动。

在进入kernel代码前,即bootloader和自解压缩阶段,ARM未开启MMU功能。因此kernel启动代码一个重要功能是设置好相应的页表,并开启MMU功能。为了支持MMU功能,kernel镜像中的所有符号,包括代码段和数据段的符号,在链接时都生成了它在开启MMU时,所在物理内存地址映射到的虚拟内存地址。

以arm kernel第一个符号(函数)stext为例,在编译链接,它生成的虚拟地址是0xc0008000,而放置它的物理地址为0x30008000(还记得这是PHYS_OFFSET+TEXT_OFFSET吗?)。实际上这个变换可以利用简单的公式进行表示:va = pa – PHYS_OFFSET + PAGE_OFFSET。Arm linux最终的kernel空间的页表,就是按照这个关系来建立。

之所以较早提及arm linux 的内存映射,原因是在进入kernel代码,里面所有符号地址值为清一色的0xCXXXXXXX地址,而此时ARM未开启MMU功能,故在执行stext函数第一条执行时,它的PC值就是stext所在的内存地址(即物理地址,0x30008000)。因此,下面有些代码,需要使用地址无关技术。

39279618b97bcd471c5d960e0da7d600.png

619db0329ed7f82c568e3ab5e472117c.png

2.一览stext函数

6be6ac8c6147db120c3427e5beda9d01.png

112666b7676242872ea98d0440776504.png

stext函数定义在Arch/arm/kernel/head.S,它的功能是获取处理器类型和机器类型信息,并创建临时的页表,然后开启MMU功能,并跳进第一个C语言函数start_kernel。

stext函数的在前置条件是:MMU, D-cache, 关闭; r0 = 0, r1 = machine nr, r2 = atags prointer.

代码如下:

bf721759f8f0bef57ee1cb89708a431d.png

c31034ecb5e24d891f409d845b23b431.png

0687e45345a1183deabcc8077efe65a0.png

62b29cf78999d3da8cc868f3fb303b8f.png

3 __lookup_processor_type 函数

4547465f008e49f8cb8c7fcd6d385e8b.png

33e43ac471d9909bc79766e83e6ade9f.png

__lookup_processor_type 函数是一个非常讲究技巧的函数,如果你将它领会,也将领会kernel了一些魔法。

Kernel代码将所有CPU信息的定义都放到.proc.info.init段中,因此可以认为.proc.info.init段就是一个数组,每个元素都定义了一个或一种CPU的信息。目前__lookup_processor_type使用该元素的前两个字段cpuid和mask来匹配当前CPUID,如果满足CPUID & mask == cpuid,则找到当前cpu的定义并返回。

下面是tqs3c2440开发板,CPU的定义信息,cpuid = 0x41009200,mask = 0xff00fff0。如果是码是运行在tqs3c2440开发板上,那么函数返回下面的定义:

13ead7dee1089060c15550dce34601b7.png

07aa66f6506515ae611c1f7f4b3e7014.png

5ac2d1bda90df6f42c4b7a6a072ffdb7.png

b11764364791f4e07e874ec7146e8d3c.png

7b23b21ef5e69fb5c8b437323e1505d1.png

054cadf3198c7cbe23b3e33cb94bbe77.png

ea5ff01fb6c9c84a683e65d218a9bc3d.png

4 __lookup_machine_type 函数

10afcb3f38a2ae21e2b669b1adb020f1.png

1c0d1acab147a665bd4a4058e5b530fc.png

__lookup_machine_type 和__lookup_processor_type像对孪生兄弟,它们的行为都是很类似的:__lookup_machine_type根据r1寄存器的机器编号到.arch.info.init段的数组中依次查找机器编号与r1相同的记录。它使了与它孪生兄弟同样的手法进行虚拟地址到物理地址的转换计算。

在介绍函数,我们先分析tqs3c2440开发板的机器信息的定义:

f6355bad184e9a4720b009879f1fc1dc.png

MACHINE_START宏用于定义一个.arch.info.init段的数组元素。.nr元素就是函数要比较的变量。Tqs3c2440开发板相应的定义如下:

46b6083b4f998c7d131116ad24d170af.png

这是一个struct machine_desc结构,在后面的C代码(start_kernel开始执行的代码)会使用该变量对象。在tqs3c2440开发中的__lookup_machine_type函数就是返回该对象指针。

这里涉及很多函数指针,它们都是在start_kernel函数里在各种阶段进行初始化的回函数。如map_io指向的tq2440_map_io就是在建立好内核页表后,再调用它来针对开发板的各种IO端口来建立相关的映射和页表。

至于__loopup_machine_type的代码就不作详细分析,请对比__lookup_processor_type来自行分析。代码如下:

c5abbcabb053d94610978f6c03c28eda.png

6cd74deb0a8b41e62cba9ff1618139e3.png

2047c6971ac1d068dfcc2f2b0d8f5adb.png

5. 为kernel建立临时页表

df18ad09d3dd80759041217ba02c933d.png

e00e97123f619e4366ed9676676df59e.png

前面提及到,kernel里面的所有符号在链接时,都使用了虚拟地址值。在完成基本的初始化后,kernel代码将跳到第一个c语言函数start_kernl来执行,在哪个时候,这些虚拟地址必须能够对它所存放在真正内存位置,否则运行将为出错。为此,CPU必须开启MMU,但在开启MMU前,必须为虚拟地址到物理地址的映射建立相应的面表。在开启MMU后,kernel指并不马上将PC值指向start_kernl,而是要做一些C语言运行期的设置,如堆栈,重定义等工作后才跳到start_kernel去执行。在此过程中,PC值还是物理地址,因此还需要为这段内存空间建立va = pa的内存映射关系。当然,本函数建立的所有页表都会在将来paging_init销毁再重建,这是临时过度性的映射关系和页表。

在介绍__create_table_pages前,先认识一个macro pgtbl,它将KERNL_RAM_PADDR – 0x4000的值赋给rd寄存器,从下面的使用中可以看它,该值是页表在物理内存的基础,也即页表放在kernel开始地址下的16K的地方。

365b6ce2936cdb3fe475ba917bb791fc.png

999ac2b372cc0866a03ccdfe2e6cdcf4.png

9f5420732b4a04cefe5cfeed4602d129.png

b56a65de6797c57351717c0363d7ba85.png

1df91d8814720bbe1f01fecd4afc9b8a.png

一口气将__create_pages_table分析完,但里涉及的代码还是需要细细品读。尤其是右移20位和18位两个地方与页表目录项的地址关系比较复杂。执行完该函数后,虚拟内存和物理内存的映射关系如下图所示:

7c7d8d5fa0615716c0739d3b2f2787cd.png

501ff2d7128dec55875f0799fc8aba59.png

27e7b34b6026d0cbff8e6be735f8663d.png

6. 开启MMU

c55ed03e02a41ac3267c2276d9676f4b.png

8fd518dcd2bdb7612c14e36be7c77198.png

看完页表的建立,想必开启MMU的代码也是小菜一碟吧。此函数的主要功能是将页表的基址加到cp15中的面表指针寄存器,同时设置域访问(domain access)寄存器。

2d8480095027baf80686b855bce88325.png

42585de378ec66861a4aee47073e94c3.png

bed10c1a9ec315234ccb7e9b8bfffe48.png

b288cd19ab744ebf874695dbe5fcf858.png

7.__mmap_switched函数

541a768a9a48d0e50566a8cbba4bbca3.png

8948404a7bb6f706a55316c294480ba3.png

__mmap_switched函数专用来设置C语言的执行环境,比如重定位工作,堆栈,以及BSS段的清零。

__switch_data变量先定义了一系里面处量的数据,如重定位和数据段的地址,BSS段的地址,pocessor_id和__mach_arch_type变量的地址等。

5b306ae6eb366ec1e4b07cfcc823eb78.png

3740901643e0ecda37c85bbb019c16cb.png

160578584_777180

160578584_777180

责任编辑:

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值