diff options
author | Alexander Graf <agraf@suse.de> | 2010-07-29 08:47:42 -0400 |
---|---|---|
committer | Avi Kivity <avi@redhat.com> | 2010-10-24 04:50:42 -0400 |
commit | 96bc451a153297bf1f99ef2d633d512ea349ae7a (patch) | |
tree | 75ab4d40387ea1295a23b180e4c8719ce7053d00 /arch/powerpc/kvm | |
parent | 34698d8c61bd3fc86b2e99c3d1ad9ef140b3eb0d (diff) |
KVM: PPC: Introduce shared page
For transparent variable sharing between the hypervisor and guest, I introduce
a shared page. This shared page will contain all the registers the guest can
read and write safely without exiting guest context.
This patch only implements the stubs required for the basic structure of the
shared page. The actual register moving follows.
Signed-off-by: Alexander Graf <agraf@suse.de>
Signed-off-by: Avi Kivity <avi@redhat.com>
Diffstat (limited to 'arch/powerpc/kvm')
-rw-r--r-- | arch/powerpc/kvm/44x.c | 7 | ||||
-rw-r--r-- | arch/powerpc/kvm/book3s.c | 9 | ||||
-rw-r--r-- | arch/powerpc/kvm/e500.c | 7 |
3 files changed, 22 insertions, 1 deletions
diff --git a/arch/powerpc/kvm/44x.c b/arch/powerpc/kvm/44x.c index 73c0a3f64ed1..e7b1f3fca5dc 100644 --- a/arch/powerpc/kvm/44x.c +++ b/arch/powerpc/kvm/44x.c | |||
@@ -123,8 +123,14 @@ struct kvm_vcpu *kvmppc_core_vcpu_create(struct kvm *kvm, unsigned int id) | |||
123 | if (err) | 123 | if (err) |
124 | goto free_vcpu; | 124 | goto free_vcpu; |
125 | 125 | ||
126 | vcpu->arch.shared = (void*)__get_free_page(GFP_KERNEL|__GFP_ZERO); | ||
127 | if (!vcpu->arch.shared) | ||
128 | goto uninit_vcpu; | ||
129 | |||
126 | return vcpu; | 130 | return vcpu; |
127 | 131 | ||
132 | uninit_vcpu: | ||
133 | kvm_vcpu_uninit(vcpu); | ||
128 | free_vcpu: | 134 | free_vcpu: |
129 | kmem_cache_free(kvm_vcpu_cache, vcpu_44x); | 135 | kmem_cache_free(kvm_vcpu_cache, vcpu_44x); |
130 | out: | 136 | out: |
@@ -135,6 +141,7 @@ void kvmppc_core_vcpu_free(struct kvm_vcpu *vcpu) | |||
135 | { | 141 | { |
136 | struct kvmppc_vcpu_44x *vcpu_44x = to_44x(vcpu); | 142 | struct kvmppc_vcpu_44x *vcpu_44x = to_44x(vcpu); |
137 | 143 | ||
144 | free_page((unsigned long)vcpu->arch.shared); | ||
138 | kvm_vcpu_uninit(vcpu); | 145 | kvm_vcpu_uninit(vcpu); |
139 | kmem_cache_free(kvm_vcpu_cache, vcpu_44x); | 146 | kmem_cache_free(kvm_vcpu_cache, vcpu_44x); |
140 | } | 147 | } |
diff --git a/arch/powerpc/kvm/book3s.c b/arch/powerpc/kvm/book3s.c index a3cef30d1d42..b3385dd6f28d 100644 --- a/arch/powerpc/kvm/book3s.c +++ b/arch/powerpc/kvm/book3s.c | |||
@@ -1242,6 +1242,10 @@ struct kvm_vcpu *kvmppc_core_vcpu_create(struct kvm *kvm, unsigned int id) | |||
1242 | if (err) | 1242 | if (err) |
1243 | goto free_shadow_vcpu; | 1243 | goto free_shadow_vcpu; |
1244 | 1244 | ||
1245 | vcpu->arch.shared = (void*)__get_free_page(GFP_KERNEL|__GFP_ZERO); | ||
1246 | if (!vcpu->arch.shared) | ||
1247 | goto uninit_vcpu; | ||
1248 | |||
1245 | vcpu->arch.host_retip = kvm_return_point; | 1249 | vcpu->arch.host_retip = kvm_return_point; |
1246 | vcpu->arch.host_msr = mfmsr(); | 1250 | vcpu->arch.host_msr = mfmsr(); |
1247 | #ifdef CONFIG_PPC_BOOK3S_64 | 1251 | #ifdef CONFIG_PPC_BOOK3S_64 |
@@ -1268,10 +1272,12 @@ struct kvm_vcpu *kvmppc_core_vcpu_create(struct kvm *kvm, unsigned int id) | |||
1268 | 1272 | ||
1269 | err = kvmppc_mmu_init(vcpu); | 1273 | err = kvmppc_mmu_init(vcpu); |
1270 | if (err < 0) | 1274 | if (err < 0) |
1271 | goto free_shadow_vcpu; | 1275 | goto uninit_vcpu; |
1272 | 1276 | ||
1273 | return vcpu; | 1277 | return vcpu; |
1274 | 1278 | ||
1279 | uninit_vcpu: | ||
1280 | kvm_vcpu_uninit(vcpu); | ||
1275 | free_shadow_vcpu: | 1281 | free_shadow_vcpu: |
1276 | kfree(vcpu_book3s->shadow_vcpu); | 1282 | kfree(vcpu_book3s->shadow_vcpu); |
1277 | free_vcpu: | 1283 | free_vcpu: |
@@ -1284,6 +1290,7 @@ void kvmppc_core_vcpu_free(struct kvm_vcpu *vcpu) | |||
1284 | { | 1290 | { |
1285 | struct kvmppc_vcpu_book3s *vcpu_book3s = to_book3s(vcpu); | 1291 | struct kvmppc_vcpu_book3s *vcpu_book3s = to_book3s(vcpu); |
1286 | 1292 | ||
1293 | free_page((unsigned long)vcpu->arch.shared); | ||
1287 | kvm_vcpu_uninit(vcpu); | 1294 | kvm_vcpu_uninit(vcpu); |
1288 | kfree(vcpu_book3s->shadow_vcpu); | 1295 | kfree(vcpu_book3s->shadow_vcpu); |
1289 | vfree(vcpu_book3s); | 1296 | vfree(vcpu_book3s); |
diff --git a/arch/powerpc/kvm/e500.c b/arch/powerpc/kvm/e500.c index e8a00b0c4449..71750f2dd5d3 100644 --- a/arch/powerpc/kvm/e500.c +++ b/arch/powerpc/kvm/e500.c | |||
@@ -117,8 +117,14 @@ struct kvm_vcpu *kvmppc_core_vcpu_create(struct kvm *kvm, unsigned int id) | |||
117 | if (err) | 117 | if (err) |
118 | goto uninit_vcpu; | 118 | goto uninit_vcpu; |
119 | 119 | ||
120 | vcpu->arch.shared = (void*)__get_free_page(GFP_KERNEL|__GFP_ZERO); | ||
121 | if (!vcpu->arch.shared) | ||
122 | goto uninit_tlb; | ||
123 | |||
120 | return vcpu; | 124 | return vcpu; |
121 | 125 | ||
126 | uninit_tlb: | ||
127 | kvmppc_e500_tlb_uninit(vcpu_e500); | ||
122 | uninit_vcpu: | 128 | uninit_vcpu: |
123 | kvm_vcpu_uninit(vcpu); | 129 | kvm_vcpu_uninit(vcpu); |
124 | free_vcpu: | 130 | free_vcpu: |
@@ -131,6 +137,7 @@ void kvmppc_core_vcpu_free(struct kvm_vcpu *vcpu) | |||
131 | { | 137 | { |
132 | struct kvmppc_vcpu_e500 *vcpu_e500 = to_e500(vcpu); | 138 | struct kvmppc_vcpu_e500 *vcpu_e500 = to_e500(vcpu); |
133 | 139 | ||
140 | free_page((unsigned long)vcpu->arch.shared); | ||
134 | kvmppc_e500_tlb_uninit(vcpu_e500); | 141 | kvmppc_e500_tlb_uninit(vcpu_e500); |
135 | kvm_vcpu_uninit(vcpu); | 142 | kvm_vcpu_uninit(vcpu); |
136 | kmem_cache_free(kvm_vcpu_cache, vcpu_e500); | 143 | kmem_cache_free(kvm_vcpu_cache, vcpu_e500); |