tests: Long P2P_LISTEN and offchannel TX
[mech_eap.git] / wlantest / readpcap.c
index d96345c..6c3853b 100644 (file)
@@ -2,23 +2,50 @@
  * PCAP capture file reader
  * Copyright (c) 2010, Jouni Malinen <j@w1.fi>
  *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License version 2 as
- * published by the Free Software Foundation.
- *
- * Alternatively, this software may be distributed under the terms of BSD
- * license.
- *
- * See README and COPYING for more details.
+ * This software may be distributed under the terms of the BSD license.
+ * See README for more details.
  */
 
 #include "utils/includes.h"
-#include <pcap/pcap.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];
@@ -27,6 +54,95 @@ int read_cap_file(struct wlantest *wt, const char *fname)
        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) {
@@ -62,7 +178,7 @@ int read_cap_file(struct wlantest *wt, const char *fname)
                        continue;
                }
                count++;
-               wlantest_process(wt, data, hdr->caplen);
+               wlantest_process_wired(wt, data, hdr->caplen);
        }
 
        pcap_close(pcap);