Replacing mpc
The output of mpc
was an Abstract Syntax Tree, but because our language is so simple, when we replace it we’re going to go directly from the input string to an S-Expressions (which is pretty much like abstract syntax tree anyway). So let us declare a function which takes some string s
, some pointer to a position in that string i
and some terminal character end
and which outputs an lval
. We’ll call it lval_read_expr
.
lval* lval_read_expr(char* s, int* i, char end);
We’ll define the implementation later. For now lets go ahead and replace the places we call mpc
with this new function. For error handling we’ll get lval_read_expr
to return an error lval
if something goes wrong. The terminal character when we’re reading from a C string is just the null character \0
.
char* input = readline("lispy> ");
/* Read from input to create an S-Expr */
int pos = 0;
lval* expr = lval_read_expr(input, &pos, '\0');
/* Evaluate and print input */
lval* x = lval_eval(e, expr);
We also need to replace mpc
in our function builtin_load
. Here, because we only support strings in our read function, we first need to load in the contents of the file. To do this we use fseek
and ftell
to find out how many characters are in the file before allocating a string of that size (plus one) and reading the file into it.
lval* builtin_load(lenv* e, lval* a) {
LASSERT_NUM("load", a, 1);
LASSERT_TYPE("load", a, 0, LVAL_STR);
/* Open file and check it exists */
FILE* f = fopen(a->cell[0]->str, "rb");
if (f == NULL) {
lval* err = lval_err("Could not load Library %s", a->cell[0]->str);
lval_del(a);
return err;
}
/* Read File Contents */
fseek(f, 0, SEEK_END);
long length = ftell(f);
fseek(f, 0, SEEK_SET);
char* input = calloc(length+1, 1);
fread(input, 1, length, f);
fclose(f);
We can then easily read from this string as we did in main
.
/* Read from input to create an S-Expr */
int pos = 0;
lval* expr = lval_read_expr(input, &pos, '\0');
free(input);
/* Evaluate all expressions contained in S-Expr */
if (expr->type != LVAL_ERR) {
while (expr->count) {
lval* x = lval_eval(e, lval_pop(expr, 0));
if (x->type == LVAL_ERR) { lval_println(x); }
lval_del(x);
}
} else {
lval_println(expr);
}
lval_del(expr);
lval_del(a);
return lval_sexpr();
}
And with those calls replaced we can start defining lval_read_expr
.