Welcome to mirror list, hosted at ThFree Co, Russian Federation.

rdns.c « src - github.com/rofl0r/proxychains-ng.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
blob: d9496b1f1132a4b5cd05e6596c30c00a34f50ec1 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
#include <sys/socket.h>
#include <stdlib.h>
#include <string.h>

#include "rdns.h"
#include "allocator_thread.h"
#include "remotedns.h"

#ifndef HAVE_SOCK_CLOEXEC
#define SOCK_CLOEXEC 0
#endif

//static enum dns_lookup_flavor dns_flavor;
#define dns_flavor rdns_get_flavor()

static struct sockaddr_in rdns_server;

size_t rdns_daemon_get_host_for_ip(ip_type4 ip, char* readbuf) {
	struct at_msg msg = {
		.h.msgtype = ATM_GETNAME,
		.h.datalen = htons(4),
		.m.ip = ip,
	};
	int fd = socket(AF_INET, SOCK_DGRAM|SOCK_CLOEXEC, 0);
	sendto(fd, &msg, sizeof(msg.h)+4, 0, (void*)&rdns_server, sizeof(rdns_server));
	recvfrom(fd, &msg, sizeof msg, 0, (void*)0, (void*)0);
	close(fd);
	msg.h.datalen = ntohs(msg.h.datalen);
	if(!msg.h.datalen || msg.h.datalen > 256) return 0;
	memcpy(readbuf, msg.m.host, msg.h.datalen);
	return msg.h.datalen - 1;
}

static ip_type4 rdns_daemon_get_ip_for_host(char* host, size_t len) {
	struct at_msg msg = {
		.h.msgtype = ATM_GETIP,
	};
	if(len >= 256) return IPT4_INT(-1);
	memcpy(msg.m.host, host, len+1);
	msg.h.datalen = htons(len+1);
	int fd = socket(AF_INET, SOCK_DGRAM|SOCK_CLOEXEC, 0);
	sendto(fd, &msg, sizeof(msg.h)+len+1, 0, (void*)&rdns_server, sizeof(rdns_server));
	recvfrom(fd, &msg, sizeof msg, 0, (void*)0, (void*)0);
	close(fd);
	if(ntohs(msg.h.datalen) != 4) return IPT4_INT(-1);
	return msg.m.ip;
}

const char *rdns_resolver_string(enum dns_lookup_flavor flavor) {
	static const char tab[][7] = {
		[DNSLF_LIBC] = "off",
		[DNSLF_FORKEXEC] = "old",
		[DNSLF_RDNS_THREAD] = "thread",
		[DNSLF_RDNS_DAEMON] = "daemon",
	};
	return tab[flavor];
}

void rdns_init(enum dns_lookup_flavor flavor) {
	static int init_done = 0;
	if(!init_done) switch(flavor) {
		case DNSLF_RDNS_THREAD:
			at_init();
			break;
		case DNSLF_RDNS_DAEMON:
		default:
			break;
	}
	init_done = 1;
}

void rdns_set_daemon(struct sockaddr_in* addr) {
	rdns_server = *addr;
}

#if 0
enum dns_lookup_flavor rdns_get_flavor(void) {
	return dns_flavor;
}
#endif

size_t rdns_get_host_for_ip(ip_type4 ip, char* readbuf) {
	switch(dns_flavor) {
		case DNSLF_RDNS_THREAD: return at_get_host_for_ip(ip, readbuf);
		case DNSLF_RDNS_DAEMON: return rdns_daemon_get_host_for_ip(ip, readbuf);
		default:
			abort();
	}
}

ip_type4 rdns_get_ip_for_host(char* host, size_t len) {
	switch(dns_flavor) {
		case DNSLF_RDNS_THREAD: return at_get_ip_for_host(host, len);
		case DNSLF_RDNS_DAEMON: return rdns_daemon_get_ip_for_host(host, len);
		default:
			abort();
	}
}