123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189 |
- /*
- * PCAP capture file reader
- * Copyright (c) 2010, Jouni Malinen <j@w1.fi>
- *
- * This software may be distributed under the terms of the BSD license.
- * See README for more details.
- */
- #include "utils/includes.h"
- #include <pcap.h>
- #include "utils/common.h"
- #include "wlantest.h"
- static void write_pcap_with_radiotap(struct wlantest *wt,
- const u8 *data, size_t data_len)
- {
- struct pcap_pkthdr h;
- u8 rtap[] = {
- 0x00 /* rev */,
- 0x00 /* pad */,
- 0x0a, 0x00, /* header len */
- 0x02, 0x00, 0x00, 0x00, /* present flags */
- 0x00, /* flags */
- 0x00 /* pad */
- };
- u8 *buf;
- size_t len;
- if (wt->assume_fcs)
- rtap[8] |= 0x10;
- os_memset(&h, 0, sizeof(h));
- h.ts = wt->write_pcap_time;
- len = sizeof(rtap) + data_len;
- buf = os_malloc(len);
- if (buf == NULL)
- return;
- os_memcpy(buf, rtap, sizeof(rtap));
- os_memcpy(buf + sizeof(rtap), data, data_len);
- h.caplen = len;
- h.len = len;
- pcap_dump(wt->write_pcap_dumper, &h, buf);
- os_free(buf);
- }
- int read_cap_file(struct wlantest *wt, const char *fname)
- {
- char errbuf[PCAP_ERRBUF_SIZE];
- pcap_t *pcap;
- unsigned int count = 0;
- struct pcap_pkthdr *hdr;
- const u_char *data;
- int res;
- int dlt;
- pcap = pcap_open_offline(fname, errbuf);
- if (pcap == NULL) {
- wpa_printf(MSG_ERROR, "Failed to read pcap file '%s': %s",
- fname, errbuf);
- return -1;
- }
- dlt = pcap_datalink(pcap);
- if (dlt != DLT_IEEE802_11_RADIO && dlt != DLT_PRISM_HEADER &&
- dlt != DLT_IEEE802_11) {
- wpa_printf(MSG_ERROR, "Unsupported pcap datalink type: %d",
- dlt);
- pcap_close(pcap);
- return -1;
- }
- wpa_printf(MSG_DEBUG, "pcap datalink type: %d", dlt);
- for (;;) {
- clear_notes(wt);
- os_free(wt->decrypted);
- wt->decrypted = NULL;
- res = pcap_next_ex(pcap, &hdr, &data);
- if (res == -2)
- break; /* No more packets */
- if (res == -1) {
- wpa_printf(MSG_INFO, "pcap_next_ex failure: %s",
- pcap_geterr(pcap));
- break;
- }
- if (res != 1) {
- wpa_printf(MSG_INFO, "Unexpected pcap_next_ex return "
- "value %d", res);
- break;
- }
- /* Packet was read without problems */
- wpa_printf(MSG_EXCESSIVE, "pcap hdr: ts=%d.%06d "
- "len=%u/%u",
- (int) hdr->ts.tv_sec, (int) hdr->ts.tv_usec,
- hdr->caplen, hdr->len);
- if (wt->write_pcap_dumper) {
- wt->write_pcap_time = hdr->ts;
- if (dlt == DLT_IEEE802_11)
- write_pcap_with_radiotap(wt, data, hdr->caplen);
- else
- pcap_dump(wt->write_pcap_dumper, hdr, data);
- if (wt->pcap_no_buffer)
- pcap_dump_flush(wt->write_pcap_dumper);
- }
- if (hdr->caplen < hdr->len) {
- add_note(wt, MSG_DEBUG, "pcap: Dropped incomplete "
- "frame (%u/%u captured)",
- hdr->caplen, hdr->len);
- write_pcapng_write_read(wt, dlt, hdr, data);
- continue;
- }
- count++;
- switch (dlt) {
- case DLT_IEEE802_11_RADIO:
- wlantest_process(wt, data, hdr->caplen);
- break;
- case DLT_PRISM_HEADER:
- wlantest_process_prism(wt, data, hdr->caplen);
- break;
- case DLT_IEEE802_11:
- wlantest_process_80211(wt, data, hdr->caplen);
- break;
- }
- write_pcapng_write_read(wt, dlt, hdr, data);
- }
- pcap_close(pcap);
- wpa_printf(MSG_DEBUG, "Read %s: %u packets", fname, count);
- return 0;
- }
- int read_wired_cap_file(struct wlantest *wt, const char *fname)
- {
- char errbuf[PCAP_ERRBUF_SIZE];
- pcap_t *pcap;
- unsigned int count = 0;
- struct pcap_pkthdr *hdr;
- const u_char *data;
- int res;
- pcap = pcap_open_offline(fname, errbuf);
- if (pcap == NULL) {
- wpa_printf(MSG_ERROR, "Failed to read pcap file '%s': %s",
- fname, errbuf);
- return -1;
- }
- for (;;) {
- res = pcap_next_ex(pcap, &hdr, &data);
- if (res == -2)
- break; /* No more packets */
- if (res == -1) {
- wpa_printf(MSG_INFO, "pcap_next_ex failure: %s",
- pcap_geterr(pcap));
- break;
- }
- if (res != 1) {
- wpa_printf(MSG_INFO, "Unexpected pcap_next_ex return "
- "value %d", res);
- break;
- }
- /* Packet was read without problems */
- wpa_printf(MSG_EXCESSIVE, "pcap hdr: ts=%d.%06d "
- "len=%u/%u",
- (int) hdr->ts.tv_sec, (int) hdr->ts.tv_usec,
- hdr->caplen, hdr->len);
- if (hdr->caplen < hdr->len) {
- wpa_printf(MSG_DEBUG, "pcap: Dropped incomplete frame "
- "(%u/%u captured)",
- hdr->caplen, hdr->len);
- continue;
- }
- count++;
- wlantest_process_wired(wt, data, hdr->caplen);
- }
- pcap_close(pcap);
- wpa_printf(MSG_DEBUG, "Read %s: %u packets", fname, count);
- return 0;
- }
|