diff options
Diffstat (limited to 'arch/x86')
-rw-r--r-- | arch/x86/kvm/mmu.c | 79 | ||||
-rw-r--r-- | arch/x86/kvm/mmu.h | 6 |
2 files changed, 82 insertions, 3 deletions
diff --git a/arch/x86/kvm/mmu.c b/arch/x86/kvm/mmu.c index 33cd7c982dd3..f7541fe22cd8 100644 --- a/arch/x86/kvm/mmu.c +++ b/arch/x86/kvm/mmu.c | |||
@@ -1097,6 +1097,7 @@ static void mmu_alloc_roots(struct kvm_vcpu *vcpu) | |||
1097 | int i; | 1097 | int i; |
1098 | gfn_t root_gfn; | 1098 | gfn_t root_gfn; |
1099 | struct kvm_mmu_page *sp; | 1099 | struct kvm_mmu_page *sp; |
1100 | int metaphysical = 0; | ||
1100 | 1101 | ||
1101 | root_gfn = vcpu->arch.cr3 >> PAGE_SHIFT; | 1102 | root_gfn = vcpu->arch.cr3 >> PAGE_SHIFT; |
1102 | 1103 | ||
@@ -1105,14 +1106,20 @@ static void mmu_alloc_roots(struct kvm_vcpu *vcpu) | |||
1105 | hpa_t root = vcpu->arch.mmu.root_hpa; | 1106 | hpa_t root = vcpu->arch.mmu.root_hpa; |
1106 | 1107 | ||
1107 | ASSERT(!VALID_PAGE(root)); | 1108 | ASSERT(!VALID_PAGE(root)); |
1109 | if (tdp_enabled) | ||
1110 | metaphysical = 1; | ||
1108 | sp = kvm_mmu_get_page(vcpu, root_gfn, 0, | 1111 | sp = kvm_mmu_get_page(vcpu, root_gfn, 0, |
1109 | PT64_ROOT_LEVEL, 0, ACC_ALL, NULL); | 1112 | PT64_ROOT_LEVEL, metaphysical, |
1113 | ACC_ALL, NULL); | ||
1110 | root = __pa(sp->spt); | 1114 | root = __pa(sp->spt); |
1111 | ++sp->root_count; | 1115 | ++sp->root_count; |
1112 | vcpu->arch.mmu.root_hpa = root; | 1116 | vcpu->arch.mmu.root_hpa = root; |
1113 | return; | 1117 | return; |
1114 | } | 1118 | } |
1115 | #endif | 1119 | #endif |
1120 | metaphysical = !is_paging(vcpu); | ||
1121 | if (tdp_enabled) | ||
1122 | metaphysical = 1; | ||
1116 | for (i = 0; i < 4; ++i) { | 1123 | for (i = 0; i < 4; ++i) { |
1117 | hpa_t root = vcpu->arch.mmu.pae_root[i]; | 1124 | hpa_t root = vcpu->arch.mmu.pae_root[i]; |
1118 | 1125 | ||
@@ -1126,7 +1133,7 @@ static void mmu_alloc_roots(struct kvm_vcpu *vcpu) | |||
1126 | } else if (vcpu->arch.mmu.root_level == 0) | 1133 | } else if (vcpu->arch.mmu.root_level == 0) |
1127 | root_gfn = 0; | 1134 | root_gfn = 0; |
1128 | sp = kvm_mmu_get_page(vcpu, root_gfn, i << 30, | 1135 | sp = kvm_mmu_get_page(vcpu, root_gfn, i << 30, |
1129 | PT32_ROOT_LEVEL, !is_paging(vcpu), | 1136 | PT32_ROOT_LEVEL, metaphysical, |
1130 | ACC_ALL, NULL); | 1137 | ACC_ALL, NULL); |
1131 | root = __pa(sp->spt); | 1138 | root = __pa(sp->spt); |
1132 | ++sp->root_count; | 1139 | ++sp->root_count; |
@@ -1160,6 +1167,36 @@ static int nonpaging_page_fault(struct kvm_vcpu *vcpu, gva_t gva, | |||
1160 | error_code & PFERR_WRITE_MASK, gfn); | 1167 | error_code & PFERR_WRITE_MASK, gfn); |
1161 | } | 1168 | } |
1162 | 1169 | ||
1170 | static int tdp_page_fault(struct kvm_vcpu *vcpu, gva_t gpa, | ||
1171 | u32 error_code) | ||
1172 | { | ||
1173 | struct page *page; | ||
1174 | int r; | ||
1175 | |||
1176 | ASSERT(vcpu); | ||
1177 | ASSERT(VALID_PAGE(vcpu->arch.mmu.root_hpa)); | ||
1178 | |||
1179 | r = mmu_topup_memory_caches(vcpu); | ||
1180 | if (r) | ||
1181 | return r; | ||
1182 | |||
1183 | down_read(¤t->mm->mmap_sem); | ||
1184 | page = gfn_to_page(vcpu->kvm, gpa >> PAGE_SHIFT); | ||
1185 | if (is_error_page(page)) { | ||
1186 | kvm_release_page_clean(page); | ||
1187 | up_read(¤t->mm->mmap_sem); | ||
1188 | return 1; | ||
1189 | } | ||
1190 | spin_lock(&vcpu->kvm->mmu_lock); | ||
1191 | kvm_mmu_free_some_pages(vcpu); | ||
1192 | r = __direct_map(vcpu, gpa, error_code & PFERR_WRITE_MASK, | ||
1193 | gpa >> PAGE_SHIFT, page, TDP_ROOT_LEVEL); | ||
1194 | spin_unlock(&vcpu->kvm->mmu_lock); | ||
1195 | up_read(¤t->mm->mmap_sem); | ||
1196 | |||
1197 | return r; | ||
1198 | } | ||
1199 | |||
1163 | static void nonpaging_free(struct kvm_vcpu *vcpu) | 1200 | static void nonpaging_free(struct kvm_vcpu *vcpu) |
1164 | { | 1201 | { |
1165 | mmu_free_roots(vcpu); | 1202 | mmu_free_roots(vcpu); |
@@ -1253,7 +1290,35 @@ static int paging32E_init_context(struct kvm_vcpu *vcpu) | |||
1253 | return paging64_init_context_common(vcpu, PT32E_ROOT_LEVEL); | 1290 | return paging64_init_context_common(vcpu, PT32E_ROOT_LEVEL); |
1254 | } | 1291 | } |
1255 | 1292 | ||
1256 | static int init_kvm_mmu(struct kvm_vcpu *vcpu) | 1293 | static int init_kvm_tdp_mmu(struct kvm_vcpu *vcpu) |
1294 | { | ||
1295 | struct kvm_mmu *context = &vcpu->arch.mmu; | ||
1296 | |||
1297 | context->new_cr3 = nonpaging_new_cr3; | ||
1298 | context->page_fault = tdp_page_fault; | ||
1299 | context->free = nonpaging_free; | ||
1300 | context->prefetch_page = nonpaging_prefetch_page; | ||
1301 | context->shadow_root_level = TDP_ROOT_LEVEL; | ||
1302 | context->root_hpa = INVALID_PAGE; | ||
1303 | |||
1304 | if (!is_paging(vcpu)) { | ||
1305 | context->gva_to_gpa = nonpaging_gva_to_gpa; | ||
1306 | context->root_level = 0; | ||
1307 | } else if (is_long_mode(vcpu)) { | ||
1308 | context->gva_to_gpa = paging64_gva_to_gpa; | ||
1309 | context->root_level = PT64_ROOT_LEVEL; | ||
1310 | } else if (is_pae(vcpu)) { | ||
1311 | context->gva_to_gpa = paging64_gva_to_gpa; | ||
1312 | context->root_level = PT32E_ROOT_LEVEL; | ||
1313 | } else { | ||
1314 | context->gva_to_gpa = paging32_gva_to_gpa; | ||
1315 | context->root_level = PT32_ROOT_LEVEL; | ||
1316 | } | ||
1317 | |||
1318 | return 0; | ||
1319 | } | ||
1320 | |||
1321 | static int init_kvm_softmmu(struct kvm_vcpu *vcpu) | ||
1257 | { | 1322 | { |
1258 | ASSERT(vcpu); | 1323 | ASSERT(vcpu); |
1259 | ASSERT(!VALID_PAGE(vcpu->arch.mmu.root_hpa)); | 1324 | ASSERT(!VALID_PAGE(vcpu->arch.mmu.root_hpa)); |
@@ -1268,6 +1333,14 @@ static int init_kvm_mmu(struct kvm_vcpu *vcpu) | |||
1268 | return paging32_init_context(vcpu); | 1333 | return paging32_init_context(vcpu); |
1269 | } | 1334 | } |
1270 | 1335 | ||
1336 | static int init_kvm_mmu(struct kvm_vcpu *vcpu) | ||
1337 | { | ||
1338 | if (tdp_enabled) | ||
1339 | return init_kvm_tdp_mmu(vcpu); | ||
1340 | else | ||
1341 | return init_kvm_softmmu(vcpu); | ||
1342 | } | ||
1343 | |||
1271 | static void destroy_kvm_mmu(struct kvm_vcpu *vcpu) | 1344 | static void destroy_kvm_mmu(struct kvm_vcpu *vcpu) |
1272 | { | 1345 | { |
1273 | ASSERT(vcpu); | 1346 | ASSERT(vcpu); |
diff --git a/arch/x86/kvm/mmu.h b/arch/x86/kvm/mmu.h index 1fce19ec7a23..e64e9f56a65e 100644 --- a/arch/x86/kvm/mmu.h +++ b/arch/x86/kvm/mmu.h | |||
@@ -3,6 +3,12 @@ | |||
3 | 3 | ||
4 | #include <linux/kvm_host.h> | 4 | #include <linux/kvm_host.h> |
5 | 5 | ||
6 | #ifdef CONFIG_X86_64 | ||
7 | #define TDP_ROOT_LEVEL PT64_ROOT_LEVEL | ||
8 | #else | ||
9 | #define TDP_ROOT_LEVEL PT32E_ROOT_LEVEL | ||
10 | #endif | ||
11 | |||
6 | static inline void kvm_mmu_free_some_pages(struct kvm_vcpu *vcpu) | 12 | static inline void kvm_mmu_free_some_pages(struct kvm_vcpu *vcpu) |
7 | { | 13 | { |
8 | if (unlikely(vcpu->kvm->arch.n_free_mmu_pages < KVM_MIN_FREE_MMU_PAGES)) | 14 | if (unlikely(vcpu->kvm->arch.n_free_mmu_pages < KVM_MIN_FREE_MMU_PAGES)) |