Commit b63c15af authored by Juliusz Chroboczek's avatar Juliusz Chroboczek

Make the flooding protocol use TCP.

This simplifies the code a lot, since it gives us the notion of a session
for free.  And congestion control, of course.

This is not quite ready yet: we don't wait for the peer's handshake before
initiating flooding, and we don't perform duplicate suppression.  Doing that
will require a different data structure.
parent d818237e
...@@ -18,30 +18,17 @@ ...@@ -18,30 +18,17 @@
#include "flood.h" #include "flood.h"
#include "util.h" #include "util.h"
int flood_socket = -1; int server_socket = -1;
int flood_port = 4444; int server_port = -1;
struct datum **data = NULL; struct datum **data = NULL;
int numdata = 0, maxdata = 0; int numdata = 0, maxdata = 0;
struct timespec flood_time = {0, 0};
static void
schedule_flood()
{
struct timespec now;
clock_gettime(CLOCK_MONOTONIC, &now);
ts_add_msec(&flood_time, &now, 1);
}
static void (*datum_callback)(struct datum *, int) = NULL; static void (*datum_callback)(struct datum *, int) = NULL;
static int buffer_update(struct neighbour *neigh, static int parse_tlv(struct neighbour *neigh);
const unsigned char *key, int keylen, int acked); static int handshake(struct neighbour *neigh);
static int record_unacked(struct neighbour *neigh, static int dump_data(struct neighbour *neigh);
const unsigned char *key, int keylen);
static int flush_unacked(struct neighbour *neigh,
const unsigned char *key, int keylen);
static int static int
seqno_compare(unsigned short s1, unsigned short s2) seqno_compare(unsigned short s1, unsigned short s2)
...@@ -222,65 +209,137 @@ extend_datum(struct datum *datum, time_t extend) ...@@ -222,65 +209,137 @@ extend_datum(struct datum *datum, time_t extend)
return 0; return 0;
} }
struct neighbour *neighbours = NULL; struct neighbour *neighs = NULL;
int numneighbours = 0, maxneighbours = 0; int numneighs = 0, maxneighs = 0;
static int send_dump_request(struct neighbour *neigh); struct neighbour *
static int send_dump_reply(struct neighbour *neigh); find_neighbour(int fd)
{
for(int i = 0; i < numneighs; i++) {
if(neighs[i].fd == fd)
return &neighs[i];
}
return NULL;
}
int struct neighbour *
flood_setup(void (*callback)(struct datum *, int)) create_neighbour()
{ {
struct sockaddr_in6 sin6; if(maxneighs <= numneighs) {
int s, rc, saved_errno; int n = maxneighs == 0 ? 8 : 2 * maxneighs;
int zero = 0, one = 1; struct neighbour *newneighs =
realloc(neighs, n * sizeof(struct neighbour));
if(newneighs != NULL) {
neighs = newneighs;
maxneighs = n;
}
}
if(maxneighs <= numneighs)
return NULL;
memset(&neighs[numneighs], 0, sizeof(struct neighbour));
neighs[numneighs].fd = -1;
numneighs++;
return &neighs[numneighs - 1];
}
void
flush_neighbour(struct neighbour *neigh)
{
int i = neigh - neighs;
assert(i >= 0 && i < numneighs);
if(neigh->fd >= 0) {
close(neigh->fd);
neigh->fd = -1;
}
s = socket(PF_INET6, SOCK_DGRAM, 0); if(neigh->sin6 != NULL) {
if(s < 0) free(neigh->sin6);
neigh->sin6 = NULL;
}
if(i < numneighs - 1)
memmove(neighs + i, neighs + i + 1,
(numneighs - i - 1) * sizeof(struct neighbour));
numneighs--;
}
static int
setup_socket(int fd)
{
int rc, saved_errno;
int zero = 0;
if(fd < 0) {
fd = socket(PF_INET6, SOCK_STREAM, 0);
if(fd < 0)
return -1; return -1;
rc = setsockopt(s, IPPROTO_IPV6, IPV6_V6ONLY, &zero, sizeof(zero)); rc = setsockopt(fd, IPPROTO_IPV6, IPV6_V6ONLY, &zero, sizeof(zero));
if(rc < 0) if(rc < 0)
goto fail; goto fail;
}
rc = fcntl(s, F_GETFL, 0); rc = fcntl(fd, F_GETFL, 0);
if(rc < 0) if(rc < 0)
goto fail; goto fail;
rc = fcntl(s, F_SETFL, (rc | O_NONBLOCK)); rc = fcntl(fd, F_SETFL, (rc | O_NONBLOCK));
if(rc < 0) if(rc < 0)
goto fail; goto fail;
rc = fcntl(s, F_GETFD, 0); rc = fcntl(fd, F_GETFD, 0);
if(rc < 0) if(rc < 0)
goto fail; goto fail;
rc = fcntl(s, F_SETFD, rc | FD_CLOEXEC); rc = fcntl(fd, F_SETFD, rc | FD_CLOEXEC);
if(rc < 0) if(rc < 0)
goto fail; goto fail;
rc = setsockopt(s, IPPROTO_IPV6, IPV6_RECVPKTINFO, &one, sizeof(one)); return fd;
if(rc < 0)
goto fail; fail:
saved_errno = errno;
close(fd);
errno = saved_errno;
return -1;
}
int
flood_setup(void (*callback)(struct datum *, int))
{
struct sockaddr_in6 sin6;
int fd, rc, saved_errno;
datum_callback = callback;
if(server_port < 0)
return 0;
fd = setup_socket(-1);
if(fd < 0)
return -1;
memset(&sin6, 0, sizeof(sin6)); memset(&sin6, 0, sizeof(sin6));
sin6.sin6_family = AF_INET6; sin6.sin6_family = AF_INET6;
sin6.sin6_port = htons(flood_port); sin6.sin6_port = htons(server_port);
rc = bind(s, (struct sockaddr*)&sin6, sizeof(sin6)); rc = bind(fd, (struct sockaddr*)&sin6, sizeof(sin6));
if(rc < 0) if(rc < 0)
goto fail; goto fail;
flood_socket = s; rc = listen(fd, 1024);
if(rc < 0)
goto fail;
datum_callback = callback; server_socket = fd;
periodic_flood();
return 1; return 1;
fail: fail:
saved_errno = errno; saved_errno = errno;
close(s); close(fd);
errno = saved_errno; errno = saved_errno;
return -1; return -1;
} }
...@@ -288,561 +347,408 @@ flood_setup(void (*callback)(struct datum *, int)) ...@@ -288,561 +347,408 @@ flood_setup(void (*callback)(struct datum *, int))
void void
flood_cleanup() flood_cleanup()
{ {
close(flood_socket); for(int i = 0; i < numneighs; i++) {
flood_socket = -1; if(neighs[i].fd >= 0) {
} close(neighs[i].fd);
neighs[i].fd = -1;
static void }
commit_neighbour(struct neighbour *neigh, int update, int permanent) }
{ if(server_socket >= 0) {
if(update) { close(server_socket);
struct timespec now; server_socket = -1;
clock_gettime(CLOCK_MONOTONIC, &now);
neigh->time = now.tv_sec;
} }
if(permanent)
neigh->permanent = 1;
} }
static int int
match(const struct sockaddr_in6 *a, const struct sockaddr_in6 *b) flood_accept()
{ {
return a->sin6_port == b->sin6_port && int fd, rc;
memcmp(&a->sin6_addr, &b->sin6_addr, 16) == 0; struct neighbour *neigh;
}
struct neighbour * fd = accept(server_socket, NULL, NULL);
find_neighbour(struct sockaddr_in6 *sin6, int create, int update, int permanent) if(fd < 0) {
{ if(errno != EAGAIN)
for(int i = 0; i < numneighbours; i++) { perror("accept");
if(match(sin6, &neighbours[i].addr)) { return 0;
commit_neighbour(&neighbours[i], update, permanent);
return &neighbours[i];
}
} }
if(!create) rc = setup_socket(fd);
return NULL; if(rc < 0) {
perror("setup_socket(accept)");
if(maxneighbours <= numneighbours) { close(fd);
int n = maxneighbours == 0 ? 8 : 2 * maxneighbours; return -1;
struct neighbour *newneighbours =
realloc(neighbours, n * sizeof(struct neighbour));
if(newneighbours != NULL) {
neighbours = newneighbours;
maxneighbours = n;
}
} }
if(maxneighbours <= numneighbours)
return NULL;
memset(&neighbours[numneighbours], 0, sizeof(struct neighbour)); neigh = create_neighbour();
memcpy(&neighbours[numneighbours].addr, sin6, sizeof(struct sockaddr_in6)); if(neigh == NULL) {
neighbours[numneighbours].dump_done = 0; close(fd);
commit_neighbour(&neighbours[numneighbours], update, permanent); return -1;
numneighbours++; }
return &neighbours[numneighbours-1];
}
void neigh->fd = fd;
flush_neighbour(struct neighbour *neigh) rc = handshake(neigh);
{ if(rc < 0) {
int i = neigh - neighbours; close(neigh->fd);
assert(i >= 0 && i < numneighbours); neigh->fd = -1;
free(neighbours[i].unacked); }
neighbours[i].unacked = NULL; return 1;
free(neighbours[i].pktinfo);
neighbours[i].pktinfo = NULL;
if(i < numneighbours - 1)
memmove(neighbours + i, neighbours + i + 1,
(numneighbours - i - 1) * sizeof(struct neighbour));
numneighbours--;
} }
static void static int
parse_packet(struct sockaddr_in6 *from, struct in6_pktinfo *info, flood_reconnect(struct neighbour *neigh)
const unsigned char *packet, int packetlen)
{ {
struct neighbour *neigh; int fd, rc;
unsigned int bodylen;
int i;
if(packetlen < 4)
return;
if(packet[0] != 44 || packet[1] != 0) fd = setup_socket(-1);
return; if(fd < 0) {
return -1;
}
DO_NTOHS(bodylen, packet + 2); rc = connect(fd, (struct sockaddr*)neigh->sin6, sizeof(struct sockaddr_in6));
if(rc < 0 && errno != EINPROGRESS) {
perror("connect");
close(fd);
/* let the connect loop recover */
return 0;
}
if(bodylen + 4 > packetlen) { neigh->fd = fd;
fprintf(stderr, "Received truncated packet.\n"); rc = handshake(neigh);
return; if(rc < 0) {
close(neigh->fd);
neigh->fd = -1;
return -1;
} }
return 1;
}
neigh = find_neighbour(from, 1, 1, 0); int
flood_connect(const struct sockaddr_in6 *sin6)
{
struct neighbour *neigh;
neigh = create_neighbour();
if(neigh == NULL) if(neigh == NULL)
return; return -1;
if(info != NULL) { if(neigh->sin6 == NULL)
if(neigh->pktinfo != NULL) { neigh->sin6 = malloc(sizeof(struct sockaddr_in6));
if(memcmp(neigh->pktinfo, info, sizeof(struct in6_pktinfo)) != 0) { if(neigh->sin6 == NULL) {
free(neigh->pktinfo); flush_neighbour(neigh);
neigh->pktinfo = NULL; return -1;
} }
memcpy(neigh->sin6, sin6, sizeof(struct sockaddr_in6));
return flood_reconnect(neigh);
}
int
flood_read(struct neighbour *neigh)
{
int rc;
if(neigh->in.cap == 0) {
neigh->in.buf = malloc(4096);
if(neigh->in.buf == NULL) {
close(neigh->fd);
neigh->fd = -1;
return -1;
} }
if(neigh->pktinfo == NULL) neigh->in.cap = 4096;
neigh->pktinfo = malloc(sizeof(struct in6_pktinfo));
if(neigh->pktinfo != NULL)
memcpy(neigh->pktinfo, info, sizeof(struct in6_pktinfo));
} }
i = 0; if(neigh->in.len >= neigh->in.cap) {
while(i < bodylen) { fprintf(stderr, "Read buffer overflow.\n");
const unsigned char *tlv = packet + 4 + i; close(neigh->fd);
int len; neigh->fd = -1;
if(tlv[0] == 0) { return -1;
i++;
continue;
} }
if(i + 1 > bodylen)
return;
len = tlv[1];
if(i + len + 2 > bodylen)
return;
switch(tlv[0]) { rc = read(neigh->fd,
case 1: neigh->in.buf + neigh->in.len,
debugf("<- PAD1\n"); neigh->in.cap - neigh->in.len);
break; if(rc <= 0) {
case 2: { if(errno == EAGAIN)
struct datum *datum; return 0;
unsigned char keylen; if(rc < 0)
unsigned short seqno; perror("read");
unsigned int time; close(neigh->fd);
int ack, doit, conflict; neigh->fd = -1;
if(len < 2) {
debugf("Truncated DATUM.\n");
goto skip;
}
ack = !!(tlv[2] & 0x80);
DO_NTOHS(seqno, tlv + 3);
DO_NTOHL(time, tlv + 5);
keylen = tlv[9];
if(len < keylen + 8) {
debugf("Truncated DATUM.\n");
goto skip;
}
debugf("<- DATUM %d (%d) %ld%s\n",
keylen <= 0 ? -1 : (int)tlv[10], keylen, (long)time,
ack ? " (ack)" : "");
datum = find_datum(tlv+10, keylen);
if(datum != NULL && seqno >= datum->seqno) {
flush_unacked(neigh, tlv+10, keylen);
}
datum = update_datum(tlv + 10, keylen, seqno,
tlv + 10 + keylen, len - keylen - 8,
time, &doit, &conflict);
if(doit && datum_callback != NULL)
datum_callback(datum, conflict);
if(doit || ack)
flood(datum, neigh, ack, doit);
} }
break; neigh->in.len += rc;
case 3:
debugf("<- DUMP\n"); while(neigh->in.len > 0) {
send_dump_reply(neigh); rc = parse_tlv(neigh);
for(int i = 0; i < numneighbours; i++) { if(rc < 0) {
for(int j = 0; j < numdata; j++) close(neigh->fd);
record_unacked(&neighbours[i], neigh->fd = -1;
datum_key(data[j]), data[j]->keylen); return -1;
}
schedule_flood();
break;
case 4:
debugf("<- DUMP-ACK\n");
neigh->dump_done = 1;
break;
default:
debugf("Unknown TLV %d\n", tlv[0]);
} }
if(rc == 0)
return 1;
skip: memmove(neigh->in.buf, neigh->in.buf + rc,
i += 2 + len; neigh->in.len - rc);
neigh->in.len -= rc;
} }
flush_updates(neigh, 1); return 1;
} }
int int
flood_listen(void) flood_write(struct neighbour *neigh)
{ {
struct sockaddr_in6 from;
struct in6_pktinfo *info;
unsigned char buf[4096];
struct iovec iov[1];
struct msghdr msg;
int cmsglen = 100;
char cmsgbuf[cmsglen];
struct cmsghdr *cmsg = (struct cmsghdr*)cmsgbuf;
int rc; int rc;
iov[0].iov_base = buf; if(neigh->fd < 0) {
iov[0].iov_len = 4096; fprintf(stderr, "flood_write called for dead neighbour!\n");
memset(&msg, 0, sizeof(msg)); return -1;
msg.msg_name = &from;
msg.msg_namelen = sizeof(from);
msg.msg_iov = iov;
msg.msg_iovlen = 1;
msg.msg_control = cmsg;
msg.msg_controllen = cmsglen;
rc = recvmsg(flood_socket, &msg, 0);
if(rc < 0)
return rc;
info = NULL;
cmsg = CMSG_FIRSTHDR(&msg);
while(cmsg != NULL) {
if ((cmsg->cmsg_level == IPPROTO_IPV6) &&
(cmsg->cmsg_type == IPV6_PKTINFO)) {
info = (struct in6_pktinfo*)CMSG_DATA(cmsg);
break;
} }
cmsg = CMSG_NXTHDR(&msg, cmsg);
if(neigh->out.len <= 0) {
fprintf(stderr, "flood_write called but nothing to do!\n");
return 0;
} }
if(info == NULL) { rc = write(neigh->fd, neigh->out.buf, neigh->out.len);
errno = EINVAL; if(rc < 0) {
if(errno == EAGAIN)
return 0;
close(neigh->fd);
neigh->fd = -1;
return -1; return -1;
} }
if(rc < neigh->out.len) {
parse_packet(&from, info, buf, rc); memmove(neigh->out.buf, neigh->out.buf + rc, neigh->out.len - rc);
neigh->out.len -= rc;
} else {
neigh->out.len = 0;
}
return 1; return 1;
} }
static int static int
send_neighbour(struct neighbour *neigh, unsigned char *buf, int buflen) parse_tlv(struct neighbour *neigh)
{ {
struct timespec now; int tpe, len;
struct msghdr msg; unsigned char *body;
struct iovec iov[1];
struct cmsghdr *cmsg;
union {
struct cmsghdr hdr;
char buf[CMSG_SPACE(sizeof(struct in6_pktinfo))];
} u;
clock_gettime(CLOCK_MONOTONIC, &now); if(!neigh->handshake_received) {
neigh->send_time = now.tv_sec; if(neigh->in.len < 4)
return 0;
iov[0].iov_base = buf; if(neigh->in.buf[0] != 44 ||
iov[0].iov_len = buflen; neigh->in.buf[1] != 1)
memset(&msg, 0, sizeof(msg)); return -1;
msg.msg_name = (struct sockaddr*)&neigh->addr; debugf("<- Handshake\n");
msg.msg_namelen = sizeof(neigh->addr); neigh->handshake_received = 1;
msg.msg_iov = iov; if(!neigh->dump_sent) {
msg.msg_iovlen = 1; int rc = dump_data(neigh);
if(neigh->pktinfo != NULL) { if(rc < 0)
memset(u.buf, 0, sizeof(u.buf)); return rc;
msg.msg_control = u.buf; }
msg.msg_controllen = CMSG_SPACE(sizeof(struct in6_pktinfo)); return 4;
cmsg = CMSG_FIRSTHDR(&msg); }
cmsg->cmsg_level = IPPROTO_IPV6;
cmsg->cmsg_type = IPV6_PKTINFO;
cmsg->cmsg_len = CMSG_LEN(sizeof(struct in6_pktinfo));
memcpy(CMSG_DATA(cmsg), neigh->pktinfo, sizeof(struct in6_pktinfo));
}
return sendmsg(flood_socket, &msg, 0);
} if(neigh->in.len < 1)
return 0;
static int if(neigh->in.buf[0] == 0) {
send_dump_request(struct neighbour *neigh) debugf("<- PAD1\n");
{ return 1;
unsigned char buf[6] = {44, 0, 0, 2, 3, 0}; }
debugf("-> DUMP\n");
return send_neighbour(neigh, buf, 6);
}
static int if(neigh->in.len < 2)
send_dump_reply(struct neighbour *neigh) return 0;
{
unsigned char buf[6] = {44, 0, 0, 2, 4, 0};
debugf("-> DUMP-ACK\n");
return send_neighbour(neigh, buf, 6);
}
int tpe = neigh->in.buf[0];
flush_updates(struct neighbour *neigh, int all) len = neigh->in.buf[1];
{
unsigned char buf[1024] = {44, 0, 0, 0};
struct timespec now;
time_t time;
int i, n, rc = 0;
if(neigh->numbuffered == 0) if(neigh->in.len < len + 2)
return 0; return 0;
clock_gettime(CLOCK_MONOTONIC, &now); body = neigh->in.buf + 2;
i = 0; switch(tpe) {
for(n = 0; n < neigh->numbuffered; n++) { case 1:
debugf("<- PADN\n");
break;
case 2: {
struct datum *datum; struct datum *datum;
unsigned char keylen;
if(i == 0) { unsigned short seqno;
buf[i] = 44; i++; unsigned int time;
buf[i] = 0; i++; int doit, conflict;
buf[i] = 0; i++; if(len < 7) {
buf[i] = 0; i++; debugf("Truncated Update.\n");
return -1;
} }
DO_NTOHS(seqno, body);
datum = find_datum(neigh->buffered[n].key, neigh->buffered[n].keylen); DO_NTOHL(time, body + 2);
free(neigh->buffered[n].key); keylen = body[6];
neigh->buffered[n].key = NULL; if(len < keylen + 7) {
if(datum == NULL) debugf("Truncated Update.\n");
continue; return -1;
time = datum->time - now.tv_sec;
if(time <= 0)
continue;
if(time > 0xFFFFFFFF)
time = 0xFFFFFFFF;
buf[i++] = 2;
buf[i++] = 1 + 2 + 4 + 1 + datum->keylen + datum->vallen;
buf[i++] = neigh->buffered[n].acked ? 0x80 : 0;
DO_HTONS(buf + i, datum->seqno); i += 2;
DO_HTONL(buf + i, time); i += 4;
buf[i++] = datum->keylen;
memcpy(buf + i, datum->datum, datum->keylen + datum->vallen);
i += datum->keylen + datum->vallen;
debugf("-> DATUM %d (%d) %ld%s\n",
datum->keylen <= 0 ? -1 : datum->datum[0], datum->keylen,
(long)time, neigh->buffered[n].acked ? " (ack)" : "");
if(i >= 1024 - 32) {
if(!all)
break;
DO_HTONS(buf + 2, i - 4);
rc = send_neighbour(neigh, buf, i);
} }
debugf("<- Update %d (%d) %ld\n",
keylen <= 0 ? -1 : (int)body[7], keylen, (long)time);
datum = find_datum(body + 7, keylen);
datum = update_datum(body + 7, keylen, seqno,
body + 7 + keylen, len - keylen - 7,
time, &doit, &conflict);
if(doit) {
if(datum_callback != NULL)
datum_callback(datum, conflict);
flood(datum, neigh);
} }
if(i > 0) {
DO_HTONS(buf + 2, i - 4);
rc = send_neighbour(neigh, buf, i);
} }
break;
if(n < neigh->numbuffered) { default:
memmove(neigh->buffered, neigh->buffered + n, debugf("Unknown TLV %d\n", tpe);
(neigh->numbuffered - n) * sizeof(struct buffered));
neigh->numbuffered -= n;
} else {
neigh->numbuffered = 0;
} }
return 2 + len;
return rc;
} }
static int static int
buffer_update(struct neighbour *neigh, const unsigned char *key, int keylen, expand_buffer(struct buffer *buf, int len)
int acked)
{ {
unsigned char *newkey; int cap;
unsigned char *b;
if(neigh->buffered == NULL) {
neigh->buffered = malloc(MAXBUFFERED * sizeof(struct buffered));
if(neigh->buffered == NULL)
return -1;
}
if(neigh->numbuffered >= MAXBUFFERED)
flush_updates(neigh, 0);
assert(neigh->numbuffered < MAXBUFFERED);
newkey = malloc(keylen); if(buf->cap - buf->len >= len)
if(newkey == NULL) return 0;
return -1;
memcpy(newkey, key, keylen);
neigh->buffered[neigh->numbuffered].key = newkey; cap = buf->len + len;
neigh->buffered[neigh->numbuffered].keylen = keylen; if(cap < buf->cap * 2)
neigh->buffered[neigh->numbuffered].acked = acked; cap = buf->cap * 2;
neigh->numbuffered++;
b = malloc(cap);
if(b == NULL)
return -1;
memcpy(b, buf->buf, buf->len);
free(buf->buf);
buf->buf = b;
buf->cap = cap;
return 1; return 1;
} }
static const unsigned char hs[4] = {44, 1, 0, 0};
static int static int
send_keepalive(struct neighbour *neigh) buffer_handshake(struct neighbour *neigh)
{ {
unsigned char buf[4] = {44, 0, 0, 0}; int rc;
debugf("-> Keepalive\n"); rc = expand_buffer(&neigh->out, 4);
return send_neighbour(neigh, buf, 4); if(rc < 0)
return -1;
memcpy(neigh->out.buf + neigh->out.len, hs, 4);
neigh->out.len += 4;
debugf("-> Handshake\n");
return 1;
} }
static int static int
neighbour_alive(struct neighbour *neigh, time_t now) buffer_tlv(struct neighbour *neigh, int tpe, int len, unsigned char *body)
{ {
if(neigh->permanent) int rc;
rc = expand_buffer(&neigh->out, 2 + len);
if(rc < 0)
return -1;
neigh->out.buf[neigh->out.len++] = tpe;
neigh->out.buf[neigh->out.len++] = len;
memcpy(neigh->out.buf + neigh->out.len, body, len);
neigh->out.len += len;
return 1; return 1;
return neigh->time > now - 240;
} }
void static int
flood(struct datum *datum, struct neighbour *neigh, int ack, int doit) buffer_update(struct neighbour *neigh, struct datum *datum)
{ {
int len = 2 + 4 + 1 + datum->keylen + datum->vallen;
unsigned char body[len];
struct timespec now; struct timespec now;
if(ack && neigh != NULL) time_t time;
buffer_update(neigh, datum_key(datum), datum->keylen, 0);
clock_gettime(CLOCK_MONOTONIC, &now); clock_gettime(CLOCK_MONOTONIC, &now);
if(doit) { time = datum->time - now.tv_sec;
for(int i = 0; i < numneighbours; i++) { if(time < 0)
if(neighbour_alive(&neighbours[i], now.tv_sec) && time = 0;
&neighbours[i] != neigh) if(time > 0xFFFFFFFF)
record_unacked(&neighbours[i], datum_key(datum), datum->keylen); time = 0xFFFFFFFF;
} DO_HTONS(body, datum->seqno);
} DO_HTONL(body + 2, (unsigned int)time);
schedule_flood(); body[6] = datum->keylen;
} memcpy(body + 7, datum->datum, datum->keylen);
memcpy(body + 7 + datum->keylen, datum->datum + datum->keylen,
static struct unacked * datum->vallen);
find_unacked(struct neighbour *neigh, const unsigned char *key, int keylen) debugf("-> Update\n");
{ return buffer_tlv(neigh, 2, len, body);
for(int i = 0; i < neigh->numunacked; i++) {
if(neigh->unacked[i].keylen == keylen &&
memcmp(neigh->unacked[i].key, key, keylen) == 0)
return &neigh->unacked[i];
}
return NULL;
} }
static int static int
record_unacked(struct neighbour *neigh, const unsigned char *key, int keylen) dump_data(struct neighbour *neigh)
{ {
struct unacked *unacked; for(int i = 0; i < numdata; i++) {
struct timespec now; int rc = buffer_update(neigh, data[i]);
unsigned char *newkey; if(rc < 0)
return rc;
clock_gettime(CLOCK_MONOTONIC, &now);
unacked = find_unacked(neigh, key, keylen);
if(unacked != NULL) {
unacked->count = 0;
unacked->time = now.tv_sec;
return 0;
}
if(neigh->numunacked >= neigh->maxunacked) {
struct unacked *n;
int count = neigh->maxunacked * 3 / 2;
if(count < 8)
count = 8;
n = realloc(neigh->unacked, count * sizeof(struct unacked));
if(n == NULL)
return -1;
neigh->unacked = n;
neigh->maxunacked = count;
} }
neigh->dump_sent = 1;
newkey = malloc(keylen);
if(newkey == NULL)
return -1;
memcpy(newkey, key, keylen);
neigh->unacked[neigh->numunacked].count = 0;
neigh->unacked[neigh->numunacked].key = newkey;
neigh->unacked[neigh->numunacked].keylen = keylen;
neigh->unacked[neigh->numunacked].time = now.tv_sec;
neigh->numunacked++;
schedule_flood();
return 1; return 1;
} }
static int static int
flush_unacked(struct neighbour *neigh, const unsigned char *key, int keylen) handshake(struct neighbour *neigh)
{ {
int i; int rc;
struct unacked *unacked; if(neigh->fd < 0)
return -1;
unacked = find_unacked(neigh, key, keylen); rc = buffer_handshake(neigh);
if(unacked == NULL) if(rc < 0)
return 0; return rc;
i = unacked - neigh->unacked;
assert(i >= 0 && i < neigh->numunacked);
free(neigh->unacked[i].key); if(neigh->handshake_received) {
neigh->unacked[i].key = NULL; rc = dump_data(neigh);
if(rc < 0)
return rc;
}
if(i < neigh->numunacked - 1)
memmove(neigh->unacked + i, neigh->unacked + i + 1,
(neigh->numunacked - i - 1) * sizeof(struct unacked));
neigh->numunacked--;
return 1; return 1;
} }
static struct timespec expire_neighbours_time = {0, 0}; void
flood(struct datum *datum, struct neighbour *neigh)
static void
expire_neighbours()
{ {
struct timespec now; struct timespec now;
int i;
clock_gettime(CLOCK_MONOTONIC, &now); clock_gettime(CLOCK_MONOTONIC, &now);
i = 0; for(int i = 0; i < numneighs; i++) {
while(i < numneighbours) { if(neighs[i].fd >= 0 && &neighs[i] != neigh) {
if(neighbour_alive(&neighbours[i], now.tv_sec)) { int rc;
if(neighbours[i].send_time < now.tv_sec - 60) rc = buffer_update(&neighs[i], datum);
send_keepalive(&neighbours[i]); if(rc < 0) {
i++; close(neighs[i].fd);
} else { neighs[i].fd = -1;
flush_neighbour(&neighbours[i]); }
} }
} }
} }
struct timespec expire_neighs_time = {0, 0};
void void
periodic_flood() expire_neighs()
{ {
struct timespec now; int i = 0;
int work = 0; while(i < numneighs) {
if(neighs[i].fd >= 0) {
clock_gettime(CLOCK_MONOTONIC, &now); i++;
} else if(neighs[i].sin6 != NULL) {
if(ts_compare(&expire_neighbours_time, &now) <= 0) { flood_reconnect(&neighs[i]);
expire_neighbours(); i++;
expire_neighbours_time = now;
expire_neighbours_time.tv_sec += 10;
}
for(int i = 0; i < numneighbours; i++) {
if(!neighbours[i].dump_done && neighbours[i].dump_request_count < 4) {
work = 1;
send_dump_request(&neighbours[i]);
neighbours[i].dump_request_count++;
}
if(neighbours[i].numunacked > 0)
work = 1;
for(int j = 0; j < neighbours[i].numunacked; j++) {
struct unacked *unacked = &neighbours[i].unacked[j];
struct timespec soon = {unacked->time, 0};
if(unacked->count > 0)
soon.tv_sec += 1 << (unacked->count - 1);
if(ts_compare(&soon, &now) <= 0) {
buffer_update(&neighbours[i], unacked->key, unacked->keylen, 1);
unacked->count++;
}
}
flush_updates(&neighbours[i], 1);
}
if(work) {
flood_time = now;
flood_time.tv_sec += 1;
} else { } else {
flood_time.tv_sec = 0; flush_neighbour(&neighs[i]);
flood_time.tv_nsec = 0;
} }
}
clock_gettime(CLOCK_MONOTONIC, &expire_neighs_time);
expire_neighs_time.tv_sec += 30;
} }
...@@ -18,13 +18,11 @@ datum_val(const struct datum *datum) ...@@ -18,13 +18,11 @@ datum_val(const struct datum *datum)
return datum->datum + datum->keylen; return datum->datum + datum->keylen;
} }
extern int flood_port; extern int server_port;
extern int flood_socket; extern int server_socket;
extern struct datum **data; extern struct datum **data;
extern int numdata, maxdata; extern int numdata, maxdata;
extern struct timespec flood_time;
struct unacked { struct unacked {
int count; int count;
unsigned char *key; unsigned char *key;
...@@ -38,24 +36,23 @@ struct buffered { ...@@ -38,24 +36,23 @@ struct buffered {
int acked; int acked;
}; };
#define MAXBUFFERED 100 struct buffer {
unsigned char *buf;
int len, cap;
};
struct neighbour { struct neighbour {
struct sockaddr_in6 addr; int fd;
struct in6_pktinfo *pktinfo; int handshake_received;
int permanent; int dump_sent;
time_t time; struct sockaddr_in6 *sin6;
time_t send_time; struct buffer in, out;
struct unacked *unacked;
int numunacked, maxunacked;
struct buffered *buffered;
int numbuffered;
int dump_request_count;
int dump_done;
}; };
extern struct neighbour *neighbours; extern struct neighbour *neighs;
extern int numneighbours, maxneighbours; extern int numneighs, maxneighs;
extern struct timespec expire_neighs_time;
struct datum *find_datum(const unsigned char *key, int keylen); struct datum *find_datum(const unsigned char *key, int keylen);
struct datum *update_datum(const unsigned char *key, int keylen, struct datum *update_datum(const unsigned char *key, int keylen,
...@@ -67,9 +64,9 @@ time_t datum_remaining(const struct datum *datum); ...@@ -67,9 +64,9 @@ time_t datum_remaining(const struct datum *datum);
int extend_datum(struct datum *datum, time_t extend); int extend_datum(struct datum *datum, time_t extend);
int flood_setup(void (*callback)(struct datum *, int)); int flood_setup(void (*callback)(struct datum *, int));
void flood_cleanup(void); void flood_cleanup(void);
int flood_listen(void); int flood_accept(void);
struct neighbour * int flood_connect(const struct sockaddr_in6* sin6);
find_neighbour(struct sockaddr_in6 *sin6, int create, int update, int permanent); int flood_read(struct neighbour *neigh);
void flood(struct datum *datum, struct neighbour *neigh, int ack, int doit); int flood_write(struct neighbour *neigh);
void periodic_flood(void); void flood(struct datum *datum, struct neighbour *except);
int flush_updates(struct neighbour *neigh, int all); void expire_neighs(void);
...@@ -119,7 +119,7 @@ update_lease(const unsigned char *mac, int ipv6, ...@@ -119,7 +119,7 @@ update_lease(const unsigned char *mac, int ipv6,
doit = extend_datum(datum, time); doit = extend_datum(datum, time);
if(doit_return) if(doit_return)
*doit_return = doit; *doit_return = doit;
flood(datum, NULL, 0, 1); flood(datum, NULL);
return datum; return datum;
} }
...@@ -136,7 +136,7 @@ update_lease(const unsigned char *mac, int ipv6, ...@@ -136,7 +136,7 @@ update_lease(const unsigned char *mac, int ipv6,
if(doit_return) if(doit_return)
*doit_return = doit; *doit_return = doit;
update_client_routes(mac, lease_address(datum, ipv6), ipv6); update_client_routes(mac, lease_address(datum, ipv6), ipv6);
flood(datum, NULL, 0, 1); flood(datum, NULL);
return datum; return datum;
} }
...@@ -199,7 +199,7 @@ update_association(struct interface *interface, const unsigned char *mac, ...@@ -199,7 +199,7 @@ update_association(struct interface *interface, const unsigned char *mac,
if(datum->vallen == 8 && if(datum->vallen == 8 &&
memcmp(datum_val(datum), myid, 8) == 0) { memcmp(datum_val(datum), myid, 8) == 0) {
extend_datum(datum, time); extend_datum(datum, time);
flood(datum, NULL, 0, 1); flood(datum, NULL);
return client; return client;
} else { } else {
seqno = datum->seqno + 1; seqno = datum->seqno + 1;
...@@ -207,7 +207,7 @@ update_association(struct interface *interface, const unsigned char *mac, ...@@ -207,7 +207,7 @@ update_association(struct interface *interface, const unsigned char *mac,
} }
datum = update_datum(key, 7, seqno, myid, 8, time, NULL, NULL); datum = update_datum(key, 7, seqno, myid, 8, time, NULL, NULL);
flood(datum, NULL, 0, 1); flood(datum, NULL);
return client; return client;
} }
...@@ -230,5 +230,5 @@ flush_association(const unsigned char *mac, int time) ...@@ -230,5 +230,5 @@ flush_association(const unsigned char *mac, int time)
seqno = datum->seqno + 1; seqno = datum->seqno + 1;
datum = update_datum(key, 7, seqno, NULL, 0, time, NULL, NULL); datum = update_datum(key, 7, seqno, NULL, 0, time, NULL, NULL);
flood(datum, NULL, 0, 1); flood(datum, NULL);
} }
...@@ -211,7 +211,7 @@ main(int argc, char **argv) ...@@ -211,7 +211,7 @@ main(int argc, char **argv)
p = strtol(optarg, &end, 0); p = strtol(optarg, &end, 0);
if(*end != '\0' || p <= 0 || p > 0xFFFF) if(*end != '\0' || p <= 0 || p > 0xFFFF)
goto usage; goto usage;
flood_port = p; server_port = p;
} }
break; break;
case 'P': { case 'P': {
...@@ -270,7 +270,7 @@ main(int argc, char **argv) ...@@ -270,7 +270,7 @@ main(int argc, char **argv)
} else } else
goto usage; goto usage;
sin6.sin6_port = htons(port); sin6.sin6_port = htons(port);
find_neighbour(&sin6, 1, 0, 1); flood_connect(&sin6);
} else { } else {
goto usage; goto usage;
} }
...@@ -339,47 +339,50 @@ main(int argc, char **argv) ...@@ -339,47 +339,50 @@ main(int argc, char **argv)
} }
while(1) { while(1) {
fd_set readfds; fd_set readfds, writefds;
int nls = netlink_socket(); int nls = netlink_socket();
int maxfd; int maxfd;
struct timespec deadline; struct timespec now, deadline;
FD_ZERO(&readfds); FD_ZERO(&readfds);
FD_ZERO(&writefds);
FD_SET(nls, &readfds); FD_SET(nls, &readfds);
maxfd = nls;
if(numinterfaces > 0) { if(numinterfaces > 0) {
FD_SET(ra_socket, &readfds); FD_SET(ra_socket, &readfds);
maxfd = max(maxfd, ra_socket);
FD_SET(dhcpv4_socket, &readfds); FD_SET(dhcpv4_socket, &readfds);
maxfd = max(maxfd, dhcpv4_socket);
} }
FD_SET(flood_socket, &readfds);
maxfd = max(nls, flood_socket); FD_SET(server_socket, &readfds);
if(numinterfaces > 0) maxfd = max(maxfd, server_socket);
maxfd = max(maxfd, max(ra_socket, dhcpv4_socket));
if(flood_time.tv_sec > 0) { for(int i = 0; i < numneighs; i++) {
struct timespec now; if(neighs[i].fd >= 0) {
clock_gettime(CLOCK_MONOTONIC, &now); if(neighs[i].out.len > 0)
ts_minus(&deadline, &flood_time, &now); FD_SET(neighs[i].fd, &writefds);
if(deadline.tv_sec < 0) { FD_SET(neighs[i].fd, &readfds);
deadline.tv_sec = 0; maxfd = max(maxfd, neighs[i].fd);
deadline.tv_nsec = 0;
} }
} }
rc = pselect(maxfd + 1, &readfds, NULL, NULL, clock_gettime(CLOCK_MONOTONIC, &now);
flood_time.tv_sec > 0 ? &deadline : NULL, NULL); ts_minus(&deadline, &expire_neighs_time, &now);
rc = pselect(maxfd + 1, &readfds, &writefds, NULL, &deadline, NULL);
if(rc < 0 && errno != EINTR) { if(rc < 0 && errno != EINTR) {
perror("pselect"); perror("pselect");
sleep(1); sleep(1);
} }
clock_gettime(CLOCK_MONOTONIC, &now);
if(exiting) if(exiting)
break; break;
if(dumping) { if(dumping) {
static const char zeroes[8] = {0}; static const char zeroes[8] = {0};
struct timespec now;
clock_gettime(CLOCK_MONOTONIC, &now);
printf("Interfaces"); printf("Interfaces");
for(int i = 0; i < numinterfaces; i++) for(int i = 0; i < numinterfaces; i++)
printf(" %s", interfaces[i].ifname); printf(" %s", interfaces[i].ifname);
...@@ -462,15 +465,8 @@ main(int argc, char **argv) ...@@ -462,15 +465,8 @@ main(int argc, char **argv)
printf(".\n"); printf(".\n");
} }
printf("\n"); printf("\n");
for(int i = 0; i < numneighbours; i++) { for(int i = 0; i < numneighs; i++) {
char buf[INET6_ADDRSTRLEN]; printf("Neighbour %d.\n", neighs[i].fd);
inet_ntop(AF_INET6, &neighbours[i].addr.sin6_addr,
buf, sizeof(buf));
printf("Neighbour %s:%d %ds %ds%s.\n",
buf, ntohs(neighbours[i].addr.sin6_port),
(int)(now.tv_sec - neighbours[i].time),
(int)(now.tv_sec - neighbours[i].send_time),
neighbours[i].permanent ? " (permanent)" : "");
} }
fflush(stdout); fflush(stdout);
...@@ -478,26 +474,24 @@ main(int argc, char **argv) ...@@ -478,26 +474,24 @@ main(int argc, char **argv)
dumping = 0; dumping = 0;
} }
if(flood_time.tv_sec > 0) { if(rc >= 0) {
struct timespec now;
clock_gettime(CLOCK_MONOTONIC, &now);
if(ts_compare(&flood_time, &now) <= 0) {
periodic_flood();
}
}
if(rc <= 0)
continue;
if(FD_ISSET(nls, &readfds)) { if(FD_ISSET(nls, &readfds)) {
rc = netlink_listen(); rc = netlink_listen();
if(rc < 0) if(rc < 0)
nl_perror(rc, "netlink_listen"); nl_perror(rc, "netlink_listen");
} }
if(FD_ISSET(flood_socket, &readfds)) if(FD_ISSET(server_socket, &readfds))
flood_listen(); flood_accept();
for(int i = 0; i < numneighs; i++) {
if(neighs[i].fd >= 0) {
if(FD_ISSET(neighs[i].fd, &readfds))
flood_read(&neighs[i]);
if(FD_ISSET(neighs[i].fd, &writefds))
flood_write(&neighs[i]);
}
}
if(numinterfaces > 0) { if(numinterfaces > 0) {
if(FD_ISSET(ra_socket, &readfds)) if(FD_ISSET(ra_socket, &readfds))
receive_rs(); receive_rs();
...@@ -507,6 +501,10 @@ main(int argc, char **argv) ...@@ -507,6 +501,10 @@ main(int argc, char **argv)
} }
} }
if(ts_compare(&now, &expire_neighs_time) >= 0)
expire_neighs();
}
client_cleanup(); client_cleanup();
if(numinterfaces > 0) { if(numinterfaces > 0) {
ra_cleanup(); ra_cleanup();
......
...@@ -38,8 +38,8 @@ DHCPv4 or IPv6 RA. This option may be specified multiple times, in which ...@@ -38,8 +38,8 @@ DHCPv4 or IPv6 RA. This option may be specified multiple times, in which
case all prefixes will be announced to clients. case all prefixes will be announced to clients.
.TP .TP
.BI \-f " port" .BI \-f " port"
Set the local UDP port used by the flooding protocol. If this is not set, Set the server TCP port used by the flooding protocol. If this is not
flooding is disabled. set, we don't act as a server.
.TP .TP
.BI \-F " address:port" .BI \-F " address:port"
Specify the address of a remote peer for the flooding protocol. There is Specify the address of a remote peer for the flooding protocol. There is
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment