2018-05-15 18:55:55 -07:00
|
|
|
#
|
2018-01-11 18:51:21 -08:00
|
|
|
This file contains definitions of operators like "+" and "and".
|
|
|
|
|
2018-02-02 15:48:28 -08:00
|
|
|
use "core/metaprogramming.nom"
|
2017-09-21 00:10:26 -07:00
|
|
|
|
2018-04-25 16:30:49 -07:00
|
|
|
# Indexing
|
|
|
|
immediately
|
2018-05-15 18:55:55 -07:00
|
|
|
# NOTE!!! It's critical that there are spaces around %key if it's a string,
|
2018-01-18 01:49:13 -08:00
|
|
|
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-04-06 16:53:30 -07:00
|
|
|
parse [..]
|
2018-01-23 19:22:20 -08:00
|
|
|
%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,
|
2018-04-06 16:53:30 -07:00
|
|
|
..as: %obj.%key
|
2017-09-21 00:10:26 -07:00
|
|
|
|
2018-01-11 18:51:21 -08:00
|
|
|
# Comparison Operators
|
2018-04-25 16:30:49 -07:00
|
|
|
immediately
|
2018-04-17 14:18:23 -07:00
|
|
|
compile [%x < %y] to: Lua value "(\(%x as lua expr) < \(%y as lua expr))"
|
|
|
|
compile [%x > %y] to: Lua value "(\(%x as lua expr) > \(%y as lua expr))"
|
|
|
|
compile [%x <= %y] to: Lua value "(\(%x as lua expr) <= \(%y as lua expr))"
|
|
|
|
compile [%x >= %y] to: Lua value "(\(%x as lua expr) >= \(%y as lua expr))"
|
2018-01-26 20:20:12 -08:00
|
|
|
# TODO: optimize case of [%x,%y] = [1,2]
|
2018-04-25 16:30:49 -07: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};
|
2018-04-25 16:04:46 -07:00
|
|
|
local a_lua, b_lua = \%a:as_lua(nomsu), \%b:as_lua(nomsu);
|
2018-01-11 18:51:21 -08:00
|
|
|
if safe[\%a.type] or safe[\%b.type] then
|
2018-05-16 18:12:56 -07:00
|
|
|
return Lua.Value(nil, "(", a_lua, " == ", b_lua, ")");
|
2018-01-11 01:03:52 -08:00
|
|
|
else
|
2018-05-16 18:12:56 -07:00
|
|
|
return Lua.Value(nil, "utils.equivalent(", a_lua, ", ", b_lua, ")");
|
2018-01-11 01:03:52 -08:00
|
|
|
end
|
2018-04-25 16:30:49 -07: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};
|
2018-04-25 16:04:46 -07:00
|
|
|
local a_lua, b_lua = \%a:as_lua(nomsu), \%b:as_lua(nomsu);
|
2018-01-11 18:51:21 -08:00
|
|
|
if safe[\%a.type] or safe[\%b.type] then
|
2018-05-16 18:12:56 -07:00
|
|
|
return Lua.Value(nil, "(", a_lua, " ~= ", b_lua, ")");
|
2018-01-11 18:51:21 -08:00
|
|
|
else
|
2018-05-16 18:12:56 -07:00
|
|
|
return Lua.Value(nil, "(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-04-17 14:18:23 -07:00
|
|
|
compile [%'s id, id of %] to: Lua value "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-04-25 16:30:49 -07:00
|
|
|
immediately
|
|
|
|
compile [%var <- %value] to
|
2018-04-25 16:04:46 -07:00
|
|
|
lua> "local \%var_lua = \%var:as_lua(nomsu);"
|
2018-05-16 18:12:56 -07:00
|
|
|
assume %var_lua.is_value or barf "Invalid target for assignment: \%var"
|
2018-04-25 16:04:46 -07:00
|
|
|
lua> "local \%value_lua = \%value:as_lua(nomsu);"
|
2018-05-16 18:12:56 -07:00
|
|
|
assume %value_lua.is_value or barf "Invalid value for assignment: \%value"
|
2018-04-13 15:29:16 -07:00
|
|
|
lua> ".."
|
2018-05-16 18:12:56 -07:00
|
|
|
local lua = Lua(nil, \%var_lua, ' = ', \%value_lua, ';');
|
2018-04-13 15:29:16 -07:00
|
|
|
if \%var.type == 'Var' then
|
2018-05-04 13:49:09 -07:00
|
|
|
lua:add_free_vars({\%var});
|
2018-04-13 15:29:16 -07:00
|
|
|
end
|
2018-04-18 17:41:40 -07:00
|
|
|
return lua;
|
2018-01-23 19:42:01 -08:00
|
|
|
|
2018-04-25 16:30:49 -07:00
|
|
|
immediately
|
2018-01-26 20:20:12 -08:00
|
|
|
# Simultaneous mutli-assignments like: x,y,z = 1,x,3;
|
2018-04-25 16:30:49 -07:00
|
|
|
compile [<- %assignments] to
|
2018-01-26 20:20:12 -08:00
|
|
|
assume ((%assignments' "type") is "Dict") or barf ".."
|
2018-05-16 18:12:56 -07:00
|
|
|
Expected a Dict for the assignments part of '<- %' statement, not \%assignments
|
2018-01-26 20:20:12 -08:00
|
|
|
lua> ".."
|
2018-05-16 18:12:56 -07:00
|
|
|
local lhs, rhs = Lua(), Lua();
|
|
|
|
for i, item in ipairs(\%assignments) do
|
|
|
|
local target, value = item[1], item[2];
|
2018-04-25 16:04:46 -07:00
|
|
|
local target_lua = target:as_lua(nomsu);
|
2018-04-13 15:29:16 -07:00
|
|
|
if not target_lua.is_value then error("Invalid target for assignment: "..target:get_src()); end
|
2018-04-25 16:04:46 -07:00
|
|
|
local value_lua = value:as_lua(nomsu);
|
2018-04-13 15:29:16 -07:00
|
|
|
if not value_lua.is_value then error("Invalid value for assignment: "..value:get_src()); end
|
2018-01-26 20:20:12 -08:00
|
|
|
if target.type == "Var" then
|
2018-05-04 13:49:09 -07:00
|
|
|
lhs:add_free_vars({target});
|
2018-01-26 20:20:12 -08:00
|
|
|
end
|
2018-04-13 15:29:16 -07:00
|
|
|
if i > 1 then
|
|
|
|
lhs:append(", ");
|
|
|
|
rhs:append(", ");
|
|
|
|
end
|
|
|
|
lhs:append(target_lua);
|
|
|
|
rhs:append(value_lua);
|
2018-01-26 20:20:12 -08:00
|
|
|
end
|
2018-05-16 18:12:56 -07:00
|
|
|
return Lua(nil, lhs, " = ", rhs, ";");
|
2018-01-26 20:20:12 -08:00
|
|
|
|
2018-04-25 16:30:49 -07:00
|
|
|
immediately
|
2018-05-09 20:35:29 -07:00
|
|
|
compile [external %var <- %value] to
|
2018-01-25 17:34:49 -08:00
|
|
|
%var_lua <- (%var as lua)
|
2018-05-16 18:12:56 -07:00
|
|
|
assume %var_lua.is_value or barf "Invalid target for assignment: \%var"
|
2018-01-25 17:34:49 -08:00
|
|
|
%value_lua <- (%value as lua)
|
2018-05-16 18:12:56 -07:00
|
|
|
assume %value_lua.is_value or barf "Invalid value for assignment: \%value"
|
2018-04-13 15:29:16 -07:00
|
|
|
return: Lua "\(%var_lua) = \(%value_lua);"
|
2018-01-23 19:42:01 -08:00
|
|
|
|
2018-05-09 20:35:29 -07:00
|
|
|
compile [with external %externs %body] to
|
2018-04-19 17:23:44 -07:00
|
|
|
%body_lua <- (%body as lua statements)
|
2018-05-16 18:12:56 -07:00
|
|
|
lua> "\%body_lua:remove_free_vars(\(%externs));"
|
2018-04-13 15:29:16 -07:00
|
|
|
return %body_lua
|
2018-01-26 20:20:12 -08:00
|
|
|
|
2018-04-25 16:30:49 -07:00
|
|
|
compile [with %assignments %body] to
|
2018-04-19 17:23:44 -07:00
|
|
|
%lua <- (%body as lua statements)
|
|
|
|
lua> ".."
|
2018-05-16 18:12:56 -07:00
|
|
|
local lhs, rhs = Lua(), Lua();
|
2018-04-19 17:23:44 -07:00
|
|
|
local vars = {};
|
2018-05-16 18:12:56 -07:00
|
|
|
for i, item in ipairs(\%assignments) do
|
|
|
|
local target, value = item[1], item[2];
|
2018-04-19 17:23:44 -07:00
|
|
|
if not target.type == "Var" then
|
2018-05-16 18:12:56 -07:00
|
|
|
error("Invalid target for 'with' assignment: "..tostring(target));
|
2018-04-19 17:23:44 -07:00
|
|
|
end
|
2018-04-25 16:04:46 -07:00
|
|
|
local target_lua = target:as_lua(nomsu);
|
|
|
|
local value_lua = value:as_lua(nomsu);
|
2018-04-19 17:23:44 -07:00
|
|
|
if not value_lua.is_value then
|
2018-05-16 18:12:56 -07:00
|
|
|
error("Invalid value for assignment: "..tostring(value));
|
2018-04-19 17:23:44 -07:00
|
|
|
end
|
|
|
|
if target.type == "Var" then
|
2018-05-04 13:49:09 -07:00
|
|
|
lhs:add_free_vars({target});
|
2018-04-19 17:23:44 -07:00
|
|
|
end
|
|
|
|
if i > 1 then
|
|
|
|
lhs:append(", ");
|
|
|
|
rhs:append(", ");
|
|
|
|
end
|
|
|
|
lhs:append(target_lua);
|
|
|
|
rhs:append(value_lua);
|
|
|
|
vars[i] = tostring(target_lua);
|
|
|
|
end
|
|
|
|
\%lua:remove_free_vars(vars);
|
2018-04-27 16:45:11 -07:00
|
|
|
\%lua:prepend("local ", lhs, " = ", rhs, ";\n");
|
2018-04-19 17:23:44 -07:00
|
|
|
return
|
|
|
|
Lua ".."
|
|
|
|
do
|
|
|
|
\%lua
|
|
|
|
end -- 'with' block
|
2018-01-25 17:34:49 -08:00
|
|
|
|
2018-04-25 16:30:49 -07:00
|
|
|
immediately
|
2018-01-23 19:22:20 -08:00
|
|
|
# Math Operators
|
2018-04-17 14:18:23 -07:00
|
|
|
compile [%x wrapped around %y, %x mod %y] to: Lua value "(\(%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-04-17 14:18:23 -07:00
|
|
|
compile [%x and %y] to: Lua value "(\(%x as lua expr) and \(%y as lua expr))"
|
|
|
|
compile [%x or %y] to: Lua value "(\(%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-04-17 14:18:23 -07:00
|
|
|
compile [%a OR %b, %a | %b] to: Lua value "bit32.bor(\(%a as lua expr), \(%b as lua expr))"
|
|
|
|
compile [%a XOR %b] to: Lua value "bit32.bxor(\(%a as lua expr), \(%b as lua expr))"
|
|
|
|
compile [%a AND %b, %a & %b] to: Lua value "bit32.band(\(%a as lua expr), \(%b as lua expr))"
|
|
|
|
compile [NOT %, ~ %] to: Lua value "bit32.bnot(\(% as lua expr))"
|
|
|
|
compile [%x LSHIFT %shift, %x << %shift] to: Lua value "bit32.lshift(\(%x as lua expr), \(%shift as lua expr))"
|
|
|
|
compile [%x RSHIFT %shift, %x >>> %shift] to: Lua value "bit32.rshift(\(%x as lua expr), \(%shift as lua expr))"
|
|
|
|
compile [%x ARSHIFT %shift, %x >> %shift] to: Lua value "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-04-17 14:18:23 -07:00
|
|
|
compile [- %] to: Lua value "(- \(% as lua expr))"
|
|
|
|
compile [not %] to: Lua value "(not \(% as lua expr))"
|
2018-05-10 22:47:03 -07:00
|
|
|
# Using custom "len()" instead of Lua's "#" operator for compatibility with luajit.
|
|
|
|
compile [length of %list] to: Lua value "len(\(%list as lua expr))"
|
2017-09-21 00:10:26 -07:00
|
|
|
|
2018-01-23 19:22:20 -08:00
|
|
|
# Update operators
|
2018-04-25 16:30:49 -07:00
|
|
|
immediately
|
2018-01-26 20:20:12 -08:00
|
|
|
parse [%var + <- %, %var +<- %] as: %var <- (%var + %)
|
|
|
|
parse [%var - <- %, %var -<- %] as: %var <- (%var - %)
|
|
|
|
parse [%var * <- %, %var *<- %] as: %var <- (%var * %)
|
|
|
|
parse [%var / <- %, %var /<- %] as: %var <- (%var / %)
|
|
|
|
parse [%var ^ <- %, %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 %)
|