xref: /kvm-unit-tests/lib/arm64/asm/pgtable.h (revision 6ca8c283190847bd2ff648232f98f70e4e6d3d0d)
1 #ifndef _ASMARM64_PGTABLE_H_
2 #define _ASMARM64_PGTABLE_H_
3 /*
4  * Adapted from arch/arm64/include/asm/pgtable.h
5  *              include/asm-generic/pgtable-nopmd.h
6  *              include/linux/mm.h
7  *
8  * Note: some Linux function APIs have been modified. Nothing crazy,
9  *       but if a function took, for example, an mm_struct, then
10  *       that was either removed or replaced.
11  *
12  * Copyright (C) 2017, Red Hat Inc, Andrew Jones <drjones@redhat.com>
13  *
14  * This work is licensed under the terms of the GNU GPL, version 2.
15  */
16 #include <alloc.h>
17 #include <asm/setup.h>
18 #include <asm/page.h>
19 #include <asm/pgtable-hwdef.h>
20 
21 /*
22  * We can convert va <=> pa page table addresses with simple casts
23  * because we always allocate their pages with alloc_page(), and
24  * alloc_page() always returns identity mapped pages.
25  */
26 #define pgtable_va(x)		((void *)(unsigned long)(x))
27 #define pgtable_pa(x)		((unsigned long)(x))
28 
29 #define pgd_none(pgd)		(!pgd_val(pgd))
30 #define pmd_none(pmd)		(!pmd_val(pmd))
31 #define pte_none(pte)		(!pte_val(pte))
32 
33 #define pgd_index(addr) \
34 	(((addr) >> PGDIR_SHIFT) & (PTRS_PER_PGD - 1))
35 #define pgd_offset(pgtable, addr) ((pgtable) + pgd_index(addr))
36 
37 #define pgd_free(pgd) free(pgd)
38 static inline pgd_t *pgd_alloc(void)
39 {
40 	pgd_t *pgd = memalign(PAGE_SIZE, PTRS_PER_PGD * sizeof(pgd_t));
41 	memset(pgd, 0, PTRS_PER_PGD * sizeof(pgd_t));
42 	return pgd;
43 }
44 
45 #define pmd_offset(pgd, addr)	((pmd_t *)pgd)
46 #define pmd_free(pmd)
47 #define pmd_alloc(pgd, addr)	pmd_offset(pgd, addr)
48 
49 static inline pte_t *pmd_page_vaddr(pmd_t pmd)
50 {
51 	return pgtable_va(pmd_val(pmd) & PHYS_MASK & (s32)PAGE_MASK);
52 }
53 
54 #define pte_index(addr) \
55 	(((addr) >> PAGE_SHIFT) & (PTRS_PER_PTE - 1))
56 #define pte_offset(pmd, addr) \
57 	(pmd_page_vaddr(*(pmd)) + pte_index(addr))
58 
59 #define pte_free(pte) free_page(pte)
60 static inline pte_t *pte_alloc_one(void)
61 {
62 	assert(PTRS_PER_PTE * sizeof(pte_t) == PAGE_SIZE);
63 	pte_t *pte = alloc_page();
64 	return pte;
65 }
66 static inline pte_t *pte_alloc(pmd_t *pmd, unsigned long addr)
67 {
68 	if (pmd_none(*pmd)) {
69 		pte_t *pte = pte_alloc_one();
70 		pmd_val(*pmd) = pgtable_pa(pte) | PMD_TYPE_TABLE;
71 	}
72 	return pte_offset(pmd, addr);
73 }
74 
75 #endif /* _ASMARM64_PGTABLE_H_ */
76