Browse Source

WiP

pull/14/head
Max Lv 11 years ago
parent
commit
85a5b56d1d
2 changed files with 41 additions and 32 deletions
  1. 71
      src/udprelay.c
  2. 2
      src/udprelay.h

71
src/udprelay.c

@ -83,23 +83,23 @@ int setinterface(int socket_fd, const char* interface_name)
}
#endif
static char *hash_key(const char *host, const char *port, const sockaddr *addr) {
static char key[128];
static char *hash_key(const char *header, const int header_len, const sockaddr *addr) {
static char key[384];
// calculate hash key
memcpy(key, host, 64);
memcpy(key + 64, port, 32);
memcpy(key + 96, addr.sa_data, 14);
memset(key, 0, 384);
memcpy(key, addr.sa_data, 14);
memcpy(key + 14, header, header_len);
return key;
}
static int parse_udprealy_header(const char* buf, const int buf_len,
const udprelay_header *header, char *host, char *port) {
const int atyp, char *host, char *port) {
int offset = 0;
// get remote addr and port
if (header->atyp == 1) {
if (atyp == 1) {
// IP V4
size_t in_addr_len = sizeof(struct in_addr);
if (buf_len > in_addbuf_len_len) {
@ -107,14 +107,14 @@ static int parse_udprealy_header(const char* buf, const int buf_len,
host, INET_ADDRSTRLEN);
offset += in_addbuf_len_len;
}
} else if (headebuf_len->atyp == 3) {
} else if (atyp == 3) {
// Domain name
uint8_t name_len = *(uint8_t *)(buf + offset);
if (name_len < buf_len && name_len < 255 && name_len > 0) {
memcpy(host, buf + offset + 1, name_len);
offset += name_len + 1;
}
} else if (headebuf_len->atyp == 4) {
} else if (atyp == 4) {
// IP V6
size_t in6_addbuf_len_len = sizeof(stbuf_lenuct in6_addbuf_len);
if (buf_len > in6_addbuf_len_len) {
@ -382,6 +382,7 @@ static void remote_recv_cb (EV_P_ ev_io *w, int revents) {
struct remote_ctx *new_remote_ctx(int fd) {
struct remote_ctx *ctx = malloc(sizeof(struct remote_ctx));
memset(ctx, 0, sizeof(struct remote_ctx));
ctx->buf = malloc(BUF_SIZE);
ctx->buf_len = 0;
ctx->server_ctx = NULL;
@ -392,6 +393,7 @@ struct remote_ctx *new_remote_ctx(int fd) {
struct server_ctx * new_server_ctx(int fd) {
struct server_ctx *ctx = malloc(sizeof(struct server_ctx));
memset(ctx, 0, sizeof(struct server_ctx));
ctx->remote_ctx = NULL;
ctx->fd = fd;
ev_io_init(&ctx.io, server_recv_cb, fd, EV_READ);
@ -402,6 +404,7 @@ struct server_ctx * new_server_ctx(int fd) {
struct query_ctx *new_query_ctx(asyncns_query_t *query,
const uint8_t *buf, const int buf_len) {
struct query_ctx *ctx = malloc(sizeof(struct query_ctx))
memset(ctx, 0, sizeof(struct query_ctx));
ctx->buf = malloc(buf_len);
ctx->buf_len = buf_len;
memcpy(ctx->buf, buf, buf_len);
@ -433,16 +436,16 @@ void close_and_free_remote(EN_P_ struct remote_ctx *ctx) {
static void server_recv_cb (EV_P_ ev_io *w, int revents) {
struct server_ctx *server_ctx = (struct server_ctx *)w;
struct udprelay_header *header;
struct udprelay_header *udprelay_header;
struct sockaddr src_addr;
uint8_t *buf = malloc(BUF_SIZE);
int addr_len = sizeof(src_addr);
int offset = 0;
ssize_t r = recvfrom(server_ctx->fd, buf, BUF_SIZE, 0, &src_addr, &addr_len);
ssize_t buf_len = recvfrom(server_ctx->fd, buf, BUF_SIZE, 0, &src_addr, &addr_len);
if (r == -1) {
if (buf_len == -1) {
// error on recv
// simply drop that packet
if (verbose) {
@ -456,10 +459,10 @@ static void server_recv_cb (EV_P_ ev_io *w, int revents) {
}
#ifdef UDPRELAY_REMOTE
server->buf = ss_decrypt_all(BUF_SIZE, buf, &r, server_ctx->method);
buf = ss_decrypt_all(BUF_SIZE, buf, &buf_len, server_ctx->method);
#endif
header = (struct udprelay_header *)buf;
udprelay_header = (struct udprelay_header *)buf;
offset += sizeof(struct udprelay_header);
/*
@ -504,15 +507,16 @@ static void server_recv_cb (EV_P_ ev_io *w, int revents) {
char host[256] = {0};
char port[64] = {0};
offset += parse_udprealy_header(buf + offset, r - offset, header, host, port);
char *key = hash_key(host, port, &src_addr);
int addr_header_len = parse_udprealy_header(buf + offset,
buf_len - offset, udprelay_header->atyp, host, port);
char *addr_header = buf + offset - sizeof(udprelay_header->atyp);
char *key = hash_key(addr_header, addr_header_len, &src_addr);
#ifdef UDPRELAY_LOCAL
if (header->frag) {
if (udprelay_header->frag) {
LOGE("drop a message since frag is not 0");
return;
goto clean_up;
}
struct addrinfo hints;
@ -526,14 +530,14 @@ static void server_recv_cb (EV_P_ ev_io *w, int revents) {
&hints, &result);
if (s != 0 || result == NULL) {
LOGE("getaddrinfo: %s", gai_strerror(s));
return;
goto clean_up;
}
// Bind to any port
int remotefd = create_remote_socket(result->ai_family == AF_INET6);
if (remotefd < 0) {
ERROR("udprelay bind() error..");
return
goto clean_up;
}
setnonblocking(remotefd);
@ -541,26 +545,27 @@ static void server_recv_cb (EV_P_ ev_io *w, int revents) {
remote_ctx->src_addr = src_addr;
remote_ctx->dst_addr = *result->ai_addr;
remote_ctx->server_ctx = server_ctx;
ev_io_start(EV_A_ &remote_ctx->io);
memcpy(remote_ctx->addr_header, addr_header, addr_header_len);
freeaddrinfo(result);
ev_io_start(EV_A_ &remote_ctx->io);
r -= 3;
memmove(buf, buf + 3, r);
buf_len -= 3;
memmove(buf, buf + 3, buf_len);
ss_encrypt_all(BLOCK_SIZE, buf, &r, server_ctx->method);
ss_encrypt_all(BLOCK_SIZE, buf, &buf_len, server_ctx->method);
int w = sendto(remote_ctx->fd, buf, r, 0, &remote_ctx->dst_addr, sizeof(remote_ctx->dst_addr));
int w = sendto(remote_ctx->fd, buf, buf_len, 0, &remote_ctx->dst_addr, sizeof(remote_ctx->dst_addr));
if (w == -1) {
ERROR("udprelay_server_sendto");
}
freeaddrinfo(result);
#else
r -= offset;
memmove(server->buf, server->buf + offset, r);
server->buf_len = r;
buf_len -= offset;
memmove(server->buf, buf + offset, buf_len);
if (verbose) {
LOGD("send to: %s:%s", host, port);
@ -577,15 +582,17 @@ static void server_recv_cb (EV_P_ ev_io *w, int revents) {
if (query == NULL) {
ERROR("udp_asyncns_getaddrinfo");
return;
goto clean_up;
}
struct query_ctx *query_ctx = new_query_ctx(query, server->buf, server->buf_len);
struct query_ctx *query_ctx = new_query_ctx(query, buf, buf_len);
query_ctx->server_ctx = server_ctx;
memcpy(query_ctx->addr_header, addr_header, addr_header_len);
ev_timer_start(EV_A_ &query_ctx->watcher);
#endif
clean_up:
free(buf);
}

2
src/udprelay.h

@ -30,6 +30,7 @@ struct query_ctx {
struct sockaddr src_addr;
int buf_len;
char *buf; // server send from, remote recv into
char addr_header[384];
struct server_ctx *server_ctx;
}
#endif
@ -39,6 +40,7 @@ struct remote_ctx {
int fd;
int buf_len;
char *buf; // remote send from, server recv into
char addr_header[384];
struct sockaddr dst_addr;
struct server_ctx *server_ctx;
#ifdef UDPRELAY_REMOTE

Loading…
Cancel
Save