bus.c 19.8 KB
Newer Older
G
Gerd Hoffmann 已提交
1 2 3
#include "hw/hw.h"
#include "hw/usb.h"
#include "hw/qdev.h"
4
#include "sysemu/sysemu.h"
5
#include "monitor/monitor.h"
6
#include "trace.h"
7 8

static void usb_bus_dev_print(Monitor *mon, DeviceState *qdev, int indent);
9 10

static char *usb_get_dev_path(DeviceState *dev);
11
static char *usb_get_fw_dev_path(DeviceState *qdev);
12
static void usb_qdev_unrealize(DeviceState *qdev, Error **errp);
13

14 15
static Property usb_props[] = {
    DEFINE_PROP_STRING("port", USBDevice, port_path),
G
Gerd Hoffmann 已提交
16
    DEFINE_PROP_STRING("serial", USBDevice, serial),
17 18
    DEFINE_PROP_BIT("full-path", USBDevice, flags,
                    USB_DEV_FLAG_FULL_PATH, true),
19 20
    DEFINE_PROP_BIT("msos-desc", USBDevice, flags,
                    USB_DEV_FLAG_MSOS_DESC_ENABLE, true),
21 22 23
    DEFINE_PROP_END_OF_LIST()
};

24 25 26
static void usb_bus_class_init(ObjectClass *klass, void *data)
{
    BusClass *k = BUS_CLASS(klass);
27
    HotplugHandlerClass *hc = HOTPLUG_HANDLER_CLASS(klass);
28 29 30 31

    k->print_dev = usb_bus_dev_print;
    k->get_dev_path = usb_get_dev_path;
    k->get_fw_dev_path = usb_get_fw_dev_path;
32
    hc->unplug = qdev_simple_device_unplug_cb;
33 34 35 36 37 38 39
}

static const TypeInfo usb_bus_info = {
    .name = TYPE_USB_BUS,
    .parent = TYPE_BUS,
    .instance_size = sizeof(USBBus),
    .class_init = usb_bus_class_init,
40 41 42 43
    .interfaces = (InterfaceInfo[]) {
        { TYPE_HOTPLUG_HANDLER },
        { }
    }
44
};
45

46
static int next_usb_bus = 0;
B
Blue Swirl 已提交
47
static QTAILQ_HEAD(, USBBus) busses = QTAILQ_HEAD_INITIALIZER(busses);
48

49 50 51 52 53 54 55 56 57
static int usb_device_post_load(void *opaque, int version_id)
{
    USBDevice *dev = opaque;

    if (dev->state == USB_STATE_NOTATTACHED) {
        dev->attached = 0;
    } else {
        dev->attached = 1;
    }
58 59
    if (dev->setup_index < 0 ||
        dev->setup_len < 0 ||
60 61
        dev->setup_index > dev->setup_len ||
        dev->setup_len > sizeof(dev->data_buf)) {
62 63
        return -EINVAL;
    }
64 65 66
    return 0;
}

G
Gerd Hoffmann 已提交
67 68 69 70
const VMStateDescription vmstate_usb_device = {
    .name = "USBDevice",
    .version_id = 1,
    .minimum_version_id = 1,
71
    .post_load = usb_device_post_load,
72
    .fields = (VMStateField[]) {
G
Gerd Hoffmann 已提交
73 74 75 76 77 78 79 80 81 82 83
        VMSTATE_UINT8(addr, USBDevice),
        VMSTATE_INT32(state, USBDevice),
        VMSTATE_INT32(remote_wakeup, USBDevice),
        VMSTATE_INT32(setup_state, USBDevice),
        VMSTATE_INT32(setup_len, USBDevice),
        VMSTATE_INT32(setup_index, USBDevice),
        VMSTATE_UINT8_ARRAY(setup_buf, USBDevice, 8),
        VMSTATE_END_OF_LIST(),
    }
};

84 85
void usb_bus_new(USBBus *bus, size_t bus_size,
                 USBBusOps *ops, DeviceState *host)
