Gitweb:     
http://git.kernel.org/git/?p=linux/kernel/git/torvalds/linux-2.6.git;a=commit;h=8b4caf6650808024c37ec4b29cf81b308af998b1
Commit:     8b4caf6650808024c37ec4b29cf81b308af998b1
Parent:     e4e03deda83b1f2fc37ccbfc1eef27e86e8ed4e9
Author:     Laurent Vivier <[EMAIL PROTECTED]>
AuthorDate: Tue Sep 18 11:27:19 2007 +0200
Committer:  Avi Kivity <[EMAIL PROTECTED]>
CommitDate: Wed Jan 30 17:52:47 2008 +0200

    KVM: x86 emulator: move all decoding process to function x86_decode_insn()
    
    Split the decoding process into a new function x86_decode_insn().
    
    Signed-off-by: Laurent Vivier <[EMAIL PROTECTED]>
    Signed-off-by: Avi Kivity <[EMAIL PROTECTED]>
---
 drivers/kvm/x86_emulate.c |   77 +++++++++++++++++++++++++++++++--------------
 1 files changed, 53 insertions(+), 24 deletions(-)

diff --git a/drivers/kvm/x86_emulate.c b/drivers/kvm/x86_emulate.c
index d7026cb..540589c 100644
--- a/drivers/kvm/x86_emulate.c
+++ b/drivers/kvm/x86_emulate.c
@@ -517,20 +517,16 @@ static int test_cc(unsigned int condition, unsigned int 
flags)
 }
 
 int
-x86_emulate_memop(struct x86_emulate_ctxt *ctxt, struct x86_emulate_ops *ops)
+x86_decode_insn(struct x86_emulate_ctxt *ctxt, struct x86_emulate_ops *ops)
 {
        struct decode_cache *c = &ctxt->decode;
        u8 sib, rex_prefix = 0;
        unsigned int i;
        int rc = 0;
-       unsigned long cr2 = ctxt->cr2;
        int mode = ctxt->mode;
        int index_reg = 0, base_reg = 0, scale, rip_relative = 0;
-       int no_wb = 0;
-       u64 msr_data;
 
        /* Shadow copy of register state. Committed on successful emulation. */
-       unsigned long _eflags = ctxt->eflags;
 
        memset(c, 0, sizeof(struct decode_cache));
        c->eip = ctxt->vcpu->rip;
@@ -622,8 +618,10 @@ done_prefixes:
                }
 
                /* Unrecognised? */
-               if (c->d == 0)
-                       goto cannot_emulate;
+               if (c->d == 0) {
+                       DPRINTF("Cannot emulate %02x\n", c->b);
+                       return -1;
+               }
        }
 
        /* ModRM and SIB bytes. */
@@ -776,7 +774,6 @@ done_prefixes:
                }
                if (c->ad_bytes != 8)
                        c->modrm_ea = (u32)c->modrm_ea;
-               cr2 = c->modrm_ea;
        modrm_done:
                ;
        }
@@ -838,13 +835,6 @@ done_prefixes:
                        break;
                }
                c->src.type = OP_MEM;
-               c->src.ptr = (unsigned long *)cr2;
-               c->src.val = 0;
-               if ((rc = ops->read_emulated((unsigned long)c->src.ptr,
-                                          &c->src.val,
-                                          c->src.bytes, ctxt->vcpu)) != 0)
-                       goto done;
-               c->src.orig_val = c->src.val;
                break;
        case SrcImm:
                c->src.type = OP_IMM;
@@ -877,7 +867,7 @@ done_prefixes:
        switch (c->d & DstMask) {
        case ImplicitOps:
                /* Special instructions do their own operand decoding. */
-               goto special_insn;
+               return 0;
        case DstReg:
                c->dst.type = OP_REG;
                if ((c->d & ByteOp)
@@ -905,14 +895,54 @@ done_prefixes:
                }
                break;
        case DstMem:
-               c->dst.type = OP_MEM;
-               c->dst.ptr = (unsigned long *)cr2;
-               c->dst.bytes = (c->d & ByteOp) ? 1 : c->op_bytes;
-               c->dst.val = 0;
                if ((c->d & ModRM) && c->modrm_mod == 3) {
                        c->dst.type = OP_REG;
                        break;
                }
+               c->dst.type = OP_MEM;
+               break;
+       }
+
+done:
+       return (rc == X86EMUL_UNHANDLEABLE) ? -1 : 0;
+}
+
+int
+x86_emulate_memop(struct x86_emulate_ctxt *ctxt, struct x86_emulate_ops *ops)
+{
+       unsigned long cr2 = ctxt->cr2;
+       int no_wb = 0;
+       u64 msr_data;
+       unsigned long _eflags = ctxt->eflags;
+       struct decode_cache *c = &ctxt->decode;
+       int rc;
+
+       rc = x86_decode_insn(ctxt, ops);
+       if (rc)
+               return rc;
+
+       if ((c->d & ModRM) && (c->modrm_mod != 3))
+               cr2 = c->modrm_ea;
+
+       if (c->src.type == OP_MEM) {
+               c->src.ptr = (unsigned long *)cr2;
+               c->src.val = 0;
+               if ((rc = ops->read_emulated((unsigned long)c->src.ptr,
+                                            &c->src.val,
+                                            c->src.bytes,
+                                            ctxt->vcpu)) != 0)
+                       goto done;
+               c->src.orig_val = c->src.val;
+       }
+
+       if ((c->d & DstMask) == ImplicitOps)
+               goto special_insn;
+
+
+       if (c->dst.type == OP_MEM) {
+               c->dst.ptr = (unsigned long *)cr2;
+               c->dst.bytes = (c->d & ByteOp) ? 1 : c->op_bytes;
+               c->dst.val = 0;
                if (c->d & BitOp) {
                        unsigned long mask = ~(c->dst.bytes * 8 - 1);
 
@@ -925,7 +955,6 @@ done_prefixes:
                                           &c->dst.val,
                                          c->dst.bytes, ctxt->vcpu)) != 0))
                        goto done;
-               break;
        }
        c->dst.orig_val = c->dst.val;
 
@@ -983,7 +1012,7 @@ done_prefixes:
                emulate_2op_SrcV("cmp", c->src, c->dst, _eflags);
                break;
        case 0x63:              /* movsxd */
-               if (mode != X86EMUL_MODE_PROT64)
+               if (ctxt->mode != X86EMUL_MODE_PROT64)
                        goto cannot_emulate;
                c->dst.val = (s32) c->src.val;
                break;
@@ -1041,7 +1070,7 @@ done_prefixes:
                break;
        case 0x8f:              /* pop (sole member of Grp1a) */
                /* 64-bit mode: POP always pops a 64-bit operand. */
-               if (mode == X86EMUL_MODE_PROT64)
+               if (ctxt->mode == X86EMUL_MODE_PROT64)
                        c->dst.bytes = 8;
                if ((rc = ops->read_std(register_address(
                                                   ctxt->ss_base,
@@ -1152,7 +1181,7 @@ done_prefixes:
                        break;
                case 6: /* push */
                        /* 64-bit mode: PUSH always pushes a 64-bit operand. */
-                       if (mode == X86EMUL_MODE_PROT64) {
+                       if (ctxt->mode == X86EMUL_MODE_PROT64) {
                                c->dst.bytes = 8;
                                if ((rc = ops->read_std(
                                                 (unsigned long)c->dst.ptr,
-
To unsubscribe from this list: send the line "unsubscribe git-commits-head" in
the body of a message to [EMAIL PROTECTED]
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Reply via email to