2022-09-12 02:18:25 +08:00
|
|
|
// Copyright (c) 2022 Cesanta Software Limited
|
|
|
|
// All rights reserved
|
|
|
|
//
|
|
|
|
// example using MIP and a TUN/TAP interface
|
|
|
|
|
2022-12-10 02:48:05 +08:00
|
|
|
static const char *s_listening_address = "http://0.0.0.0:8000";
|
|
|
|
|
2022-11-16 02:08:09 +08:00
|
|
|
#include <sys/socket.h>
|
2022-12-10 02:48:05 +08:00
|
|
|
#ifndef __OpenBSD__
|
2022-09-12 02:18:25 +08:00
|
|
|
#include <linux/if.h>
|
|
|
|
#include <linux/if_tun.h>
|
2022-12-10 02:48:05 +08:00
|
|
|
#else
|
|
|
|
#include <net/if.h>
|
|
|
|
#include <net/if_tun.h>
|
|
|
|
#include <net/if_types.h>
|
|
|
|
#endif
|
2022-09-12 02:18:25 +08:00
|
|
|
#include <sys/ioctl.h>
|
2022-11-09 21:11:22 +08:00
|
|
|
#include "mongoose.h"
|
2022-09-12 02:18:25 +08:00
|
|
|
|
|
|
|
static int s_signo;
|
|
|
|
void signal_handler(int signo) {
|
|
|
|
s_signo = signo;
|
|
|
|
}
|
|
|
|
|
2022-12-19 19:43:25 +08:00
|
|
|
static size_t tap_tx(const void *buf, size_t len, struct mip_if *ifp) {
|
|
|
|
ssize_t res = write(*(int*) ifp->driver_data, buf, len);
|
2022-09-12 02:18:25 +08:00
|
|
|
if (res < 0) {
|
|
|
|
MG_ERROR(("tap_tx failed: %d", errno));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return (size_t) res;
|
|
|
|
}
|
|
|
|
|
2022-12-19 19:43:25 +08:00
|
|
|
static bool tap_up(struct mip_if *ifp) {
|
|
|
|
return ifp->driver_data ? true : false;
|
2022-09-12 02:18:25 +08:00
|
|
|
}
|
|
|
|
|
2022-12-19 19:43:25 +08:00
|
|
|
static size_t tap_rx(void *buf, size_t len, struct mip_if *ifp) {
|
|
|
|
ssize_t received = read(*(int *) ifp->driver_data, buf, len);
|
2022-09-12 02:18:25 +08:00
|
|
|
usleep(1); // This is to avoid 100% CPU
|
|
|
|
if (received < 0) return 0;
|
|
|
|
return (size_t) received;
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char *argv[]) {
|
|
|
|
const char *iface = "tap0"; // Network iface
|
2022-11-16 01:52:43 +08:00
|
|
|
const char *mac = "02:00:01:02:03:77"; // MAC address
|
2022-09-12 02:18:25 +08:00
|
|
|
|
|
|
|
// 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], "-v") == 0 && i + 1 < argc) {
|
|
|
|
mg_log_set(atoi(argv[++i]));
|
|
|
|
} else {
|
|
|
|
MG_ERROR(("unknown option %s", argv[i]));
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Open network interface
|
2022-12-10 02:48:05 +08:00
|
|
|
#ifndef __OpenBSD__
|
|
|
|
const char* tuntap_device = "/dev/net/tun";
|
|
|
|
#else
|
|
|
|
const char* tuntap_device = "/dev/tap0";
|
|
|
|
#endif
|
|
|
|
int fd = open(tuntap_device, O_RDWR);
|
2022-09-12 02:18:25 +08:00
|
|
|
struct ifreq ifr;
|
|
|
|
memset(&ifr, 0, sizeof(ifr));
|
|
|
|
strncpy(ifr.ifr_name, iface, IFNAMSIZ);
|
2022-12-10 02:48:05 +08:00
|
|
|
#ifndef __OpenBSD__
|
2022-09-12 02:18:25 +08:00
|
|
|
ifr.ifr_flags = IFF_TAP | IFF_NO_PI;
|
|
|
|
if (ioctl(fd, TUNSETIFF, (void *) &ifr) < 0) {
|
|
|
|
MG_ERROR(("Failed to setup TAP interface: %s", ifr.ifr_name));
|
2022-12-10 02:48:05 +08:00
|
|
|
abort(); // return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
ifr.ifr_flags = (short)(IFF_UP | IFF_BROADCAST | IFF_MULTICAST);
|
|
|
|
if (ioctl(fd, TUNSIFMODE, (void *) &ifr) < 0) {
|
|
|
|
MG_ERROR(("Failed to setup TAP interface: %s", ifr.ifr_name));
|
|
|
|
abort(); // return EXIT_FAILURE;
|
2022-09-12 02:18:25 +08:00
|
|
|
}
|
2022-12-10 02:48:05 +08:00
|
|
|
#endif
|
2022-09-12 02:18:25 +08:00
|
|
|
fcntl(fd, F_SETFL, fcntl(fd, F_GETFL, 0) | O_NONBLOCK); // Non-blocking mode
|
|
|
|
|
|
|
|
MG_INFO(("Opened TAP interface: %s", iface));
|
|
|
|
signal(SIGINT, signal_handler);
|
|
|
|
signal(SIGTERM, signal_handler);
|
|
|
|
|
|
|
|
struct mg_mgr mgr; // Event manager
|
|
|
|
mg_mgr_init(&mgr); // Initialise event manager
|
|
|
|
|
|
|
|
struct mip_driver driver = {.tx = tap_tx, .up = tap_up, .rx = tap_rx};
|
2022-12-19 19:43:25 +08:00
|
|
|
struct mip_if mif = {.driver = &driver, .driver_data = &fd};
|
2022-11-09 21:11:22 +08:00
|
|
|
sscanf(mac, "%hhx:%hhx:%hhx:%hhx:%hhx:%hhx", &mif.mac[0], &mif.mac[1],
|
|
|
|
&mif.mac[2], &mif.mac[3], &mif.mac[4], &mif.mac[5]);
|
|
|
|
mip_init(&mgr, &mif);
|
2022-09-12 02:18:25 +08:00
|
|
|
MG_INFO(("Init done, starting main loop"));
|
|
|
|
|
2022-12-10 02:48:05 +08:00
|
|
|
// Start infinite event loop
|
|
|
|
MG_INFO(("Mongoose version : v%s", MG_VERSION));
|
|
|
|
MG_INFO(("Listening on : %s", s_listening_address));
|
|
|
|
|
2022-09-12 02:18:25 +08:00
|
|
|
extern void device_dashboard_fn(struct mg_connection *, int, void *, void *);
|
2023-01-30 06:39:01 +08:00
|
|
|
mg_http_listen(&mgr, s_listening_address, device_dashboard_fn, NULL);
|
2022-09-12 02:18:25 +08:00
|
|
|
|
|
|
|
while (s_signo == 0) mg_mgr_poll(&mgr, 100); // Infinite event loop
|
|
|
|
|
|
|
|
mg_mgr_free(&mgr);
|
|
|
|
close(fd);
|
|
|
|
printf("Exiting on signal %d\n", s_signo);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|