small optimizations (relational operators)

This commit is contained in:
Roberto Ierusalimschy 2000-01-19 14:50:30 -02:00
parent 33d3504889
commit 512b15b601
3 changed files with 80 additions and 40 deletions

View File

@ -1,5 +1,5 @@
/* /*
** $Id: lbuiltin.c,v 1.90 1999/12/28 19:23:41 roberto Exp roberto $ ** $Id: lbuiltin.c,v 1.91 1999/12/30 18:27:03 roberto Exp roberto $
** Built-in functions ** Built-in functions
** See Copyright Notice in lua.h ** See Copyright Notice in lua.h
*/ */
@ -541,10 +541,9 @@ static int sort_comp (lua_State *L, lua_Object f, const TObject *a,
luaD_call(L, L->top-3, 1); luaD_call(L, L->top-3, 1);
} }
else { /* a < b? */ else { /* a < b? */
*(L->top) = *a; *(L->top++) = *a;
*(L->top+1) = *b; *(L->top++) = *b;
luaV_comparison(L, L->top+2, LUA_T_NUMBER, LUA_T_NIL, LUA_T_NIL, IM_LT); luaV_comparison(L);
L->top++; /* result of comparison */
} }
return ttype(--(L->top)) != LUA_T_NIL; return ttype(--(L->top)) != LUA_T_NIL;
} }

106
lvm.c
View File

