mirror of
https://github.com/servalproject/serval-dna.git
synced 2025-01-13 16:29:50 +00:00
228 lines
5.8 KiB
C
228 lines
5.8 KiB
C
|
|
/*
|
|
|
|
Integration with olsr routing.
|
|
|
|
- requires olsrd to be running on the same machine with plugin X loaded with the following configuration;
|
|
LoadPlugin "name..."{
|
|
PlParam "BindPort" "1000"
|
|
PlParam "DestPort" "1001"
|
|
PlParam "MagicNumber" "123"
|
|
}
|
|
|
|
This plugin will be used to forward broadcast mdp payloads to other instances of servald running on the network.
|
|
|
|
*/
|
|
|
|
#include "serval.h"
|
|
#include "overlay_packet.h"
|
|
#include "overlay_buffer.h"
|
|
#include "overlay_address.h"
|
|
|
|
#define PACKET_FORMAT_NUMBER 123
|
|
static int local_port =4131;
|
|
static int remote_port =4130;
|
|
|
|
static void olsr_read(struct sched_ent *alarm);
|
|
|
|
static struct profile_total read_timing={
|
|
.name="olsr_read",
|
|
};
|
|
|
|
static struct sched_ent read_watch={
|
|
.function=olsr_read,
|
|
.stats=&read_timing,
|
|
.poll.fd=-1,
|
|
.poll.events=POLLIN,
|
|
};
|
|
|
|
int olsr_init_socket(void){
|
|
int fd;
|
|
int reuseP = 1;
|
|
|
|
if (read_watch.poll.fd>=0)
|
|
return 0;
|
|
|
|
if (!confValueGetBoolean("olsr.enabled",0))
|
|
return 0;
|
|
|
|
local_port = confValueGetInt64Range("olsr.local.port", local_port, 1LL, 0xFFFFLL);
|
|
remote_port = confValueGetInt64Range("olsr.remote.port", remote_port, 1LL, 0xFFFFLL);
|
|
|
|
INFOF("Initialising olsr broadcast forwarding via ports %d-%d", local_port, remote_port);
|
|
struct sockaddr_in addr = {
|
|
.sin_family = AF_INET,
|
|
.sin_addr.s_addr = htonl(INADDR_LOOPBACK),
|
|
.sin_port = htons(local_port),
|
|
};
|
|
|
|
fd = socket(AF_INET,SOCK_DGRAM,0);
|
|
if (fd < 0) {
|
|
return WHY_perror("Error creating socket");
|
|
}
|
|
|
|
if (setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &reuseP, sizeof(reuseP)) < 0) {
|
|
WHY_perror("setsockopt(SO_REUSEADR)");
|
|
close(fd);
|
|
return -1;
|
|
}
|
|
|
|
#ifdef SO_REUSEPORT
|
|
if (setsockopt(fd, SOL_SOCKET, SO_REUSEPORT, &reuseP, sizeof(reuseP)) < 0) {
|
|
WHY_perror("setsockopt(SO_REUSEPORT)");
|
|
close(fd);
|
|
return -1;
|
|
}
|
|
#endif
|
|
|
|
/* Automatically close socket on calls to exec().
|
|
This makes life easier when we restart with an exec after receiving
|
|
a bad signal. */
|
|
fcntl(fd, F_SETFL, fcntl(fd, F_GETFL, NULL) | O_CLOEXEC);
|
|
|
|
if (bind(fd, (struct sockaddr *)&addr, sizeof(struct sockaddr_in))) {
|
|
WHY_perror("Bind failed");
|
|
close(fd);
|
|
return -1;
|
|
}
|
|
|
|
// we only need to see packets from destPort
|
|
addr.sin_port=htons(remote_port);
|
|
if (connect(fd, (struct sockaddr *)&addr, sizeof(struct sockaddr_in))){
|
|
WHY_perror("Connect failed");
|
|
// ignore error
|
|
}
|
|
|
|
read_watch.poll.fd = fd;
|
|
|
|
watch(&read_watch);
|
|
return 0;
|
|
}
|
|
|
|
static void parse_frame(struct overlay_buffer *buff){
|
|
struct overlay_frame frame;
|
|
u_int8_t addr_len;
|
|
struct in_addr *addr;
|
|
|
|
// parse the incoming olsr header
|
|
if ((PACKET_FORMAT_NUMBER & 0xFF) != ob_get(buff))
|
|
return;
|
|
|
|
frame.ttl = ob_get(buff);
|
|
addr_len = ob_get(buff);
|
|
|
|
// Note IP6 not yet supported
|
|
if (addr_len!=4)
|
|
return;
|
|
|
|
addr = (struct in_addr *)ob_get_bytes_ptr(buff, addr_len);
|
|
|
|
// read source subscriber
|
|
if (overlay_address_parse(buff, NULL, &frame.source))
|
|
return;
|
|
|
|
if (frame.source->reachable==REACHABLE_NONE){
|
|
// locate the interface we should send outgoing unicast packets to
|
|
frame.source->interface = overlay_interface_find(*addr);
|
|
if (frame.source->interface){
|
|
frame.source->reachable=REACHABLE_DIRECT;
|
|
frame.source->address.sin_family=AF_INET;
|
|
frame.source->address.sin_addr=*addr;
|
|
// assume the port number of the other servald matches our local port number configuration
|
|
frame.source->address.sin_port=frame.source->interface->port;
|
|
}
|
|
}
|
|
|
|
// read source broadcast id
|
|
// assume each packet may arrive multiple times due to routing loops between servald overlay and olsr.
|
|
if (overlay_address_parse(buff, &frame.broadcast_id, NULL))
|
|
return;
|
|
|
|
// the remaining bytes are an mdp payload, process it
|
|
frame.payload = buff;
|
|
|
|
overlay_saw_mdp_containing_frame(&frame, gettime_ms());
|
|
|
|
// TODO relay this packet to other non-olsr networks.
|
|
}
|
|
|
|
static void olsr_read(struct sched_ent *alarm){
|
|
if (alarm->poll.revents & POLLIN) {
|
|
unsigned char buff[1600];
|
|
struct sockaddr_in addr;
|
|
socklen_t size = sizeof(struct sockaddr_in);
|
|
|
|
int msg_len = recvfrom(read_watch.poll.fd, buff, sizeof(buff), 0, (struct sockaddr *)&addr, &size);
|
|
if (msg_len<3)
|
|
return;
|
|
|
|
// drop packets from other port numbers
|
|
if (ntohs(addr.sin_port)!=remote_port){
|
|
WHYF("Dropping unexpected packet from port %d", ntohs(addr.sin_port));
|
|
return;
|
|
}
|
|
|
|
struct overlay_buffer *b = ob_static(buff, msg_len);
|
|
parse_frame(b);
|
|
|
|
ob_free(b);
|
|
}
|
|
|
|
if (alarm->poll.revents & (POLLHUP | POLLERR)) {
|
|
unwatch(alarm);
|
|
close(alarm->poll.fd);
|
|
alarm->poll.fd=-1;
|
|
WHY("Olsr socket closed due to error");
|
|
}
|
|
}
|
|
|
|
static int send_packet(unsigned char *header, int header_len, unsigned char *payload, int payload_len){
|
|
struct sockaddr_in addr={
|
|
.sin_family=AF_INET,
|
|
.sin_addr.s_addr=htonl(INADDR_LOOPBACK),
|
|
.sin_port=htons(remote_port),
|
|
};
|
|
|
|
struct iovec iov[]={
|
|
{
|
|
.iov_base=header,
|
|
.iov_len=header_len,
|
|
},
|
|
{
|
|
.iov_base=payload,
|
|
.iov_len=payload_len,
|
|
},
|
|
};
|
|
|
|
struct msghdr msg={
|
|
.msg_name=&addr,
|
|
.msg_namelen=sizeof(struct sockaddr_in),
|
|
.msg_iov=iov,
|
|
.msg_iovlen=2,
|
|
};
|
|
|
|
if (sendmsg(read_watch.poll.fd, &msg, 0)<0){
|
|
return WHY_perror("Sending packet");
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int olsr_send(struct overlay_frame *frame){
|
|
if (read_watch.poll.fd<0)
|
|
return 0;
|
|
// only send broadcasts
|
|
if (frame->destination)
|
|
return 0;
|
|
|
|
struct overlay_buffer *b=ob_new();
|
|
// build olsr specific frame header
|
|
ob_append_byte(b, PACKET_FORMAT_NUMBER);
|
|
ob_append_byte(b, frame->ttl);
|
|
overlay_address_append(b, frame->source);
|
|
overlay_broadcast_append(b, &frame->broadcast_id);
|
|
// send the packet
|
|
int ret = send_packet(b->bytes, b->position, frame->payload->bytes, frame->payload->sizeLimit);
|
|
ob_free(b);
|
|
return ret;
|
|
}
|