bus.c 20.0 KB
Newer Older
P
Peter Maydell 已提交
1
#include "qemu/osdep.h"
G
Gerd Hoffmann 已提交
2 3 4
#include "hw/hw.h"
#include "hw/usb.h"
#include "hw/qdev.h"
5
#include "qapi/error.h"
6
#include "qemu/error-report.h"
7
#include "sysemu/sysemu.h"
8
#include "monitor/monitor.h"
9
#include "trace.h"
10
#include "qemu/cutils.h"
11 12

static void usb_bus_dev_print(Monitor *mon, DeviceState *qdev, int indent);
13 14

static char *usb_get_dev_path(DeviceState *dev);
15
static char *usb_get_fw_dev_path(DeviceState *qdev);
16
static void usb_qdev_unrealize(DeviceState *qdev, Error **errp);
17

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

28 29 30
static void usb_bus_class_init(ObjectClass *klass, void *data)
{
    BusClass *k = BUS_CLASS(klass);
31
    HotplugHandlerClass *hc = HOTPLUG_HANDLER_CLASS(klass);
32 33 34 35

    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;
36
    hc->unplug = qdev_simple_device_unplug_cb;
37 38 39 40 41 42 43
}

static const TypeInfo usb_bus_info = {
    .name = TYPE_USB_BUS,
    .parent = TYPE_BUS,
    .instance_size = sizeof(USBBus),
    .class_init = usb_bus_class_init,
44 45 46 47
    .interfaces = (InterfaceInfo[]) {
        { TYPE_HOTPLUG_HANDLER },
        { }
    }
48
};
49

50
static int next_usb_bus = 0;
B
Blue Swirl 已提交
51
static QTAILQ_HEAD(, USBBus) busses = QTAILQ_HEAD_INITIALIZER(busses);
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) {
58
        dev->attached = false;
59
    } else {
60
        dev->attached = true;
61
    }
62 63
    if (dev->setup_index < 0 ||
        dev->setup_len < 0 ||
64 65
        dev->setup_index > dev->setup_len ||
        dev->setup_len > sizeof(dev->data_buf)) {
66 67
        return -EINVAL;
    }
68 69 70
    return 0;
}

G
Gerd Hoffmann 已提交
71 72 73 74
const VMStateDescription vmstate_usb_device = {
    .name = "USBDevice",
    .version_id = 1,
    .minimum_version_id = 1,
75
    .post_load = usb_device_post_load,
76
    .fields = (VMStateField[]) {
G
Gerd Hoffmann 已提交
77 78 79 80 81 82 83 84 85 86 87
        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(),
    }
};

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

G
Gonglei 已提交
100 101 102 103 104 105 106
void usb_bus_release(USBBus *bus)
{
    assert(next_usb_bus > 0);

    QTAILQ_REMOVE(&busses, bus, next);
}

107 108 109 110 111
USBBus *usb_bus_find(int busnr)
{
    USBBus *bus;

    if (-1 == busnr)
B
Blue Swirl 已提交
112 113
        return QTAILQ_FIRST(&busses);
    QTAILQ_FOREACH(bus, &busses, next) {
114 115 116 117 118 119
        if (bus->busnr == busnr)
            return bus;
    }
    return NULL;
}

120
static void usb_device_realize(USBDevice *dev, Error **errp)
121 122
{
    USBDeviceClass *klass = USB_DEVICE_GET_CLASS(dev);
123 124 125

    if (klass->realize) {
        klass->realize(dev, errp);
126 127 128
    }
}

G
Gerd Hoffmann 已提交
129
USBDevice *usb_device_find_device(USBDevice *dev, uint8_t addr)
130 131
{
    USBDeviceClass *klass = USB_DEVICE_GET_CLASS(dev);
G
Gerd Hoffmann 已提交
132 133
    if (klass->find_device) {
        return klass->find_device(dev, addr);
134
    }
G
Gerd Hoffmann 已提交
135
    return NULL;
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 166 167 168 169
}

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

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

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

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);
196 197 198
    if (dev->usb_desc) {
        return dev->usb_desc;
    }
199 200 201 202 203 204 205 206 207 208 209 210
    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);
    }
}

