forked from BackSlasher/dns-override
-
Notifications
You must be signed in to change notification settings - Fork 0
/
dns-override.c
64 lines (54 loc) · 1.77 KB
/
dns-override.c
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
#define _GNU_SOURCE
#include <stdio.h>
#include <dlfcn.h>
#include <string.h>
#include <stdlib.h>
// make code compatible with glibc 2.2.5.
__asm__(".symver dlsym,dlsym@GLIBC_2.2.5");
__asm__(".symver fmemopen,fmemopen@GLIBC_2.2.5");
static const char *RESOLV_CONF = "/etc/resolv.conf";
static const int RESOLV_SIZE = 16;
static const char *RESOLV_ENV = "RESOLV_CONF";
static FILE *(*original_fopen)(const char*, const char*) = NULL;
static FILE *(*original_fopen64)(const char*, const char*) = NULL;
static void init_fopen_override() {
if (original_fopen == NULL)
original_fopen = dlsym(RTLD_NEXT, "fopen");
if (original_fopen64 == NULL)
original_fopen64 = dlsym(RTLD_NEXT, "fopen64");
}
static char *better_resolv = NULL;
static int better_resolv_inited=0;
// No fmemopen64 so implementing the same for both functions
static FILE *init_better_resolv() {
// Singleton
if (!better_resolv_inited) {
better_resolv_inited=1;
better_resolv = getenv(RESOLV_ENV);
// Print warning message
if (!better_resolv)
fprintf(stderr, "dns-override: No env var %s, will return unmodified file\n", RESOLV_ENV);
}
if (better_resolv)
return fmemopen(better_resolv, strlen(better_resolv), "r");
else
return NULL;
}
FILE *fopen(const char *path, const char *mode) {
init_fopen_override();
// If it's our file
if(!strncmp(RESOLV_CONF, path, RESOLV_SIZE)) {
FILE *res = init_better_resolv();
// Try returning our file
if (res) return res;
}
return (*original_fopen)(path, mode);
}
FILE *fopen64(const char *path, const char *mode) {
init_fopen_override();
if(!strncmp(RESOLV_CONF, path, RESOLV_SIZE)) {
FILE *res = init_better_resolv();
if (res) return res;
}
return (*original_fopen64)(path, mode);
}