2026/1/9 5:57:46
网站建设
项目流程
网站建设优化网站排名,专业网站设计公司排行榜,深圳企业宣传片,拼多多seo搜索优化文章目录1. 前言2. 概述2.1 两级分页2.2 三级分页3. 参考资料1. 前言
限于作者能力水平#xff0c;本文可能存在谬误#xff0c;因此而给读者带来的损失#xff0c;作者不做任何承诺。
2. 概述
本文以 Linux 4.14.x 在 ARMv7 架构下#xff0c;分别对两级和三级分页进行…文章目录1. 前言2. 概述2.1 两级分页2.2 三级分页3. 参考资料1. 前言限于作者能力水平本文可能存在谬误因此而给读者带来的损失作者不做任何承诺。2. 概述本文以Linux 4.14.x在ARMv7架构下分别对两级和三级分页进行讨论。在讨论之前先假定4G虚拟地址空间按1G:3G划分中断向量位于4G虚拟地址最高地址位置(Vector High)一如下图2.1 两级分页在没有启用LPAE(Large Physical Address Extension)功能的情形下内核最多使用两级分页进行寻址。注意这里的最多表示有些情形下可以一级分页两级分页是上限。先上图Linux 下两级分页寻址过程如下仅用到TTBR0来存储第一级页表物理基址用来进行page table walk其值初始在__enable_mmu中设置后续会随着进程切换变化而TTBR1仅用来备份内核页表 swapper_pg_dir的物理地址其值在proc-v7-2level.S:v7_ttb_setup中设置在切换页表期间临时拷贝覆盖TTBR0避免非法的地址访问。来简单看一下上图形成的过程。先看TTBR0TTBR1初始化过程__v7_ca7mp_setup:...__errata_finish:mov r10,#0...#ifdefCONFIG_MMU.../* * r10 0 * r8 内核页表 swapper_pg_dir 物理基址 */v7_ttb_setup r10,r4,r5,r8,r3 TTBCR,TTBRx setup...#endif.macro v7_ttb_setup,zero,ttbr0l,ttbr0h,ttbr1,tmp// TTBCR 0mcr p15,0,\zero,c2,c0,2 TTB controlregisterALT_SMP(orr \ttbr0l,\ttbr0l,#TTB_FLAGS_SMP)// \ttbr0l | TTB_FLAGS_SMP (r4 | TTB_FLAGS_SMP)...ALT_SMP(orr \ttbr1,\ttbr1,#TTB_FLAGS_SMP)// \ttbr1 | TTB_FLAGS_SMP (r8 | TTB_FLAGS_SMP)...// TTBR1 swapper_pg_dir 页表物理地址mcr p15,0,\ttbr1,c2,c0,1 load TTB1.endmBOOT CPU上TTBCRTTBR0TTBR1从初始化如上分析。注意MMU是每 CPU的那自然TTBCRTTBR0TTBR1也是每 CPU的。在非BOOT CPU上除了TTBR0初始为idmap_pgd的物理地址外TTBR1仍然初始化为swapper_pg_dir页表物理地址这里就不做展开了。再简单看下内核lowmem的映射建立包括内核镜像区间start_kernel()setup_arch()paging_init()map_lowmem()map_lowmem()建立了lowmem的页表映射内核区间用1MB section大小进行映射只使用了一级映射正如前面提到的。接下来看下进程第一级页表的创建copy_process()copy_mm()dup_mm()mm_init()mm_alloc_pgd()pgd_alloc()/* * need to get a 16k page for level 1 */pgd_t*pgd_alloc(structmm_struct*mm){pgd_t*new_pgd,*init_pgd;pud_t*new_pud,*init_pud;pmd_t*new_pmd,*init_pmd;pte_t*new_pte,*init_pte;/* 分配页目录表(第一级页表)空间(表项为未配置状态) */new_pgd__pgd_alloc();if(!new_pgd)gotono_pgd;/* 页目录表(第一级页表)用户空间部分的所有表项清 0 */memset(new_pgd,0,USER_PTRS_PER_PGD*sizeof(pgd_t));/* * Copy over the kernel and IO PGD entries *//* 拷贝 内核空间页目录表映射表项 到新分配页目录表的内核空间映射部分 */init_pgdpgd_offset_k(0);memcpy(new_pgdUSER_PTRS_PER_PGD,init_pgdUSER_PTRS_PER_PGD,(PTRS_PER_PGD-USER_PTRS_PER_PGD)*sizeof(pgd_t));/* 清除新页目录表空间的 cache */clean_dcache_area(new_pgd,PTRS_PER_PGD*sizeof(pgd_t));returnnew_pgd;}最后看一下进程切换时的页表切换__schedule()context_switch()switch_mm_irqs_off()switch_mm()check_and_switch_context()voidcheck_and_switch_context(structmm_struct*mm,structtask_struct*tsk){unsignedlongflags;unsignedintcpusmp_processor_id();u64 asid;.../* TTBR0 TTBR1 swapper_pg_dir 物理地址 */cpu_set_reserved_ttbr0();...switch_mm_fastpath:/* 切换到目标进程 tsk 的页表: TTBR0 标进程 tsk 的页表 */cpu_switch_mm(mm-pgd,mm);}#definecpu_switch_mm(pgd,mm)cpu_do_switch_mm(virt_to_phys(pgd),mm)// cpu_v7_switch_mm2.2 三级分页在启用了LPAE(Large Physical Address Extension)功能的情形下内核最多使用三级分页进行寻址。照样先上图Linux 下三级分页寻址过程如下Linux 下三级分页寻址过程同时使用TTBR0和TTBR1进行page table walkTTBR1指向内核空间 1GB 页表物理基址且其值始终不会变化TTBR0用来存储每进程第一级页表物理基址。来简单看一下上图形成的过程。先看TTBR0TTBR1初始化过程和二级分页基本一致只不过TTBCRTTBR1设置是通过proc-v7-3level.S:v7_ttb_setup设置时TTBR1并不是指向内核页表swapper_pg_dir的物理首地址而是指向了第二级的最后一个 PMD 页表的物理首地址为啥这样来看看。三级分页使用Long-descriptor格式我们讨论的场景是1G:3G划分所以swapper_pg_dir包含的最后一个 PMD 页表刚好映射内核的1G空间。那为什么不让TTBR1指向swapper_pg_dir的物理首地址呢这和硬件的实现有关看一下相关原文简单来说在启用了LPAE(Large Physical Address Extension)功能的情形下寻址内核 [0xC0000000, 0xFFFFFFFF] 空间时从第二级 PMD 页表开始 page table walk 的过程这就是TTBR1指向最后一个第二级 PMD 页表的原因。寻址时VA的 PGD Index 部分可以不用关心了因为确定寻址的内核空间。Linux 下三级分页的进程第一级页表的创建和页表切换过程和二级分页基本相同这里就不再赘述了。3. 参考资料[1] DDI0406C_d_armv7ar_arm.pdf[2] TLB原理