]>
Commit | Line | Data |
---|---|---|
62aed765 AL |
1 | #ifndef QEMU_USB_H |
2 | #define QEMU_USB_H | |
3 | ||
bb36d470 FB |
4 | /* |
5 | * QEMU USB API | |
5fafdf24 | 6 | * |
bb36d470 | 7 | * Copyright (c) 2005 Fabrice Bellard |
5fafdf24 | 8 | * |
bb36d470 FB |
9 | * Permission is hereby granted, free of charge, to any person obtaining a copy |
10 | * of this software and associated documentation files (the "Software"), to deal | |
11 | * in the Software without restriction, including without limitation the rights | |
12 | * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | |
13 | * copies of the Software, and to permit persons to whom the Software is | |
14 | * furnished to do so, subject to the following conditions: | |
15 | * | |
16 | * The above copyright notice and this permission notice shall be included in | |
17 | * all copies or substantial portions of the Software. | |
18 | * | |
19 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | |
20 | * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | |
21 | * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL | |
22 | * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | |
23 | * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | |
24 | * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | |
25 | * THE SOFTWARE. | |
26 | */ | |
c0f4ce77 | 27 | |
d4842052 | 28 | #include "exec/memory.h" |
a27bd6c7 | 29 | #include "hw/qdev-core.h" |
daf015ef | 30 | #include "qemu/iov.h" |
1de7afc9 | 31 | #include "qemu/queue.h" |
db1015e9 | 32 | #include "qom/object.h" |
c0f4ce77 | 33 | |
8e257816 BH |
34 | /* Constants related to the USB / PCI interaction */ |
35 | #define USB_SBRN 0x60 /* Serial Bus Release Number Register */ | |
36 | #define USB_RELEASE_1 0x10 /* USB 1.0 */ | |
37 | #define USB_RELEASE_2 0x20 /* USB 2.0 */ | |
38 | #define USB_RELEASE_3 0x30 /* USB 3.0 */ | |
39 | ||
bb36d470 FB |
40 | #define USB_TOKEN_SETUP 0x2d |
41 | #define USB_TOKEN_IN 0x69 /* device -> host */ | |
42 | #define USB_TOKEN_OUT 0xe1 /* host -> device */ | |
43 | ||
9a77a0f5 | 44 | #define USB_RET_SUCCESS (0) |
36dfe324 HG |
45 | #define USB_RET_NODEV (-1) |
46 | #define USB_RET_NAK (-2) | |
47 | #define USB_RET_STALL (-3) | |
48 | #define USB_RET_BABBLE (-4) | |
49 | #define USB_RET_IOERROR (-5) | |
50 | #define USB_RET_ASYNC (-6) | |
51 | #define USB_RET_ADD_TO_QUEUE (-7) | |
0cae7b1a | 52 | #define USB_RET_REMOVE_FROM_QUEUE (-8) |
bb36d470 FB |
53 | |
54 | #define USB_SPEED_LOW 0 | |
55 | #define USB_SPEED_FULL 1 | |
56 | #define USB_SPEED_HIGH 2 | |
843d4e0c GH |
57 | #define USB_SPEED_SUPER 3 |
58 | ||
59 | #define USB_SPEED_MASK_LOW (1 << USB_SPEED_LOW) | |
60 | #define USB_SPEED_MASK_FULL (1 << USB_SPEED_FULL) | |
61 | #define USB_SPEED_MASK_HIGH (1 << USB_SPEED_HIGH) | |
62 | #define USB_SPEED_MASK_SUPER (1 << USB_SPEED_SUPER) | |
bb36d470 FB |
63 | |
64 | #define USB_STATE_NOTATTACHED 0 | |
65 | #define USB_STATE_ATTACHED 1 | |
66 | //#define USB_STATE_POWERED 2 | |
67 | #define USB_STATE_DEFAULT 3 | |
68 | //#define USB_STATE_ADDRESS 4 | |
69 | //#define USB_STATE_CONFIGURED 5 | |
70 | #define USB_STATE_SUSPENDED 6 | |
71 | ||
a594cfbf FB |
72 | #define USB_CLASS_AUDIO 1 |
73 | #define USB_CLASS_COMM 2 | |
74 | #define USB_CLASS_HID 3 | |
75 | #define USB_CLASS_PHYSICAL 5 | |
76 | #define USB_CLASS_STILL_IMAGE 6 | |
77 | #define USB_CLASS_PRINTER 7 | |
78 | #define USB_CLASS_MASS_STORAGE 8 | |
79 | #define USB_CLASS_HUB 9 | |
80 | #define USB_CLASS_CDC_DATA 0x0a | |
81 | #define USB_CLASS_CSCID 0x0b | |
82 | #define USB_CLASS_CONTENT_SEC 0x0d | |
83 | #define USB_CLASS_APP_SPEC 0xfe | |
84 | #define USB_CLASS_VENDOR_SPEC 0xff | |
85 | ||
b870472d PA |
86 | #define USB_SUBCLASS_UNDEFINED 0 |
87 | #define USB_SUBCLASS_AUDIO_CONTROL 1 | |
88 | #define USB_SUBCLASS_AUDIO_STREAMING 2 | |
89 | #define USB_SUBCLASS_AUDIO_MIDISTREAMING 3 | |
90 | ||
bb36d470 FB |
91 | #define USB_DIR_OUT 0 |
92 | #define USB_DIR_IN 0x80 | |
93 | ||
94 | #define USB_TYPE_MASK (0x03 << 5) | |
95 | #define USB_TYPE_STANDARD (0x00 << 5) | |
96 | #define USB_TYPE_CLASS (0x01 << 5) | |
97 | #define USB_TYPE_VENDOR (0x02 << 5) | |
98 | #define USB_TYPE_RESERVED (0x03 << 5) | |
99 | ||
100 | #define USB_RECIP_MASK 0x1f | |
101 | #define USB_RECIP_DEVICE 0x00 | |
102 | #define USB_RECIP_INTERFACE 0x01 | |
103 | #define USB_RECIP_ENDPOINT 0x02 | |
104 | #define USB_RECIP_OTHER 0x03 | |
105 | ||
106 | #define DeviceRequest ((USB_DIR_IN|USB_TYPE_STANDARD|USB_RECIP_DEVICE)<<8) | |
107 | #define DeviceOutRequest ((USB_DIR_OUT|USB_TYPE_STANDARD|USB_RECIP_DEVICE)<<8) | |
2b81ba53 GH |
108 | #define VendorDeviceRequest ((USB_DIR_IN|USB_TYPE_VENDOR|USB_RECIP_DEVICE)<<8) |
109 | #define VendorDeviceOutRequest \ | |
110 | ((USB_DIR_OUT|USB_TYPE_VENDOR|USB_RECIP_DEVICE)<<8) | |
111 | ||
112 | #define InterfaceRequest \ | |
59ae540c FB |
113 | ((USB_DIR_IN|USB_TYPE_STANDARD|USB_RECIP_INTERFACE)<<8) |
114 | #define InterfaceOutRequest \ | |
115 | ((USB_DIR_OUT|USB_TYPE_STANDARD|USB_RECIP_INTERFACE)<<8) | |
f3571b1a MR |
116 | #define ClassInterfaceRequest \ |
117 | ((USB_DIR_IN|USB_TYPE_CLASS|USB_RECIP_INTERFACE)<<8) | |
118 | #define ClassInterfaceOutRequest \ | |
119 | ((USB_DIR_OUT|USB_TYPE_CLASS|USB_RECIP_INTERFACE)<<8) | |
2b81ba53 GH |
120 | #define VendorInterfaceRequest \ |
121 | ((USB_DIR_IN|USB_TYPE_VENDOR|USB_RECIP_INTERFACE)<<8) | |
122 | #define VendorInterfaceOutRequest \ | |
123 | ((USB_DIR_OUT|USB_TYPE_VENDOR|USB_RECIP_INTERFACE)<<8) | |
124 | ||
125 | #define EndpointRequest ((USB_DIR_IN|USB_TYPE_STANDARD|USB_RECIP_ENDPOINT)<<8) | |
126 | #define EndpointOutRequest \ | |
127 | ((USB_DIR_OUT|USB_TYPE_STANDARD|USB_RECIP_ENDPOINT)<<8) | |
bb36d470 FB |
128 | |
129 | #define USB_REQ_GET_STATUS 0x00 | |
130 | #define USB_REQ_CLEAR_FEATURE 0x01 | |
131 | #define USB_REQ_SET_FEATURE 0x03 | |
132 | #define USB_REQ_SET_ADDRESS 0x05 | |
133 | #define USB_REQ_GET_DESCRIPTOR 0x06 | |
134 | #define USB_REQ_SET_DESCRIPTOR 0x07 | |
135 | #define USB_REQ_GET_CONFIGURATION 0x08 | |
136 | #define USB_REQ_SET_CONFIGURATION 0x09 | |
137 | #define USB_REQ_GET_INTERFACE 0x0A | |
138 | #define USB_REQ_SET_INTERFACE 0x0B | |
139 | #define USB_REQ_SYNCH_FRAME 0x0C | |
811ad5d8 GH |
140 | #define USB_REQ_SET_SEL 0x30 |
141 | #define USB_REQ_SET_ISOCH_DELAY 0x31 | |
bb36d470 FB |
142 | |
143 | #define USB_DEVICE_SELF_POWERED 0 | |
144 | #define USB_DEVICE_REMOTE_WAKEUP 1 | |
145 | ||
146 | #define USB_DT_DEVICE 0x01 | |
147 | #define USB_DT_CONFIG 0x02 | |
148 | #define USB_DT_STRING 0x03 | |
149 | #define USB_DT_INTERFACE 0x04 | |
150 | #define USB_DT_ENDPOINT 0x05 | |
25620cba GH |
151 | #define USB_DT_DEVICE_QUALIFIER 0x06 |
152 | #define USB_DT_OTHER_SPEED_CONFIG 0x07 | |
a7fb71d1 | 153 | #define USB_DT_DEBUG 0x0A |
c6d3ad0f | 154 | #define USB_DT_INTERFACE_ASSOC 0x0B |
2077469b GH |
155 | #define USB_DT_BOS 0x0F |
156 | #define USB_DT_DEVICE_CAPABILITY 0x10 | |
b870472d PA |
157 | #define USB_DT_CS_INTERFACE 0x24 |
158 | #define USB_DT_CS_ENDPOINT 0x25 | |
b43a2851 | 159 | #define USB_DT_ENDPOINT_COMPANION 0x30 |
bb36d470 | 160 | |
2077469b GH |
161 | #define USB_DEV_CAP_WIRELESS 0x01 |
162 | #define USB_DEV_CAP_USB2_EXT 0x02 | |
163 | #define USB_DEV_CAP_SUPERSPEED 0x03 | |
164 | ||
bd93976a PK |
165 | #define USB_CFG_ATT_ONE (1 << 7) /* should always be set */ |
166 | #define USB_CFG_ATT_SELFPOWER (1 << 6) | |
167 | #define USB_CFG_ATT_WAKEUP (1 << 5) | |
168 | #define USB_CFG_ATT_BATTERY (1 << 4) | |
169 | ||
942ac052 AZ |
170 | #define USB_ENDPOINT_XFER_CONTROL 0 |
171 | #define USB_ENDPOINT_XFER_ISOC 1 | |
172 | #define USB_ENDPOINT_XFER_BULK 2 | |
173 | #define USB_ENDPOINT_XFER_INT 3 | |
d8e17efd | 174 | #define USB_ENDPOINT_XFER_INVALID 255 |
942ac052 | 175 | |
7c37e6a4 GH |
176 | #define USB_INTERFACE_INVALID 255 |
177 | ||
07771f6f | 178 | typedef struct USBBusOps USBBusOps; |
bb36d470 FB |
179 | typedef struct USBPort USBPort; |
180 | typedef struct USBDevice USBDevice; | |
4d611c9a | 181 | typedef struct USBPacket USBPacket; |
a552a966 | 182 | typedef struct USBCombinedPacket USBCombinedPacket; |
d8e17efd | 183 | typedef struct USBEndpoint USBEndpoint; |
bb36d470 | 184 | |
37fb59d3 GH |
185 | typedef struct USBDesc USBDesc; |
186 | typedef struct USBDescID USBDescID; | |
187 | typedef struct USBDescDevice USBDescDevice; | |
188 | typedef struct USBDescConfig USBDescConfig; | |
6e625fc7 | 189 | typedef struct USBDescIfaceAssoc USBDescIfaceAssoc; |
37fb59d3 GH |
190 | typedef struct USBDescIface USBDescIface; |
191 | typedef struct USBDescEndpoint USBDescEndpoint; | |
192 | typedef struct USBDescOther USBDescOther; | |
132a3f55 | 193 | typedef struct USBDescString USBDescString; |
5319dc7b | 194 | typedef struct USBDescMSOS USBDescMSOS; |
132a3f55 GH |
195 | |
196 | struct USBDescString { | |
197 | uint8_t index; | |
198 | char *str; | |
199 | QLIST_ENTRY(USBDescString) next; | |
200 | }; | |
37fb59d3 | 201 | |
1de14d43 GH |
202 | #define USB_MAX_ENDPOINTS 15 |
203 | #define USB_MAX_INTERFACES 16 | |
204 | ||
d8e17efd | 205 | struct USBEndpoint { |
63095ab5 GH |
206 | uint8_t nr; |
207 | uint8_t pid; | |
d8e17efd | 208 | uint8_t type; |
82f02fe9 | 209 | uint8_t ifnum; |
f003397c | 210 | int max_packet_size; |
04b300f8 | 211 | int max_streams; |
7936e0f0 | 212 | bool pipeline; |
0132b4b6 | 213 | bool halted; |
25d5de7d | 214 | USBDevice *dev; |
db4be873 | 215 | QTAILQ_HEAD(, USBPacket) queue; |
d8e17efd GH |
216 | }; |
217 | ||
eeb0cf9a | 218 | enum USBDeviceFlags { |
be41efde | 219 | USB_DEV_FLAG_IS_HOST, |
5319dc7b GH |
220 | USB_DEV_FLAG_MSOS_DESC_ENABLE, |
221 | USB_DEV_FLAG_MSOS_DESC_IN_USE, | |
eeb0cf9a GH |
222 | }; |
223 | ||
bb36d470 FB |
224 | /* definition of a USB device */ |
225 | struct USBDevice { | |
806b6024 | 226 | DeviceState qdev; |
618c169b | 227 | USBPort *port; |
5f69076b | 228 | char *port_path; |
71938a09 | 229 | char *serial; |
bb36d470 | 230 | void *opaque; |
eeb0cf9a | 231 | uint32_t flags; |
89b9b79f | 232 | |
0f6dba14 GH |
233 | char *pcap_filename; |
234 | FILE *pcap; | |
235 | ||
ba3f9bfb | 236 | /* Actual connected speed */ |
806b6024 | 237 | int speed; |
ba3f9bfb HG |
238 | /* Supported speeds, not in info because it may be variable (hostdevs) */ |
239 | int speedmask; | |
806b6024 | 240 | uint8_t addr; |
0fe6d12e | 241 | char product_desc[32]; |
61e094c0 | 242 | int auto_attach; |
eb19d2b9 | 243 | bool attached; |
806b6024 | 244 | |
c1ecb40a | 245 | int32_t state; |
806b6024 | 246 | uint8_t setup_buf[8]; |
19f33223 | 247 | uint8_t data_buf[4096]; |
c1ecb40a GH |
248 | int32_t remote_wakeup; |
249 | int32_t setup_state; | |
250 | int32_t setup_len; | |
251 | int32_t setup_index; | |
132a3f55 | 252 | |
25d5de7d | 253 | USBEndpoint ep_ctl; |
d8e17efd GH |
254 | USBEndpoint ep_in[USB_MAX_ENDPOINTS]; |
255 | USBEndpoint ep_out[USB_MAX_ENDPOINTS]; | |
256 | ||
132a3f55 | 257 | QLIST_HEAD(, USBDescString) strings; |
386ab487 | 258 | const USBDesc *usb_desc; /* Overrides class usb_desc if not NULL */ |
a980a065 | 259 | const USBDescDevice *device; |
65360511 GH |
260 | |
261 | int configuration; | |
262 | int ninterfaces; | |
1de14d43 | 263 | int altsetting[USB_MAX_INTERFACES]; |
a980a065 | 264 | const USBDescConfig *config; |
1de14d43 | 265 | const USBDescIface *ifaces[USB_MAX_INTERFACES]; |
806b6024 GH |
266 | }; |
267 | ||
62aed765 | 268 | #define TYPE_USB_DEVICE "usb-device" |
a489d195 | 269 | OBJECT_DECLARE_TYPE(USBDevice, USBDeviceClass, USB_DEVICE) |
62aed765 | 270 | |
7d553f27 | 271 | typedef void (*USBDeviceRealize)(USBDevice *dev, Error **errp); |
b69c3c21 | 272 | typedef void (*USBDeviceUnrealize)(USBDevice *dev); |
7d553f27 | 273 | |
db1015e9 | 274 | struct USBDeviceClass { |
62aed765 AL |
275 | DeviceClass parent_class; |
276 | ||
7d553f27 GA |
277 | USBDeviceRealize realize; |
278 | USBDeviceUnrealize unrealize; | |
279 | ||
73796fe6 GH |
280 | /* |
281 | * Walk (enabled) downstream ports, check for a matching device. | |
282 | * Only hubs implement this. | |
283 | */ | |
284 | USBDevice *(*find_device)(USBDevice *dev, uint8_t addr); | |
285 | ||
eb5e680a GH |
286 | /* |
287 | * Called when a packet is canceled. | |
288 | */ | |
289 | void (*cancel_packet)(USBDevice *dev, USBPacket *p); | |
290 | ||
b6f77fbe GH |
291 | /* |
292 | * Attach the device | |
293 | */ | |
294 | void (*handle_attach)(USBDevice *dev); | |
295 | ||
89b9b79f AL |
296 | /* |
297 | * Reset the device | |
806b6024 | 298 | */ |
059809e4 | 299 | void (*handle_reset)(USBDevice *dev); |
89b9b79f AL |
300 | |
301 | /* | |
302 | * Process control request. | |
303 | * Called from handle_packet(). | |
304 | * | |
9a77a0f5 | 305 | * Status gets stored in p->status, and if p->status == USB_RET_SUCCESS |
993d46ce | 306 | * then the number of bytes transferred is stored in p->actual_length |
89b9b79f | 307 | */ |
9a77a0f5 HG |
308 | void (*handle_control)(USBDevice *dev, USBPacket *p, int request, int value, |
309 | int index, int length, uint8_t *data); | |
89b9b79f AL |
310 | |
311 | /* | |
312 | * Process data transfers (both BULK and ISOC). | |
313 | * Called from handle_packet(). | |
314 | * | |
9a77a0f5 | 315 | * Status gets stored in p->status, and if p->status == USB_RET_SUCCESS |
993d46ce | 316 | * then the number of bytes transferred is stored in p->actual_length |
89b9b79f | 317 | */ |
9a77a0f5 | 318 | void (*handle_data)(USBDevice *dev, USBPacket *p); |
0958b4cc | 319 | |
1de14d43 GH |
320 | void (*set_interface)(USBDevice *dev, int interface, |
321 | int alt_old, int alt_new); | |
322 | ||
36dfe324 HG |
323 | /* |
324 | * Called when the hcd is done queuing packets for an endpoint, only | |
325 | * necessary for devices which can return USB_RET_ADD_TO_QUEUE. | |
326 | */ | |
327 | void (*flush_ep_queue)(USBDevice *dev, USBEndpoint *ep); | |
328 | ||
f79738b0 HG |
329 | /* |
330 | * Called by the hcd to let the device know the queue for an endpoint | |
331 | * has been unlinked / stopped. Optional may be NULL. | |
332 | */ | |
333 | void (*ep_stopped)(USBDevice *dev, USBEndpoint *ep); | |
334 | ||
3b444ead HG |
335 | /* |
336 | * Called by the hcd to alloc / free streams on a bulk endpoint. | |
337 | * Optional may be NULL. | |
338 | */ | |
339 | int (*alloc_streams)(USBDevice *dev, USBEndpoint **eps, int nr_eps, | |
340 | int streams); | |
341 | void (*free_streams)(USBDevice *dev, USBEndpoint **eps, int nr_eps); | |
342 | ||
06384698 | 343 | const char *product_desc; |
37fb59d3 | 344 | const USBDesc *usb_desc; |
1e351dc3 | 345 | bool attached_settable; |
db1015e9 | 346 | }; |
bb36d470 | 347 | |
0d86d2be | 348 | typedef struct USBPortOps { |
618c169b GH |
349 | void (*attach)(USBPort *port); |
350 | void (*detach)(USBPort *port); | |
4706ab6c HG |
351 | /* |
352 | * This gets called when a device downstream from the device attached to | |
353 | * the port (iow attached through a hub) gets detached. | |
354 | */ | |
355 | void (*child_detach)(USBPort *port, USBDevice *child); | |
d47e59b8 HG |
356 | void (*wakeup)(USBPort *port); |
357 | /* | |
358 | * Note that port->dev will be different then the device from which | |
f53c398a | 359 | * the packet originated when a hub is involved. |
d47e59b8 HG |
360 | */ |
361 | void (*complete)(USBPort *port, USBPacket *p); | |
0d86d2be | 362 | } USBPortOps; |
0d92ed30 | 363 | |
bb36d470 FB |
364 | /* USB port on which a device can be connected */ |
365 | struct USBPort { | |
a594cfbf | 366 | USBDevice *dev; |
843d4e0c | 367 | int speedmask; |
c24e4aac | 368 | int hubcount; |
c7a2196a | 369 | char path[16]; |
0d86d2be | 370 | USBPortOps *ops; |
bb36d470 FB |
371 | void *opaque; |
372 | int index; /* internal port index, may be used with the opaque */ | |
72cf2d4f | 373 | QTAILQ_ENTRY(USBPort) next; |
bb36d470 FB |
374 | }; |
375 | ||
4d611c9a PB |
376 | typedef void USBCallback(USBPacket * packet, void *opaque); |
377 | ||
f53c398a GH |
378 | typedef enum USBPacketState { |
379 | USB_PACKET_UNDEFINED = 0, | |
380 | USB_PACKET_SETUP, | |
db4be873 | 381 | USB_PACKET_QUEUED, |
f53c398a GH |
382 | USB_PACKET_ASYNC, |
383 | USB_PACKET_COMPLETE, | |
384 | USB_PACKET_CANCELED, | |
385 | } USBPacketState; | |
386 | ||
db4be873 | 387 | /* Structure used to hold information about an active USB packet. */ |
4d611c9a PB |
388 | struct USBPacket { |
389 | /* Data fields for use by the driver. */ | |
390 | int pid; | |
e983395d | 391 | uint64_t id; |
f53c398a | 392 | USBEndpoint *ep; |
8550a02d | 393 | unsigned int stream; |
4f4321c1 | 394 | QEMUIOVector iov; |
1b4b29a1 | 395 | uint64_t parameter; /* control transfers */ |
6ba43f1f | 396 | bool short_not_ok; |
a6fb2ddb | 397 | bool int_req; |
9a77a0f5 | 398 | int status; /* USB_RET_* status code */ |
993d46ce | 399 | int actual_length; /* Number of bytes actually transferred */ |
4d611c9a | 400 | /* Internal use by the USB layer. */ |
f53c398a | 401 | USBPacketState state; |
a552a966 | 402 | USBCombinedPacket *combined; |
db4be873 | 403 | QTAILQ_ENTRY(USBPacket) queue; |
a552a966 HG |
404 | QTAILQ_ENTRY(USBPacket) combined_entry; |
405 | }; | |
406 | ||
407 | struct USBCombinedPacket { | |
408 | USBPacket *first; | |
eae3eb3e | 409 | QTAILQ_HEAD(, USBPacket) packets; |
a552a966 | 410 | QEMUIOVector iov; |
4d611c9a PB |
411 | }; |
412 | ||
4f4321c1 | 413 | void usb_packet_init(USBPacket *p); |
db4be873 | 414 | void usb_packet_set_state(USBPacket *p, USBPacketState state); |
5ac2731c | 415 | void usb_packet_check_state(USBPacket *p, USBPacketState expected); |
8550a02d GH |
416 | void usb_packet_setup(USBPacket *p, int pid, |
417 | USBEndpoint *ep, unsigned int stream, | |
418 | uint64_t id, bool short_not_ok, bool int_req); | |
4f4321c1 GH |
419 | void usb_packet_addbuf(USBPacket *p, void *ptr, size_t len); |
420 | int usb_packet_map(USBPacket *p, QEMUSGList *sgl); | |
e2f89926 | 421 | void usb_packet_unmap(USBPacket *p, QEMUSGList *sgl); |
4f4321c1 GH |
422 | void usb_packet_copy(USBPacket *p, void *ptr, size_t bytes); |
423 | void usb_packet_skip(USBPacket *p, size_t bytes); | |
6a98d1c0 | 424 | size_t usb_packet_size(USBPacket *p); |
4f4321c1 GH |
425 | void usb_packet_cleanup(USBPacket *p); |
426 | ||
f53c398a GH |
427 | static inline bool usb_packet_is_inflight(USBPacket *p) |
428 | { | |
db4be873 GH |
429 | return (p->state == USB_PACKET_QUEUED || |
430 | p->state == USB_PACKET_ASYNC); | |
f53c398a GH |
431 | } |
432 | ||
73796fe6 GH |
433 | USBDevice *usb_find_device(USBPort *port, uint8_t addr); |
434 | ||
9a77a0f5 | 435 | void usb_handle_packet(USBDevice *dev, USBPacket *p); |
4ff658fb | 436 | void usb_packet_complete(USBDevice *dev, USBPacket *p); |
d0ff81b8 | 437 | void usb_packet_complete_one(USBDevice *dev, USBPacket *p); |
4ff658fb | 438 | void usb_cancel_packet(USBPacket * p); |
53aa8c0e | 439 | |
d8e17efd | 440 | void usb_ep_init(USBDevice *dev); |
19deaa08 | 441 | void usb_ep_reset(USBDevice *dev); |
5b6780d0 | 442 | void usb_ep_dump(USBDevice *dev); |
d8e17efd GH |
443 | struct USBEndpoint *usb_ep_get(USBDevice *dev, int pid, int ep); |
444 | uint8_t usb_ep_get_type(USBDevice *dev, int pid, int ep); | |
445 | void usb_ep_set_type(USBDevice *dev, int pid, int ep, uint8_t type); | |
82f02fe9 | 446 | void usb_ep_set_ifnum(USBDevice *dev, int pid, int ep, uint8_t ifnum); |
f003397c GH |
447 | void usb_ep_set_max_packet_size(USBDevice *dev, int pid, int ep, |
448 | uint16_t raw); | |
04b300f8 | 449 | void usb_ep_set_max_streams(USBDevice *dev, int pid, int ep, uint8_t raw); |
e382d966 | 450 | void usb_ep_set_halted(USBDevice *dev, int pid, int ep, bool halted); |
c13a9e61 HG |
451 | USBPacket *usb_ep_find_packet_by_id(USBDevice *dev, int pid, int ep, |
452 | uint64_t id); | |
d8e17efd | 453 | |
a552a966 HG |
454 | void usb_ep_combine_input_packets(USBEndpoint *ep); |
455 | void usb_combined_input_packet_complete(USBDevice *dev, USBPacket *p); | |
456 | void usb_combined_packet_cancel(USBDevice *dev, USBPacket *p); | |
457 | ||
b791c3b3 | 458 | void usb_pick_speed(USBPort *port); |
891fb2cd GH |
459 | void usb_attach(USBPort *port); |
460 | void usb_detach(USBPort *port); | |
d28f4e2d GH |
461 | void usb_port_reset(USBPort *port); |
462 | void usb_device_reset(USBDevice *dev); | |
8550a02d | 463 | void usb_wakeup(USBEndpoint *ep, unsigned int stream); |
50b7963e | 464 | void usb_generic_async_ctrl_complete(USBDevice *s, USBPacket *p); |
4d611c9a | 465 | |
bb36d470 | 466 | /* usb-linux.c */ |
1ce6be24 | 467 | void hmp_info_usbhost(Monitor *mon, const QDict *qdict); |
b99260eb | 468 | bool usb_host_dev_is_scsi_storage(USBDevice *usbdev); |
59ae540c | 469 | |
87ecb68b PB |
470 | /* usb ports of the VM */ |
471 | ||
87ecb68b PB |
472 | #define VM_USB_HUB_SIZE 8 |
473 | ||
806b6024 GH |
474 | /* usb-bus.c */ |
475 | ||
0d936928 | 476 | #define TYPE_USB_BUS "usb-bus" |
8063396b | 477 | OBJECT_DECLARE_SIMPLE_TYPE(USBBus, USB_BUS) |
0d936928 | 478 | |
806b6024 GH |
479 | struct USBBus { |
480 | BusState qbus; | |
07771f6f | 481 | USBBusOps *ops; |
806b6024 GH |
482 | int busnr; |
483 | int nfree; | |
484 | int nused; | |
72cf2d4f BS |
485 | QTAILQ_HEAD(, USBPort) free; |
486 | QTAILQ_HEAD(, USBPort) used; | |
487 | QTAILQ_ENTRY(USBBus) next; | |
806b6024 GH |
488 | }; |
489 | ||
07771f6f | 490 | struct USBBusOps { |
f4bbaaf5 MA |
491 | void (*register_companion)(USBBus *bus, USBPort *ports[], |
492 | uint32_t portcount, uint32_t firstport, | |
493 | Error **errp); | |
8550a02d | 494 | void (*wakeup_endpoint)(USBBus *bus, USBEndpoint *ep, unsigned int stream); |
07771f6f GH |
495 | }; |
496 | ||
c889b3a5 AF |
497 | void usb_bus_new(USBBus *bus, size_t bus_size, |
498 | USBBusOps *ops, DeviceState *host); | |
e5a9bece | 499 | void usb_bus_release(USBBus *bus); |
806b6024 | 500 | USBBus *usb_bus_find(int busnr); |
ba02430f | 501 | void usb_legacy_register(const char *typename, const char *usbdevice_name, |
405cf80c | 502 | USBDevice *(*usbdevice_init)(void)); |
32aaaebe MA |
503 | USBDevice *usb_new(const char *name); |
504 | bool usb_realize_and_unref(USBDevice *dev, USBBus *bus, Error **errp); | |
806b6024 | 505 | USBDevice *usb_create_simple(USBBus *bus, const char *name); |
0958b4cc | 506 | USBDevice *usbdevice_create(const char *cmdline); |
a5d2f727 | 507 | void usb_register_port(USBBus *bus, USBPort *port, void *opaque, int index, |
ace1318b | 508 | USBPortOps *ops, int speedmask); |
f4bbaaf5 MA |
509 | void usb_register_companion(const char *masterbus, USBPort *ports[], |
510 | uint32_t portcount, uint32_t firstport, | |
511 | void *opaque, USBPortOps *ops, int speedmask, | |
512 | Error **errp); | |
c7a2196a | 513 | void usb_port_location(USBPort *downstream, USBPort *upstream, int portnr); |
a8e662b5 | 514 | void usb_unregister_port(USBBus *bus, USBPort *port); |
7d553f27 | 515 | void usb_claim_port(USBDevice *dev, Error **errp); |
891fb2cd | 516 | void usb_release_port(USBDevice *dev); |
7d553f27 | 517 | void usb_device_attach(USBDevice *dev, Error **errp); |
a8e662b5 | 518 | int usb_device_detach(USBDevice *dev); |
594a5360 | 519 | void usb_check_attach(USBDevice *dev, Error **errp); |
a5d2f727 GH |
520 | |
521 | static inline USBBus *usb_bus_from_device(USBDevice *d) | |
522 | { | |
523 | return DO_UPCAST(USBBus, qbus, d->qdev.parent_bus); | |
524 | } | |
701a8f76 PB |
525 | |
526 | extern const VMStateDescription vmstate_usb_device; | |
527 | ||
528 | #define VMSTATE_USB_DEVICE(_field, _state) { \ | |
529 | .name = (stringify(_field)), \ | |
530 | .size = sizeof(USBDevice), \ | |
531 | .vmsd = &vmstate_usb_device, \ | |
532 | .flags = VMS_STRUCT, \ | |
533 | .offset = vmstate_offset_value(_state, _field, USBDevice), \ | |
534 | } | |
535 | ||
73796fe6 GH |
536 | USBDevice *usb_device_find_device(USBDevice *dev, uint8_t addr); |
537 | ||
62aed765 AL |
538 | void usb_device_cancel_packet(USBDevice *dev, USBPacket *p); |
539 | ||
540 | void usb_device_handle_attach(USBDevice *dev); | |
541 | ||
542 | void usb_device_handle_reset(USBDevice *dev); | |
543 | ||
9a77a0f5 HG |
544 | void usb_device_handle_control(USBDevice *dev, USBPacket *p, int request, |
545 | int val, int index, int length, uint8_t *data); | |
62aed765 | 546 | |
9a77a0f5 | 547 | void usb_device_handle_data(USBDevice *dev, USBPacket *p); |
62aed765 AL |
548 | |
549 | void usb_device_set_interface(USBDevice *dev, int interface, | |
550 | int alt_old, int alt_new); | |
551 | ||
36dfe324 HG |
552 | void usb_device_flush_ep_queue(USBDevice *dev, USBEndpoint *ep); |
553 | ||
f79738b0 HG |
554 | void usb_device_ep_stopped(USBDevice *dev, USBEndpoint *ep); |
555 | ||
3b444ead HG |
556 | int usb_device_alloc_streams(USBDevice *dev, USBEndpoint **eps, int nr_eps, |
557 | int streams); | |
558 | void usb_device_free_streams(USBDevice *dev, USBEndpoint **eps, int nr_eps); | |
559 | ||
62aed765 AL |
560 | const char *usb_device_get_product_desc(USBDevice *dev); |
561 | ||
562 | const USBDesc *usb_device_get_usb_desc(USBDevice *dev); | |
563 | ||
b2d1fe67 HG |
564 | /* quirks.c */ |
565 | ||
566 | /* In bulk endpoints are streaming data sources (iow behave like isoc eps) */ | |
567 | #define USB_QUIRK_BUFFER_BULK_IN 0x01 | |
568 | /* Bulk pkts in FTDI format, need special handling when combining packets */ | |
569 | #define USB_QUIRK_IS_FTDI 0x02 | |
701a8f76 | 570 | |
b2d1fe67 HG |
571 | int usb_get_quirks(uint16_t vendor_id, uint16_t product_id, |
572 | uint8_t interface_class, uint8_t interface_subclass, | |
573 | uint8_t interface_protocol); | |
574 | ||
0f6dba14 GH |
575 | /* pcap.c */ |
576 | void usb_pcap_init(FILE *fp); | |
577 | void usb_pcap_ctrl(USBPacket *p, bool setup); | |
578 | void usb_pcap_data(USBPacket *p, bool setup); | |
579 | ||
b2d1fe67 | 580 | #endif |