Skip to content

Commit c0aa454

Browse files
Mike PallBuristan
Mike Pall
authored andcommitted
Reject negative getfenv()/setfenv() levels to prevent compiler warning.
Thanks to Sergey Kaplun. (cherry picked from commit 9d77734) When the number represented the level value is given to the `getfenv()`/`setfenv()`, it is cast to the `int`. Assume the given value is `2^31`, i.e. the resulting value after the cast is `INT_MIN`. After this, it will be decremented in `lj_debug_level()` and underflowed to the `INT_MAX`. That produces the UBSan warning about signed integer overflow. This patch raises the error early in the aforementioned functions, since a negative level value is meaningless. Sergey Kaplun: * added the description and the test for the problem Part of tarantool/tarantool#11055
1 parent 29e0827 commit c0aa454

File tree

2 files changed

+31
-0
lines changed

2 files changed

+31
-0
lines changed

src/lib_base.c

+4
Original file line numberDiff line numberDiff line change
@@ -144,6 +144,8 @@ LJLIB_CF(getfenv) LJLIB_REC(.)
144144
cTValue *o = L->base;
145145
if (!(o < L->top && tvisfunc(o))) {
146146
int level = lj_lib_optint(L, 1, 1);
147+
if (level < 0)
148+
lj_err_arg(L, 1, LJ_ERR_INVLVL);
147149
o = lj_debug_frame(L, level, &level);
148150
if (o == NULL)
149151
lj_err_arg(L, 1, LJ_ERR_INVLVL);
@@ -166,6 +168,8 @@ LJLIB_CF(setfenv)
166168
setgcref(L->env, obj2gco(t));
167169
return 0;
168170
}
171+
if (level < 0)
172+
lj_err_arg(L, 1, LJ_ERR_INVLVL);
169173
o = lj_debug_frame(L, level, &level);
170174
if (o == NULL)
171175
lj_err_arg(L, 1, LJ_ERR_INVLVL);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
local tap = require('tap')
2+
3+
-- The test file to demonstrate UBSan warning for `setfenv()` and
4+
-- `getfenv()` with a huge `level` value.
5+
-- See also: https://github.com/LuaJIT/LuaJIT/issues/1329.
6+
local test = tap.test('lj-1329-getfenv-setfenv-negative')
7+
8+
test:plan(4)
9+
10+
-- This number will be equal to `INT_MIN` when casted to `int`.
11+
-- After this, it will be decremented in `lj_debug_level()` and
12+
-- underflowed to the `INT_MAX`. That produces the UBSan warning
13+
-- about signed integer overflow.
14+
local LEVEL = 2 ^ 31
15+
local ERRMSG = 'invalid level'
16+
17+
-- Tests check the UBSan runtime error. Add assertions just to be
18+
-- sure that we don't change the behaviour.
19+
local status, errmsg = pcall(getfenv, LEVEL)
20+
test:ok(not status, 'getfenv: correct status')
21+
test:like(errmsg, ERRMSG, 'getfenv: correct error message')
22+
23+
status, errmsg = pcall(setfenv, LEVEL, {})
24+
test:ok(not status, 'setfenv: correct status')
25+
test:like(errmsg, ERRMSG, 'setfenv: correct error message')
26+
27+
test:done(true)

0 commit comments

Comments
 (0)