aboutsummaryrefslogtreecommitdiff
path: root/nDPId.c
diff options
context:
space:
mode:
authorToni <matzeton@googlemail.com>2024-08-19 18:33:18 +0200
committerGitHub <noreply@github.com>2024-08-19 18:33:18 +0200
commit5e4005162b804c5501fccf4d066c5b1b99c38b89 (patch)
tree73a090e6c0fd79de4a2d5fc950be8d52185bf905 /nDPId.c
parenta230eaf061e4c570acfa3e2d494baa6c604acc22 (diff)
Add PF_RING support. (#38)
Diffstat (limited to 'nDPId.c')
-rw-r--r--nDPId.c529
1 files changed, 348 insertions, 181 deletions
diff --git a/nDPId.c b/nDPId.c
index 4ae87dca9..ae197e597 100644
--- a/nDPId.c
+++ b/nDPId.c
@@ -32,6 +32,9 @@
#include "config.h"
#include "nDPIsrvd.h"
#include "nio.h"
+#ifdef ENABLE_PFRING
+#include "npfring.h"
+#endif
#include "utils.h"
#ifndef UNIX_PATH_MAX
@@ -264,6 +267,9 @@ struct nDPId_flow
struct nDPId_workflow
{
+#ifdef ENABLE_PFRING
+ struct npfring npf;
+#endif
pcap_t * pcap_handle;
MT_VALUE(error_or_eof, uint8_t);
@@ -483,6 +489,9 @@ static struct
#ifdef ENABLE_EPOLL
uint8_t use_poll;
#endif
+#ifdef ENABLE_PFRING
+ uint8_t use_pfring;
+#endif
/* subopts */
unsigned long long int max_flows_per_thread;
unsigned long long int max_idle_flows_per_thread;
@@ -1252,54 +1261,94 @@ static struct nDPId_workflow * init_workflow(char const * const file_or_device)
MT_INIT2(workflow->error_or_eof, 0);
- errno = 0;
- if (access(file_or_device, R_OK) != 0 && errno == ENOENT)
+#ifdef ENABLE_PFRING
+ if (nDPId_options.use_pfring != 0)
{
- workflow->pcap_handle = pcap_open_live(file_or_device, 65535, 1, 250, pcap_error_buffer);
+ errno = 0;
+
+ if (npfring_init(file_or_device, PFRING_BUFFER_SIZE, &workflow->npf) != 0)
+ {
+ logger_early(1, "PF_RING open device %s failed: %s", file_or_device, strerror(errno));
+ free_workflow(&workflow);
+ return NULL;
+ }
+
+ if (is_cmdarg_set(&nDPId_options.bpf_str) != 0)
+ {
+ if (npfring_set_bpf(&workflow->npf, get_cmdarg(&nDPId_options.bpf_str)) != 0)
+ {
+ logger_early(1, "%s", "PF_RING set bpf filter failed");
+ free_workflow(&workflow);
+ return NULL;
+ }
+ }
}
else
+#endif
{
- workflow->pcap_handle =
- pcap_open_offline_with_tstamp_precision(file_or_device, PCAP_TSTAMP_PRECISION_MICRO, pcap_error_buffer);
- workflow->is_pcap_file = 1;
- }
-
- if (workflow->pcap_handle == NULL)
- {
- logger_early(1,
- (workflow->is_pcap_file == 0 ? "pcap_open_live: %.*s"
- : "pcap_open_offline_with_tstamp_precision: %.*s"),
- (int)PCAP_ERRBUF_SIZE,
- pcap_error_buffer);
- free_workflow(&workflow);
- return NULL;
- }
+ errno = 0;
- if (workflow->is_pcap_file == 0 && pcap_setnonblock(workflow->pcap_handle, 1, pcap_error_buffer) == PCAP_ERROR)
- {
- logger_early(1, "pcap_setnonblock: %.*s", (int)PCAP_ERRBUF_SIZE, pcap_error_buffer);
- free_workflow(&workflow);
- return NULL;
- }
+ if (access(file_or_device, R_OK) != 0 && errno == ENOENT)
+ {
+ workflow->pcap_handle = pcap_open_live(file_or_device, 65535, 1, 250, pcap_error_buffer);
+ }
+ else
+ {
+ workflow->pcap_handle =
+ pcap_open_offline_with_tstamp_precision(file_or_device, PCAP_TSTAMP_PRECISION_MICRO, pcap_error_buffer);
+ workflow->is_pcap_file = 1;
+ }
- if (is_cmdarg_set(&nDPId_options.bpf_str) != 0)
- {
- struct bpf_program fp;
- if (pcap_compile(workflow->pcap_handle, &fp, get_cmdarg(&nDPId_options.bpf_str), 1, PCAP_NETMASK_UNKNOWN) != 0)
+ if (workflow->pcap_handle == NULL)
{
- logger_early(1, "pcap_compile: %s", pcap_geterr(workflow->pcap_handle));
+ logger_early(1,
+ (workflow->is_pcap_file == 0 ? "pcap_open_live: %.*s"
+ : "pcap_open_offline_with_tstamp_precision: %.*s"),
+ (int)PCAP_ERRBUF_SIZE,
+ pcap_error_buffer);
free_workflow(&workflow);
return NULL;
}
- if (pcap_setfilter(workflow->pcap_handle, &fp) != 0)
+
+ if (workflow->is_pcap_file == 0 && pcap_setnonblock(workflow->pcap_handle, 1, pcap_error_buffer) == PCAP_ERROR)
{
- logger_early(1, "pcap_setfilter: %s", pcap_geterr(workflow->pcap_handle));
+ logger_early(1, "pcap_setnonblock: %.*s", (int)PCAP_ERRBUF_SIZE, pcap_error_buffer);
free_workflow(&workflow);
+ return NULL;
+ }
+
+ if (is_cmdarg_set(&nDPId_options.bpf_str) != 0)
+ {
+ struct bpf_program fp;
+ if (pcap_compile(workflow->pcap_handle, &fp, get_cmdarg(&nDPId_options.bpf_str), 1, PCAP_NETMASK_UNKNOWN) !=
+ 0)
+ {
+ logger_early(1, "pcap_compile: %s", pcap_geterr(workflow->pcap_handle));
+ free_workflow(&workflow);
+ return NULL;
+ }
+ if (pcap_setfilter(workflow->pcap_handle, &fp) != 0)
+ {
+ logger_early(1, "pcap_setfilter: %s", pcap_geterr(workflow->pcap_handle));
+ free_workflow(&workflow);
+ pcap_freecode(&fp);
+ return NULL;
+ }
pcap_freecode(&fp);
+ }
+ }
+
+#ifdef ENABLE_PFRING
+ if (nDPId_options.use_pfring != 0)
+ {
+ if (npfring_enable(&workflow->npf) != 0)
+ {
+ logger_early(1, "%s", "Could not enable PF_RING");
+ free_workflow(&workflow);
return NULL;
}
- pcap_freecode(&fp);
}
+#endif
workflow->ndpi_struct = ndpi_init_detection_module(NULL);
if (workflow->ndpi_struct == NULL)
@@ -1489,6 +1538,13 @@ static void free_workflow(struct nDPId_workflow ** const workflow)
return;
}
+#ifdef ENABLE_PFRING
+ if (nDPId_options.use_pfring != 0)
+ {
+ npfring_close(&w->npf);
+ }
+#endif
+
if (w->pcap_handle != NULL)
{
pcap_close(w->pcap_handle);
@@ -2125,6 +2181,33 @@ static void jsonize_daemon(struct nDPId_reader_thread * const reader_thread, enu
case DAEMON_EVENT_SHUTDOWN:
ndpi_serialize_string_uint64(&workflow->ndpi_serializer, "packets-captured", workflow->packets_captured);
ndpi_serialize_string_uint64(&workflow->ndpi_serializer, "packets-processed", workflow->packets_processed);
+#ifdef ENABLE_PFRING
+ {
+ int rc;
+ struct npfring_stats stats = {};
+
+ if (nDPId_options.use_pfring != 0) {
+ if ((rc = npfring_stats(&workflow->npf, &stats)) != 0)
+ {
+ logger(1, "[%8llu] PF_RING stats returned: %d", reader_thread->workflow->packets_processed, rc);
+ }
+ ndpi_serialize_string_boolean(&workflow->ndpi_serializer, "pfring_active", 0);
+ ndpi_serialize_string_uint64(&workflow->ndpi_serializer, "pfring_recv", 0);
+ ndpi_serialize_string_uint64(&workflow->ndpi_serializer, "pfring_drop", 0);
+ ndpi_serialize_string_uint64(&workflow->ndpi_serializer, "pfring_shunt", 0);
+ } else {
+ ndpi_serialize_string_boolean(&workflow->ndpi_serializer, "pfring_active", nDPId_options.use_pfring);
+ ndpi_serialize_string_uint64(&workflow->ndpi_serializer, "pfring_recv", stats.recv);
+ ndpi_serialize_string_uint64(&workflow->ndpi_serializer, "pfring_drop", stats.drop);
+ ndpi_serialize_string_uint64(&workflow->ndpi_serializer, "pfring_shunt", stats.shunt);
+ }
+ }
+#else
+ ndpi_serialize_string_boolean(&workflow->ndpi_serializer, "pfring_active", 0);
+ ndpi_serialize_string_uint64(&workflow->ndpi_serializer, "pfring_recv", 0);
+ ndpi_serialize_string_uint64(&workflow->ndpi_serializer, "pfring_drop", 0);
+ ndpi_serialize_string_uint64(&workflow->ndpi_serializer, "pfring_shunt", 0);
+#endif
ndpi_serialize_string_uint64(&workflow->ndpi_serializer,
"total-skipped-flows",
workflow->total_skipped_flows);
@@ -2669,9 +2752,20 @@ static void jsonize_packet_event(struct nDPId_reader_thread * const reader_threa
get_l4_protocol_idle_time_external(flow_ext->flow_basic.l4_protocol));
}
- ndpi_serialize_string_int32(&workflow->ndpi_serializer,
- "pkt_datalink",
- pcap_datalink(reader_thread->workflow->pcap_handle));
+#ifdef ENABLE_PFRING
+ if (nDPId_options.use_pfring != 0)
+ {
+ ndpi_serialize_string_int32(&workflow->ndpi_serializer,
+ "pkt_datalink",
+ npfring_datalink(&reader_thread->workflow->npf));
+ }
+ else
+#endif
+ {
+ ndpi_serialize_string_int32(&workflow->ndpi_serializer,
+ "pkt_datalink",
+ pcap_datalink(reader_thread->workflow->pcap_handle));
+ }
ndpi_serialize_string_uint32(&workflow->ndpi_serializer, "pkt_caplen", header->caplen);
ndpi_serialize_string_uint32(&workflow->ndpi_serializer, "pkt_type", pkt_type);
ndpi_serialize_string_uint32(&workflow->ndpi_serializer, "pkt_l3_offset", pkt_l3_offset);
@@ -2736,9 +2830,20 @@ static void jsonize_flow_event(struct nDPId_reader_thread * const reader_thread,
case FLOW_EVENT_IDLE:
case FLOW_EVENT_UPDATE:
case FLOW_EVENT_ANALYSE:
- ndpi_serialize_string_int32(&workflow->ndpi_serializer,
- "flow_datalink",
- pcap_datalink(reader_thread->workflow->pcap_handle));
+#ifdef ENABLE_PFRING
+ if (nDPId_options.use_pfring != 0)
+ {
+ ndpi_serialize_string_int32(&workflow->ndpi_serializer,
+ "flow_datalink",
+ npfring_datalink(&reader_thread->workflow->npf));
+ }
+ else
+#endif
+ {
+ ndpi_serialize_string_int32(&workflow->ndpi_serializer,
+ "flow_datalink",
+ pcap_datalink(reader_thread->workflow->pcap_handle));
+ }
ndpi_serialize_string_uint32(&workflow->ndpi_serializer,
"flow_max_packets",
nDPId_options.max_packets_per_flow_to_send);
@@ -3164,9 +3269,20 @@ static int process_datalink_layer(struct nDPId_reader_thread * const reader_thre
uint16_t * layer3_type)
{
const uint16_t eth_offset = 0;
- const int datalink_type = pcap_datalink(reader_thread->workflow->pcap_handle);
+ int datalink_type;
const struct ndpi_ethhdr * ethernet;
+#ifdef ENABLE_PFRING
+ if (nDPId_options.use_pfring != 0)
+ {
+ datalink_type = npfring_datalink(&reader_thread->workflow->npf);
+ }
+ else
+#endif
+ {
+ datalink_type = pcap_datalink(reader_thread->workflow->pcap_handle);
+ }
+
switch (datalink_type)
{
case DLT_NULL:
@@ -4402,168 +4518,208 @@ static void log_all_flows(struct nDPId_reader_thread const * const reader_thread
}
#endif
-static void run_pcap_loop(struct nDPId_reader_thread * const reader_thread)
+static void run_capture_loop(struct nDPId_reader_thread * const reader_thread)
{
- if (reader_thread->workflow != NULL && reader_thread->workflow->pcap_handle != NULL)
+ if (reader_thread->workflow == NULL || (reader_thread->workflow->pcap_handle == NULL
+#ifdef ENABLE_PFRING
+ && reader_thread->workflow->npf.pfring_desc == NULL
+#endif
+ ))
{
- if (reader_thread->workflow->is_pcap_file != 0)
- {
- switch (pcap_loop(reader_thread->workflow->pcap_handle, -1, &ndpi_process_packet, (uint8_t *)reader_thread))
- {
- case PCAP_ERROR:
- logger(1, "Error while reading pcap file: '%s'", pcap_geterr(reader_thread->workflow->pcap_handle));
- MT_GET_AND_ADD(reader_thread->workflow->error_or_eof, 1);
- return;
- case PCAP_ERROR_BREAK:
- MT_GET_AND_ADD(reader_thread->workflow->error_or_eof, 1);
- return;
- default:
- return;
- }
- }
- else
+ return;
+ }
+
+ if (reader_thread->workflow->is_pcap_file != 0)
+ {
+ switch (pcap_loop(reader_thread->workflow->pcap_handle, -1, &ndpi_process_packet, (uint8_t *)reader_thread))
{
-#if !defined(__FreeBSD__) && !defined(__APPLE__)
- sigset_t thread_signal_set, old_signal_set;
- sigfillset(&thread_signal_set);
- if (pthread_sigmask(SIG_BLOCK, &thread_signal_set, &old_signal_set) != 0)
- {
- logger(1, "pthread_sigmask: %s", strerror(errno));
+ case PCAP_ERROR:
+ logger(1, "Error while reading pcap file: '%s'", pcap_geterr(reader_thread->workflow->pcap_handle));
MT_GET_AND_ADD(reader_thread->workflow->error_or_eof, 1);
return;
- }
-
- sigaddset(&thread_signal_set, SIGINT);
- sigaddset(&thread_signal_set, SIGTERM);
- sigaddset(&thread_signal_set, SIGUSR1);
- int signal_fd = signalfd(-1, &thread_signal_set, SFD_NONBLOCK);
- if (signal_fd < 0 || set_fd_cloexec(signal_fd) < 0)
- {
- logger(1, "signalfd: %s", strerror(errno));
+ case PCAP_ERROR_BREAK:
MT_GET_AND_ADD(reader_thread->workflow->error_or_eof, 1);
return;
- }
+ default:
+ return;
+ }
+ }
+ else
+ {
+#if !defined(__FreeBSD__) && !defined(__APPLE__)
+ sigset_t thread_signal_set, old_signal_set;
+ sigfillset(&thread_signal_set);
+ if (pthread_sigmask(SIG_BLOCK, &thread_signal_set, &old_signal_set) != 0)
+ {
+ logger(1, "pthread_sigmask: %s", strerror(errno));
+ MT_GET_AND_ADD(reader_thread->workflow->error_or_eof, 1);
+ return;
+ }
+
+ sigaddset(&thread_signal_set, SIGINT);
+ sigaddset(&thread_signal_set, SIGTERM);
+ sigaddset(&thread_signal_set, SIGUSR1);
+ int signal_fd = signalfd(-1, &thread_signal_set, SFD_NONBLOCK);
+ if (signal_fd < 0 || set_fd_cloexec(signal_fd) < 0)
+ {
+ logger(1, "signalfd: %s", strerror(errno));
+ MT_GET_AND_ADD(reader_thread->workflow->error_or_eof, 1);
+ return;
+ }
#endif
- int pcap_fd = pcap_get_selectable_fd(reader_thread->workflow->pcap_handle);
- if (pcap_fd < 0)
- {
- logger(1, "%s", "Got an invalid PCAP fd");
- MT_GET_AND_ADD(reader_thread->workflow->error_or_eof, 1);
- return;
- }
+ int capture_fd = -1;
+#ifdef ENABLE_PFRING
+ if (nDPId_options.use_pfring != 0)
+ {
+ capture_fd = npfring_get_selectable_fd(&reader_thread->workflow->npf);
+ }
+ else
+#endif
+ {
+ capture_fd = pcap_get_selectable_fd(reader_thread->workflow->pcap_handle);
+ }
+ if (capture_fd < 0)
+ {
+ logger(1,
+ "Got an invalid %s fd",
+ (
+#ifdef ENABLE_PFRING
+ nDPId_options.use_pfring != 0 ? "PF_RING" :
+#endif
+ "PCAP"));
+ MT_GET_AND_ADD(reader_thread->workflow->error_or_eof, 1);
+ return;
+ }
- struct nio io;
- nio_init(&io);
+ struct nio io;
+ nio_init(&io);
#ifdef ENABLE_EPOLL
- if ((nDPId_options.use_poll == 0 && nio_use_epoll(&io, 32) != NIO_SUCCESS) ||
- (nDPId_options.use_poll != 0 && nio_use_poll(&io, nDPIsrvd_MAX_REMOTE_DESCRIPTORS) != NIO_SUCCESS))
+ if ((nDPId_options.use_poll == 0 && nio_use_epoll(&io, 32) != NIO_SUCCESS) ||
+ (nDPId_options.use_poll != 0 && nio_use_poll(&io, nDPIsrvd_MAX_REMOTE_DESCRIPTORS) != NIO_SUCCESS))
#else
- if (nio_use_poll(&io, nDPIsrvd_MAX_REMOTE_DESCRIPTORS) != NIO_SUCCESS)
+ if (nio_use_poll(&io, nDPIsrvd_MAX_REMOTE_DESCRIPTORS) != NIO_SUCCESS)
+#endif
+ {
+ logger(1, "%s", "Event I/O poll/epoll setup failed");
+ MT_GET_AND_ADD(reader_thread->workflow->error_or_eof, 1);
+ nio_free(&io);
+ return;
+ }
+
+ errno = 0;
+ if (nio_add_fd(&io, capture_fd, NIO_EVENT_INPUT, NULL) != NIO_SUCCESS)
+ {
+ logger(1, "Could not add pcap fd to event queue: %s", (errno != 0 ? strerror(errno) : "Internal Error"));
+ MT_GET_AND_ADD(reader_thread->workflow->error_or_eof, 1);
+ nio_free(&io);
+ return;
+ }
+#if !defined(__FreeBSD__) && !defined(__APPLE__)
+ errno = 0;
+ if (nio_add_fd(&io, signal_fd, NIO_EVENT_INPUT, NULL) != NIO_SUCCESS)
+ {
+ logger(1, "Could not add signal fd to event queue: %s", (errno != 0 ? strerror(errno) : "Internal Error"));
+ MT_GET_AND_ADD(reader_thread->workflow->error_or_eof, 1);
+ nio_free(&io);
+ return;
+ }
#endif
- {
- logger(1, "%s", "Event I/O poll/epoll setup failed");
- MT_GET_AND_ADD(reader_thread->workflow->error_or_eof, 1);
- nio_free(&io);
- return;
- }
+ int const timeout_ms = 1000; /* TODO: Configurable? */
+ struct timeval tval_before_epoll, tval_after_epoll;
+ while (MT_GET_AND_ADD(nDPId_main_thread_shutdown, 0) == 0 && processing_threads_error_or_eof() == 0)
+ {
+ get_current_time(&tval_before_epoll);
errno = 0;
- if (nio_add_fd(&io, pcap_fd, NIO_EVENT_INPUT, NULL) != NIO_SUCCESS)
+ if (nio_run(&io, timeout_ms) != NIO_SUCCESS)
{
- logger(1,
- "Could not add pcap fd to event queue: %s",
- (errno != 0 ? strerror(errno) : "Internal Error"));
+ logger(1, "Event I/O returned error: %s", strerror(errno));
MT_GET_AND_ADD(reader_thread->workflow->error_or_eof, 1);
- nio_free(&io);
- return;
+ break;
}
-#if !defined(__FreeBSD__) && !defined(__APPLE__)
- errno = 0;
- if (nio_add_fd(&io, signal_fd, NIO_EVENT_INPUT, NULL) != NIO_SUCCESS)
+
+ int nready = nio_get_nready(&io);
+
+ if (nready == 0)
{
- logger(1,
- "Could not add signal fd to event queue: %s",
- (errno != 0 ? strerror(errno) : "Internal Error"));
- MT_GET_AND_ADD(reader_thread->workflow->error_or_eof, 1);
- nio_free(&io);
- return;
+ struct timeval tval_diff;
+ get_current_time(&tval_after_epoll);
+ ndpi_timer_sub(&tval_after_epoll, &tval_before_epoll, &tval_diff);
+ uint64_t tdiff_us = tval_diff.tv_sec * 1000 * 1000 + tval_diff.tv_usec;
+
+ reader_thread->workflow->last_global_time += tdiff_us;
+ reader_thread->workflow->last_thread_time += tdiff_us;
+
+ do_periodically_work(reader_thread);
}
-#endif
- int const timeout_ms = 1000; /* TODO: Configurable? */
- struct timeval tval_before_epoll, tval_after_epoll;
- while (MT_GET_AND_ADD(nDPId_main_thread_shutdown, 0) == 0 && processing_threads_error_or_eof() == 0)
+ for (int i = 0; i < nready; ++i)
{
- get_current_time(&tval_before_epoll);
- errno = 0;
- if (nio_run(&io, timeout_ms) != NIO_SUCCESS)
+ if (nio_has_error(&io, i) == NIO_SUCCESS)
{
- logger(1, "Event I/O returned error: %s", strerror(errno));
+ logger(1, "%s", "Event I/O error");
MT_GET_AND_ADD(reader_thread->workflow->error_or_eof, 1);
- break;
}
- int nready = nio_get_nready(&io);
-
- if (nready == 0)
- {
- struct timeval tval_diff;
- get_current_time(&tval_after_epoll);
- ndpi_timer_sub(&tval_after_epoll, &tval_before_epoll, &tval_diff);
- uint64_t tdiff_us = tval_diff.tv_sec * 1000 * 1000 + tval_diff.tv_usec;
-
- reader_thread->workflow->last_global_time += tdiff_us;
- reader_thread->workflow->last_thread_time += tdiff_us;
+ int fd = nio_get_fd(&io, i);
- do_periodically_work(reader_thread);
- }
-
- for (int i = 0; i < nready; ++i)
+#if !defined(__FreeBSD__) && !defined(__APPLE__)
+ if (fd == signal_fd)
{
- if (nio_has_error(&io, i) == NIO_SUCCESS)
+ struct signalfd_siginfo fdsi;
+ if (read(signal_fd, &fdsi, sizeof(fdsi)) != sizeof(fdsi))
{
- logger(1, "%s", "Event I/O error");
- MT_GET_AND_ADD(reader_thread->workflow->error_or_eof, 1);
+ if (errno != EAGAIN)
+ {
+ logger(1, "Could not read signal data from fd %d: %s", signal_fd, strerror(errno));
+ }
}
-
- int fd = nio_get_fd(&io, i);
-
-#if !defined(__FreeBSD__) && !defined(__APPLE__)
- if (fd == signal_fd)
+ else
{
- struct signalfd_siginfo fdsi;
- if (read(signal_fd, &fdsi, sizeof(fdsi)) != sizeof(fdsi))
+ char const * signame = "unknown";
+ switch (fdsi.ssi_signo)
{
- if (errno != EAGAIN)
- {
- logger(1, "Could not read signal data from fd %d: %s", signal_fd, strerror(errno));
- }
+ case SIGINT:
+ signame = "SIGINT";
+ sighandler(SIGINT);
+ break;
+ case SIGTERM:
+ signame = "SIGTERM";
+ sighandler(SIGTERM);
+ break;
+ case SIGUSR1:
+ signame = "SIGUSR1";
+ log_all_flows(reader_thread);
+ break;
}
- else
+ logger(1, "Received signal %d (%s)", fdsi.ssi_signo, signame);
+ }
+ }
+ else
+#endif
+ if (fd == capture_fd)
+ {
+#ifdef ENABLE_PFRING
+ if (nDPId_options.use_pfring != 0)
+ {
+ struct pcap_pkthdr hdr;
+
+ int rc = npfring_recv(&reader_thread->workflow->npf, &hdr);
+ if (rc == 0)
{
- char const * signame = "unknown";
- switch (fdsi.ssi_signo)
- {
- case SIGINT:
- signame = "SIGINT";
- sighandler(SIGINT);
- break;
- case SIGTERM:
- signame = "SIGTERM";
- sighandler(SIGTERM);
- break;
- case SIGUSR1:
- signame = "SIGUSR1";
- log_all_flows(reader_thread);
- break;
- }
- logger(1, "Received signal %d (%s)", fdsi.ssi_signo, signame);
+ logger(1, "Error while reading packets from PF_RING: %d", rc);
+ MT_GET_AND_ADD(reader_thread->workflow->error_or_eof, 1);
+ nio_free(&io);
+ return;
}
+
+ ndpi_process_packet((uint8_t *)reader_thread,
+ &hdr,
+ &reader_thread->workflow->npf.pfring_buffer[0]);
}
else
#endif
- if (fd == pcap_fd)
{
switch (pcap_dispatch(
reader_thread->workflow->pcap_handle, -1, ndpi_process_packet, (uint8_t *)reader_thread))
@@ -4582,15 +4738,15 @@ static void run_pcap_loop(struct nDPId_reader_thread * const reader_thread)
break;
}
}
- else
- {
- logger(1, "Unknown event descriptor or data returned: %p", nio_get_ptr(&io, i));
- }
+ }
+ else
+ {
+ logger(1, "Unknown event descriptor or data returned: %p", nio_get_ptr(&io, i));
}
}
-
- nio_free(&io);
}
+
+ nio_free(&io);
}
}
@@ -4622,7 +4778,7 @@ static void * processing_thread(void * const ndpi_thread_arg)
jsonize_daemon(reader_thread, DAEMON_EVENT_INIT);
}
- run_pcap_loop(reader_thread);
+ run_capture_loop(reader_thread);
set_collector_block(reader_thread);
MT_GET_AND_ADD(reader_thread->workflow->error_or_eof, 1);
return NULL;
@@ -4975,6 +5131,9 @@ static void print_usage(char const * const arg0)
"\t \t"
"[-v] [-h]\n\n"
"\t-i\tInterface or file from where to read packets from.\n"
+#ifdef ENABLE_PFRING
+ "\t-r\tUse PFRING to capture packets instead of libpcap.\n"
+#endif
"\t-I\tProcess only packets where the source address of the first packet\n"
"\t \tis part of the interface subnet. (Internal mode)\n"
"\t-E\tProcess only packets where the source address of the first packet\n"
@@ -5022,38 +5181,46 @@ static void print_usage(char const * const arg0)
static void nDPId_print_deps_version(FILE * const out)
{
fprintf(out,
- "------------------------------------------------------\n"
+ "-------------------------------------------------------\n"
#ifdef LIBNDPI_STATIC
- "nDPI version: %s (statically linked)\n"
+ "nDPI version...: %s (statically linked)\n"
#else
- "nDPI version: %s\n"
+ "nDPI version...: %s\n"
#endif
- " API version: %u\n"
- "pcap version: %s\n"
- "------------------------------------------------------\n",
+ " API version...: %u\n"
+ "pcap version...: %s\n",
ndpi_revision(),
ndpi_get_api_version(),
pcap_lib_version() + strlen("libpcap version "));
if (ndpi_get_gcrypt_version() != NULL)
{
- fprintf(out,
- "gcrypt version: %s\n"
- "----------------------------------\n",
- ndpi_get_gcrypt_version());
+ fprintf(out, "gcrypt version.: %s\n", ndpi_get_gcrypt_version());
}
+#ifdef ENABLE_PFRING
+ npfring_print_version(out);
+#endif
+ fprintf(out, "%s", "-------------------------------------------------------\n");
}
static int nDPId_parse_options(int argc, char ** argv)
{
int opt;
- while ((opt = getopt(argc, argv, "i:IEB:lL:c:edp:u:g:P:C:J:S:a:Azo:vh")) != -1)
+ while ((opt = getopt(argc, argv, "i:rIEB:lL:c:edp:u:g:P:C:J:S:a:Azo:vh")) != -1)
{
switch (opt)
{
case 'i':
set_cmdarg(&nDPId_options.pcap_file_or_interface, optarg);
break;
+ case 'r':
+#ifdef ENABLE_PFRING
+ nDPId_options.use_pfring = 1;
+ break;
+#else
+ logger_early(1, "%s", "nDPId was built w/o PFRING support");
+ return 1;
+#endif
case 'I':
nDPId_options.process_internal_initial_direction = 1;
break;