86
{
87
    qbus_create_inplace(bus, bus_size, TYPE_USB_BUS, host, NULL);
88
    qbus_set_bus_hotplug_handler(BUS(bus), &error_abort);
89
    bus->ops = ops;
90
    bus->busnr = next_usb_bus++;
B
Blue Swirl 已提交
91 92 93
    QTAILQ_INIT(&bus->free);
    QTAILQ_INIT(&bus->used);
    QTAILQ_INSERT_TAIL(&busses, bus, next);
94 95
}

G
Gonglei 已提交
96 97 98 99 100 101 102
void usb_bus_release(USBBus *bus)
{
    assert(next_usb_bus > 0);

    QTAILQ_REMOVE(&busses, bus, next);
}

103 104 105 106 107
USBBus *usb_bus_find(int busnr)
{
    USBBus *bus;

    if (-1 == busnr)
B
Blue Swirl 已提交
108 109
        return QTAILQ_FIRST(&busses);
    QTAILQ_FOREACH(bus, &busses, next) {
110 111 112 113 114 115
        if (bus->busnr == busnr)
            return bus;
    }
    return NULL;
}

116
static void usb_device_realize(USBDevice *dev, Error **errp)
117 118
{
    USBDeviceClass *klass = USB_DEVICE_GET_CLASS(dev);
119 120 121

    if (klass->realize) {
        klass->realize(dev, errp);
122 123 124
    }
}

G
Gerd Hoffmann 已提交
125
USBDevice *usb_device_find_device(USBDevice *dev, uint8_t addr)
126 127
{
    USBDeviceClass *klass = USB_DEVICE_GET_CLASS(dev);
G
Gerd Hoffmann 已提交
128 129
    if (klass->find_device) {
        return klass->find_device(dev, addr);
130
    }
G
Gerd Hoffmann 已提交
131
    return NULL;
132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165
}

static void usb_device_handle_destroy(USBDevice *dev)
{
    USBDeviceClass *klass = USB_DEVICE_GET_CLASS(dev);
    if (klass->handle_destroy) {
        klass->handle_destroy(dev);
    }
}

void usb_device_cancel_packet(USBDevice *dev, USBPacket *p)
{
    USBDeviceClass *klass = USB_DEVICE_GET_CLASS(dev);
    if (klass->cancel_packet) {
        klass->cancel_packet(dev, p);
    }
}

void usb_device_handle_attach(USBDevice *dev)
{
    USBDeviceClass *klass = USB_DEVICE_GET_CLASS(dev);
    if (klass->handle_attach) {
        klass->handle_attach(dev);
    }
}

void usb_device_handle_reset(USBDevice *dev)
{
    USBDeviceClass *klass = USB_DEVICE_GET_CLASS(dev);
    if (klass->handle_reset) {
        klass->handle_reset(dev);
    }
}

166 167
void usb_device_handle_control(USBDevice *dev, USBPacket *p, int request,
                               int value, int index, int length, uint8_t *data)
168 169 170
{
    USBDeviceClass *klass = USB_DEVICE_GET_CLASS(dev);
    if (klass->handle_control) {
171
        klass->handle_control(dev, p, request, value, index, length, data);
172 173 174
    }
}

175
void usb_device_handle_data(USBDevice *dev, USBPacket *p)
176 177 178
{
    USBDeviceClass *klass = USB_DEVICE_GET_CLASS(dev);
    if (klass->handle_data) {
179
        klass->handle_data(dev, p);
180 181 182 183 184 185 186 187 188 189 190 191
    }
}

const char *usb_device_get_product_desc(USBDevice *dev)
{
    USBDeviceClass *klass = USB_DEVICE_GET_CLASS(dev);
    return klass->product_desc;
}

const USBDesc *usb_device_get_usb_desc(USBDevice *dev)
{
    USBDeviceClass *klass = USB_DEVICE_GET_CLASS(dev);
192 193 194
    if (dev->usb_desc) {
        return dev->usb_desc;
    }
195 196 197 198 199 200 201 202 203 204 205 206
    return klass->usb_desc;
}

