-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathvm.c
230 lines (198 loc) · 5.44 KB
/
vm.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
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
#include <stdio.h>
#include <stdarg.h>
#include <string.h>
#include "common.h"
#include "object.h"
#include "memory.h"
#include "vm.h"
#include "compiler.h"
#include "debug.h"
#include "value.h"
/*
* The virtual machine is a singleton.
* At first glance, this will make it simpler to implement the language. It
* would be a more mature idea to operate the vm as a pointer and pass it
* around to functions. This can be a future improvement.
*/
vm_t vm;
static void reset_stack() {
vm.stack_top = vm.stack;
}
static void runtime_error(const char* format, ...) {
// Variadic functions take a varying number of arguments.
// Thus we will use the variadic flavour of printf called vfprintf
va_list args;
va_start(args, format);
vfprintf(stderr, format, args);
va_end(args);
fputs("\n", stderr);
size_t instruction = vm.ip - vm.chunk->code -1;
int line = get_line_num(&(vm.chunk->lines), instruction);
fprintf(stderr, "[line %d] in script\n", line);
reset_stack();
}
void init_vm() {
reset_stack();
vm.objects = NULL;
init_table(&vm.strings);
}
void free_vm() {
free_objects();
free_table(&vm.strings);
}
void push(value_t value) {
*vm.stack_top = value;
vm.stack_top++;
}
value_t pop() {
vm.stack_top--;
return *vm.stack_top;
}
static value_t peek(int distance) {
// `distance` is how far down from the top of the stack to look:
// zero is the top, one is one slot down, etc.
return vm.stack_top[-1 - distance];
}
static bool is_falsey(value_t value) {
// Implementation of falseables.
// `null` and `false` are falseables. Everything else is true.
return IS_NULL(value) || (IS_BOOL(value) && !AS_BOOL(value));
}
static void concatenate() {
obj_string_t* b = AS_STRING(pop());
obj_string_t* a = AS_STRING(pop());
int length = a->length + b->length;
char* chars = ALLOCATE(char, length + 1);
memcpy(chars, a->chars, a->length);
memcpy(chars + a->length, b->chars, b->length);
chars[length] = '\0';
obj_string_t* result = take_string(chars, length);
push(OBJ_VAL(result));
}
static interpret_result_t run() {
// returns the byte currently pointed by ip and then advances the
// instruction pointer to the next byte.
#define READ_BYTE() (*vm.ip++)
// calls READ_BYTE, treating the result number as an index for the constants
// array. Returns the constant value from the constant pool.
#define READ_CONSTANT() (vm.chunk->constants.values[READ_BYTE()])
// The following macro comes in a do/while block so that it can be used with
// trailing ";" without causing problems even if inside `if` conditionals.
#define BINARY_OP(value_type, op) \
do { \
if (!IS_NUMBER(peek(0)) || !IS_NUMBER(peek(1))) { \
runtime_error("Operands must be numbers."); \
return INTERPRET_RUNTIME_ERROR; \
} \
double b = AS_NUMBER(pop()); \
double a = AS_NUMBER(pop()); \
push(value_type(a op b)); \
} while (false)
for (;;) {
#ifdef DEBUG_TRACE_EXECUTION
printf(" ");
for (value_t* slot = vm.stack; slot < vm.stack_top; slot++) {
printf("[ ");
print_value(*slot);
printf(" ]");
}
printf("\n");
// pointer math to convert ip back to a relative offset from the beginning
// of the bytecode.
disassemble_instruction(vm.chunk, (int)(vm.ip - vm.chunk->code));
#endif
uint8_t instruction;
switch (instruction = READ_BYTE()) {
case OP_CONSTANT: {
value_t constant = READ_CONSTANT();
push(constant);
break;
};
case OP_TRUE: {
push(BOOL_VAL(true));
break;
}
case OP_FALSE: {
push(BOOL_VAL(false));
break;
}
case OP_EQUAL: {
value_t b = pop();
value_t a = pop();
push(BOOL_VAL(values_equal(a, b)));
break;
}
case OP_GREATER: {
BINARY_OP(BOOL_VAL, >);
break;
}
case OP_LESS: {
BINARY_OP(BOOL_VAL, <);
break;
}
case OP_NULL: {
push(NULL_VAL);
break;
}
case OP_NEGATE:
if (!IS_NUMBER(peek(0))) {
runtime_error("Operant must be a number.");
return INTERPRET_RUNTIME_ERROR;
}
push(NUMBER_VAL(-AS_NUMBER(pop())));
break;
case OP_ADD: {
if (IS_STRING(peek(0)) && IS_STRING(peek(1))) {
concatenate();
} else if (IS_NUMBER(peek(0)) && IS_NUMBER(peek(1))) {
double b = AS_NUMBER(pop());
double a = AS_NUMBER(pop());
push(NUMBER_VAL(a + b));
} else {
runtime_error(
"Operands must be two numbers or two strings."
);
return INTERPRET_RUNTIME_ERROR;
}
break;
}
case OP_SUBTRACT: {
BINARY_OP(NUMBER_VAL, -);
break;
}
case OP_MULTIPLY: {
BINARY_OP(NUMBER_VAL, *);
break;
}
case OP_DIVIDE: {
BINARY_OP(NUMBER_VAL, /);
break;
}
case OP_NOT: {
push(BOOL_VAL(is_falsey(pop())));
break;
}
case OP_RETURN: {
print_value(pop());
printf("\n");
return INTERPRET_OK;
}
}
}
#undef BINARY_OP
#undef READ_BYTE
#undef READ_CONSTANT
}
interpret_result_t interpret(const char* source) {
chunk_t chunk;
init_chunk(&chunk);
if (!compile(source, &chunk)) {
free_chunk(&chunk);
return INTERPRET_COMPILE_ERROR;
}
vm.chunk = &chunk;
vm.ip = vm.chunk->code;
interpret_result_t result = run();
free_chunk(&chunk);
return result;
}