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",\
|
.driver = "virtio-serial-pci",\
|
||||||
.property = "vectors",\
|
.property = "vectors",\
|
||||||
.value = stringify(0),\
|
.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 = {
|
static QEMUMachine pc_machine_v0_12 = {
|
||||||
|
@ -13,6 +13,7 @@ static int usb_qdev_exit(DeviceState *qdev);
|
|||||||
|
|
||||||
static Property usb_props[] = {
|
static Property usb_props[] = {
|
||||||
DEFINE_PROP_STRING("port", USBDevice, port_path),
|
DEFINE_PROP_STRING("port", USBDevice, port_path),
|
||||||
|
DEFINE_PROP_STRING("serial", USBDevice, serial),
|
||||||
DEFINE_PROP_BIT("full-path", USBDevice, flags,
|
DEFINE_PROP_BIT("full-path", USBDevice, flags,
|
||||||
USB_DEV_FLAG_FULL_PATH, true),
|
USB_DEV_FLAG_FULL_PATH, true),
|
||||||
DEFINE_PROP_END_OF_LIST()
|
DEFINE_PROP_END_OF_LIST()
|
||||||
|
@ -566,6 +566,12 @@ void usb_desc_create_serial(USBDevice *dev)
|
|||||||
char *path;
|
char *path;
|
||||||
int dst;
|
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);
|
assert(index != 0 && desc->str[index] != NULL);
|
||||||
dst = snprintf(serial, sizeof(serial), "%s", desc->str[index]);
|
dst = snprintf(serial, sizeof(serial), "%s", desc->str[index]);
|
||||||
path = qdev_get_dev_path(hcd);
|
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);
|
USBHIDState *us = DO_UPCAST(USBHIDState, dev, dev);
|
||||||
|
|
||||||
|
if (dev->serial) {
|
||||||
|
usb_desc_set_string(dev, STR_SERIALNUMBER, dev->serial);
|
||||||
|
}
|
||||||
usb_desc_init(dev);
|
usb_desc_init(dev);
|
||||||
us->intr = usb_ep_get(dev, USB_TOKEN_IN, 1);
|
us->intr = usb_ep_get(dev, USB_TOKEN_IN, 1);
|
||||||
hid_init(&us->hid, kind, usb_hid_changed);
|
hid_init(&us->hid, kind, usb_hid_changed);
|
||||||
|
@ -58,7 +58,6 @@ typedef struct {
|
|||||||
USBPacket *packet;
|
USBPacket *packet;
|
||||||
/* usb-storage only */
|
/* usb-storage only */
|
||||||
BlockConf conf;
|
BlockConf conf;
|
||||||
char *serial;
|
|
||||||
uint32_t removable;
|
uint32_t removable;
|
||||||
} MSDState;
|
} MSDState;
|
||||||
|
|
||||||
@ -602,7 +601,7 @@ static int usb_msd_initfn_storage(USBDevice *dev)
|
|||||||
return -1;
|
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
|
* 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);
|
bdrv_detach_dev(bs, &s->dev.qdev);
|
||||||
s->conf.bs = NULL;
|
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);
|
usb_desc_init(dev);
|
||||||
scsi_bus_new(&s->bus, &s->dev.qdev, &usb_msd_scsi_info_storage, NULL);
|
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,
|
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) {
|
if (!scsi_dev) {
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
@ -734,7 +728,6 @@ static const VMStateDescription vmstate_usb_msd = {
|
|||||||
|
|
||||||
static Property msd_properties[] = {
|
static Property msd_properties[] = {
|
||||||
DEFINE_BLOCK_PROPERTIES(MSDState, conf),
|
DEFINE_BLOCK_PROPERTIES(MSDState, conf),
|
||||||
DEFINE_PROP_STRING("serial", MSDState, serial),
|
|
||||||
DEFINE_PROP_BIT("removable", MSDState, removable, 0, false),
|
DEFINE_PROP_BIT("removable", MSDState, removable, 0, false),
|
||||||
DEFINE_PROP_END_OF_LIST(),
|
DEFINE_PROP_END_OF_LIST(),
|
||||||
};
|
};
|
||||||
|
@ -60,20 +60,30 @@ static int usb_ehci_pci_initfn(PCIDevice *dev)
|
|||||||
pci_conf[0x6e] = 0x00;
|
pci_conf[0x6e] = 0x00;
|
||||||
pci_conf[0x6f] = 0xc0; /* USBLEFCTLSTS */
|
pci_conf[0x6f] = 0xc0; /* USBLEFCTLSTS */
|
||||||
|
|
||||||
s->caps[0x09] = 0x68; /* EECP */
|
|
||||||
|
|
||||||
s->irq = dev->irq[3];
|
s->irq = dev->irq[3];
|
||||||
s->as = pci_get_address_space(dev);
|
s->as = pci_get_address_space(dev);
|
||||||
|
|
||||||
s->capsbase = 0x00;
|
usb_ehci_realize(s, DEVICE(dev), NULL);
|
||||||
s->opregbase = 0x20;
|
|
||||||
|
|
||||||
usb_ehci_initfn(s, DEVICE(dev));
|
|
||||||
pci_register_bar(dev, 0, PCI_BASE_ADDRESS_SPACE_MEMORY, &s->mem);
|
pci_register_bar(dev, 0, PCI_BASE_ADDRESS_SPACE_MEMORY, &s->mem);
|
||||||
|
|
||||||
return 0;
|
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,
|
static void usb_ehci_pci_write_config(PCIDevice *dev, uint32_t addr,
|
||||||
uint32_t val, int l)
|
uint32_t val, int l)
|
||||||
{
|
{
|
||||||
@ -122,6 +132,7 @@ static const TypeInfo ehci_pci_type_info = {
|
|||||||
.name = TYPE_PCI_EHCI,
|
.name = TYPE_PCI_EHCI,
|
||||||
.parent = TYPE_PCI_DEVICE,
|
.parent = TYPE_PCI_DEVICE,
|
||||||
.instance_size = sizeof(EHCIPCIState),
|
.instance_size = sizeof(EHCIPCIState),
|
||||||
|
.instance_init = usb_ehci_pci_init,
|
||||||
.abstract = true,
|
.abstract = true,
|
||||||
.class_init = ehci_class_init,
|
.class_init = ehci_class_init,
|
||||||
};
|
};
|
||||||
|
@ -32,28 +32,42 @@ static Property ehci_sysbus_properties[] = {
|
|||||||
DEFINE_PROP_END_OF_LIST(),
|
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);
|
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;
|
EHCIState *s = &i->ehci;
|
||||||
|
|
||||||
s->capsbase = sec->capsbase;
|
s->capsbase = sec->capsbase;
|
||||||
s->opregbase = sec->opregbase;
|
s->opregbase = sec->opregbase;
|
||||||
|
s->portscbase = sec->portscbase;
|
||||||
|
s->portnr = sec->portnr;
|
||||||
s->as = &address_space_memory;
|
s->as = &address_space_memory;
|
||||||
|
|
||||||
usb_ehci_initfn(s, DEVICE(dev));
|
usb_ehci_init(s, DEVICE(obj));
|
||||||
sysbus_init_irq(dev, &s->irq);
|
sysbus_init_mmio(d, &s->mem);
|
||||||
sysbus_init_mmio(dev, &s->mem);
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ehci_sysbus_class_init(ObjectClass *klass, void *data)
|
static void ehci_sysbus_class_init(ObjectClass *klass, void *data)
|
||||||
{
|
{
|
||||||
DeviceClass *dc = DEVICE_CLASS(klass);
|
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->vmsd = &vmstate_ehci_sysbus;
|
||||||
dc->props = ehci_sysbus_properties;
|
dc->props = ehci_sysbus_properties;
|
||||||
}
|
}
|
||||||
@ -62,6 +76,7 @@ static const TypeInfo ehci_type_info = {
|
|||||||
.name = TYPE_SYS_BUS_EHCI,
|
.name = TYPE_SYS_BUS_EHCI,
|
||||||
.parent = TYPE_SYS_BUS_DEVICE,
|
.parent = TYPE_SYS_BUS_DEVICE,
|
||||||
.instance_size = sizeof(EHCISysBusState),
|
.instance_size = sizeof(EHCISysBusState),
|
||||||
|
.instance_init = ehci_sysbus_init,
|
||||||
.abstract = true,
|
.abstract = true,
|
||||||
.class_init = ehci_sysbus_class_init,
|
.class_init = ehci_sysbus_class_init,
|
||||||
.class_size = sizeof(SysBusEHCIClass),
|
.class_size = sizeof(SysBusEHCIClass),
|
||||||
@ -95,11 +110,101 @@ static const TypeInfo ehci_exynos4210_type_info = {
|
|||||||
.class_init = ehci_exynos4210_class_init,
|
.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)
|
static void ehci_sysbus_register_types(void)
|
||||||
{
|
{
|
||||||
type_register_static(&ehci_type_info);
|
type_register_static(&ehci_type_info);
|
||||||
type_register_static(&ehci_xlnx_type_info);
|
type_register_static(&ehci_xlnx_type_info);
|
||||||
type_register_static(&ehci_exynos4210_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)
|
type_init(ehci_sysbus_register_types)
|
||||||
|
@ -995,7 +995,7 @@ static uint64_t ehci_port_read(void *ptr, hwaddr addr,
|
|||||||
uint32_t val;
|
uint32_t val;
|
||||||
|
|
||||||
val = s->portsc[addr >> 2];
|
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;
|
return val;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1036,7 +1036,7 @@ static void ehci_port_write(void *ptr, hwaddr addr,
|
|||||||
uint32_t old = *portsc;
|
uint32_t old = *portsc;
|
||||||
USBDevice *dev = s->ports[port].dev;
|
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 */
|
/* Clear rwc bits */
|
||||||
*portsc &= ~(val & PORTSC_RWC_MASK);
|
*portsc &= ~(val & PORTSC_RWC_MASK);
|
||||||
@ -1069,7 +1069,7 @@ static void ehci_port_write(void *ptr, hwaddr addr,
|
|||||||
|
|
||||||
*portsc &= ~PORTSC_RO_MASK;
|
*portsc &= ~PORTSC_RO_MASK;
|
||||||
*portsc |= val;
|
*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,
|
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;
|
int i;
|
||||||
|
|
||||||
/* 2.2 host controller interface version */
|
if (s->portnr > NB_PORTS) {
|
||||||
s->caps[0x00] = (uint8_t)(s->opregbase - s->capsbase);
|
error_setg(errp, "Too many ports! Max. port number is %d.",
|
||||||
s->caps[0x01] = 0x00;
|
NB_PORTS);
|
||||||
s->caps[0x02] = 0x00;
|
return;
|
||||||
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;
|
|
||||||
|
|
||||||
usb_bus_new(&s->bus, &ehci_bus_ops, dev);
|
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_register_port(&s->bus, &s->ports[i], s, i, &ehci_port_ops,
|
||||||
USB_SPEED_MASK_HIGH);
|
USB_SPEED_MASK_HIGH);
|
||||||
s->ports[i].dev = 0;
|
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->frame_timer = qemu_new_timer_ns(vm_clock, ehci_frame_timer, s);
|
||||||
s->async_bh = qemu_bh_new(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_register_reset(ehci_reset, s);
|
||||||
qemu_add_vm_change_state_handler(usb_ehci_vm_state_change, 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(&s->mem, "ehci", MMIO_SIZE);
|
||||||
memory_region_init_io(&s->mem_caps, &ehci_mmio_caps_ops, s,
|
memory_region_init_io(&s->mem_caps, &ehci_mmio_caps_ops, s,
|
||||||
"capabilities", CAPA_SIZE);
|
"capabilities", CAPA_SIZE);
|
||||||
memory_region_init_io(&s->mem_opreg, &ehci_mmio_opreg_ops, s,
|
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,
|
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->capsbase, &s->mem_caps);
|
||||||
memory_region_add_subregion(&s->mem, s->opregbase, &s->mem_opreg);
|
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);
|
&s->mem_ports);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -40,11 +40,7 @@
|
|||||||
#define MMIO_SIZE 0x1000
|
#define MMIO_SIZE 0x1000
|
||||||
#define CAPA_SIZE 0x10
|
#define CAPA_SIZE 0x10
|
||||||
|
|
||||||
#define PORTSC 0x0044
|
#define NB_PORTS 6 /* Max. Number of downstream ports */
|
||||||
#define PORTSC_BEGIN PORTSC
|
|
||||||
#define PORTSC_END (PORTSC + 4 * NB_PORTS)
|
|
||||||
|
|
||||||
#define NB_PORTS 6 /* Number of downstream ports */
|
|
||||||
|
|
||||||
typedef struct EHCIPacket EHCIPacket;
|
typedef struct EHCIPacket EHCIPacket;
|
||||||
typedef struct EHCIQueue EHCIQueue;
|
typedef struct EHCIQueue EHCIQueue;
|
||||||
@ -268,6 +264,8 @@ struct EHCIState {
|
|||||||
int companion_count;
|
int companion_count;
|
||||||
uint16_t capsbase;
|
uint16_t capsbase;
|
||||||
uint16_t opregbase;
|
uint16_t opregbase;
|
||||||
|
uint16_t portscbase;
|
||||||
|
uint16_t portnr;
|
||||||
|
|
||||||
/* properties */
|
/* properties */
|
||||||
uint32_t maxframes;
|
uint32_t maxframes;
|
||||||
@ -278,7 +276,7 @@ struct EHCIState {
|
|||||||
*/
|
*/
|
||||||
uint8_t caps[CAPA_SIZE];
|
uint8_t caps[CAPA_SIZE];
|
||||||
union {
|
union {
|
||||||
uint32_t opreg[PORTSC_BEGIN/sizeof(uint32_t)];
|
uint32_t opreg[0x44/sizeof(uint32_t)];
|
||||||
struct {
|
struct {
|
||||||
uint32_t usbcmd;
|
uint32_t usbcmd;
|
||||||
uint32_t usbsts;
|
uint32_t usbsts;
|
||||||
@ -322,7 +320,8 @@ struct EHCIState {
|
|||||||
|
|
||||||
extern const VMStateDescription vmstate_ehci;
|
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 TYPE_PCI_EHCI "pci-ehci-usb"
|
||||||
#define PCI_EHCI(obj) OBJECT_CHECK(EHCIPCIState, (obj), TYPE_PCI_EHCI)
|
#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_SYS_BUS_EHCI "sysbus-ehci-usb"
|
||||||
#define TYPE_EXYNOS4210_EHCI "exynos4210-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) \
|
#define SYS_BUS_EHCI(obj) \
|
||||||
OBJECT_CHECK(EHCISysBusState, (obj), TYPE_SYS_BUS_EHCI)
|
OBJECT_CHECK(EHCISysBusState, (obj), TYPE_SYS_BUS_EHCI)
|
||||||
@ -361,6 +362,19 @@ typedef struct SysBusEHCIClass {
|
|||||||
|
|
||||||
uint16_t capsbase;
|
uint16_t capsbase;
|
||||||
uint16_t opregbase;
|
uint16_t opregbase;
|
||||||
|
uint16_t portscbase;
|
||||||
|
uint16_t portnr;
|
||||||
} SysBusEHCIClass;
|
} 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
|
#endif
|
||||||
|
@ -241,7 +241,11 @@ static int usb_host_get_port(libusb_device *dev, char *port, size_t len)
|
|||||||
size_t off;
|
size_t off;
|
||||||
int rc, i;
|
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);
|
rc = libusb_get_port_path(ctx, dev, path, 7);
|
||||||
|
#endif
|
||||||
if (rc < 0) {
|
if (rc < 0) {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -891,6 +895,7 @@ static int usb_host_initfn(USBDevice *udev)
|
|||||||
USBHostDevice *s = USB_HOST_DEVICE(udev);
|
USBHostDevice *s = USB_HOST_DEVICE(udev);
|
||||||
|
|
||||||
loglevel = s->loglevel;
|
loglevel = s->loglevel;
|
||||||
|
udev->flags |= (1 << USB_DEV_FLAG_IS_HOST);
|
||||||
udev->auto_attach = 0;
|
udev->auto_attach = 0;
|
||||||
QTAILQ_INIT(&s->requests);
|
QTAILQ_INIT(&s->requests);
|
||||||
QTAILQ_INIT(&s->isorings);
|
QTAILQ_INIT(&s->isorings);
|
||||||
|
@ -205,6 +205,7 @@ struct USBDevice {
|
|||||||
DeviceState qdev;
|
DeviceState qdev;
|
||||||
USBPort *port;
|
USBPort *port;
|
||||||
char *port_path;
|
char *port_path;
|
||||||
|
char *serial;
|
||||||
void *opaque;
|
void *opaque;
|
||||||
uint32_t flags;
|
uint32_t flags;
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user