211 212 213 214 215 216 217 218
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);
    }
}

219 220 221 222 223 224 225 226
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);
    }
}

227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244
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);
    }
}

245
static void usb_qdev_realize(DeviceState *qdev, Error **errp)
246
{
247
    USBDevice *dev = USB_DEVICE(qdev);
248
    Error *local_err = NULL;
249

250 251
    pstrcpy(dev->product_desc, sizeof(dev->product_desc),
            usb_device_get_product_desc(dev));
G
Gerd Hoffmann 已提交
252
    dev->auto_attach = 1;
253
    QLIST_INIT(&dev->strings);
G
Gerd Hoffmann 已提交
254
    usb_ep_init(dev);
255 256 257 258 259

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

    usb_device_realize(dev, &local_err);
    if (local_err) {
264
        usb_release_port(dev);
265 266
        error_propagate(errp, local_err);
        return;
267
    }
268

269
    if (dev->auto_attach) {
270 271 272 273 274
        usb_device_attach(dev, &local_err);
        if (local_err) {
            usb_qdev_unrealize(qdev, NULL);
            error_propagate(errp, local_err);
            return;
275
        }
276
    }
277 278
}

279
static void usb_qdev_unrealize(DeviceState *qdev, Error **errp)
280
{
281
    USBDevice *dev = USB_DEVICE(qdev);
282

283 284 285
    if (dev->attached) {
        usb_device_detach(dev);
    }
286
    usb_device_handle_destroy(dev);
287 288 289
    if (dev->port) {
        usb_release_port(dev);
    }
290 291
}

292
typedef struct LegacyUSBFactory
293
{
294 295
    const char *name;
    const char *usbdevice_name;
296
    USBDevice *(*usbdevice_init)(USBBus *bus, const char *params);
297
} LegacyUSBFactory;
298

299 300
static GSList *legacy_usb_factory;

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

314
USBDevice *usb_create(USBBus *bus, const char *name)
315 316 317 318
{
    DeviceState *dev;

    dev = qdev_create(&bus->qbus, name);
319
    return USB_DEVICE(dev);
320
}
321

322 323
static USBDevice *usb_try_create_simple(USBBus *bus, const char *name,
                                        Error **errp)
324
{
325 326
    Error *err = NULL;
    USBDevice *dev;
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) {
335 336 337
        error_propagate(errp, err);
        error_prepend(errp, "Failed to initialize USB device '%s': ",
                      name);
338
        object_unparent(OBJECT(dev));
339
        return NULL;
P
Paul Brook 已提交
340
    }
341 342 343
    return dev;
}

344 345
USBDevice *usb_create_simple(USBBus *bus, const char *name)
{
346
    return usb_try_create_simple(bus, name, &error_abort);
347 348
}

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

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 已提交
363
    QTAILQ_INSERT_TAIL(&bus->free, port, next);
364 365 366
    bus->nfree++;
}

367 368 369 370
void usb_register_companion(const char *masterbus, USBPort *ports[],
                            uint32_t portcount, uint32_t firstport,
                            void *opaque, USBPortOps *ops, int speedmask,
                            Error **errp)
371 372 373 374 375 376 377 378 379 380
{
    USBBus *bus;
    int i;

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

381 382 383 384 385 386 387 388
    if (!bus) {
        error_setg(errp, "USB bus '%s' not found", masterbus);
        return;
    }
    if (!bus->ops->register_companion) {
        error_setg(errp, "Can't use USB bus '%s' as masterbus,"
                   " it doesn't support companion controllers",
                   masterbus);
389
        return;
390 391 392 393 394 395
    }

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

396
    bus->ops->register_companion(bus, ports, portcount, firstport, errp);
397 398
}

399 400 401 402 403
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 已提交
404
        downstream->hubcount = upstream->hubcount + 1;
405 406
    } else {
        snprintf(downstream->path, sizeof(downstream->path), "%d", portnr);
G
Gerd Hoffmann 已提交
407
        downstream->hubcount = 0;
408 409 410
    }
}

411 412
void usb_unregister_port(USBBus *bus, USBPort *port)
{
413 414 415
    if (port->dev) {
        object_unparent(OBJECT(port->dev));
    }
416 417 418 419
    QTAILQ_REMOVE(&bus->free, port, next);
    bus->nfree--;
}

