]> git.proxmox.com Git - mirror_qemu.git/blame - hw/usb/pcap.c
virtio-sound: handle VIRTIO_SND_R_PCM_RELEASE
[mirror_qemu.git] / hw / usb / pcap.c
CommitLineData
0f6dba14
GH
1/*
2 * usb packet capture
3 *
4 * Copyright (c) 2021 Gerd Hoffmann <kraxel@redhat.com>
5 *
6 * This work is licensed under the terms of the GNU GPL, version 2 or later.
7 * See the COPYING file in the top-level directory.
8 */
9
10#include "qemu/osdep.h"
11#include "hw/usb.h"
12
13#define PCAP_MAGIC 0xa1b2c3d4
14#define PCAP_MAJOR 2
15#define PCAP_MINOR 4
16
17/* https://wiki.wireshark.org/Development/LibpcapFileFormat */
18
19struct pcap_hdr {
20 uint32_t magic_number; /* magic number */
21 uint16_t version_major; /* major version number */
22 uint16_t version_minor; /* minor version number */
23 int32_t thiszone; /* GMT to local correction */
24 uint32_t sigfigs; /* accuracy of timestamps */
25 uint32_t snaplen; /* max length of captured packets, in octets */
26 uint32_t network; /* data link type */
27};
28
29struct pcaprec_hdr {
30 uint32_t ts_sec; /* timestamp seconds */
31 uint32_t ts_usec; /* timestamp microseconds */
32 uint32_t incl_len; /* number of octets of packet saved in file */
33 uint32_t orig_len; /* actual length of packet */
34};
35
36/* https://www.tcpdump.org/linktypes.html */
37/* linux: Documentation/usb/usbmon.rst */
38/* linux: drivers/usb/mon/mon_bin.c */
39
40#define LINKTYPE_USB_LINUX 189 /* first 48 bytes only */
41#define LINKTYPE_USB_LINUX_MMAPPED 220 /* full 64 byte header */
42
43struct usbmon_packet {
44 uint64_t id; /* 0: URB ID - from submission to callback */
45 unsigned char type; /* 8: Same as text; extensible. */
46 unsigned char xfer_type; /* ISO (0), Intr, Control, Bulk (3) */
47 unsigned char epnum; /* Endpoint number and transfer direction */
48 unsigned char devnum; /* Device address */
49 uint16_t busnum; /* 12: Bus number */
50 char flag_setup; /* 14: Same as text */
51 char flag_data; /* 15: Same as text; Binary zero is OK. */
52 int64_t ts_sec; /* 16: gettimeofday */
53 int32_t ts_usec; /* 24: gettimeofday */
54 int32_t status; /* 28: */
55 unsigned int length; /* 32: Length of data (submitted or actual) */
56 unsigned int len_cap; /* 36: Delivered length */
57 union { /* 40: */
58 unsigned char setup[8]; /* Only for Control S-type */
59 struct iso_rec { /* Only for ISO */
60 int32_t error_count;
61 int32_t numdesc;
62 } iso;
63 } s;
64 int32_t interval; /* 48: Only for Interrupt and ISO */
65 int32_t start_frame; /* 52: For ISO */
66 uint32_t xfer_flags; /* 56: copy of URB's transfer_flags */
67 uint32_t ndesc; /* 60: Actual number of ISO descriptors */
68}; /* 64 total length */
69
70/* ------------------------------------------------------------------------ */
71
72#define CTRL_LEN 4096
73#define DATA_LEN 256
74
75static int usbmon_status(USBPacket *p)
76{
77 switch (p->status) {
78 case USB_RET_SUCCESS:
79 return 0;
80 case USB_RET_NODEV:
81 return -19; /* -ENODEV */
82 default:
83 return -121; /* -EREMOTEIO */
84 }
85}
86
87static unsigned int usbmon_epnum(USBPacket *p)
88{
89 unsigned epnum = 0;
90
91 epnum |= p->ep->nr;
92 epnum |= (p->pid == USB_TOKEN_IN) ? 0x80 : 0;
93 return epnum;
94}
95
96static unsigned char usbmon_xfer_type[] = {
97 [USB_ENDPOINT_XFER_CONTROL] = 2,
98 [USB_ENDPOINT_XFER_ISOC] = 0,
99 [USB_ENDPOINT_XFER_BULK] = 3,
100 [USB_ENDPOINT_XFER_INT] = 1,
101};
102
103static void do_usb_pcap_header(FILE *fp, struct usbmon_packet *packet)
104{
105 struct pcaprec_hdr header;
106 struct timeval tv;
107
108 gettimeofday(&tv, NULL);
109 packet->ts_sec = tv.tv_sec;
110 packet->ts_usec = tv.tv_usec;
111
112 header.ts_sec = packet->ts_sec;
113 header.ts_usec = packet->ts_usec;
114 header.incl_len = packet->len_cap;
115 header.orig_len = packet->length + sizeof(*packet);
116 fwrite(&header, sizeof(header), 1, fp);
117 fwrite(packet, sizeof(*packet), 1, fp);
118}
119
120static void do_usb_pcap_ctrl(FILE *fp, USBPacket *p, bool setup)
121{
122 USBDevice *dev = p->ep->dev;
123 bool in = dev->setup_buf[0] & USB_DIR_IN;
124 struct usbmon_packet packet = {
125 .id = 0,
126 .type = setup ? 'S' : 'C',
127 .xfer_type = usbmon_xfer_type[USB_ENDPOINT_XFER_CONTROL],
128 .epnum = in ? 0x80 : 0,
129 .devnum = dev->addr,
6ba5a437 130 .flag_setup = setup ? 0 : '-',
0f6dba14
GH
131 .flag_data = '=',
132 .length = dev->setup_len,
133 };
134 int data_len = dev->setup_len;
135
136 if (data_len > CTRL_LEN) {
137 data_len = CTRL_LEN;
138 }
139 if (setup) {
140 memcpy(packet.s.setup, dev->setup_buf, 8);
141 } else {
142 packet.status = usbmon_status(p);
143 }
144
145 if (in && setup) {
146 packet.flag_data = '<';
147 packet.length = 0;
148 data_len = 0;
149 }
150 if (!in && !setup) {
151 packet.flag_data = '>';
152 packet.length = 0;
153 data_len = 0;
154 }
155
156 packet.len_cap = data_len + sizeof(packet);
157 do_usb_pcap_header(fp, &packet);
158 if (data_len) {
159 fwrite(dev->data_buf, data_len, 1, fp);
160 }
161
162 fflush(fp);
163}
164
165static void do_usb_pcap_data(FILE *fp, USBPacket *p, bool setup)
166{
167 struct usbmon_packet packet = {
168 .id = p->id,
169 .type = setup ? 'S' : 'C',
170 .xfer_type = usbmon_xfer_type[p->ep->type],
171 .epnum = usbmon_epnum(p),
172 .devnum = p->ep->dev->addr,
6ba5a437 173 .flag_setup = '-',
0f6dba14
GH
174 .flag_data = '=',
175 .length = p->iov.size,
176 };
177 int data_len = p->iov.size;
178
179 if (p->ep->nr == 0) {
180 /* ignore control pipe packets */
181 return;
182 }
183
184 if (data_len > DATA_LEN) {
185 data_len = DATA_LEN;
186 }
187 if (!setup) {
188 packet.status = usbmon_status(p);
189 if (packet.length > p->actual_length) {
190 packet.length = p->actual_length;
191 }
192 if (data_len > p->actual_length) {
193 data_len = p->actual_length;
194 }
195 }
196
197 if (p->pid == USB_TOKEN_IN && setup) {
198 packet.flag_data = '<';
199 packet.length = 0;
200 data_len = 0;
201 }
202 if (p->pid == USB_TOKEN_OUT && !setup) {
203 packet.flag_data = '>';
204 packet.length = 0;
205 data_len = 0;
206 }
207
208 packet.len_cap = data_len + sizeof(packet);
209 do_usb_pcap_header(fp, &packet);
210 if (data_len) {
211 void *buf = g_malloc(data_len);
212 iov_to_buf(p->iov.iov, p->iov.niov, 0, buf, data_len);
213 fwrite(buf, data_len, 1, fp);
214 g_free(buf);
215 }
216
217 fflush(fp);
218}
219
220void usb_pcap_init(FILE *fp)
221{
222 struct pcap_hdr header = {
223 .magic_number = PCAP_MAGIC,
224 .version_major = 2,
225 .version_minor = 4,
226 .snaplen = MAX(CTRL_LEN, DATA_LEN) + sizeof(struct usbmon_packet),
227 .network = LINKTYPE_USB_LINUX_MMAPPED,
228 };
229
230 fwrite(&header, sizeof(header), 1, fp);
231}
232
233void usb_pcap_ctrl(USBPacket *p, bool setup)
234{
235 FILE *fp = p->ep->dev->pcap;
236
237 if (!fp) {
238 return;
239 }
240
241 do_usb_pcap_ctrl(fp, p, setup);
242}
243
244void usb_pcap_data(USBPacket *p, bool setup)
245{
246 FILE *fp = p->ep->dev->pcap;
247
248 if (!fp) {
249 return;
250 }
251
252 do_usb_pcap_data(fp, p, setup);
253}