void usb_device_set_interface(USBDevice *dev, int interface,
                              int alt_old, int alt_new)
{
    USBDeviceClass *klass = USB_DEVICE_GET_CLASS(dev);
    if (klass->set_interface) {
        klass->set_interface(dev, interface, alt_old, alt_new);
    }
}

207 208 209 210 211 212 213 214
void usb_device_flush_ep_queue(USBDevice *dev, USBEndpoint *ep)
{
    USBDeviceClass *klass = USB_DEVICE_GET_CLASS(dev);
    if (klass->flush_ep_queue) {
        klass->flush_ep_queue(dev, ep);
    }
}

215 216 217 218 219 220 221 222
void usb_device_ep_stopped(USBDevice *dev, USBEndpoint *ep)
{
    USBDeviceClass *klass = USB_DEVICE_GET_CLASS(dev);
    if (klass->ep_stopped) {
        klass->ep_stopped(dev, ep);
    }
}

223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240
int usb_device_alloc_streams(USBDevice *dev, USBEndpoint **eps, int nr_eps,
                             int streams)
{
    USBDeviceClass *klass = USB_DEVICE_GET_CLASS(dev);
    if (klass->alloc_streams) {
        return klass->alloc_streams(dev, eps, nr_eps, streams);
    }
    return 0;
}

void usb_device_free_streams(USBDevice *dev, USBEndpoint **eps, int nr_eps)
{
    USBDeviceClass *klass = USB_DEVICE_GET_CLASS(dev);
    if (klass->free_streams) {
        klass->free_streams(dev, eps, nr_eps);
    }
}

241
static void usb_qdev_realize(DeviceState *qdev, Error **errp)
242
{
243
    USBDevice *dev = USB_DEVICE(qdev);
244
    Error *local_err = NULL;
245

246 247
    pstrcpy(dev->product_desc, sizeof(dev->product_desc),
            usb_device_get_product_desc(dev));
G
Gerd Hoffmann 已提交
248
    dev->auto_attach = 1;
249
    QLIST_INIT(&dev->strings);
G
Gerd Hoffmann 已提交
250
    usb_ep_init(dev);
251 252 253 254 255

    usb_claim_port(dev, &local_err);
    if (local_err) {
        error_propagate(errp, local_err);
        return;
256
    }
257 258 259

    usb_device_realize(dev, &local_err);
    if (local_err) {
260
        usb_release_port(dev);
261 262
        error_propagate(errp, local_err);
        return;
263
    }
264

265
    if (dev->auto_attach) {
266 267 268 269 270
        usb_device_attach(dev, &local_err);
        if (local_err) {
            usb_qdev_unrealize(qdev, NULL);
            error_propagate(errp, local_err);
            return;
271
        }
272
    }
273 274
}

275
static void usb_qdev_unrealize(DeviceState *qdev, Error **errp)
276
{
277
    USBDevice *dev = USB_DEVICE(qdev);
278

279 280 281
    if (dev->attached) {
        usb_device_detach(dev);
    }
282
    usb_device_handle_destroy(dev);
283 284 285
    if (dev->port) {
        usb_release_port(dev);
    }
286 287
}

288
typedef struct LegacyUSBFactory
289
{
290 291
    const char *name;
    const char *usbdevice_name;
292
    USBDevice *(*usbdevice_init)(USBBus *bus, const char *params);
293
} LegacyUSBFactory;
294

295 296
static GSList *legacy_usb_factory;

297
void usb_legacy_register(const char *typename, const char *usbdevice_name,
298 299
                         USBDevice *(*usbdevice_init)(USBBus *bus,
                                                      const char *params))
300
{
301 302
    if (usbdevice_name) {
        LegacyUSBFactory *f = g_malloc0(sizeof(*f));
303
        f->name = typename;
304 305 306
        f->usbdevice_name = usbdevice_name;
        f->usbdevice_init = usbdevice_init;
        legacy_usb_factory = g_slist_append(legacy_usb_factory, f);
307 308 309
    }
}

310
USBDevice *usb_create(USBBus *bus, const char *name)
311 312 313 314
{
    DeviceState *dev;

    dev = qdev_create(&bus->qbus, name);
315
    return USB_DEVICE(dev);
316
}
317

