bus.c 21.2 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);
M
Marc-André Lureau 已提交
282 283 284 285 286 287 288
    USBDescString *s, *next;

    QLIST_FOREACH_SAFE(s, &dev->strings, next, next) {
        QLIST_REMOVE(s, next);
        g_free(s->str);
        g_free(s);
    }
289

290 291 292
    if (dev->attached) {
        usb_device_detach(dev);
    }
293
    usb_device_handle_destroy(dev);
294 295 296
    if (dev->port) {
        usb_release_port(dev);
    }
297 298
}

299
typedef struct LegacyUSBFactory
300
{
301 302
    const char *name;
    const char *usbdevice_name;
303
    USBDevice *(*usbdevice_init)(USBBus *bus, const char *params);
304
} LegacyUSBFactory;
305

306 307
static GSList *legacy_usb_factory;

308
void usb_legacy_register(const char *typename, const char *usbdevice_name,
309 310
                         USBDevice *(*usbdevice_init)(USBBus *bus,
                                                      const char *params))
311
{
312 313
    if (usbdevice_name) {
        LegacyUSBFactory *f = g_malloc0(sizeof(*f));
314
        f->name = typename;
315 316 317
        f->usbdevice_name = usbdevice_name;
        f->usbdevice_init = usbdevice_init;
        legacy_usb_factory = g_slist_append(legacy_usb_factory, f);
318 319 320
    }
}

321
USBDevice *usb_create(USBBus *bus, const char *name)
322 323 324 325
{
    DeviceState *dev;

    dev = qdev_create(&bus->qbus, name);
326
    return USB_DEVICE(dev);
327
}
328

329 330
static USBDevice *usb_try_create_simple(USBBus *bus, const char *name,
                                        Error **errp)
331
{
332 333
    Error *err = NULL;
    USBDevice *dev;
334

335 336 337 338 339 340 341
    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) {
342 343 344
        error_propagate(errp, err);
        error_prepend(errp, "Failed to initialize USB device '%s': ",
                      name);
345
        object_unparent(OBJECT(dev));
346
        return NULL;
P
Paul Brook 已提交
347
    }
348 349 350
    return dev;
}

351 352
USBDevice *usb_create_simple(USBBus *bus, const char *name)
{
353
    return usb_try_create_simple(bus, name, &error_abort);
354 355
}

356 357
static void usb_fill_port(USBPort *port, void *opaque, int index,
                          USBPortOps *ops, int speedmask)
358
{
359 360 361
    port->opaque = opaque;
    port->index = index;
    port->ops = ops;
G
Gerd Hoffmann 已提交
362
    port->speedmask = speedmask;
363
    usb_port_location(port, NULL, index + 1);
364 365 366 367 368 369
}

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 已提交
370
    QTAILQ_INSERT_TAIL(&bus->free, port, next);
371 372 373
    bus->nfree++;
}

374 375 376 377
void usb_register_companion(const char *masterbus, USBPort *ports[],
                            uint32_t portcount, uint32_t firstport,
                            void *opaque, USBPortOps *ops, int speedmask,
                            Error **errp)
378 379 380 381 382 383 384 385 386 387
{
    USBBus *bus;
    int i;

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

388 389 390 391 392 393 394 395
    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);
396
        return;
397 398 399 400 401 402
    }

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

403
    bus->ops->register_companion(bus, ports, portcount, firstport, errp);
404 405
}

406 407 408 409 410
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 已提交
411
        downstream->hubcount = upstream->hubcount + 1;
412 413
    } else {
        snprintf(downstream->path, sizeof(downstream->path), "%d", portnr);
G
Gerd Hoffmann 已提交
414
        downstream->hubcount = 0;
415 416 417
    }
}

418 419
void usb_unregister_port(USBBus *bus, USBPort *port)
{
420 421 422
    if (port->dev) {
        object_unparent(OBJECT(port->dev));
    }
423 424 425 426
    QTAILQ_REMOVE(&bus->free, port, next);
    bus->nfree--;
}

427
void usb_claim_port(USBDevice *dev, Error **errp)
428 429 430 431
{
    USBBus *bus = usb_bus_from_device(dev);
    USBPort *port;

432 433
    assert(dev->port == NULL);

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

B
Blue Swirl 已提交
459
    QTAILQ_REMOVE(&bus->free, port, next);
460 461
    bus->nfree--;

462 463
    dev->port = port;
    port->dev = dev;
464

B
Blue Swirl 已提交
465
    QTAILQ_INSERT_TAIL(&bus->used, port, next);
466 467 468
    bus->nused++;
}

469
void usb_release_port(USBDevice *dev)
470 471
{
    USBBus *bus = usb_bus_from_device(dev);
472
    USBPort *port = dev->port;
473

474 475 476 477 478 479 480 481 482 483 484
    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++;
485 486
}

487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508
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);
        }
    }
}

509
void usb_check_attach(USBDevice *dev, Error **errp)
510 511
{
    USBBus *bus = usb_bus_from_device(dev);
512
    USBPort *port = dev->port;
513
    char devspeed[32], portspeed[32];
514

515 516
    assert(port != NULL);
    assert(!dev->attached);
517 518 519 520
    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);
521 522

    if (!(port->speedmask & dev->speedmask)) {
523 524 525 526 527 528
        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;
529
    }
530 531 532 533 534 535 536 537 538 539 540 541
}

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

543
    dev->attached = true;
544 545 546 547 548 549 550
    usb_attach(port);
}

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

552 553 554
    assert(port != NULL);
    assert(dev->attached);
    trace_usb_port_detach(bus->busnr, port->path);
555

556
    usb_detach(port);
557
    dev->attached = false;
558 559 560
    return 0;
}

