2022-06-13 02:03:52 +01:00
|
|
|
// Copyright (c) 2022 Cesanta Software Limited
|
|
|
|
// All rights reserved
|
|
|
|
//
|
|
|
|
// SNTP example using MIP and pcap driver
|
|
|
|
|
|
|
|
#include <pcap.h>
|
|
|
|
#include "mongoose.h"
|
|
|
|
|
2022-09-07 13:35:05 +01:00
|
|
|
#define MQTT_URL "mqtt://broker.hivemq.com:1883" // MQTT broker to connect to
|
|
|
|
#define MQTT_TOPIC "t/123" // Topic to subscribe to
|
2022-06-13 02:03:52 +01:00
|
|
|
|
|
|
|
static int s_signo;
|
|
|
|
void signal_handler(int signo) {
|
|
|
|
s_signo = signo;
|
|
|
|
}
|
|
|
|
|
2022-09-07 13:35:05 +01:00
|
|
|
static size_t pcap_tx(const void *buf, size_t len, void *userdata) {
|
|
|
|
int res = pcap_inject((pcap_t *) userdata, buf, len);
|
2022-06-13 02:03:52 +01:00
|
|
|
if (res == PCAP_ERROR) {
|
|
|
|
MG_ERROR(("pcap_inject: %d", res));
|
|
|
|
}
|
2022-09-07 13:35:05 +01:00
|
|
|
return res == PCAP_ERROR ? 0 : len;
|
2022-06-13 02:03:52 +01:00
|
|
|
}
|
|
|
|
|
2022-08-31 08:44:05 +01:00
|
|
|
static bool pcap_up(void *userdata) {
|
2022-06-13 02:03:52 +01:00
|
|
|
return userdata ? true : false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static size_t pcap_rx(void *buf, size_t len, void *userdata) {
|
|
|
|
size_t received = 0;
|
|
|
|
struct pcap_pkthdr *hdr = NULL;
|
|
|
|
const unsigned char *pkt = NULL;
|
2022-09-07 20:48:23 +01:00
|
|
|
|
|
|
|
// To avoid busy-loop and 100% CPU time, wait on pcap for some time
|
|
|
|
int fd = pcap_get_selectable_fd((pcap_t *) userdata); // Pcap file descriptor
|
|
|
|
struct timeval tv = {0, 50000}; // 50 ms
|
|
|
|
fd_set rset;
|
|
|
|
FD_ZERO(&rset);
|
|
|
|
FD_SET(fd, &rset);
|
|
|
|
if (select(fd + 1, &rset, 0, 0, &tv) == 1 && // Have data ?
|
|
|
|
pcap_next_ex((pcap_t *) userdata, &hdr, &pkt) == 1) { // Yes, read
|
2022-06-13 02:03:52 +01:00
|
|
|
received = hdr->len < len ? hdr->len : len;
|
|
|
|
memcpy(buf, pkt, received);
|
|
|
|
}
|
2022-09-07 13:35:05 +01:00
|
|
|
|
2022-09-07 20:48:23 +01:00
|
|
|
return received;
|
2022-09-07 13:35:05 +01:00
|
|
|
}
|
|
|
|
|
2022-06-13 02:03:52 +01:00
|
|
|
int main(int argc, char *argv[]) {
|
|
|
|
const char *iface = "lo0"; // Network iface
|
2022-09-07 13:35:05 +01:00
|
|
|
const char *mac = "00:00:01:02:03:77"; // MAC address
|
2022-06-13 02:03:52 +01:00
|
|
|
const char *bpf = NULL; // "host x.x.x.x or ether host ff:ff:ff:ff:ff:ff";
|
|
|
|
char errbuf[PCAP_ERRBUF_SIZE] = "";
|
|
|
|
|
|
|
|
// Parse options
|
|
|
|
for (int i = 1; i < argc; i++) {
|
|
|
|
if (strcmp(argv[i], "-i") == 0 && i + 1 < argc) {
|
|
|
|
iface = argv[++i];
|
|
|
|
} else if (strcmp(argv[i], "-mac") == 0 && i + 1 < argc) {
|
|
|
|
mac = argv[++i];
|
|
|
|
} else if (strcmp(argv[i], "-bpf") == 0 && i + 1 < argc) {
|
|
|
|
bpf = argv[++i];
|
|
|
|
} else {
|
2022-09-07 13:35:05 +01:00
|
|
|
MG_ERROR(("unknown option %s", argv[i]));
|
|
|
|
return EXIT_FAILURE;
|
2022-06-13 02:03:52 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Open network interface
|
|
|
|
pcap_t *ph = pcap_open_live(iface, 0xffff, 1, 1, errbuf);
|
|
|
|
if (ph == NULL) {
|
|
|
|
pcap_if_t *devs, *d;
|
|
|
|
MG_ERROR(("Failed to open interface %s. Available interfaces:", iface));
|
|
|
|
if (pcap_findalldevs(&devs, errbuf) == 0) {
|
|
|
|
for (d = devs; d != NULL; d = d->next) {
|
|
|
|
MG_ERROR(("%s (%s)", d->name, d->description ? d->description : ""));
|
|
|
|
}
|
|
|
|
pcap_freealldevs(devs);
|
|
|
|
}
|
2022-09-07 13:35:05 +01:00
|
|
|
return EXIT_FAILURE;
|
2022-06-13 02:03:52 +01:00
|
|
|
}
|
|
|
|
pcap_setnonblock(ph, 1, errbuf);
|
2022-09-07 13:35:05 +01:00
|
|
|
pcap_setdirection(ph, PCAP_D_IN);
|
|
|
|
|
2022-06-13 02:03:52 +01:00
|
|
|
// Apply BPF to reduce noise. Let in only broadcasts and our own traffic
|
|
|
|
if (bpf != NULL) {
|
|
|
|
struct bpf_program bpfp;
|
2022-09-07 13:35:05 +01:00
|
|
|
if (pcap_compile(ph, &bpfp, bpf, 1, 0)) MG_ERROR(("BPF compile failed\n"));
|
2022-06-13 02:03:52 +01:00
|
|
|
pcap_setfilter(ph, &bpfp);
|
|
|
|
pcap_freecode(&bpfp);
|
|
|
|
}
|
|
|
|
|
2022-09-07 13:35:05 +01:00
|
|
|
MG_INFO(("Opened interface %s", iface));
|
2022-06-13 02:03:52 +01:00
|
|
|
signal(SIGINT, signal_handler);
|
|
|
|
signal(SIGTERM, signal_handler);
|
|
|
|
|
2022-09-07 13:35:05 +01:00
|
|
|
struct mg_mgr mgr; // Event manager
|
|
|
|
mg_mgr_init(&mgr); // Initialise event manager
|
|
|
|
mg_log_set(MG_LL_DEBUG); // Set debug log level
|
|
|
|
|
2022-08-31 08:44:05 +01:00
|
|
|
struct mip_cfg c = {.ip = 0, .mask = 0, .gw = 0};
|
|
|
|
sscanf(mac, "%hhx:%hhx:%hhx:%hhx:%hhx:%hhx", &c.mac[0], &c.mac[1], &c.mac[2],
|
|
|
|
&c.mac[3], &c.mac[4], &c.mac[5]);
|
2022-06-13 02:03:52 +01:00
|
|
|
|
2022-08-31 08:44:05 +01:00
|
|
|
struct mip_driver driver = {.tx = pcap_tx, .up = pcap_up, .rx = pcap_rx};
|
|
|
|
mip_init(&mgr, &c, &driver, ph);
|
2022-06-13 02:03:52 +01:00
|
|
|
MG_INFO(("Init done, starting main loop"));
|
|
|
|
|
2022-09-07 20:48:23 +01:00
|
|
|
extern void device_dashboard_fn(struct mg_connection *, int, void *, void *);
|
|
|
|
mg_http_listen(&mgr, "http://0.0.0.0:8000", device_dashboard_fn, &mgr);
|
2022-09-07 13:35:05 +01:00
|
|
|
|
2022-09-07 20:48:23 +01:00
|
|
|
while (s_signo == 0) mg_mgr_poll(&mgr, 100); // Infinite event loop
|
2022-06-13 02:03:52 +01:00
|
|
|
|
|
|
|
mg_mgr_free(&mgr);
|
|
|
|
pcap_close(ph);
|
|
|
|
printf("Exiting on signal %d\n", s_signo);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|