mirror of
https://github.com/lua/lua.git
synced 2025-01-14 05:43:00 +08:00
optimizations for gettable (temporary)
This commit is contained in:
parent
1fe280df72
commit
f8279f6cd8
98
lvm.c
98
lvm.c
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
** $Id: lvm.c,v 1.242 2002/06/24 14:11:14 roberto Exp roberto $
|
** $Id: lvm.c,v 1.243 2002/06/24 15:07:21 roberto Exp roberto $
|
||||||
** Lua virtual machine
|
** Lua virtual machine
|
||||||
** See Copyright Notice in lua.h
|
** See Copyright Notice in lua.h
|
||||||
*/
|
*/
|
||||||
@ -32,7 +32,7 @@
|
|||||||
|
|
||||||
|
|
||||||
/* limit for table tag-method chains (to avoid loops) */
|
/* limit for table tag-method chains (to avoid loops) */
|
||||||
#define MAXTAGLOOP 10000
|
#define MAXTAGLOOP 100
|
||||||
|
|
||||||
|
|
||||||
static void luaV_checkGC (lua_State *L, StkId top) {
|
static void luaV_checkGC (lua_State *L, StkId top) {
|
||||||
@ -108,34 +108,46 @@ static void callTM (lua_State *L, const TObject *f,
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static const TObject *luaV_index (lua_State *L, const TObject *t,
|
||||||
|
TObject *key, int loop) {
|
||||||
|
const TObject *tm = fasttm(L, hvalue(t)->metatable, TM_INDEX);
|
||||||
|
if (tm == NULL) return &luaO_nilobject; /* no TM */
|
||||||
|
if (ttype(tm) == LUA_TFUNCTION) {
|
||||||
|
callTMres(L, tm, t, key);
|
||||||
|
return L->top;
|
||||||
|
}
|
||||||
|
else return luaV_gettable(L, tm, key, loop);
|
||||||
|
}
|
||||||
|
|
||||||
|
static const TObject *luaV_getnotable (lua_State *L, const TObject *t,
|
||||||
|
TObject *key, int loop) {
|
||||||
|
const TObject *tm = luaT_gettmbyobj(L, t, TM_GETTABLE);
|
||||||
|
if (ttype(tm) == LUA_TNIL)
|
||||||
|
luaG_typeerror(L, t, "index");
|
||||||
|
if (ttype(tm) == LUA_TFUNCTION) {
|
||||||
|
callTMres(L, tm, t, key);
|
||||||
|
return L->top;
|
||||||
|
}
|
||||||
|
else return luaV_gettable(L, tm, key, loop);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
** Function to index a table.
|
** Function to index a table.
|
||||||
** Receives the table at `t' and the key at `key'.
|
** Receives the table at `t' and the key at `key'.
|
||||||
** leaves the result at `res'.
|
** leaves the result at `res'.
|
||||||
*/
|
*/
|
||||||
const TObject *luaV_gettable (lua_State *L, const TObject *t, TObject *key) {
|
const TObject *luaV_gettable (lua_State *L, const TObject *t, TObject *key,
|
||||||
const TObject *tm;
|
int loop) {
|
||||||
int loop = 0;
|
if (loop > MAXTAGLOOP)
|
||||||
do {
|
luaG_runerror(L, "loop in gettable");
|
||||||
if (ttype(t) == LUA_TTABLE) { /* `t' is a table? */
|
if (ttype(t) == LUA_TTABLE) { /* `t' is a table? */
|
||||||
Table *h = hvalue(t);
|
Table *h = hvalue(t);
|
||||||
const TObject *v = luaH_get(h, key); /* do a primitive get */
|
const TObject *v = luaH_get(h, key); /* do a primitive get */
|
||||||
if (ttype(v) != LUA_TNIL || /* result is no nil? */
|
if (ttype(v) != LUA_TNIL) return v;
|
||||||
(tm = fasttm(L, h->metatable, TM_INDEX)) == NULL) { /* or no TM? */
|
else return luaV_index(L, t, key, loop+1);
|
||||||
return v;
|
}
|
||||||
}
|
else return luaV_getnotable(L, t, key, loop+1);
|
||||||
/* else will try the tag method */
|
|
||||||
}
|
|
||||||
else if (ttype(tm = luaT_gettmbyobj(L, t, TM_GETTABLE)) == LUA_TNIL)
|
|
||||||
luaG_typeerror(L, t, "index");
|
|
||||||
if (ttype(tm) == LUA_TFUNCTION) {
|
|
||||||
callTMres(L, tm, t, key);
|
|
||||||
return L->top;
|
|
||||||
}
|
|
||||||
t = tm; /* else repeat access with `tm' */
|
|
||||||
} while (++loop <= MAXTAGLOOP);
|
|
||||||
luaG_runerror(L, "loop in gettable");
|
|
||||||
return NULL; /* to avoid warnings */
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -392,12 +404,26 @@ StkId luaV_execute (lua_State *L) {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case OP_GETGLOBAL: {
|
case OP_GETGLOBAL: {
|
||||||
lua_assert(ttype(KBx(i)) == LUA_TSTRING && ttype(&cl->g) == LUA_TTABLE);
|
StkId rb = KBx(i);
|
||||||
setobj(RA(i), luaV_gettable(L, &cl->g, KBx(i)));
|
const TObject *v;
|
||||||
|
lua_assert(ttype(rb) == LUA_TSTRING && ttype(&cl->g) == LUA_TTABLE);
|
||||||
|
v = luaH_getstr(hvalue(&cl->g), tsvalue(rb));
|
||||||
|
if (ttype(v) != LUA_TNIL) { setobj(ra, v); }
|
||||||
|
else
|
||||||
|
setobj(RA(i), luaV_index(L, &cl->g, rb, 0));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case OP_GETTABLE: {
|
case OP_GETTABLE: {
|
||||||
setobj(RA(i), luaV_gettable(L, RB(i), RKC(i)));
|
StkId rb = RB(i);
|
||||||
|
TObject *rc = RKC(i);
|
||||||
|
if (ttype(rb) == LUA_TTABLE) {
|
||||||
|
const TObject *v = luaH_get(hvalue(rb), rc);
|
||||||
|
if (ttype(v) != LUA_TNIL) { setobj(ra, v); }
|
||||||
|
else
|
||||||
|
setobj(RA(i), luaV_index(L, rb, rc, 0));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
setobj(RA(i), luaV_getnotable(L, rb, rc, 0));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case OP_SETGLOBAL: {
|
case OP_SETGLOBAL: {
|
||||||
@ -423,8 +449,17 @@ StkId luaV_execute (lua_State *L) {
|
|||||||
}
|
}
|
||||||
case OP_SELF: {
|
case OP_SELF: {
|
||||||
StkId rb = RB(i);
|
StkId rb = RB(i);
|
||||||
|
TObject *rc = RKC(i);
|
||||||
|
runtime_check(L, ttype(rc) == LUA_TSTRING);
|
||||||
setobj(ra+1, rb);
|
setobj(ra+1, rb);
|
||||||
setobj(RA(i), luaV_gettable(L, rb, RKC(i)));
|
if (ttype(rb) == LUA_TTABLE) {
|
||||||
|
const TObject *v = luaH_getstr(hvalue(rb), tsvalue(rc));
|
||||||
|
if (ttype(v) != LUA_TNIL) { setobj(ra, v); }
|
||||||
|
else
|
||||||
|
setobj(RA(i), luaV_index(L, rb, rc, 0));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
setobj(RA(i), luaV_getnotable(L, rb, rc, 0));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case OP_ADD: {
|
case OP_ADD: {
|
||||||
@ -620,11 +655,10 @@ StkId luaV_execute (lua_State *L) {
|
|||||||
else dojump(pc, GETARG_sBx(*pc) + 1); /* else jump back */
|
else dojump(pc, GETARG_sBx(*pc) + 1); /* else jump back */
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case OP_TFORPREP: {
|
case OP_TFORPREP: { /* for compatibility only */
|
||||||
if (ttype(ra) == LUA_TTABLE) {
|
if (ttype(ra) == LUA_TTABLE) {
|
||||||
setobj(ra+1, ra);
|
setobj(ra+1, ra);
|
||||||
setsvalue(ra, luaS_new(L, "next"));
|
setobj(ra, luaH_getstr(hvalue(gt(L)), luaS_new(L, "next")));
|
||||||
setobj(RA(i), luaV_gettable(L, gt(L), ra));
|
|
||||||
}
|
}
|
||||||
dojump(pc, GETARG_sBx(i));
|
dojump(pc, GETARG_sBx(i));
|
||||||
break;
|
break;
|
||||||
|
5
lvm.h
5
lvm.h
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
** $Id: lvm.h,v 1.42 2002/06/13 13:39:55 roberto Exp roberto $
|
** $Id: lvm.h,v 1.43 2002/06/24 13:08:45 roberto Exp roberto $
|
||||||
** Lua virtual machine
|
** Lua virtual machine
|
||||||
** See Copyright Notice in lua.h
|
** See Copyright Notice in lua.h
|
||||||
*/
|
*/
|
||||||
@ -26,7 +26,8 @@ int luaV_lessthan (lua_State *L, const TObject *l, const TObject *r);
|
|||||||
int luaV_equalval (lua_State *L, const TObject *t1, const TObject *t2);
|
int luaV_equalval (lua_State *L, const TObject *t1, const TObject *t2);
|
||||||
const TObject *luaV_tonumber (const TObject *obj, TObject *n);
|
const TObject *luaV_tonumber (const TObject *obj, TObject *n);
|
||||||
int luaV_tostring (lua_State *L, TObject *obj);
|
int luaV_tostring (lua_State *L, TObject *obj);
|
||||||
const TObject *luaV_gettable (lua_State *L, const TObject *t, TObject *key);
|
const TObject *luaV_gettable (lua_State *L, const TObject *t, TObject *key,
|
||||||
|
int loop);
|
||||||
void luaV_settable (lua_State *L, const TObject *t, TObject *key, StkId val);
|
void luaV_settable (lua_State *L, const TObject *t, TObject *key, StkId val);
|
||||||
StkId luaV_execute (lua_State *L);
|
StkId luaV_execute (lua_State *L);
|
||||||
void luaV_concat (lua_State *L, int total, int last);
|
void luaV_concat (lua_State *L, int total, int last);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user