318 319
static USBDevice *usb_try_create_simple(USBBus *bus, const char *name,
                                        Error **errp)
320
{
321 322
    Error *err = NULL;
    USBDevice *dev;
323

324 325 326 327 328 329 330 331 332 333 334
    dev = USB_DEVICE(qdev_try_create(&bus->qbus, name));
    if (!dev) {
        error_setg(errp, "Failed to create USB device '%s'", name);
        return NULL;
    }
    object_property_set_bool(OBJECT(dev), true, "realized", &err);
    if (err) {
        error_setg(errp, "Failed to initialize USB device '%s': %s",
                   name, error_get_pretty(err));
        error_free(err);
        object_unparent(OBJECT(dev));
335
        return NULL;
P
Paul Brook 已提交
336
    }
337 338 339
    return dev;
}

340 341
USBDevice *usb_create_simple(USBBus *bus, const char *name)
{
342
    return usb_try_create_simple(bus, name, &error_abort);
343 344
}

345 346
static void usb_fill_port(USBPort *port, void *opaque, int index,
                          USBPortOps *ops, int speedmask)
347
{
348 349 350
    port->opaque = opaque;
    port->index = index;
    port->ops = ops;
G
Gerd Hoffmann 已提交
351
    port->speedmask = speedmask;
352
    usb_port_location(port, NULL, index + 1);
353 354 355 356 357 358
}

void usb_register_port(USBBus *bus, USBPort *port, void *opaque, int index,
                       USBPortOps *ops, int speedmask)
{
    usb_fill_port(port, opaque, index, ops, speedmask);
B
Blue Swirl 已提交
359
    QTAILQ_INSERT_TAIL(&bus->free, port, next);
360 361 362
    bus->nfree++;
}

363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393
int usb_register_companion(const char *masterbus, USBPort *ports[],
                           uint32_t portcount, uint32_t firstport,
                           void *opaque, USBPortOps *ops, int speedmask)
{
    USBBus *bus;
    int i;

    QTAILQ_FOREACH(bus, &busses, next) {
        if (strcmp(bus->qbus.name, masterbus) == 0) {
            break;
        }
    }

    if (!bus || !bus->ops->register_companion) {
        qerror_report(QERR_INVALID_PARAMETER_VALUE, "masterbus",
                      "an USB masterbus");
        if (bus) {
            error_printf_unless_qmp(
                "USB bus '%s' does not allow companion controllers\n",
                masterbus);
        }
        return -1;
    }

    for (i = 0; i < portcount; i++) {
        usb_fill_port(ports[i], opaque, i, ops, speedmask);
    }

    return bus->ops->register_companion(bus, ports, portcount, firstport);
}

394 395 396 397 398
void usb_port_location(USBPort *downstream, USBPort *upstream, int portnr)
{
    if (upstream) {
        snprintf(downstream->path, sizeof(downstream->path), "%s.%d",
                 upstream->path, portnr);
G
Gerd Hoffmann 已提交
399
        downstream->hubcount = upstream->hubcount + 1;
400 401
    } else {
        snprintf(downstream->path, sizeof(downstream->path), "%d", portnr);
G
Gerd Hoffmann 已提交
402
        downstream->hubcount = 0;
403 404 405
    }
}

406 407
void usb_unregister_port(USBBus *bus, USBPort *port)
{
408 409 410
    if (port->dev) {
        object_unparent(OBJECT(port->dev));
    }
411 412 413 414
    QTAILQ_REMOVE(&bus->free, port, next);
    bus->nfree--;
}

