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
|
#include <dlfcn.h>
#include <errno.h>
#include <fcntl.h>
#include <limits.h>
#include <stddef.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#ifndef LINKER
#error LINKER must be defined
#endif
static char exe[PATH_MAX], *linker;
static ssize_t (*real_readlink)(const char *, char *, size_t);
ssize_t readlink(const char *path, char *buf, size_t len)
{
if (real_readlink == NULL) {
real_readlink = dlsym(RTLD_NEXT, "readlink");
if (real_readlink == NULL) {
errno = ENOSYS;
return -1;
}
}
if (!strcmp(path, "/proc/self/exe")) {
int fd;
if (exe[0] == '\0') {
if (linker == NULL) {
linker = realpath(LINKER, NULL);
if (linker == NULL) {
return -1;
}
}
if (real_readlink(path, exe, sizeof(exe)) < 1) {
goto fail;
}
if (!strcmp(exe, linker)) {
char c;
int arg = 0;
ssize_t arglen;
fd = open("/proc/self/cmdline",
O_RDONLY | O_CLOEXEC);
if (fd < 0) {
goto fail;
}
/* Skip the --argv0/--preload ldso args.
* This number must be kept in sync with the
* argument order in loader/loader.c */
while (arg < 6) {
if (read(fd, &c, 1) != 1) {
goto fail_close;
}
if (c == '\0') {
++arg;
}
}
/* Read the executable path from the cmdline. */
arglen = read(fd, exe, sizeof(exe));
if (arglen < 1) {
goto fail_close;
}
close(fd);
/* Ensure the path exists, fits, and has NUL. */
if (exe[0] == '\0') {
goto fail;
}
if (strnlen(exe, arglen) == (size_t) arglen) {
goto fail;
}
}
}
return stpncpy(buf, exe, len) - buf;
fail_close:
close(fd);
fail:
exe[0] = '\0';
errno = EIO;
return -1;
}
return real_readlink(path, buf, len);
}
|