qemu

Форк
0
/
pcap.c 
253 строки · 7.1 Кб
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

Использование cookies

Мы используем файлы cookie в соответствии с Политикой конфиденциальности и Политикой использования cookies.

Нажимая кнопку «Принимаю», Вы даете АО «СберТех» согласие на обработку Ваших персональных данных в целях совершенствования нашего веб-сайта и Сервиса GitVerse, а также повышения удобства их использования.

Запретить использование cookies Вы можете самостоятельно в настройках Вашего браузера.