Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/lib/libnvmm Handle more corner cases, clean up a little, and...
details: https://anonhg.NetBSD.org/src/rev/35a4591e24a3
branches: trunk
changeset: 995983:35a4591e24a3
user: maxv <maxv%NetBSD.org@localhost>
date: Sun Jan 13 10:43:22 2019 +0000
description:
Handle more corner cases, clean up a little, and add a set of instructions
in Group1.
diffstat:
lib/libnvmm/libnvmm_x86.c | 216 ++++++++++++++++++++++++---------------------
1 files changed, 116 insertions(+), 100 deletions(-)
diffs (truncated from 455 to 300 lines):
diff -r 7f2cdc857cff -r 35a4591e24a3 lib/libnvmm/libnvmm_x86.c
--- a/lib/libnvmm/libnvmm_x86.c Sun Jan 13 10:07:50 2019 +0000
+++ b/lib/libnvmm/libnvmm_x86.c Sun Jan 13 10:43:22 2019 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: libnvmm_x86.c,v 1.14 2019/01/08 07:34:22 maxv Exp $ */
+/* $NetBSD: libnvmm_x86.c,v 1.15 2019/01/13 10:43:22 maxv Exp $ */
/*
* Copyright (c) 2018 The NetBSD Foundation, Inc.
@@ -77,13 +77,15 @@
printf("| -> RAX=%p\n", (void *)state.gprs[NVMM_X64_GPR_RAX]);
printf("| -> RBX=%p\n", (void *)state.gprs[NVMM_X64_GPR_RBX]);
printf("| -> RCX=%p\n", (void *)state.gprs[NVMM_X64_GPR_RCX]);
+ printf("| -> RFLAGS=%p\n", (void *)state.gprs[NVMM_X64_GPR_RFLAGS]);
for (i = 0; i < NVMM_X64_NSEG; i++) {
- printf("| -> %s: sel=0x%lx base=%p, limit=%p, P=%d, D=%d\n",
+ printf("| -> %s: sel=0x%lx base=%p, limit=%p, P=%d, D=%d L=%d\n",
segnames[i],
state.segs[i].selector,
(void *)state.segs[i].base,
(void *)state.segs[i].limit,
- state.segs[i].attrib.p, state.segs[i].attrib.def32);
+ state.segs[i].attrib.p, state.segs[i].attrib.def32,
+ state.segs[i].attrib.lng);
}
printf("| -> MSR_EFER=%p\n", (void *)state.msrs[NVMM_X64_MSR_EFER]);
printf("| -> CR0=%p\n", (void *)state.crs[NVMM_X64_CR_CR0]);
@@ -392,7 +394,7 @@
gva &= ~PAGE_MASK;
is_pae = (state->crs[NVMM_X64_CR_CR4] & CR4_PAE) != 0;
- is_lng = (state->msrs[NVMM_X64_MSR_EFER] & EFER_LME) != 0;
+ is_lng = (state->msrs[NVMM_X64_MSR_EFER] & EFER_LMA) != 0;
has_pse = (state->crs[NVMM_X64_CR_CR4] & CR4_PSE) != 0;
cr3 = state->crs[NVMM_X64_CR_CR3];
@@ -437,6 +439,12 @@
/* -------------------------------------------------------------------------- */
static inline bool
+is_long_mode(struct nvmm_x64_state *state)
+{
+ return (state->msrs[NVMM_X64_MSR_EFER] & EFER_LMA) != 0;
+}
+
+static inline bool
is_64bit(struct nvmm_x64_state *state)
{
return (state->segs[NVMM_X64_SEG_CS].attrib.lng != 0);
@@ -456,14 +464,8 @@
(state->segs[NVMM_X64_SEG_CS].attrib.def32 == 0);
}
-static inline bool
-is_long_mode(struct nvmm_x64_state *state)
-{
- return (state->msrs[NVMM_X64_MSR_EFER] & EFER_LME) != 0;
-}
-
static int
-segment_apply(struct nvmm_x64_state_seg *seg, gvaddr_t *gva, size_t size)
+segment_check(struct nvmm_x64_state_seg *seg, gvaddr_t gva, size_t size)
{
uint64_t limit;
@@ -480,11 +482,10 @@
limit *= PAGE_SIZE;
}
- if (__predict_false(*gva + size > limit)) {
+ if (__predict_false(gva + size > limit)) {
goto error;
}
- *gva += seg->base;
return 0;
error:
@@ -492,17 +493,25 @@
return -1;
}
-static uint64_t
-mask_from_adsize(size_t adsize)
+static inline void
+segment_apply(struct nvmm_x64_state_seg *seg, gvaddr_t *gva)
{
- switch (adsize) {
- case 8:
- return 0xFFFFFFFFFFFFFFFF;
+ *gva += seg->base;
+}
+
+static inline uint64_t
+size_to_mask(size_t size)
+{
+ switch (size) {
+ case 1:
+ return 0x00000000000000FF;
+ case 2:
+ return 0x000000000000FFFF;
case 4:
return 0x00000000FFFFFFFF;
- case 2:
- default: /* impossible */
- return 0x000000000000FFFF;
+ case 8:
+ default:
+ return 0xFFFFFFFFFFFFFFFF;
}
}
@@ -511,7 +520,7 @@
{
uint64_t mask, cnt;
- mask = mask_from_adsize(adsize);
+ mask = size_to_mask(adsize);
cnt = state->gprs[NVMM_X64_GPR_RCX] & mask;
return cnt;
@@ -522,28 +531,12 @@
{
uint64_t mask;
- mask = mask_from_adsize(adsize);
+ /* XXX: should we zero-extend? */
+ mask = size_to_mask(adsize);
state->gprs[NVMM_X64_GPR_RCX] &= ~mask;
state->gprs[NVMM_X64_GPR_RCX] |= cnt;
}
-static uint64_t
-rep_dec_apply(struct nvmm_x64_state *state, size_t adsize)
-{
- uint64_t mask, cnt;
-
- mask = mask_from_adsize(adsize);
-
- cnt = state->gprs[NVMM_X64_GPR_RCX] & mask;
- cnt -= 1;
- cnt &= mask;
-
- state->gprs[NVMM_X64_GPR_RCX] &= ~mask;
- state->gprs[NVMM_X64_GPR_RCX] |= cnt;
-
- return cnt;
-}
-
static int
read_guest_memory(struct nvmm_machine *mach, struct nvmm_x64_state *state,
gvaddr_t gva, uint8_t *data, size_t size)
@@ -693,7 +686,7 @@
uint64_t cnt = 0; /* GCC */
uint8_t iobuf[8];
int iocnt = 1;
- gvaddr_t gva;
+ gvaddr_t gva = 0; /* GCC */
int reg = 0; /* GCC */
int ret, seg;
bool psld = false;
@@ -717,7 +710,8 @@
if (exit->u.io.rep) {
cnt = rep_get_cnt(&state, exit->u.io.address_size);
if (__predict_false(cnt == 0)) {
- return 0;
+ state.gprs[NVMM_X64_GPR_RIP] = exit->u.io.npc;
+ goto out;
}
}
@@ -736,24 +730,29 @@
}
gva = state.gprs[reg];
- gva &= mask_from_adsize(exit->u.io.address_size);
-
- if (!is_long_mode(&state)) {
- if (exit->u.io.seg != -1) {
- seg = exit->u.io.seg;
+ gva &= size_to_mask(exit->u.io.address_size);
+
+ if (exit->u.io.seg != -1) {
+ seg = exit->u.io.seg;
+ } else {
+ if (io.in) {
+ seg = NVMM_X64_SEG_ES;
} else {
- if (io.in) {
- seg = NVMM_X64_SEG_ES;
- } else {
- seg = fetch_segment(mach, &state);
- if (seg == -1)
- return -1;
- }
+ seg = fetch_segment(mach, &state);
+ if (seg == -1)
+ return -1;
}
-
- ret = segment_apply(&state.segs[seg], &gva, io.size);
+ }
+
+ if (__predict_true(is_long_mode(&state))) {
+ if (seg == NVMM_X64_SEG_GS || seg == NVMM_X64_SEG_FS) {
+ segment_apply(&state.segs[seg], &gva);
+ }
+ } else {
+ ret = segment_check(&state.segs[seg], gva, io.size);
if (ret == -1)
return -1;
+ segment_apply(&state.segs[seg], &gva);
}
if (exit->u.io.rep && !psld) {
@@ -780,6 +779,10 @@
if (io.in) {
if (!exit->u.io.str) {
memcpy(&state.gprs[NVMM_X64_GPR_RAX], io.data, io.size);
+ if (io.size == 4) {
+ /* Zero-extend to 64 bits. */
+ state.gprs[NVMM_X64_GPR_RAX] &= size_to_mask(4);
+ }
} else {
ret = write_guest_memory(mach, &state, gva, io.data,
io.size);
@@ -807,6 +810,7 @@
state.gprs[NVMM_X64_GPR_RIP] = exit->u.io.npc;
}
+out:
ret = nvmm_vcpu_setstate(mach, cpuid, &state, NVMM_X64_STATE_GPRS);
if (ret == -1)
return -1;
@@ -907,7 +911,6 @@
};
struct x86_immediate {
- size_t size; /* 1/2/4/8 */
uint64_t data;
};
@@ -1016,6 +1019,19 @@
* Group1
*/
{
+ /* Ev, Iz */
+ .byte = 0x81,
+ .regmodrm = true,
+ .regtorm = true,
+ .szoverride = true,
+ .defsize = -1,
+ .allsize = OPSIZE_WORD|OPSIZE_DOUB|OPSIZE_QUAD,
+ .group1 = true,
+ .immediate = true,
+ .flags = FLAG_immz,
+ .emul = NULL /* group1 */
+ },
+ {
/* Ev, Ib */
.byte = 0x83,
.regmodrm = true,
@@ -1769,7 +1785,7 @@
return 0;
}
-static uint64_t
+static inline uint64_t
sign_extend(uint64_t val, int size)
{
if (size == 1) {
@@ -1806,15 +1822,13 @@
}
store->type = STORE_IMM;
- store->u.imm.size = immsize;
if (fsm_read(fsm, (uint8_t *)&store->u.imm.data, immsize) == -1) {
return -1;
}
- fsm_advance(fsm, store->u.imm.size, NULL);
+ fsm_advance(fsm, immsize, NULL);
if (sesize != 0) {
store->u.imm.data = sign_extend(store->u.imm.data, sesize);
- store->u.imm.size = sesize;
}
return 0;
@@ -2204,6 +2218,11 @@
instr->operand_size = get_operand_size(fsm, instr);
instr->address_size = get_address_size(fsm, instr);
+ if (fsm->is64bit && (instr->operand_size == 4)) {
+ /* Zero-extend to 64 bits. */
+ instr->zeroextend_mask = ~size_to_mask(4);
+ }
+
if (opcode->regmodrm) {
fsm_advance(fsm, 1, node_regmodrm);
} else if (opcode->dmo) {
Home |
Main Index |
Thread Index |
Old Index