mirror of
https://github.com/antirez/aocla
synced 2025-01-13 20:01:40 +01:00
get@.
This commit is contained in:
parent
cc7fd74d87
commit
03e6c46d7c
1 changed files with 86 additions and 12 deletions
98
aocla.c
98
aocla.c
|
@ -23,6 +23,7 @@
|
||||||
#define OBJ_TYPE_STRING (1<<3)
|
#define OBJ_TYPE_STRING (1<<3)
|
||||||
#define OBJ_TYPE_SYMBOL (1<<4)
|
#define OBJ_TYPE_SYMBOL (1<<4)
|
||||||
#define OBJ_TYPE_BOOL (1<<5)
|
#define OBJ_TYPE_BOOL (1<<5)
|
||||||
|
#define OBJ_TYPE_ANY INT_MAX /* All bits set. For checkStackType(). */
|
||||||
typedef struct obj {
|
typedef struct obj {
|
||||||
int type; /* OBJ_TYPE_... */
|
int type; /* OBJ_TYPE_... */
|
||||||
int refcount; /* Reference count. */
|
int refcount; /* Reference count. */
|
||||||
|
@ -147,6 +148,7 @@ obj *newObject(int type) {
|
||||||
int issymbol(int c) {
|
int issymbol(int c) {
|
||||||
if (isalpha(c)) return 1;
|
if (isalpha(c)) return 1;
|
||||||
switch(c) {
|
switch(c) {
|
||||||
|
case '@':
|
||||||
case '$':
|
case '$':
|
||||||
case '+':
|
case '+':
|
||||||
case '-':
|
case '-':
|
||||||
|
@ -443,9 +445,26 @@ obj *newBool(int b) {
|
||||||
/* Deep copy the passed object. Return an object with refcount = 1. */
|
/* Deep copy the passed object. Return an object with refcount = 1. */
|
||||||
obj *deepCopy(obj *o) {
|
obj *deepCopy(obj *o) {
|
||||||
if (o == NULL) return NULL;
|
if (o == NULL) return NULL;
|
||||||
/* TODO: actually implement it. */
|
obj *c = newObject(o->type);
|
||||||
printf("IMPLEMENT ME \\o/\n");
|
switch(o->type) {
|
||||||
exit(1);
|
case OBJ_TYPE_INT: c->i = o->i; break;
|
||||||
|
case OBJ_TYPE_BOOL: c->istrue = o->istrue; break;
|
||||||
|
case OBJ_TYPE_LIST:
|
||||||
|
case OBJ_TYPE_TUPLE:
|
||||||
|
c->l.len = o->l.len;
|
||||||
|
c->l.ele = myalloc(sizeof(obj*)*o->l.len);
|
||||||
|
for (size_t j = 0; j < o->l.len; j++)
|
||||||
|
c->l.ele[j] = deepCopy(o->l.ele[j]);
|
||||||
|
break;
|
||||||
|
case OBJ_TYPE_STRING:
|
||||||
|
case OBJ_TYPE_SYMBOL:
|
||||||
|
c->str.len = o->str.len;
|
||||||
|
c->str.quoted = o->str.quoted; /* Only useful for symbols. */
|
||||||
|
c->str.ptr = myalloc(o->str.len+1);
|
||||||
|
memcpy(c->str.ptr,o->str.ptr,o->str.len+1);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return c;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* This function performs a deep copy of the object if it has a refcount > 1.
|
/* This function performs a deep copy of the object if it has a refcount > 1.
|
||||||
|
@ -518,10 +537,11 @@ obj *stackPop(aoclactx *ctx) {
|
||||||
return ctx->stack[--ctx->stacklen];
|
return ctx->stack[--ctx->stacklen];
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Return the pointer to the last object on the stack or NULL. */
|
/* Return the pointer to the last object (if offset == 0) on the stack
|
||||||
obj *stackPeek(aoclactx *ctx) {
|
* or NULL. Offset of 1 means penultimate and so forth. */
|
||||||
if (ctx->stacklen == 0) return NULL;
|
obj *stackPeek(aoclactx *ctx, size_t offset) {
|
||||||
return ctx->stack[ctx->stacklen-1];
|
if (ctx->stacklen <= offset) return NULL;
|
||||||
|
return ctx->stack[ctx->stacklen-1-offset];
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Show the current content of the stack. */
|
/* Show the current content of the stack. */
|
||||||
|
@ -583,10 +603,15 @@ int eval(aoclactx *ctx, obj *l) {
|
||||||
/* Quoted symbols don't generate a procedure call, but like
|
/* Quoted symbols don't generate a procedure call, but like
|
||||||
* any other object they get pushed on the stack. */
|
* any other object they get pushed on the stack. */
|
||||||
if (o->str.quoted) {
|
if (o->str.quoted) {
|
||||||
stackPush(ctx,o);
|
obj *notq = deepCopy(o);
|
||||||
retain(o);
|
notq->str.quoted = 0;
|
||||||
|
stackPush(ctx,notq);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Not quoted symbols get looked up and executed if they
|
||||||
|
* don't start with "$". Otherwise are handled as locals
|
||||||
|
* push on the stack. */
|
||||||
if (o->str.ptr[0] == '$') { /* Push local var. */
|
if (o->str.ptr[0] == '$') { /* Push local var. */
|
||||||
int idx = o->str.ptr[1];
|
int idx = o->str.ptr[1];
|
||||||
if (ctx->frame->locals[idx] == NULL) {
|
if (ctx->frame->locals[idx] == NULL) {
|
||||||
|
@ -787,6 +812,9 @@ int procDef(aoclactx *ctx) {
|
||||||
}
|
}
|
||||||
|
|
||||||
/* if, ifelse, while.
|
/* if, ifelse, while.
|
||||||
|
*
|
||||||
|
* (list) => (result) // if
|
||||||
|
* (list list) => (result) // ifelse and while
|
||||||
*
|
*
|
||||||
* We could implement while in AOCLA itself, once we have ifelse, however
|
* We could implement while in AOCLA itself, once we have ifelse, however
|
||||||
* this way we would build everything on a recursive implementation (still
|
* this way we would build everything on a recursive implementation (still
|
||||||
|
@ -838,7 +866,7 @@ rterr: /* Cleanup. We jump here on error with retval = 1. */
|
||||||
return retval;
|
return retval;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Evaluate the given list. */
|
/* Evaluate the given list, consuming it. */
|
||||||
int procEval(aoclactx *ctx) {
|
int procEval(aoclactx *ctx) {
|
||||||
if (checkStackType(ctx,1,OBJ_TYPE_LIST)) return 1;
|
if (checkStackType(ctx,1,OBJ_TYPE_LIST)) return 1;
|
||||||
obj *l = stackPop(ctx);
|
obj *l = stackPop(ctx);
|
||||||
|
@ -847,7 +875,7 @@ int procEval(aoclactx *ctx) {
|
||||||
return retval;
|
return retval;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Print the top object to stdout. */
|
/* Print the top object to stdout, consuming it */
|
||||||
int procPrint(aoclactx *ctx) {
|
int procPrint(aoclactx *ctx) {
|
||||||
if (checkStackLen(ctx,1)) return 1;
|
if (checkStackLen(ctx,1)) return 1;
|
||||||
obj *o = stackPop(ctx);
|
obj *o = stackPop(ctx);
|
||||||
|
@ -856,7 +884,8 @@ int procPrint(aoclactx *ctx) {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Len replace the object on top with its length. Works with many types. */
|
/* Len -- gets object len. Works with many types.
|
||||||
|
* (object) => (len) */
|
||||||
int procLen(aoclactx *ctx) {
|
int procLen(aoclactx *ctx) {
|
||||||
if (checkStackType(ctx,1,OBJ_TYPE_LIST|OBJ_TYPE_TUPLE|OBJ_TYPE_STRING|
|
if (checkStackType(ctx,1,OBJ_TYPE_LIST|OBJ_TYPE_TUPLE|OBJ_TYPE_STRING|
|
||||||
OBJ_TYPE_SYMBOL)) return 1;
|
OBJ_TYPE_SYMBOL)) return 1;
|
||||||
|
@ -872,6 +901,48 @@ int procLen(aoclactx *ctx) {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Implements -> and <-, appending element x in list with stack
|
||||||
|
*
|
||||||
|
* ([1 2 3] x) => ([1 2 3 x]) | ([x 1 2 3])
|
||||||
|
*
|
||||||
|
* <- is very inefficient as it memmoves all N elements. */
|
||||||
|
int procListAppend(aoclactx *ctx) {
|
||||||
|
int tail = ctx->frame->curproc->name[0] == '-'; /* Append on tail? */
|
||||||
|
if (checkStackType(ctx,2,OBJ_TYPE_ANY,OBJ_TYPE_LIST)) return 1;
|
||||||
|
obj *l = getUnsharedObject(stackPop(ctx));
|
||||||
|
obj *ele = stackPop(ctx);
|
||||||
|
l->l.ele = myrealloc(l->l.ele,sizeof(obj*)*(l->l.len+1));
|
||||||
|
if (tail) {
|
||||||
|
l->l.ele[l->l.len] = ele;
|
||||||
|
} else {
|
||||||
|
memmove(l->l.ele+1,l->l.ele,sizeof(obj*)*l->l.len);
|
||||||
|
l->l.ele[0] = ele;
|
||||||
|
}
|
||||||
|
l->l.len++;
|
||||||
|
stackPush(ctx,l);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* @idx -- get element at index:
|
||||||
|
* (list index) => (list element)
|
||||||
|
*
|
||||||
|
* TODO: this should work for strings as well. */
|
||||||
|
int procListGetAt(aoclactx *ctx) {
|
||||||
|
if (checkStackType(ctx,2,OBJ_TYPE_LIST,OBJ_TYPE_INT)) return 1;
|
||||||
|
obj *idx = stackPop(ctx);
|
||||||
|
obj *list = stackPeek(ctx,0);
|
||||||
|
int i = idx->i;
|
||||||
|
if (i < 0) i = list->l.len+i; /* -1 is last element, and so forth. */
|
||||||
|
release(idx);
|
||||||
|
if (i < 0 || (size_t)i >= list->l.len) {
|
||||||
|
stackPush(ctx,newBool(0)); // Out of index? Just push false.
|
||||||
|
} else {
|
||||||
|
stackPush(ctx,list->l.ele[i]);
|
||||||
|
retain(list->l.ele[i]);
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
void loadLibrary(aoclactx *ctx) {
|
void loadLibrary(aoclactx *ctx) {
|
||||||
addProc(ctx,"+",procBasicMath,NULL);
|
addProc(ctx,"+",procBasicMath,NULL);
|
||||||
addProc(ctx,"-",procBasicMath,NULL);
|
addProc(ctx,"-",procBasicMath,NULL);
|
||||||
|
@ -891,6 +962,9 @@ void loadLibrary(aoclactx *ctx) {
|
||||||
addProc(ctx,"eval",procEval,NULL);
|
addProc(ctx,"eval",procEval,NULL);
|
||||||
addProc(ctx,"print",procPrint,NULL);
|
addProc(ctx,"print",procPrint,NULL);
|
||||||
addProc(ctx,"len",procLen,NULL);
|
addProc(ctx,"len",procLen,NULL);
|
||||||
|
addProc(ctx,"->",procListAppend,NULL);
|
||||||
|
addProc(ctx,"<-",procListAppend,NULL);
|
||||||
|
addProc(ctx,"get@",procListGetAt,NULL);
|
||||||
addProcString(ctx,"dup","[(x) $x $x]");
|
addProcString(ctx,"dup","[(x) $x $x]");
|
||||||
addProcString(ctx,"swap","[(x y) $y $x]");
|
addProcString(ctx,"swap","[(x y) $y $x]");
|
||||||
addProcString(ctx,"drop","[(_)]");
|
addProcString(ctx,"drop","[(_)]");
|
||||||
|
|
Loading…
Reference in a new issue