/* ** $Id: lapi.c,v 1.65 1999/12/23 18:19:57 roberto Exp roberto $ ** Lua API ** See Copyright Notice in lua.h */ #include #define LUA_REENTRANT #include "lapi.h" #include "lauxlib.h" #include "ldo.h" #include "lfunc.h" #include "lgc.h" #include "lmem.h" #include "lobject.h" #include "lref.h" #include "lstate.h" #include "lstring.h" #include "ltable.h" #include "ltm.h" #include "lua.h" #include "lvm.h" const char lua_ident[] = "$Lua: " LUA_VERSION " " LUA_COPYRIGHT " $\n" "$Authors: " LUA_AUTHORS " $"; const lua_Type luaA_normtype[] = { /* ORDER LUA_T */ LUA_T_USERDATA, LUA_T_NUMBER, LUA_T_STRING, LUA_T_ARRAY, LUA_T_LPROTO, LUA_T_CPROTO, LUA_T_NIL, LUA_T_LCLOSURE, LUA_T_CCLOSURE, LUA_T_LCLOSURE, LUA_T_CCLOSURE, /* LUA_T_LCLMARK, LUA_T_CCLMARK */ LUA_T_LPROTO, LUA_T_CPROTO /* LUA_T_LMARK, LUA_T_CMARK */ }; void luaA_setnormalized (TObject *d, const TObject *s) { d->value = s->value; d->ttype = luaA_normalizedtype(s); } const TObject *luaA_protovalue (const TObject *o) { switch (luaA_normalizedtype(o)) { case LUA_T_CCLOSURE: case LUA_T_LCLOSURE: return protovalue(o); default: LUA_ASSERT(L, luaA_normalizedtype(o) == LUA_T_LPROTO || luaA_normalizedtype(o) == LUA_T_CPROTO, "invalid `function'"); return o; } } void luaA_checkCparams (lua_State *L, int nParams) { if (nParams > L->top-L->Cstack.base) lua_error(L, "API error - wrong number of arguments in C2lua stack"); } lua_Object luaA_putluaObject (lua_State *L, const TObject *o) { luaD_openstack(L, L->Cstack.base); *L->Cstack.base++ = *o; return L->Cstack.base-1; } lua_Object luaA_putObjectOnTop (lua_State *L) { luaD_openstack(L, L->Cstack.base); *L->Cstack.base++ = *(--L->top); return L->Cstack.base-1; } static void top2LC (lua_State *L, int n) { /* Put the `n' elements on the top as the Lua2C contents */ L->Cstack.base = L->top; /* new base */ L->Cstack.lua2C = L->Cstack.base-n; /* position of the new results */ L->Cstack.num = n; /* number of results */ } lua_Object lua_pop (lua_State *L) { luaA_checkCparams(L, 1); return luaA_putObjectOnTop(L); } /* ** Get a parameter, returning the object handle or LUA_NOOBJECT on error. ** `number' must be 1 to get the first parameter. */ lua_Object lua_lua2C (lua_State *L, int number) { if (number <= 0 || number > L->Cstack.num) return LUA_NOOBJECT; return L->Cstack.lua2C+number-1; } int lua_callfunction (lua_State *L, lua_Object function) { if (function == LUA_NOOBJECT) return 1; else { luaD_openstack(L, L->Cstack.base); luaA_setnormalized(L->Cstack.base, function); return luaD_protectedrun(L); } } lua_Object lua_gettagmethod (lua_State *L, int tag, const char *event) { return luaA_putluaObject(L, luaT_gettagmethod(L, tag, event)); } lua_Object lua_settagmethod (lua_State *L, int tag, const char *event) { luaA_checkCparams(L, 1); luaT_settagmethod(L, tag, event, L->top-1); return luaA_putObjectOnTop(L); } lua_Object lua_seterrormethod (lua_State *L) { lua_Object temp; luaA_checkCparams(L, 1); temp = lua_getglobal(L, "_ERRORMESSAGE"); lua_setglobal(L, "_ERRORMESSAGE"); return temp; } lua_Object lua_gettable (lua_State *L) { luaA_checkCparams(L, 2); luaV_gettable(L); return luaA_putObjectOnTop(L); } lua_Object lua_rawgettable (lua_State *L) { luaA_checkCparams(L, 2); if (ttype(L->top-2) != LUA_T_ARRAY) lua_error(L, "indexed expression not a table in rawgettable"); *(L->top-2) = *luaH_get(L, avalue(L->top-2), L->top-1); --L->top; return luaA_putObjectOnTop(L); } void lua_settable (lua_State *L) { luaA_checkCparams(L, 3); luaV_settable(L, L->top-3); L->top -= 2; /* pop table and index */ } void lua_rawsettable (lua_State *L) { luaA_checkCparams(L, 3); luaV_rawsettable(L, L->top-3); } lua_Object lua_createtable (lua_State *L) { TObject o; luaC_checkGC(L); avalue(&o) = luaH_new(L, 0); ttype(&o) = LUA_T_ARRAY; return luaA_putluaObject(L, &o); } lua_Object lua_getglobal (lua_State *L, const char *name) { luaD_checkstack(L, 2); /* may need that to call a tag method */ luaV_getglobal(L, luaS_assertglobalbyname(L, name)); return luaA_putObjectOnTop(L); } lua_Object lua_rawgetglobal (lua_State *L, const char *name) { GlobalVar *gv = luaS_assertglobalbyname(L, name); return luaA_putluaObject(L, &gv->value); } void lua_setglobal (lua_State *L, const char *name) { luaA_checkCparams(L, 1); luaD_checkstack(L, 2); /* may need that to call a tag method */ luaV_setglobal(L, luaS_assertglobalbyname(L, name)); } void lua_rawsetglobal (lua_State *L, const char *name) { GlobalVar *gv = luaS_assertglobalbyname(L, name); luaA_checkCparams(L, 1); gv->value = *(--L->top); } const char *lua_type (lua_State *L, lua_Object o) { UNUSED(L); return (o == LUA_NOOBJECT) ? "NOOBJECT" : luaO_typename(o); } int lua_isnil (lua_State *L, lua_Object o) { UNUSED(L); return (o != LUA_NOOBJECT) && (ttype(o) == LUA_T_NIL); } int lua_istable (lua_State *L, lua_Object o) { UNUSED(L); return (o != LUA_NOOBJECT) && (ttype(o) == LUA_T_ARRAY); } int lua_isuserdata (lua_State *L, lua_Object o) { UNUSED(L); return (o != LUA_NOOBJECT) && (ttype(o) == LUA_T_USERDATA); } int lua_iscfunction (lua_State *L, lua_Object o) { return (lua_tag(L, o) == LUA_T_CPROTO); } int lua_isnumber (lua_State *L, lua_Object o) { UNUSED(L); return (o != LUA_NOOBJECT) && (tonumber(o) == 0); } int lua_isstring (lua_State *L, lua_Object o) { UNUSED(L); return (o != LUA_NOOBJECT && (ttype(o) == LUA_T_STRING || ttype(o) == LUA_T_NUMBER)); } int lua_isfunction (lua_State *L, lua_Object o) { return *lua_type(L, o) == 'f'; } int lua_equal(lua_State *L, lua_Object o1, lua_Object o2) { UNUSED(L); if (o1 == LUA_NOOBJECT || o2 == LUA_NOOBJECT) return (o1 == o2); else { TObject obj1, obj2; luaA_setnormalized(&obj1, o1); luaA_setnormalized(&obj2, o2); return luaO_equalObj(&obj1, &obj2); } } double lua_getnumber (lua_State *L, lua_Object obj) { UNUSED(L); if (obj == LUA_NOOBJECT || tonumber(obj)) return 0.0; else return (nvalue(obj)); } const char *lua_getstring (lua_State *L, lua_Object obj) { luaC_checkGC(L); /* `tostring' may create a new string */ if (obj == LUA_NOOBJECT || tostring(L, obj)) return NULL; else return (svalue(obj)); } long lua_strlen (lua_State *L, lua_Object obj) { if (obj == LUA_NOOBJECT || tostring(L, obj)) return 0L; else return (tsvalue(obj)->u.s.len); } void *lua_getuserdata (lua_State *L, lua_Object obj) { UNUSED(L); if (obj == LUA_NOOBJECT || ttype(obj) != LUA_T_USERDATA) return NULL; else return tsvalue(obj)->u.d.value; } lua_CFunction lua_getcfunction (lua_State *L, lua_Object obj) { if (!lua_iscfunction(L, obj)) return NULL; else return fvalue(luaA_protovalue(obj)); } void lua_pushnil (lua_State *L) { ttype(L->top) = LUA_T_NIL; incr_top; } void lua_pushnumber (lua_State *L, double n) { ttype(L->top) = LUA_T_NUMBER; nvalue(L->top) = n; incr_top; } void lua_pushlstring (lua_State *L, const char *s, long len) { tsvalue(L->top) = luaS_newlstr(L, s, len); ttype(L->top) = LUA_T_STRING; incr_top; luaC_checkGC(L); } void lua_pushstring (lua_State *L, const char *s) { if (s == NULL) lua_pushnil(L); else lua_pushlstring(L, s, strlen(s)); } void lua_pushcclosure (lua_State *L, lua_CFunction fn, int n) { if (fn == NULL) lua_error(L, "API error - attempt to push a NULL Cfunction"); luaA_checkCparams(L, n); ttype(L->top) = LUA_T_CPROTO; fvalue(L->top) = fn; incr_top; luaV_closure(L, n); luaC_checkGC(L); } void lua_pushusertag (lua_State *L, void *u, int tag) { if (tag < 0 && tag != LUA_ANYTAG) luaT_realtag(L, tag); /* error if tag is not valid */ tsvalue(L->top) = luaS_createudata(L, u, tag); ttype(L->top) = LUA_T_USERDATA; incr_top; luaC_checkGC(L); } void luaA_pushobject (lua_State *L, const TObject *o) { *L->top = *o; incr_top; } void lua_pushobject (lua_State *L, lua_Object o) { if (o == LUA_NOOBJECT) lua_error(L, "API error - attempt to push a NOOBJECT"); luaA_setnormalized(L->top, o); incr_top; } int lua_tag (lua_State *L, lua_Object o) { UNUSED(L); if (o == LUA_NOOBJECT) return LUA_T_NIL; else if (ttype(o) == LUA_T_USERDATA) /* to allow `old' tags (deprecated) */ return o->value.ts->u.d.tag; else return luaT_effectivetag(o); } void lua_settag (lua_State *L, int tag) { luaA_checkCparams(L, 1); luaT_realtag(L, tag); switch (ttype(L->top-1)) { case LUA_T_ARRAY: (L->top-1)->value.a->htag = tag; break; case LUA_T_USERDATA: (L->top-1)->value.ts->u.d.tag = tag; break; default: luaL_verror(L, "cannot change the tag of a %.20s", luaO_typename(L->top-1)); } L->top--; } GlobalVar *luaA_nextvar (lua_State *L, TaggedString *ts) { GlobalVar *gv; if (ts == NULL) gv = L->rootglobal; /* first variable */ else { /* check whether name is in global var list */ luaL_arg_check(L, ts->u.s.gv, 1, "variable name expected"); gv = ts->u.s.gv->next; /* get next */ } while (gv && gv->value.ttype == LUA_T_NIL) /* skip globals with nil */ gv = gv->next; if (gv) { ttype(L->top) = LUA_T_STRING; tsvalue(L->top) = gv->name; incr_top; luaA_pushobject(L, &gv->value); } return gv; } const char *lua_nextvar (lua_State *L, const char *varname) { TaggedString *ts = (varname == NULL) ? NULL : luaS_new(L, varname); GlobalVar *gv = luaA_nextvar(L, ts); if (gv) { top2LC(L, 2); return gv->name->str; } else { top2LC(L, 0); return NULL; } } int luaA_next (lua_State *L, const Hash *t, int i) { int tsize = t->size; for (; i