415
void usb_claim_port(USBDevice *dev, Error **errp)
416 417 418 419
{
    USBBus *bus = usb_bus_from_device(dev);
    USBPort *port;

420 421
    assert(dev->port == NULL);

G
Gerd Hoffmann 已提交
422 423 424 425 426 427 428
    if (dev->port_path) {
        QTAILQ_FOREACH(port, &bus->free, next) {
            if (strcmp(port->path, dev->port_path) == 0) {
                break;
            }
        }
        if (port == NULL) {
429
            error_setg(errp, "usb port %s (bus %s) not found (in use?)",
430 431
                       dev->port_path, bus->qbus.name);
            return;
G
Gerd Hoffmann 已提交
432 433
        }
    } else {
434
        if (bus->nfree == 1 && strcmp(object_get_typename(OBJECT(dev)), "usb-hub") != 0) {
435
            /* Create a new hub and chain it on */
436
            usb_try_create_simple(bus, "usb-hub", NULL);
437 438
        }
        if (bus->nfree == 0) {
439
            error_setg(errp, "tried to attach usb device %s to a bus "
440 441
                       "with no free ports", dev->product_desc);
            return;
442
        }
G
Gerd Hoffmann 已提交
443 444
        port = QTAILQ_FIRST(&bus->free);
    }
445
    trace_usb_port_claim(bus->busnr, port->path);
446

B
Blue Swirl 已提交
447
    QTAILQ_REMOVE(&bus->free, port, next);
448 449
    bus->nfree--;

450 451
    dev->port = port;
    port->dev = dev;
452

B
Blue Swirl 已提交
453
    QTAILQ_INSERT_TAIL(&bus->used, port, next);
454 455 456
    bus->nused++;
}

457
void usb_release_port(USBDevice *dev)
458 459
{
    USBBus *bus = usb_bus_from_device(dev);
460
    USBPort *port = dev->port;
461

462 463 464 465 466 467 468 469 470 471 472
    assert(port != NULL);
    trace_usb_port_release(bus->busnr, port->path);

    QTAILQ_REMOVE(&bus->used, port, next);
    bus->nused--;

    dev->port = NULL;
    port->dev = NULL;

    QTAILQ_INSERT_TAIL(&bus->free, port, next);
    bus->nfree++;
473 474
}

475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496
static void usb_mask_to_str(char *dest, size_t size,
                            unsigned int speedmask)
{
    static const struct {
        unsigned int mask;
        const char *name;
    } speeds[] = {
        { .mask = USB_SPEED_MASK_FULL,  .name = "full"  },
        { .mask = USB_SPEED_MASK_HIGH,  .name = "high"  },
        { .mask = USB_SPEED_MASK_SUPER, .name = "super" },
    };
    int i, pos = 0;

    for (i = 0; i < ARRAY_SIZE(speeds); i++) {
        if (speeds[i].mask & speedmask) {
            pos += snprintf(dest + pos, size - pos, "%s%s",
                            pos ? "+" : "",
                            speeds[i].name);
        }
    }
}

497
void usb_check_attach(USBDevice *dev, Error **errp)
498 499
{
    USBBus *bus = usb_bus_from_device(dev);
500
    USBPort *port = dev->port;
501
    char devspeed[32], portspeed[32];
502

503 504
    assert(port != NULL);
    assert(!dev->attached);
505 506 507 508
    usb_mask_to_str(devspeed, sizeof(devspeed), dev->speedmask);
    usb_mask_to_str(portspeed, sizeof(portspeed), port->speedmask);
    trace_usb_port_attach(bus->busnr, port->path,
                          devspeed, portspeed);
509 510

    if (!(port->speedmask & dev->speedmask)) {
511 512 513 514 515 516
        error_setg(errp, "Warning: speed mismatch trying to attach"
                   " usb device \"%s\" (%s speed)"
                   " to bus \"%s\", port \"%s\" (%s speed)",
                   dev->product_desc, devspeed,
                   bus->qbus.name, port->path, portspeed);
        return;
517
    }
518 519 520 521 522 523 524 525 526 527 528 529
}

void usb_device_attach(USBDevice *dev, Error **errp)
{
    USBPort *port = dev->port;
    Error *local_err = NULL;

    usb_check_attach(dev, &local_err);
    if (local_err) {
        error_propagate(errp, local_err);
        return;
    }
530

531 532 533 534 535 536 537 538
    dev->attached++;
    usb_attach(port);
}

int usb_device_detach(USBDevice *dev)
{
    USBBus *bus = usb_bus_from_device(dev);
    USBPort *port = dev->port;
539

540 541 542
    assert(port != NULL);
    assert(dev->attached);
    trace_usb_port_detach(bus->busnr, port->path);
543

544 545
    usb_detach(port);
    dev->attached--;
546 547 548
    return 0;
}

