Compare commits
14 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
35a9bf27df | ||
|
|
ba0af8cc20 | ||
|
|
ec85be5c6a | ||
|
|
fe989851ab | ||
|
|
46842dd200 | ||
|
|
e5eb30598d | ||
|
|
c81ad0a7c6 | ||
|
|
97641d6dda | ||
|
|
e48dac775c | ||
|
|
6282f36ac7 | ||
|
|
cf022af4a9 | ||
|
|
597c586657 | ||
|
|
fc7f9aa0c8 | ||
|
|
7492f977b6 |
@@ -13,7 +13,9 @@ glorytun_SOURCES = \
|
|||||||
src/option.c \
|
src/option.c \
|
||||||
src/option.h \
|
src/option.h \
|
||||||
src/tun.c \
|
src/tun.c \
|
||||||
src/tun.h
|
src/tun.h \
|
||||||
|
src/db.c \
|
||||||
|
src/db.h
|
||||||
|
|
||||||
EXTRA_DIST = \
|
EXTRA_DIST = \
|
||||||
LICENSE \
|
LICENSE \
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ AC_INIT([glorytun],
|
|||||||
[https://github.com/angt/glorytun/issues],
|
[https://github.com/angt/glorytun/issues],
|
||||||
[glorytun],
|
[glorytun],
|
||||||
[https://github.com/angt/glorytun])
|
[https://github.com/angt/glorytun])
|
||||||
|
AC_DEFINE_UNQUOTED([VERSION_MAJOR], [m4_esyscmd([./version.sh major])])
|
||||||
AC_CONFIG_SRCDIR([src/common.h])
|
AC_CONFIG_SRCDIR([src/common.h])
|
||||||
AC_CONFIG_AUX_DIR([build-aux])
|
AC_CONFIG_AUX_DIR([build-aux])
|
||||||
AC_CONFIG_MACRO_DIR([m4])
|
AC_CONFIG_MACRO_DIR([m4])
|
||||||
|
|||||||
@@ -50,16 +50,18 @@ static inline int str_empty (const char *restrict str)
|
|||||||
}
|
}
|
||||||
|
|
||||||
_pure_
|
_pure_
|
||||||
static inline int str_cmp (const char *restrict sa, const char *restrict sb)
|
static inline size_t str_cmp (const char *restrict sa, const char *restrict sb)
|
||||||
{
|
{
|
||||||
if (!sa || !sb)
|
if (!sa || !sb)
|
||||||
return 1;
|
return 1;
|
||||||
|
|
||||||
while (*sa==*sb++)
|
size_t i = 0;
|
||||||
if (!*sa++)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
return 1;
|
while (sa[i]==sb[i])
|
||||||
|
if (!sa[i++])
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
return i+1;
|
||||||
}
|
}
|
||||||
|
|
||||||
_pure_
|
_pure_
|
||||||
|
|||||||
@@ -36,7 +36,7 @@ void gt_fatal (const char *fmt, ...)
|
|||||||
|
|
||||||
void gt_na (const char *name)
|
void gt_na (const char *name)
|
||||||
{
|
{
|
||||||
gt_log("%s is not available on your platform!\n", name);
|
gt_log("%s is not available on your platform\n", name);
|
||||||
}
|
}
|
||||||
|
|
||||||
int gt_tohex (char *dst, size_t dst_size, const uint8_t *src, size_t src_size)
|
int gt_tohex (char *dst, size_t dst_size, const uint8_t *src, size_t src_size)
|
||||||
|
|||||||
14
src/common.h
14
src/common.h
@@ -17,12 +17,14 @@
|
|||||||
#define _1_(x) (__builtin_expect((x), 1))
|
#define _1_(x) (__builtin_expect((x), 1))
|
||||||
#define _0_(x) (__builtin_expect((x), 0))
|
#define _0_(x) (__builtin_expect((x), 0))
|
||||||
|
|
||||||
#define _printf_(A,B) __attribute__((format(printf,A,B)))
|
#define CLZ(x) (__builtin_clz(x))
|
||||||
#define _noreturn_ __attribute__((noreturn))
|
|
||||||
#define _unused_ __attribute__((unused))
|
#define _printf_(A,B) __attribute__ ((format(printf,A,B)))
|
||||||
#define _pure_ __attribute__((pure))
|
#define _noreturn_ __attribute__ ((noreturn))
|
||||||
#define _const_ __attribute__((const))
|
#define _unused_ __attribute__ ((unused))
|
||||||
#define _align_(...) __attribute__((aligned(__VA_ARGS__)))
|
#define _pure_ __attribute__ ((pure))
|
||||||
|
#define _const_ __attribute__ ((const))
|
||||||
|
#define _align_(...) __attribute__ ((aligned(__VA_ARGS__)))
|
||||||
|
|
||||||
typedef struct buffer buffer_t;
|
typedef struct buffer buffer_t;
|
||||||
|
|
||||||
|
|||||||
153
src/db.c
Normal file
153
src/db.c
Normal file
@@ -0,0 +1,153 @@
|
|||||||
|
#include "db.h"
|
||||||
|
#include "common-static.h"
|
||||||
|
|
||||||
|
#define CBIT(X) (1&(intptr_t)(X))
|
||||||
|
#define CBIT_PTR(X) (uint8_t *)(1|(intptr_t)(X))
|
||||||
|
#define CBIT_NODE(X) (struct node *)(1^(intptr_t)(X))
|
||||||
|
|
||||||
|
struct node {
|
||||||
|
uint8_t *child[2];
|
||||||
|
uint32_t point;
|
||||||
|
};
|
||||||
|
|
||||||
|
_pure_
|
||||||
|
static inline size_t db_size (const uint8_t *a)
|
||||||
|
{
|
||||||
|
return (a[0]?:str_len((char *)a+1))+1;
|
||||||
|
}
|
||||||
|
|
||||||
|
_pure_
|
||||||
|
static inline size_t db_cmp (const uint8_t *a, const uint8_t *b)
|
||||||
|
{
|
||||||
|
const size_t size = a[0];
|
||||||
|
|
||||||
|
if (size!=b[0])
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
if (!size) {
|
||||||
|
size_t i = str_cmp((char *)a+1, (char *)b+1);
|
||||||
|
return i?i+1:0;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (size_t i=1; i<=size; i++) {
|
||||||
|
if (a[i]!=b[i])
|
||||||
|
return i+1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
_pure_
|
||||||
|
static inline int db_dir (const uint32_t point, uint8_t *data, const size_t size)
|
||||||
|
{
|
||||||
|
const size_t pos = point>>8;
|
||||||
|
|
||||||
|
if (pos>=size)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
return ((point|data[pos])&255)==255;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint8_t *db_search (uint8_t **p, uint8_t *data)
|
||||||
|
{
|
||||||
|
if _0_(!*p)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
uint8_t *r = *p;
|
||||||
|
const size_t size = db_size(data);
|
||||||
|
|
||||||
|
while (CBIT(r)) {
|
||||||
|
struct node *node = CBIT_NODE(r);
|
||||||
|
r = node->child[db_dir(node->point, data, size)];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!db_cmp(r, data))
|
||||||
|
return r;
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint8_t *db_insert (uint8_t **p, uint8_t *data)
|
||||||
|
{
|
||||||
|
if _0_(CBIT(data))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
if _0_(!*p) {
|
||||||
|
*p = data;
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint8_t *r = *p;
|
||||||
|
size_t size = db_size(data);
|
||||||
|
|
||||||
|
while (CBIT(r)) {
|
||||||
|
struct node *node = CBIT_NODE(r);
|
||||||
|
r = node->child[db_dir(node->point, data, size)];
|
||||||
|
}
|
||||||
|
|
||||||
|
const size_t diff = db_cmp(r, data);
|
||||||
|
|
||||||
|
if _0_(!diff)
|
||||||
|
return r;
|
||||||
|
|
||||||
|
const size_t pos = diff-1;
|
||||||
|
const uint8_t mask = ~((1u<<31)>>CLZ(r[pos]^data[pos]));
|
||||||
|
const size_t point = (pos<<8)|mask;
|
||||||
|
|
||||||
|
while (CBIT(*p)) {
|
||||||
|
struct node *node = CBIT_NODE(*p);
|
||||||
|
|
||||||
|
if (node->point>point)
|
||||||
|
break;
|
||||||
|
|
||||||
|
p = node->child+db_dir(node->point, data, size);
|
||||||
|
}
|
||||||
|
|
||||||
|
struct node *node = malloc(sizeof(struct node));
|
||||||
|
|
||||||
|
if _0_(!node)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
const int dir = (mask|r[pos])==255;
|
||||||
|
|
||||||
|
node->child[dir] = *p;
|
||||||
|
node->child[1-dir] = data;
|
||||||
|
node->point = point;
|
||||||
|
|
||||||
|
*p = CBIT_PTR(node);
|
||||||
|
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint8_t *db_remove (uint8_t **p, uint8_t *data)
|
||||||
|
{
|
||||||
|
if _0_(!*p)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
const size_t size = db_size(data);
|
||||||
|
|
||||||
|
uint8_t **p_old = NULL;
|
||||||
|
struct node *node = NULL;
|
||||||
|
int dir = 0;
|
||||||
|
|
||||||
|
while (CBIT(*p)) {
|
||||||
|
p_old = p;
|
||||||
|
node = CBIT_NODE(*p);
|
||||||
|
dir = db_dir(node->point, data, size);
|
||||||
|
p = node->child+dir;
|
||||||
|
}
|
||||||
|
|
||||||
|
if _0_(db_cmp(data, *p))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
uint8_t *r = *p;
|
||||||
|
|
||||||
|
if (p_old) {
|
||||||
|
*p_old = node->child[1-dir];
|
||||||
|
free(node);
|
||||||
|
} else {
|
||||||
|
*p = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
return r;
|
||||||
|
}
|
||||||
7
src/db.h
Normal file
7
src/db.h
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <stdint.h>
|
||||||
|
|
||||||
|
uint8_t *db_search (uint8_t **, uint8_t *);
|
||||||
|
uint8_t *db_insert (uint8_t **, uint8_t *);
|
||||||
|
uint8_t *db_remove (uint8_t **, uint8_t *);
|
||||||
132
src/main.c
132
src/main.c
@@ -2,9 +2,11 @@
|
|||||||
#include <limits.h>
|
#include <limits.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <signal.h>
|
#include <signal.h>
|
||||||
|
#include <string.h>
|
||||||
#include <poll.h>
|
#include <poll.h>
|
||||||
#include <sys/time.h>
|
#include <sys/time.h>
|
||||||
#include <sys/fcntl.h>
|
#include <sys/fcntl.h>
|
||||||
|
#include <sys/stat.h>
|
||||||
|
|
||||||
#ifndef __FAVOR_BSD
|
#ifndef __FAVOR_BSD
|
||||||
#define __FAVOR_BSD
|
#define __FAVOR_BSD
|
||||||
@@ -341,7 +343,7 @@ static void gt_set_signal (void)
|
|||||||
|
|
||||||
static ssize_t fd_read (int fd, void *data, size_t size)
|
static ssize_t fd_read (int fd, void *data, size_t size)
|
||||||
{
|
{
|
||||||
if (!size)
|
if ((fd==-1) || !size)
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
ssize_t ret = read(fd, data, size);
|
ssize_t ret = read(fd, data, size);
|
||||||
@@ -361,7 +363,7 @@ static ssize_t fd_read (int fd, void *data, size_t size)
|
|||||||
|
|
||||||
static ssize_t fd_write (int fd, const void *data, size_t size)
|
static ssize_t fd_write (int fd, const void *data, size_t size)
|
||||||
{
|
{
|
||||||
if (!size)
|
if ((fd==-1) || !size)
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
ssize_t ret = write(fd, data, size);
|
ssize_t ret = write(fd, data, size);
|
||||||
@@ -382,6 +384,11 @@ static ssize_t fd_write (int fd, const void *data, size_t size)
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static ssize_t fd_write_str (int fd, const char *str)
|
||||||
|
{
|
||||||
|
return fd_write(fd, str, str_len(str));
|
||||||
|
}
|
||||||
|
|
||||||
static size_t fd_read_all (int fd, void *data, size_t size)
|
static size_t fd_read_all (int fd, void *data, size_t size)
|
||||||
{
|
{
|
||||||
size_t done = 0;
|
size_t done = 0;
|
||||||
@@ -505,11 +512,37 @@ static int gt_decrypt (struct crypto_ctx *ctx, buffer_t *dst, buffer_t *src)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void gt_print_hdr (const int ip_version, uint8_t *data, size_t ip_size, const char *sockname)
|
_pure_
|
||||||
|
static inline uint32_t sum16 (uint32_t sum, const uint8_t *data, const size_t size)
|
||||||
|
{
|
||||||
|
const size_t lim = size&~1u;
|
||||||
|
|
||||||
|
for (size_t i=0; i<lim; i+=2)
|
||||||
|
sum += (data[i]<<8)|data[i+1];
|
||||||
|
|
||||||
|
if (size&1)
|
||||||
|
sum += data[size-1]<<8;
|
||||||
|
|
||||||
|
return sum;
|
||||||
|
}
|
||||||
|
|
||||||
|
_const_
|
||||||
|
static inline uint16_t sum16_final (uint32_t sum)
|
||||||
|
{
|
||||||
|
sum = (sum>>16)+(sum&0xFFFF);
|
||||||
|
return ~(sum+(sum>>16));
|
||||||
|
}
|
||||||
|
|
||||||
|
static void gt_print_hdr (const int ip_version, uint8_t *data, size_t ip_size)
|
||||||
{
|
{
|
||||||
const ssize_t ip_proto = ip_get_proto(ip_version, data, ip_size);
|
const ssize_t ip_proto = ip_get_proto(ip_version, data, ip_size);
|
||||||
const ssize_t ip_hdr_size = ip_get_hdr_size(ip_version, data, ip_size);
|
const ssize_t ip_hdr_size = ip_get_hdr_size(ip_version, data, ip_size);
|
||||||
|
|
||||||
|
if (ip_proto<0 || ip_hdr_size<=0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
uint32_t sum = (size_t)ip_proto+ip_size-(size_t)ip_hdr_size;
|
||||||
|
|
||||||
char ip_src[INET6_ADDRSTRLEN];
|
char ip_src[INET6_ADDRSTRLEN];
|
||||||
char ip_dst[INET6_ADDRSTRLEN];
|
char ip_dst[INET6_ADDRSTRLEN];
|
||||||
|
|
||||||
@@ -517,40 +550,45 @@ static void gt_print_hdr (const int ip_version, uint8_t *data, size_t ip_size, c
|
|||||||
case 4:
|
case 4:
|
||||||
inet_ntop(AF_INET, &data[12], ip_src, sizeof(ip_src));
|
inet_ntop(AF_INET, &data[12], ip_src, sizeof(ip_src));
|
||||||
inet_ntop(AF_INET, &data[16], ip_dst, sizeof(ip_dst));
|
inet_ntop(AF_INET, &data[16], ip_dst, sizeof(ip_dst));
|
||||||
|
sum = sum16(sum, &data[12], 2*4);
|
||||||
break;
|
break;
|
||||||
case 6:
|
case 6:
|
||||||
inet_ntop(AF_INET6, &data[9], ip_src, sizeof(ip_src));
|
inet_ntop(AF_INET6, &data[9], ip_src, sizeof(ip_src));
|
||||||
inet_ntop(AF_INET6, &data[25], ip_dst, sizeof(ip_dst));
|
inet_ntop(AF_INET6, &data[25], ip_dst, sizeof(ip_dst));
|
||||||
|
sum = sum16(sum, &data[9], 2*16); // XXX
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
gt_log("%s: version=%i size=%zi proto=%zi src=%s dst=%s\n", sockname, ip_version, ip_size, ip_proto, ip_src, ip_dst);
|
if (ip_proto==IPPROTO_TCP) {
|
||||||
|
|
||||||
if (ip_hdr_size<=0)
|
|
||||||
return;
|
|
||||||
|
|
||||||
if (ip_proto==6) {
|
|
||||||
struct tcphdr tcp;
|
struct tcphdr tcp;
|
||||||
|
|
||||||
byte_cpy(&tcp, &data[ip_hdr_size], sizeof(tcp));
|
byte_cpy(&tcp, &data[ip_hdr_size], sizeof(tcp));
|
||||||
|
|
||||||
|
uint16_t tcp_sum = ntohs(tcp.th_sum);
|
||||||
|
tcp.th_sum = 0;
|
||||||
|
|
||||||
|
sum = sum16(sum, (uint8_t *)&tcp, sizeof(tcp));
|
||||||
|
sum = sum16(sum, &data[ip_hdr_size+sizeof(tcp)], ip_size-ip_hdr_size-sizeof(tcp));
|
||||||
|
uint16_t computed_sum = sum16_final(sum);
|
||||||
|
|
||||||
tcp.th_sport = ntohs(tcp.th_sport);
|
tcp.th_sport = ntohs(tcp.th_sport);
|
||||||
tcp.th_dport = ntohs(tcp.th_dport);
|
tcp.th_dport = ntohs(tcp.th_dport);
|
||||||
tcp.th_seq = ntohl(tcp.th_seq);
|
tcp.th_seq = ntohl(tcp.th_seq);
|
||||||
tcp.th_ack = ntohl(tcp.th_ack);
|
tcp.th_ack = ntohl(tcp.th_ack);
|
||||||
tcp.th_win = ntohs(tcp.th_win);
|
tcp.th_win = ntohs(tcp.th_win);
|
||||||
|
|
||||||
gt_log("%s: tcp src=%u dst=%u seq=%u ack=%u win=%u %c%c%c%c%c%c\n",
|
gt_print("proto:%zi\tsrc:%s.%u\tdst:%s.%u\tseq:%u\tack:%u\twin:%u\tsize:%zu\tflags:%c%c%c%c%c%c\tsum:%i\n",
|
||||||
sockname, tcp.th_sport, tcp.th_dport, tcp.th_seq, tcp.th_ack, tcp.th_win,
|
ip_proto, ip_src, tcp.th_sport, ip_dst, tcp.th_dport,
|
||||||
|
tcp.th_seq, tcp.th_ack, tcp.th_win, ip_size-ip_hdr_size+tcp.th_off*4,
|
||||||
(tcp.th_flags&TH_FIN) ?'F':'.',
|
(tcp.th_flags&TH_FIN) ?'F':'.',
|
||||||
(tcp.th_flags&TH_SYN) ?'S':'.',
|
(tcp.th_flags&TH_SYN) ?'S':'.',
|
||||||
(tcp.th_flags&TH_RST) ?'R':'.',
|
(tcp.th_flags&TH_RST) ?'R':'.',
|
||||||
(tcp.th_flags&TH_PUSH)?'P':'.',
|
(tcp.th_flags&TH_PUSH)?'P':'.',
|
||||||
(tcp.th_flags&TH_ACK) ?'A':'.',
|
(tcp.th_flags&TH_ACK) ?'A':'.',
|
||||||
(tcp.th_flags&TH_URG) ?'U':'.');
|
(tcp.th_flags&TH_URG) ?'U':'.',
|
||||||
}
|
(computed_sum==tcp_sum));
|
||||||
|
|
||||||
if (ip_proto==17) {
|
} else if (ip_proto==IPPROTO_UDP) {
|
||||||
struct udphdr udp;
|
struct udphdr udp;
|
||||||
|
|
||||||
byte_cpy(&udp, &data[ip_hdr_size], sizeof(udp));
|
byte_cpy(&udp, &data[ip_hdr_size], sizeof(udp));
|
||||||
@@ -559,8 +597,11 @@ static void gt_print_hdr (const int ip_version, uint8_t *data, size_t ip_size, c
|
|||||||
udp.uh_dport = ntohs(udp.uh_dport);
|
udp.uh_dport = ntohs(udp.uh_dport);
|
||||||
udp.uh_ulen = ntohs(udp.uh_ulen);
|
udp.uh_ulen = ntohs(udp.uh_ulen);
|
||||||
|
|
||||||
gt_log("%s: udp src=%u dst=%u len=%u\n",
|
gt_print("proto:%zi\tsrc:%s.%u\tdst:%s.%u\tsize:%u\n",
|
||||||
sockname, udp.uh_sport, udp.uh_dport, udp.uh_ulen);
|
ip_proto, ip_src, udp.uh_sport, ip_dst, udp.uh_dport, udp.uh_ulen-8);
|
||||||
|
} else {
|
||||||
|
gt_print("proto:%zi\tsrc:%s\tdst:%s\tsize:%zu\n",
|
||||||
|
ip_proto, ip_src, ip_dst, ip_size);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -610,10 +651,13 @@ static int gt_setup_secretkey (struct crypto_ctx *ctx, char *keyfile)
|
|||||||
|
|
||||||
static int gt_setup_crypto (struct crypto_ctx *ctx, int fd, int listener)
|
static int gt_setup_crypto (struct crypto_ctx *ctx, int fd, int listener)
|
||||||
{
|
{
|
||||||
|
const uint8_t gt[] = {'G', 'T', VERSION_MAJOR, 0 };
|
||||||
|
|
||||||
|
const size_t size = 96;
|
||||||
|
const size_t hash_size = 32;
|
||||||
|
|
||||||
const size_t nonce_size = crypto_aead_aes256gcm_NPUBBYTES;
|
const size_t nonce_size = crypto_aead_aes256gcm_NPUBBYTES;
|
||||||
const size_t public_size = crypto_scalarmult_SCALARBYTES;
|
const size_t public_size = crypto_scalarmult_SCALARBYTES;
|
||||||
const size_t hash_size = crypto_generichash_BYTES;
|
|
||||||
const size_t size = nonce_size + public_size + hash_size;
|
|
||||||
|
|
||||||
uint8_t secret[crypto_scalarmult_SCALARBYTES];
|
uint8_t secret[crypto_scalarmult_SCALARBYTES];
|
||||||
uint8_t shared[crypto_scalarmult_BYTES];
|
uint8_t shared[crypto_scalarmult_BYTES];
|
||||||
@@ -625,10 +669,14 @@ static int gt_setup_crypto (struct crypto_ctx *ctx, int fd, int listener)
|
|||||||
|
|
||||||
crypto_generichash_state state;
|
crypto_generichash_state state;
|
||||||
|
|
||||||
|
byte_set(data_w, 0, size);
|
||||||
randombytes_buf(data_w, nonce_size);
|
randombytes_buf(data_w, nonce_size);
|
||||||
|
|
||||||
randombytes_buf(secret, sizeof(secret));
|
randombytes_buf(secret, sizeof(secret));
|
||||||
crypto_scalarmult_base(&data_w[nonce_size], secret);
|
crypto_scalarmult_base(&data_w[nonce_size], secret);
|
||||||
|
|
||||||
|
byte_cpy(&data_w[size-hash_size-sizeof(gt)], gt, sizeof(gt));
|
||||||
|
|
||||||
crypto_generichash(&data_w[size-hash_size], hash_size,
|
crypto_generichash(&data_w[size-hash_size], hash_size,
|
||||||
data_w, size-hash_size, ctx->skey, sizeof(ctx->skey));
|
data_w, size-hash_size, ctx->skey, sizeof(ctx->skey));
|
||||||
|
|
||||||
@@ -638,6 +686,9 @@ static int gt_setup_crypto (struct crypto_ctx *ctx, int fd, int listener)
|
|||||||
if (fd_read_all(fd, data_r, size)!=size)
|
if (fd_read_all(fd, data_r, size)!=size)
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
|
if (memcmp(&data_r[size-hash_size-sizeof(gt)], gt, sizeof(gt)))
|
||||||
|
return -2;
|
||||||
|
|
||||||
crypto_generichash(hash, hash_size,
|
crypto_generichash(hash, hash_size,
|
||||||
data_r, size-hash_size, ctx->skey, sizeof(ctx->skey));
|
data_r, size-hash_size, ctx->skey, sizeof(ctx->skey));
|
||||||
|
|
||||||
@@ -698,6 +749,7 @@ int main (int argc, char **argv)
|
|||||||
char *dev = NULL;
|
char *dev = NULL;
|
||||||
char *keyfile = NULL;
|
char *keyfile = NULL;
|
||||||
char *congestion = NULL;
|
char *congestion = NULL;
|
||||||
|
char *statefile = NULL;
|
||||||
|
|
||||||
long buffer_size = GT_BUFFER_SIZE;
|
long buffer_size = GT_BUFFER_SIZE;
|
||||||
|
|
||||||
@@ -739,6 +791,7 @@ int main (int argc, char **argv)
|
|||||||
{ "noquickack", NULL, option_option },
|
{ "noquickack", NULL, option_option },
|
||||||
{ "retry", &retry_opts, option_option },
|
{ "retry", &retry_opts, option_option },
|
||||||
{ "daemon", NULL, option_option },
|
{ "daemon", NULL, option_option },
|
||||||
|
{ "statefile", &statefile, option_str },
|
||||||
{ "debug", NULL, option_option },
|
{ "debug", NULL, option_option },
|
||||||
{ "version", NULL, option_option },
|
{ "version", NULL, option_option },
|
||||||
{ NULL },
|
{ NULL },
|
||||||
@@ -760,7 +813,7 @@ int main (int argc, char **argv)
|
|||||||
|
|
||||||
if (buffer_size < 2048) {
|
if (buffer_size < 2048) {
|
||||||
buffer_size = 2048;
|
buffer_size = 2048;
|
||||||
gt_log("buffer size must be greater than 2048!\n");
|
gt_log("buffer size must be greater than 2048\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!listener) {
|
if (!listener) {
|
||||||
@@ -773,8 +826,13 @@ int main (int argc, char **argv)
|
|||||||
retry_count = 0;
|
retry_count = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (statefile && statefile[0]!='/') {
|
||||||
|
gt_log("statefile must be an absolute path\n");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
if (sodium_init()==-1) {
|
if (sodium_init()==-1) {
|
||||||
gt_log("libsodium initialization has failed!\n");
|
gt_log("libsodium initialization has failed\n");
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -844,8 +902,34 @@ int main (int argc, char **argv)
|
|||||||
chdir("/");
|
chdir("/");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int state_fd = -1;
|
||||||
|
|
||||||
|
if (statefile) {
|
||||||
|
state_fd = open(statefile, O_WRONLY);
|
||||||
|
|
||||||
|
if (state_fd==-1) {
|
||||||
|
if (errno!=EINTR)
|
||||||
|
perror("open statefile");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct stat st = {0};
|
||||||
|
|
||||||
|
if (fstat(state_fd, &st)==-1) {
|
||||||
|
perror("stat statefile");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!S_ISFIFO(st.st_mode)) {
|
||||||
|
gt_log("`%s' is not a fifo\n", statefile);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
long retry = 0;
|
long retry = 0;
|
||||||
|
|
||||||
|
fd_write_str(state_fd, "INITIALIZED");
|
||||||
|
|
||||||
while (!gt_close) {
|
while (!gt_close) {
|
||||||
if (retry_count>=0 && retry>=retry_count+1) {
|
if (retry_count>=0 && retry>=retry_count+1) {
|
||||||
gt_log("couldn't %s (%d attempt%s)\n", listener?"listen":"connect",
|
gt_log("couldn't %s (%d attempt%s)\n", listener?"listen":"connect",
|
||||||
@@ -911,7 +995,7 @@ int main (int argc, char **argv)
|
|||||||
|
|
||||||
retry = 0;
|
retry = 0;
|
||||||
|
|
||||||
gt_log("%s: running\n", sockname);
|
fd_write_str(state_fd, "STARTED");
|
||||||
|
|
||||||
fd_set rfds;
|
fd_set rfds;
|
||||||
FD_ZERO(&rfds);
|
FD_ZERO(&rfds);
|
||||||
@@ -985,7 +1069,7 @@ int main (int argc, char **argv)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if _0_(debug)
|
if _0_(debug)
|
||||||
gt_print_hdr(ip_version, data, ip_size, sockname);
|
gt_print_hdr(ip_version, data, ip_size);
|
||||||
|
|
||||||
blks[blk_write++].size = r;
|
blks[blk_write++].size = r;
|
||||||
blk_count++;
|
blk_count++;
|
||||||
@@ -1075,6 +1159,8 @@ int main (int argc, char **argv)
|
|||||||
ssize_t r = tun_write(tun.fd, tun.write.read, ip_size);
|
ssize_t r = tun_write(tun.fd, tun.write.read, ip_size);
|
||||||
|
|
||||||
if (r>0) {
|
if (r>0) {
|
||||||
|
if _0_(debug)
|
||||||
|
gt_print_hdr(ip_version, tun.write.read, ip_size);
|
||||||
tun.write.read += r;
|
tun.write.read += r;
|
||||||
} else {
|
} else {
|
||||||
gt_close |= !r;
|
gt_close |= !r;
|
||||||
@@ -1093,6 +1179,8 @@ int main (int argc, char **argv)
|
|||||||
close(sock.fd);
|
close(sock.fd);
|
||||||
sock.fd = -1;
|
sock.fd = -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fd_write_str(state_fd, "STOPPED");
|
||||||
}
|
}
|
||||||
|
|
||||||
freeaddrinfo(ai);
|
freeaddrinfo(ai);
|
||||||
|
|||||||
@@ -6,4 +6,6 @@
|
|||||||
[ -z "${VERSION}" ] && VERSION=`basename \`pwd\`` \
|
[ -z "${VERSION}" ] && VERSION=`basename \`pwd\`` \
|
||||||
&& VERSION=${VERSION#*-}
|
&& VERSION=${VERSION#*-}
|
||||||
|
|
||||||
|
[ "$1" = "major" ] && VERSION=${VERSION%%.*}
|
||||||
|
|
||||||
printf ${VERSION}
|
printf ${VERSION}
|
||||||
|
|||||||
Reference in New Issue
Block a user