xref: /openbmc/qemu/hw/usb/pcap.c (revision 7d87775f)
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 
19 struct 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 
29 struct 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 
43 struct 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 
75 static 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 
87 static 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 
96 static 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 
103 static 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 
120 static 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,
130         .flag_setup = setup ? 0 : '-',
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 
165 static 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,
173         .flag_setup = '-',
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 
220 void 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 
233 void 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 
244 void 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 }
254