549 550 551 552 553 554 555 556 557 558
int usb_device_delete_addr(int busnr, int addr)
{
    USBBus *bus;
    USBPort *port;
    USBDevice *dev;

    bus = usb_bus_find(busnr);
    if (!bus)
        return -1;

B
Blue Swirl 已提交
559
    QTAILQ_FOREACH(port, &bus->used, next) {
560 561 562 563 564 565 566
        if (port->dev->addr == addr)
            break;
    }
    if (!port)
        return -1;
    dev = port->dev;

567
    object_unparent(OBJECT(dev));
568 569 570 571 572 573 574 575 576
    return 0;
}

static const char *usb_speed(unsigned int speed)
{
    static const char *txt[] = {
        [ USB_SPEED_LOW  ] = "1.5",
        [ USB_SPEED_FULL ] = "12",
        [ USB_SPEED_HIGH ] = "480",
577
        [ USB_SPEED_SUPER ] = "5000",
578 579 580 581 582 583 584 585
    };
    if (speed >= ARRAY_SIZE(txt))
        return "?";
    return txt[speed];
}

static void usb_bus_dev_print(Monitor *mon, DeviceState *qdev, int indent)
{
586
    USBDevice *dev = USB_DEVICE(qdev);
587 588
    USBBus *bus = usb_bus_from_device(dev);

589
    monitor_printf(mon, "%*saddr %d.%d, port %s, speed %s, name %s%s\n",
590
                   indent, "", bus->busnr, dev->addr,
591
                   dev->port ? dev->port->path : "-",
592
                   usb_speed(dev->speed), dev->product_desc,
593
                   dev->attached ? ", attached" : "");
594 595
}

596 597
static char *usb_get_dev_path(DeviceState *qdev)
{
598
    USBDevice *dev = USB_DEVICE(qdev);
G
Gerd Hoffmann 已提交
599 600 601
    DeviceState *hcd = qdev->parent_bus->parent;
    char *id = NULL;

602 603
    if (dev->flags & (1 << USB_DEV_FLAG_FULL_PATH)) {
        id = qdev_get_dev_path(hcd);
G
Gerd Hoffmann 已提交
604 605 606 607 608 609 610 611
    }
    if (id) {
        char *ret = g_strdup_printf("%s/%s", id, dev->port->path);
        g_free(id);
        return ret;
    } else {
        return g_strdup(dev->port->path);
    }
612 613
}

614 615
static char *usb_get_fw_dev_path(DeviceState *qdev)
{
616
    USBDevice *dev = USB_DEVICE(qdev);
617
    char *fw_path, *in;
B
Blue Swirl 已提交
618
    ssize_t pos = 0, fw_len;
619 620
    long nr;

B
Blue Swirl 已提交
621
    fw_len = 32 + strlen(dev->port->path) * 6;
622
    fw_path = g_malloc(fw_len);
623
    in = dev->port->path;
B
Blue Swirl 已提交
624
    while (fw_len - pos > 0) {
625 626 627
        nr = strtol(in, &in, 10);
        if (in[0] == '.') {
            /* some hub between root port and device */
628
            pos += snprintf(fw_path + pos, fw_len - pos, "hub@%lx/", nr);
629 630 631
            in++;
        } else {
            /* the device itself */
632
            pos += snprintf(fw_path + pos, fw_len - pos, "%s@%lx",
B
Blue Swirl 已提交
633
                            qdev_fw_name(qdev), nr);
634 635 636 637 638 639
            break;
        }
    }
    return fw_path;
}

640
void hmp_info_usb(Monitor *mon, const QDict *qdict)
641 642 643 644 645
{
    USBBus *bus;
    USBDevice *dev;
    USBPort *port;

B
Blue Swirl 已提交
646
    if (QTAILQ_EMPTY(&busses)) {
647 648 649 650
        monitor_printf(mon, "USB support not enabled\n");
        return;
    }

B
Blue Swirl 已提交
651 652
    QTAILQ_FOREACH(bus, &busses, next) {
        QTAILQ_FOREACH(port, &bus->used, next) {
653 654 655
            dev = port->dev;
            if (!dev)
                continue;
656 657
            monitor_printf(mon, "  Device %d.%d, Port %s, Speed %s Mb/s, Product %s\n",
                           bus->busnr, dev->addr, port->path, usb_speed(dev->speed),
658
                           dev->product_desc);
659 660 661 662
        }
    }
}

