mirror of
https://github.com/lua/lua.git
synced 2025-01-14 05:43:00 +08:00
creation of function do_protectedrun, that executes lua code enclosed
in a setjmp, with error recovery. Elimination of functions lua_isnil, etc.
This commit is contained in:
parent
3db06a95a3
commit
c635044f2f
336
opcode.c
336
opcode.c
@ -3,7 +3,7 @@
|
||||
** TecCGraf - PUC-Rio
|
||||
*/
|
||||
|
||||
char *rcs_opcode="$Id: opcode.c,v 2.12 1994/11/01 18:25:20 roberto Exp roberto $";
|
||||
char *rcs_opcode="$Id: opcode.c,v 3.1 1994/11/02 20:30:53 roberto Exp roberto $";
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
@ -30,27 +30,36 @@ static Long maxstack;
|
||||
static Object *stack=NULL;
|
||||
static Object *top;
|
||||
|
||||
static int CBase; /* when Lua calls C or C calls Lua, points to the */
|
||||
/* first slot after the last parameter. */
|
||||
static int CnResults = 0; /* when Lua calls C, has the number of parameters; */
|
||||
/* when C calls Lua, has the number of results. */
|
||||
|
||||
static jmp_buf *errorJmp;
|
||||
static int CBase = 0; /* when Lua calls C or C calls Lua, points to the */
|
||||
/* first slot after the last parameter. */
|
||||
static int CnResults = 0; /* when Lua calls C, has the number of parameters; */
|
||||
/* when C calls Lua, has the number of results. */
|
||||
|
||||
static jmp_buf *errorJmp = NULL; /* current error recover point */
|
||||
|
||||
|
||||
static int lua_execute (Byte *pc, int base);
|
||||
|
||||
|
||||
static void lua_message (char *s)
|
||||
{
|
||||
fprintf (stderr, "lua: %s\n", s);
|
||||
}
|
||||
|
||||
/*
|
||||
** Reports an error, and jumps up to the available recover label
|
||||
*/
|
||||
void lua_error (char *s)
|
||||
{
|
||||
fprintf (stderr, "lua: %s\n", s);
|
||||
lua_message(s);
|
||||
if (errorJmp)
|
||||
longjmp(*errorJmp, 1);
|
||||
else
|
||||
{
|
||||
fprintf (stderr, "lua: exit(1). Unable to recover\n");
|
||||
exit(1);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@ -73,7 +82,7 @@ static void lua_initstack (void)
|
||||
static void lua_checkstack (Word n)
|
||||
{
|
||||
if (stack == NULL)
|
||||
return lua_initstack();
|
||||
lua_initstack();
|
||||
if (n > maxstack)
|
||||
{
|
||||
int t = top-stack;
|
||||
@ -99,12 +108,6 @@ static char *lua_strconc (char *l, char *r)
|
||||
return strcat(strcpy(buffer,l),r);
|
||||
}
|
||||
|
||||
static int ToReal (char* s, float* f)
|
||||
{
|
||||
char c;
|
||||
float t;
|
||||
if (sscanf(s,"%f %c",&t,&c) == 1) { *f=t; return 1; } else return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
** Convert, if possible, to a number object.
|
||||
@ -112,31 +115,18 @@ static int ToReal (char* s, float* f)
|
||||
*/
|
||||
static int lua_tonumber (Object *obj)
|
||||
{
|
||||
char c;
|
||||
float t;
|
||||
if (tag(obj) != LUA_T_STRING)
|
||||
return 1;;
|
||||
if (!ToReal(svalue(obj), &nvalue(obj)))
|
||||
return 2;
|
||||
tag(obj) = LUA_T_NUMBER;
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
** Test if it is possible to convert an object to a number object.
|
||||
** If possible, return the converted object, otherwise return nil object.
|
||||
*/
|
||||
static Object *lua_convtonumber (Object *obj)
|
||||
{
|
||||
static Object cvt;
|
||||
if (tag(obj) == LUA_T_NUMBER)
|
||||
return 1;
|
||||
else if (sscanf(svalue(obj), "%f %c",&t,&c) == 1)
|
||||
{
|
||||
cvt = *obj;
|
||||
return &cvt;
|
||||
nvalue(obj) = t;
|
||||
tag(obj) = LUA_T_NUMBER;
|
||||
return 0;
|
||||
}
|
||||
if (tag(obj) == LUA_T_STRING && ToReal(svalue(obj), &nvalue(&cvt)))
|
||||
tag(&cvt) = LUA_T_NUMBER;
|
||||
else
|
||||
tag(&cvt) = LUA_T_NIL;
|
||||
return &cvt;
|
||||
return 2;
|
||||
}
|
||||
|
||||
|
||||
@ -182,7 +172,8 @@ static int callC (lua_CFunction func, int base)
|
||||
int oldCnResults = CnResults;
|
||||
int firstResult;
|
||||
CnResults = (top-stack) - base;
|
||||
CBase = base+CnResults; /* incorporate parameters on the stack */
|
||||
/* incorporate parameters on the stack */
|
||||
CBase = base+CnResults;
|
||||
(*func)();
|
||||
firstResult = CBase;
|
||||
CBase = oldBase;
|
||||
@ -221,22 +212,17 @@ static void do_call (Object *func, int base, int nResults, int whereRes)
|
||||
|
||||
|
||||
/*
|
||||
** Function to index the values on the top
|
||||
** Function to index a table. Receives the table at top-2 and the index
|
||||
** at top-1. Remove them from stack and push the result.
|
||||
*/
|
||||
int lua_pushsubscript (void)
|
||||
static void pushsubscript (void)
|
||||
{
|
||||
--top;
|
||||
if (tag(top-1) != LUA_T_ARRAY)
|
||||
{
|
||||
lua_reportbug ("indexed expression not a table");
|
||||
return 1;
|
||||
}
|
||||
{
|
||||
Object *h = lua_hashget (avalue(top-1), top);
|
||||
if (h == NULL) return 1;
|
||||
Object *h;
|
||||
if (tag(top-2) != LUA_T_ARRAY)
|
||||
lua_reportbug ("indexed expression not a table");
|
||||
h = lua_hashget (avalue(top-2), top-1);
|
||||
--top;
|
||||
*(top-1) = *h;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
@ -272,63 +258,36 @@ void lua_travstack (void (*fn)(Object *))
|
||||
|
||||
|
||||
/*
|
||||
** Executes a main procedure. Uses as Base the top of the stack, as it
|
||||
** uses no parameters and left no results.
|
||||
** Execute a protected call. If function is null compiles the pre-set input.
|
||||
** Leave nResults on the stack.
|
||||
*/
|
||||
static void do_main (Byte *main)
|
||||
{
|
||||
if (main)
|
||||
{
|
||||
Object f;
|
||||
tag(&f) = LUA_T_FUNCTION; bvalue(&f) = main;
|
||||
do_call(&f, top-stack, 0, top-stack);
|
||||
free(main);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
** Open file, generate opcode and execute global statement. Return 0 on
|
||||
** success or 1 on error.
|
||||
*/
|
||||
int lua_dofile (char *filename)
|
||||
static int do_protectedrun (Object *function, int nResults)
|
||||
{
|
||||
Object f;
|
||||
jmp_buf myErrorJmp;
|
||||
int status;
|
||||
int oldCBase = CBase;
|
||||
jmp_buf *oldErr = errorJmp;
|
||||
errorJmp = &myErrorJmp;
|
||||
if (setjmp(myErrorJmp) == 0)
|
||||
{
|
||||
lua_openfile (filename);
|
||||
do_main(lua_parse());
|
||||
if (function == NULL)
|
||||
{
|
||||
function = &f;
|
||||
tag(function) = LUA_T_FUNCTION;
|
||||
bvalue(function) = lua_parse();
|
||||
}
|
||||
do_call(function, CBase, nResults, CBase);
|
||||
CnResults = (top-stack) - CBase; /* number of results */
|
||||
CBase += CnResults; /* incorporate results on the stack */
|
||||
status = 0;
|
||||
}
|
||||
else
|
||||
status = 1;
|
||||
lua_closefile();
|
||||
errorJmp = oldErr;
|
||||
return status;
|
||||
}
|
||||
|
||||
/*
|
||||
** Generate opcode stored on string and execute global statement. Return 0 on
|
||||
** success or 1 on error.
|
||||
*/
|
||||
int lua_dostring (char *string)
|
||||
{
|
||||
jmp_buf myErrorJmp;
|
||||
int status;
|
||||
jmp_buf *oldErr = errorJmp;
|
||||
errorJmp = &myErrorJmp;
|
||||
if (setjmp(myErrorJmp) == 0)
|
||||
{
|
||||
lua_openstring(string);
|
||||
do_main(lua_parse());
|
||||
status = 0;
|
||||
}
|
||||
else
|
||||
CBase = oldCBase;
|
||||
top = stack+CBase;
|
||||
status = 1;
|
||||
lua_closestring();
|
||||
}
|
||||
errorJmp = oldErr;
|
||||
return status;
|
||||
}
|
||||
@ -338,23 +297,50 @@ int lua_dostring (char *string)
|
||||
*/
|
||||
int lua_callfunction (Object *function)
|
||||
{
|
||||
jmp_buf myErrorJmp;
|
||||
int status;
|
||||
jmp_buf *oldErr = errorJmp;
|
||||
errorJmp = &myErrorJmp;
|
||||
if (setjmp(myErrorJmp) == 0)
|
||||
{
|
||||
do_call(function, CBase, MULT_RET, CBase);
|
||||
CnResults = (top-stack) - CBase; /* number of results */
|
||||
CBase += CnResults; /* incorporate results on the stack */
|
||||
status = 0;
|
||||
}
|
||||
if (function == NULL)
|
||||
return 1;
|
||||
else
|
||||
status = 1;
|
||||
errorJmp = oldErr;
|
||||
return do_protectedrun (function, MULT_RET);
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
** Open file, generate opcode and execute global statement. Return 0 on
|
||||
** success or 1 on error.
|
||||
*/
|
||||
int lua_dofile (char *filename)
|
||||
{
|
||||
int status;
|
||||
char *message = lua_openfile (filename);
|
||||
if (message)
|
||||
{
|
||||
lua_message(message);
|
||||
return 1;
|
||||
}
|
||||
status = do_protectedrun(NULL, 0);
|
||||
lua_closefile();
|
||||
return status;
|
||||
}
|
||||
|
||||
/*
|
||||
** Generate opcode stored on string and execute global statement. Return 0 on
|
||||
** success or 1 on error.
|
||||
*/
|
||||
int lua_dostring (char *string)
|
||||
{
|
||||
int status;
|
||||
char *message = lua_openstring(string);
|
||||
if (message)
|
||||
{
|
||||
lua_message(message);
|
||||
return 1;
|
||||
}
|
||||
status = do_protectedrun(NULL, 0);
|
||||
lua_closestring();
|
||||
return status;
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
** Get a parameter, returning the object handle or NULL on error.
|
||||
** 'number' must be 1 to get the first parameter.
|
||||
@ -425,42 +411,6 @@ void *lua_gettable (Object *object)
|
||||
else return (avalue(object));
|
||||
}
|
||||
|
||||
/*
|
||||
** Given an object handle and a field name, return its field object.
|
||||
** On error, return NULL.
|
||||
*/
|
||||
Object *lua_getfield (Object *object, char *field)
|
||||
{
|
||||
if (object == NULL) return NULL;
|
||||
if (tag(object) != LUA_T_ARRAY)
|
||||
return NULL;
|
||||
else
|
||||
{
|
||||
Object ref;
|
||||
tag(&ref) = LUA_T_STRING;
|
||||
svalue(&ref) = lua_constant[lua_findconstant(field)];
|
||||
return (lua_hashget(avalue(object), &ref));
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
** Given an object handle and an index, return its indexed object.
|
||||
** On error, return NULL.
|
||||
*/
|
||||
Object *lua_getindexed (Object *object, float index)
|
||||
{
|
||||
if (object == NULL) return NULL;
|
||||
if (tag(object) != LUA_T_ARRAY)
|
||||
return NULL;
|
||||
else
|
||||
{
|
||||
Object ref;
|
||||
tag(&ref) = LUA_T_NUMBER;
|
||||
nvalue(&ref) = index;
|
||||
return (lua_hashget(avalue(object), &ref));
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
** Get a global object. Return the object handle or NULL on error.
|
||||
*/
|
||||
@ -550,7 +500,7 @@ int lua_storeglobal (char *name)
|
||||
{
|
||||
int n = lua_findsymbol (name);
|
||||
if (n < 0) return 1;
|
||||
if (tag(top-1) == LUA_T_MARK) return 1;
|
||||
if (top-stack <= CBase) return 1;
|
||||
s_object(n) = *(--top);
|
||||
return 0;
|
||||
}
|
||||
@ -598,73 +548,14 @@ int lua_storeindexed (lua_Object object, float index)
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
** Given an object handle, return if it is nil.
|
||||
*/
|
||||
int lua_isnil (Object *object)
|
||||
int lua_type (lua_Object o)
|
||||
{
|
||||
return (object != NULL && tag(object) == LUA_T_NIL);
|
||||
if (o == NULL)
|
||||
return LUA_T_NIL;
|
||||
else
|
||||
return tag(o);
|
||||
}
|
||||
|
||||
/*
|
||||
** Given an object handle, return if it is a number one.
|
||||
*/
|
||||
int lua_isnumber (Object *object)
|
||||
{
|
||||
return (object != NULL && tag(object) == LUA_T_NUMBER);
|
||||
}
|
||||
|
||||
/*
|
||||
** Given an object handle, return if it is a string one.
|
||||
*/
|
||||
int lua_isstring (Object *object)
|
||||
{
|
||||
return (object != NULL && tag(object) == LUA_T_STRING);
|
||||
}
|
||||
|
||||
/*
|
||||
** Given an object handle, return if it is an array one.
|
||||
*/
|
||||
int lua_istable (Object *object)
|
||||
{
|
||||
return (object != NULL && tag(object) == LUA_T_ARRAY);
|
||||
}
|
||||
|
||||
/*
|
||||
** Given an object handle, return if it is a lua function.
|
||||
*/
|
||||
int lua_isfunction (Object *object)
|
||||
{
|
||||
return (object != NULL && tag(object) == LUA_T_FUNCTION);
|
||||
}
|
||||
|
||||
/*
|
||||
** Given an object handle, return if it is a cfunction one.
|
||||
*/
|
||||
int lua_iscfunction (Object *object)
|
||||
{
|
||||
return (object != NULL && tag(object) == LUA_T_CFUNCTION);
|
||||
}
|
||||
|
||||
/*
|
||||
** Given an object handle, return if it is an user data one.
|
||||
*/
|
||||
int lua_isuserdata (Object *object)
|
||||
{
|
||||
return (object != NULL && tag(object) == LUA_T_USERDATA);
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
** Internal function: convert an object to a number
|
||||
*/
|
||||
void lua_obj2number (void)
|
||||
{
|
||||
Object *o = lua_getparam(1);
|
||||
lua_pushobject (lua_convtonumber(o));
|
||||
}
|
||||
|
||||
|
||||
|
||||
/*
|
||||
** Execute the given opcode, until a RET. Parameters are between
|
||||
@ -674,8 +565,7 @@ void lua_obj2number (void)
|
||||
static int lua_execute (Byte *pc, int base)
|
||||
{
|
||||
lua_debugline = 0; /* reset debug flag */
|
||||
if (stack == NULL)
|
||||
lua_initstack();
|
||||
lua_checkstack(STACKGAP+MAX_TEMPS+base);
|
||||
while (1)
|
||||
{
|
||||
OpCode opcode;
|
||||
@ -737,16 +627,13 @@ static int lua_execute (Byte *pc, int base)
|
||||
break;
|
||||
|
||||
case PUSHINDEXED:
|
||||
{
|
||||
int s = lua_pushsubscript();
|
||||
if (s == 1) return 1;
|
||||
}
|
||||
break;
|
||||
pushsubscript();
|
||||
break;
|
||||
|
||||
case PUSHSELF:
|
||||
{
|
||||
Object receiver = *(top-2);
|
||||
if (lua_pushsubscript() == 1) return 1;
|
||||
pushsubscript();
|
||||
*(top++) = receiver;
|
||||
break;
|
||||
}
|
||||
@ -779,10 +666,7 @@ static int lua_execute (Byte *pc, int base)
|
||||
{
|
||||
int n = *pc++;
|
||||
if (tag(top-3-n) != LUA_T_ARRAY)
|
||||
{
|
||||
lua_reportbug ("indexed expression not a table");
|
||||
return 1;
|
||||
}
|
||||
lua_reportbug ("indexed expression not a table");
|
||||
{
|
||||
Object *h = lua_hashdefine (avalue(top-3-n), top-2-n);
|
||||
if (h == NULL) return 1;
|
||||
@ -802,10 +686,7 @@ static int lua_execute (Byte *pc, int base)
|
||||
n = *(pc++);
|
||||
arr = top-n-1;
|
||||
if (tag(arr) != LUA_T_ARRAY)
|
||||
{
|
||||
lua_reportbug ("internal error - table expected");
|
||||
return 1;
|
||||
}
|
||||
lua_reportbug ("internal error - table expected");
|
||||
while (n)
|
||||
{
|
||||
tag(top) = LUA_T_NUMBER; nvalue(top) = n+m;
|
||||
@ -821,10 +702,7 @@ static int lua_execute (Byte *pc, int base)
|
||||
int n = *(pc++);
|
||||
Object *arr = top-n-1;
|
||||
if (tag(arr) != LUA_T_ARRAY)
|
||||
{
|
||||
lua_reportbug ("internal error - table expected");
|
||||
return 1;
|
||||
}
|
||||
lua_reportbug ("internal error - table expected");
|
||||
while (n)
|
||||
{
|
||||
CodeWord code;
|
||||
@ -851,8 +729,6 @@ static int lua_execute (Byte *pc, int base)
|
||||
get_word(size,pc);
|
||||
top++;
|
||||
avalue(top-1) = lua_createarray(size.w);
|
||||
if (avalue(top-1) == NULL)
|
||||
return 1;
|
||||
tag(top-1) = LUA_T_ARRAY;
|
||||
}
|
||||
break;
|
||||
@ -870,7 +746,7 @@ static int lua_execute (Byte *pc, int base)
|
||||
switch (tag(l))
|
||||
{
|
||||
case LUA_T_NIL:
|
||||
res = 0; break;
|
||||
res = 1; break;
|
||||
case LUA_T_NUMBER:
|
||||
res = (nvalue(l) == nvalue(r)); break;
|
||||
case LUA_T_ARRAY:
|
||||
|
Loading…
x
Reference in New Issue
Block a user