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.
|
|
|
|
|
|
|
|
# Macros:
|
|
|
|
|
|
|
|
lua block ".."
|
2017-09-21 02:33:04 -07:00
|
|
|
|local function make_fn(wrap_in_block)
|
2017-09-22 11:56:46 -07:00
|
|
|
| return function(nomsu, vars, kind)
|
2017-09-21 21:11:13 -07:00
|
|
|
# Do a minimal amount of pre-processing (get the aliases and the source)
|
2017-09-22 11:56:46 -07:00
|
|
|
| local aliases = nomsu:repr(nomsu:get_aliases(vars.macro_def))
|
|
|
|
| local src = nomsu:repr(vars.user_macro.src)
|
|
|
|
| local user_macro = nomsu:tree_to_lua(vars.user_macro)
|
2017-09-21 00:10:26 -07:00
|
|
|
# Then produce a block of code that creates the macro at runtime
|
2017-09-21 02:33:04 -07:00
|
|
|
| local lua = [[
|
2017-09-22 11:56:46 -07:00
|
|
|
|nomsu:defmacro(%s, (function(nomsu, vars, kind)
|
|
|
|
| if kind == "Expression" then
|
|
|
|
| nomsu:error("Macro "..%s.." was defined to be a block, but is being used as an expression")
|
|
|
|
| end
|
|
|
|
| local user_macro = %s
|
|
|
|
| local lua = user_macro(nomsu, vars)
|
|
|
|
| %s
|
|
|
|
| return lua, true
|
|
|
|
|end), %s)]]
|
|
|
|
| lua = lua:format(aliases, nomsu:repr(aliases), user_macro,
|
2017-09-21 02:33:04 -07:00
|
|
|
| wrap_in_block and [[lua = "do\\n "..lua.."\\nend"]] or "", src)
|
|
|
|
| return lua, true
|
2017-09-21 00:10:26 -07:00
|
|
|
| end
|
2017-09-21 02:33:04 -07:00
|
|
|
|end
|
2017-09-22 11:56:46 -07:00
|
|
|
|nomsu:defmacro("macro statement %macro_def = %user_macro", make_fn(false), "see:lib/metaprogramming.nom")
|
|
|
|
|nomsu:defmacro("macro block %macro_def = %user_macro", make_fn(true), "see:lib/metaprogramming.nom")
|
2017-09-21 00:10:26 -07:00
|
|
|
|
2017-09-22 11:44:07 -07:00
|
|
|
macro statement [macro %macro_def = %user_macro] =:
|
2017-09-22 11:56:46 -07:00
|
|
|
".."|nomsu:defmacro(
|
|
|
|
| \lua expr "nomsu:get_aliases(vars.macro_def)"\,
|
|
|
|
| \lua expr "nomsu:tree_to_lua(vars.user_macro)"\,
|
|
|
|
| \lua expr "nomsu:repr(vars.user_macro.src)"\)
|
|
|
|
|
|
|
|
macro [nomsu] =: "nomsu"
|
|
|
|
macro [nomsu's %key] =: ".."|nomsu[\%key as lua\]
|
|
|
|
macro [nomsu %method %args] =:
|
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-22 11:56:46 -07:00
|
|
|
|return "nomsu["..nomsu:repr(nomsu:tree_to_value(vars.method, vars)).."]("..table.concat(args, ", ")..")"
|
|
|
|
macro [nomsu utils %method %args] =:
|
2017-09-21 00:10:26 -07:00
|
|
|
lua block ".."
|
|
|
|
|local args = {}
|
|
|
|
|for i,arg in ipairs(vars.args.value) do
|
2017-09-22 11:56:46 -07:00
|
|
|
| args[i] = nomsu:tree_to_lua(arg)
|
2017-09-21 00:10:26 -07:00
|
|
|
|end
|
2017-09-22 11:56:46 -07:00
|
|
|
|return "nomsu.utils["..nomsu:repr(nomsu:tree_to_value(vars.method, vars)).."]("..table.concat(args, ", ")..")"
|
2017-09-21 00:10:26 -07:00
|
|
|
|
|
|
|
# Macro that lets you make new rules
|
|
|
|
#..
|
|
|
|
This is a macro instead of a rule because it needs to pre-empt processing the list of
|
2017-09-21 21:11:13 -07:00
|
|
|
function calls and convert it into a list of strings (rather than call a function that
|
2017-09-21 00:10:26 -07:00
|
|
|
is currently in the middle of being defined). Being a macro also allows us to snatch
|
|
|
|
the source code and store that
|
2017-09-22 11:44:07 -07:00
|
|
|
macro statement [rule %rule_def = %body] =: ".."
|
2017-09-22 11:56:46 -07:00
|
|
|
|nomsu:def(
|
|
|
|
| \nomsu "repr" [nomsu "get_aliases" [%rule_def]]\,
|
|
|
|
| \nomsu "tree_to_lua" [%body]\,
|
|
|
|
| \nomsu "repr" [lua expr "vars.body.src"]\)
|
2017-09-21 21:11:13 -07:00
|
|
|
|
|
|
|
rule [fart]=: lua block "print('poot')"
|
|
|
|
|
|
|
|
macro block [alias %aliases = %aliased] =:
|
|
|
|
lua block ".."
|
2017-09-22 11:56:46 -07:00
|
|
|
|local aliases = nomsu:get_aliases(vars.aliases)
|
|
|
|
|aliases = nomsu:repr(aliases)
|
2017-09-21 21:11:13 -07:00
|
|
|
|if vars.aliased.type ~= "Thunk" then
|
2017-09-22 11:56:46 -07:00
|
|
|
| nomsu:error("Right hand side of alias % = % should be a Thunk, but got "..vars.aliased.type
|
2017-09-21 21:11:13 -07:00
|
|
|
| ..". Maybe you used = instead of =: by mistake?")
|
|
|
|
|end
|
2017-09-22 11:56:46 -07:00
|
|
|
|local aliased = next(nomsu:get_aliases(vars.aliased.value))
|
|
|
|
|aliased = nomsu:repr(aliased)
|
2017-09-21 21:11:13 -07:00
|
|
|
|local lua = ([[
|
2017-09-22 11:56:46 -07:00
|
|
|
|nomsu:add_aliases(%s, nomsu.defs[%s])
|
2017-09-21 21:11:13 -07:00
|
|
|
|]]):format(aliases, aliased)
|
|
|
|
|return lua
|
2017-09-21 00:10:26 -07:00
|
|
|
|
|
|
|
# Get the source code for a function
|
2017-09-21 21:11:13 -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-22 11:56:46 -07:00
|
|
|
| nomsu:writeln("rule "..nomsu:repr(nomsu.utils.keys(fn_def.aliases))
|
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
|
|
|
|
|
|
|
|
|
|
|
|
# Macro helper functions
|
|
|
|
rule [%tree as value] =:
|
2017-09-22 11:56:46 -07:00
|
|
|
lua expr "nomsu:tree_to_value(vars.tree, vars)"
|
2017-09-21 00:10:26 -07:00
|
|
|
|
2017-09-21 02:33:04 -07:00
|
|
|
rule [%tree as lua] =:
|
2017-09-22 11:56:46 -07:00
|
|
|
lua expr "nomsu:tree_to_lua(vars.tree)"
|
2017-09-21 00:10:26 -07:00
|
|
|
|
2017-09-21 13:30:59 -07:00
|
|
|
rule [test, foobar] =: lua expr "print('yup')"
|
|
|
|
|
2017-09-21 00:10:26 -07:00
|
|
|
# Compiler tools
|
2017-09-22 11:56:46 -07:00
|
|
|
rule [eval %code, run %code] =: nomsu "run" [%code]
|
2017-09-21 00:10:26 -07:00
|
|
|
|
|
|
|
rule [source code from tree %tree] =:
|
|
|
|
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
|
|
|
|
|
|
|
|
macro [source code %body] =:
|
2017-09-22 11:56:46 -07:00
|
|
|
nomsu "repr" [nomsu "call" ["source code from tree %", %body]]
|
2017-09-21 00:10:26 -07:00
|
|
|
|
|
|
|
macro [parse tree %code] =:
|
2017-09-22 11:56:46 -07:00
|
|
|
nomsu "repr" [nomsu "stringify_tree" [lua expr "vars.code.value"]]
|
2017-09-21 00:10:26 -07:00
|
|
|
|