]> git.proxmox.com Git - mirror_ubuntu-eoan-kernel.git/blame - arch/arm64/include/asm/stage2_pgtable.h
Merge remote-tracking branch 'asoc/fix/cs4271' into asoc-linus
[mirror_ubuntu-eoan-kernel.git] / arch / arm64 / include / asm / stage2_pgtable.h
CommitLineData
c0ef6326
SP
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/*
da04fa04
SP
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).
c0ef6326 35 */
da04fa04 36#define STAGE2_PGTABLE_LEVELS ARM64_HW_PGTABLE_LEVELS(KVM_PHYS_SHIFT - 4)
c0ef6326
SP
37
38/*
da04fa04
SP
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...
c0ef6326 51 */
da04fa04
SP
52#if STAGE2_PGTABLE_LEVELS > CONFIG_PGTABLE_LEVELS
53#error "Unsupported combination of guest IPA and host VA_BITS."
c0ef6326
SP
54#endif
55
da04fa04
SP
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
c0ef6326
SP
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
da04fa04
SP
87#define stage2_pud_table_empty(pudp) kvm_page_empty(pudp)
88
89static 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
c0ef6326
SP
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)
da04fa04
SP
113#define stage2_pmd_table_empty(pmdp) kvm_page_empty(pmdp)
114
115static 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;
c0ef6326 118
da04fa04
SP
119 return (boundary - 1 < end - 1) ? boundary : end;
120}
121
122#endif /* STAGE2_PGTABLE_LEVELS > 2 */
c0ef6326
SP
123
124#define stage2_pte_table_empty(ptep) kvm_page_empty(ptep)
c0ef6326 125
da04fa04
SP
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>
c0ef6326
SP
130#endif
131
da04fa04
SP
132
133#define stage2_pgd_index(addr) (((addr) >> S2_PGDIR_SHIFT) & (PTRS_PER_S2_PGD - 1))
134
135static 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}
c0ef6326
SP
141
142#endif /* __ARM64_S2_PGTABLE_H_ */