summaryrefslogtreecommitdiff
path: root/pcap_rw.c
diff options
context:
space:
mode:
Diffstat (limited to 'pcap_rw.c')
-rw-r--r--pcap_rw.c79
1 files changed, 79 insertions, 0 deletions
diff --git a/pcap_rw.c b/pcap_rw.c
new file mode 100644
index 0000000..22d613f
--- /dev/null
+++ b/pcap_rw.c
@@ -0,0 +1,79 @@
+/*
+ * netsniff-ng - the packet sniffing beast
+ * Copyright 2009 - 2013 Daniel Borkmann.
+ * Subject to the GPL, version 2.
+ */
+
+#include <stdio.h>
+#include <stdint.h>
+#include <stdbool.h>
+#include <stdlib.h>
+#include <unistd.h>
+#include <errno.h>
+
+#include "pcap_io.h"
+#include "built_in.h"
+#include "xutils.h"
+#include "xio.h"
+#include "die.h"
+
+static ssize_t pcap_rw_write(int fd, pcap_pkthdr_t *phdr, enum pcap_type type,
+ const uint8_t *packet, size_t len)
+{
+ ssize_t ret, hdrsize = pcap_get_hdr_length(phdr, type), hdrlen = 0;
+
+ ret = write_or_die(fd, &phdr->raw, hdrsize);
+ if (unlikely(ret != hdrsize))
+ panic("Failed to write pkt header!\n");
+
+ hdrlen = pcap_get_length(phdr, type);
+ if (unlikely(hdrlen != len))
+ return -EINVAL;
+
+ ret = write_or_die(fd, packet, hdrlen);
+ if (unlikely(ret != hdrlen))
+ panic("Failed to write pkt payload!\n");
+
+ return hdrsize + hdrlen;
+}
+
+static ssize_t pcap_rw_read(int fd, pcap_pkthdr_t *phdr, enum pcap_type type,
+ uint8_t *packet, size_t len)
+{
+ ssize_t ret, hdrsize = pcap_get_hdr_length(phdr, type), hdrlen = 0;
+
+ ret = read_or_die(fd, &phdr->raw, hdrsize);
+ if (unlikely(ret != hdrsize))
+ return -EIO;
+
+ hdrlen = pcap_get_length(phdr, type);
+ if (unlikely(hdrlen == 0 || hdrlen > len))
+ return -EINVAL;
+
+ ret = read(fd, packet, hdrlen);
+ if (unlikely(ret != hdrlen))
+ return -EIO;
+
+ return hdrsize + hdrlen;
+}
+
+static int pcap_rw_prepare_access(int fd, enum pcap_mode mode, bool jumbo)
+{
+ set_ioprio_rt();
+
+ return 0;
+}
+
+static void pcap_rw_fsync(int fd)
+{
+ fdatasync(fd);
+}
+
+const struct pcap_file_ops pcap_rw_ops = {
+ .pull_fhdr_pcap = pcap_generic_pull_fhdr,
+ .push_fhdr_pcap = pcap_generic_push_fhdr,
+ .prepare_access_pcap = pcap_rw_prepare_access,
+ .read_pcap = pcap_rw_read,
+ .write_pcap = pcap_rw_write,
+ .fsync_pcap = pcap_rw_fsync,
+};