@ -1,5 +1,5 @@
/* /*
** $Id: lvm.c,v 1.79 2000/01/13 15:56:03 roberto Exp roberto $ ** $Id: lvm.c,v 1.80 2000/01/19 12:00:45 roberto Exp roberto $
** Lua virtual machine ** Lua virtual machine
** See Copyright Notice in lua.h ** See Copyright Notice in lua.h
*/ */
@ -236,7 +236,11 @@ static void call_arith (lua_State *L, StkId top, IMS event) {
} }
static int luaV_strcomp (const char *l, long ll, const char *r, long lr) { static int luaV_strcomp (const TaggedString *ls, const TaggedString *rs) {
const char *l = ls->str;
long ll = ls->u.s.len;
const char *r = rs->str;
long lr = rs->u.s.len;
for (;;) { for (;;) {
long temp = strcoll(l, r); long temp = strcoll(l, r);
if (temp != 0) return temp; if (temp != 0) return temp;
@ -252,23 +256,25 @@ static int luaV_strcomp (const char *l, long ll, const char *r, long lr) {
} }
} }
void luaV_comparison (lua_State *L, StkId top, lua_Type ttype_less, void luaV_comparison (lua_State *L) {
lua_Type ttype_equal, lua_Type ttype_great, IMS op) { const TObject *l = L->top-2;
const TObject *l = top-2; const TObject *r = L->top-1;
const TObject *r = top-1; int result;
real result;
if (ttype(l) == LUA_T_NUMBER && ttype(r) == LUA_T_NUMBER) if (ttype(l) == LUA_T_NUMBER && ttype(r) == LUA_T_NUMBER)
result = nvalue(l)-nvalue(r); result = nvalue(l) < nvalue(r);
else if (ttype(l) == LUA_T_STRING && ttype(r) == LUA_T_STRING) else if (ttype(l) == LUA_T_STRING && ttype(r) == LUA_T_STRING)
result = luaV_strcomp(svalue(l), tsvalue(l)->u.s.len, result = luaV_strcomp(tsvalue(l), tsvalue(r)) < 0;
svalue(r), tsvalue(r)->u.s.len);
else { else {
call_binTM(L, top, op, "unexpected type in comparison"); call_binTM(L, L->top, IM_LT, "unexpected type in comparison");
return; return;
} }
nvalue(top-2) = 1; L->top--;
ttype(top-2) = (result < 0) ? ttype_less : if (result) {
(result == 0) ? ttype_equal : ttype_great; nvalue(L->top-1) = 1.0;
ttype(L->top-1) = LUA_T_NUMBER;
}
else
ttype(L->top-1) = LUA_T_NIL;
} }
@ -380,6 +386,7 @@ StkId luaV_execute (lua_State *L, const Closure *cl, const TProtoFunc *tf,
case GETGLOBALW: aux += highbyte(L, *pc++); case GETGLOBALW: aux += highbyte(L, *pc++);
case GETGLOBAL: aux += *pc++; case GETGLOBAL: aux += *pc++;
L->top = top; L->top = top;
LUA_ASSERT(L, ttype(&consts[aux]) == LUA_T_STRING, "unexpected type");
luaV_getglobal(L, tsvalue(&consts[aux])->u.s.gv); luaV_getglobal(L, tsvalue(&consts[aux])->u.s.gv);
top++; top++;
LUA_ASSERT(L, top==L->top, "top's not synchronized"); LUA_ASSERT(L, top==L->top, "top's not synchronized");
@ -394,7 +401,9 @@ StkId luaV_execute (lua_State *L, const Closure *cl, const TProtoFunc *tf,
case GETDOTTEDW: aux += highbyte(L, *pc++); case GETDOTTEDW: aux += highbyte(L, *pc++);
case GETDOTTED: aux += *pc++; case GETDOTTED: aux += *pc++;
*top++ = consts[aux]; LUA_ASSERT(L, ttype(&consts[aux]) == LUA_T_STRING, "unexpected type");
ttype(top) = LUA_T_STRING;
tsvalue(top++) = tsvalue(&consts[aux]);
L->top = top; L->top = top;
luaV_gettable(L); luaV_gettable(L);
top--; top--;
@ -405,7 +414,9 @@ StkId luaV_execute (lua_State *L, const Closure *cl, const TProtoFunc *tf,
case PUSHSELF: aux += *pc++; { case PUSHSELF: aux += *pc++; {
TObject receiver; TObject receiver;
receiver = *(top-1); receiver = *(top-1);
*top++ = consts[aux]; LUA_ASSERT(L, ttype(&consts[aux]) == LUA_T_STRING, "unexpected type");
ttype(top) = LUA_T_STRING;
tsvalue(top++) = tsvalue(&consts[aux]);
L->top = top; L->top = top;
luaV_gettable(L); luaV_gettable(L);
*(top-1) = receiver; *(top-1) = receiver;
@ -427,6 +438,7 @@ StkId luaV_execute (lua_State *L, const Closure *cl, const TProtoFunc *tf,
case SETGLOBALW: aux += highbyte(L, *pc++); case SETGLOBALW: aux += highbyte(L, *pc++);
case SETGLOBAL: aux += *pc++; case SETGLOBAL: aux += *pc++;
LUA_ASSERT(L, ttype(&consts[aux]) == LUA_T_STRING, "unexpected type");
L->top = top; L->top = top;
luaV_setglobal(L, tsvalue(&consts[aux])->u.s.gv); luaV_setglobal(L, tsvalue(&consts[aux])->u.s.gv);
top--; top--;
@ -469,34 +481,64 @@ StkId luaV_execute (lua_State *L, const Closure *cl, const TProtoFunc *tf,
} }
case NEQOP: aux = 1; case NEQOP: aux = 1;
case EQOP: { case EQOP:
int res = luaO_equalObj(top-2, top-1);
if (aux) res = !res;
top--; top--;
ttype(top-1) = res ? LUA_T_NUMBER : LUA_T_NIL; aux = (luaO_equalObj(top-1, top) != aux);
nvalue(top-1) = 1; booleanresult:
if (aux) {
ttype(top-1) = LUA_T_NUMBER;
nvalue(top-1) = 1.0;
}
else ttype(top-1) = LUA_T_NIL;
break; break;
}
case LTOP: case LTOP:
luaV_comparison(L, top, LUA_T_NUMBER, LUA_T_NIL, LUA_T_NIL, IM_LT); top--;
top--; if (ttype(top-1) == LUA_T_NUMBER && ttype(top) == LUA_T_NUMBER)
break; aux = nvalue(top-1) < nvalue(top);
else if (ttype(top-1) == LUA_T_STRING && ttype(top) == LUA_T_STRING)
aux = luaV_strcomp(tsvalue(top-1), tsvalue(top)) < 0;
else {
call_binTM(L, top+1, IM_LT, "unexpected type in comparison");
break;
}
goto booleanresult;
case LEOP: case LEOP:
luaV_comparison(L, top, LUA_T_NUMBER, LUA_T_NUMBER, LUA_T_NIL, IM_LE);
top--; top--;
break; if (ttype(top-1) == LUA_T_NUMBER && ttype(top) == LUA_T_NUMBER)
aux = nvalue(top-1) <= nvalue(top);
else if (ttype(top-1) == LUA_T_STRING && ttype(top) == LUA_T_STRING)
aux = luaV_strcomp(tsvalue(top-1), tsvalue(top)) <= 0;
else {
call_binTM(L, top+1, IM_LE, "unexpected type in comparison");
break;
}
goto booleanresult;
case GTOP: case GTOP:
luaV_comparison(L, top, LUA_T_NIL, LUA_T_NIL, LUA_T_NUMBER, IM_GT);
top--; top--;
break; if (ttype(top-1) == LUA_T_NUMBER && ttype(top) == LUA_T_NUMBER)
aux = nvalue(top-1) > nvalue(top);
else if (ttype(top-1) == LUA_T_STRING && ttype(top) == LUA_T_STRING)
aux = luaV_strcomp(tsvalue(top-1), tsvalue(top)) > 0;
else {
call_binTM(L, top+1, IM_GT, "unexpected type in comparison");
break;
}
goto booleanresult;
case GEOP: case GEOP:
luaV_comparison(L, top, LUA_T_NIL, LUA_T_NUMBER, LUA_T_NUMBER, IM_GE);
top--; top--;
break; if (ttype(top-1) == LUA_T_NUMBER && ttype(top) == LUA_T_NUMBER)
aux = nvalue(top-1) >= nvalue(top);
else if (ttype(top-1) == LUA_T_STRING && ttype(top) == LUA_T_STRING)
aux = luaV_strcomp(tsvalue(top-1), tsvalue(top)) >= 0;
else {
call_binTM(L, top+1, IM_GE, "unexpected type in comparison");
break;
}
goto booleanresult;
case ADDOP: case ADDOP:
if (tonumber(top-1) || tonumber(top-2)) if (tonumber(top-1) || tonumber(top-2))

5
lvm.h
View File

@ -1,5 +1,5 @@
/* /*
** $Id: lvm.h,v 1.12 1999/11/22 13:12:07 roberto Exp roberto $ ** $Id: lvm.h,v 1.13 1999/12/01 19:50:08 roberto Exp roberto $
** Lua virtual machine ** Lua virtual machine
** See Copyright Notice in lua.h ** See Copyright Notice in lua.h
*/ */
@ -28,7 +28,6 @@ void luaV_getglobal (lua_State *L, GlobalVar *gv);
void luaV_setglobal (lua_State *L, GlobalVar *gv); void luaV_setglobal (lua_State *L, GlobalVar *gv);
StkId luaV_execute (lua_State *L, const Closure *cl, const TProtoFunc *tf, StkId base); StkId luaV_execute (lua_State *L, const Closure *cl, const TProtoFunc *tf, StkId base);
void luaV_closure (lua_State *L, int nelems); void luaV_closure (lua_State *L, int nelems);
void luaV_comparison (lua_State *L, StkId top, lua_Type ttype_less, void luaV_comparison (lua_State *L);
lua_Type ttype_equal, lua_Type ttype_great, IMS op);
#endif #endif