KVM: x86 emulator: Don't overwrite decode cache
Currently if we an instruction spans a page boundary, when we fetch the second half we overwrite the first half. This prevents us from tracing the full instruction opcodes. Fix by appending the second half to the first. Signed-off-by: Avi Kivity <avi@redhat.com> Signed-off-by: Marcelo Tosatti <mtosatti@redhat.com>
This commit is contained in:
parent
4496f97482
commit
2fb53ad811
|
@ -646,21 +646,22 @@ static unsigned long ss_base(struct x86_emulate_ctxt *ctxt)
|
||||||
|
|
||||||
static int do_fetch_insn_byte(struct x86_emulate_ctxt *ctxt,
|
static int do_fetch_insn_byte(struct x86_emulate_ctxt *ctxt,
|
||||||
struct x86_emulate_ops *ops,
|
struct x86_emulate_ops *ops,
|
||||||
unsigned long linear, u8 *dest)
|
unsigned long eip, u8 *dest)
|
||||||
{
|
{
|
||||||
struct fetch_cache *fc = &ctxt->decode.fetch;
|
struct fetch_cache *fc = &ctxt->decode.fetch;
|
||||||
int rc;
|
int rc;
|
||||||
int size;
|
int size, cur_size;
|
||||||
|
|
||||||
if (linear < fc->start || linear >= fc->end) {
|
if (eip == fc->end) {
|
||||||
size = min(15UL, PAGE_SIZE - offset_in_page(linear));
|
cur_size = fc->end - fc->start;
|
||||||
rc = ops->fetch(linear, fc->data, size, ctxt->vcpu, NULL);
|
size = min(15UL - cur_size, PAGE_SIZE - offset_in_page(eip));
|
||||||
|
rc = ops->fetch(ctxt->cs_base + eip, fc->data + cur_size,
|
||||||
|
size, ctxt->vcpu, NULL);
|
||||||
if (rc != X86EMUL_CONTINUE)
|
if (rc != X86EMUL_CONTINUE)
|
||||||
return rc;
|
return rc;
|
||||||
fc->start = linear;
|
fc->end += size;
|
||||||
fc->end = linear + size;
|
|
||||||
}
|
}
|
||||||
*dest = fc->data[linear - fc->start];
|
*dest = fc->data[eip - fc->start];
|
||||||
return X86EMUL_CONTINUE;
|
return X86EMUL_CONTINUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -673,7 +674,6 @@ static int do_insn_fetch(struct x86_emulate_ctxt *ctxt,
|
||||||
/* x86 instructions are limited to 15 bytes. */
|
/* x86 instructions are limited to 15 bytes. */
|
||||||
if (eip + size - ctxt->eip > 15)
|
if (eip + size - ctxt->eip > 15)
|
||||||
return X86EMUL_UNHANDLEABLE;
|
return X86EMUL_UNHANDLEABLE;
|
||||||
eip += ctxt->cs_base;
|
|
||||||
while (size--) {
|
while (size--) {
|
||||||
rc = do_fetch_insn_byte(ctxt, ops, eip++, dest++);
|
rc = do_fetch_insn_byte(ctxt, ops, eip++, dest++);
|
||||||
if (rc != X86EMUL_CONTINUE)
|
if (rc != X86EMUL_CONTINUE)
|
||||||
|
@ -935,6 +935,7 @@ x86_decode_insn(struct x86_emulate_ctxt *ctxt, struct x86_emulate_ops *ops)
|
||||||
/* Shadow copy of register state. Committed on successful emulation. */
|
/* Shadow copy of register state. Committed on successful emulation. */
|
||||||
memset(c, 0, sizeof(struct decode_cache));
|
memset(c, 0, sizeof(struct decode_cache));
|
||||||
c->eip = ctxt->eip;
|
c->eip = ctxt->eip;
|
||||||
|
c->fetch.start = c->fetch.end = c->eip;
|
||||||
ctxt->cs_base = seg_base(ctxt, VCPU_SREG_CS);
|
ctxt->cs_base = seg_base(ctxt, VCPU_SREG_CS);
|
||||||
memcpy(c->regs, ctxt->vcpu->arch.regs, sizeof c->regs);
|
memcpy(c->regs, ctxt->vcpu->arch.regs, sizeof c->regs);
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue