xref: /kvm-unit-tests/lib/arm64/asm/pgtable.h (revision 02f1cdc883df50e65bc84386f99c6b66b906103b)
130b1bc86SAndrew Jones #ifndef _ASMARM64_PGTABLE_H_
230b1bc86SAndrew Jones #define _ASMARM64_PGTABLE_H_
330b1bc86SAndrew Jones /*
430b1bc86SAndrew Jones  * Adapted from arch/arm64/include/asm/pgtable.h
530b1bc86SAndrew Jones  *              include/asm-generic/pgtable-nopmd.h
630b1bc86SAndrew Jones  *              include/linux/mm.h
730b1bc86SAndrew Jones  *
830b1bc86SAndrew Jones  * Note: some Linux function APIs have been modified. Nothing crazy,
930b1bc86SAndrew Jones  *       but if a function took, for example, an mm_struct, then
1030b1bc86SAndrew Jones  *       that was either removed or replaced.
1149f758b8SAndrew Jones  *
1249f758b8SAndrew Jones  * Copyright (C) 2017, Red Hat Inc, Andrew Jones <drjones@redhat.com>
1349f758b8SAndrew Jones  *
1449f758b8SAndrew Jones  * This work is licensed under the terms of the GNU GPL, version 2.
1530b1bc86SAndrew Jones  */
1630b1bc86SAndrew Jones #include <alloc.h>
177510bc97SAlexandru Elisei #include <alloc_page.h>
1830b1bc86SAndrew Jones #include <asm/setup.h>
1930b1bc86SAndrew Jones #include <asm/page.h>
2030b1bc86SAndrew Jones #include <asm/pgtable-hwdef.h>
2130b1bc86SAndrew Jones 
2270cea146SAlexandru Elisei #include <linux/compiler.h>
2370cea146SAlexandru Elisei 
24f02b6363SAndrew Jones /*
25f02b6363SAndrew Jones  * We can convert va <=> pa page table addresses with simple casts
26f02b6363SAndrew Jones  * because we always allocate their pages with alloc_page(), and
27f02b6363SAndrew Jones  * alloc_page() always returns identity mapped pages.
28f02b6363SAndrew Jones  */
29f02b6363SAndrew Jones #define pgtable_va(x)		((void *)(unsigned long)(x))
30f02b6363SAndrew Jones #define pgtable_pa(x)		((unsigned long)(x))
31f02b6363SAndrew Jones 
3230b1bc86SAndrew Jones #define pgd_none(pgd)		(!pgd_val(pgd))
3330b1bc86SAndrew Jones #define pmd_none(pmd)		(!pmd_val(pmd))
3430b1bc86SAndrew Jones #define pte_none(pte)		(!pte_val(pte))
3530b1bc86SAndrew Jones 
36*02f1cdc8SAlexandru Elisei #define pgd_valid(pgd)		(pgd_val(pgd) & PGD_VALID)
37*02f1cdc8SAlexandru Elisei #define pmd_valid(pmd)		(pmd_val(pmd) & PMD_SECT_VALID)
38*02f1cdc8SAlexandru Elisei #define pte_valid(pte)		(pte_val(pte) & PTE_VALID)
39*02f1cdc8SAlexandru Elisei 
40*02f1cdc8SAlexandru Elisei #define pmd_huge(pmd)	\
41*02f1cdc8SAlexandru Elisei 	((pmd_val(pmd) & PMD_TYPE_MASK) == PMD_TYPE_SECT)
42*02f1cdc8SAlexandru Elisei 
4330b1bc86SAndrew Jones #define pgd_index(addr) \
4430b1bc86SAndrew Jones 	(((addr) >> PGDIR_SHIFT) & (PTRS_PER_PGD - 1))
4530b1bc86SAndrew Jones #define pgd_offset(pgtable, addr) ((pgtable) + pgd_index(addr))
4630b1bc86SAndrew Jones 
4730b1bc86SAndrew Jones #define pgd_free(pgd) free(pgd)
4830b1bc86SAndrew Jones static inline pgd_t *pgd_alloc(void)
4930b1bc86SAndrew Jones {
5030b1bc86SAndrew Jones 	pgd_t *pgd = memalign(PAGE_SIZE, PTRS_PER_PGD * sizeof(pgd_t));
5130b1bc86SAndrew Jones 	memset(pgd, 0, PTRS_PER_PGD * sizeof(pgd_t));
5230b1bc86SAndrew Jones 	return pgd;
5330b1bc86SAndrew Jones }
5430b1bc86SAndrew Jones 
550226e262SPaolo Bonzini #define pmd_offset(pgd, addr)	((pmd_t *)pgd)
5630b1bc86SAndrew Jones #define pmd_free(pmd)
570226e262SPaolo Bonzini #define pmd_alloc(pgd, addr)	pmd_offset(pgd, addr)
5830b1bc86SAndrew Jones 
5930b1bc86SAndrew Jones static inline pte_t *pmd_page_vaddr(pmd_t pmd)
6030b1bc86SAndrew Jones {
61f02b6363SAndrew Jones 	return pgtable_va(pmd_val(pmd) & PHYS_MASK & (s32)PAGE_MASK);
6230b1bc86SAndrew Jones }
6330b1bc86SAndrew Jones 
6430b1bc86SAndrew Jones #define pte_index(addr) \
6530b1bc86SAndrew Jones 	(((addr) >> PAGE_SHIFT) & (PTRS_PER_PTE - 1))
6630b1bc86SAndrew Jones #define pte_offset(pmd, addr) \
6730b1bc86SAndrew Jones 	(pmd_page_vaddr(*(pmd)) + pte_index(addr))
6830b1bc86SAndrew Jones 
69031755dbSPaolo Bonzini #define pte_free(pte) free_page(pte)
7030b1bc86SAndrew Jones static inline pte_t *pte_alloc_one(void)
7130b1bc86SAndrew Jones {
72031755dbSPaolo Bonzini 	assert(PTRS_PER_PTE * sizeof(pte_t) == PAGE_SIZE);
73031755dbSPaolo Bonzini 	pte_t *pte = alloc_page();
7430b1bc86SAndrew Jones 	return pte;
7530b1bc86SAndrew Jones }
7630b1bc86SAndrew Jones static inline pte_t *pte_alloc(pmd_t *pmd, unsigned long addr)
7730b1bc86SAndrew Jones {
7830b1bc86SAndrew Jones 	if (pmd_none(*pmd)) {
7970cea146SAlexandru Elisei 		pmd_t entry;
8070cea146SAlexandru Elisei 		pmd_val(entry) = pgtable_pa(pte_alloc_one()) | PMD_TYPE_TABLE;
8170cea146SAlexandru Elisei 		WRITE_ONCE(*pmd, entry);
8230b1bc86SAndrew Jones 	}
8330b1bc86SAndrew Jones 	return pte_offset(pmd, addr);
8430b1bc86SAndrew Jones }
8530b1bc86SAndrew Jones 
8630b1bc86SAndrew Jones #endif /* _ASMARM64_PGTABLE_H_ */
87