bus.c 19.9 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 "qemu/error-report.h"
6
#include "sysemu/sysemu.h"
7
#include "monitor/monitor.h"
8
#include "trace.h"
9 10

static void usb_bus_dev_print(Monitor *mon, DeviceState *qdev, int indent);
11 12

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

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

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

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

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

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

51 52 53 54 55 56 57 58 59
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;
    }
60 61
    if (dev->setup_index < 0 ||
        dev->setup_len < 0 ||
62 63
        dev->setup_index > dev->setup_len ||
        dev->setup_len > sizeof(dev->data_buf)) {
64 65
        return -EINVAL;
    }
66 67 68
    return 0;
}

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

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

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

    QTAILQ_REMOVE(&busses, bus, next);
}

105 106 107 108 109
USBBus *usb_bus_find(int busnr)
{
    USBBus *bus;

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

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

    if (klass->realize) {
        klass->realize(dev, errp);
124 125 126
    }
}

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

297 298
static GSList *legacy_usb_factory;

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

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

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

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

326 327 328 329 330 331 332
    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) {
333 334 335
        error_propagate(errp, err);
        error_prepend(errp, "Failed to initialize USB device '%s': ",
                      name);
336
        object_unparent(OBJECT(dev));
337
        return NULL;
P
Paul Brook 已提交
338
    }
339 340 341
    return dev;
}

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

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

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

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

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

379 380 381 382 383 384 385 386
    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);
387
        return;
388 389 390 391 392 393
    }

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

394
    bus->ops->register_companion(bus, ports, portcount, firstport, errp);
395 396
}

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

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

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

423 424
    assert(dev->port == NULL);

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

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

453 454
    dev->port = port;
    port->dev = dev;
455

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

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

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

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

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

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

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

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

534 535 536 537 538 539 540 541
    dev->attached++;
    usb_attach(port);
}

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

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

547 548
    usb_detach(port);
    dev->attached--;
549 550 551
    return 0;
}

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

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

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

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

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

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

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

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

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

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

B
Blue Swirl 已提交
654 655
    QTAILQ_FOREACH(bus, &busses, next) {
        QTAILQ_FOREACH(port, &bus->used, next) {
656 657 658
            dev = port->dev;
            if (!dev)
                continue;
659 660 661 662 663 664
            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 ?: "");
665 666 667 668
        }
    }
}

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

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

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

707 708 709 710 711 712 713
    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;
    }

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

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

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

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

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