aboutsummaryrefslogtreecommitdiffstats
path: root/arch/powerpc/kvm
diff options
context:
space:
mode:
authorAlexander Graf <agraf@suse.de>2010-07-29 08:47:48 -0400
committerAvi Kivity <avi@redhat.com>2010-10-24 04:50:45 -0400
commit2a342ed57756ad5d8af5456959433884367e5ab2 (patch)
treebb3f1d707916bc53f48919ace0d0f757c7e2083b /arch/powerpc/kvm
parenta73a9599e03eef1324d5aeecaebc1b339d2e1664 (diff)
KVM: PPC: Implement hypervisor interface
To communicate with KVM directly we need to plumb some sort of interface between the guest and KVM. Usually those interfaces use hypercalls. This hypercall implementation is described in the last patch of the series in a special documentation file. Please read that for further information. This patch implements stubs to handle KVM PPC hypercalls on the host and guest side alike. 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/book3s.c9
-rw-r--r--arch/powerpc/kvm/booke.c10
-rw-r--r--arch/powerpc/kvm/powerpc.c32
3 files changed, 48 insertions, 3 deletions
diff --git a/arch/powerpc/kvm/book3s.c b/arch/powerpc/kvm/book3s.c
index cfd7fe5c3a62..5cb5f0d9381f 100644
--- a/arch/powerpc/kvm/book3s.c
+++ b/arch/powerpc/kvm/book3s.c
@@ -947,10 +947,10 @@ program_interrupt:
947 break; 947 break;
948 } 948 }
949 case BOOK3S_INTERRUPT_SYSCALL: 949 case BOOK3S_INTERRUPT_SYSCALL:
950 // XXX make user settable
951 if (vcpu->arch.osi_enabled && 950 if (vcpu->arch.osi_enabled &&
952 (((u32)kvmppc_get_gpr(vcpu, 3)) == OSI_SC_MAGIC_R3) && 951 (((u32)kvmppc_get_gpr(vcpu, 3)) == OSI_SC_MAGIC_R3) &&
953 (((u32)kvmppc_get_gpr(vcpu, 4)) == OSI_SC_MAGIC_R4)) { 952 (((u32)kvmppc_get_gpr(vcpu, 4)) == OSI_SC_MAGIC_R4)) {
953 /* MOL hypercalls */
954 u64 *gprs = run->osi.gprs; 954 u64 *gprs = run->osi.gprs;
955 int i; 955 int i;
956 956
@@ -959,8 +959,13 @@ program_interrupt:
959 gprs[i] = kvmppc_get_gpr(vcpu, i); 959 gprs[i] = kvmppc_get_gpr(vcpu, i);
960 vcpu->arch.osi_needed = 1; 960 vcpu->arch.osi_needed = 1;
961 r = RESUME_HOST_NV; 961 r = RESUME_HOST_NV;
962 962 } else if (!(vcpu->arch.shared->msr & MSR_PR) &&
963 (((u32)kvmppc_get_gpr(vcpu, 0)) == KVM_SC_MAGIC_R0)) {
964 /* KVM PV hypercalls */
965 kvmppc_set_gpr(vcpu, 3, kvmppc_kvm_pv(vcpu));
966 r = RESUME_GUEST;
963 } else { 967 } else {
968 /* Guest syscalls */
964 vcpu->stat.syscall_exits++; 969 vcpu->stat.syscall_exits++;
965 kvmppc_book3s_queue_irqprio(vcpu, exit_nr); 970 kvmppc_book3s_queue_irqprio(vcpu, exit_nr);
966 r = RESUME_GUEST; 971 r = RESUME_GUEST;
diff --git a/arch/powerpc/kvm/booke.c b/arch/powerpc/kvm/booke.c
index b2c8c423c4d5..13e0747178e3 100644
--- a/arch/powerpc/kvm/booke.c
+++ b/arch/powerpc/kvm/booke.c
@@ -338,7 +338,15 @@ int kvmppc_handle_exit(struct kvm_run *run, struct kvm_vcpu *vcpu,
338 break; 338 break;
339 339
340 case BOOKE_INTERRUPT_SYSCALL: 340 case BOOKE_INTERRUPT_SYSCALL:
341 kvmppc_booke_queue_irqprio(vcpu, BOOKE_IRQPRIO_SYSCALL); 341 if (!(vcpu->arch.shared->msr & MSR_PR) &&
342 (((u32)kvmppc_get_gpr(vcpu, 0)) == KVM_SC_MAGIC_R0)) {
343 /* KVM PV hypercalls */
344 kvmppc_set_gpr(vcpu, 3, kvmppc_kvm_pv(vcpu));
345 r = RESUME_GUEST;
346 } else {
347 /* Guest syscalls */
348 kvmppc_booke_queue_irqprio(vcpu, BOOKE_IRQPRIO_SYSCALL);
349 }
342 kvmppc_account_exit(vcpu, SYSCALL_EXITS); 350 kvmppc_account_exit(vcpu, SYSCALL_EXITS);
343 r = RESUME_GUEST; 351 r = RESUME_GUEST;
344 break; 352 break;
diff --git a/arch/powerpc/kvm/powerpc.c b/arch/powerpc/kvm/powerpc.c
index 22f6fa2982f2..a4cf4b47e232 100644
--- a/arch/powerpc/kvm/powerpc.c
+++ b/arch/powerpc/kvm/powerpc.c
@@ -42,6 +42,38 @@ int kvm_arch_vcpu_runnable(struct kvm_vcpu *v)
42 !!(v->arch.pending_exceptions); 42 !!(v->arch.pending_exceptions);
43} 43}
44 44
45int kvmppc_kvm_pv(struct kvm_vcpu *vcpu)
46{
47 int nr = kvmppc_get_gpr(vcpu, 11);
48 int r;
49 unsigned long __maybe_unused param1 = kvmppc_get_gpr(vcpu, 3);
50 unsigned long __maybe_unused param2 = kvmppc_get_gpr(vcpu, 4);
51 unsigned long __maybe_unused param3 = kvmppc_get_gpr(vcpu, 5);
52 unsigned long __maybe_unused param4 = kvmppc_get_gpr(vcpu, 6);
53 unsigned long r2 = 0;
54
55 if (!(vcpu->arch.shared->msr & MSR_SF)) {
56 /* 32 bit mode */
57 param1 &= 0xffffffff;
58 param2 &= 0xffffffff;
59 param3 &= 0xffffffff;
60 param4 &= 0xffffffff;
61 }
62
63 switch (nr) {
64 case HC_VENDOR_KVM | KVM_HC_FEATURES:
65 r = HC_EV_SUCCESS;
66
67 /* Second return value is in r4 */
68 kvmppc_set_gpr(vcpu, 4, r2);
69 break;
70 default:
71 r = HC_EV_UNIMPLEMENTED;
72 break;
73 }
74
75 return r;
76}
45 77
46int kvmppc_emulate_mmio(struct kvm_run *run, struct kvm_vcpu *vcpu) 78int kvmppc_emulate_mmio(struct kvm_run *run, struct kvm_vcpu *vcpu)
47{ 79{