Lunatic t.: allow hex literals with >9 digits if the high ones are all 'F's.

(Or '0's, but that case is handled automatically.) The resulting generated
number is taken by looking at the 8 lower nibbles and interpreting them as
a signed 32-bit integer.
Also add some number parsing tests to test/nlcf_break.con.

git-svn-id: https://svn.eduke32.com/eduke32@3571 1a8010ca-5511-0410-912e-c29ae57300e0
This commit is contained in:
helixhorned 2013-03-17 18:11:53 +00:00
parent 15d9165199
commit d909173065
2 changed files with 43 additions and 9 deletions

View file

@ -462,21 +462,33 @@ function warnprintf(fmt, ...) -- local
end end
local function parse_number(pos, numstr) local function parse_number(pos, numstr)
-- <numstr> is a full number string, potentially prefixed with a minus sign.
local num = tonumber((numstr:gsub("h$", ""))) local num = tonumber((numstr:gsub("h$", "")))
-- local onum = num
local hex = numstr:match("0[xX]([^h]*)h?") -- get hex digits, if any
-- num==nil for Rio Lua, which doesn't handle large hex literals. -- num==nil for Rio Lua, which doesn't handle large hex literals.
if (num==nil or not (num >= -0x80000000 and num <= 0xffffffff)) then if (num==nil or not (num >= -0x80000000 and num <= 0xffffffff)) then
perrprintf(pos, "number %s out of the range of a 32-bit integer", numstr) -- number is <INT32_MIN or >UINT32_MAX or NaN
-- Be careful not to write bound checks like if (hex and #hex>8 and hex:sub(1,#hex-8):match("^[fF]$")) then
-- "if (i<LOWBOUND or i>HIGHBOUND) then error('...') end": -- Too many hex digits, but they're all Fs.
num = NaN pwarnprintf(pos, "number %s truncated to 32 bits", numstr)
elseif (num >= 0x80000000 and numstr:sub(1,2):lower()~="0x") then num = bit.band(num, 0xffffffff)
if (g_warn["number-conversion"]) then else
perrprintf(pos, "number %s out of the range of a 32-bit integer", numstr)
-- Be careful not to write bound checks like
-- "if (i<LOWBOUND or i>HIGHBOUND) then error('...') end":
num = NaN
end
elseif (num >= 0x80000000) then
if (not hex and g_warn["number-conversion"]) then
pwarnprintf(pos, "number %s converted to a negative one", numstr) pwarnprintf(pos, "number %s converted to a negative one", numstr)
end end
num = num-(0xffffffff+1) num = bit.band(num, 0xffffffff)
end end
-- printf("numstr:%s, num=%d (0x%s) '%s', resnum=%d (0x%s)",
-- numstr, onum, bit.tohex(onum), hex, num, bit.tohex(num))
return num return num
end end
@ -2869,7 +2881,7 @@ local Grammar = Pat{
t_number = POS() * lpeg.C( t_number = POS() * lpeg.C(
tok.maybe_minus * ((Pat("0x") + "0X") * Range("09", "af", "AF")^1 * Pat("h")^-1 tok.maybe_minus * ((Pat("0x") + "0X") * Range("09", "af", "AF")^1 * Pat("h")^-1
+ Range("09")^1) + Range("09")^1)
) / parse_number, ) / parse_number,
t_identifier_all = lpeg.C(t_broken_identifier + t_good_identifier), t_identifier_all = lpeg.C(t_broken_identifier + t_good_identifier),
@ -2935,7 +2947,7 @@ local Grammar = Pat{
+ Keyw("whilevarn") * sp1 * tok.rvar * sp1 * tok.define / on.while_begin + Keyw("whilevarn") * sp1 * tok.rvar * sp1 * tok.define / on.while_begin
* sp1 * Var("single_stmt") * (lpeg.Cc(nil) / on.while_end), * sp1 * Var("single_stmt") * (lpeg.Cc(nil) / on.while_end),
stmt_common = Keyw("{") * sp1 * "}" / "do end" -- space separation of commands in CON is for a reason! stmt_common = Keyw("{") * sp1 * "}" / "" -- space separation of commands in CON is for a reason!
+ Keyw("{") * sp1 * lpeg.Ct(stmt_list) * sp1 * "}" + Keyw("{") * sp1 * lpeg.Ct(stmt_list) * sp1 * "}"
+ con_inner_command + Var("switch_stmt") + lpeg.Ct(Var("while_stmt")), + con_inner_command + Var("switch_stmt") + lpeg.Ct(Var("while_stmt")),

View file

@ -1,3 +1,5 @@
// NOTE: This file is misnamed, "break" is local control flow, of course.
state teststate_break state teststate_break
ifvare 1 1 ifvare 1 1
{ {
@ -20,3 +22,23 @@ onevent EVENT_INIT
redefinequote 114 STILL LIVE OUTER redefinequote 114 STILL LIVE OUTER
echo 114 echo 114
endevent endevent
// Test number parsing
gamevar MINUS_ONE -1 0
gamevar INT32_MIN 0x80000000 0
gamevar INT32_MIN_ -0x80000000 0
gamevar INT32_MINh 0x80000000h 0
gamevar INT32_MIN_h -0x80000000h 0
gamevar INT32_MIN_P1 0x80000001 0
gamevar INT32_MIN_P1h 0x80000001h 0
//gamevar INT32_MAX_ -0x80000001 0 // error
gamevar INT32_MAX 0x7fffffff 0
gamevar INT32_MAXh 0x7fffffffh 0
// 9 digits? truncate it, but only because there's an F there.
gamevar SOMEHEX 0xFFEFC0001 0
// This one throws an error:
//gamevar BADHEX 0xBFEFC0001 0