2018-01-11 18:51:21 -08:00
|
|
|
#..
|
|
|
|
This file contains definitions of operators like "+" and "and".
|
|
|
|
|
2018-01-11 03:35:35 -08:00
|
|
|
use "lib/metaprogramming.nom"
|
2017-09-21 00:10:26 -07:00
|
|
|
|
2018-01-11 18:51:21 -08:00
|
|
|
# Indexing:
|
2018-01-19 17:29:44 -08:00
|
|
|
immediately
|
2018-01-18 01:49:13 -08:00
|
|
|
#.. NOTE!!! It's critical that there are spaces around %key if it's a string,
|
|
|
|
otherwise, Lua will get confused and interpret %obj[[[foo]]] as %obj("[foo]")
|
|
|
|
instead of %obj[ "foo" ].
|
|
|
|
It's also critical to have parens around %obj, otherwise Lua is too dumb to
|
|
|
|
realize that {x=1}["x"] is the same as ({x=1})["x"] or that
|
|
|
|
{x=1}.x is the same as ({x=1}).x
|
2018-01-23 19:22:20 -08:00
|
|
|
compile [..]
|
|
|
|
%obj' %key, %obj's %key, %key in %obj, %key'th in %obj, %key of %obj,
|
|
|
|
%key st in %obj, %key nd in %obj, %key rd in %obj, %key th in %obj,
|
|
|
|
..to
|
2018-01-18 01:49:13 -08:00
|
|
|
lua> ".."
|
2018-01-25 17:34:49 -08:00
|
|
|
local obj_lua = \(%obj as lua expr);
|
2018-01-18 01:49:13 -08:00
|
|
|
if not obj_lua:sub(-1,-1):match("[a-zA-Z)]") then
|
|
|
|
obj_lua = "("..obj_lua..")";
|
|
|
|
end
|
2018-01-25 17:34:49 -08:00
|
|
|
local key_lua = \(%key as lua expr);
|
2018-01-18 01:49:13 -08:00
|
|
|
local key_attr = (key_lua:match("'([a-zA-Z][a-zA-Z0-9]*)'")
|
|
|
|
or key_lua:match('"([a-zA-Z][a-zA-Z0-9]*)"'));
|
|
|
|
if key_attr then
|
2018-01-25 17:34:49 -08:00
|
|
|
return {expr=obj_lua.."."..key_attr};
|
2018-01-18 01:49:13 -08:00
|
|
|
elseif key_lua:sub(1,1) == "[" then
|
|
|
|
key_lua = " "..key_lua.." ";
|
|
|
|
end
|
2018-01-25 17:34:49 -08:00
|
|
|
return {expr=obj_lua.."["..key_lua.."]"};
|
2017-09-21 00:10:26 -07:00
|
|
|
|
2018-01-11 18:51:21 -08:00
|
|
|
# Comparison Operators
|
2018-01-19 17:29:44 -08:00
|
|
|
immediately
|
2018-01-25 17:34:49 -08:00
|
|
|
compile [%x < %y] to {expr:"(\(%x as lua expr) < \(%y as lua expr))"}
|
|
|
|
compile [%x > %y] to {expr:"(\(%x as lua expr) > \(%y as lua expr))"}
|
|
|
|
compile [%x <= %y] to {expr:"(\(%x as lua expr) <= \(%y as lua expr))"}
|
|
|
|
compile [%x >= %y] to {expr:"(\(%x as lua expr) >= \(%y as lua expr))"}
|
2018-01-19 17:29:44 -08:00
|
|
|
compile [%a is %b, %a = %b, %a == %b] to
|
2018-01-11 18:51:21 -08:00
|
|
|
lua> ".."
|
|
|
|
local safe = {Text=true, Number=true};
|
|
|
|
local a_lua, b_lua = nomsu:tree_to_lua(\%a).expr, nomsu:tree_to_lua(\%b).expr;
|
|
|
|
if safe[\%a.type] or safe[\%b.type] then
|
2018-01-25 17:34:49 -08:00
|
|
|
return {expr="("..a_lua.." == "..b_lua..")"};
|
2018-01-11 01:03:52 -08:00
|
|
|
else
|
2018-01-25 17:34:49 -08:00
|
|
|
return {expr="utils.equivalent("..a_lua..", "..b_lua..")"};
|
2018-01-11 01:03:52 -08:00
|
|
|
end
|
2018-01-19 17:29:44 -08:00
|
|
|
compile [%a isn't %b, %a is not %b, %a not= %b, %a != %b] to
|
2018-01-11 18:51:21 -08:00
|
|
|
lua> ".."
|
|
|
|
local safe = {Text=true, Number=true};
|
|
|
|
local a_lua, b_lua = nomsu:tree_to_lua(\%a).expr, nomsu:tree_to_lua(\%b).expr;
|
|
|
|
if safe[\%a.type] or safe[\%b.type] then
|
2018-01-25 17:34:49 -08:00
|
|
|
return {expr="("..a_lua.." ~= "..b_lua..")"};
|
2018-01-11 18:51:21 -08:00
|
|
|
else
|
2018-01-25 17:34:49 -08:00
|
|
|
return {expr="(not utils.equivalent("..a_lua..", "..b_lua.."))"};
|
2018-01-11 18:51:21 -08:00
|
|
|
end
|
|
|
|
# For strict identity checking, use (%x's id) is (%y's id)
|
2018-01-25 17:34:49 -08:00
|
|
|
compile [%'s id, id of %] to {expr:"nomsu.ids[\(% as lua expr)]"}
|
2018-01-07 16:51:29 -08:00
|
|
|
|
2018-01-23 19:42:01 -08:00
|
|
|
# Variable assignment operator
|
2018-01-19 17:29:44 -08:00
|
|
|
immediately
|
2018-01-25 17:34:49 -08:00
|
|
|
compile [%var <- %value] to
|
|
|
|
lua> "local \%var_lua = nomsu:tree_to_lua(\%var);"
|
|
|
|
assume (%var_lua's "expr") or barf "Invalid target for assignment: \(%var's source code)"
|
|
|
|
lua> "local \%value_lua = nomsu:tree_to_lua(\%value);"
|
|
|
|
assume (%value_lua's "expr") or barf "Invalid value for assignment: \(%value's source code)"
|
|
|
|
return {..}
|
|
|
|
statements:"\(%var_lua's "expr") = \(%value_lua's "expr");"
|
|
|
|
locals: =lua "(\%var.type == 'Var' and {\%var_lua.expr} or nil)"
|
2018-01-23 19:42:01 -08:00
|
|
|
|
2018-01-25 17:34:49 -08:00
|
|
|
immediately
|
|
|
|
compile [export %var <- %value] to
|
|
|
|
%var_lua <- (%var as lua)
|
|
|
|
assume (%var_lua's "expr") or barf "Invalid target for assignment: \(%var's source code)"
|
|
|
|
%value_lua <- (%value as lua)
|
|
|
|
assume (%value_lua's "expr") or barf "Invalid value for assignment: \(%value's source code)"
|
|
|
|
return {statements:"\(%var_lua's "expr") = \(%value_lua's "expr");"}
|
2018-01-23 19:42:01 -08:00
|
|
|
|
2018-01-25 17:34:49 -08:00
|
|
|
compile [with %assignments %body] to
|
|
|
|
%body_lua <- (%body as lua)
|
|
|
|
%locals <- []
|
|
|
|
%declarations <- []
|
|
|
|
%leftover_locals <- (=lua "{unpack(\%body_lua.locals or {})}")
|
|
|
|
assume ((%assignments' "type") is "List") or barf ".."
|
|
|
|
Expected a List for the assignments part of 'with' statement, not \(%assignments' source code)
|
|
|
|
lua> ".."
|
2018-01-23 19:22:20 -08:00
|
|
|
for i, item in ipairs(\%assignments.value) do
|
|
|
|
if item.type == "Var" then
|
|
|
|
local var = nomsu:tree_to_lua(item).expr;
|
2018-01-25 17:34:49 -08:00
|
|
|
utils.remove_from_list(\%leftover_locals, var);
|
|
|
|
table.insert(\%locals, var);
|
2018-01-23 19:22:20 -08:00
|
|
|
else
|
2018-01-25 17:34:49 -08:00
|
|
|
if not (item.type == "FunctionCall" and #item.value == 3 and item.value[2].value == "<-") then
|
|
|
|
error("'with' statement expects entries of the form: '%var <- %value', not: "..item:get_src());
|
|
|
|
end
|
2018-01-23 19:22:20 -08:00
|
|
|
local target, value = item.value[1], item.value[3];
|
2018-01-25 17:34:49 -08:00
|
|
|
local target_lua = nomsu:tree_to_lua(target);
|
|
|
|
if not target_lua.expr then error("Invalid target for assignment: "..target:get_src()); end
|
|
|
|
local value_lua = nomsu:tree_to_lua(value);
|
|
|
|
if not value_lua.expr then error("Invalid value for assignment: "..value:get_src()); end
|
2018-01-23 19:22:20 -08:00
|
|
|
if target.type == "Var" then
|
2018-01-25 17:34:49 -08:00
|
|
|
utils.remove_from_list(\%leftover_locals, target_lua.expr);
|
2018-01-23 19:22:20 -08:00
|
|
|
end
|
2018-01-25 17:34:49 -08:00
|
|
|
table.insert(\%declarations, (("local %s = %s;\\n "):format(
|
|
|
|
target_lua.expr, value_lua.expr)));
|
2018-01-23 19:22:20 -08:00
|
|
|
end
|
2018-01-11 18:51:21 -08:00
|
|
|
end
|
2018-01-25 17:34:49 -08:00
|
|
|
local locals_code = "";
|
|
|
|
if #\%locals > 0 then
|
|
|
|
locals_code = "\\nlocal "..table.concat(\%locals, ", ")..";";
|
|
|
|
end
|
|
|
|
local declaration_code = "";
|
|
|
|
if #\%declarations > 0 then
|
|
|
|
declaration_code = "\\n"..table.concat(\%declarations, "\\n");
|
2018-01-23 19:42:01 -08:00
|
|
|
end
|
2018-01-25 17:34:49 -08:00
|
|
|
return {locals=\%leftover_locals, statements=([[
|
|
|
|
do%s%s
|
|
|
|
%s
|
|
|
|
end]]):format(locals_code, declaration_code, \%body_lua.statements or (\%body_lua.expr..";"))};
|
|
|
|
|
|
|
|
compile [exporting %exported %body] to
|
|
|
|
%body_lua <- (%body as lua)
|
|
|
|
%leftover_locals <- (=lua "{unpack(\%body_lua.locals or {})}")
|
|
|
|
assume ((%exported's "type") = "List") or barf ".."
|
|
|
|
Expected a List for the export part of 'exporting' statement, not \(%exported's source code)
|
|
|
|
lua> ".."
|
2018-01-23 19:42:01 -08:00
|
|
|
for i, item in ipairs(\%exported.value) do
|
2018-01-25 17:34:49 -08:00
|
|
|
if item.type ~= "Var" then
|
|
|
|
error("'exporting' statement expects Vars, not: "..item:get_src());
|
|
|
|
end
|
2018-01-23 19:42:01 -08:00
|
|
|
local var = nomsu:tree_to_lua(item).expr;
|
2018-01-25 17:34:49 -08:00
|
|
|
utils.remove_from_list(leftover_locals, var);
|
2018-01-23 19:42:01 -08:00
|
|
|
end
|
2018-01-25 17:34:49 -08:00
|
|
|
return {locals:%leftover_locals, statements:=lua "\%body_lua.statements or (\%body_lua.expr..';')"}
|
2018-01-23 19:42:01 -08:00
|
|
|
|
2018-01-23 19:22:20 -08:00
|
|
|
immediately
|
|
|
|
# Math Operators
|
2018-01-25 17:34:49 -08:00
|
|
|
compile [%x + %y] to {expr:"(\(%x as lua expr) + \(%y as lua expr))"}
|
|
|
|
compile [%x - %y] to {expr:"(\(%x as lua expr) - \(%y as lua expr))"}
|
|
|
|
compile [%x * %y] to {expr:"(\(%x as lua expr) * \(%y as lua expr))"}
|
|
|
|
compile [%x / %y] to {expr:"(\(%x as lua expr) / \(%y as lua expr))"}
|
|
|
|
compile [%x ^ %y] to {expr:"(\(%x as lua expr) ^ \(%y as lua expr))"}
|
|
|
|
compile [%x wrapped around %y, %x mod %y] to {expr:"(\(%x as lua expr) % \(%y as lua expr))"}
|
2017-09-21 00:10:26 -07:00
|
|
|
|
2018-01-23 19:22:20 -08:00
|
|
|
# 3-part chained comparisons
|
|
|
|
# (uses a lambda to avoid re-evaluating middle value, while still being an expression)
|
|
|
|
parse [%x < %y < %z] as: =lua "(function(x,y,z) return x < y and y < z; end)(\%x,\%y,\%z)"
|
|
|
|
parse [%x <= %y < %z] as: =lua "(function(x,y,z) return x <= y and y < z; end)(\%x,\%y,\%z)"
|
|
|
|
parse [%x < %y <= %z] as: =lua "(function(x,y,z) return x < y and y <= z; end)(\%x,\%y,\%z)"
|
|
|
|
parse [%x <= %y <= %z] as: =lua "(function(x,y,z) return x <= y and y <= z; end)(\%x,\%y,\%z)"
|
|
|
|
parse [%x > %y > %z] as: =lua "(function(x,y,z) return x > y and y > z; end)(\%x,\%y,\%z)"
|
|
|
|
parse [%x >= %y > %z] as: =lua "(function(x,y,z) return x >= y and y > z; end)(\%x,\%y,\%z)"
|
|
|
|
parse [%x > %y >= %z] as: =lua "(function(x,y,z) return x > y and y >= z; end)(\%x,\%y,\%z)"
|
|
|
|
parse [%x >= %y >= %z] as: =lua "(function(x,y,z) return x >= y and y >= z; end)(\%x,\%y,\%z)"
|
|
|
|
# TODO: optimize for common case where x,y,z are all either variables or number literals
|
2018-01-07 18:03:37 -08:00
|
|
|
|
2018-01-23 19:22:20 -08:00
|
|
|
# Boolean Operators
|
2018-01-25 17:34:49 -08:00
|
|
|
compile [%x and %y] to {expr:"(\(%x as lua expr) and \(%y as lua expr))"}
|
|
|
|
compile [%x or %y] to {expr:"(\(%x as lua expr) or \(%y as lua expr))"}
|
2018-01-07 18:03:37 -08:00
|
|
|
|
2018-01-23 19:22:20 -08:00
|
|
|
# Bitwise Operators
|
2018-01-25 17:34:49 -08:00
|
|
|
compile [%a OR %b, %a | %b] to {expr:"bit32.bor(\(%a as lua expr), \(%b as lua expr))"}
|
|
|
|
compile [%a XOR %b] to {expr:"bit32.bxor(\(%a as lua expr), \(%b as lua expr))"}
|
|
|
|
compile [%a AND %b, %a & %b] to {expr:"bit32.band(\(%a as lua expr), \(%b as lua expr))"}
|
|
|
|
compile [NOT %, ~ %] to {expr:"bit32.bnot(\(% as lua expr))"}
|
|
|
|
compile [%x LSHIFT %shift, %x << %shift] to {expr:"bit32.lshift(\(%x as lua expr), \(%shift as lua expr))"}
|
|
|
|
compile [%x RSHIFT %shift, %x >>> %shift] to {expr:"bit32.rshift(\(%x as lua expr), \(%shift as lua expr))"}
|
|
|
|
compile [%x ARSHIFT %shift, %x >> %shift] to {expr:"bit32.arshift(\(%x as lua expr), \(%shift as lua expr))"}
|
2018-01-23 19:22:20 -08:00
|
|
|
# TODO: implement OR, XOR, AND for multiple operands?
|
2017-12-13 16:29:15 -08:00
|
|
|
|
2018-01-23 19:22:20 -08:00
|
|
|
# Unary operators
|
2018-01-25 17:34:49 -08:00
|
|
|
compile [- %] to {expr:"(- \(% as lua expr))"}
|
|
|
|
compile [not %] to {expr:"(not \(% as lua expr))"}
|
2017-09-21 00:10:26 -07:00
|
|
|
|
2018-01-23 19:22:20 -08:00
|
|
|
# Update operators
|
|
|
|
immediately
|
|
|
|
parse [<- %var + %] as: %var <- (%var + %)
|
|
|
|
parse [<- %var - %] as: %var <- (%var - %)
|
|
|
|
parse [<- %var * %] as: %var <- (%var * %)
|
|
|
|
parse [<- %var / %] as: %var <- (%var / %)
|
|
|
|
parse [<- %var ^ %] as: %var <- (%var ^ %)
|
|
|
|
parse [<- %var and %] as: %var <- (%var and %)
|
|
|
|
parse [<- %var or %] as: %var <- (%var or %)
|
2018-01-25 17:34:49 -08:00
|
|
|
parse [wrap %var around %] as: %var <- (%var wrapped around %)
|