mirror of
https://github.com/luau-lang/luau.git
synced 2025-01-07 11:59:11 +00:00
011c1afbde
I've decided to take a stab at implementing `bit32.byteswap` from the [recently merged RFC](https://github.com/Roblox/luau/blob/master/rfcs/function-bit32-byteswap.md). I asked on Discord for some guidance, but for the sake of posterity: this is my first time doing this and I am likely to have made some mistakes. The biggest gaps in this implementation are the lack of tests and the lack of native codegen support. I'd appreciate help with those since I'm not sure what's relevant for me to touch for tests, and I'm told that relevant assembler instructions don't exist publicly yet. Intuition tells me that Luau-side tests would go into `tests/conformance/bitwise.luau` but this is not well documented and I'm not sure how I'm meant to test built-in implementations. The current implementation compiles down to `bswap` and `rev` on x86 and ARM respectively when optimized. --------- Co-authored-by: Arseny Kapoulkine <arseny.kapoulkine@gmail.com>
174 lines
6 KiB
Lua
174 lines
6 KiB
Lua
-- This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
|
-- This file is based on Lua 5.x tests -- https://github.com/lua/lua/tree/master/testes
|
|
print("testing bitwise operations")
|
|
|
|
assert(bit32.band() == bit32.bnot(0))
|
|
assert(bit32.btest() == true)
|
|
assert(bit32.bor() == 0)
|
|
assert(bit32.bxor() == 0)
|
|
|
|
assert(bit32.band() == bit32.band(0xffffffff))
|
|
assert(bit32.band(1,2) == 0)
|
|
|
|
|
|
-- out-of-range numbers
|
|
assert(bit32.band(-1) == 0xffffffff)
|
|
assert(bit32.band(2^33 - 1) == 0xffffffff)
|
|
assert(bit32.band(-2^33 - 1) == 0xffffffff)
|
|
assert(bit32.band(2^33 + 1) == 1)
|
|
assert(bit32.band(-2^33 + 1) == 1)
|
|
assert(bit32.band(-2^40) == 0)
|
|
assert(bit32.band(2^40) == 0)
|
|
assert(bit32.band(-2^40 - 2) == 0xfffffffe)
|
|
assert(bit32.band(2^40 - 4) == 0xfffffffc)
|
|
|
|
assert(bit32.lrotate(0, -1) == 0)
|
|
assert(bit32.lrotate(0, 7) == 0)
|
|
assert(bit32.lrotate(0x12345678, 4) == 0x23456781)
|
|
assert(bit32.rrotate(0x12345678, -4) == 0x23456781)
|
|
assert(bit32.lrotate(0x12345678, -8) == 0x78123456)
|
|
assert(bit32.rrotate(0x12345678, 8) == 0x78123456)
|
|
assert(bit32.lrotate(0xaaaaaaaa, 2) == 0xaaaaaaaa)
|
|
assert(bit32.lrotate(0xaaaaaaaa, -2) == 0xaaaaaaaa)
|
|
for i = -50, 50 do
|
|
assert(bit32.lrotate(0x89abcdef, i) == bit32.lrotate(0x89abcdef, i%32))
|
|
end
|
|
|
|
assert(bit32.lshift(0x12345678, 4) == 0x23456780)
|
|
assert(bit32.lshift(0x12345678, 8) == 0x34567800)
|
|
assert(bit32.lshift(0x12345678, -4) == 0x01234567)
|
|
assert(bit32.lshift(0x12345678, -8) == 0x00123456)
|
|
assert(bit32.lshift(0x12345678, 32) == 0)
|
|
assert(bit32.lshift(0x12345678, -32) == 0)
|
|
assert(bit32.rshift(0x12345678, 4) == 0x01234567)
|
|
assert(bit32.rshift(0x12345678, 8) == 0x00123456)
|
|
assert(bit32.rshift(0x12345678, 32) == 0)
|
|
assert(bit32.rshift(0x12345678, -32) == 0)
|
|
assert(bit32.arshift(0x12345678, 0) == 0x12345678)
|
|
assert(bit32.arshift(0x12345678, 1) == 0x12345678 / 2)
|
|
assert(bit32.arshift(0x12345678, -1) == 0x12345678 * 2)
|
|
assert(bit32.arshift(-1, 1) == 0xffffffff)
|
|
assert(bit32.arshift(-1, 24) == 0xffffffff)
|
|
assert(bit32.arshift(-1, 32) == 0xffffffff)
|
|
assert(bit32.arshift(-1, -1) == (-1 * 2) % 2^32)
|
|
|
|
print("+")
|
|
-- some special cases
|
|
local c = {0, 1, 2, 3, 10, 0x80000000, 0xaaaaaaaa, 0x55555555,
|
|
0xffffffff, 0x7fffffff}
|
|
|
|
for _, b in pairs(c) do
|
|
assert(bit32.band(b) == b)
|
|
assert(bit32.band(b, b) == b)
|
|
assert(bit32.btest(b, b) == (b ~= 0))
|
|
assert(bit32.band(b, b, b) == b)
|
|
assert(bit32.btest(b, b, b) == (b ~= 0))
|
|
assert(bit32.band(b, bit32.bnot(b)) == 0)
|
|
assert(bit32.bor(b, bit32.bnot(b)) == bit32.bnot(0))
|
|
assert(bit32.bor(b) == b)
|
|
assert(bit32.bor(b, b) == b)
|
|
assert(bit32.bor(b, b, b) == b)
|
|
assert(bit32.bxor(b) == b)
|
|
assert(bit32.bxor(b, b) == 0)
|
|
assert(bit32.bxor(b, 0) == b)
|
|
assert(bit32.bxor(b, b, b) == b)
|
|
assert(bit32.bnot(b) ~= b)
|
|
assert(bit32.bnot(bit32.bnot(b)) == b)
|
|
assert(bit32.bnot(b) == 2^32 - 1 - b)
|
|
assert(bit32.lrotate(b, 32) == b)
|
|
assert(bit32.rrotate(b, 32) == b)
|
|
assert(bit32.lshift(bit32.lshift(b, -4), 4) == bit32.band(b, bit32.bnot(0xf)))
|
|
assert(bit32.rshift(bit32.rshift(b, 4), -4) == bit32.band(b, bit32.bnot(0xf)))
|
|
for i = -40, 40 do
|
|
assert(bit32.lshift(b, i) == math.floor((b * 2^i) % 2^32))
|
|
end
|
|
end
|
|
|
|
assert(not pcall(bit32.band, {}))
|
|
assert(not pcall(bit32.bnot, "a"))
|
|
assert(not pcall(bit32.lshift, 45))
|
|
assert(not pcall(bit32.lshift, 45, print))
|
|
assert(not pcall(bit32.rshift, 45, print))
|
|
|
|
print("+")
|
|
|
|
|
|
-- testing extract/replace
|
|
|
|
assert(bit32.extract(0x12345678, 0, 4) == 8)
|
|
assert(bit32.extract(0x12345678, 4, 4) == 7)
|
|
assert(bit32.extract(0xa0001111, 28, 4) == 0xa)
|
|
assert(bit32.extract(0xa0001111, 31, 1) == 1)
|
|
assert(bit32.extract(0x50000111, 31, 1) == 0)
|
|
assert(bit32.extract(0xf2345679, 0, 32) == 0xf2345679)
|
|
assert(bit32.extract(0xa0001111, 16) == 0)
|
|
assert(bit32.extract(0xa0001111, 31) == 1)
|
|
assert(bit32.extract(42, 1, 3) == 5)
|
|
|
|
local pos pos = 1
|
|
assert(bit32.extract(42, pos, 3) == 5) -- test bit32.extract builtin instead of bit32.extractk
|
|
|
|
assert(not pcall(bit32.extract, 0, -1))
|
|
assert(not pcall(bit32.extract, 0, 32))
|
|
assert(not pcall(bit32.extract, 0, 0, 33))
|
|
assert(not pcall(bit32.extract, 0, 31, 2))
|
|
|
|
assert(bit32.replace(0x12345678, 5, 28, 4) == 0x52345678)
|
|
assert(bit32.replace(0x12345678, 0x87654321, 0, 32) == 0x87654321)
|
|
assert(bit32.replace(0, 1, 2) == 2^2)
|
|
assert(bit32.replace(0, -1, 4) == 2^4)
|
|
assert(bit32.replace(-1, 0, 31) == 2^31 - 1)
|
|
assert(bit32.replace(-1, 0, 1, 2) == 2^32 - 7)
|
|
|
|
-- testing countlz/countrc
|
|
assert(bit32.countlz(0) == 32)
|
|
assert(bit32.countlz(42) == 26)
|
|
assert(bit32.countlz(0xffffffff) == 0)
|
|
assert(bit32.countlz(0x80000000) == 0)
|
|
assert(bit32.countlz(0x7fffffff) == 1)
|
|
|
|
assert(bit32.countrz(0) == 32)
|
|
assert(bit32.countrz(1) == 0)
|
|
assert(bit32.countrz(42) == 1)
|
|
assert(bit32.countrz(0x80000000) == 31)
|
|
assert(bit32.countrz(0x40000000) == 30)
|
|
assert(bit32.countrz(0x7fffffff) == 0)
|
|
|
|
-- testing byteswap
|
|
assert(bit32.byteswap(0x10203040) == 0x40302010)
|
|
assert(bit32.byteswap(0) == 0)
|
|
assert(bit32.byteswap(-1) == 0xffffffff)
|
|
|
|
--[[
|
|
This test verifies a fix in luauF_replace() where if the 4th
|
|
parameter was not a number, but the first three are numbers, it will
|
|
cause the Luau math library to crash.
|
|
]]--
|
|
|
|
assert(bit32.replace(-1, 0, 1, "2") == 2^32 - 7)
|
|
|
|
-- many of the tests above go through fastcall path
|
|
-- to make sure the basic implementations are also correct we test some functions with string->number coercions
|
|
assert(bit32.lrotate("0x12345678", 4) == 0x23456781)
|
|
assert(bit32.rrotate("0x12345678", -4) == 0x23456781)
|
|
assert(bit32.arshift("0x12345678", 1) == 0x12345678 / 2)
|
|
assert(bit32.arshift("-1", 32) == 0xffffffff)
|
|
assert(bit32.arshift("-1", 1) == 0xffffffff)
|
|
assert(bit32.bnot("1") == 0xfffffffe)
|
|
assert(bit32.band("1", 3) == 1)
|
|
assert(bit32.band(1, "3") == 1)
|
|
assert(bit32.band(1, 3, "5") == 1)
|
|
assert(bit32.bor("1", 2) == 3)
|
|
assert(bit32.bor(1, "2") == 3)
|
|
assert(bit32.bor(1, 3, "5") == 7)
|
|
assert(bit32.bxor("1", 3) == 2)
|
|
assert(bit32.bxor(1, "3") == 2)
|
|
assert(bit32.bxor(1, 3, "5") == 7)
|
|
assert(bit32.btest(1, "3") == true)
|
|
assert(bit32.btest("1", 3) == true)
|
|
assert(bit32.countlz("42") == 26)
|
|
assert(bit32.countrz("42") == 1)
|
|
assert(bit32.extract("42", 1, 3) == 5)
|
|
assert(bit32.byteswap("0xa1b2c3d4") == 0xd4c3b2a1)
|
|
|
|
return('OK')
|