鴻蒙輕內(nèi)核A核源碼分析系列之虛實(shí)映射(4)虛實(shí)映射查詢
51CTO和華為官方合作共建的鴻蒙技術(shù)社區(qū)
4、虛實(shí)映射查詢函數(shù)LOS_ArchMmuQuery
給定一個(gè)虛擬內(nèi)存地址,可以查詢其映射到的物理內(nèi)存地址,還可以查詢映射標(biāo)簽屬性信息,函數(shù)LOS_ArchMmuQuery負(fù)責(zé)完成這些信息的查詢。
4.1 函數(shù)LOS_ArchMmuQuery
函數(shù)LOS_ArchMmuQuery用于獲取進(jìn)程空間虛擬地址對應(yīng)的物理地址以及映射標(biāo)簽屬性,其中輸入?yún)?shù)為虛擬內(nèi)存地址vaddr,輸出參數(shù)為物理內(nèi)存地址*paddr和標(biāo)簽屬性*flags。⑴處獲取虛擬地址對應(yīng)的頁表項(xiàng)。⑵處如果虛擬地址對應(yīng)的頁表項(xiàng)描述符類型無效,返回錯(cuò)誤碼。⑶處如果頁表項(xiàng)描述符類型為L1頁表Section類型映射,則執(zhí)行⑷獲取映射的物理地址,其中MMU_DESCRIPTOR_L1_SECTION_ADDR(l1Entry)為L1頁表項(xiàng)的高12位,(vaddr & (MMU_DESCRIPTOR_L1_SMALL_SIZE - 1))為虛擬地址的低20位,即頁內(nèi)偏移值??梢院蜕衔牧私獾降闹R(shí)相對應(yīng),物理內(nèi)存地址的計(jì)算方式為頁表項(xiàng)的高12位加上虛擬內(nèi)存地址的低20位,如下圖所示。⑸處獲取映射的標(biāo)簽屬性,把MMU標(biāo)簽轉(zhuǎn)換為內(nèi)存區(qū)域標(biāo)簽。

如果虛擬地址對應(yīng)的頁表項(xiàng)描述符類型為頁表Page Table,則執(zhí)行⑹調(diào)用內(nèi)聯(lián)函數(shù)OsGetPte2BasePtr()計(jì)算L2頁表項(xiàng)基地址,計(jì)算方法為:取L1頁表項(xiàng)的高22位,低10位置0,得到L2頁表項(xiàng)物理內(nèi)存基地址,然后轉(zhuǎn)化為L2頁表項(xiàng)虛擬內(nèi)存基地址。⑺處計(jì)算虛擬地址對應(yīng)的L2頁表項(xiàng)數(shù)值,從上文可知,L2頁表項(xiàng)的指針地址在頁表項(xiàng)基地址加上虛擬內(nèi)存地址的高20位,取該地址的數(shù)據(jù)即為L2頁表項(xiàng)數(shù)據(jù)。如果L2頁表項(xiàng)描述符類型為小頁,則執(zhí)行⑻計(jì)算物理內(nèi)存地址,其中MMU_DESCRIPTOR_L2_SMALL_PAGE_ADDR(l2Entry)為L2頁表項(xiàng)的高20位;vaddr & (MMU_DESCRIPTOR_L2_SMALL_SIZE - 1)為虛擬地址的低12位,如下圖所示。然后計(jì)算相應(yīng)的標(biāo)簽值。⑼處表示當(dāng)前輕內(nèi)核還不支持大頁類型。

- STATUS_T LOS_ArchMmuQuery(const LosArchMmu *archMmu, VADDR_T vaddr, PADDR_T *paddr, UINT32 *flags)
- {
- ⑴ PTE_T l1Entry = OsGetPte1(archMmu->virtTtb, vaddr);
- PTE_T l2Entry;
- PTE_T* l2Base = NULL;
- ⑵ if (OsIsPte1Invalid(l1Entry)) {
- return LOS_ERRNO_VM_NOT_FOUND;
- ⑶ } else if (OsIsPte1Section(l1Entry)) {
- if (paddr != NULL) {
- ⑷ *paddr = MMU_DESCRIPTOR_L1_SECTION_ADDR(l1Entry) + (vaddr & (MMU_DESCRIPTOR_L1_SMALL_SIZE - 1));
- }
- if (flags != NULL) {
- ⑸ OsCvtSecAttsToFlags(l1Entry, flags);
- }
- } else if (OsIsPte1PageTable(l1Entry)) {
- ⑹ l2Base = OsGetPte2BasePtr(l1Entry);
- if (l2Base == NULL) {
- return LOS_ERRNO_VM_NOT_FOUND;
- }
- ⑺ l2Entry = OsGetPte2(l2Base, vaddr);
- if (OsIsPte2SmallPage(l2Entry) || OsIsPte2SmallPageXN(l2Entry)) {
- if (paddr != NULL) {
- ⑻ *paddr = MMU_DESCRIPTOR_L2_SMALL_PAGE_ADDR(l2Entry) + (vaddr & (MMU_DESCRIPTOR_L2_SMALL_SIZE - 1));
- }
- if (flags != NULL) {
- OsCvtPte2AttsToFlags(l1Entry, l2Entry, flags);
- }
- ⑼ } else if (OsIsPte2LargePage(l2Entry)) {
- LOS_Panic("%s %d, large page unimplemented\n", __FUNCTION__, __LINE__);
- } else {
- return LOS_ERRNO_VM_NOT_FOUND;
- }
- }
- return LOS_OK;
- }
51CTO和華為官方合作共建的鴻蒙技術(shù)社區(qū)