内核代码一开始,在完成了el2_setup, set_cpu_boot_mode_flag之后,就开始__create_page_tables
108 ENTRY(stext)
109 bl preserve_boot_args
110 bl el2_setup // Drop to EL1, w0=cpu_boot_mode
111 adrp x23, __PHYS_OFFSET
112 and x23, x23, MIN_KIMG_ALIGN - 1 // KASLR offset, defaults to 0
113 bl set_cpu_boot_mode_flag
114 bl __create_page_tables
[~/work/qemu_work/kernel/linux-5.4.220/arch/arm64/kernel/head.S][asm]
本文主要概述__create_page_tables 的作用和过程。代码没必要一行一行分析,都是基础的汇编代码。并且内核本身有很全面的注释。本文目的是把这个过程拆解成很多个小段,从而更容易理解。
Invalidate the init page tables
adrp x0, init_pg_dir
adrp x1, init_pg_end
sub x1, x1, x0
bl __inval_dcache_area
如上Invalidate了init_pg_dir 到init_pg_end 这一段的dcache。
接着又清除了这一段内存。
init_pg_dir 及 init_pg_end 的定义见 init_pg_dir 的大小及作用
Create the identity mapping
这里主要使用map_memory 来创建映射页表。
恒等映射表定义的位置
.macro map_memory, tbl, rtbl, vstart, vend, flags, phys, pgds, istart, iend, tmp, count, sv
/* map_memory x0, x1, x3, x6, x7, x3, x4, x10, x11, x12, x13, x14
map memory 宏部分参数说明如下
参数 | 变量 | 地址 |
---|---|---|
tbl: 页表的起始地址。页表基地址是 idmap pg dir。 | idmap_pg_dir | x0: 0x41bfa000 |
rtbl: 下一级页表的起始地址。下一级页表的基地址通常是 tbl+ PAGE SIZE | x1 | x1: 0x0 |
vstart: 要映射的虚拟地址的起始地址。这里是 idmap text start。 | __pa(__idmap_text_start) | x3: 0x418e5000 |
vend: 要映射的虚拟地址的结束地址。这里是 idmap text end。 | __pa(__idmap_text_end) | x6: 0x418e56b8 |
flags: 最后一级页表的一些属性 | SWAPPER_MM_MMUFLAGS | x7: 0x711 |
phys:映射对应的物理地址。物理地址的起始地址是 dmap text start。 | __pa(__idmap_text_start) | x3: 0x418e5000 |
pgds: PGD 页表项的个数。 | PTRS_PER_PGD == 0x200 == 512 | x4: 0x200 |
其他参数都是临时使用的。
映射的汇编代码暂时不看,只验证这一步的结果。即虚拟地址 0x418e5000 - 0x418e56b8,被映射到了相同的物理地址0x418e5000 - 0x418e56b8上。详细验证的过程见这个链接
Map the kernel image (starting with PHYS_OFFSET).
map kernel image 时,memory 宏部分参数说明如下
.macro map_memory, tbl, rtbl, vstart, vend, flags, phys, pgds, istart, iend, tmp, count, sv
/* map_memory x0, x1, x5, x6, x7, x3, x4, x10, x11, x12, x13, x14
参数 | 变量 | 地址 |
---|---|---|
tbl: 页表的起始地址。页表基地址是 idmap pg dir。 | init_pg_dir | x0: 0x4223c000 |
rtbl: 下一级页表的起始地址。下一级页表的基地址通常是 tbl+ PAGE SIZE | x1 | x1: 0x41bdd000 |
vstart: 要映射的虚拟地址的起始地址。 | KIMAGE_VADDR + TEXT_OFFSET | x5: 0xffff000010080000 |
vend: 要映射的虚拟地址的结束地址。 | runtime __pa(_end) | x6: 0xffff000012241000 |
flags: 最后一级页表的一些属性 | SWAPPER_MM_MMUFLAGS | x7: 0x711 |
phys:映射对应的物理地址。物理地址的起始地址是 dmap text start。 | runtime __pa(_text) | x3: 0x40080000 |
pgds: PGD 页表项的个数。 | PTRS_PER_PGD == 0x200 == 512 | x4: 0x200 |
如下验证一下,可以看到 0xffff000010080000 确实被映射到了0x40080000