-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathrunner.c
138 lines (117 loc) · 2.96 KB
/
runner.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
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
#include <stdio.h>
#include <stdint.h>
#include <stdlib.h>
#include <fcntl.h>
#include <sys/stat.h>
#include <sys/mman.h>
#include <unistd.h>
#include <errno.h>
#include <sinter.h>
#define eprintf(...) fprintf(stderr, __VA_ARGS__)
static const char *fault_names[] = {
"no fault",
"out of memory",
"type error",
"divide by zero",
"stack overflow",
"stack underflow",
"uninitialised load",
"invalid load",
"invalid program",
"internal error",
"incorrect function arity",
"program called error()",
"uninitialised heap"
};
static const char *type_names[] = {
"unknown",
"undefined",
"null",
"boolean",
"integer",
"float",
"string",
"array",
"function"
};
void setup_internals(void);
void display_object_result(sinter_value_t *res, _Bool is_error);
ssize_t check_posix(ssize_t result, const char *msg) {
if (result == -1 && errno) {
perror(msg);
_exit(1);
}
return result;
}
static void print_string(const char *s, bool is_error) {
(void) is_error;
printf("%s", s);
}
static void print_integer(int32_t v, bool is_error) {
(void) is_error;
printf("%d", v);
}
static void print_float(float v, bool is_error) {
(void) is_error;
printf("%f", v);
}
static void print_flush(bool is_error) {
(void) is_error;
printf("\n");
}
int main(int argc, char *argv[]) {
if (argc < 2) {
eprintf("Usage: %s <program>\n", argv[0]);
return 1;
}
int program_fd = check_posix(open(argv[1], O_RDONLY), "Failed to open program");
off_t size;
{
struct stat stat_buf;
check_posix(fstat(program_fd, &stat_buf), "fstat failed");
size = stat_buf.st_size;
}
const unsigned char *program = mmap(NULL, size, PROT_READ, MAP_SHARED, program_fd, 0);
if (program == MAP_FAILED) {
check_posix(-1, "mmap failed");
}
sinter_printer_float = print_float;
sinter_printer_string = print_string;
sinter_printer_integer = print_integer;
sinter_printer_flush = print_flush;
setup_internals();
sinter_value_t result = { 0 };
sinter_fault_t fault = sinter_run(program, size, &result);
printf("Program exited with fault %s and result type %s: ",
fault >= (sizeof(fault_names)/sizeof(fault_names[0])) ? "(unknown fault)" : fault_names[fault],
result.type >= (sizeof(type_names)/sizeof(type_names[0])) ? "(unknown type)" : type_names[result.type]);
switch (result.type) {
case sinter_type_undefined:
printf("undefined");
break;
case sinter_type_null:
printf("null");
break;
case sinter_type_boolean:
printf("%s", result.boolean_value ? "true" : "false");
break;
case sinter_type_integer:
printf("%d", result.integer_value);
break;
case sinter_type_float:
printf("%f", result.float_value);
break;
case sinter_type_string:
printf("%s", result.string_value);
break;
case sinter_type_array:
case sinter_type_function:
display_object_result(&result, false);
break;
default:
printf("(unable to print value)");
break;
}
printf("\n");
return 0;
}