mirror of
https://github.com/antirez/aocla
synced 2025-01-15 03:41:01 +01:00
REPL loop and parsing changes.
This commit is contained in:
parent
58cc7dcc97
commit
15fa5fefd8
1 changed files with 94 additions and 20 deletions
114
aocla.c
114
aocla.c
|
@ -43,23 +43,23 @@ typedef struct aproc {
|
||||||
/* We have local vars, so we need a stack frame. We start with a top level
|
/* We have local vars, so we need a stack frame. We start with a top level
|
||||||
* stack frame. Each time a procedure is called, we create a new stack frame
|
* stack frame. Each time a procedure is called, we create a new stack frame
|
||||||
* and free it once the procedure returns. */
|
* and free it once the procedure returns. */
|
||||||
#define NUMVARS ('z'-'a'+1)
|
#define AOCLA_NUMVARS ('z'-'a'+1)
|
||||||
typedef struct stackframe {
|
typedef struct stackframe {
|
||||||
obj *locals[NUMVARS]; /* Local var names are limited to a,b,c,...,z. */
|
obj *locals[AOCLA_NUMVARS];/* Local var names are limited to a,b,c,...,z. */
|
||||||
int lstate[NUMVARS]; /* Local state. When a local is assigned, it's set
|
int lstate[AOCLA_NUMVARS]; /* Local state. When a local is assigned, it's
|
||||||
to 1. If a local is pushed, it drops to zero
|
set to 1. If a local is pushed, it drops to zero
|
||||||
(but lcoals[N] will still be not NULL). So
|
(but lcoals[N] will still be not NULL). So
|
||||||
next time it is pushed, we know that we need
|
next time it is pushed, we know that we need
|
||||||
to perform a deep copy of the object. */
|
to perform a deep copy of the object. */
|
||||||
} stackframe;
|
} stackframe;
|
||||||
|
|
||||||
/* Interpreter state. */
|
/* Interpreter state. */
|
||||||
typedef struct ainterp {
|
typedef struct aoclactx {
|
||||||
size_t maxstack, sl; /* Stack max len and stack current len. */
|
size_t maxstack, sl; /* Stack max len and stack current len. */
|
||||||
obj **stack;
|
obj **stack;
|
||||||
aproc *proc; /* Procedures. Lists bound to specific names. */
|
aproc *proc; /* Procedures. Lists bound to specific names. */
|
||||||
stackframe *frame; /* Stack frame with locals. */
|
stackframe *frame; /* Stack frame with locals. */
|
||||||
} ainterp;
|
} aoclactx;
|
||||||
|
|
||||||
/* ================================= Utils ================================== */
|
/* ================================= Utils ================================== */
|
||||||
|
|
||||||
|
@ -98,13 +98,30 @@ void freeobj(obj *o) {
|
||||||
free(o);
|
free(o);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Return true if the character 'c' is within the Aocla symbols charset. */
|
||||||
|
int issymbol(int c) {
|
||||||
|
if (isalpha(c)) return 1;
|
||||||
|
switch(c) {
|
||||||
|
case '+':
|
||||||
|
case '-':
|
||||||
|
case '*':
|
||||||
|
case '/':
|
||||||
|
case '=':
|
||||||
|
case '?':
|
||||||
|
case '%':
|
||||||
|
return 1;
|
||||||
|
default:
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* Given the string 's' return the obj representing the list or
|
/* Given the string 's' return the obj representing the list or
|
||||||
* NULL on syntax error. '*next' is set to the next byte to parse, after
|
* NULL on syntax error. '*next' is set to the next byte to parse, after
|
||||||
* the current e was completely parsed. */
|
* the current e was completely parsed. */
|
||||||
obj *parseList(const char *s, const char **next) {
|
obj *parseList(const char *s, const char **next) {
|
||||||
obj *o = myalloc(sizeof(*o));
|
obj *o = myalloc(sizeof(*o));
|
||||||
while(isspace(s[0])) s++;
|
while(isspace(s[0])) s++;
|
||||||
if (s[0] == '-' || isdigit(s[0])) {
|
if (s[0] == '-' || isdigit(s[0])) { /* Integer. */
|
||||||
char buf[64];
|
char buf[64];
|
||||||
size_t len = 0;
|
size_t len = 0;
|
||||||
while((*s == '-' || isdigit(*s)) && len < sizeof(buf)-1)
|
while((*s == '-' || isdigit(*s)) && len < sizeof(buf)-1)
|
||||||
|
@ -114,7 +131,7 @@ obj *parseList(const char *s, const char **next) {
|
||||||
o->i = atoi(buf);
|
o->i = atoi(buf);
|
||||||
if (next) *next = s;
|
if (next) *next = s;
|
||||||
return o;
|
return o;
|
||||||
} else if (s[0] == '[') {
|
} else if (s[0] == '[') { /* List. */
|
||||||
o->type = OBJ_TYPE_LIST;
|
o->type = OBJ_TYPE_LIST;
|
||||||
o->l.len = 0;
|
o->l.len = 0;
|
||||||
o->l.ele = NULL;
|
o->l.ele = NULL;
|
||||||
|
@ -140,12 +157,7 @@ obj *parseList(const char *s, const char **next) {
|
||||||
o->l.ele[o->l.len++] = element;
|
o->l.ele[o->l.len++] = element;
|
||||||
s = nextptr; /* Continue from first byte not parsed. */
|
s = nextptr; /* Continue from first byte not parsed. */
|
||||||
|
|
||||||
while(isspace(s[0])) s++;
|
continue; /* Parse next element. */
|
||||||
if (s[0] == ']') continue; /* Will be handled by the loop. */
|
|
||||||
if (s[0] == ',') {
|
|
||||||
s++;
|
|
||||||
continue; /* Parse next element. */
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Syntax error. */
|
/* Syntax error. */
|
||||||
freeobj(o);
|
freeobj(o);
|
||||||
|
@ -154,23 +166,21 @@ obj *parseList(const char *s, const char **next) {
|
||||||
/* Syntax error (list not closed). */
|
/* Syntax error (list not closed). */
|
||||||
freeobj(o);
|
freeobj(o);
|
||||||
return NULL;
|
return NULL;
|
||||||
} else if (isalpha(s[0])) {
|
} else if (issymbol(s[0])) { /* Symbol. */
|
||||||
o->type = OBJ_TYPE_SYMBOL;
|
o->type = OBJ_TYPE_SYMBOL;
|
||||||
const char *end = s;
|
const char *end = s;
|
||||||
while(isalpha(*end)) end++;
|
while(issymbol(*end)) end++;
|
||||||
o->sym.len = end-s;
|
o->sym.len = end-s;
|
||||||
char *dest = malloc(o->sym.len+1);
|
char *dest = malloc(o->sym.len+1);
|
||||||
o->sym.ptr = dest;
|
o->sym.ptr = dest;
|
||||||
memcpy(dest,s,o->sym.len);
|
memcpy(dest,s,o->sym.len);
|
||||||
dest[o->sym.len] = 0;
|
dest[o->sym.len] = 0;
|
||||||
*next = end;
|
*next = end;
|
||||||
} else if (s[0] == '"') {
|
} else if (s[0] == '"') { /* String. */
|
||||||
printf("IMPLEMENT STRING PARSING\n");
|
printf("IMPLEMENT STRING PARSING\n");
|
||||||
exit(1);
|
exit(1);
|
||||||
} else {
|
} else {
|
||||||
/* In a serious program you don't printf() in the middle of
|
/* Syntax error. */
|
||||||
* a function. Just return NULL. */
|
|
||||||
fprintf(stderr,"Syntax error parsing '%s'\n", s);
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
return o;
|
return o;
|
||||||
|
@ -230,6 +240,9 @@ void printobj(obj *obj) {
|
||||||
case OBJ_TYPE_INT:
|
case OBJ_TYPE_INT:
|
||||||
printf("%d",obj->i);
|
printf("%d",obj->i);
|
||||||
break;
|
break;
|
||||||
|
case OBJ_TYPE_SYMBOL:
|
||||||
|
printf("%s",obj->sym.ptr);
|
||||||
|
break;
|
||||||
case OBJ_TYPE_LIST:
|
case OBJ_TYPE_LIST:
|
||||||
printf("[");
|
printf("[");
|
||||||
for (size_t j = 0; j < obj->l.len; j++) {
|
for (size_t j = 0; j < obj->l.len; j++) {
|
||||||
|
@ -241,6 +254,43 @@ void printobj(obj *obj) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* ========================== Interpreter state ============================= */
|
||||||
|
|
||||||
|
/* Create a new stack frame. */
|
||||||
|
stackframe *newStackFrame(void) {
|
||||||
|
stackframe *sf = myalloc(sizeof(*sf));
|
||||||
|
memset(sf->locals,0,sizeof(sf->locals));
|
||||||
|
memset(sf->lstate,0,sizeof(sf->lstate));
|
||||||
|
return sf;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Free a stack frame. */
|
||||||
|
void freeStackFrame(stackframe *sf) {
|
||||||
|
for (int j = 0; j < AOCLA_NUMVARS; j++)
|
||||||
|
if (sf->locals[j]) freeobj(sf->locals[j]);
|
||||||
|
free(sf);
|
||||||
|
}
|
||||||
|
|
||||||
|
#define AOCLA_STACK_MAX 256
|
||||||
|
aoclactx *newInterpreter(void) {
|
||||||
|
aoclactx *i = myalloc(sizeof(*i));
|
||||||
|
i->maxstack = AOCLA_STACK_MAX;
|
||||||
|
i->sl = 0;
|
||||||
|
i->stack = myalloc(sizeof(obj*)*i->maxstack);
|
||||||
|
i->proc = NULL; /* That's a linked list. Starts empty. */
|
||||||
|
i->frame = newStackFrame();
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================ Eval ==================================== */
|
||||||
|
|
||||||
|
/* Evaluate the program in the list 'l' in the specified context 'ctx'. */
|
||||||
|
void eval(aoclactx *ctx, obj *l) {
|
||||||
|
if (l->type != OBJ_TYPE_LIST) return;
|
||||||
|
printobj(l);
|
||||||
|
printf("\n");
|
||||||
|
}
|
||||||
|
|
||||||
/* ================================ CLI ===================================== */
|
/* ================================ CLI ===================================== */
|
||||||
|
|
||||||
/* Read the lists contained in the file 'fp', parse them into an obj
|
/* Read the lists contained in the file 'fp', parse them into an obj
|
||||||
|
@ -261,7 +311,31 @@ int readLists(FILE *fp, obj **v, size_t vlen) {
|
||||||
return idx;
|
return idx;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Real Eval Print Loop. */
|
||||||
void repl(void) {
|
void repl(void) {
|
||||||
|
char buf[1024];
|
||||||
|
aoclactx *ctx = newInterpreter();
|
||||||
|
while(1) {
|
||||||
|
printf("aocla> "); fflush(stdout);
|
||||||
|
if (fgets(buf,sizeof(buf)-2,stdin) == NULL) break;
|
||||||
|
size_t l = strlen(buf);
|
||||||
|
if (l && buf[l-1] == '\n') buf[--l] = 0;
|
||||||
|
if (l == 0) continue;
|
||||||
|
|
||||||
|
/* Aocla programs are Aocla lists, so when users just write
|
||||||
|
* in the REPL we need to surround with []. */
|
||||||
|
memmove(buf+1,buf,l);
|
||||||
|
buf[0] = '[';
|
||||||
|
buf[l+1] = ']';
|
||||||
|
buf[l+2] = 0;
|
||||||
|
|
||||||
|
obj *list = parseList(buf,NULL);
|
||||||
|
if (!list) {
|
||||||
|
printf("Syntax error\n");
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
eval(ctx,list);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void evalFile(const char *filename, char **argv, int argc) {
|
void evalFile(const char *filename, char **argv, int argc) {
|
||||||
|
|
Loading…
Reference in a new issue