aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorRoberto Ierusalimschy <roberto@inf.puc-rio.br>2021-07-22 13:48:43 -0300
committerRoberto Ierusalimschy <roberto@inf.puc-rio.br>2021-07-22 13:48:43 -0300
commit439e45a2f69549b674d6a6e2023e8debfa00a2b8 (patch)
tree416db940cc3e4826119fda14d47f579b60cfb99b
parent62fb93442753cbfb828335cd172e71471dffd536 (diff)
downloadlua-439e45a2f69549b674d6a6e2023e8debfa00a2b8.tar.gz
lua-439e45a2f69549b674d6a6e2023e8debfa00a2b8.tar.bz2
lua-439e45a2f69549b674d6a6e2023e8debfa00a2b8.zip
Bug: luaL_tolstring may get confused with negative index
When object has a '__name' metafield, 'luaL_tolstring' used the received index after pushing a string on the stack.
-rw-r--r--lauxlib.c1
-rw-r--r--ltests.c3
-rw-r--r--testes/errors.lua16
3 files changed, 20 insertions, 0 deletions
diff --git a/lauxlib.c b/lauxlib.c
index 94835ef9..8ed1da11 100644
--- a/lauxlib.c
+++ b/lauxlib.c
@@ -881,6 +881,7 @@ LUALIB_API lua_Integer luaL_len (lua_State *L, int idx) {
881 881
882 882
883LUALIB_API const char *luaL_tolstring (lua_State *L, int idx, size_t *len) { 883LUALIB_API const char *luaL_tolstring (lua_State *L, int idx, size_t *len) {
884 idx = lua_absindex(L,idx);
884 if (luaL_callmeta(L, idx, "__tostring")) { /* metafield? */ 885 if (luaL_callmeta(L, idx, "__tostring")) { /* metafield? */
885 if (!lua_isstring(L, -1)) 886 if (!lua_isstring(L, -1))
886 luaL_error(L, "'__tostring' must return a string"); 887 luaL_error(L, "'__tostring' must return a string");
diff --git a/ltests.c b/ltests.c
index a50f7830..97834e38 100644
--- a/ltests.c
+++ b/ltests.c
@@ -1743,6 +1743,9 @@ static struct X { int x; } x;
1743 (void)s1; /* to avoid warnings */ 1743 (void)s1; /* to avoid warnings */
1744 lua_longassert((s == NULL && s1 == NULL) || strcmp(s, s1) == 0); 1744 lua_longassert((s == NULL && s1 == NULL) || strcmp(s, s1) == 0);
1745 } 1745 }
1746 else if EQ("Ltolstring") {
1747 luaL_tolstring(L1, getindex, NULL);
1748 }
1746 else if EQ("type") { 1749 else if EQ("type") {
1747 lua_pushstring(L1, luaL_typename(L1, getnum)); 1750 lua_pushstring(L1, luaL_typename(L1, getnum));
1748 } 1751 }
diff --git a/testes/errors.lua b/testes/errors.lua
index 825f37c2..a7dc479a 100644
--- a/testes/errors.lua
+++ b/testes/errors.lua
@@ -228,6 +228,22 @@ do -- named objects (field '__name')
228 checkmessage("return {} < XX", "table with My Type") 228 checkmessage("return {} < XX", "table with My Type")
229 checkmessage("return XX < io.stdin", "My Type with FILE*") 229 checkmessage("return XX < io.stdin", "My Type with FILE*")
230 _G.XX = nil 230 _G.XX = nil
231
232 if T then -- extra tests for 'luaL_tolstring'
233 -- bug in 5.4.3; 'luaL_tolstring' with negative indices
234 local x = setmetatable({}, {__name="TABLE"})
235 assert(T.testC("Ltolstring -1; return 1", x) == tostring(x))
236
237 local a, b = T.testC("pushint 10; Ltolstring -2; return 2", x)
238 assert(a == 10 and b == tostring(x))
239
240 setmetatable(x, {__tostring=function (o)
241 assert(o == x)
242 return "ABC"
243 end})
244 a, b, c = T.testC("pushint 10; Ltolstring -2; return 3", x)
245 assert(a == x and b == 10 and c == "ABC")
246 end
231end 247end
232 248
233-- global functions 249-- global functions