summaryrefslogtreecommitdiffstats
path: root/sys/amd64/vmm
diff options
context:
space:
mode:
authorneel <neel@FreeBSD.org>2014-10-26 03:03:41 +0000
committerneel <neel@FreeBSD.org>2014-10-26 03:03:41 +0000
commita76d8ee4e9b6e0f73c4210907a7f9773ca25c154 (patch)
treecc31c393c958b9dcd0718e9156fbc3bf9df3a47e /sys/amd64/vmm
parent5b3100bae021caf26b7f3faa62d0f9a956bb2b52 (diff)
downloadFreeBSD-src-a76d8ee4e9b6e0f73c4210907a7f9773ca25c154.zip
FreeBSD-src-a76d8ee4e9b6e0f73c4210907a7f9773ca25c154.tar.gz
Don't pass the 'error' return from an I/O port handler directly to vm_run().
Most I/O port handlers return -1 to signal an error. If this value is returned without modification to vm_run() then it leads to incorrect behavior because '-1' is interpreted as ERESTART at the system call level. Fix this by always returning EIO to signal an error from an I/O port handler. MFC after: 1 week
Diffstat (limited to 'sys/amd64/vmm')
-rw-r--r--sys/amd64/vmm/vmm_ioport.c48
1 files changed, 27 insertions, 21 deletions
diff --git a/sys/amd64/vmm/vmm_ioport.c b/sys/amd64/vmm/vmm_ioport.c
index 4ce1403..564ca74 100644
--- a/sys/amd64/vmm/vmm_ioport.c
+++ b/sys/amd64/vmm/vmm_ioport.c
@@ -106,15 +106,14 @@ emulate_inout_port(struct vm *vm, int vcpuid, struct vm_exit *vmexit,
uint32_t mask, val;
int error;
- error = 0;
- *retu = true;
-
- if (vmexit->u.inout.port >= MAX_IOPORTS)
- goto done;
-
- handler = ioport_handler[vmexit->u.inout.port];
- if (handler == NULL)
- goto done;
+ /*
+ * If there is no handler for the I/O port then punt to userspace.
+ */
+ if (vmexit->u.inout.port >= MAX_IOPORTS ||
+ (handler = ioport_handler[vmexit->u.inout.port]) == NULL) {
+ *retu = true;
+ return (0);
+ }
mask = vie_size2mask(vmexit->u.inout.bytes);
@@ -124,20 +123,27 @@ emulate_inout_port(struct vm *vm, int vcpuid, struct vm_exit *vmexit,
error = (*handler)(vm, vcpuid, vmexit->u.inout.in,
vmexit->u.inout.port, vmexit->u.inout.bytes, &val);
+ if (error) {
+ /*
+ * The value returned by this function is also the return value
+ * of vm_run(). This needs to be a positive number otherwise it
+ * can be interpreted as a "pseudo-error" like ERESTART.
+ *
+ * Enforce this by mapping all errors to EIO.
+ */
+ return (EIO);
+ }
- if (!error) {
- *retu = false;
- if (vmexit->u.inout.in) {
- vmexit->u.inout.eax &= ~mask;
- vmexit->u.inout.eax |= val & mask;
- error = vm_set_register(vm, vcpuid,
- VM_REG_GUEST_RAX, vmexit->u.inout.eax);
- KASSERT(error == 0, ("emulate_ioport: error %d "
- "setting guest rax register", error));
- }
+ if (vmexit->u.inout.in) {
+ vmexit->u.inout.eax &= ~mask;
+ vmexit->u.inout.eax |= val & mask;
+ error = vm_set_register(vm, vcpuid, VM_REG_GUEST_RAX,
+ vmexit->u.inout.eax);
+ KASSERT(error == 0, ("emulate_ioport: error %d setting guest "
+ "rax register", error));
}
-done:
- return (error);
+ *retu = false;
+ return (0);
}
static int
OpenPOWER on IntegriCloud