Merge branch 'for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/jikos/hid
[cascardo/linux.git] / arch / arm64 / include / asm / stage2_pgtable.h
1 /*
2  * Copyright (C) 2016 - ARM Ltd
3  *
4  * stage2 page table helpers
5  *
6  * This program is free software; you can redistribute it and/or modify
7  * it under the terms of the GNU General Public License version 2 as
8  * published by the Free Software Foundation.
9  *
10  * This program is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13  * GNU General Public License for more details.
14  *
15  * You should have received a copy of the GNU General Public License
16  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
17  */
18
19 #ifndef __ARM64_S2_PGTABLE_H_
20 #define __ARM64_S2_PGTABLE_H_
21
22 #include <asm/pgtable.h>
23
24 /*
25  * The hardware supports concatenation of up to 16 tables at stage2 entry level
26  * and we use the feature whenever possible.
27  *
28  * Now, the minimum number of bits resolved at any level is (PAGE_SHIFT - 3).
29  * On arm64, the smallest PAGE_SIZE supported is 4k, which means
30  *             (PAGE_SHIFT - 3) > 4 holds for all page sizes.
31  * This implies, the total number of page table levels at stage2 expected
32  * by the hardware is actually the number of levels required for (KVM_PHYS_SHIFT - 4)
33  * in normal translations(e.g, stage1), since we cannot have another level in
34  * the range (KVM_PHYS_SHIFT, KVM_PHYS_SHIFT - 4).
35  */
36 #define STAGE2_PGTABLE_LEVELS           ARM64_HW_PGTABLE_LEVELS(KVM_PHYS_SHIFT - 4)
37
38 /*
39  * With all the supported VA_BITs and 40bit guest IPA, the following condition
40  * is always true:
41  *
42  *       STAGE2_PGTABLE_LEVELS <= CONFIG_PGTABLE_LEVELS
43  *
44  * We base our stage-2 page table walker helpers on this assumption and
45  * fall back to using the host version of the helper wherever possible.
46  * i.e, if a particular level is not folded (e.g, PUD) at stage2, we fall back
47  * to using the host version, since it is guaranteed it is not folded at host.
48  *
49  * If the condition breaks in the future, we can rearrange the host level
50  * definitions and reuse them for stage2. Till then...
51  */
52 #if STAGE2_PGTABLE_LEVELS > CONFIG_PGTABLE_LEVELS
53 #error "Unsupported combination of guest IPA and host VA_BITS."
54 #endif
55
56 /* S2_PGDIR_SHIFT is the size mapped by top-level stage2 entry */
57 #define S2_PGDIR_SHIFT                  ARM64_HW_PGTABLE_LEVEL_SHIFT(4 - STAGE2_PGTABLE_LEVELS)
58 #define S2_PGDIR_SIZE                   (_AC(1, UL) << S2_PGDIR_SHIFT)
59 #define S2_PGDIR_MASK                   (~(S2_PGDIR_SIZE - 1))
60
61 /*
62  * The number of PTRS across all concatenated stage2 tables given by the
63  * number of bits resolved at the initial level.
64  */
65 #define PTRS_PER_S2_PGD                 (1 << (KVM_PHYS_SHIFT - S2_PGDIR_SHIFT))
66
67 /*
68  * KVM_MMU_CACHE_MIN_PAGES is the number of stage2 page table translation
69  * levels in addition to the PGD.
70  */
71 #define KVM_MMU_CACHE_MIN_PAGES         (STAGE2_PGTABLE_LEVELS - 1)
72
73
74 #if STAGE2_PGTABLE_LEVELS > 3
75
76 #define S2_PUD_SHIFT                    ARM64_HW_PGTABLE_LEVEL_SHIFT(1)
77 #define S2_PUD_SIZE                     (_AC(1, UL) << S2_PUD_SHIFT)
78 #define S2_PUD_MASK                     (~(S2_PUD_SIZE - 1))
79
80 #define stage2_pgd_none(pgd)                            pgd_none(pgd)
81 #define stage2_pgd_clear(pgd)                           pgd_clear(pgd)
82 #define stage2_pgd_present(pgd)                         pgd_present(pgd)
83 #define stage2_pgd_populate(pgd, pud)                   pgd_populate(NULL, pgd, pud)
84 #define stage2_pud_offset(pgd, address)                 pud_offset(pgd, address)
85 #define stage2_pud_free(pud)                            pud_free(NULL, pud)
86
87 #define stage2_pud_table_empty(pudp)                    kvm_page_empty(pudp)
88
89 static inline phys_addr_t stage2_pud_addr_end(phys_addr_t addr, phys_addr_t end)
90 {
91         phys_addr_t boundary = (addr + S2_PUD_SIZE) & S2_PUD_MASK;
92
93         return (boundary - 1 < end - 1) ? boundary : end;
94 }
95
96 #endif          /* STAGE2_PGTABLE_LEVELS > 3 */
97
98
99 #if STAGE2_PGTABLE_LEVELS > 2
100
101 #define S2_PMD_SHIFT                    ARM64_HW_PGTABLE_LEVEL_SHIFT(2)
102 #define S2_PMD_SIZE                     (_AC(1, UL) << S2_PMD_SHIFT)
103 #define S2_PMD_MASK                     (~(S2_PMD_SIZE - 1))
104
105 #define stage2_pud_none(pud)                            pud_none(pud)
106 #define stage2_pud_clear(pud)                           pud_clear(pud)
107 #define stage2_pud_present(pud)                         pud_present(pud)
108 #define stage2_pud_populate(pud, pmd)                   pud_populate(NULL, pud, pmd)
109 #define stage2_pmd_offset(pud, address)                 pmd_offset(pud, address)
110 #define stage2_pmd_free(pmd)                            pmd_free(NULL, pmd)
111
112 #define stage2_pud_huge(pud)                            pud_huge(pud)
113 #define stage2_pmd_table_empty(pmdp)                    kvm_page_empty(pmdp)
114
115 static inline phys_addr_t stage2_pmd_addr_end(phys_addr_t addr, phys_addr_t end)
116 {
117         phys_addr_t boundary = (addr + S2_PMD_SIZE) & S2_PMD_MASK;
118
119         return (boundary - 1 < end - 1) ? boundary : end;
120 }
121
122 #endif          /* STAGE2_PGTABLE_LEVELS > 2 */
123
124 #define stage2_pte_table_empty(ptep)                    kvm_page_empty(ptep)
125
126 #if STAGE2_PGTABLE_LEVELS == 2
127 #include <asm/stage2_pgtable-nopmd.h>
128 #elif STAGE2_PGTABLE_LEVELS == 3
129 #include <asm/stage2_pgtable-nopud.h>
130 #endif
131
132
133 #define stage2_pgd_index(addr)                          (((addr) >> S2_PGDIR_SHIFT) & (PTRS_PER_S2_PGD - 1))
134
135 static inline phys_addr_t stage2_pgd_addr_end(phys_addr_t addr, phys_addr_t end)
136 {
137         phys_addr_t boundary = (addr + S2_PGDIR_SIZE) & S2_PGDIR_MASK;
138
139         return (boundary - 1 < end - 1) ? boundary : end;
140 }
141
142 #endif  /* __ARM64_S2_PGTABLE_H_ */