hw/xen: Expose handle_bufioreq in xen_register_ioreq
Expose handle_bufioreq in xen_register_ioreq(). This is to allow machines to enable or disable buffered ioreqs. No functional change since all callers still set it to HVM_IOREQSRV_BUFIOREQ_ATOMIC. Reviewed-by: Stefano Stabellini <sstabellini@kernel.org> Signed-off-by: Edgar E. Iglesias <edgar.iglesias@amd.com>
This commit is contained in:
parent
abdfd6549d
commit
b2150e403a
@ -614,7 +614,9 @@ void xen_hvm_init_pc(PCMachineState *pcms, MemoryRegion **ram_memory)
|
|||||||
|
|
||||||
state = g_new0(XenIOState, 1);
|
state = g_new0(XenIOState, 1);
|
||||||
|
|
||||||
xen_register_ioreq(state, max_cpus, &xen_memory_listener);
|
xen_register_ioreq(state, max_cpus,
|
||||||
|
HVM_IOREQSRV_BUFIOREQ_ATOMIC,
|
||||||
|
&xen_memory_listener);
|
||||||
|
|
||||||
xen_is_stubdomain = xen_check_stubdomain(state->xenstore);
|
xen_is_stubdomain = xen_check_stubdomain(state->xenstore);
|
||||||
|
|
||||||
|
@ -667,6 +667,8 @@ static int xen_map_ioreq_server(XenIOState *state)
|
|||||||
xen_pfn_t ioreq_pfn;
|
xen_pfn_t ioreq_pfn;
|
||||||
xen_pfn_t bufioreq_pfn;
|
xen_pfn_t bufioreq_pfn;
|
||||||
evtchn_port_t bufioreq_evtchn;
|
evtchn_port_t bufioreq_evtchn;
|
||||||
|
unsigned long num_frames = 1;
|
||||||
|
unsigned long frame = 1;
|
||||||
int rc;
|
int rc;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -675,59 +677,78 @@ static int xen_map_ioreq_server(XenIOState *state)
|
|||||||
*/
|
*/
|
||||||
QEMU_BUILD_BUG_ON(XENMEM_resource_ioreq_server_frame_bufioreq != 0);
|
QEMU_BUILD_BUG_ON(XENMEM_resource_ioreq_server_frame_bufioreq != 0);
|
||||||
QEMU_BUILD_BUG_ON(XENMEM_resource_ioreq_server_frame_ioreq(0) != 1);
|
QEMU_BUILD_BUG_ON(XENMEM_resource_ioreq_server_frame_ioreq(0) != 1);
|
||||||
|
|
||||||
|
if (state->has_bufioreq) {
|
||||||
|
frame = 0;
|
||||||
|
num_frames = 2;
|
||||||
|
}
|
||||||
state->fres = xenforeignmemory_map_resource(xen_fmem, xen_domid,
|
state->fres = xenforeignmemory_map_resource(xen_fmem, xen_domid,
|
||||||
XENMEM_resource_ioreq_server,
|
XENMEM_resource_ioreq_server,
|
||||||
state->ioservid, 0, 2,
|
state->ioservid,
|
||||||
|
frame, num_frames,
|
||||||
&addr,
|
&addr,
|
||||||
PROT_READ | PROT_WRITE, 0);
|
PROT_READ | PROT_WRITE, 0);
|
||||||
if (state->fres != NULL) {
|
if (state->fres != NULL) {
|
||||||
trace_xen_map_resource_ioreq(state->ioservid, addr);
|
trace_xen_map_resource_ioreq(state->ioservid, addr);
|
||||||
state->buffered_io_page = addr;
|
state->shared_page = addr;
|
||||||
state->shared_page = addr + XC_PAGE_SIZE;
|
if (state->has_bufioreq) {
|
||||||
|
state->buffered_io_page = addr;
|
||||||
|
state->shared_page = addr + XC_PAGE_SIZE;
|
||||||
|
}
|
||||||
} else if (errno != EOPNOTSUPP) {
|
} else if (errno != EOPNOTSUPP) {
|
||||||
error_report("failed to map ioreq server resources: error %d handle=%p",
|
error_report("failed to map ioreq server resources: error %d handle=%p",
|
||||||
errno, xen_xc);
|
errno, xen_xc);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
rc = xen_get_ioreq_server_info(xen_domid, state->ioservid,
|
/*
|
||||||
(state->shared_page == NULL) ?
|
* If we fail to map the shared page with xenforeignmemory_map_resource()
|
||||||
&ioreq_pfn : NULL,
|
* or if we're using buffered ioreqs, we need xen_get_ioreq_server_info()
|
||||||
(state->buffered_io_page == NULL) ?
|
* to provide the the addresses to map the shared page and/or to get the
|
||||||
&bufioreq_pfn : NULL,
|
* event-channel port for buffered ioreqs.
|
||||||
&bufioreq_evtchn);
|
*/
|
||||||
if (rc < 0) {
|
if (state->shared_page == NULL || state->has_bufioreq) {
|
||||||
error_report("failed to get ioreq server info: error %d handle=%p",
|
rc = xen_get_ioreq_server_info(xen_domid, state->ioservid,
|
||||||
errno, xen_xc);
|
(state->shared_page == NULL) ?
|
||||||
return rc;
|
&ioreq_pfn : NULL,
|
||||||
}
|
(state->has_bufioreq &&
|
||||||
|
state->buffered_io_page == NULL) ?
|
||||||
|
&bufioreq_pfn : NULL,
|
||||||
|
&bufioreq_evtchn);
|
||||||
|
if (rc < 0) {
|
||||||
|
error_report("failed to get ioreq server info: error %d handle=%p",
|
||||||
|
errno, xen_xc);
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
if (state->shared_page == NULL) {
|
if (state->shared_page == NULL) {
|
||||||
trace_xen_map_ioreq_server_shared_page(ioreq_pfn);
|
trace_xen_map_ioreq_server_shared_page(ioreq_pfn);
|
||||||
|
|
||||||
state->shared_page = xenforeignmemory_map(xen_fmem, xen_domid,
|
state->shared_page = xenforeignmemory_map(xen_fmem, xen_domid,
|
||||||
PROT_READ | PROT_WRITE,
|
PROT_READ | PROT_WRITE,
|
||||||
1, &ioreq_pfn, NULL);
|
1, &ioreq_pfn, NULL);
|
||||||
|
}
|
||||||
if (state->shared_page == NULL) {
|
if (state->shared_page == NULL) {
|
||||||
error_report("map shared IO page returned error %d handle=%p",
|
error_report("map shared IO page returned error %d handle=%p",
|
||||||
errno, xen_xc);
|
errno, xen_xc);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
if (state->buffered_io_page == NULL) {
|
if (state->has_bufioreq && state->buffered_io_page == NULL) {
|
||||||
trace_xen_map_ioreq_server_buffered_io_page(bufioreq_pfn);
|
trace_xen_map_ioreq_server_buffered_io_page(bufioreq_pfn);
|
||||||
|
|
||||||
state->buffered_io_page = xenforeignmemory_map(xen_fmem, xen_domid,
|
state->buffered_io_page = xenforeignmemory_map(xen_fmem, xen_domid,
|
||||||
PROT_READ | PROT_WRITE,
|
PROT_READ | PROT_WRITE,
|
||||||
1, &bufioreq_pfn,
|
1, &bufioreq_pfn,
|
||||||
NULL);
|
NULL);
|
||||||
if (state->buffered_io_page == NULL) {
|
if (state->buffered_io_page == NULL) {
|
||||||
error_report("map buffered IO page returned error %d", errno);
|
error_report("map buffered IO page returned error %d", errno);
|
||||||
return -1;
|
return -1;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (state->shared_page == NULL || state->buffered_io_page == NULL) {
|
if (state->shared_page == NULL ||
|
||||||
|
(state->has_bufioreq && state->buffered_io_page == NULL)) {
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -830,14 +851,15 @@ static void xen_do_ioreq_register(XenIOState *state,
|
|||||||
state->ioreq_local_port[i] = rc;
|
state->ioreq_local_port[i] = rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
rc = qemu_xen_evtchn_bind_interdomain(state->xce_handle, xen_domid,
|
if (state->has_bufioreq) {
|
||||||
state->bufioreq_remote_port);
|
rc = qemu_xen_evtchn_bind_interdomain(state->xce_handle, xen_domid,
|
||||||
if (rc == -1) {
|
state->bufioreq_remote_port);
|
||||||
error_report("buffered evtchn bind error %d", errno);
|
if (rc == -1) {
|
||||||
goto err;
|
error_report("buffered evtchn bind error %d", errno);
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
|
state->bufioreq_local_port = rc;
|
||||||
}
|
}
|
||||||
state->bufioreq_local_port = rc;
|
|
||||||
|
|
||||||
/* Init RAM management */
|
/* Init RAM management */
|
||||||
#ifdef XEN_COMPAT_PHYSMAP
|
#ifdef XEN_COMPAT_PHYSMAP
|
||||||
xen_map_cache_init(xen_phys_offset_to_gaddr, state);
|
xen_map_cache_init(xen_phys_offset_to_gaddr, state);
|
||||||
@ -865,6 +887,7 @@ err:
|
|||||||
}
|
}
|
||||||
|
|
||||||
void xen_register_ioreq(XenIOState *state, unsigned int max_cpus,
|
void xen_register_ioreq(XenIOState *state, unsigned int max_cpus,
|
||||||
|
uint8_t handle_bufioreq,
|
||||||
const MemoryListener *xen_memory_listener)
|
const MemoryListener *xen_memory_listener)
|
||||||
{
|
{
|
||||||
int rc;
|
int rc;
|
||||||
@ -883,7 +906,8 @@ void xen_register_ioreq(XenIOState *state, unsigned int max_cpus,
|
|||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
rc = xen_create_ioreq_server(xen_domid, &state->ioservid);
|
state->has_bufioreq = handle_bufioreq != HVM_IOREQSRV_BUFIOREQ_OFF;
|
||||||
|
rc = xen_create_ioreq_server(xen_domid, handle_bufioreq, &state->ioservid);
|
||||||
if (!rc) {
|
if (!rc) {
|
||||||
xen_do_ioreq_register(state, max_cpus, xen_memory_listener);
|
xen_do_ioreq_register(state, max_cpus, xen_memory_listener);
|
||||||
} else {
|
} else {
|
||||||
|
@ -194,7 +194,9 @@ static void xen_pvh_init(MachineState *ms)
|
|||||||
}
|
}
|
||||||
|
|
||||||
xen_pvh_init_ram(s, sysmem);
|
xen_pvh_init_ram(s, sysmem);
|
||||||
xen_register_ioreq(&s->ioreq, ms->smp.max_cpus, &xen_memory_listener);
|
xen_register_ioreq(&s->ioreq, ms->smp.max_cpus,
|
||||||
|
HVM_IOREQSRV_BUFIOREQ_ATOMIC,
|
||||||
|
&xen_memory_listener);
|
||||||
|
|
||||||
if (s->cfg.virtio_mmio_num) {
|
if (s->cfg.virtio_mmio_num) {
|
||||||
xen_create_virtio_mmio_devices(s);
|
xen_create_virtio_mmio_devices(s);
|
||||||
|
@ -81,6 +81,8 @@ typedef struct XenIOState {
|
|||||||
QLIST_HEAD(, XenPciDevice) dev_list;
|
QLIST_HEAD(, XenPciDevice) dev_list;
|
||||||
DeviceListener device_listener;
|
DeviceListener device_listener;
|
||||||
|
|
||||||
|
bool has_bufioreq;
|
||||||
|
|
||||||
Notifier exit;
|
Notifier exit;
|
||||||
} XenIOState;
|
} XenIOState;
|
||||||
|
|
||||||
@ -95,6 +97,7 @@ void xen_device_unrealize(DeviceListener *listener, DeviceState *dev);
|
|||||||
|
|
||||||
void xen_hvm_change_state_handler(void *opaque, bool running, RunState rstate);
|
void xen_hvm_change_state_handler(void *opaque, bool running, RunState rstate);
|
||||||
void xen_register_ioreq(XenIOState *state, unsigned int max_cpus,
|
void xen_register_ioreq(XenIOState *state, unsigned int max_cpus,
|
||||||
|
uint8_t handle_bufioreq,
|
||||||
const MemoryListener *xen_memory_listener);
|
const MemoryListener *xen_memory_listener);
|
||||||
|
|
||||||
void cpu_ioreq_pio(ioreq_t *req);
|
void cpu_ioreq_pio(ioreq_t *req);
|
||||||
|
@ -464,10 +464,11 @@ static inline void xen_unmap_pcidev(domid_t dom,
|
|||||||
}
|
}
|
||||||
|
|
||||||
static inline int xen_create_ioreq_server(domid_t dom,
|
static inline int xen_create_ioreq_server(domid_t dom,
|
||||||
|
int handle_bufioreq,
|
||||||
ioservid_t *ioservid)
|
ioservid_t *ioservid)
|
||||||
{
|
{
|
||||||
int rc = xendevicemodel_create_ioreq_server(xen_dmod, dom,
|
int rc = xendevicemodel_create_ioreq_server(xen_dmod, dom,
|
||||||
HVM_IOREQSRV_BUFIOREQ_ATOMIC,
|
handle_bufioreq,
|
||||||
ioservid);
|
ioservid);
|
||||||
|
|
||||||
if (rc == 0) {
|
if (rc == 0) {
|
||||||
|
Loading…
Reference in New Issue
Block a user