420
void usb_claim_port(USBDevice *dev, Error **errp)
421 422 423 424
{
    USBBus *bus = usb_bus_from_device(dev);
    USBPort *port;

425 426
    assert(dev->port == NULL);

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

B
Blue Swirl 已提交
452
    QTAILQ_REMOVE(&bus->free, port, next);
453 454
    bus->nfree--;

455 456
    dev->port = port;
    port->dev = dev;
457

B
Blue Swirl 已提交
458
    QTAILQ_INSERT_TAIL(&bus->used, port, next);
459 460 461
    bus->nused++;
}

462
void usb_release_port(USBDevice *dev)
463 464
{
    USBBus *bus = usb_bus_from_device(dev);
465
    USBPort *port = dev->port;
466

467 468 469 470 471 472 473 474 475 476 477
    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++;
478 479
}

480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501
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);
        }
    }
}

502
void usb_check_attach(USBDevice *dev, Error **errp)
503 504
{
    USBBus *bus = usb_bus_from_device(dev);
505
    USBPort *port = dev->port;
506
    char devspeed[32], portspeed[32];
507

508 509
    assert(port != NULL);
    assert(!dev->attached);
510 511 512 513
    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);
514 515

    if (!(port->speedmask & dev->speedmask)) {
516 517 518 519 520 521
        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;
522
    }
523 524 525 526 527 528 529 530 531 532 533 534
}

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;
    }
535

536
    dev->attached = true;
537 538 539 540 541 542 543
    usb_attach(port);
}

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

545 546 547
    assert(port != NULL);
    assert(dev->attached);
    trace_usb_port_detach(bus->busnr, port->path);
548

549
    usb_detach(port);
550
    dev->attached = false;
551 552 553
    return 0;
}

554 555 556 557 558 559 560 561 562 563
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 已提交
564
    QTAILQ_FOREACH(port, &bus->used, next) {
565 566 567 568 569 570 571
        if (port->dev->addr == addr)
            break;
    }
    if (!port)
        return -1;
    dev = port->dev;

572
    object_unparent(OBJECT(dev));
573 574 575 576 577 578 579 580 581
    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",
582
        [ USB_SPEED_SUPER ] = "5000",
583 584 585 586 587 588 589 590
    };
    if (speed >= ARRAY_SIZE(txt))
        return "?";
    return txt[speed];
}

static void usb_bus_dev_print(Monitor *mon, DeviceState *qdev, int indent)
{
591
    USBDevice *dev = USB_DEVICE(qdev);
592 593
    USBBus *bus = usb_bus_from_device(dev);

594
    monitor_printf(mon, "%*saddr %d.%d, port %s, speed %s, name %s%s\n",
595
                   indent, "", bus->busnr, dev->addr,
596
                   dev->port ? dev->port->path : "-",
597
                   usb_speed(dev->speed), dev->product_desc,
598
                   dev->attached ? ", attached" : "");
599 600
}

601 602
static char *usb_get_dev_path(DeviceState *qdev)
{
603
    USBDevice *dev = USB_DEVICE(qdev);
G
Gerd Hoffmann 已提交
604 605 606
    DeviceState *hcd = qdev->parent_bus->parent;
    char *id = NULL;

607 608
    if (dev->flags & (1 << USB_DEV_FLAG_FULL_PATH)) {
        id = qdev_get_dev_path(hcd);
G
Gerd Hoffmann 已提交
609 610 611 612 613 614 615 616
    }
    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);
    }
617 618
}

