forked from evmar/c-repl
-
Notifications
You must be signed in to change notification settings - Fork 1
/
child.c
95 lines (80 loc) · 2.11 KB
/
child.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
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
/* c-repl -- a C read-eval-print loop.
* Copyright (C) 2006 Evan Martin <[email protected]>
*/
/* The child process is what actually runs the code.
* It reads in a number from stdin,
* then loads dl#.so and executes dl#().
*/
#include <assert.h>
#include <dlfcn.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
static int debug = 0;
/* Load dl<id>.so and run dl<id>(). */
static void load_and_run(int id) {
char buf[1024];
sprintf(buf, "./.c-repl/dl%d.so", id);
if (debug)
fprintf(stderr, "CHILD> loading %s\n", buf);
void *so = dlopen(buf, RTLD_LAZY | RTLD_GLOBAL);
if (!so) {
fprintf(stderr, "CHILD> error loading library: %s\n", dlerror());
assert(so);
}
sprintf(buf, "dl%d", id);
void (*f)() = dlsym(so, buf);
if (!f) {
fprintf(stderr, "CHILD> error loading function: %s\n", dlerror());
assert(f);
}
//printf("child executing '%s':\n", buf);
// XXX fork here to do the segfault -> undo magic?
f();
}
int main(int argc, char **argv) {
if (argc < 3) {
fprintf(stderr, "bad arguments\n");
return 1;
}
const int command_fd = atoi(argv[1]);
const int response_fd = atoi(argv[2]);
FILE* command_pipe = fdopen(command_fd, "rb");
if (!command_pipe) {
perror("fdopen(command_fd)");
return 1;
}
setlinebuf(command_pipe);
FILE* response_pipe = fdopen(response_fd, "wb");
if (!response_pipe) {
perror("fdopen(response_fd)");
return 1;
}
setlinebuf(response_pipe);
if (fprintf(response_pipe, "%d\n", getpid()) < 0) {
perror("CHILD> fputs");
return 1;
}
char buf[1024];
int highest_id = 0;
while (fgets(buf, sizeof(buf), command_pipe)) {
const int id = atoi(buf);
if (id > 0)
for ( ; highest_id < id; highest_id++)
load_and_run(highest_id+1);
/* If we get here, we succeeded.
* Let the parent know. */
if (fputs(buf, response_pipe) < 0) {
perror("CHILD> fputs");
break;
}
fflush(response_pipe);
}
if (debug)
fprintf(stderr, "CHILD> exiting\n");
fclose(command_pipe);
fclose(response_pipe);
return 0;
};
/* vim: set ts=2 sw=2 et cino=(0 : */