Xen: Use the ioreq-server API when available
The ioreq-server API added to Xen 4.5 offers better security than the existing Xen/QEMU interface because the shared pages that are used to pass emulation request/results back and forth are removed from the guest's memory space before any requests are serviced. This prevents the guest from mapping these pages (they are in a well known location) and attempting to attack QEMU by synthesizing its own request structures. Hence, this patch modifies configure to detect whether the API is available, and adds the necessary code to use the API if it is. Signed-off-by: Paul Durrant <paul.durrant@citrix.com> Signed-off-by: Stefano Stabellini <stefano.stabellini@eu.citrix.com> Acked-by: Stefano Stabellini <stefano.stabellini@eu.citrix.com>
This commit is contained in:
parent
707ff80021
commit
3996e85c18
29
configure
vendored
29
configure
vendored
@ -1869,6 +1869,32 @@ EOF
|
|||||||
#if !defined(HVM_MAX_VCPUS)
|
#if !defined(HVM_MAX_VCPUS)
|
||||||
# error HVM_MAX_VCPUS not defined
|
# error HVM_MAX_VCPUS not defined
|
||||||
#endif
|
#endif
|
||||||
|
int main(void) {
|
||||||
|
xc_interface *xc;
|
||||||
|
xs_daemon_open();
|
||||||
|
xc = xc_interface_open(0, 0, 0);
|
||||||
|
xc_hvm_set_mem_type(0, 0, HVMMEM_ram_ro, 0, 0);
|
||||||
|
xc_gnttab_open(NULL, 0);
|
||||||
|
xc_domain_add_to_physmap(0, 0, XENMAPSPACE_gmfn, 0, 0);
|
||||||
|
xc_hvm_inject_msi(xc, 0, 0xf0000000, 0x00000000);
|
||||||
|
xc_hvm_create_ioreq_server(xc, 0, 0, NULL);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
EOF
|
||||||
|
compile_prog "" "$xen_libs"
|
||||||
|
then
|
||||||
|
xen_ctrl_version=450
|
||||||
|
xen=yes
|
||||||
|
|
||||||
|
elif
|
||||||
|
cat > $TMPC <<EOF &&
|
||||||
|
#include <xenctrl.h>
|
||||||
|
#include <xenstore.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <xen/hvm/hvm_info_table.h>
|
||||||
|
#if !defined(HVM_MAX_VCPUS)
|
||||||
|
# error HVM_MAX_VCPUS not defined
|
||||||
|
#endif
|
||||||
int main(void) {
|
int main(void) {
|
||||||
xc_interface *xc;
|
xc_interface *xc;
|
||||||
xs_daemon_open();
|
xs_daemon_open();
|
||||||
@ -4283,6 +4309,9 @@ if test -n "$sparc_cpu"; then
|
|||||||
echo "Target Sparc Arch $sparc_cpu"
|
echo "Target Sparc Arch $sparc_cpu"
|
||||||
fi
|
fi
|
||||||
echo "xen support $xen"
|
echo "xen support $xen"
|
||||||
|
if test "$xen" = "yes" ; then
|
||||||
|
echo "xen ctrl version $xen_ctrl_version"
|
||||||
|
fi
|
||||||
echo "brlapi support $brlapi"
|
echo "brlapi support $brlapi"
|
||||||
echo "bluez support $bluez"
|
echo "bluez support $bluez"
|
||||||
echo "Documentation $docs"
|
echo "Documentation $docs"
|
||||||
|
@ -16,7 +16,9 @@
|
|||||||
|
|
||||||
#include "hw/hw.h"
|
#include "hw/hw.h"
|
||||||
#include "hw/xen/xen.h"
|
#include "hw/xen/xen.h"
|
||||||
|
#include "hw/pci/pci.h"
|
||||||
#include "qemu/queue.h"
|
#include "qemu/queue.h"
|
||||||
|
#include "trace.h"
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* We don't support Xen prior to 3.3.0.
|
* We don't support Xen prior to 3.3.0.
|
||||||
@ -179,4 +181,225 @@ static inline int xen_get_vmport_regs_pfn(XenXC xc, domid_t dom,
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/* Xen before 4.5 */
|
||||||
|
#if CONFIG_XEN_CTRL_INTERFACE_VERSION < 450
|
||||||
|
|
||||||
|
#ifndef HVM_PARAM_BUFIOREQ_EVTCHN
|
||||||
|
#define HVM_PARAM_BUFIOREQ_EVTCHN 26
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define IOREQ_TYPE_PCI_CONFIG 2
|
||||||
|
|
||||||
|
typedef uint32_t ioservid_t;
|
||||||
|
|
||||||
|
static inline void xen_map_memory_section(XenXC xc, domid_t dom,
|
||||||
|
ioservid_t ioservid,
|
||||||
|
MemoryRegionSection *section)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void xen_unmap_memory_section(XenXC xc, domid_t dom,
|
||||||
|
ioservid_t ioservid,
|
||||||
|
MemoryRegionSection *section)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void xen_map_io_section(XenXC xc, domid_t dom,
|
||||||
|
ioservid_t ioservid,
|
||||||
|
MemoryRegionSection *section)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void xen_unmap_io_section(XenXC xc, domid_t dom,
|
||||||
|
ioservid_t ioservid,
|
||||||
|
MemoryRegionSection *section)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void xen_map_pcidev(XenXC xc, domid_t dom,
|
||||||
|
ioservid_t ioservid,
|
||||||
|
PCIDevice *pci_dev)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void xen_unmap_pcidev(XenXC xc, domid_t dom,
|
||||||
|
ioservid_t ioservid,
|
||||||
|
PCIDevice *pci_dev)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline int xen_create_ioreq_server(XenXC xc, domid_t dom,
|
||||||
|
ioservid_t *ioservid)
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void xen_destroy_ioreq_server(XenXC xc, domid_t dom,
|
||||||
|
ioservid_t ioservid)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline int xen_get_ioreq_server_info(XenXC xc, domid_t dom,
|
||||||
|
ioservid_t ioservid,
|
||||||
|
xen_pfn_t *ioreq_pfn,
|
||||||
|
xen_pfn_t *bufioreq_pfn,
|
||||||
|
evtchn_port_t *bufioreq_evtchn)
|
||||||
|
{
|
||||||
|
unsigned long param;
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
rc = xc_get_hvm_param(xc, dom, HVM_PARAM_IOREQ_PFN, ¶m);
|
||||||
|
if (rc < 0) {
|
||||||
|
fprintf(stderr, "failed to get HVM_PARAM_IOREQ_PFN\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
*ioreq_pfn = param;
|
||||||
|
|
||||||
|
rc = xc_get_hvm_param(xc, dom, HVM_PARAM_BUFIOREQ_PFN, ¶m);
|
||||||
|
if (rc < 0) {
|
||||||
|
fprintf(stderr, "failed to get HVM_PARAM_BUFIOREQ_PFN\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
*bufioreq_pfn = param;
|
||||||
|
|
||||||
|
rc = xc_get_hvm_param(xc, dom, HVM_PARAM_BUFIOREQ_EVTCHN,
|
||||||
|
¶m);
|
||||||
|
if (rc < 0) {
|
||||||
|
fprintf(stderr, "failed to get HVM_PARAM_BUFIOREQ_EVTCHN\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
*bufioreq_evtchn = param;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline int xen_set_ioreq_server_state(XenXC xc, domid_t dom,
|
||||||
|
ioservid_t ioservid,
|
||||||
|
bool enable)
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Xen 4.5 */
|
||||||
|
#else
|
||||||
|
|
||||||
|
static inline void xen_map_memory_section(XenXC xc, domid_t dom,
|
||||||
|
ioservid_t ioservid,
|
||||||
|
MemoryRegionSection *section)
|
||||||
|
{
|
||||||
|
hwaddr start_addr = section->offset_within_address_space;
|
||||||
|
ram_addr_t size = int128_get64(section->size);
|
||||||
|
hwaddr end_addr = start_addr + size - 1;
|
||||||
|
|
||||||
|
trace_xen_map_mmio_range(ioservid, start_addr, end_addr);
|
||||||
|
xc_hvm_map_io_range_to_ioreq_server(xc, dom, ioservid, 1,
|
||||||
|
start_addr, end_addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void xen_unmap_memory_section(XenXC xc, domid_t dom,
|
||||||
|
ioservid_t ioservid,
|
||||||
|
MemoryRegionSection *section)
|
||||||
|
{
|
||||||
|
hwaddr start_addr = section->offset_within_address_space;
|
||||||
|
ram_addr_t size = int128_get64(section->size);
|
||||||
|
hwaddr end_addr = start_addr + size - 1;
|
||||||
|
|
||||||
|
trace_xen_unmap_mmio_range(ioservid, start_addr, end_addr);
|
||||||
|
xc_hvm_unmap_io_range_from_ioreq_server(xc, dom, ioservid, 1,
|
||||||
|
start_addr, end_addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void xen_map_io_section(XenXC xc, domid_t dom,
|
||||||
|
ioservid_t ioservid,
|
||||||
|
MemoryRegionSection *section)
|
||||||
|
{
|
||||||
|
hwaddr start_addr = section->offset_within_address_space;
|
||||||
|
ram_addr_t size = int128_get64(section->size);
|
||||||
|
hwaddr end_addr = start_addr + size - 1;
|
||||||
|
|
||||||
|
trace_xen_map_portio_range(ioservid, start_addr, end_addr);
|
||||||
|
xc_hvm_map_io_range_to_ioreq_server(xc, dom, ioservid, 0,
|
||||||
|
start_addr, end_addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void xen_unmap_io_section(XenXC xc, domid_t dom,
|
||||||
|
ioservid_t ioservid,
|
||||||
|
MemoryRegionSection *section)
|
||||||
|
{
|
||||||
|
hwaddr start_addr = section->offset_within_address_space;
|
||||||
|
ram_addr_t size = int128_get64(section->size);
|
||||||
|
hwaddr end_addr = start_addr + size - 1;
|
||||||
|
|
||||||
|
trace_xen_unmap_portio_range(ioservid, start_addr, end_addr);
|
||||||
|
xc_hvm_unmap_io_range_from_ioreq_server(xc, dom, ioservid, 0,
|
||||||
|
start_addr, end_addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void xen_map_pcidev(XenXC xc, domid_t dom,
|
||||||
|
ioservid_t ioservid,
|
||||||
|
PCIDevice *pci_dev)
|
||||||
|
{
|
||||||
|
trace_xen_map_pcidev(ioservid, pci_bus_num(pci_dev->bus),
|
||||||
|
PCI_SLOT(pci_dev->devfn), PCI_FUNC(pci_dev->devfn));
|
||||||
|
xc_hvm_map_pcidev_to_ioreq_server(xc, dom, ioservid,
|
||||||
|
0, pci_bus_num(pci_dev->bus),
|
||||||
|
PCI_SLOT(pci_dev->devfn),
|
||||||
|
PCI_FUNC(pci_dev->devfn));
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void xen_unmap_pcidev(XenXC xc, domid_t dom,
|
||||||
|
ioservid_t ioservid,
|
||||||
|
PCIDevice *pci_dev)
|
||||||
|
{
|
||||||
|
trace_xen_unmap_pcidev(ioservid, pci_bus_num(pci_dev->bus),
|
||||||
|
PCI_SLOT(pci_dev->devfn), PCI_FUNC(pci_dev->devfn));
|
||||||
|
xc_hvm_unmap_pcidev_from_ioreq_server(xc, dom, ioservid,
|
||||||
|
0, pci_bus_num(pci_dev->bus),
|
||||||
|
PCI_SLOT(pci_dev->devfn),
|
||||||
|
PCI_FUNC(pci_dev->devfn));
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline int xen_create_ioreq_server(XenXC xc, domid_t dom,
|
||||||
|
ioservid_t *ioservid)
|
||||||
|
{
|
||||||
|
int rc = xc_hvm_create_ioreq_server(xc, dom, 1, ioservid);
|
||||||
|
|
||||||
|
if (rc == 0) {
|
||||||
|
trace_xen_ioreq_server_create(*ioservid);
|
||||||
|
}
|
||||||
|
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void xen_destroy_ioreq_server(XenXC xc, domid_t dom,
|
||||||
|
ioservid_t ioservid)
|
||||||
|
{
|
||||||
|
trace_xen_ioreq_server_destroy(ioservid);
|
||||||
|
xc_hvm_destroy_ioreq_server(xc, dom, ioservid);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline int xen_get_ioreq_server_info(XenXC xc, domid_t dom,
|
||||||
|
ioservid_t ioservid,
|
||||||
|
xen_pfn_t *ioreq_pfn,
|
||||||
|
xen_pfn_t *bufioreq_pfn,
|
||||||
|
evtchn_port_t *bufioreq_evtchn)
|
||||||
|
{
|
||||||
|
return xc_hvm_get_ioreq_server_info(xc, dom, ioservid,
|
||||||
|
ioreq_pfn, bufioreq_pfn,
|
||||||
|
bufioreq_evtchn);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline int xen_set_ioreq_server_state(XenXC xc, domid_t dom,
|
||||||
|
ioservid_t ioservid,
|
||||||
|
bool enable)
|
||||||
|
{
|
||||||
|
trace_xen_ioreq_server_state(ioservid, enable);
|
||||||
|
return xc_hvm_set_ioreq_server_state(xc, dom, ioservid, enable);
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
#endif /* QEMU_HW_XEN_COMMON_H */
|
#endif /* QEMU_HW_XEN_COMMON_H */
|
||||||
|
@ -897,6 +897,15 @@ pvscsi_tx_rings_num_pages(const char* label, uint32_t num) "Number of %s pages:
|
|||||||
# xen-hvm.c
|
# xen-hvm.c
|
||||||
xen_ram_alloc(unsigned long ram_addr, unsigned long size) "requested: %#lx, size %#lx"
|
xen_ram_alloc(unsigned long ram_addr, unsigned long size) "requested: %#lx, size %#lx"
|
||||||
xen_client_set_memory(uint64_t start_addr, unsigned long size, bool log_dirty) "%#"PRIx64" size %#lx, log_dirty %i"
|
xen_client_set_memory(uint64_t start_addr, unsigned long size, bool log_dirty) "%#"PRIx64" size %#lx, log_dirty %i"
|
||||||
|
xen_ioreq_server_create(uint32_t id) "id: %u"
|
||||||
|
xen_ioreq_server_destroy(uint32_t id) "id: %u"
|
||||||
|
xen_ioreq_server_state(uint32_t id, bool enable) "id: %u: enable: %i"
|
||||||
|
xen_map_mmio_range(uint32_t id, uint64_t start_addr, uint64_t end_addr) "id: %u start: %#"PRIx64" end: %#"PRIx64
|
||||||
|
xen_unmap_mmio_range(uint32_t id, uint64_t start_addr, uint64_t end_addr) "id: %u start: %#"PRIx64" end: %#"PRIx64
|
||||||
|
xen_map_portio_range(uint32_t id, uint64_t start_addr, uint64_t end_addr) "id: %u start: %#"PRIx64" end: %#"PRIx64
|
||||||
|
xen_unmap_portio_range(uint32_t id, uint64_t start_addr, uint64_t end_addr) "id: %u start: %#"PRIx64" end: %#"PRIx64
|
||||||
|
xen_map_pcidev(uint32_t id, uint8_t bus, uint8_t dev, uint8_t func) "id: %u bdf: %02x.%02x.%02x"
|
||||||
|
xen_unmap_pcidev(uint32_t id, uint8_t bus, uint8_t dev, uint8_t func) "id: %u bdf: %02x.%02x.%02x"
|
||||||
|
|
||||||
# xen-mapcache.c
|
# xen-mapcache.c
|
||||||
xen_map_cache(uint64_t phys_addr) "want %#"PRIx64
|
xen_map_cache(uint64_t phys_addr) "want %#"PRIx64
|
||||||
|
160
xen-hvm.c
160
xen-hvm.c
@ -85,9 +85,6 @@ static inline ioreq_t *xen_vcpu_ioreq(shared_iopage_t *shared_page, int vcpu)
|
|||||||
}
|
}
|
||||||
# define FMT_ioreq_size "u"
|
# define FMT_ioreq_size "u"
|
||||||
#endif
|
#endif
|
||||||
#ifndef HVM_PARAM_BUFIOREQ_EVTCHN
|
|
||||||
#define HVM_PARAM_BUFIOREQ_EVTCHN 26
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#define BUFFER_IO_MAX_DELAY 100
|
#define BUFFER_IO_MAX_DELAY 100
|
||||||
/* Leave some slack so that hvmloader does not complain about lack of
|
/* Leave some slack so that hvmloader does not complain about lack of
|
||||||
@ -107,6 +104,7 @@ typedef struct XenPhysmap {
|
|||||||
} XenPhysmap;
|
} XenPhysmap;
|
||||||
|
|
||||||
typedef struct XenIOState {
|
typedef struct XenIOState {
|
||||||
|
ioservid_t ioservid;
|
||||||
shared_iopage_t *shared_page;
|
shared_iopage_t *shared_page;
|
||||||
shared_vmport_iopage_t *shared_vmport_page;
|
shared_vmport_iopage_t *shared_vmport_page;
|
||||||
buffered_iopage_t *buffered_io_page;
|
buffered_iopage_t *buffered_io_page;
|
||||||
@ -123,6 +121,8 @@ typedef struct XenIOState {
|
|||||||
|
|
||||||
struct xs_handle *xenstore;
|
struct xs_handle *xenstore;
|
||||||
MemoryListener memory_listener;
|
MemoryListener memory_listener;
|
||||||
|
MemoryListener io_listener;
|
||||||
|
DeviceListener device_listener;
|
||||||
QLIST_HEAD(, XenPhysmap) physmap;
|
QLIST_HEAD(, XenPhysmap) physmap;
|
||||||
hwaddr free_phys_offset;
|
hwaddr free_phys_offset;
|
||||||
const XenPhysmap *log_for_dirtybit;
|
const XenPhysmap *log_for_dirtybit;
|
||||||
@ -491,12 +491,23 @@ static void xen_set_memory(struct MemoryListener *listener,
|
|||||||
bool log_dirty = memory_region_is_logging(section->mr);
|
bool log_dirty = memory_region_is_logging(section->mr);
|
||||||
hvmmem_type_t mem_type;
|
hvmmem_type_t mem_type;
|
||||||
|
|
||||||
|
if (section->mr == &ram_memory) {
|
||||||
|
return;
|
||||||
|
} else {
|
||||||
|
if (add) {
|
||||||
|
xen_map_memory_section(xen_xc, xen_domid, state->ioservid,
|
||||||
|
section);
|
||||||
|
} else {
|
||||||
|
xen_unmap_memory_section(xen_xc, xen_domid, state->ioservid,
|
||||||
|
section);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (!memory_region_is_ram(section->mr)) {
|
if (!memory_region_is_ram(section->mr)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!(section->mr != &ram_memory
|
if (log_dirty != add) {
|
||||||
&& ( (log_dirty && add) || (!log_dirty && !add)))) {
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -539,6 +550,50 @@ static void xen_region_del(MemoryListener *listener,
|
|||||||
memory_region_unref(section->mr);
|
memory_region_unref(section->mr);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void xen_io_add(MemoryListener *listener,
|
||||||
|
MemoryRegionSection *section)
|
||||||
|
{
|
||||||
|
XenIOState *state = container_of(listener, XenIOState, io_listener);
|
||||||
|
|
||||||
|
memory_region_ref(section->mr);
|
||||||
|
|
||||||
|
xen_map_io_section(xen_xc, xen_domid, state->ioservid, section);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void xen_io_del(MemoryListener *listener,
|
||||||
|
MemoryRegionSection *section)
|
||||||
|
{
|
||||||
|
XenIOState *state = container_of(listener, XenIOState, io_listener);
|
||||||
|
|
||||||
|
xen_unmap_io_section(xen_xc, xen_domid, state->ioservid, section);
|
||||||
|
|
||||||
|
memory_region_unref(section->mr);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void xen_device_realize(DeviceListener *listener,
|
||||||
|
DeviceState *dev)
|
||||||
|
{
|
||||||
|
XenIOState *state = container_of(listener, XenIOState, device_listener);
|
||||||
|
|
||||||
|
if (object_dynamic_cast(OBJECT(dev), TYPE_PCI_DEVICE)) {
|
||||||
|
PCIDevice *pci_dev = PCI_DEVICE(dev);
|
||||||
|
|
||||||
|
xen_map_pcidev(xen_xc, xen_domid, state->ioservid, pci_dev);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void xen_device_unrealize(DeviceListener *listener,
|
||||||
|
DeviceState *dev)
|
||||||
|
{
|
||||||
|
XenIOState *state = container_of(listener, XenIOState, device_listener);
|
||||||
|
|
||||||
|
if (object_dynamic_cast(OBJECT(dev), TYPE_PCI_DEVICE)) {
|
||||||
|
PCIDevice *pci_dev = PCI_DEVICE(dev);
|
||||||
|
|
||||||
|
xen_unmap_pcidev(xen_xc, xen_domid, state->ioservid, pci_dev);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static void xen_sync_dirty_bitmap(XenIOState *state,
|
static void xen_sync_dirty_bitmap(XenIOState *state,
|
||||||
hwaddr start_addr,
|
hwaddr start_addr,
|
||||||
ram_addr_t size)
|
ram_addr_t size)
|
||||||
@ -639,6 +694,17 @@ static MemoryListener xen_memory_listener = {
|
|||||||
.priority = 10,
|
.priority = 10,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static MemoryListener xen_io_listener = {
|
||||||
|
.region_add = xen_io_add,
|
||||||
|
.region_del = xen_io_del,
|
||||||
|
.priority = 10,
|
||||||
|
};
|
||||||
|
|
||||||
|
static DeviceListener xen_device_listener = {
|
||||||
|
.realize = xen_device_realize,
|
||||||
|
.unrealize = xen_device_unrealize,
|
||||||
|
};
|
||||||
|
|
||||||
/* get the ioreq packets from share mem */
|
/* get the ioreq packets from share mem */
|
||||||
static ioreq_t *cpu_get_ioreq_from_shared_memory(XenIOState *state, int vcpu)
|
static ioreq_t *cpu_get_ioreq_from_shared_memory(XenIOState *state, int vcpu)
|
||||||
{
|
{
|
||||||
@ -887,6 +953,27 @@ static void handle_ioreq(XenIOState *state, ioreq_t *req)
|
|||||||
case IOREQ_TYPE_INVALIDATE:
|
case IOREQ_TYPE_INVALIDATE:
|
||||||
xen_invalidate_map_cache();
|
xen_invalidate_map_cache();
|
||||||
break;
|
break;
|
||||||
|
case IOREQ_TYPE_PCI_CONFIG: {
|
||||||
|
uint32_t sbdf = req->addr >> 32;
|
||||||
|
uint32_t val;
|
||||||
|
|
||||||
|
/* Fake a write to port 0xCF8 so that
|
||||||
|
* the config space access will target the
|
||||||
|
* correct device model.
|
||||||
|
*/
|
||||||
|
val = (1u << 31) |
|
||||||
|
((req->addr & 0x0f00) << 16) |
|
||||||
|
((sbdf & 0xffff) << 8) |
|
||||||
|
(req->addr & 0xfc);
|
||||||
|
do_outp(0xcf8, 4, val);
|
||||||
|
|
||||||
|
/* Now issue the config space access via
|
||||||
|
* port 0xCFC
|
||||||
|
*/
|
||||||
|
req->addr = 0xcfc | (req->addr & 0x03);
|
||||||
|
cpu_ioreq_pio(req);
|
||||||
|
break;
|
||||||
|
}
|
||||||
default:
|
default:
|
||||||
hw_error("Invalid ioreq type 0x%x\n", req->type);
|
hw_error("Invalid ioreq type 0x%x\n", req->type);
|
||||||
}
|
}
|
||||||
@ -1017,9 +1104,15 @@ static void xen_main_loop_prepare(XenIOState *state)
|
|||||||
static void xen_hvm_change_state_handler(void *opaque, int running,
|
static void xen_hvm_change_state_handler(void *opaque, int running,
|
||||||
RunState rstate)
|
RunState rstate)
|
||||||
{
|
{
|
||||||
|
XenIOState *state = opaque;
|
||||||
|
|
||||||
if (running) {
|
if (running) {
|
||||||
xen_main_loop_prepare((XenIOState *)opaque);
|
xen_main_loop_prepare(state);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
xen_set_ioreq_server_state(xen_xc, xen_domid,
|
||||||
|
state->ioservid,
|
||||||
|
(rstate == RUN_STATE_RUNNING));
|
||||||
}
|
}
|
||||||
|
|
||||||
static void xen_exit_notifier(Notifier *n, void *data)
|
static void xen_exit_notifier(Notifier *n, void *data)
|
||||||
@ -1088,8 +1181,9 @@ int xen_hvm_init(ram_addr_t *below_4g_mem_size, ram_addr_t *above_4g_mem_size,
|
|||||||
MemoryRegion **ram_memory)
|
MemoryRegion **ram_memory)
|
||||||
{
|
{
|
||||||
int i, rc;
|
int i, rc;
|
||||||
unsigned long ioreq_pfn;
|
xen_pfn_t ioreq_pfn;
|
||||||
unsigned long bufioreq_evtchn;
|
xen_pfn_t bufioreq_pfn;
|
||||||
|
evtchn_port_t bufioreq_evtchn;
|
||||||
XenIOState *state;
|
XenIOState *state;
|
||||||
|
|
||||||
state = g_malloc0(sizeof (XenIOState));
|
state = g_malloc0(sizeof (XenIOState));
|
||||||
@ -1106,6 +1200,12 @@ int xen_hvm_init(ram_addr_t *below_4g_mem_size, ram_addr_t *above_4g_mem_size,
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
rc = xen_create_ioreq_server(xen_xc, xen_domid, &state->ioservid);
|
||||||
|
if (rc < 0) {
|
||||||
|
perror("xen: ioreq server create");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
state->exit.notify = xen_exit_notifier;
|
state->exit.notify = xen_exit_notifier;
|
||||||
qemu_add_exit_notifier(&state->exit);
|
qemu_add_exit_notifier(&state->exit);
|
||||||
|
|
||||||
@ -1115,8 +1215,18 @@ int xen_hvm_init(ram_addr_t *below_4g_mem_size, ram_addr_t *above_4g_mem_size,
|
|||||||
state->wakeup.notify = xen_wakeup_notifier;
|
state->wakeup.notify = xen_wakeup_notifier;
|
||||||
qemu_register_wakeup_notifier(&state->wakeup);
|
qemu_register_wakeup_notifier(&state->wakeup);
|
||||||
|
|
||||||
xc_get_hvm_param(xen_xc, xen_domid, HVM_PARAM_IOREQ_PFN, &ioreq_pfn);
|
rc = xen_get_ioreq_server_info(xen_xc, xen_domid, state->ioservid,
|
||||||
|
&ioreq_pfn, &bufioreq_pfn,
|
||||||
|
&bufioreq_evtchn);
|
||||||
|
if (rc < 0) {
|
||||||
|
hw_error("failed to get ioreq server info: error %d handle=" XC_INTERFACE_FMT,
|
||||||
|
errno, xen_xc);
|
||||||
|
}
|
||||||
|
|
||||||
DPRINTF("shared page at pfn %lx\n", ioreq_pfn);
|
DPRINTF("shared page at pfn %lx\n", ioreq_pfn);
|
||||||
|
DPRINTF("buffered io page at pfn %lx\n", bufioreq_pfn);
|
||||||
|
DPRINTF("buffered io evtchn is %x\n", bufioreq_evtchn);
|
||||||
|
|
||||||
state->shared_page = xc_map_foreign_range(xen_xc, xen_domid, XC_PAGE_SIZE,
|
state->shared_page = xc_map_foreign_range(xen_xc, xen_domid, XC_PAGE_SIZE,
|
||||||
PROT_READ|PROT_WRITE, ioreq_pfn);
|
PROT_READ|PROT_WRITE, ioreq_pfn);
|
||||||
if (state->shared_page == NULL) {
|
if (state->shared_page == NULL) {
|
||||||
@ -1138,10 +1248,10 @@ int xen_hvm_init(ram_addr_t *below_4g_mem_size, ram_addr_t *above_4g_mem_size,
|
|||||||
hw_error("get vmport regs pfn returned error %d, rc=%d", errno, rc);
|
hw_error("get vmport regs pfn returned error %d, rc=%d", errno, rc);
|
||||||
}
|
}
|
||||||
|
|
||||||
xc_get_hvm_param(xen_xc, xen_domid, HVM_PARAM_BUFIOREQ_PFN, &ioreq_pfn);
|
state->buffered_io_page = xc_map_foreign_range(xen_xc, xen_domid,
|
||||||
DPRINTF("buffered io page at pfn %lx\n", ioreq_pfn);
|
XC_PAGE_SIZE,
|
||||||
state->buffered_io_page = xc_map_foreign_range(xen_xc, xen_domid, XC_PAGE_SIZE,
|
PROT_READ|PROT_WRITE,
|
||||||
PROT_READ|PROT_WRITE, ioreq_pfn);
|
bufioreq_pfn);
|
||||||
if (state->buffered_io_page == NULL) {
|
if (state->buffered_io_page == NULL) {
|
||||||
hw_error("map buffered IO page returned error %d", errno);
|
hw_error("map buffered IO page returned error %d", errno);
|
||||||
}
|
}
|
||||||
@ -1149,6 +1259,12 @@ int xen_hvm_init(ram_addr_t *below_4g_mem_size, ram_addr_t *above_4g_mem_size,
|
|||||||
/* Note: cpus is empty at this point in init */
|
/* Note: cpus is empty at this point in init */
|
||||||
state->cpu_by_vcpu_id = g_malloc0(max_cpus * sizeof(CPUState *));
|
state->cpu_by_vcpu_id = g_malloc0(max_cpus * sizeof(CPUState *));
|
||||||
|
|
||||||
|
rc = xen_set_ioreq_server_state(xen_xc, xen_domid, state->ioservid, true);
|
||||||
|
if (rc < 0) {
|
||||||
|
hw_error("failed to enable ioreq server info: error %d handle=" XC_INTERFACE_FMT,
|
||||||
|
errno, xen_xc);
|
||||||
|
}
|
||||||
|
|
||||||
state->ioreq_local_port = g_malloc0(max_cpus * sizeof (evtchn_port_t));
|
state->ioreq_local_port = g_malloc0(max_cpus * sizeof (evtchn_port_t));
|
||||||
|
|
||||||
/* FIXME: how about if we overflow the page here? */
|
/* FIXME: how about if we overflow the page here? */
|
||||||
@ -1156,22 +1272,16 @@ int xen_hvm_init(ram_addr_t *below_4g_mem_size, ram_addr_t *above_4g_mem_size,
|
|||||||
rc = xc_evtchn_bind_interdomain(state->xce_handle, xen_domid,
|
rc = xc_evtchn_bind_interdomain(state->xce_handle, xen_domid,
|
||||||
xen_vcpu_eport(state->shared_page, i));
|
xen_vcpu_eport(state->shared_page, i));
|
||||||
if (rc == -1) {
|
if (rc == -1) {
|
||||||
fprintf(stderr, "bind interdomain ioctl error %d\n", errno);
|
fprintf(stderr, "shared evtchn %d bind error %d\n", i, errno);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
state->ioreq_local_port[i] = rc;
|
state->ioreq_local_port[i] = rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
rc = xc_get_hvm_param(xen_xc, xen_domid, HVM_PARAM_BUFIOREQ_EVTCHN,
|
|
||||||
&bufioreq_evtchn);
|
|
||||||
if (rc < 0) {
|
|
||||||
fprintf(stderr, "failed to get HVM_PARAM_BUFIOREQ_EVTCHN\n");
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
rc = xc_evtchn_bind_interdomain(state->xce_handle, xen_domid,
|
rc = xc_evtchn_bind_interdomain(state->xce_handle, xen_domid,
|
||||||
(uint32_t)bufioreq_evtchn);
|
bufioreq_evtchn);
|
||||||
if (rc == -1) {
|
if (rc == -1) {
|
||||||
fprintf(stderr, "bind interdomain ioctl error %d\n", errno);
|
fprintf(stderr, "buffered evtchn bind error %d\n", errno);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
state->bufioreq_local_port = rc;
|
state->bufioreq_local_port = rc;
|
||||||
@ -1187,6 +1297,12 @@ int xen_hvm_init(ram_addr_t *below_4g_mem_size, ram_addr_t *above_4g_mem_size,
|
|||||||
memory_listener_register(&state->memory_listener, &address_space_memory);
|
memory_listener_register(&state->memory_listener, &address_space_memory);
|
||||||
state->log_for_dirtybit = NULL;
|
state->log_for_dirtybit = NULL;
|
||||||
|
|
||||||
|
state->io_listener = xen_io_listener;
|
||||||
|
memory_listener_register(&state->io_listener, &address_space_io);
|
||||||
|
|
||||||
|
state->device_listener = xen_device_listener;
|
||||||
|
device_listener_register(&state->device_listener);
|
||||||
|
|
||||||
/* Initialize backend core & drivers */
|
/* Initialize backend core & drivers */
|
||||||
if (xen_be_init() != 0) {
|
if (xen_be_init() != 0) {
|
||||||
fprintf(stderr, "%s: xen backend core setup failed\n", __FUNCTION__);
|
fprintf(stderr, "%s: xen backend core setup failed\n", __FUNCTION__);
|
||||||
|
Loading…
Reference in New Issue
Block a user