619 620
static char *usb_get_fw_dev_path(DeviceState *qdev)
{
621
    USBDevice *dev = USB_DEVICE(qdev);
622
    char *fw_path, *in;
B
Blue Swirl 已提交
623
    ssize_t pos = 0, fw_len;
624 625
    long nr;

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

645
void hmp_info_usb(Monitor *mon, const QDict *qdict)
646 647 648 649 650
{
    USBBus *bus;
    USBDevice *dev;
    USBPort *port;

B
Blue Swirl 已提交
651
    if (QTAILQ_EMPTY(&busses)) {
652 653 654 655
        monitor_printf(mon, "USB support not enabled\n");
        return;
    }

B
Blue Swirl 已提交
656 657
    QTAILQ_FOREACH(bus, &busses, next) {
        QTAILQ_FOREACH(port, &bus->used, next) {
658 659 660
            dev = port->dev;
            if (!dev)
                continue;
661 662 663 664 665 666
            monitor_printf(mon, "  Device %d.%d, Port %s, Speed %s Mb/s, "
                           "Product %s%s%s\n",
                           bus->busnr, dev->addr, port->path,
                           usb_speed(dev->speed), dev->product_desc,
                           dev->qdev.id ? ", ID: " : "",
                           dev->qdev.id ?: "");
667 668 669 670
        }
    }
}

G
Gerd Hoffmann 已提交
671 672 673 674
/* handle legacy -usbdevice cmd line option */
USBDevice *usbdevice_create(const char *cmdline)
{
    USBBus *bus = usb_bus_find(-1 /* any */);
675
    LegacyUSBFactory *f = NULL;
676
    Error *err = NULL;
677
    GSList *i;
678 679
    char driver[32];
    const char *params;
G
Gerd Hoffmann 已提交
680
    int len;
681
    USBDevice *dev;
G
Gerd Hoffmann 已提交
682 683 684 685 686 687 688 689 690

    params = strchr(cmdline,':');
    if (params) {
        params++;
        len = params - cmdline;
        if (len > sizeof(driver))
            len = sizeof(driver);
        pstrcpy(driver, len, cmdline);
    } else {
691
        params = "";
G
Gerd Hoffmann 已提交
692 693 694
        pstrcpy(driver, sizeof(driver), cmdline);
    }

695 696 697 698 699
    for (i = legacy_usb_factory; i; i = i->next) {
        f = i->data;
        if (strcmp(f->usbdevice_name, driver) == 0) {
            break;
        }
G
Gerd Hoffmann 已提交
700
    }
701
    if (i == NULL) {
G
Gerd Hoffmann 已提交
702 703
#if 0
        /* no error because some drivers are not converted (yet) */
704
        error_report("usbdevice %s not found", driver);
G
Gerd Hoffmann 已提交
705 706 707 708
#endif
        return NULL;
    }

709 710 711 712 713 714 715
    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;
    }

716 717 718
    if (f->usbdevice_init) {
        dev = f->usbdevice_init(bus, params);
    } else {
T
TeLeMan 已提交
719
        if (*params) {
720
            error_report("usbdevice %s accepts no params", driver);
G
Gerd Hoffmann 已提交
721 722
            return NULL;
        }
723 724 725 726 727
        dev = usb_create(bus, f->name);
    }
    if (!dev) {
        error_report("Failed to create USB device '%s'", f->name);
        return NULL;
G
Gerd Hoffmann 已提交
728
    }
729 730
    object_property_set_bool(OBJECT(dev), true, "realized", &err);
    if (err) {
731 732
        error_reportf_err(err, "Failed to initialize USB device '%s': ",
                          f->name);
733 734 735 736
        object_unparent(OBJECT(dev));
        return NULL;
    }
    return dev;
G
Gerd Hoffmann 已提交
737
}
738

739 740 741
static void usb_device_class_init(ObjectClass *klass, void *data)
{
    DeviceClass *k = DEVICE_CLASS(klass);
742
    k->bus_type = TYPE_USB_BUS;
743 744
    k->realize  = usb_qdev_realize;
    k->unrealize = usb_qdev_unrealize;
745
    k->props    = usb_props;
746 747
}

748
static const TypeInfo usb_device_type_info = {
749 750 751 752 753
    .name = TYPE_USB_DEVICE,
    .parent = TYPE_DEVICE,
    .instance_size = sizeof(USBDevice),
    .abstract = true,
    .class_size = sizeof(USBDeviceClass),
754
    .class_init = usb_device_class_init,
755 756
};

A
Andreas Färber 已提交
757
static void usb_register_types(void)
758
{
759
    type_register_static(&usb_bus_info);
760 761 762
    type_register_static(&usb_device_type_info);
}

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