diff options
Diffstat (limited to 'ldo.c')
-rw-r--r-- | ldo.c | 34 |
1 files changed, 22 insertions, 12 deletions
@@ -1,5 +1,5 @@ | |||
1 | /* | 1 | /* |
2 | ** $Id: ldo.c,v 2.7 2004/06/02 19:07:55 roberto Exp roberto $ | 2 | ** $Id: ldo.c,v 2.8 2004/09/03 15:48:56 roberto Exp roberto $ |
3 | ** Stack and Call structure of Lua | 3 | ** Stack and Call structure of Lua |
4 | ** See Copyright Notice in lua.h | 4 | ** See Copyright Notice in lua.h |
5 | */ | 5 | */ |
@@ -93,7 +93,7 @@ int luaD_rawrunprotected (lua_State *L, Pfunc f, void *ud) { | |||
93 | 93 | ||
94 | 94 | ||
95 | static void restore_stack_limit (lua_State *L) { | 95 | static void restore_stack_limit (lua_State *L) { |
96 | L->stack_last = L->stack+L->stacksize-1; | 96 | lua_assert(L->stack_last - L->stack == L->stacksize - EXTRA_STACK - 1); |
97 | if (L->size_ci > LUA_MAXCALLS) { /* there was an overflow? */ | 97 | if (L->size_ci > LUA_MAXCALLS) { /* there was an overflow? */ |
98 | int inuse = (L->ci - L->base_ci); | 98 | int inuse = (L->ci - L->base_ci); |
99 | if (inuse + 1 < LUA_MAXCALLS) /* can `undo' overflow? */ | 99 | if (inuse + 1 < LUA_MAXCALLS) /* can `undo' overflow? */ |
@@ -121,9 +121,11 @@ static void correctstack (lua_State *L, TValue *oldstack) { | |||
121 | 121 | ||
122 | void luaD_reallocstack (lua_State *L, int newsize) { | 122 | void luaD_reallocstack (lua_State *L, int newsize) { |
123 | TValue *oldstack = L->stack; | 123 | TValue *oldstack = L->stack; |
124 | luaM_reallocvector(L, L->stack, L->stacksize, newsize, TValue); | 124 | int realsize = newsize + 1 + EXTRA_STACK; |
125 | L->stacksize = newsize; | 125 | lua_assert(L->stack_last - L->stack == L->stacksize - EXTRA_STACK - 1); |
126 | L->stack_last = L->stack+newsize-1-EXTRA_STACK; | 126 | luaM_reallocvector(L, L->stack, L->stacksize, realsize, TValue); |
127 | L->stacksize = realsize; | ||
128 | L->stack_last = L->stack+newsize; | ||
127 | correctstack(L, oldstack); | 129 | correctstack(L, oldstack); |
128 | } | 130 | } |
129 | 131 | ||
@@ -133,7 +135,7 @@ void luaD_reallocCI (lua_State *L, int newsize) { | |||
133 | luaM_reallocvector(L, L->base_ci, L->size_ci, newsize, CallInfo); | 135 | luaM_reallocvector(L, L->base_ci, L->size_ci, newsize, CallInfo); |
134 | L->size_ci = cast(unsigned short, newsize); | 136 | L->size_ci = cast(unsigned short, newsize); |
135 | L->ci = (L->ci - oldci) + L->base_ci; | 137 | L->ci = (L->ci - oldci) + L->base_ci; |
136 | L->end_ci = L->base_ci + L->size_ci; | 138 | L->end_ci = L->base_ci + L->size_ci - 1; |
137 | } | 139 | } |
138 | 140 | ||
139 | 141 | ||
@@ -141,11 +143,11 @@ void luaD_growstack (lua_State *L, int n) { | |||
141 | if (n <= L->stacksize) /* double size is enough? */ | 143 | if (n <= L->stacksize) /* double size is enough? */ |
142 | luaD_reallocstack(L, 2*L->stacksize); | 144 | luaD_reallocstack(L, 2*L->stacksize); |
143 | else | 145 | else |
144 | luaD_reallocstack(L, L->stacksize + n + EXTRA_STACK); | 146 | luaD_reallocstack(L, L->stacksize + n); |
145 | } | 147 | } |
146 | 148 | ||
147 | 149 | ||
148 | static void luaD_growCI (lua_State *L) { | 150 | static CallInfo *luaD_growCI (lua_State *L) { |
149 | if (L->size_ci > LUA_MAXCALLS) /* overflow while handling overflow? */ | 151 | if (L->size_ci > LUA_MAXCALLS) /* overflow while handling overflow? */ |
150 | luaD_throw(L, LUA_ERRERR); | 152 | luaD_throw(L, LUA_ERRERR); |
151 | else { | 153 | else { |
@@ -153,6 +155,7 @@ static void luaD_growCI (lua_State *L) { | |||
153 | if (L->size_ci > LUA_MAXCALLS) | 155 | if (L->size_ci > LUA_MAXCALLS) |
154 | luaG_runerror(L, "stack overflow"); | 156 | luaG_runerror(L, "stack overflow"); |
155 | } | 157 | } |
158 | return ++L->ci; | ||
156 | } | 159 | } |
157 | 160 | ||
158 | 161 | ||
@@ -170,6 +173,7 @@ void luaD_callhook (lua_State *L, int event, int line) { | |||
170 | ar.i_ci = L->ci - L->base_ci; | 173 | ar.i_ci = L->ci - L->base_ci; |
171 | luaD_checkstack(L, LUA_MINSTACK); /* ensure minimum stack size */ | 174 | luaD_checkstack(L, LUA_MINSTACK); /* ensure minimum stack size */ |
172 | L->ci->top = L->top + LUA_MINSTACK; | 175 | L->ci->top = L->top + LUA_MINSTACK; |
176 | lua_assert(L->ci->top <= L->stack_last); | ||
173 | L->allowhook = 0; /* cannot call hooks inside a hook */ | 177 | L->allowhook = 0; /* cannot call hooks inside a hook */ |
174 | lua_unlock(L); | 178 | lua_unlock(L); |
175 | (*hook)(L, &ar); | 179 | (*hook)(L, &ar); |
@@ -232,14 +236,18 @@ static StkId tryfuncTM (lua_State *L, StkId func) { | |||
232 | } | 236 | } |
233 | 237 | ||
234 | 238 | ||
239 | |||
240 | #define inc_ci(L) \ | ||
241 | ((L->ci == L->end_ci) ? luaD_growCI(L) : \ | ||
242 | (condhardstacktests(luaD_reallocCI(L, L->size_ci)), ++L->ci)) | ||
243 | |||
244 | |||
235 | int luaD_precall (lua_State *L, StkId func, int nresults) { | 245 | int luaD_precall (lua_State *L, StkId func, int nresults) { |
236 | LClosure *cl; | 246 | LClosure *cl; |
237 | ptrdiff_t funcr; | 247 | ptrdiff_t funcr; |
238 | if (!ttisfunction(func)) /* `func' is not a function? */ | 248 | if (!ttisfunction(func)) /* `func' is not a function? */ |
239 | func = tryfuncTM(L, func); /* check the `function' tag method */ | 249 | func = tryfuncTM(L, func); /* check the `function' tag method */ |
240 | funcr = savestack(L, func); | 250 | funcr = savestack(L, func); |
241 | if (L->ci + 1 == L->end_ci) luaD_growCI(L); | ||
242 | else condhardstacktests(luaD_reallocCI(L, L->size_ci)); | ||
243 | cl = &clvalue(func)->l; | 251 | cl = &clvalue(func)->l; |
244 | if (!cl->isC) { /* Lua function? prepare its call */ | 252 | if (!cl->isC) { /* Lua function? prepare its call */ |
245 | CallInfo *ci; | 253 | CallInfo *ci; |
@@ -256,10 +264,11 @@ int luaD_precall (lua_State *L, StkId func, int nresults) { | |||
256 | func = restorestack(L, funcr); | 264 | func = restorestack(L, funcr); |
257 | base = func + 1; | 265 | base = func + 1; |
258 | } | 266 | } |
259 | ci = ++L->ci; /* now `enter' new function */ | 267 | ci = inc_ci(L); /* now `enter' new function */ |
260 | ci->func = func; | 268 | ci->func = func; |
261 | L->base = ci->base = base; | 269 | L->base = ci->base = base; |
262 | ci->top = L->base + p->maxstacksize; | 270 | ci->top = L->base + p->maxstacksize; |
271 | lua_assert(ci->top <= L->stack_last); | ||
263 | ci->savedpc = p->code; /* starting point */ | 272 | ci->savedpc = p->code; /* starting point */ |
264 | ci->tailcalls = 0; | 273 | ci->tailcalls = 0; |
265 | ci->nresults = nresults; | 274 | ci->nresults = nresults; |
@@ -272,10 +281,11 @@ int luaD_precall (lua_State *L, StkId func, int nresults) { | |||
272 | CallInfo *ci; | 281 | CallInfo *ci; |
273 | int n; | 282 | int n; |
274 | luaD_checkstack(L, LUA_MINSTACK); /* ensure minimum stack size */ | 283 | luaD_checkstack(L, LUA_MINSTACK); /* ensure minimum stack size */ |
275 | ci = ++L->ci; /* now `enter' new function */ | 284 | ci = inc_ci(L); /* now `enter' new function */ |
276 | ci->func = restorestack(L, funcr); | 285 | ci->func = restorestack(L, funcr); |
277 | L->base = ci->base = ci->func + 1; | 286 | L->base = ci->base = ci->func + 1; |
278 | ci->top = L->top + LUA_MINSTACK; | 287 | ci->top = L->top + LUA_MINSTACK; |
288 | lua_assert(ci->top <= L->stack_last); | ||
279 | if (L->hookmask & LUA_MASKCALL) | 289 | if (L->hookmask & LUA_MASKCALL) |
280 | luaD_callhook(L, LUA_HOOKCALL, -1); | 290 | luaD_callhook(L, LUA_HOOKCALL, -1); |
281 | lua_unlock(L); | 291 | lua_unlock(L); |