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:
Edgar E. Iglesias 2024-09-04 19:28:06 +02:00
parent abdfd6549d
commit b2150e403a
5 changed files with 73 additions and 41 deletions

View File

@ -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);

View File

@ -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 {

View File

@ -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);

View File

@ -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);

View File

@ -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) {