Skip to content

Commit

Permalink
FFI: Fix 64 bit shift fold rules.
Browse files Browse the repository at this point in the history
Thanks to Peter Cawley.

(cherry picked from commit 9e04372)

For `IR_BSHR`, `IR_BROL`, `IR_BROR` during `kfold_int64arith()` the left
argument is truncated down to 32 bits, which leads to incorrect results
if the right argument is >= 2^32.

Also, `IR_BSAR` does an unsigned shift rather than a signed shift, but
since this case branch is unreachable, it is harmless for now.

This patch fixes all misbehaviours (including possible for `IR_BSAR`) to
preserve IR semantics.

Sergey Kaplun:
* added the description and the test for the problem

Part of tarantool/tarantool#10199

Reviewed-by: Sergey Bronnikov <[email protected]>
Reviewed-by: Maxim Kokryashkin <[email protected]>
Signed-off-by: Sergey Kaplun <[email protected]>
(cherry picked from commit f0bc089)
  • Loading branch information
Mike Pall authored and Buristan committed Oct 17, 2024
1 parent 85ffbbf commit ac1cb36
Show file tree
Hide file tree
Showing 2 changed files with 81 additions and 4 deletions.
8 changes: 4 additions & 4 deletions src/lj_opt_fold.c
Original file line number Diff line number Diff line change
Expand Up @@ -382,10 +382,10 @@ static uint64_t kfold_int64arith(jit_State *J, uint64_t k1, uint64_t k2,
case IR_BOR: k1 |= k2; break;
case IR_BXOR: k1 ^= k2; break;
case IR_BSHL: k1 <<= (k2 & 63); break;
case IR_BSHR: k1 = (int32_t)((uint32_t)k1 >> (k2 & 63)); break;
case IR_BSAR: k1 >>= (k2 & 63); break;
case IR_BROL: k1 = (int32_t)lj_rol((uint32_t)k1, (k2 & 63)); break;
case IR_BROR: k1 = (int32_t)lj_ror((uint32_t)k1, (k2 & 63)); break;
case IR_BSHR: k1 >>= (k2 & 63); break;
case IR_BSAR: k1 = (uint64_t)((int64_t)k1 >> (k2 & 63)); break;
case IR_BROL: k1 = lj_rol(k1, (k2 & 63)); break;
case IR_BROR: k1 = lj_ror(k1, (k2 & 63)); break;
default: lj_assertJ(0, "bad IR op %d", op); break;
}
#else
Expand Down
77 changes: 77 additions & 0 deletions test/tarantool-tests/lj-1079-fix-64-bitshift-folds.test.lua
Original file line number Diff line number Diff line change
@@ -0,0 +1,77 @@
local tap = require('tap')

-- Test file to demonstrate LuaJIT misbehaviour on folding
-- for bitshift operations.
-- See also, https://github.com/LuaJIT/LuaJIT/issues/1079.

local test = tap.test('lj-1079-fix-64-bitshift-folds'):skipcond({
['Test requires JIT enabled'] = not jit.status(),
})

local bit = require('bit')

test:plan(4)

-- Generic function for `bit.ror()`, `bit.rol()`.
local function bitop_rotation(bitop_func)
local r = {}
for i = 1, 4 do
-- (i & k1) o k2 ==> (i o k2) & (k1 o k2)
-- XXX: Don't use named constants here to match folding rules.
-- `7LL` is just some mask, that doesn't change the `i` value.
-- `32` is used for the half bit-width rotation.
local int64 = bit.band(i, 7LL)
r[i] = tonumber(bitop_func(int64, 32))
end
return r
end

-- Similar function for `bit.rshift()`.
local function bitop_rshift_signed()
local r = {}
for i = 1, 4 do
-- (i & k1) o k2 ==> (i o k2) & (k1 o k2)
-- XXX: Use `-i` instead of `i` to prevent other folding due
-- to IR difference so the IRs don't match fold rule mask.
-- (-i & 7LL) < 1 << 32 => result == 0.
local int64 = bit.band(-i, 7LL)
r[i] = tonumber(bit.rshift(int64, 32))
end
return r
end

-- A little bit different example, which leads to the assertion
-- failure due to the incorrect recording.
local function bitop_rshift_huge()
local r = {}
for i = 1, 4 do
-- (i & k1) o k2 ==> (i o k2) & (k1 o k2)
-- XXX: Need to use cast to the int64_t via `+ 0LL`, see the
-- documentation [1] for the details.
-- [1]: https://bitop.luajit.org/semantics.html
local int64 = bit.band(2 ^ 33 + i, 2 ^ 33 + 0LL)
r[i] = tonumber(bit.rshift(int64, 32))
end
return r
end

local function test_64bitness(subtest, payload_func, bitop_func)
subtest:plan(1)

jit.off()
jit.flush()
local results_joff = payload_func(bitop_func)
jit.on()
-- Reset hotcounters.
jit.opt.start('hotloop=1')
local results_jon = payload_func(bitop_func)
subtest:is_deeply(results_jon, results_joff,
'same results for VM and JIT for ' .. subtest.name)
end

test:test('rshift signed', test_64bitness, bitop_rshift_signed)
test:test('rshift huge', test_64bitness, bitop_rshift_huge)
test:test('rol', test_64bitness, bitop_rotation, bit.rol)
test:test('ror', test_64bitness, bitop_rotation, bit.ror)

test:done(true)

0 comments on commit ac1cb36

Please sign in to comment.