鸿蒙轻内核A核源码分析系列之虚实映射(4)虚实映射查询

开发 前端
虚实映射其实就是一个建立页表的过程。MMU支持多级页表,LiteOS-A内核采用二级页表描述进程空间。首先介绍下一级页表和二级页表。

[[438478]]

想了解更多内容,请访问:

51CTO和华为官方合作共建的鸿蒙技术社区

https://harmonyos.51cto.com

4、虚实映射查询函数LOS_ArchMmuQuery

给定一个虚拟内存地址,可以查询其映射到的物理内存地址,还可以查询映射标签属性信息,函数LOS_ArchMmuQuery负责完成这些信息的查询。

4.1 函数LOS_ArchMmuQuery

函数LOS_ArchMmuQuery用于获取进程空间虚拟地址对应的物理地址以及映射标签属性,其中输入参数为虚拟内存地址vaddr,输出参数为物理内存地址*paddr和标签属性*flags。⑴处获取虚拟地址对应的页表项。⑵处如果虚拟地址对应的页表项描述符类型无效,返回错误码。⑶处如果页表项描述符类型为L1页表Section类型映射,则执行⑷获取映射的物理地址,其中MMU_DESCRIPTOR_L1_SECTION_ADDR(l1Entry)为L1页表项的高12位,(vaddr & (MMU_DESCRIPTOR_L1_SMALL_SIZE - 1))为虚拟地址的低20位,即页内偏移值。可以和上文了解到的知识相对应,物理内存地址的计算方式为页表项的高12位加上虚拟内存地址的低20位,如下图所示。⑸处获取映射的标签属性,把MMU标签转换为内存区域标签。

鸿蒙轻内核A核源码分析系列五 虚实映射(4)虚实映射查询-鸿蒙HarmonyOS技术社区

如果虚拟地址对应的页表项描述符类型为页表Page Table,则执行⑹调用内联函数OsGetPte2BasePtr()计算L2页表项基地址,计算方法为:取L1页表项的高22位,低10位置0,得到L2页表项物理内存基地址,然后转化为L2页表项虚拟内存基地址。⑺处计算虚拟地址对应的L2页表项数值,从上文可知,L2页表项的指针地址在页表项基地址加上虚拟内存地址的高20位,取该地址的数据即为L2页表项数据。如果L2页表项描述符类型为小页,则执行⑻计算物理内存地址,其中MMU_DESCRIPTOR_L2_SMALL_PAGE_ADDR(l2Entry)为L2页表项的高20位;vaddr & (MMU_DESCRIPTOR_L2_SMALL_SIZE - 1)为虚拟地址的低12位,如下图所示。然后计算相应的标签值。⑼处表示当前轻内核还不支持大页类型。

鸿蒙轻内核A核源码分析系列五 虚实映射(4)虚实映射查询-鸿蒙HarmonyOS技术社区
  1. STATUS_T LOS_ArchMmuQuery(const LosArchMmu *archMmu, VADDR_T vaddr, PADDR_T *paddr, UINT32 *flags) 
  2. ⑴  PTE_T l1Entry = OsGetPte1(archMmu->virtTtb, vaddr); 
  3.     PTE_T l2Entry; 
  4.     PTE_T* l2Base = NULL
  5.  
  6. ⑵  if (OsIsPte1Invalid(l1Entry)) { 
  7.         return LOS_ERRNO_VM_NOT_FOUND; 
  8. ⑶  } else if (OsIsPte1Section(l1Entry)) { 
  9.         if (paddr != NULL) { 
  10. ⑷          *paddr = MMU_DESCRIPTOR_L1_SECTION_ADDR(l1Entry) + (vaddr & (MMU_DESCRIPTOR_L1_SMALL_SIZE - 1)); 
  11.         } 
  12.  
  13.         if (flags != NULL) { 
  14. ⑸          OsCvtSecAttsToFlags(l1Entry, flags); 
  15.         } 
  16.     } else if (OsIsPte1PageTable(l1Entry)) { 
  17. ⑹      l2Base = OsGetPte2BasePtr(l1Entry); 
  18.         if (l2Base == NULL) { 
  19.             return LOS_ERRNO_VM_NOT_FOUND; 
  20.         } 
  21. ⑺      l2Entry = OsGetPte2(l2Base, vaddr); 
  22.         if (OsIsPte2SmallPage(l2Entry) || OsIsPte2SmallPageXN(l2Entry)) { 
  23.             if (paddr != NULL) { 
  24. ⑻               *paddr = MMU_DESCRIPTOR_L2_SMALL_PAGE_ADDR(l2Entry) + (vaddr & (MMU_DESCRIPTOR_L2_SMALL_SIZE - 1)); 
  25.             } 
  26.  
  27.             if (flags != NULL) { 
  28.                 OsCvtPte2AttsToFlags(l1Entry, l2Entry, flags); 
  29.             } 
  30. ⑼      } else if (OsIsPte2LargePage(l2Entry)) { 
  31.             LOS_Panic("%s %d, large page unimplemented\n", __FUNCTION__, __LINE__); 
  32.         } else { 
  33.             return LOS_ERRNO_VM_NOT_FOUND; 
  34.         } 
  35.     } 
  36.  
  37.     return LOS_OK; 

想了解更多内容,请访问:

51CTO和华为官方合作共建的鸿蒙技术社区

https://harmonyos.51cto.com

 

责任编辑:jianghua 来源: 鸿蒙社区
相关推荐

2021-12-02 15:08:23

鸿蒙HarmonyOS应用

2021-12-03 16:20:26

鸿蒙HarmonyOS应用

2021-12-01 15:59:22

鸿蒙HarmonyOS应用

2022-03-11 20:23:14

鸿蒙源码分析进程管理

2022-01-12 10:50:23

鸿蒙HarmonyOS应用

2022-01-10 15:31:44

鸿蒙HarmonyOS应用

2022-04-13 11:02:12

鸿蒙事件模块事件Event

2022-03-03 18:28:28

Harmony进程任务管理模块

2021-06-04 09:57:49

鸿蒙HarmonyOS应用

2021-05-25 09:28:34

鸿蒙HarmonyOS应用

2021-10-20 16:08:57

鸿蒙HarmonyOS应用

2022-04-13 11:12:43

鸿蒙轻内核信号量模块操作系统

2021-05-08 15:14:50

鸿蒙HarmonyOS应用

2021-11-05 15:00:33

鸿蒙HarmonyOS应用

2021-05-17 09:28:59

鸿蒙HarmonyOS应用

2021-11-08 15:06:15

鸿蒙HarmonyOS应用

2021-06-04 14:15:10

鸿蒙HarmonyOS应用

2022-01-14 08:39:47

鸿蒙HarmonyOS应用

2021-06-17 09:36:07

鸿蒙HarmonyOS应用

2021-05-31 20:30:55

鸿蒙HarmonyOS应用
点赞
收藏

51CTO技术栈公众号