mirror of
https://github.com/rofl0r/proxychains-ng
synced 2025-01-25 19:12:53 +08:00
360 lines
8.4 KiB
C
360 lines
8.4 KiB
C
#undef _GNU_SOURCE
|
|
#define _GNU_SOURCE
|
|
#undef _POSIX_C_SOURCE
|
|
#define _POSIX_C_SOURCE 200809L
|
|
#define _DARWIN_C_SOURCE
|
|
#define _XOPEN_SOURCE 600
|
|
#include <limits.h>
|
|
#include <pthread.h>
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <sys/select.h>
|
|
#include <assert.h>
|
|
#include <string.h>
|
|
#include <stdint.h>
|
|
#include <stddef.h>
|
|
#include <errno.h>
|
|
#include <sys/mman.h>
|
|
#include "allocator_thread.h"
|
|
#include "debug.h"
|
|
#include "ip_type.h"
|
|
#include "mutex.h"
|
|
#include "hash.h"
|
|
|
|
/* stuff for our internal translation table */
|
|
|
|
typedef struct {
|
|
uint32_t hash;
|
|
char* string;
|
|
} string_hash_tuple;
|
|
|
|
typedef struct {
|
|
uint32_t counter;
|
|
uint32_t capa;
|
|
string_hash_tuple** list;
|
|
} internal_ip_lookup_table;
|
|
|
|
static void *dumpstring(char* s, size_t len) {
|
|
char* p = malloc(len);
|
|
if(p) memcpy(p, s, len);
|
|
return p;
|
|
}
|
|
|
|
static pthread_mutex_t *internal_ips_lock;
|
|
static internal_ip_lookup_table *internal_ips;
|
|
|
|
uint32_t index_from_internal_ip(ip_type4 internalip) {
|
|
PFUNC();
|
|
ip_type4 tmp = internalip;
|
|
uint32_t ret;
|
|
ret = tmp.octet[3] + (tmp.octet[2] << 8) + (tmp.octet[1] << 16);
|
|
ret -= 1;
|
|
return ret;
|
|
}
|
|
|
|
char *string_from_internal_ip(ip_type4 internalip) {
|
|
PFUNC();
|
|
char *res = NULL;
|
|
uint32_t index = index_from_internal_ip(internalip);
|
|
if(index < internal_ips->counter)
|
|
res = internal_ips->list[index]->string;
|
|
return res;
|
|
}
|
|
|
|
extern unsigned int remote_dns_subnet;
|
|
ip_type4 make_internal_ip(uint32_t index) {
|
|
ip_type4 ret;
|
|
index++; // so we can start at .0.0.1
|
|
if(index > 0xFFFFFF)
|
|
return ip_type_invalid.addr.v4;
|
|
ret.octet[0] = remote_dns_subnet & 0xFF;
|
|
ret.octet[1] = (index & 0xFF0000) >> 16;
|
|
ret.octet[2] = (index & 0xFF00) >> 8;
|
|
ret.octet[3] = index & 0xFF;
|
|
return ret;
|
|
}
|
|
|
|
static ip_type4 ip_from_internal_list(char* name, size_t len) {
|
|
uint32_t hash = dalias_hash((char *) name);
|
|
size_t i;
|
|
ip_type4 res;
|
|
void* new_mem;
|
|
// see if we already have this dns entry saved.
|
|
if(internal_ips->counter) {
|
|
for(i = 0; i < internal_ips->counter; i++) {
|
|
if(internal_ips->list[i]->hash == hash && !strcmp(name, internal_ips->list[i]->string)) {
|
|
res = make_internal_ip(i);
|
|
PDEBUG("got cached ip for %s\n", name);
|
|
goto have_ip;
|
|
}
|
|
}
|
|
}
|
|
// grow list if needed.
|
|
if(internal_ips->capa < internal_ips->counter + 1) {
|
|
PDEBUG("realloc\n");
|
|
new_mem = realloc(internal_ips->list, (internal_ips->capa + 16) * sizeof(void *));
|
|
if(new_mem) {
|
|
internal_ips->capa += 16;
|
|
internal_ips->list = new_mem;
|
|
} else {
|
|
oom:
|
|
PDEBUG("out of mem\n");
|
|
goto err_plus_unlock;
|
|
}
|
|
}
|
|
|
|
res = make_internal_ip(internal_ips->counter);
|
|
if(res.as_int == ip_type_invalid.addr.v4.as_int)
|
|
goto err_plus_unlock;
|
|
|
|
string_hash_tuple tmp = { 0 };
|
|
new_mem = dumpstring((char*) &tmp, sizeof(string_hash_tuple));
|
|
if(!new_mem)
|
|
goto oom;
|
|
|
|
PDEBUG("creating new entry %d for ip of %s\n", (int) internal_ips->counter, name);
|
|
|
|
internal_ips->list[internal_ips->counter] = new_mem;
|
|
internal_ips->list[internal_ips->counter]->hash = hash;
|
|
|
|
new_mem = dumpstring((char*) name, len + 1);
|
|
|
|
if(!new_mem) {
|
|
internal_ips->list[internal_ips->counter] = 0;
|
|
goto oom;
|
|
}
|
|
internal_ips->list[internal_ips->counter]->string = new_mem;
|
|
|
|
internal_ips->counter += 1;
|
|
|
|
have_ip:
|
|
|
|
return res;
|
|
err_plus_unlock:
|
|
|
|
PDEBUG("return err\n");
|
|
return ip_type_invalid.addr.v4;
|
|
}
|
|
|
|
/* stuff for communication with the allocator thread */
|
|
|
|
enum at_msgtype {
|
|
ATM_GETIP,
|
|
ATM_GETNAME,
|
|
ATM_EXIT,
|
|
};
|
|
|
|
enum at_direction {
|
|
ATD_SERVER = 0,
|
|
ATD_CLIENT,
|
|
ATD_MAX,
|
|
};
|
|
|
|
struct at_msghdr {
|
|
enum at_msgtype msgtype;
|
|
size_t datalen;
|
|
};
|
|
|
|
static pthread_t allocator_thread;
|
|
int req_pipefd[2];
|
|
int resp_pipefd[2];
|
|
|
|
static int wait_data(int readfd) {
|
|
PFUNC();
|
|
fd_set fds;
|
|
FD_ZERO(&fds);
|
|
FD_SET(readfd, &fds);
|
|
int ret;
|
|
while((ret = select(readfd+1, &fds, NULL, NULL, NULL)) <= 0) {
|
|
if(ret < 0) {
|
|
int e = errno;
|
|
if(e == EINTR) continue;
|
|
#ifdef __GLIBC__
|
|
char emsg[1024];
|
|
char* x = strerror_r(errno, emsg, sizeof emsg);
|
|
dprintf(2, "select2: %s\n", x);
|
|
#endif
|
|
return 0;
|
|
}
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
static int trywrite(int fd, void* buf, size_t bytes) {
|
|
ssize_t ret;
|
|
unsigned char *out = buf;
|
|
again:
|
|
ret = write(fd, out, bytes);
|
|
switch(ret) {
|
|
case -1:
|
|
if(errno == EINTR) goto again;
|
|
case 0:
|
|
return 0;
|
|
default:
|
|
if(ret == bytes || !bytes) return 1;
|
|
out += ret;
|
|
bytes -= ret;
|
|
goto again;
|
|
}
|
|
}
|
|
|
|
static int sendmessage(enum at_direction dir, struct at_msghdr *hdr, void* data) {
|
|
static int* destfd[ATD_MAX] = { [ATD_SERVER] = &req_pipefd[1], [ATD_CLIENT] = &resp_pipefd[1] };
|
|
int ret = trywrite(*destfd[dir], hdr, sizeof *hdr);
|
|
if(ret && hdr->datalen) {
|
|
assert(hdr->datalen <= MSG_LEN_MAX);
|
|
ret = trywrite(*destfd[dir], data, hdr->datalen);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static int tryread(int fd, void* buf, size_t bytes) {
|
|
ssize_t ret;
|
|
unsigned char *out = buf;
|
|
again:
|
|
ret = read(fd, out, bytes);
|
|
switch(ret) {
|
|
case -1:
|
|
if(errno == EINTR) goto again;
|
|
case 0:
|
|
return 0;
|
|
default:
|
|
if(ret == bytes || !bytes) return 1;
|
|
out += ret;
|
|
bytes -= ret;
|
|
goto again;
|
|
}
|
|
}
|
|
|
|
static int getmessage(enum at_direction dir, struct at_msghdr *hdr, void* data) {
|
|
static int* readfd[ATD_MAX] = { [ATD_SERVER] = &req_pipefd[0], [ATD_CLIENT] = &resp_pipefd[0] };
|
|
ssize_t ret;
|
|
if((ret = wait_data(*readfd[dir]))) {
|
|
if(!tryread(*readfd[dir], hdr, sizeof *hdr))
|
|
return 0;
|
|
assert(hdr->datalen <= MSG_LEN_MAX);
|
|
if(hdr->datalen) {
|
|
ret = tryread(*readfd[dir], data, hdr->datalen);
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static void* threadfunc(void* x) {
|
|
(void) x;
|
|
int ret;
|
|
struct at_msghdr msg;
|
|
union {
|
|
char host[MSG_LEN_MAX];
|
|
ip_type4 ip;
|
|
} readbuf;
|
|
while((ret = getmessage(ATD_SERVER, &msg, &readbuf))) {
|
|
switch(msg.msgtype) {
|
|
case ATM_GETIP:
|
|
/* client wants an ip for a DNS name. iterate our list and check if we have an existing entry.
|
|
* if not, create a new one. */
|
|
readbuf.ip = ip_from_internal_list(readbuf.host, msg.datalen - 1);
|
|
msg.datalen = sizeof(ip_type4);
|
|
break;
|
|
case ATM_GETNAME: {
|
|
char *host = string_from_internal_ip(readbuf.ip);
|
|
if(host) {
|
|
size_t l = strlen(host);
|
|
assert(l < MSG_LEN_MAX);
|
|
memcpy(readbuf.host, host, l + 1);
|
|
msg.datalen = l + 1;
|
|
}
|
|
break;
|
|
}
|
|
case ATM_EXIT:
|
|
return 0;
|
|
default:
|
|
abort();
|
|
}
|
|
ret = sendmessage(ATD_CLIENT, &msg, &readbuf);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/* API to access the internal ip mapping */
|
|
|
|
ip_type4 at_get_ip_for_host(char* host, size_t len) {
|
|
ip_type4 readbuf;
|
|
MUTEX_LOCK(internal_ips_lock);
|
|
if(len > MSG_LEN_MAX) goto inv;
|
|
struct at_msghdr msg = {.msgtype = ATM_GETIP, .datalen = len + 1 };
|
|
if(sendmessage(ATD_SERVER, &msg, host) &&
|
|
getmessage(ATD_CLIENT, &msg, &readbuf));
|
|
else {
|
|
inv:
|
|
readbuf = ip_type_invalid.addr.v4;
|
|
}
|
|
assert(msg.msgtype == ATM_GETIP);
|
|
MUTEX_UNLOCK(internal_ips_lock);
|
|
return readbuf;
|
|
}
|
|
|
|
size_t at_get_host_for_ip(ip_type4 ip, char* readbuf) {
|
|
struct at_msghdr msg = {.msgtype = ATM_GETNAME, .datalen = sizeof(ip_type4) };
|
|
size_t res = 0;
|
|
MUTEX_LOCK(internal_ips_lock);
|
|
if(sendmessage(ATD_SERVER, &msg, &ip) && getmessage(ATD_CLIENT, &msg, readbuf)) {
|
|
if((ptrdiff_t) msg.datalen <= 0) res = 0;
|
|
else res = msg.datalen - 1;
|
|
}
|
|
assert(msg.msgtype == ATM_GETNAME);
|
|
MUTEX_UNLOCK(internal_ips_lock);
|
|
return res;
|
|
}
|
|
|
|
|
|
static void initpipe(int* fds) {
|
|
if(pipe(fds) == -1) {
|
|
perror("pipe");
|
|
exit(1);
|
|
}
|
|
}
|
|
|
|
#ifndef MAX
|
|
#define MAX(x, y) ((x) > (y) ? (x) : (y))
|
|
#endif
|
|
|
|
#if !defined(PTHREAD_STACK_MIN) || defined(__APPLE__)
|
|
/* MAC says its min is 8KB, but then crashes in our face. thx hunkOLard */
|
|
#define PTHREAD_STACK_MIN 64*1024
|
|
#endif
|
|
|
|
/* initialize with pointers to shared memory. these will
|
|
* be used to place responses and arguments */
|
|
void at_init(void) {
|
|
PFUNC();
|
|
void *shm = mmap(0, 4096, PROT_WRITE|PROT_READ, MAP_ANON|MAP_SHARED, -1, 0);
|
|
assert(shm);
|
|
internal_ips_lock = shm;
|
|
internal_ips = (void*)((char*)shm + 2048);
|
|
|
|
MUTEX_INIT(internal_ips_lock);
|
|
memset(internal_ips, 0, sizeof *internal_ips);
|
|
initpipe(req_pipefd);
|
|
initpipe(resp_pipefd);
|
|
pthread_attr_t allocator_thread_attr;
|
|
pthread_attr_init(&allocator_thread_attr);
|
|
pthread_attr_setstacksize(&allocator_thread_attr, MAX(16 * 1024, PTHREAD_STACK_MIN));
|
|
pthread_create(&allocator_thread, &allocator_thread_attr, threadfunc, 0);
|
|
pthread_attr_destroy(&allocator_thread_attr);
|
|
}
|
|
|
|
void at_close(void) {
|
|
PFUNC();
|
|
const int msg = ATM_EXIT;
|
|
write(req_pipefd[1], &msg, sizeof(int));
|
|
pthread_join(allocator_thread, NULL);
|
|
close(req_pipefd[0]);
|
|
close(req_pipefd[1]);
|
|
close(resp_pipefd[0]);
|
|
close(resp_pipefd[1]);
|
|
MUTEX_DESTROY(internal_ips_lock);
|
|
}
|