Added support for serializing rules.

This commit is contained in:
Bruce Hill 2017-10-31 16:19:08 -07:00
parent c1ec00d5fc
commit 6ba79a8ff1
3 changed files with 43 additions and 7 deletions

View File

@ -9,8 +9,8 @@ lua> ".."
| local body = nomsu:typecheck(vars, "body", "Thunk"); | local body = nomsu:typecheck(vars, "body", "Thunk");
| return ([[ | return ([[
|nomsu:def(%s, %s, %s) |nomsu:def(%s, %s, %s)
|]]):format(nomsu:repr(signature), nomsu:tree_to_lua(body), nomsu:repr(body.src)), nil; |]]):format(nomsu:repr(signature), nomsu:tree_to_lua(body), nomsu:repr(("rule %s\\n..=%s"):format(vars.signature.src, vars.body.src))), nil;
|end), "<source can be found in lib/metaprogramming.nom>"); |end));
# Rule to make nomsu macros: # Rule to make nomsu macros:
rule [escaped parse %shorthand as %longhand] =: rule [escaped parse %shorthand as %longhand] =:
@ -33,7 +33,7 @@ rule [escaped parse %shorthand as %longhand] =:
| end | end
| return nil, table.concat(lua_bits, "\\n"); | return nil, table.concat(lua_bits, "\\n");
|end |end
|nomsu:defmacro(aliases, parsing_as, template.src); |nomsu:defmacro(aliases, parsing_as, ("parse %s\\n..as %s"):format(vars.shorthand.src, vars.longhand.src));
escaped parse \[parse %shorthand as %longhand] as \: escaped parse \%shorthand as \%longhand escaped parse \[parse %shorthand as %longhand] as \: escaped parse \%shorthand as \%longhand
# Rule to make lua macros: # Rule to make lua macros:
@ -42,14 +42,14 @@ rule [escaped compile %macro_def to %body] =:
|local aliases = nomsu:get_stubs(nomsu:typecheck(vars, "macro_def", "List").value); |local aliases = nomsu:get_stubs(nomsu:typecheck(vars, "macro_def", "List").value);
|local body = nomsu:typecheck(vars, "body", "Thunk"); |local body = nomsu:typecheck(vars, "body", "Thunk");
|local thunk = nomsu:tree_to_value(body); |local thunk = nomsu:tree_to_value(body);
|nomsu:defmacro(aliases, thunk, body.src); |nomsu:defmacro(aliases, thunk, ("compile %s\\n..to %s"):format(vars.macro_def.src, body.src));
rule [escaped compile %macro_def to code %body] =: rule [escaped compile %macro_def to code %body] =:
lua> ".." lua> ".."
|local aliases = nomsu:get_stubs(nomsu:typecheck(vars, "macro_def", "List").value); |local aliases = nomsu:get_stubs(nomsu:typecheck(vars, "macro_def", "List").value);
|local body = nomsu:typecheck(vars, "body", "Thunk"); |local body = nomsu:typecheck(vars, "body", "Thunk");
|local thunk = nomsu:tree_to_value(body); |local thunk = nomsu:tree_to_value(body);
|local thunk_wrapper = function(nomsu, vars) return nil, thunk(nomsu, vars); end |local thunk_wrapper = function(nomsu, vars) return nil, thunk(nomsu, vars); end
|nomsu:defmacro(aliases, thunk_wrapper, body.src); |nomsu:defmacro(aliases, thunk_wrapper, ("compile %s\\n..to code %s"):format(vars.macro_def.src, body.src));
parse [compile %macro_def to %body] as: escaped compile \%macro_def to \%body parse [compile %macro_def to %body] as: escaped compile \%macro_def to \%body
parse [compile %macro_def to code %body] as: escaped compile \%macro_def to code \%body parse [compile %macro_def to code %body] as: escaped compile \%macro_def to code \%body

View File

