KVM: s390: Decoding helper functions.
authorCornelia Huck <cornelia.huck@de.ibm.com>
Thu, 20 Dec 2012 14:32:07 +0000 (15:32 +0100)
committerMarcelo Tosatti <mtosatti@redhat.com>
Mon, 7 Jan 2013 21:53:38 +0000 (19:53 -0200)
Introduce helper functions for decoding the various base/displacement
instruction formats.

Reviewed-by: Alexander Graf <agraf@suse.de>
Signed-off-by: Cornelia Huck <cornelia.huck@de.ibm.com>
Signed-off-by: Marcelo Tosatti <mtosatti@redhat.com>
arch/s390/kvm/intercept.c
arch/s390/kvm/kvm-s390.h
arch/s390/kvm/priv.c
arch/s390/kvm/sigp.c

index 0c08b3f1c1c569c6bfecb471a92f40cf93ac7611..df6c0ad085aa6ad0bbb1ba0e536f7ffd92e8a5ba 100644 (file)
@@ -26,9 +26,6 @@ static int handle_lctlg(struct kvm_vcpu *vcpu)
 {
        int reg1 = (vcpu->arch.sie_block->ipa & 0x00f0) >> 4;
        int reg3 = vcpu->arch.sie_block->ipa & 0x000f;
-       int base2 = vcpu->arch.sie_block->ipb >> 28;
-       int disp2 = ((vcpu->arch.sie_block->ipb & 0x0fff0000) >> 16) +
-                       ((vcpu->arch.sie_block->ipb & 0xff00) << 4);
        u64 useraddr;
        int reg, rc;
 
@@ -36,17 +33,15 @@ static int handle_lctlg(struct kvm_vcpu *vcpu)
        if ((vcpu->arch.sie_block->ipb & 0xff) != 0x2f)
                return -EOPNOTSUPP;
 
-       useraddr = disp2;
-       if (base2)
-               useraddr += vcpu->run->s.regs.gprs[base2];
+       useraddr = kvm_s390_get_base_disp_rsy(vcpu);
 
        if (useraddr & 7)
                return kvm_s390_inject_program_int(vcpu, PGM_SPECIFICATION);
 
        reg = reg1;
 
-       VCPU_EVENT(vcpu, 5, "lctlg r1:%x, r3:%x,b2:%x,d2:%x", reg1, reg3, base2,
-                  disp2);
+       VCPU_EVENT(vcpu, 5, "lctlg r1:%x, r3:%x, addr:%llx", reg1, reg3,
+                  useraddr);
        trace_kvm_s390_handle_lctl(vcpu, 1, reg1, reg3, useraddr);
 
        do {
@@ -68,23 +63,19 @@ static int handle_lctl(struct kvm_vcpu *vcpu)
 {
        int reg1 = (vcpu->arch.sie_block->ipa & 0x00f0) >> 4;
        int reg3 = vcpu->arch.sie_block->ipa & 0x000f;
-       int base2 = vcpu->arch.sie_block->ipb >> 28;
-       int disp2 = ((vcpu->arch.sie_block->ipb & 0x0fff0000) >> 16);
        u64 useraddr;
        u32 val = 0;
        int reg, rc;
 
        vcpu->stat.instruction_lctl++;
 
-       useraddr = disp2;
-       if (base2)
-               useraddr += vcpu->run->s.regs.gprs[base2];
+       useraddr = kvm_s390_get_base_disp_rs(vcpu);
 
        if (useraddr & 3)
                return kvm_s390_inject_program_int(vcpu, PGM_SPECIFICATION);
 
-       VCPU_EVENT(vcpu, 5, "lctl r1:%x, r3:%x,b2:%x,d2:%x", reg1, reg3, base2,
-                  disp2);
+       VCPU_EVENT(vcpu, 5, "lctl r1:%x, r3:%x, addr:%llx", reg1, reg3,
+                  useraddr);
        trace_kvm_s390_handle_lctl(vcpu, 0, reg1, reg3, useraddr);
 
        reg = reg1;
index d75bc5e92c5bf4c53a3a19bf2479007646834f49..dccc0242b7cac1976ec6af24bef34441fab4df79 100644 (file)
@@ -65,6 +65,43 @@ static inline void kvm_s390_set_prefix(struct kvm_vcpu *vcpu, u32 prefix)
        vcpu->arch.sie_block->ihcpu  = 0xffff;
 }
 
+static inline u64 kvm_s390_get_base_disp_s(struct kvm_vcpu *vcpu)
+{
+       int base2 = vcpu->arch.sie_block->ipb >> 28;
+       int disp2 = ((vcpu->arch.sie_block->ipb & 0x0fff0000) >> 16);
+
+       return (base2 ? vcpu->run->s.regs.gprs[base2] : 0) + disp2;
+}
+
+static inline void kvm_s390_get_base_disp_sse(struct kvm_vcpu *vcpu,
+                                             u64 *address1, u64 *address2)
+{
+       int base1 = (vcpu->arch.sie_block->ipb & 0xf0000000) >> 28;
+       int disp1 = (vcpu->arch.sie_block->ipb & 0x0fff0000) >> 16;
+       int base2 = (vcpu->arch.sie_block->ipb & 0xf000) >> 12;
+       int disp2 = vcpu->arch.sie_block->ipb & 0x0fff;
+
+       *address1 = (base1 ? vcpu->run->s.regs.gprs[base1] : 0) + disp1;
+       *address2 = (base2 ? vcpu->run->s.regs.gprs[base2] : 0) + disp2;
+}
+
+static inline u64 kvm_s390_get_base_disp_rsy(struct kvm_vcpu *vcpu)
+{
+       int base2 = vcpu->arch.sie_block->ipb >> 28;
+       int disp2 = ((vcpu->arch.sie_block->ipb & 0x0fff0000) >> 16) +
+                       ((vcpu->arch.sie_block->ipb & 0xff00) << 4);
+
+       return (base2 ? vcpu->run->s.regs.gprs[base2] : 0) + disp2;
+}
+
+static inline u64 kvm_s390_get_base_disp_rs(struct kvm_vcpu *vcpu)
+{
+       int base2 = vcpu->arch.sie_block->ipb >> 28;
+       int disp2 = ((vcpu->arch.sie_block->ipb & 0x0fff0000) >> 16);
+
+       return (base2 ? vcpu->run->s.regs.gprs[base2] : 0) + disp2;
+}
+
 int kvm_s390_handle_wait(struct kvm_vcpu *vcpu);
 enum hrtimer_restart kvm_s390_idle_wakeup(struct hrtimer *timer);
 void kvm_s390_tasklet(unsigned long parm);
index 1aeb9335f9e26e4ee1130d8d8bac3efb5cd00fe7..d715842f56ca122b9bab3c93895d608f8e478e94 100644 (file)
 
 static int handle_set_prefix(struct kvm_vcpu *vcpu)
 {
-       int base2 = vcpu->arch.sie_block->ipb >> 28;
-       int disp2 = ((vcpu->arch.sie_block->ipb & 0x0fff0000) >> 16);
        u64 operand2;
        u32 address = 0;
        u8 tmp;
 
        vcpu->stat.instruction_spx++;
 
-       operand2 = disp2;
-       if (base2)
-               operand2 += vcpu->run->s.regs.gprs[base2];
+       operand2 = kvm_s390_get_base_disp_s(vcpu);
 
        /* must be word boundary */
        if (operand2 & 3) {
@@ -67,15 +63,12 @@ out:
 
 static int handle_store_prefix(struct kvm_vcpu *vcpu)
 {
-       int base2 = vcpu->arch.sie_block->ipb >> 28;
-       int disp2 = ((vcpu->arch.sie_block->ipb & 0x0fff0000) >> 16);
        u64 operand2;
        u32 address;
 
        vcpu->stat.instruction_stpx++;
-       operand2 = disp2;
-       if (base2)
-               operand2 += vcpu->run->s.regs.gprs[base2];
+
+       operand2 = kvm_s390_get_base_disp_s(vcpu);
 
        /* must be word boundary */
        if (operand2 & 3) {
@@ -100,15 +93,12 @@ out:
 
 static int handle_store_cpu_address(struct kvm_vcpu *vcpu)
 {
-       int base2 = vcpu->arch.sie_block->ipb >> 28;
-       int disp2 = ((vcpu->arch.sie_block->ipb & 0x0fff0000) >> 16);
        u64 useraddr;
        int rc;
 
        vcpu->stat.instruction_stap++;
-       useraddr = disp2;
-       if (base2)
-               useraddr += vcpu->run->s.regs.gprs[base2];
+
+       useraddr = kvm_s390_get_base_disp_s(vcpu);
 
        if (useraddr & 1) {
                kvm_s390_inject_program_int(vcpu, PGM_SPECIFICATION);
@@ -178,15 +168,12 @@ static int handle_stfl(struct kvm_vcpu *vcpu)
 
 static int handle_stidp(struct kvm_vcpu *vcpu)
 {
-       int base2 = vcpu->arch.sie_block->ipb >> 28;
-       int disp2 = ((vcpu->arch.sie_block->ipb & 0x0fff0000) >> 16);
        u64 operand2;
        int rc;
 
        vcpu->stat.instruction_stidp++;
-       operand2 = disp2;
-       if (base2)
-               operand2 += vcpu->run->s.regs.gprs[base2];
+
+       operand2 = kvm_s390_get_base_disp_s(vcpu);
 
        if (operand2 & 7) {
                kvm_s390_inject_program_int(vcpu, PGM_SPECIFICATION);
@@ -240,17 +227,13 @@ static int handle_stsi(struct kvm_vcpu *vcpu)
        int fc = (vcpu->run->s.regs.gprs[0] & 0xf0000000) >> 28;
        int sel1 = vcpu->run->s.regs.gprs[0] & 0xff;
        int sel2 = vcpu->run->s.regs.gprs[1] & 0xffff;
-       int base2 = vcpu->arch.sie_block->ipb >> 28;
-       int disp2 = ((vcpu->arch.sie_block->ipb & 0x0fff0000) >> 16);
        u64 operand2;
        unsigned long mem;
 
        vcpu->stat.instruction_stsi++;
        VCPU_EVENT(vcpu, 4, "stsi: fc: %x sel1: %x sel2: %x", fc, sel1, sel2);
 
-       operand2 = disp2;
-       if (base2)
-               operand2 += vcpu->run->s.regs.gprs[base2];
+       operand2 = kvm_s390_get_base_disp_s(vcpu);
 
        if (operand2 & 0xfff && fc > 0)
                return kvm_s390_inject_program_int(vcpu, PGM_SPECIFICATION);
@@ -335,17 +318,14 @@ int kvm_s390_handle_b2(struct kvm_vcpu *vcpu)
 
 static int handle_tprot(struct kvm_vcpu *vcpu)
 {
-       int base1 = (vcpu->arch.sie_block->ipb & 0xf0000000) >> 28;
-       int disp1 = (vcpu->arch.sie_block->ipb & 0x0fff0000) >> 16;
-       int base2 = (vcpu->arch.sie_block->ipb & 0xf000) >> 12;
-       int disp2 = vcpu->arch.sie_block->ipb & 0x0fff;
-       u64 address1 = disp1 + base1 ? vcpu->run->s.regs.gprs[base1] : 0;
-       u64 address2 = disp2 + base2 ? vcpu->run->s.regs.gprs[base2] : 0;
+       u64 address1, address2;
        struct vm_area_struct *vma;
        unsigned long user_address;
 
        vcpu->stat.instruction_tprot++;
 
+       kvm_s390_get_base_disp_sse(vcpu, &address1, &address2);
+
        /* we only handle the Linux memory detection case:
         * access key == 0
         * guest DAT == off
index 566ddf6e8dfb54290afdb69d706cceadd7b78dbb..461e84179db1077fc32ba3ad0b266600f04e7f63 100644 (file)
@@ -324,8 +324,6 @@ int kvm_s390_handle_sigp(struct kvm_vcpu *vcpu)
 {
        int r1 = (vcpu->arch.sie_block->ipa & 0x00f0) >> 4;
        int r3 = vcpu->arch.sie_block->ipa & 0x000f;
-       int base2 = vcpu->arch.sie_block->ipb >> 28;
-       int disp2 = ((vcpu->arch.sie_block->ipb & 0x0fff0000) >> 16);
        u32 parameter;
        u16 cpu_addr = vcpu->run->s.regs.gprs[r3];
        u8 order_code;
@@ -336,9 +334,7 @@ int kvm_s390_handle_sigp(struct kvm_vcpu *vcpu)
                return kvm_s390_inject_program_int(vcpu,
                                                   PGM_PRIVILEGED_OPERATION);
 
-       order_code = disp2;
-       if (base2)
-               order_code += vcpu->run->s.regs.gprs[base2];
+       order_code = kvm_s390_get_base_disp_rs(vcpu);
 
        if (r1 % 2)
                parameter = vcpu->run->s.regs.gprs[r1];