1.前言
本文基于高通8996平台,kernel版本为3.18.31。
本文主要介绍head.S的calc_phys_offset执行流程
2. 几个宏定义
- PHYS_OFFSET
#arch/arm64/include/asm/memory.h
/* PHYS_OFFSET - the physical address of the start of memory. */
#define PHYS_OFFSET ({ memstart_addr; })
内存的起始物理地址
- PAGE_OFFSET
#arch/arm64/include/asm/memory.h
#define PAGE_OFFSET (UL(0xffffffffffffffff) << (VA_BITS - 1))
kernel image起始的虚拟内存地址
定义在kernel/arch/arm64/include/asm/memory.h中,我们使用CONFIG_ARM64_VA_BITS为39,PAGE_OFFSET等于0xffffffc000000000
3.__calc_phys_offset代码说明
物理地址相对虚拟地址的偏移保存到x28,计算kernel_image的物理地址保存到x24,这个就是PHYS_OFFSET
/*
* Calculate the start of physical memory.
*/
__calc_phys_offset:
// adr伪指令用于将一个地址加载到寄存器,
// 取到的是相对于PC寄存器的地址,由于此刻PC寄存器中值是物理地址,
// 所以x0中取到的即是标号1处的物理地址
adr x0, 1f
// 将标号1处的的前八字节(.的虚拟地址)给x1,后八字节即PAGE_OFFSET给x2
ldp x1, x2, [x0]
// .处的物理地址减去虚拟地址,即x28中保存的是物理地址相对虚拟地址的偏移
sub x28, x0, x1
// 计算出kernel image起始的物理地址给x24
// PAGE_OFFSET+虚拟地址与物理地址偏移
add x24, x2, x28
ret
ENDPROC(__calc_phys_offset)
.align 3
// 用来定义一个quad word也就是4字(8字节),“.”表示当前虚拟地址
1: .quad .
//此处是说将PAGE_OFFSET放置在这个位置,占用8个字节
.quad PAGE_OFFSE
4.总结
__calc_phys_offset的主要作用就是通过计算物理地址与虚拟地址偏移,保存到x28, 而kernel image的虚拟地址为PAGE_OFFSET,因此可以获知kernel image的物理地址,将kernel image的物理地址保存到x24中