@ -242,6 +242,7 @@ do
assert(type(thunk) == 'function', "Bad thunk: " .. tostring(repr(thunk))) assert(type(thunk) == 'function', "Bad thunk: " .. tostring(repr(thunk)))
local canonical_args = nil local canonical_args = nil
local aliases = { } local aliases = { }
self.def_number = self.def_number + 1
for _index_0 = 1, #signature do for _index_0 = 1, #signature do
local _des_0 = signature[_index_0] local _des_0 = signature[_index_0]
local stub, arg_names local stub, arg_names
@ -269,13 +270,33 @@ do
arg_names = arg_names, arg_names = arg_names,
src = src, src = src,
is_macro = is_macro, is_macro = is_macro,
aliases = aliases aliases = aliases,
def_number = self.def_number
} }
end end
end, end,
defmacro = function(self, signature, thunk, src) defmacro = function(self, signature, thunk, src)
return self:def(signature, thunk, src, true) return self:def(signature, thunk, src, true)
end, end,
serialize_defs = function(self, after)
if after == nil then
after = 0
end
defs = { }
for _, def in pairs(self.defs) do
defs[def.def_number] = def.src or ""
end
local keys = utils.keys(defs)
table.sort(keys)
local buff = { }
for _index_0 = 1, #keys do
local i = keys[_index_0]
if i > after and #defs[i] > 0 then
insert(buff, defs[i])
end
end
return concat(buff, "\n")
end,
call = function(self, stub, line_no, ...) call = function(self, stub, line_no, ...)
local def = self.defs[stub] local def = self.defs[stub]
if def and def.is_macro and self.callstack[#self.callstack] ~= "#macro" then if def and def.is_macro and self.callstack[#self.callstack] ~= "#macro" then
@ -1063,6 +1084,7 @@ end)]]):format(concat(lua_bits, "\n"))
self.defs = setmetatable({ }, { self.defs = setmetatable({ }, {
__index = parent and parent.defs __index = parent and parent.defs
}) })
self.def_number = 0
self.callstack = { } self.callstack = { }
self.debug = false self.debug = false
self.utils = utils self.utils = utils

View File

@ -194,6 +194,7 @@ class NomsuCompiler
@write = (...)=> io.write(...) @write = (...)=> io.write(...)
@write_err = (...)=> io.stderr\write(...) @write_err = (...)=> io.stderr\write(...)
@defs = setmetatable({}, {__index:parent and parent.defs}) @defs = setmetatable({}, {__index:parent and parent.defs})
@def_number = 0
@callstack = {} @callstack = {}
@debug = false @debug = false
@utils = utils @utils = utils
@ -219,6 +220,7 @@ class NomsuCompiler
assert type(thunk) == 'function', "Bad thunk: #{repr thunk}" assert type(thunk) == 'function', "Bad thunk: #{repr thunk}"
canonical_args = nil canonical_args = nil
aliases = {} aliases = {}
@def_number += 1
for {stub, arg_names} in *signature for {stub, arg_names} in *signature
assert stub, "NO STUB FOUND: #{repr signature}" assert stub, "NO STUB FOUND: #{repr signature}"
if @debug then @writeln "#{colored.bright "DEFINING RULE:"} #{colored.underscore colored.magenta repr(stub)} #{colored.bright "WITH ARGS"} #{colored.dim repr(arg_names)}" if @debug then @writeln "#{colored.bright "DEFINING RULE:"} #{colored.underscore colored.magenta repr(stub)} #{colored.bright "WITH ARGS"} #{colored.dim repr(arg_names)}"
@ -228,10 +230,22 @@ class NomsuCompiler
assert utils.equivalent(utils.set(arg_names), canonical_args), "Mismatched args" assert utils.equivalent(utils.set(arg_names), canonical_args), "Mismatched args"
else canonical_args = utils.set(arg_names) else canonical_args = utils.set(arg_names)
insert aliases, stub insert aliases, stub
@defs[stub] = {:thunk, :stub, :arg_names, :src, :is_macro, :aliases} @defs[stub] = {:thunk, :stub, :arg_names, :src, :is_macro, :aliases, def_number:@def_number}
defmacro: (signature, thunk, src)=> defmacro: (signature, thunk, src)=>
@def(signature, thunk, src, true) @def(signature, thunk, src, true)
serialize_defs: (after=0)=>
defs = {}
for _, def in pairs(@defs)
defs[def.def_number] = def.src or ""
keys = utils.keys(defs)
table.sort(keys)
buff = {}
for i in *keys
if i > after and #defs[i] > 0
insert buff, defs[i]
return concat buff, "\n"
call: (stub,line_no,...)=> call: (stub,line_no,...)=>
def = @defs[stub] def = @defs[stub]