xref: /openbmc/qemu/hw/usb/pcap.c (revision 9f34101db00eabd8f424e98b481c2394e6509198)
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