2021-10-30 04:25:12 +08:00
-- This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
print ( " testing language/library basics " )
function concat ( head , ... )
if select ( ' # ' , ... ) == 0 then
return tostring ( head )
else
return tostring ( head ) .. " , " .. concat ( ... )
end
end
-- constants
assert ( tostring ( 1 ) == " 1 " )
assert ( tostring ( - 1 ) == " -1 " )
assert ( tostring ( 1.125 ) == " 1.125 " )
assert ( tostring ( true ) == " true " )
assert ( tostring ( nil ) == " nil " )
-- empty return
assert ( select ( ' # ' , ( function ( ) end ) ( ) ) == 0 )
assert ( select ( ' # ' , ( function ( ) return end ) ( ) ) == 0 )
-- locals
assert ( ( function ( ) local a = 1 return a end ) ( ) == 1 )
assert ( ( function ( ) local a , b , c = 1 , 2 , 3 return c end ) ( ) == 3 )
assert ( ( function ( ) local a , b , c = 1 , 2 return c end ) ( ) == nil )
assert ( ( function ( ) local a = 1 , 2 return a end ) ( ) == 1 )
-- function calls
local function foo ( a , b ) return b end
assert ( foo ( 1 ) == nil )
assert ( foo ( 1 , 2 ) == 2 )
assert ( foo ( 1 , 2 , 3 ) == 2 )
-- pcall
assert ( concat ( pcall ( function ( ) end ) ) == " true " )
assert ( concat ( pcall ( function ( ) return nil end ) ) == " true,nil " )
assert ( concat ( pcall ( function ( ) return 1 , 2 , 3 end ) ) == " true,1,2,3 " )
assert ( concat ( pcall ( function ( ) error ( " oops " ) end ) ) == " false,basic.lua:39: oops " )
-- assignments
assert ( ( function ( ) local a = 1 a = 2 return a end ) ( ) == 2 )
assert ( ( function ( ) a = 1 a = 2 return a end ) ( ) == 2 )
assert ( ( function ( ) local a = 1 a , b = 1 return a end ) ( ) == 1 )
assert ( ( function ( ) local a = 1 a , b = 1 return b end ) ( ) == nil )
assert ( ( function ( ) local a = 1 b = 2 a , b = b , a return a end ) ( ) == 2 )
assert ( ( function ( ) local a = 1 b = 2 a , b = b , a return b end ) ( ) == 1 )
assert ( ( function ( ) _G.foo = 1 return _G [ ' foo ' ] end ) ( ) == 1 )
assert ( ( function ( ) _G [ ' bar ' ] = 1 return _G.bar end ) ( ) == 1 )
assert ( ( function ( ) local a = 1 ( function ( ) a = 2 end ) ( ) return a end ) ( ) == 2 )
2022-08-12 04:42:54 +08:00
-- assignments with local conflicts
assert ( ( function ( ) local a , b = 1 , { } a , b [ a ] = 43 , - 1 return a + b [ 1 ] end ) ( ) == 42 )
assert ( ( function ( ) local a = { } local b = a a [ 1 ] , a = 43 , - 1 return a + b [ 1 ] end ) ( ) == 42 )
assert ( ( function ( ) local a , b = 1 , { } a , b [ a ] = ( function ( ) return 43 , - 1 end ) ( ) return a + b [ 1 ] end ) ( ) == 42 )
assert ( ( function ( ) local a = { } local b = a a [ 1 ] , a = ( function ( ) return 43 , - 1 end ) ( ) return a + b [ 1 ] end ) ( ) == 42 )
2021-10-30 04:25:12 +08:00
-- upvalues
assert ( ( function ( ) local a = 1 function foo ( ) return a end return foo ( ) end ) ( ) == 1 )
-- check upvalue propagation - foo must have numupvalues=1
assert ( ( function ( ) local a = 1 function foo ( ) return function ( ) return a end end return foo ( ) ( ) end ) ( ) == 1 )
-- check that function args are properly closed over
assert ( ( function ( ) function foo ( a ) return function ( ) return a end end return foo ( 1 ) ( ) end ) ( ) == 1 )
-- this checks local aliasing - b & a should share the same local slot, but the capture must return 1 instead of 2
assert ( ( function ( ) function foo ( ) local f do local a = 1 f = function ( ) return a end end local b = 2 return f end return foo ( ) ( ) end ) ( ) == 1 )
-- this checks local mutability - we capture a ref to 1 but must return 2
assert ( ( function ( ) function foo ( ) local a = 1 local function f ( ) return a end a = 2 return f end return foo ( ) ( ) end ) ( ) == 2 )
-- this checks upval mutability - we change the value from a context where it's upval
assert ( ( function ( ) function foo ( ) local a = 1 ( function ( ) a = 2 end ) ( ) return a end return foo ( ) end ) ( ) == 2 )
-- check self capture: does self go into any upvalues?
assert ( ( function ( ) local t = { f = 5 } function t : get ( ) return ( function ( ) return self.f end ) ( ) end return t : get ( ) end ) ( ) == 5 )
-- check self capture & close: is self copied to upval?
assert ( ( function ( ) function foo ( ) local t = { f = 5 } function t : get ( ) return function ( ) return self.f end end return t : get ( ) end return foo ( ) ( ) end ) ( ) == 5 )
-- if
assert ( ( function ( ) local a = 1 if a then a = 2 end return a end ) ( ) == 2 )
assert ( ( function ( ) local a if a then a = 2 end return a end ) ( ) == nil )
assert ( ( function ( ) local a = 0 if a then a = 1 else a = 2 end return a end ) ( ) == 1 )
assert ( ( function ( ) local a if a then a = 1 else a = 2 end return a end ) ( ) == 2 )
-- binary ops
assert ( ( function ( ) local a = 1 a = a + 2 return a end ) ( ) == 3 )
assert ( ( function ( ) local a = 1 a = a - 2 return a end ) ( ) == - 1 )
assert ( ( function ( ) local a = 1 a = a * 2 return a end ) ( ) == 2 )
assert ( ( function ( ) local a = 1 a = a / 2 return a end ) ( ) == 0.5 )
assert ( ( function ( ) local a = 5 a = a % 2 return a end ) ( ) == 1 )
assert ( ( function ( ) local a = 3 a = a ^ 2 return a end ) ( ) == 9 )
2022-10-14 06:59:53 +08:00
assert ( ( function ( ) local a = 9 a = a ^ 0.5 return a end ) ( ) == 3 )
2021-10-30 04:25:12 +08:00
assert ( ( function ( ) local a = ' 1 ' a = a .. ' 2 ' return a end ) ( ) == " 12 " )
assert ( ( function ( ) local a = ' 1 ' a = a .. ' 2 ' .. ' 3 ' return a end ) ( ) == " 123 " )
assert ( concat ( pcall ( function ( ) return ' 1 ' .. nil .. ' 2 ' end ) ) : match ( " ^false,.*attempt to concatenate nil with string " ) )
assert ( ( function ( ) local a = 1 a = a == 2 return a end ) ( ) == false )
assert ( ( function ( ) local a = 1 a = a ~= 2 return a end ) ( ) == true )
assert ( ( function ( ) local a = 1 a = a < 2 return a end ) ( ) == true )
assert ( ( function ( ) local a = 1 a = a <= 2 return a end ) ( ) == true )
assert ( ( function ( ) local a = 1 a = a > 2 return a end ) ( ) == false )
assert ( ( function ( ) local a = 1 a = a >= 2 return a end ) ( ) == false )
assert ( ( function ( ) local a = 1 a = a and 2 return a end ) ( ) == 2 )
assert ( ( function ( ) local a = nil a = a and 2 return a end ) ( ) == nil )
assert ( ( function ( ) local a = 1 a = a or 2 return a end ) ( ) == 1 )
assert ( ( function ( ) local a = nil a = a or 2 return a end ) ( ) == 2 )
2022-09-02 07:00:14 +08:00
assert ( ( function ( ) local a a = 1 local b = 2 b = a and b return b end ) ( ) == 2 )
assert ( ( function ( ) local a a = nil local b = 2 b = a and b return b end ) ( ) == nil )
assert ( ( function ( ) local a a = 1 local b = 2 b = a or b return b end ) ( ) == 1 )
assert ( ( function ( ) local a a = nil local b = 2 b = a or b return b end ) ( ) == 2 )
2021-10-30 04:25:12 +08:00
-- binary arithmetics coerces strings to numbers (sadly)
assert ( 1 + " 2 " == 3 )
assert ( 2 * " 0xa " == 20 )
-- unary ops
assert ( ( function ( ) local a = true a = not a return a end ) ( ) == false )
assert ( ( function ( ) local a = false a = not a return a end ) ( ) == true )
assert ( ( function ( ) local a = nil a = not a return a end ) ( ) == true )
assert ( ( function ( ) return # _G end ) ( ) == 0 )
assert ( ( function ( ) return # { 1 , 2 } end ) ( ) == 2 )
assert ( ( function ( ) return # ' g ' end ) ( ) == 1 )
assert ( ( function ( ) local a = 1 a = - a return a end ) ( ) == - 1 )
2022-07-01 07:29:02 +08:00
-- __len metamethod
assert ( ( function ( ) local ud = newproxy ( true ) getmetatable ( ud ) . __len = function ( ) return 42 end return # ud end ) ( ) == 42 )
assert ( ( function ( ) local t = { } setmetatable ( t , { __len = function ( ) return 42 end } ) return # t end ) ( ) == 42 )
2021-10-30 04:25:12 +08:00
-- while/repeat
assert ( ( function ( ) local a = 10 local b = 1 while a > 1 do b = b * 2 a = a - 1 end return b end ) ( ) == 512 )
assert ( ( function ( ) local a = 10 local b = 1 repeat b = b * 2 a = a - 1 until a == 1 return b end ) ( ) == 512 )
assert ( ( function ( ) local a = 10 local b = 1 while true do b = b * 2 a = a - 1 if a == 1 then break end end return b end ) ( ) == 512 )
assert ( ( function ( ) local a = 10 local b = 1 while true do b = b * 2 a = a - 1 if a == 1 then break else end end return b end ) ( ) == 512 )
assert ( ( function ( ) local a = 10 local b = 1 repeat b = b * 2 a = a - 1 if a == 1 then break end until false return b end ) ( ) == 512 )
assert ( ( function ( ) local a = 10 local b = 1 repeat b = b * 2 a = a - 1 if a == 1 then break else end until false return b end ) ( ) == 512 )
-- this makes sure a - 4 doesn't clobber a (which would happen if the lifetime of locals inside the repeat..until block is contained within
-- the block and ends before the condition is evaluated
assert ( ( function ( ) repeat local a = 5 until a - 4 < 0 or a - 4 >= 0 end ) ( ) == nil )
-- numeric for
-- basic tests with positive/negative step sizes
assert ( ( function ( ) local a = 1 for b = 1 , 9 do a = a * 2 end return a end ) ( ) == 512 )
assert ( ( function ( ) local a = 1 for b = 1 , 9 , 2 do a = a * 2 end return a end ) ( ) == 32 )
assert ( ( function ( ) local a = 1 for b = 1 , 9 , - 2 do a = a * 2 end return a end ) ( ) == 1 )
assert ( ( function ( ) local a = 1 for b = 9 , 1 , - 2 do a = a * 2 end return a end ) ( ) == 32 )
-- make sure break works
assert ( ( function ( ) local a = 1 for b = 1 , 9 do a = a * 2 if a == 128 then break end end return a end ) ( ) == 128 )
assert ( ( function ( ) local a = 1 for b = 1 , 9 do a = a * 2 if a == 128 then break else end end return a end ) ( ) == 128 )
-- make sure internal index is protected against modification
assert ( ( function ( ) local a = 1 for b = 9 , 1 , - 2 do a = a * 2 b = nil end return a end ) ( ) == 32 )
-- generic for
-- ipairs
assert ( ( function ( ) local a = ' ' for k in ipairs ( { 5 , 6 , 7 } ) do a = a .. k end return a end ) ( ) == " 123 " )
assert ( ( function ( ) local a = ' ' for k , v in ipairs ( { 5 , 6 , 7 } ) do a = a .. k end return a end ) ( ) == " 123 " )
assert ( ( function ( ) local a = ' ' for k , v in ipairs ( { 5 , 6 , 7 } ) do a = a .. v end return a end ) ( ) == " 567 " )
-- ipairs with gaps
assert ( ( function ( ) local a = ' ' for k in ipairs ( { 5 , 6 , 7 , nil , 8 } ) do a = a .. k end return a end ) ( ) == " 123 " )
assert ( ( function ( ) local a = ' ' for k , v in ipairs ( { 5 , 6 , 7 , nil , 8 } ) do a = a .. k end return a end ) ( ) == " 123 " )
assert ( ( function ( ) local a = ' ' for k , v in ipairs ( { 5 , 6 , 7 , nil , 8 } ) do a = a .. v end return a end ) ( ) == " 567 " )
-- manual ipairs/inext
local inext = ipairs ( { 5 , 6 , 7 } )
assert ( concat ( inext ( { 5 , 6 , 7 } , 2 ) ) == " 3,7 " )
-- pairs on array
assert ( ( function ( ) local a = ' ' for k in pairs ( { 5 , 6 , 7 } ) do a = a .. k end return a end ) ( ) == " 123 " )
assert ( ( function ( ) local a = ' ' for k , v in pairs ( { 5 , 6 , 7 } ) do a = a .. k end return a end ) ( ) == " 123 " )
assert ( ( function ( ) local a = ' ' for k , v in pairs ( { 5 , 6 , 7 } ) do a = a .. v end return a end ) ( ) == " 567 " )
-- pairs on array with gaps
assert ( ( function ( ) local a = ' ' for k in pairs ( { 5 , 6 , 7 , nil , 8 } ) do a = a .. k end return a end ) ( ) == " 1235 " )
assert ( ( function ( ) local a = ' ' for k , v in pairs ( { 5 , 6 , 7 , nil , 8 } ) do a = a .. k end return a end ) ( ) == " 1235 " )
assert ( ( function ( ) local a = ' ' for k , v in pairs ( { 5 , 6 , 7 , nil , 8 } ) do a = a .. v end return a end ) ( ) == " 5678 " )
-- pairs on table
assert ( ( function ( ) local a = { } for k in pairs ( { a = 1 , b = 2 , c = 3 } ) do a [ k ] = 1 end return a.a + a.b + a.c end ) ( ) == 3 )
assert ( ( function ( ) local a = { } for k , v in pairs ( { a = 1 , b = 2 , c = 3 } ) do a [ k ] = 1 end return a.a + a.b + a.c end ) ( ) == 3 )
assert ( ( function ( ) local a = { } for k , v in pairs ( { a = 1 , b = 2 , c = 3 } ) do a [ k ] = v end return a.a + a.b + a.c end ) ( ) == 6 )
-- pairs on mixed array/table + gaps in the array portion
-- note that a,b,c results in a,c,b during traversal since index is based on hash & size
assert ( ( function ( ) local a = { } for k , v in pairs ( { 1 , 2 , 3 , a = 5 , b = 6 , c = 7 } ) do a [ # a + 1 ] = v end return table.concat ( a , ' , ' ) end ) ( ) == " 1,2,3,5,7,6 " )
assert ( ( function ( ) local a = { } for k , v in pairs ( { 1 , 2 , 3 , nil , 4 , a = 5 , b = 6 , c = 7 } ) do a [ # a + 1 ] = v end return table.concat ( a , ' , ' ) end ) ( ) == " 1,2,3,4,5,7,6 " )
-- pairs manually
assert ( ( function ( ) local a = ' ' for k in next , { 5 , 6 , 7 } do a = a .. k end return a end ) ( ) == " 123 " )
assert ( ( function ( ) local a = ' ' for k , v in next , { 5 , 6 , 7 } do a = a .. k end return a end ) ( ) == " 123 " )
assert ( ( function ( ) local a = ' ' for k , v in next , { 5 , 6 , 7 } do a = a .. v end return a end ) ( ) == " 567 " )
assert ( ( function ( ) local a = { } for k in next , { a = 1 , b = 2 , c = 3 } do a [ k ] = 1 end return a.a + a.b + a.c end ) ( ) == 3 )
assert ( ( function ( ) local a = { } for k , v in next , { a = 1 , b = 2 , c = 3 } do a [ k ] = 1 end return a.a + a.b + a.c end ) ( ) == 3 )
assert ( ( function ( ) local a = { } for k , v in next , { a = 1 , b = 2 , c = 3 } do a [ k ] = v end return a.a + a.b + a.c end ) ( ) == 6 )
-- too many vars
assert ( ( function ( ) local a = ' ' for k , v , p in pairs ( { a = 1 , b = 2 , c = 3 } ) do a = a .. tostring ( p ) end return a end ) ( ) == " nilnilnil " )
-- make sure break works
assert ( ( function ( ) local a = 1 for _ in pairs ( { 1 , 2 , 3 } ) do a = a * 2 if a == 4 then break end end return a end ) ( ) == 4 )
assert ( ( function ( ) local a = 1 for _ in pairs ( { 1 , 2 , 3 } ) do a = a * 2 if a == 4 then break else end end return a end ) ( ) == 4 )
-- make sure internal index is protected against modification
assert ( ( function ( ) local a = 1 for b in pairs ( { 1 , 2 , 3 } ) do a = a * 2 b = nil end return a end ) ( ) == 8 )
-- make sure custom iterators work! example is from PIL 7.1
function list_iter ( t )
local i = 0
local n = table.getn ( t )
return function ( )
i = i + 1
if i <= n then return t [ i ] end
end
end
assert ( ( function ( ) local a = ' ' for e in list_iter ( { 4 , 2 , 1 } ) do a = a .. e end return a end ) ( ) == " 421 " )
-- make sure multret works in context of pairs() - this is a very painful to handle combination due to complex internal details
assert ( ( function ( ) local function f ( ) return { 5 , 6 , 7 } , 8 , 9 , 0 end local a = ' ' for k , v in ipairs ( f ( ) ) do a = a .. v end return a end ) ( ) == " 567 " )
-- table literals
-- basic tests
assert ( ( function ( ) local t = { } return # t end ) ( ) == 0 )
assert ( ( function ( ) local t = { 1 , 2 } return # t end ) ( ) == 2 )
assert ( ( function ( ) local t = { 1 , 2 } return t [ 1 ] + t [ 2 ] end ) ( ) == 3 )
assert ( ( function ( ) local t = { data = 4 } return t.data end ) ( ) == 4 )
assert ( ( function ( ) local t = { [ 1 + 2 ] = 4 } return t [ 3 ] end ) ( ) == 4 )
assert ( ( function ( ) local t = { data = 4 , [ 1 + 2 ] = 5 } return t.data + t [ 3 ] end ) ( ) == 9 )
assert ( ( function ( ) local t = { [ 1 ] = 1 , [ 2 ] = 2 } return t [ 1 ] + t [ 2 ] end ) ( ) == 3 )
-- since table ctor is chunked in groups of 16, we should be careful with edge cases around that
assert ( ( function ( ) return table.concat ( { } , ' , ' ) end ) ( ) == " " )
assert ( ( function ( ) return table.concat ( { 1 } , ' , ' ) end ) ( ) == " 1 " )
assert ( ( function ( ) return table.concat ( { 1 , 2 } , ' , ' ) end ) ( ) == " 1,2 " )
assert ( ( function ( ) return table.concat ( { 1 , 2 , 3 , 4 , 5 , 6 , 7 , 8 , 9 , 10 , 11 , 12 , 13 , 14 , 15 } , ' , ' ) end ) ( ) ==
" 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15 " )
assert ( ( function ( ) return table.concat ( { 1 , 2 , 3 , 4 , 5 , 6 , 7 , 8 , 9 , 10 , 11 , 12 , 13 , 14 , 15 , 16 } , ' , ' ) end ) ( ) == " 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16 " )
assert ( ( function ( ) return table.concat ( { 1 , 2 , 3 , 4 , 5 , 6 , 7 , 8 , 9 , 10 , 11 , 12 , 13 , 14 , 15 , 16 , 17 } , ' , ' ) end ) ( ) == " 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17 " )
-- some scripts rely on exact table traversal order; while it's evil to do so, let's check that it works
assert ( ( function ( )
local kSelectedBiomes = {
[ ' Mountains ' ] = true ,
[ ' Canyons ' ] = true ,
[ ' Dunes ' ] = true ,
[ ' Arctic ' ] = true ,
[ ' Lavaflow ' ] = true ,
[ ' Hills ' ] = true ,
[ ' Plains ' ] = true ,
[ ' Marsh ' ] = true ,
[ ' Water ' ] = true ,
}
local result = " "
for k in pairs ( kSelectedBiomes ) do result = result .. k end
return result
end ) ( ) == " ArcticDunesCanyonsWaterMountainsHillsLavaflowPlainsMarsh " )
-- multiple returns
-- local=
assert ( ( function ( ) function foo ( ) return 2 , 3 , 4 end local a , b , c = foo ( ) return ' ' .. a .. b .. c end ) ( ) == " 234 " )
assert ( ( function ( ) function foo ( ) return 2 , 3 , 4 end local a , b , c = 1 , foo ( ) return ' ' .. a .. b .. c end ) ( ) == " 123 " )
assert ( ( function ( ) function foo ( ) return 2 end local a , b , c = 1 , foo ( ) return ' ' .. a .. b .. tostring ( c ) end ) ( ) == " 12nil " )
-- assignments
assert ( ( function ( ) function foo ( ) return 2 , 3 end a , b , c , d = 1 , foo ( ) return ' ' .. a .. b .. c .. tostring ( d ) end ) ( ) == " 123nil " )
assert ( ( function ( ) function foo ( ) return 2 , 3 end local a , b , c , d a , b , c , d = 1 , foo ( ) return ' ' .. a .. b .. c .. tostring ( d ) end ) ( ) == " 123nil " )
-- varargs
-- local=
assert ( ( function ( ) function foo ( ... ) local a , b , c = ... return a + b + c end return foo ( 1 , 2 , 3 ) end ) ( ) == 6 )
assert ( ( function ( ) function foo ( x , ... ) local a , b , c = ... return a + b + c end return foo ( 1 , 2 , 3 , 4 ) end ) ( ) == 9 )
-- assignments
assert ( ( function ( ) function foo ( ... ) a , b , c = ... return a + b + c end return foo ( 1 , 2 , 3 ) end ) ( ) == 6 )
assert ( ( function ( ) function foo ( x , ... ) a , b , c = ... return a + b + c end return foo ( 1 , 2 , 3 , 4 ) end ) ( ) == 9 )
-- extra nils
assert ( ( function ( ) function foo ( ... ) local a , b , c = ... return tostring ( a ) .. tostring ( b ) .. tostring ( c ) end return foo ( 1 , 2 ) end ) ( ) == " 12nil " )
-- varargs + multiple returns
-- return
assert ( ( function ( ) function foo ( ... ) return ... end return concat ( foo ( 1 , 2 , 3 ) ) end ) ( ) == " 1,2,3 " )
assert ( ( function ( ) function foo ( ... ) return ... end return foo ( ) end ) ( ) == nil )
assert ( ( function ( ) function foo ( a , ... ) return a + 10 , ... end return concat ( foo ( 1 , 2 , 3 ) ) end ) ( ) == " 11,2,3 " )
-- call
assert ( ( function ( ) function foo ( ... ) return ... end function bar ( ... ) return foo ( ... ) end return concat ( bar ( 1 , 2 , 3 ) ) end ) ( ) == " 1,2,3 " )
assert ( ( function ( ) function foo ( ... ) return ... end function bar ( ... ) return foo ( ... ) end return bar ( ) end ) ( ) == nil )
assert ( ( function ( ) function foo ( a , ... ) return a + 10 , ... end function bar ( a , ... ) return foo ( a * 2 , ... ) end return concat ( bar ( 1 , 2 , 3 ) ) end ) ( ) == " 12,2,3 " )
-- manual pack
assert ( ( function ( ) function pack ( first , ... ) if not first then return { } end local t = pack ( ... ) table.insert ( t , 1 , first ) return t end function foo ( ... ) return pack ( ... ) end return # foo ( 0 , 1 , 2 ) end ) ( ) == 3 )
-- multret + table literals
-- basic tests
assert ( ( function ( ) function foo ( ... ) return { ... } end return # ( foo ( ) ) end ) ( ) == 0 )
assert ( ( function ( ) function foo ( ... ) return { ... } end return # ( foo ( 1 , 2 , 3 ) ) end ) ( ) == 3 )
assert ( ( function ( ) function foo ( ) return 1 , 2 , 3 end return # ( { foo ( ) } ) end ) ( ) == 3 )
-- since table ctor is chunked in groups of 16, we should be careful with edge cases around that
assert ( ( function ( ) function foo ( ) return 1 , 2 , 3 end return table.concat ( { foo ( ) } , ' , ' ) end ) ( ) == " 1,2,3 " )
assert ( ( function ( ) function foo ( ) return 1 , 2 , 3 end return table.concat ( { 1 , 2 , 3 , 4 , 5 , 6 , 7 , 8 , 9 , 10 , 11 , 12 , 13 , 14 , foo ( ) } , ' , ' ) end ) ( ) == " 1,2,3,4,5,6,7,8,9,10,11,12,13,14,1,2,3 " )
assert ( ( function ( ) function foo ( ) return 1 , 2 , 3 end return table.concat ( { 1 , 2 , 3 , 4 , 5 , 6 , 7 , 8 , 9 , 10 , 11 , 12 , 13 , 14 , 15 , foo ( ) } , ' , ' ) end ) ( ) == " 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,1,2,3 " )
assert ( ( function ( ) function foo ( ) return 1 , 2 , 3 end return table.concat ( { 1 , 2 , 3 , 4 , 5 , 6 , 7 , 8 , 9 , 10 , 11 , 12 , 13 , 14 , 15 , 16 , foo ( ) } , ' , ' ) end ) ( ) == " 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,1,2,3 " )
assert ( ( function ( ) function foo ( ) return 1 , 2 , 3 end return table.concat ( { 1 , 2 , 3 , 4 , 5 , 6 , 7 , 8 , 9 , 10 , 11 , 12 , 13 , 14 , 15 , 16 , 17 , foo ( ) } , ' , ' ) end ) ( ) == " 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,1,2,3 " )
-- table access
assert ( ( function ( ) local t = { 6 , 9 , 7 } return t [ 2 ] end ) ( ) == 9 )
assert ( ( function ( ) local t = { 6 , 9 , 7 } return t [ 0 ] end ) ( ) == nil )
assert ( ( function ( ) local t = { 6 , 9 , 7 } return t [ 4 ] end ) ( ) == nil )
assert ( ( function ( ) local t = { 6 , 9 , 7 } return t [ 4.5 ] end ) ( ) == nil )
assert ( ( function ( ) local t = { 6 , 9 , 7 , [ 4.5 ] = 11 } return t [ 4.5 ] end ) ( ) == 11 )
assert ( ( function ( ) local t = { 6 , 9 , 7 , a = 11 } return t [ ' a ' ] end ) ( ) == 11 )
assert ( ( function ( ) local t = { 6 , 9 , 7 } setmetatable ( t , { __index = function ( t , i ) return i * 10 end } ) return concat ( t [ 2 ] , t [ 5 ] ) end ) ( ) == " 9,50 " )
assert ( ( function ( ) local t = { 6 , 9 , 7 } t [ 2 ] = 10 return t [ 2 ] end ) ( ) == 10 )
assert ( ( function ( ) local t = { 6 , 9 , 7 } t [ 0 ] = 5 return t [ 0 ] end ) ( ) == 5 )
assert ( ( function ( ) local t = { 6 , 9 , 7 } t [ 4 ] = 10 return t [ 4 ] end ) ( ) == 10 )
assert ( ( function ( ) local t = { 6 , 9 , 7 } t [ 4.5 ] = 10 return t [ 4.5 ] end ) ( ) == 10 )
assert ( ( function ( ) local t = { 6 , 9 , 7 } t [ ' a ' ] = 11 return t [ ' a ' ] end ) ( ) == 11 )
assert ( ( function ( ) local t = { 6 , 9 , 7 } setmetatable ( t , { __newindex = function ( t , i , v ) rawset ( t , i * 10 , v ) end } ) t [ 1 ] = 17 t [ 5 ] = 1 return concat ( t [ 1 ] , t [ 5 ] , t [ 50 ] ) end ) ( ) == " 17,nil,1 " )
2022-02-12 02:43:14 +08:00
-- userdata access
assert ( ( function ( ) local ud = newproxy ( true ) getmetatable ( ud ) . __index = function ( ud , i ) return i * 10 end return ud [ 2 ] end ) ( ) == 20 )
assert ( ( function ( ) local ud = newproxy ( true ) getmetatable ( ud ) . __index = function ( ) return function ( self , i ) return i * 10 end end return ud : meow ( 2 ) end ) ( ) == 20 )
2021-10-30 04:25:12 +08:00
-- and/or
-- rhs is a constant
assert ( ( function ( ) local a = 1 a = a and 2 return a end ) ( ) == 2 )
assert ( ( function ( ) local a = nil a = a and 2 return a end ) ( ) == nil )
assert ( ( function ( ) local a = 1 a = a or 2 return a end ) ( ) == 1 )
assert ( ( function ( ) local a = nil a = a or 2 return a end ) ( ) == 2 )
-- rhs is a local
assert ( ( function ( ) local a = 1 local b = 2 a = a and b return a end ) ( ) == 2 )
assert ( ( function ( ) local a = nil local b = 2 a = a and b return a end ) ( ) == nil )
assert ( ( function ( ) local a = 1 local b = 2 a = a or b return a end ) ( ) == 1 )
assert ( ( function ( ) local a = nil local b = 2 a = a or b return a end ) ( ) == 2 )
-- rhs is a global (prevents optimizations)
assert ( ( function ( ) local a = 1 b = 2 a = a and b return a end ) ( ) == 2 )
assert ( ( function ( ) local a = nil b = 2 a = a and b return a end ) ( ) == nil )
assert ( ( function ( ) local a = 1 b = 2 a = a or b return a end ) ( ) == 1 )
assert ( ( function ( ) local a = nil b = 2 a = a or b return a end ) ( ) == 2 )
-- table access: method calls + fake oop via mt
assert ( ( function ( )
local Class = { }
Class.__index = Class
function Class . new ( )
local self = { }
setmetatable ( self , Class )
self.field = 5
return self
end
function Class : GetField ( )
return self.field
end
local object = Class.new ( )
return object : GetField ( )
end ) ( ) == 5 )
-- table access: evil indexer
assert ( ( function ( )
local a = { 5 }
local b = { 6 }
local mt = { __index = function ( ) return b [ 1 ] end }
setmetatable ( a , mt )
b = a.hi
return b
end ) ( ) == 6 )
-- table access: fast-path tests for array lookup
-- in-bounds array lookup shouldn't call into Lua, but if the element isn't there we'll still call the metatable
assert ( ( function ( ) local a = { 9 , [ 1.5 ] = 7 } return a [ 1 ] , a [ 2 ] , a [ 1.5 ] end ) ( ) == 9 , nil , 7 )
assert ( ( function ( ) local a = { 9 , [ 1.5 ] = 7 } setmetatable ( a , { __index = function ( ) return 5 end } ) return concat ( a [ 1 ] , a [ 2 ] , a [ 1.5 ] ) end ) ( ) == " 9,5,7 " )
assert ( ( function ( ) local a = { 9 , nil , 11 } setmetatable ( a , { __index = function ( ) return 5 end } ) return concat ( a [ 1 ] , a [ 2 ] , a [ 3 ] , a [ 4 ] ) end ) ( ) == " 9,5,11,5 " )
-- namecall for userdata: technically not officially supported but hard to test in a different way!
-- warning: this test may break at any time as we may decide that we'll only use userdata-namecall on tagged user data objects
assert ( ( function ( )
local obj = newproxy ( true )
getmetatable ( obj ) . __namecall = function ( self , arg ) return 42 + arg end
return obj : Foo ( 10 )
end ) ( ) == 52 )
assert ( ( function ( )
local obj = newproxy ( true )
local t = { }
setmetatable ( t , { __call = function ( self1 , self2 , arg ) return 42 + arg end } )
getmetatable ( obj ) . __namecall = t
return obj : Foo ( 10 )
end ) ( ) == 52 )
-- namecall for oop to test fast paths
assert ( ( function ( )
local Class = { }
Class.__index = Class
function Class : new ( klass , v ) -- note, this isn't necessarily common but it exercises additional namecall paths
local self = { value = v }
setmetatable ( self , Class )
return self
end
function Class : get ( )
return self.value
end
function Class : set ( v )
self.value = v
end
local n = Class : new ( 32 )
n : set ( 42 )
return n : get ( )
end ) ( ) == 42 )
-- comparison
-- basic types
assert ( ( function ( ) a = nil return concat ( a == nil , a ~= nil ) end ) ( ) == " true,false " )
assert ( ( function ( ) a = nil return concat ( a == 1 , a ~= 1 ) end ) ( ) == " false,true " )
assert ( ( function ( ) a = 1 return concat ( a == 1 , a ~= 1 ) end ) ( ) == " true,false " )
assert ( ( function ( ) a = 1 return concat ( a == 2 , a ~= 2 ) end ) ( ) == " false,true " )
assert ( ( function ( ) a = true return concat ( a == true , a ~= true ) end ) ( ) == " true,false " )
assert ( ( function ( ) a = true return concat ( a == false , a ~= false ) end ) ( ) == " false,true " )
assert ( ( function ( ) a = ' a ' return concat ( a == ' a ' , a ~= ' a ' ) end ) ( ) == " true,false " )
assert ( ( function ( ) a = ' a ' return concat ( a == ' b ' , a ~= ' b ' ) end ) ( ) == " false,true " )
-- tables, reference equality (no mt)
assert ( ( function ( ) a = { } return concat ( a == a , a ~= a ) end ) ( ) == " true,false " )
assert ( ( function ( ) a = { } b = { } return concat ( a == b , a ~= b ) end ) ( ) == " false,true " )
-- tables, reference equality (mt without __eq)
assert ( ( function ( ) a = { } setmetatable ( a , { } ) return concat ( a == a , a ~= a ) end ) ( ) == " true,false " )
assert ( ( function ( ) a = { } b = { } mt = { } setmetatable ( a , mt ) setmetatable ( b , mt ) return concat ( a == b , a ~= b ) end ) ( ) == " false,true " )
-- tables, __eq with same mt/different mt but same function/different function
assert ( ( function ( ) a = { } b = { } mt = { __eq = function ( l , r ) return # l == # r end } setmetatable ( a , mt ) setmetatable ( b , mt ) return concat ( a == b , a ~= b ) end ) ( ) == " true,false " )
assert ( ( function ( ) a = { } b = { } function eq ( l , r ) return # l == # r end setmetatable ( a , { __eq = eq } ) setmetatable ( b , { __eq = eq } ) return concat ( a == b , a ~= b ) end ) ( ) == " true,false " )
assert ( ( function ( ) a = { } b = { } setmetatable ( a , { __eq = function ( l , r ) return # l == # r end } ) setmetatable ( b , { __eq = function ( l , r ) return # l == # r end } ) return concat ( a == b , a ~= b ) end ) ( ) == " false,true " )
2021-12-03 07:20:08 +08:00
-- userdata, reference equality (no mt or mt.__eq)
assert ( ( function ( ) a = newproxy ( ) return concat ( a == newproxy ( ) , a ~= newproxy ( ) ) end ) ( ) == " false,true " )
2021-10-30 04:25:12 +08:00
assert ( ( function ( ) a = newproxy ( true ) return concat ( a == newproxy ( true ) , a ~= newproxy ( true ) ) end ) ( ) == " false,true " )
-- rawequal
assert ( rawequal ( true , 5 ) == false )
assert ( rawequal ( nil , nil ) == true )
assert ( rawequal ( true , false ) == false )
assert ( rawequal ( true , true ) == true )
assert ( rawequal ( 0 , - 0 ) == true )
assert ( rawequal ( 1 , 2 ) == false )
assert ( rawequal ( " a " , " a " ) == true )
assert ( rawequal ( " a " , " b " ) == false )
assert ( ( function ( ) a = { } b = { } mt = { __eq = function ( l , r ) return # l == # r end } setmetatable ( a , mt ) setmetatable ( b , mt ) return concat ( a == b , rawequal ( a , b ) ) end ) ( ) == " true,false " )
2022-10-14 06:59:53 +08:00
-- rawequal fallback
assert ( concat ( pcall ( rawequal , " a " , " a " ) ) == " true,true " )
assert ( concat ( pcall ( rawequal , " a " , " b " ) ) == " true,false " )
assert ( concat ( pcall ( rawequal , " a " , nil ) ) == " true,false " )
assert ( pcall ( rawequal , " a " ) == false )
2021-10-30 04:25:12 +08:00
-- metatable ops
local function vec3t ( x , y , z )
2022-02-12 02:43:14 +08:00
return setmetatable ( { x = x , y = y , z = z } , {
2021-10-30 04:25:12 +08:00
__add = function ( l , r ) return vec3t ( l.x + r.x , l.y + r.y , l.z + r.z ) end ,
__sub = function ( l , r ) return vec3t ( l.x - r.x , l.y - r.y , l.z - r.z ) end ,
__mul = function ( l , r ) return type ( r ) == " number " and vec3t ( l.x * r , l.y * r , l.z * r ) or vec3t ( l.x * r.x , l.y * r.y , l.z * r.z ) end ,
__div = function ( l , r ) return type ( r ) == " number " and vec3t ( l.x / r , l.y / r , l.z / r ) or vec3t ( l.x / r.x , l.y / r.y , l.z / r.z ) end ,
__unm = function ( v ) return vec3t ( - v.x , - v.y , - v.z ) end ,
__tostring = function ( v ) return string.format ( " %g, %g, %g " , v.x , v.y , v.z ) end
} )
end
-- reg vs reg
assert ( ( function ( ) return tostring ( vec3t ( 1 , 2 , 3 ) + vec3t ( 4 , 5 , 6 ) ) end ) ( ) == " 5, 7, 9 " )
assert ( ( function ( ) return tostring ( vec3t ( 1 , 2 , 3 ) - vec3t ( 4 , 5 , 6 ) ) end ) ( ) == " -3, -3, -3 " )
assert ( ( function ( ) return tostring ( vec3t ( 1 , 2 , 3 ) * vec3t ( 4 , 5 , 6 ) ) end ) ( ) == " 4, 10, 18 " )
assert ( ( function ( ) return tostring ( vec3t ( 1 , 2 , 3 ) / vec3t ( 2 , 4 , 8 ) ) end ) ( ) == " 0.5, 0.5, 0.375 " )
-- reg vs constant
assert ( ( function ( ) return tostring ( vec3t ( 1 , 2 , 3 ) * 2 ) end ) ( ) == " 2, 4, 6 " )
assert ( ( function ( ) return tostring ( vec3t ( 1 , 2 , 3 ) / 2 ) end ) ( ) == " 0.5, 1, 1.5 " )
-- unary
assert ( ( function ( ) return tostring ( - vec3t ( 1 , 2 , 3 ) ) end ) ( ) == " -1, -2, -3 " )
-- string comparison
assert ( ( function ( ) function cmp ( a , b ) return a < b , a <= b , a > b , a >= b end return concat ( cmp ( ' a ' , ' b ' ) ) end ) ( ) == " true,true,false,false " )
assert ( ( function ( ) function cmp ( a , b ) return a < b , a <= b , a > b , a >= b end return concat ( cmp ( ' a ' , ' a ' ) ) end ) ( ) == " false,true,false,true " )
assert ( ( function ( ) function cmp ( a , b ) return a < b , a <= b , a > b , a >= b end return concat ( cmp ( ' a ' , ' ' ) ) end ) ( ) == " false,false,true,true " )
assert ( ( function ( ) function cmp ( a , b ) return a < b , a <= b , a > b , a >= b end return concat ( cmp ( ' ' , ' \\ 0 ' ) ) end ) ( ) == " true,true,false,false " )
assert ( ( function ( ) function cmp ( a , b ) return a < b , a <= b , a > b , a >= b end return concat ( cmp ( ' abc ' , ' abd ' ) ) end ) ( ) == " true,true,false,false " )
assert ( ( function ( ) function cmp ( a , b ) return a < b , a <= b , a > b , a >= b end return concat ( cmp ( ' ab \\ 0c ' , ' ab \\ 0d ' ) ) end ) ( ) == " true,true,false,false " )
assert ( ( function ( ) function cmp ( a , b ) return a < b , a <= b , a > b , a >= b end return concat ( cmp ( ' ab \\ 0c ' , ' ab \\ 0 ' ) ) end ) ( ) == " false,false,true,true " )
2022-10-07 07:55:58 +08:00
assert ( ( function ( ) function cmp ( a , b ) return a < b , a <= b , a > b , a >= b end return concat ( cmp ( ' \\ 0a ' , ' \\ 0b ' ) ) end ) ( ) == " true,true,false,false " )
assert ( ( function ( ) function cmp ( a , b ) return a < b , a <= b , a > b , a >= b end return concat ( cmp ( ' a ' , ' a \\ 0 ' ) ) end ) ( ) == " true,true,false,false " )
assert ( ( function ( ) function cmp ( a , b ) return a < b , a <= b , a > b , a >= b end return concat ( cmp ( ' a ' , ' \200 ' ) ) end ) ( ) == " true,true,false,false " )
2021-10-30 04:25:12 +08:00
-- array access
assert ( ( function ( ) local a = { 4 , 5 , 6 } return a [ 3 ] end ) ( ) == 6 )
assert ( ( function ( ) local a = { 4 , 5 , nil , 6 } return a [ 3 ] end ) ( ) == nil )
assert ( ( function ( ) local a = { 4 , 5 , nil , 6 } setmetatable ( a , { __index = function ( ) return 42 end } ) return a [ 4 ] end ) ( ) == 6 )
assert ( ( function ( ) local a = { 4 , 5 , nil , 6 } setmetatable ( a , { __index = function ( ) return 42 end } ) return a [ 3 ] end ) ( ) == 42 )
assert ( ( function ( ) local a = { 4 , 5 , 6 } a [ 3 ] = 8 return a [ 3 ] end ) ( ) == 8 )
assert ( ( function ( ) local a = { 4 , 5 , nil , 6 } a [ 3 ] = 8 return a [ 3 ] end ) ( ) == 8 )
assert ( ( function ( ) local a = { 4 , 5 , nil , 6 } setmetatable ( a , { __newindex = function ( t , i ) rawset ( t , i , 42 ) end } ) a [ 4 ] = 0 return a [ 4 ] end ) ( ) == 0 )
assert ( ( function ( ) local a = { 4 , 5 , nil , 6 } setmetatable ( a , { __newindex = function ( t , i ) rawset ( t , i , 42 ) end } ) a [ 3 ] = 0 return a [ 3 ] end ) ( ) == 42 )
-- array index for literal
assert ( ( function ( ) local a = { 4 , 5 , nil , 6 } return concat ( a [ 1 ] , a [ 3 ] , a [ 4 ] , a [ 100 ] ) end ) ( ) == " 4,nil,6,nil " )
assert ( ( function ( ) local a = { 4 , 5 , nil , 6 } a [ 1 ] = 42 a [ 3 ] = 0 a [ 100 ] = 75 return concat ( a [ 1 ] , a [ 3 ] , a [ 75 ] , a [ 100 ] ) end ) ( ) == " 42,0,nil,75 " )
-- load error
assert ( ( function ( ) return concat ( loadstring ( ' hello world ' ) ) end ) ( ) == " nil,[string \" hello world \" ]:1: Incomplete statement: expected assignment or a function call " )
-- many arguments & locals
function f ( p1 , p2 , p3 , p4 , p5 , p6 , p7 , p8 , p9 , p10 ,
p11 , p12 , p13 , p14 , p15 , p16 , p17 , p18 , p19 , p20 ,
p21 , p22 , p23 , p24 , p25 , p26 , p27 , p28 , p29 , p30 ,
p31 , p32 , p33 , p34 , p35 , p36 , p37 , p38 , p39 , p40 ,
p41 , p42 , p43 , p44 , p45 , p46 , p48 , p49 , p50 , ... )
local a1 , a2 , a3 , a4 , a5 , a6 , a7 , a8 , a9 , a10 , a11 , a12 , a13 , a14
end
assert ( f ( ) == nil )
-- upvalues & loops (validates timely closing)
assert ( ( function ( )
local res = { }
for i = 1 , 5 do
res [ # res + 1 ] = ( function ( ) return i end )
end
local sum = 0
for i , f in pairs ( res ) do sum = sum + f ( ) end
return sum
end ) ( ) == 15 )
assert ( ( function ( )
local res = { }
for i in ipairs ( { 1 , 2 , 3 , 4 , 5 } ) do
res [ # res + 1 ] = ( function ( ) return i end )
end
local sum = 0
for i , f in pairs ( res ) do sum = sum + f ( ) end
return sum
end ) ( ) == 15 )
assert ( ( function ( )
local res = { }
local i = 0
while i <= 5 do
local j = i
res [ # res + 1 ] = ( function ( ) return j end )
i = i + 1
end
local sum = 0
for i , f in pairs ( res ) do sum = sum + f ( ) end
return sum
end ) ( ) == 15 )
assert ( ( function ( )
local res = { }
local i = 0
repeat
local j = i
res [ # res + 1 ] = ( function ( ) return j end )
i = i + 1
until i > 5
local sum = 0
for i , f in pairs ( res ) do sum = sum + f ( ) end
return sum
end ) ( ) == 15 )
-- upvalues & loops & break!
assert ( ( function ( )
local res = { }
for i = 1 , 10 do
res [ # res + 1 ] = ( function ( ) return i end )
if i == 5 then
break
end
end
local sum = 0
for i , f in pairs ( res ) do sum = sum + f ( ) end
return sum
end ) ( ) == 15 )
assert ( ( function ( )
local res = { }
for i in ipairs ( { 1 , 2 , 3 , 4 , 5 , 6 , 7 , 8 , 9 , 10 } ) do
res [ # res + 1 ] = ( function ( ) return i end )
if i == 5 then
break
end
end
local sum = 0
for i , f in pairs ( res ) do sum = sum + f ( ) end
return sum
end ) ( ) == 15 )
assert ( ( function ( )
local res = { }
local i = 0
while i < 10 do
local j = i
res [ # res + 1 ] = ( function ( ) return j end )
if i == 5 then
break
end
i = i + 1
end
local sum = 0
for i , f in pairs ( res ) do sum = sum + f ( ) end
return sum
end ) ( ) == 15 )
assert ( ( function ( )
local res = { }
local i = 0
repeat
local j = i
res [ # res + 1 ] = ( function ( ) return j end )
if i == 5 then
break
end
i = i + 1
until i >= 10
local sum = 0
for i , f in pairs ( res ) do sum = sum + f ( ) end
return sum
end ) ( ) == 15 )
-- the reason why this test is interesting is that the table created here has arraysize=0 and a single hash element with key = 1.0
-- ipairs must iterate through that
assert ( ( function ( )
local arr = { [ 1 ] = 42 }
local sum = 0
for i , v in ipairs ( arr ) do
sum = sum + v
end
return sum
end ) ( ) == 42 )
-- the reason why this test is interesting is it ensures we do correct mutability analysis for locals
local function chainTest ( n )
local first = nil
local last = nil
-- Build chain of n equality constraints
for i = 0 , n do
local name = " v " .. i ;
if i == 0 then first = name end
if i == n then last = name end
end
return concat ( first , last )
end
assert ( chainTest ( 100 ) == " v0,v100 " )
-- this validates import fallbacks
assert ( pcall ( function ( ) return idontexist.a end ) == false )
-- make sure that NaN is preserved by the bytecode compiler
local realnan = tostring ( math.abs ( 0 ) / math.abs ( 0 ) )
assert ( tostring ( 0 / 0 * 0 ) == realnan )
assert ( tostring ( ( - 1 ) ^ ( 1 / 2 ) ) == realnan )
-- make sure that negative zero is preserved by bytecode compiler
assert ( tostring ( 0 ) == " 0 " )
assert ( tostring ( - 0 ) == " -0 " )
-- test newline handling in long strings
assert ( ( function ( )
local s1 = [ [
] ]
local s2 = [ [
] ]
local s3 = [ [
foo
bar ] ]
local s4 = [ [
foo
bar
] ]
return concat ( s1 , s2 , s3 , s4 )
end ) ( ) == " , \n ,foo \n bar,foo \n bar \n " )
-- fastcall
-- positive tests for all simple examples; note that in this case the call is a multret call (nresults=LUA_MULTRET)
assert ( ( function ( ) return math.abs ( - 5 ) end ) ( ) == 5 )
assert ( ( function ( ) local abs = math.abs return abs ( - 5 ) end ) ( ) == 5 )
assert ( ( function ( ) local abs = math.abs function foo ( ) return abs ( - 5 ) end return foo ( ) end ) ( ) == 5 )
-- vararg testing - in this case nparams = LUA_MULTRET and it gets adjusted before execution
assert ( ( function ( ) function foo ( ... ) return math.abs ( ... ) end return foo ( - 5 ) end ) ( ) == 5 )
assert ( ( function ( ) function foo ( ... ) local abs = math.abs return abs ( ... ) end return foo ( - 5 ) end ) ( ) == 5 )
assert ( ( function ( ) local abs = math.abs function foo ( ... ) return abs ( ... ) end return foo ( - 5 ) end ) ( ) == 5 )
-- NOTE: getfenv breaks fastcalls for the remainder of the source! hence why this is delayed until the end
function testgetfenv ( )
getfenv ( )
2022-07-15 06:39:35 +08:00
-- declare constant so that at O2 this test doesn't interfere with constant folding which we can't deoptimize
local negfive negfive = - 5
2021-10-30 04:25:12 +08:00
-- getfenv breaks fastcalls (we assume we can't rely on knowing the semantics), but behavior shouldn't change
2022-07-15 06:39:35 +08:00
assert ( ( function ( ) return math.abs ( negfive ) end ) ( ) == 5 )
assert ( ( function ( ) local abs = math.abs return abs ( negfive ) end ) ( ) == 5 )
assert ( ( function ( ) local abs = math.abs function foo ( ) return abs ( negfive ) end return foo ( ) end ) ( ) == 5 )
2021-10-30 04:25:12 +08:00
-- ... unless you actually reassign the function :D
getfenv ( ) . math = { abs = function ( n ) return n * n end }
2022-07-15 06:39:35 +08:00
assert ( ( function ( ) return math.abs ( negfive ) end ) ( ) == 25 )
assert ( ( function ( ) local abs = math.abs return abs ( negfive ) end ) ( ) == 25 )
assert ( ( function ( ) local abs = math.abs function foo ( ) return abs ( negfive ) end return foo ( ) end ) ( ) == 25 )
2021-10-30 04:25:12 +08:00
end
-- you need to have enough arguments and arguments of the right type; if you don't, we'll fallback to the regular code. This checks coercions
-- first to make sure all fallback paths work
assert ( ( function ( ) return math.abs ( ' -5 ' ) end ) ( ) == 5 )
assert ( ( function ( ) local abs = math.abs return abs ( ' -5 ' ) end ) ( ) == 5 )
assert ( ( function ( ) local abs = math.abs function foo ( ) return abs ( ' -5 ' ) end return foo ( ) end ) ( ) == 5 )
-- if you don't have enough arguments or types are wrong, we fall back to the regular execution; this checks that the error generated is actually correct
assert ( concat ( pcall ( function ( ) return math.abs ( ) end ) ) : match ( " missing argument #1 to 'abs' " ) )
assert ( concat ( pcall ( function ( ) return math.abs ( nil ) end ) ) : match ( " invalid argument #1 to 'abs' " ) )
assert ( concat ( pcall ( function ( ) return math.abs ( { } ) end ) ) : match ( " invalid argument #1 to 'abs' " ) )
-- very large unpack
assert ( select ( ' # ' , table.unpack ( { 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 } ) ) == 263 )
-- basic continue in for/while/repeat loops
assert ( ( function ( ) local a = 1 for i = 1 , 8 do a = a + 1 if a < 5 then continue end a = a * 2 end return a end ) ( ) == 190 )
assert ( ( function ( ) local a = 1 while a < 100 do a = a + 1 if a < 5 then continue end a = a * 2 end return a end ) ( ) == 190 )
assert ( ( function ( ) local a = 1 repeat a = a + 1 if a < 5 then continue end a = a * 2 until a > 100 return a end ) ( ) == 190 )
-- upvalues, loops, continue
assert ( ( function ( )
local res = { }
for i = 1 , 10 do
res [ # res + 1 ] = ( function ( ) return i end )
if i == 5 then
continue
end
i = i * 2
end
local sum = 0
for i , f in pairs ( res ) do sum = sum + f ( ) end
return sum
end ) ( ) == 105 )
assert ( ( function ( )
local res = { }
for i in ipairs ( { 1 , 2 , 3 , 4 , 5 , 6 , 7 , 8 , 9 , 10 } ) do
res [ # res + 1 ] = ( function ( ) return i end )
if i == 5 then
continue
end
i = i * 2
end
local sum = 0
for i , f in pairs ( res ) do sum = sum + f ( ) end
return sum
end ) ( ) == 105 )
assert ( ( function ( )
local res = { }
local i = 1
while i <= 10 do
local j = i
res [ # res + 1 ] = ( function ( ) return j end )
if i == 5 then
i = i + 1
continue
end
i = i + 1
j = j * 2
end
local sum = 0
for i , f in pairs ( res ) do sum = sum + f ( ) end
return sum
end ) ( ) == 105 )
assert ( ( function ( )
local res = { }
local i = 1
repeat
local j = i
res [ # res + 1 ] = ( function ( ) return j end )
if i == 5 then
i = i + 1
continue
end
i = i + 1
j = j * 2
until i > 10
local sum = 0
for i , f in pairs ( res ) do sum = sum + f ( ) end
return sum
end ) ( ) == 105 )
2022-03-18 08:06:25 +08:00
-- shrinking array part
assert ( ( function ( )
local t = table.create ( 100 , 42 )
for i = 1 , 90 do t [ i ] = nil end
t [ 101 ] = 42
local sum = 0
for _ , v in ipairs ( t ) do sum += v end
for _ , v in pairs ( t ) do sum += v end
return sum
end ) ( ) == 462 )
2021-10-30 04:25:12 +08:00
-- upvalues: recursive capture
assert ( ( function ( ) local function fact ( n ) return n < 1 and 1 or n * fact ( n - 1 ) end return fact ( 5 ) end ) ( ) == 120 )
-- basic compound assignment
assert ( ( function ( )
local a = 1
b = 2
local c = { value = 3 }
local d = { 4 }
local e = 3
local f = 2
a += 5
b -= a
c.value *= 3
d [ 1 ] /= b
e %= 2
f ^= 4
return concat ( a , b , c.value , d [ 1 ] , e , f )
end ) ( ) == " 6,-4,9,-1,1,16 " )
-- compound concat
assert ( ( function ( )
local a = ' a '
a .. = ' b '
a .. = ' c ' .. ' d '
a .. = ' e ' .. ' f ' .. a
return a
end ) ( ) == " abcdefabcd " )
-- compound assignment with side effects validates lhs is evaluated once
assert ( ( function ( )
local res = { 1 , 2 , 3 }
local count = 0
res [ ( function ( ) count += 1 return count end ) ( ) ] += 5
res [ ( function ( ) count += 1 return count end ) ( ) ] += 6
res [ ( function ( ) count += 1 return count end ) ( ) ] += 7
return table.concat ( res , ' , ' )
end ) ( ) == " 6,8,10 " )
2022-07-01 07:29:02 +08:00
-- typeof and type require an argument
assert ( pcall ( typeof ) == false )
assert ( pcall ( type ) == false )
2021-10-30 04:25:12 +08:00
-- typeof == type in absence of custom userdata
assert ( concat ( typeof ( 5 ) , typeof ( nil ) , typeof ( { } ) , typeof ( newproxy ( ) ) ) == " number,nil,table,userdata " )
2022-03-18 08:06:25 +08:00
-- type/typeof/newproxy interaction with metatables: __type doesn't work intentionally to avoid spoofing
assert ( ( function ( )
local ud = newproxy ( true )
getmetatable ( ud ) . __type = " number "
return concat ( type ( ud ) , typeof ( ud ) )
end ) ( ) == " userdata,userdata " )
2021-10-30 04:25:12 +08:00
testgetfenv ( ) -- DONT MOVE THIS LINE
2021-12-03 07:20:08 +08:00
return ' OK '