mirror of https://github.com/xqemu/xqemu.git
Merge remote-tracking branch 'kraxel/usb.84' into staging
# By Andreas Färber (3) and others # Via Gerd Hoffmann * kraxel/usb.84: usb: fix serial number for hid devices usb: add serial bus property usb-host-libusb: set USB_DEV_FLAG_IS_HOST usb/host-libusb: Fix building with libusb git master code usb/hcd-ehci: Add Faraday FUSBH200 support usb/hcd-ehci: Replace PORTSC macros with variables usb/hcd-ehci: Add Tegra2 SysBus EHCI device usb/hcd-ehci: Split off instance_init from realize usb/hcd-ehci-sysbus: Convert to QOM realize
This commit is contained in:
commit
89f204d2c6
|
@ -622,6 +622,18 @@ static QEMUMachine pc_machine_v0_13 = {
|
|||
.driver = "virtio-serial-pci",\
|
||||
.property = "vectors",\
|
||||
.value = stringify(0),\
|
||||
},{\
|
||||
.driver = "usb-mouse",\
|
||||
.property = "serial",\
|
||||
.value = "1",\
|
||||
},{\
|
||||
.driver = "usb-tablet",\
|
||||
.property = "serial",\
|
||||
.value = "1",\
|
||||
},{\
|
||||
.driver = "usb-kbd",\
|
||||
.property = "serial",\
|
||||
.value = "1",\
|
||||
}
|
||||
|
||||
static QEMUMachine pc_machine_v0_12 = {
|
||||
|
|
|
@ -13,6 +13,7 @@ static int usb_qdev_exit(DeviceState *qdev);
|
|||
|
||||
static Property usb_props[] = {
|
||||
DEFINE_PROP_STRING("port", USBDevice, port_path),
|
||||
DEFINE_PROP_STRING("serial", USBDevice, serial),
|
||||
DEFINE_PROP_BIT("full-path", USBDevice, flags,
|
||||
USB_DEV_FLAG_FULL_PATH, true),
|
||||
DEFINE_PROP_END_OF_LIST()
|
||||
|
|
|
@ -566,6 +566,12 @@ void usb_desc_create_serial(USBDevice *dev)
|
|||
char *path;
|
||||
int dst;
|
||||
|
||||
if (dev->serial) {
|
||||
/* 'serial' usb bus property has priority if present */
|
||||
usb_desc_set_string(dev, index, dev->serial);
|
||||
return;
|
||||
}
|
||||
|
||||
assert(index != 0 && desc->str[index] != NULL);
|
||||
dst = snprintf(serial, sizeof(serial), "%s", desc->str[index]);
|
||||
path = qdev_get_dev_path(hcd);
|
||||
|
|
|
@ -560,6 +560,9 @@ static int usb_hid_initfn(USBDevice *dev, int kind)
|
|||
{
|
||||
USBHIDState *us = DO_UPCAST(USBHIDState, dev, dev);
|
||||
|
||||
if (dev->serial) {
|
||||
usb_desc_set_string(dev, STR_SERIALNUMBER, dev->serial);
|
||||
}
|
||||
usb_desc_init(dev);
|
||||
us->intr = usb_ep_get(dev, USB_TOKEN_IN, 1);
|
||||
hid_init(&us->hid, kind, usb_hid_changed);
|
||||
|
|
|
@ -58,7 +58,6 @@ typedef struct {
|
|||
USBPacket *packet;
|
||||
/* usb-storage only */
|
||||
BlockConf conf;
|
||||
char *serial;
|
||||
uint32_t removable;
|
||||
} MSDState;
|
||||
|
||||
|
@ -602,7 +601,7 @@ static int usb_msd_initfn_storage(USBDevice *dev)
|
|||
return -1;
|
||||
}
|
||||
|
||||
blkconf_serial(&s->conf, &s->serial);
|
||||
blkconf_serial(&s->conf, &dev->serial);
|
||||
|
||||
/*
|
||||
* Hack alert: this pretends to be a block device, but it's really
|
||||
|
@ -616,16 +615,11 @@ static int usb_msd_initfn_storage(USBDevice *dev)
|
|||
bdrv_detach_dev(bs, &s->dev.qdev);
|
||||
s->conf.bs = NULL;
|
||||
|
||||
if (s->serial) {
|
||||
usb_desc_set_string(dev, STR_SERIALNUMBER, s->serial);
|
||||
} else {
|
||||
usb_desc_create_serial(dev);
|
||||
}
|
||||
|
||||
usb_desc_create_serial(dev);
|
||||
usb_desc_init(dev);
|
||||
scsi_bus_new(&s->bus, &s->dev.qdev, &usb_msd_scsi_info_storage, NULL);
|
||||
scsi_dev = scsi_bus_legacy_add_drive(&s->bus, bs, 0, !!s->removable,
|
||||
s->conf.bootindex, s->serial);
|
||||
s->conf.bootindex, dev->serial);
|
||||
if (!scsi_dev) {
|
||||
return -1;
|
||||
}
|
||||
|
@ -734,7 +728,6 @@ static const VMStateDescription vmstate_usb_msd = {
|
|||
|
||||
static Property msd_properties[] = {
|
||||
DEFINE_BLOCK_PROPERTIES(MSDState, conf),
|
||||
DEFINE_PROP_STRING("serial", MSDState, serial),
|
||||
DEFINE_PROP_BIT("removable", MSDState, removable, 0, false),
|
||||
DEFINE_PROP_END_OF_LIST(),
|
||||
};
|
||||
|
|
|
@ -60,20 +60,30 @@ static int usb_ehci_pci_initfn(PCIDevice *dev)
|
|||
pci_conf[0x6e] = 0x00;
|
||||
pci_conf[0x6f] = 0xc0; /* USBLEFCTLSTS */
|
||||
|
||||
s->caps[0x09] = 0x68; /* EECP */
|
||||
|
||||
s->irq = dev->irq[3];
|
||||
s->as = pci_get_address_space(dev);
|
||||
|
||||
s->capsbase = 0x00;
|
||||
s->opregbase = 0x20;
|
||||
|
||||
usb_ehci_initfn(s, DEVICE(dev));
|
||||
usb_ehci_realize(s, DEVICE(dev), NULL);
|
||||
pci_register_bar(dev, 0, PCI_BASE_ADDRESS_SPACE_MEMORY, &s->mem);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void usb_ehci_pci_init(Object *obj)
|
||||
{
|
||||
EHCIPCIState *i = PCI_EHCI(obj);
|
||||
EHCIState *s = &i->ehci;
|
||||
|
||||
s->caps[0x09] = 0x68; /* EECP */
|
||||
|
||||
s->capsbase = 0x00;
|
||||
s->opregbase = 0x20;
|
||||
s->portscbase = 0x44;
|
||||
s->portnr = NB_PORTS;
|
||||
|
||||
usb_ehci_init(s, DEVICE(obj));
|
||||
}
|
||||
|
||||
static void usb_ehci_pci_write_config(PCIDevice *dev, uint32_t addr,
|
||||
uint32_t val, int l)
|
||||
{
|
||||
|
@ -122,6 +132,7 @@ static const TypeInfo ehci_pci_type_info = {
|
|||
.name = TYPE_PCI_EHCI,
|
||||
.parent = TYPE_PCI_DEVICE,
|
||||
.instance_size = sizeof(EHCIPCIState),
|
||||
.instance_init = usb_ehci_pci_init,
|
||||
.abstract = true,
|
||||
.class_init = ehci_class_init,
|
||||
};
|
||||
|
|
|
@ -32,28 +32,42 @@ static Property ehci_sysbus_properties[] = {
|
|||
DEFINE_PROP_END_OF_LIST(),
|
||||
};
|
||||
|
||||
static int usb_ehci_sysbus_initfn(SysBusDevice *dev)
|
||||
static void usb_ehci_sysbus_realize(DeviceState *dev, Error **errp)
|
||||
{
|
||||
SysBusDevice *d = SYS_BUS_DEVICE(dev);
|
||||
EHCISysBusState *i = SYS_BUS_EHCI(dev);
|
||||
SysBusEHCIClass *sec = SYS_BUS_EHCI_GET_CLASS(dev);
|
||||
EHCIState *s = &i->ehci;
|
||||
|
||||
usb_ehci_realize(s, dev, errp);
|
||||
sysbus_init_irq(d, &s->irq);
|
||||
}
|
||||
|
||||
static void ehci_sysbus_init(Object *obj)
|
||||
{
|
||||
SysBusDevice *d = SYS_BUS_DEVICE(obj);
|
||||
EHCISysBusState *i = SYS_BUS_EHCI(obj);
|
||||
SysBusEHCIClass *sec = SYS_BUS_EHCI_GET_CLASS(obj);
|
||||
EHCIState *s = &i->ehci;
|
||||
|
||||
s->capsbase = sec->capsbase;
|
||||
s->opregbase = sec->opregbase;
|
||||
s->portscbase = sec->portscbase;
|
||||
s->portnr = sec->portnr;
|
||||
s->as = &address_space_memory;
|
||||
|
||||
usb_ehci_initfn(s, DEVICE(dev));
|
||||
sysbus_init_irq(dev, &s->irq);
|
||||
sysbus_init_mmio(dev, &s->mem);
|
||||
return 0;
|
||||
usb_ehci_init(s, DEVICE(obj));
|
||||
sysbus_init_mmio(d, &s->mem);
|
||||
}
|
||||
|
||||
static void ehci_sysbus_class_init(ObjectClass *klass, void *data)
|
||||
{
|
||||
DeviceClass *dc = DEVICE_CLASS(klass);
|
||||
SysBusDeviceClass *k = SYS_BUS_DEVICE_CLASS(klass);
|
||||
SysBusEHCIClass *sec = SYS_BUS_EHCI_CLASS(klass);
|
||||
|
||||
k->init = usb_ehci_sysbus_initfn;
|
||||
sec->portscbase = 0x44;
|
||||
sec->portnr = NB_PORTS;
|
||||
|
||||
dc->realize = usb_ehci_sysbus_realize;
|
||||
dc->vmsd = &vmstate_ehci_sysbus;
|
||||
dc->props = ehci_sysbus_properties;
|
||||
}
|
||||
|
@ -62,6 +76,7 @@ static const TypeInfo ehci_type_info = {
|
|||
.name = TYPE_SYS_BUS_EHCI,
|
||||
.parent = TYPE_SYS_BUS_DEVICE,
|
||||
.instance_size = sizeof(EHCISysBusState),
|
||||
.instance_init = ehci_sysbus_init,
|
||||
.abstract = true,
|
||||
.class_init = ehci_sysbus_class_init,
|
||||
.class_size = sizeof(SysBusEHCIClass),
|
||||
|
@ -95,11 +110,101 @@ static const TypeInfo ehci_exynos4210_type_info = {
|
|||
.class_init = ehci_exynos4210_class_init,
|
||||
};
|
||||
|
||||
static void ehci_tegra2_class_init(ObjectClass *oc, void *data)
|
||||
{
|
||||
SysBusEHCIClass *sec = SYS_BUS_EHCI_CLASS(oc);
|
||||
|
||||
sec->capsbase = 0x100;
|
||||
sec->opregbase = 0x140;
|
||||
}
|
||||
|
||||
static const TypeInfo ehci_tegra2_type_info = {
|
||||
.name = TYPE_TEGRA2_EHCI,
|
||||
.parent = TYPE_SYS_BUS_EHCI,
|
||||
.class_init = ehci_tegra2_class_init,
|
||||
};
|
||||
|
||||
/*
|
||||
* Faraday FUSBH200 USB 2.0 EHCI
|
||||
*/
|
||||
|
||||
/**
|
||||
* FUSBH200EHCIRegs:
|
||||
* @FUSBH200_REG_EOF_ASTR: EOF/Async. Sleep Timer Register
|
||||
* @FUSBH200_REG_BMCSR: Bus Monitor Control/Status Register
|
||||
*/
|
||||
enum FUSBH200EHCIRegs {
|
||||
FUSBH200_REG_EOF_ASTR = 0x34,
|
||||
FUSBH200_REG_BMCSR = 0x40,
|
||||
};
|
||||
|
||||
static uint64_t fusbh200_ehci_read(void *opaque, hwaddr addr, unsigned size)
|
||||
{
|
||||
EHCIState *s = opaque;
|
||||
hwaddr off = s->opregbase + s->portscbase + 4 * s->portnr + addr;
|
||||
|
||||
switch (off) {
|
||||
case FUSBH200_REG_EOF_ASTR:
|
||||
return 0x00000041;
|
||||
case FUSBH200_REG_BMCSR:
|
||||
/* High-Speed, VBUS valid, interrupt level-high active */
|
||||
return (2 << 9) | (1 << 8) | (1 << 3);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void fusbh200_ehci_write(void *opaque, hwaddr addr, uint64_t val,
|
||||
unsigned size)
|
||||
{
|
||||
}
|
||||
|
||||
static const MemoryRegionOps fusbh200_ehci_mmio_ops = {
|
||||
.read = fusbh200_ehci_read,
|
||||
.write = fusbh200_ehci_write,
|
||||
.valid.min_access_size = 4,
|
||||
.valid.max_access_size = 4,
|
||||
.endianness = DEVICE_LITTLE_ENDIAN,
|
||||
};
|
||||
|
||||
static void fusbh200_ehci_init(Object *obj)
|
||||
{
|
||||
EHCISysBusState *i = SYS_BUS_EHCI(obj);
|
||||
FUSBH200EHCIState *f = FUSBH200_EHCI(obj);
|
||||
EHCIState *s = &i->ehci;
|
||||
|
||||
memory_region_init_io(&f->mem_vendor, &fusbh200_ehci_mmio_ops, s,
|
||||
"fusbh200", 0x4c);
|
||||
memory_region_add_subregion(&s->mem,
|
||||
s->opregbase + s->portscbase + 4 * s->portnr,
|
||||
&f->mem_vendor);
|
||||
}
|
||||
|
||||
static void fusbh200_ehci_class_init(ObjectClass *oc, void *data)
|
||||
{
|
||||
SysBusEHCIClass *sec = SYS_BUS_EHCI_CLASS(oc);
|
||||
|
||||
sec->capsbase = 0x0;
|
||||
sec->opregbase = 0x10;
|
||||
sec->portscbase = 0x20;
|
||||
sec->portnr = 1;
|
||||
}
|
||||
|
||||
static const TypeInfo ehci_fusbh200_type_info = {
|
||||
.name = TYPE_FUSBH200_EHCI,
|
||||
.parent = TYPE_SYS_BUS_EHCI,
|
||||
.instance_size = sizeof(FUSBH200EHCIState),
|
||||
.instance_init = fusbh200_ehci_init,
|
||||
.class_init = fusbh200_ehci_class_init,
|
||||
};
|
||||
|
||||
static void ehci_sysbus_register_types(void)
|
||||
{
|
||||
type_register_static(&ehci_type_info);
|
||||
type_register_static(&ehci_xlnx_type_info);
|
||||
type_register_static(&ehci_exynos4210_type_info);
|
||||
type_register_static(&ehci_tegra2_type_info);
|
||||
type_register_static(&ehci_fusbh200_type_info);
|
||||
}
|
||||
|
||||
type_init(ehci_sysbus_register_types)
|
||||
|
|
|
@ -995,7 +995,7 @@ static uint64_t ehci_port_read(void *ptr, hwaddr addr,
|
|||
uint32_t val;
|
||||
|
||||
val = s->portsc[addr >> 2];
|
||||
trace_usb_ehci_portsc_read(addr + PORTSC_BEGIN, addr >> 2, val);
|
||||
trace_usb_ehci_portsc_read(addr + s->portscbase, addr >> 2, val);
|
||||
return val;
|
||||
}
|
||||
|
||||
|
@ -1036,7 +1036,7 @@ static void ehci_port_write(void *ptr, hwaddr addr,
|
|||
uint32_t old = *portsc;
|
||||
USBDevice *dev = s->ports[port].dev;
|
||||
|
||||
trace_usb_ehci_portsc_write(addr + PORTSC_BEGIN, addr >> 2, val);
|
||||
trace_usb_ehci_portsc_write(addr + s->portscbase, addr >> 2, val);
|
||||
|
||||
/* Clear rwc bits */
|
||||
*portsc &= ~(val & PORTSC_RWC_MASK);
|
||||
|
@ -1069,7 +1069,7 @@ static void ehci_port_write(void *ptr, hwaddr addr,
|
|||
|
||||
*portsc &= ~PORTSC_RO_MASK;
|
||||
*portsc |= val;
|
||||
trace_usb_ehci_portsc_change(addr + PORTSC_BEGIN, addr >> 2, *portsc, old);
|
||||
trace_usb_ehci_portsc_change(addr + s->portscbase, addr >> 2, *portsc, old);
|
||||
}
|
||||
|
||||
static void ehci_opreg_write(void *ptr, hwaddr addr,
|
||||
|
@ -2508,25 +2508,18 @@ const VMStateDescription vmstate_ehci = {
|
|||
}
|
||||
};
|
||||
|
||||
void usb_ehci_initfn(EHCIState *s, DeviceState *dev)
|
||||
void usb_ehci_realize(EHCIState *s, DeviceState *dev, Error **errp)
|
||||
{
|
||||
int i;
|
||||
|
||||
/* 2.2 host controller interface version */
|
||||
s->caps[0x00] = (uint8_t)(s->opregbase - s->capsbase);
|
||||
s->caps[0x01] = 0x00;
|
||||
s->caps[0x02] = 0x00;
|
||||
s->caps[0x03] = 0x01; /* HC version */
|
||||
s->caps[0x04] = NB_PORTS; /* Number of downstream ports */
|
||||
s->caps[0x05] = 0x00; /* No companion ports at present */
|
||||
s->caps[0x06] = 0x00;
|
||||
s->caps[0x07] = 0x00;
|
||||
s->caps[0x08] = 0x80; /* We can cache whole frame, no 64-bit */
|
||||
s->caps[0x0a] = 0x00;
|
||||
s->caps[0x0b] = 0x00;
|
||||
if (s->portnr > NB_PORTS) {
|
||||
error_setg(errp, "Too many ports! Max. port number is %d.",
|
||||
NB_PORTS);
|
||||
return;
|
||||
}
|
||||
|
||||
usb_bus_new(&s->bus, &ehci_bus_ops, dev);
|
||||
for(i = 0; i < NB_PORTS; i++) {
|
||||
for (i = 0; i < s->portnr; i++) {
|
||||
usb_register_port(&s->bus, &s->ports[i], s, i, &ehci_port_ops,
|
||||
USB_SPEED_MASK_HIGH);
|
||||
s->ports[i].dev = 0;
|
||||
|
@ -2534,24 +2527,41 @@ void usb_ehci_initfn(EHCIState *s, DeviceState *dev)
|
|||
|
||||
s->frame_timer = qemu_new_timer_ns(vm_clock, ehci_frame_timer, s);
|
||||
s->async_bh = qemu_bh_new(ehci_frame_timer, s);
|
||||
QTAILQ_INIT(&s->aqueues);
|
||||
QTAILQ_INIT(&s->pqueues);
|
||||
usb_packet_init(&s->ipacket);
|
||||
|
||||
qemu_register_reset(ehci_reset, s);
|
||||
qemu_add_vm_change_state_handler(usb_ehci_vm_state_change, s);
|
||||
}
|
||||
|
||||
void usb_ehci_init(EHCIState *s, DeviceState *dev)
|
||||
{
|
||||
/* 2.2 host controller interface version */
|
||||
s->caps[0x00] = (uint8_t)(s->opregbase - s->capsbase);
|
||||
s->caps[0x01] = 0x00;
|
||||
s->caps[0x02] = 0x00;
|
||||
s->caps[0x03] = 0x01; /* HC version */
|
||||
s->caps[0x04] = s->portnr; /* Number of downstream ports */
|
||||
s->caps[0x05] = 0x00; /* No companion ports at present */
|
||||
s->caps[0x06] = 0x00;
|
||||
s->caps[0x07] = 0x00;
|
||||
s->caps[0x08] = 0x80; /* We can cache whole frame, no 64-bit */
|
||||
s->caps[0x0a] = 0x00;
|
||||
s->caps[0x0b] = 0x00;
|
||||
|
||||
QTAILQ_INIT(&s->aqueues);
|
||||
QTAILQ_INIT(&s->pqueues);
|
||||
usb_packet_init(&s->ipacket);
|
||||
|
||||
memory_region_init(&s->mem, "ehci", MMIO_SIZE);
|
||||
memory_region_init_io(&s->mem_caps, &ehci_mmio_caps_ops, s,
|
||||
"capabilities", CAPA_SIZE);
|
||||
memory_region_init_io(&s->mem_opreg, &ehci_mmio_opreg_ops, s,
|
||||
"operational", PORTSC_BEGIN);
|
||||
"operational", s->portscbase);
|
||||
memory_region_init_io(&s->mem_ports, &ehci_mmio_port_ops, s,
|
||||
"ports", PORTSC_END - PORTSC_BEGIN);
|
||||
"ports", 4 * s->portnr);
|
||||
|
||||
memory_region_add_subregion(&s->mem, s->capsbase, &s->mem_caps);
|
||||
memory_region_add_subregion(&s->mem, s->opregbase, &s->mem_opreg);
|
||||
memory_region_add_subregion(&s->mem, s->opregbase + PORTSC_BEGIN,
|
||||
memory_region_add_subregion(&s->mem, s->opregbase + s->portscbase,
|
||||
&s->mem_ports);
|
||||
}
|
||||
|
||||
|
|
|
@ -40,11 +40,7 @@
|
|||
#define MMIO_SIZE 0x1000
|
||||
#define CAPA_SIZE 0x10
|
||||
|
||||
#define PORTSC 0x0044
|
||||
#define PORTSC_BEGIN PORTSC
|
||||
#define PORTSC_END (PORTSC + 4 * NB_PORTS)
|
||||
|
||||
#define NB_PORTS 6 /* Number of downstream ports */
|
||||
#define NB_PORTS 6 /* Max. Number of downstream ports */
|
||||
|
||||
typedef struct EHCIPacket EHCIPacket;
|
||||
typedef struct EHCIQueue EHCIQueue;
|
||||
|
@ -268,6 +264,8 @@ struct EHCIState {
|
|||
int companion_count;
|
||||
uint16_t capsbase;
|
||||
uint16_t opregbase;
|
||||
uint16_t portscbase;
|
||||
uint16_t portnr;
|
||||
|
||||
/* properties */
|
||||
uint32_t maxframes;
|
||||
|
@ -278,7 +276,7 @@ struct EHCIState {
|
|||
*/
|
||||
uint8_t caps[CAPA_SIZE];
|
||||
union {
|
||||
uint32_t opreg[PORTSC_BEGIN/sizeof(uint32_t)];
|
||||
uint32_t opreg[0x44/sizeof(uint32_t)];
|
||||
struct {
|
||||
uint32_t usbcmd;
|
||||
uint32_t usbsts;
|
||||
|
@ -322,7 +320,8 @@ struct EHCIState {
|
|||
|
||||
extern const VMStateDescription vmstate_ehci;
|
||||
|
||||
void usb_ehci_initfn(EHCIState *s, DeviceState *dev);
|
||||
void usb_ehci_init(EHCIState *s, DeviceState *dev);
|
||||
void usb_ehci_realize(EHCIState *s, DeviceState *dev, Error **errp);
|
||||
|
||||
#define TYPE_PCI_EHCI "pci-ehci-usb"
|
||||
#define PCI_EHCI(obj) OBJECT_CHECK(EHCIPCIState, (obj), TYPE_PCI_EHCI)
|
||||
|
@ -338,6 +337,8 @@ typedef struct EHCIPCIState {
|
|||
|
||||
#define TYPE_SYS_BUS_EHCI "sysbus-ehci-usb"
|
||||
#define TYPE_EXYNOS4210_EHCI "exynos4210-ehci-usb"
|
||||
#define TYPE_TEGRA2_EHCI "tegra2-ehci-usb"
|
||||
#define TYPE_FUSBH200_EHCI "fusbh200-ehci-usb"
|
||||
|
||||
#define SYS_BUS_EHCI(obj) \
|
||||
OBJECT_CHECK(EHCISysBusState, (obj), TYPE_SYS_BUS_EHCI)
|
||||
|
@ -361,6 +362,19 @@ typedef struct SysBusEHCIClass {
|
|||
|
||||
uint16_t capsbase;
|
||||
uint16_t opregbase;
|
||||
uint16_t portscbase;
|
||||
uint16_t portnr;
|
||||
} SysBusEHCIClass;
|
||||
|
||||
#define FUSBH200_EHCI(obj) \
|
||||
OBJECT_CHECK(FUSBH200EHCIState, (obj), TYPE_FUSBH200_EHCI)
|
||||
|
||||
typedef struct FUSBH200EHCIState {
|
||||
/*< private >*/
|
||||
EHCISysBusState parent_obj;
|
||||
/*< public >*/
|
||||
|
||||
MemoryRegion mem_vendor;
|
||||
} FUSBH200EHCIState;
|
||||
|
||||
#endif
|
||||
|
|
|
@ -241,7 +241,11 @@ static int usb_host_get_port(libusb_device *dev, char *port, size_t len)
|
|||
size_t off;
|
||||
int rc, i;
|
||||
|
||||
#if LIBUSBX_API_VERSION >= 0x01000102
|
||||
rc = libusb_get_port_numbers(dev, path, 7);
|
||||
#else
|
||||
rc = libusb_get_port_path(ctx, dev, path, 7);
|
||||
#endif
|
||||
if (rc < 0) {
|
||||
return 0;
|
||||
}
|
||||
|
@ -891,6 +895,7 @@ static int usb_host_initfn(USBDevice *udev)
|
|||
USBHostDevice *s = USB_HOST_DEVICE(udev);
|
||||
|
||||
loglevel = s->loglevel;
|
||||
udev->flags |= (1 << USB_DEV_FLAG_IS_HOST);
|
||||
udev->auto_attach = 0;
|
||||
QTAILQ_INIT(&s->requests);
|
||||
QTAILQ_INIT(&s->isorings);
|
||||
|
|
|
@ -205,6 +205,7 @@ struct USBDevice {
|
|||
DeviceState qdev;
|
||||
USBPort *port;
|
||||
char *port_path;
|
||||
char *serial;
|
||||
void *opaque;
|
||||
uint32_t flags;
|
||||
|
||||
|
|
Loading…
Reference in New Issue