561 562 563 564 565 566 567 568 569 570
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 已提交
571
    QTAILQ_FOREACH(port, &bus->used, next) {
572 573 574 575 576 577 578
        if (port->dev->addr == addr)
            break;
    }
    if (!port)
        return -1;
    dev = port->dev;

579
    object_unparent(OBJECT(dev));
580 581 582 583 584 585 586 587 588
    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",
589
        [ USB_SPEED_SUPER ] = "5000",
590 591 592 593 594 595 596 597
    };
    if (speed >= ARRAY_SIZE(txt))
        return "?";
    return txt[speed];
}

static void usb_bus_dev_print(Monitor *mon, DeviceState *qdev, int indent)
{
598
    USBDevice *dev = USB_DEVICE(qdev);
599 600
    USBBus *bus = usb_bus_from_device(dev);

601
    monitor_printf(mon, "%*saddr %d.%d, port %s, speed %s, name %s%s\n",
602
                   indent, "", bus->busnr, dev->addr,
603
                   dev->port ? dev->port->path : "-",
604
                   usb_speed(dev->speed), dev->product_desc,
605
                   dev->attached ? ", attached" : "");
606 607
}

608 609
static char *usb_get_dev_path(DeviceState *qdev)
{
610
    USBDevice *dev = USB_DEVICE(qdev);
G
Gerd Hoffmann 已提交
611 612 613
    DeviceState *hcd = qdev->parent_bus->parent;
    char *id = NULL;

614 615
    if (dev->flags & (1 << USB_DEV_FLAG_FULL_PATH)) {
        id = qdev_get_dev_path(hcd);
G
Gerd Hoffmann 已提交
616 617 618 619 620 621 622 623
    }
    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);
    }
624 625
}

626 627
static char *usb_get_fw_dev_path(DeviceState *qdev)
{
628
    USBDevice *dev = USB_DEVICE(qdev);
629
    char *fw_path, *in;
B
Blue Swirl 已提交
630
    ssize_t pos = 0, fw_len;
631 632
    long nr;

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

652
void hmp_info_usb(Monitor *mon, const QDict *qdict)
653 654 655 656 657
{
    USBBus *bus;
    USBDevice *dev;
    USBPort *port;

B
Blue Swirl 已提交
658
    if (QTAILQ_EMPTY(&busses)) {
659 660 661 662
        monitor_printf(mon, "USB support not enabled\n");
        return;
    }

B
Blue Swirl 已提交
663 664
    QTAILQ_FOREACH(bus, &busses, next) {
        QTAILQ_FOREACH(port, &bus->used, next) {
665 666 667
            dev = port->dev;
            if (!dev)
                continue;
668 669 670 671 672 673
            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 ?: "");
674 675 676 677
        }
    }
}

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

    params = strchr(cmdline,':');
    if (params) {
        params++;
        len = params - cmdline;
        if (len > sizeof(driver))
            len = sizeof(driver);
        pstrcpy(driver, len, cmdline);
    } else {
698
        params = "";
G
Gerd Hoffmann 已提交
699 700 701
        pstrcpy(driver, sizeof(driver), cmdline);
    }

702 703 704 705 706
    for (i = legacy_usb_factory; i; i = i->next) {
        f = i->data;
        if (strcmp(f->usbdevice_name, driver) == 0) {
            break;
        }
G
Gerd Hoffmann 已提交
707
    }
708
    if (i == NULL) {
G
Gerd Hoffmann 已提交
709 710
#if 0
        /* no error because some drivers are not converted (yet) */
711
        error_report("usbdevice %s not found", driver);
G
Gerd Hoffmann 已提交
712 713 714 715
#endif
        return NULL;
    }

716 717 718 719 720 721 722
    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;
    }

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

746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787
static bool usb_get_attached(Object *obj, Error **errp)
{
    USBDevice *dev = USB_DEVICE(obj);

    return dev->attached;
}

static void usb_set_attached(Object *obj, bool value, Error **errp)
{
    USBDevice *dev = USB_DEVICE(obj);
    Error *err = NULL;

    if (dev->attached == value) {
        return;
    }

    if (value) {
        usb_device_attach(dev, &err);
        if (err) {
            error_propagate(errp, err);
        }
    } else {
        usb_device_detach(dev);
    }
}

static void usb_device_instance_init(Object *obj)
{
    USBDevice *dev = USB_DEVICE(obj);
    USBDeviceClass *klass = USB_DEVICE_GET_CLASS(dev);

    if (klass->attached_settable) {
        object_property_add_bool(obj, "attached",
                                 usb_get_attached, usb_set_attached,
                                 NULL);
    } else {
        object_property_add_bool(obj, "attached",
                                 usb_get_attached, NULL,
                                 NULL);
    }
}

788 789 790
static void usb_device_class_init(ObjectClass *klass, void *data)
{
    DeviceClass *k = DEVICE_CLASS(klass);
791
    k->bus_type = TYPE_USB_BUS;
792 793
    k->realize  = usb_qdev_realize;
    k->unrealize = usb_qdev_unrealize;
794
    k->props    = usb_props;
795 796
}

797
static const TypeInfo usb_device_type_info = {
798 799 800
    .name = TYPE_USB_DEVICE,
    .parent = TYPE_DEVICE,
    .instance_size = sizeof(USBDevice),
801
    .instance_init = usb_device_instance_init,
802 803
    .abstract = true,
    .class_size = sizeof(USBDeviceClass),
804
    .class_init = usb_device_class_init,
805 806
};

A
Andreas Färber 已提交
807
static void usb_register_types(void)
808
{
809
    type_register_static(&usb_bus_info);
810 811 812
    type_register_static(&usb_device_type_info);
}

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