2017-09-21 00:10:26 -07:00
|
|
|
#..
|
|
|
|
This File contains rules for making rules and macros and some helper functions to make
|
|
|
|
that easier.
|
|
|
|
|
2017-09-24 20:20:27 -07:00
|
|
|
# Nil
|
|
|
|
lua code ".."
|
|
|
|
|nomsu:defmacro("nil", function(nomsu, vars) return "nil", nil end)
|
2017-09-21 00:10:26 -07:00
|
|
|
|
2017-09-24 20:20:27 -07:00
|
|
|
# Macros:
|
|
|
|
lua code ".."
|
2017-09-25 17:02:00 -07:00
|
|
|
|local function parse_as(nomsu, vars)
|
|
|
|
| if vars.shorthand.type ~= "Block" then
|
|
|
|
| nomsu:error("Expected shorthand to be Block, but got "..vars.shorthand.type)
|
|
|
|
| end
|
|
|
|
| if vars.longhand.type ~= "Block" then
|
|
|
|
| nomsu:error("Expected longhand to be Block, but got "..vars.longhand.type)
|
|
|
|
| end
|
2017-09-24 20:20:27 -07:00
|
|
|
| local template = vars.longhand
|
2017-09-25 17:02:00 -07:00
|
|
|
| local function parsing_as(nomsu, vars)
|
|
|
|
| local expanded = nomsu:replaced_vars(template, vars)
|
|
|
|
| local expr,statement = nomsu:tree_to_lua(expanded)
|
|
|
|
| return expr, statement
|
|
|
|
| end
|
|
|
|
| for _,call in ipairs(vars.shorthand.value) do
|
|
|
|
| nomsu:defmacro(call, parsing_as)
|
|
|
|
| end
|
|
|
|
|end
|
|
|
|
|nomsu:def("parse nomsu %shorthand as nomsu %longhand", parse_as)
|
|
|
|
parse nomsu \(parse %shorthand as %longhand) as nomsu \(parse nomsu \%shorthand as nomsu \%longhand)
|
|
|
|
parse (foo %x) as (baz %x)
|
|
|
|
|
|
|
|
lua code ".."
|
|
|
|
|nomsu:defmacro("lua expr %code", function(nomsu, vars)
|
|
|
|
| return nomsu:tree_to_value(vars.code, vars), nil
|
|
|
|
|end)
|
|
|
|
|
|
|
|
|
|
|
|
# Rule that lets you make new rules
|
|
|
|
lua code ".."
|
|
|
|
|nomsu:defmacro("rule %rule_def = %body", function(nomsu, vars)
|
|
|
|
| if vars.rule_def.type ~= "Block" then
|
|
|
|
| nomsu:error("Wrong type for rule definition, expected Block, but got "..vars.rule_def.type)
|
|
|
|
| end
|
|
|
|
| local thunk = nomsu:tree_to_lua({type="Thunk", value=vars.body, src=vars.body.src})
|
|
|
|
| local fn_name = "fn_"..nomsu:var_to_lua_identifier(nomsu:get_stub(vars.rule_def.value[1]))
|
|
|
|
| local lua = ([[
|
|
|
|
|do
|
|
|
|
| local %s = %s
|
|
|
|
| for _,alias in ipairs(%s) do
|
|
|
|
| nomsu:def(alias, %s)
|
|
|
|
| end
|
|
|
|
|end]]):format(fn_name, thunk, nomsu:repr(vars.rule_def.value), fn_name)
|
|
|
|
| return nil, lua
|
2017-09-24 20:20:27 -07:00
|
|
|
|end)
|
2017-09-25 17:02:00 -07:00
|
|
|
parse (rule %rule_name) as: lua expr "nomsu.defs[(nomsu:get_stub(\(%rule_name).value[1]))]"
|
|
|
|
|
|
|
|
|
|
|
|
# Macro helper functions
|
|
|
|
rule (%tree as value) =:
|
|
|
|
lua expr "nomsu:tree_to_value(vars.tree, vars)"
|
|
|
|
|
|
|
|
rule (%tree as lua) =:
|
|
|
|
lua expr "nomsu:tree_to_lua(vars.tree)"
|
2017-09-24 20:20:27 -07:00
|
|
|
|
2017-09-25 17:02:00 -07:00
|
|
|
|
|
|
|
parse (lua block %block) as:
|
2017-09-24 20:20:27 -07:00
|
|
|
lua code ".."
|
|
|
|
|do
|
|
|
|
| \(%block)
|
|
|
|
|end
|
2017-09-22 11:56:46 -07:00
|
|
|
|
2017-09-25 17:02:00 -07:00
|
|
|
parse (nomsu) as: lua expr "nomsu"
|
|
|
|
parse (nomsu's %key) as:
|
|
|
|
lua expr "nomsu[\(%key as lua)]"
|
|
|
|
parse (nomsu %method %args) as:
|
2017-09-21 00:10:26 -07:00
|
|
|
lua block ".."
|
2017-09-22 11:56:46 -07:00
|
|
|
|local args = {"nomsu"}
|
2017-09-21 21:11:13 -07:00
|
|
|
|for _,arg in ipairs(vars.args.value) do
|
2017-09-22 11:56:46 -07:00
|
|
|
| table.insert(args, nomsu:tree_to_lua(arg))
|
2017-09-21 00:10:26 -07:00
|
|
|
|end
|
2017-09-24 20:20:27 -07:00
|
|
|
|local method_name = nomsu:repr(nomsu:tree_to_value(vars.method, vars))
|
|
|
|
..with value ".."
|
|
|
|
|("nomsu[%s](%s)"):format(method_name, table.concat(args, ", "))
|
2017-09-25 17:02:00 -07:00
|
|
|
parse (repr %) as: nomsu "repr" [%]
|
2017-09-21 00:10:26 -07:00
|
|
|
|
|
|
|
# Get the source code for a function
|
2017-09-24 20:20:27 -07:00
|
|
|
rule (help %rule) =:
|
2017-09-21 00:10:26 -07:00
|
|
|
lua block ".."
|
2017-09-22 11:56:46 -07:00
|
|
|
|local fn_def = nomsu:get_fn_def(vars.rule)
|
2017-09-21 21:11:13 -07:00
|
|
|
|if not fn_def then
|
2017-09-22 11:56:46 -07:00
|
|
|
| nomsu:writeln("Rule not found: "..nomsu:repr(vars.rule))
|
2017-09-21 00:10:26 -07:00
|
|
|
|else
|
2017-09-25 17:02:00 -07:00
|
|
|
| nomsu:writeln("rule "..nomsu:repr(nomsu.utils.keys(fn_def.invocation))
|
2017-09-21 21:11:13 -07:00
|
|
|
| .." ="..(fn_def.src or ":\\n <unknown source code>"))
|
2017-09-21 00:10:26 -07:00
|
|
|
|end
|
|
|
|
|
|
|
|
# Compiler tools
|
2017-09-24 20:20:27 -07:00
|
|
|
rule (eval %code; run %code) =: nomsu "run" [%code]
|
|
|
|
rule (source code from tree %tree) =:
|
2017-09-21 00:10:26 -07:00
|
|
|
lua block ".."
|
|
|
|
|local _,_,leading_space = vars.tree.src:find("\\n(%s*)%S")
|
|
|
|
|if leading_space then
|
|
|
|
| local chunk1, chunk2 = vars.tree.src:match(":%s*([^\\n]*)(\\n.*)")
|
|
|
|
| chunk2 = chunk2:gsub("\\n"..leading_space, "\\n")
|
|
|
|
| return chunk1..chunk2.."\\n"
|
|
|
|
|else
|
|
|
|
| return vars.tree.src:match(":%s*(%S.*)").."\\n"
|
|
|
|
|end
|
|
|
|
|
2017-09-25 17:02:00 -07:00
|
|
|
parse (source code %body) as: source code from tree \%body
|
2017-09-21 00:10:26 -07:00
|
|
|
|
2017-09-25 17:02:00 -07:00
|
|
|
parse (parse tree %code) as: repr (nomsu "tree_to_str" [\%code])
|
2017-09-21 00:10:26 -07:00
|
|
|
|