arm64_linux head.S的执行流程- 4.calc_phys_offset

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中

上一篇:海思芯片文件及查看方法


下一篇:linux 设备树中 dwc3 节点的phys参数含义