G
Gerd Hoffmann 已提交
663 664 665 666
/* handle legacy -usbdevice cmd line option */
USBDevice *usbdevice_create(const char *cmdline)
{
    USBBus *bus = usb_bus_find(-1 /* any */);
667
    LegacyUSBFactory *f = NULL;
668
    Error *err = NULL;
669
    GSList *i;
670 671
    char driver[32];
    const char *params;
G
Gerd Hoffmann 已提交
672
    int len;
673
    USBDevice *dev;
G
Gerd Hoffmann 已提交
674 675 676 677 678 679 680 681 682

    params = strchr(cmdline,':');
    if (params) {
        params++;
        len = params - cmdline;
        if (len > sizeof(driver))
            len = sizeof(driver);
        pstrcpy(driver, len, cmdline);
    } else {
683
        params = "";
G
Gerd Hoffmann 已提交
684 685 686
        pstrcpy(driver, sizeof(driver), cmdline);
    }

687 688 689 690 691
    for (i = legacy_usb_factory; i; i = i->next) {
        f = i->data;
        if (strcmp(f->usbdevice_name, driver) == 0) {
            break;
        }
G
Gerd Hoffmann 已提交
692
    }
693
    if (i == NULL) {
G
Gerd Hoffmann 已提交
694 695
#if 0
        /* no error because some drivers are not converted (yet) */
696
        error_report("usbdevice %s not found", driver);
G
Gerd Hoffmann 已提交
697 698 699 700
#endif
        return NULL;
    }

701 702 703 704 705 706 707
    if (!bus) {
        error_report("Error: no usb bus to attach usbdevice %s, "
                     "please try -machine usb=on and check that "
                     "the machine model supports USB", driver);
        return NULL;
    }

708 709 710
    if (f->usbdevice_init) {
        dev = f->usbdevice_init(bus, params);
    } else {
T
TeLeMan 已提交
711
        if (*params) {
712
            error_report("usbdevice %s accepts no params", driver);
G
Gerd Hoffmann 已提交
713 714
            return NULL;
        }
715 716 717 718 719
        dev = usb_create(bus, f->name);
    }
    if (!dev) {
        error_report("Failed to create USB device '%s'", f->name);
        return NULL;
G
Gerd Hoffmann 已提交
720
    }
721 722 723 724 725 726 727 728 729
    object_property_set_bool(OBJECT(dev), true, "realized", &err);
    if (err) {
        error_report("Failed to initialize USB device '%s': %s",
                     f->name, error_get_pretty(err));
        error_free(err);
        object_unparent(OBJECT(dev));
        return NULL;
    }
    return dev;
G
Gerd Hoffmann 已提交
730
}
731

732 733 734
static void usb_device_class_init(ObjectClass *klass, void *data)
{
    DeviceClass *k = DEVICE_CLASS(klass);
735
    k->bus_type = TYPE_USB_BUS;
736 737
    k->realize  = usb_qdev_realize;
    k->unrealize = usb_qdev_unrealize;
738
    k->props    = usb_props;
739 740
}

741
static const TypeInfo usb_device_type_info = {
742 743 744 745 746
    .name = TYPE_USB_DEVICE,
    .parent = TYPE_DEVICE,
    .instance_size = sizeof(USBDevice),
    .abstract = true,
    .class_size = sizeof(USBDeviceClass),
747
    .class_init = usb_device_class_init,
748 749
};

A
Andreas Färber 已提交
750
static void usb_register_types(void)
751
{
752
    type_register_static(&usb_bus_info);
753 754 755
    type_register_static(&usb_device_type_info);
}

A
Andreas Färber 已提交
756
type_init(usb_register_types)