Cleanups.

This commit is contained in:
Bruce Hill 2018-06-20 15:22:03 -07:00
parent c1cec2ac84
commit d73cbf0aa5
10 changed files with 170 additions and 76 deletions

View File

@ -50,10 +50,10 @@ compile [append %item to %list, add %item to %list, to %list add %item, to %list
Lua "table.insert(\(%list as lua expr), \(%item as lua expr))" Lua "table.insert(\(%list as lua expr), \(%item as lua expr))"
compile [pop from %list, remove last from %list] to compile [pop from %list, remove last from %list] to
Lua "table.remove(\(%list as lua expr))" Lua value "table.remove(\(%list as lua expr))"
compile [remove index %index from %list] to compile [remove index %index from %list] to
Lua "table.remove(\(%list as lua expr), \(%index as lua expr))" Lua value "table.remove(\(%list as lua expr), \(%index as lua expr))"
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

View File

@ -135,6 +135,11 @@ compile [stop %var] to
compile [do next %var] to compile [do next %var] to
Lua "goto continue_\(%var as lua identifier)" Lua "goto continue_\(%var as lua identifier)"
compile [=== stop %var ===, --- stop %var ---, *** stop %var ***] to
Lua "::stop_\(%var as lua identifier)::"
compile [=== next %var ===, --- next %var ---, *** next %var ***] to
Lua "::continue_\(%var as lua identifier)::"
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
# Numeric range for loops # Numeric range for loops
@ -153,7 +158,7 @@ compile [..]
(%.type = "Action") and (%.type = "Action") and
(%.stub is "do next %") and (%.stub is "do next %") and
%.(3).1 = %var.1 %.(3).1 = %var.1
..: to %lua write "\n ::continue_\(%var as lua identifier)::" ..: to %lua write "\n \(compile as: === next %var ===)"
to %lua write "\nend --numeric for-loop" to %lua write "\nend --numeric for-loop"
if if
@ -166,7 +171,7 @@ compile [..]
Lua ".." Lua ".."
do -- scope for stopping for-loop do -- scope for stopping for-loop
\%lua \%lua
::stop_\(%var as lua identifier):: \(compile as: === stop %var ===)
end -- end of scope for stopping for-loop end -- end of scope for stopping for-loop
return %lua return %lua
@ -188,7 +193,7 @@ compile [for %var in %iterable %body] to
(%.type = "Action") and (%.type = "Action") and
(%.stub is "do next %") and (%.stub is "do next %") and
%.3.(1) = %var.(1) %.3.(1) = %var.(1)
..: to %lua write (Lua "\n::continue_\(%var as lua identifier)::") ..: to %lua write (Lua "\n\(compile as: === next %var ===)")
to %lua write "\nend --foreach-loop" to %lua write "\nend --foreach-loop"
if if
%body has subtree % where %body has subtree % where
@ -200,7 +205,7 @@ compile [for %var in %iterable %body] to
Lua ".." Lua ".."
do -- scope for stopping for-loop do -- scope for stopping for-loop
\%lua \%lua
::stop_\(%var as lua identifier):: \(compile as: === stop %var ===)
end -- end of scope for stopping for-loop end -- end of scope for stopping for-loop
return %lua return %lua
@ -221,14 +226,14 @@ compile [..]
(%.type = "Action") and (%.type = "Action") and
(%.stub is "do next %") and (%.stub is "do next %") and
%.3.(1) = %key.(1) %.3.(1) = %key.(1)
..: to %lua write (Lua "\n::continue_\(%key as lua identifier)::") ..: to %lua write (Lua "\n\(compile as: === next %key ===)")
if if
%body has subtree % where %body has subtree % where
(%.type = "Action") and (%.type = "Action") and
(%.stub is "do next %") and (%.stub is "do next %") and
%.3.(1) = %value.(1) %.3.(1) = %value.(1)
..: to %lua write (Lua "\n::continue_\(%value as lua identifier)::") ..: to %lua write (Lua "\n\(compile as: === next %value ===)")
to %lua write "\nend --foreach-loop" to %lua write "\nend --foreach-loop"
%stop_labels <- (Lua "") %stop_labels <- (Lua "")
@ -237,14 +242,14 @@ compile [..]
(%.type = "Action") and (%.type = "Action") and
(%.stub is "stop %") and (%.stub is "stop %") and
%.2.(1) = %key.(1) %.2.(1) = %key.(1)
..: to %stop_labels write "\n::stop_\(%key as lua identifier)::" ..: to %stop_labels write "\n\(compile as: === stop %key ===)"
if if
%body has subtree % where %body has subtree % where
(%.type = "Action") and (%.type = "Action") and
(%.stub is "stop %") and (%.stub is "stop %") and
%.2.(1) = %value.(1) %.2.(1) = %value.(1)
..: to %stop_labels write "\n::stop_\(%value as lua identifier)::" ..: to %stop_labels write "\n\(compile as: === stop %value ===)"
if: (length of "\%stop_labels") > 0 if: (length of "\%stop_labels") > 0
%lua <- %lua <-

View File

@ -6,9 +6,9 @@ use "core/metaprogramming.nom"
compile [say %message] to compile [say %message] to
lua> ".." lua> ".."
if \%message.type == "Text" then if \%message.type == "Text" then
return LuaCode(tree.source, "io.write(", \(%message as lua expr), ", '\\\\n');"); return LuaCode(tree.source, "print(", \(%message as lua expr), ");");
else else
return LuaCode(tree.source, "io.write(tostring(", \(%message as lua expr), "), '\\\\n');"); return LuaCode(tree.source, "print(tostring(", \(%message as lua expr), "));");
end end
compile [ask %prompt] to compile [ask %prompt] to

View File

@ -4,7 +4,7 @@
lua> ".." lua> ".."
nomsu.COMPILE_ACTIONS["% -> %"] = function(nomsu, tree, \%args, \%body) nomsu.COMPILE_ACTIONS["% -> %"] = function(nomsu, tree, \%args, \%body)
local lua = LuaCode(tree.source, "function(") local lua = LuaCode.Value(tree.source, "function(")
if AST.is_syntax_tree(\%args, "Action") then \%args = \%args:get_args() end if AST.is_syntax_tree(\%args, "Action") then \%args = \%args:get_args() end
local lua_args = table.map(\%args, function(a) return AST.is_syntax_tree(a) and tostring(nomsu:compile(a)) or a end) local lua_args = table.map(\%args, function(a) return AST.is_syntax_tree(a) and tostring(nomsu:compile(a)) or a end)
lua:concat_append(lua_args, ", ") lua:concat_append(lua_args, ", ")
@ -163,7 +163,7 @@ compile [declare locals %locals in %code] to
compile [remove free vars %vars from %code] to compile [remove free vars %vars from %code] to
Lua "\(%code as lua expr):remove_free_vars(\(%vars as lua expr));" Lua "\(%code as lua expr):remove_free_vars(\(%vars as lua expr));"
parse [%lua <-write %code, to %lua write %code] as: lua> "\%lua:append(\%code);" compile [%lua <-write %code, to %lua write %code] to: Lua "\(%lua as lua expr):append(\(%code as lua expr));"
compile [quote %s] to compile [quote %s] to
Lua value ".." Lua value ".."

View File

@ -130,6 +130,16 @@ compile [with %assignments %body] to
\%lua \%lua
end -- 'with' block end -- 'with' block
compile [local %var_or_vars] to
%lua <- (Lua "")
lua> ".."
if \%var_or_vars.type == "List" then
\%lua:add_free_vars(table.map(\%var_or_vars, function(v) return tostring(nomsu:compile(v)) end))
else
\%lua:add_free_vars({tostring(nomsu:compile(\%var_or_vars))})
end
return %lua
# Math Operators # Math Operators
compile [%x wrapped around %y, %x mod %y] to: Lua value "(\(%x as lua expr) % \(%y as lua expr))" compile [%x wrapped around %y, %x mod %y] to: Lua value "(\(%x as lua expr) % \(%y as lua expr))"

View File

@ -160,7 +160,7 @@ print_error = function(error_message, stack_offset)
end end
end end
end end
if file and calling_fn.short_src:match("%.moon$") and type(MOON_SOURCE_MAP[file]) == 'table' then if file and (calling_fn.short_src:match("%.moon$") or file:match("^#![^\n]*moon\n")) and type(MOON_SOURCE_MAP[file]) == 'table' then
local char = MOON_SOURCE_MAP[file][calling_fn.currentline] local char = MOON_SOURCE_MAP[file][calling_fn.currentline]
line_num = 1 line_num = 1
for _ in file:sub(1, char):gmatch("\n") do for _ in file:sub(1, char):gmatch("\n") do

View File

@ -98,7 +98,8 @@ print_error = (error_message, stack_offset=3)->
name = "upvalue '#{varname}'" name = "upvalue '#{varname}'"
if not varname\match("%(") if not varname\match("%(")
break break
if file and calling_fn.short_src\match("%.moon$") and type(MOON_SOURCE_MAP[file]) == 'table'
if file and (calling_fn.short_src\match("%.moon$") or file\match("^#![^\n]*moon\n")) and type(MOON_SOURCE_MAP[file]) == 'table'
char = MOON_SOURCE_MAP[file][calling_fn.currentline] char = MOON_SOURCE_MAP[file][calling_fn.currentline]
line_num = 1 line_num = 1
for _ in file\sub(1,char)\gmatch("\n") do line_num += 1 for _ in file\sub(1,char)\gmatch("\n") do line_num += 1

View File

@ -273,24 +273,26 @@ action [square root of %n]
=lua "math.sqrt(\%n)" =lua "math.sqrt(\%n)"
say "The square root of 2 is \(square root of 2)" say "The square root of 2 is \(square root of 2)"
# The "immediately %" macro forces the indented code below it to run before the rest of # Macros can be defined to transform one bit of nomsu code into another using "parse % as %":
the file finishes compiling, so it's often useful for writing your own macros. parse [if %condition is untrue %body] as
immediately if (not %condition) %body
# Macros can be defined to transform one bit of nomsu code into another using "parse % as %":
parse [if %condition is untrue %body] as
if (not %condition) %body
# Or to transform nomsu code into custom lua code using "compile % to %" # Or to transform nomsu code into custom lua code using "compile % to %"
compile [if %condition on opposite day %body] to compile [if %condition on opposite day %body] to
Lua ".." Lua ".."
if not \(%condition as lua expr) then if not \(%condition as lua expr) then
\(%body as lua statements) \(%body as lua statements)
end end
# Constants can be defined as macros # Constants can be defined as macros
parse [TWENTY] as: 20 parse [TWENTY] as: 20
# When they're invoked, they'll need parentheses just like a function call # When they're invoked, they'll need parentheses just like a function call
parse [TWENTY ONE] as: 21 parse [TWENTY ONE] as: 21
# If you need to use compile-time actions in the same file that they're defined in, you
can add a line of tildes (3 or more), and the file will be split into chunks, and each
chunk will run before the next one compiles. (note that each chunk has its own scope)
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
if (1 > (TWENTY)) is untrue if (1 > (TWENTY)) is untrue
say "Nomsu parsing macros work!" say "Nomsu parsing macros work!"
@ -305,55 +307,131 @@ if (1 > (TWENTY)) on opposite day
# Well... it's always *possible* to fall back to Lua behavior for something like this: # Well... it's always *possible* to fall back to Lua behavior for something like this:
action [best of %items according to %key-fn] action [best of %items according to %key-fn]
<- {%best:nil, %best-key:nil} <- {%best:nil, %best-key:nil}
for % in %items for %item in %items
%key <- (=lua "\%key-fn(\%)") %key <- (=lua "\%key-fn(\%item)")
if: (%best is (nil)) or (%key > %best-key) if: (%best is (nil)) or (%key > %best-key)
<- {%best:%, %best-key:%key} <- {%best:%item, %best-key:%key}
return %best return %best
immediately say: best of [2,-3,4,-8] according to ([%x] -> (%x * %x))
compile [function %var %body] to
Lua value ".."
(function(\(%var as lua expr))
return \(%body as lua expr)
end)
say: best of [2,-3,4,-8] according to (function %: % * %)
# But nomsu was mostly designed so that you don't *need* to. Instead of creating a # But nomsu was mostly designed so that you don't *need* to. Instead of creating a
one-off function to pass to another function and get called a bunch of times, you one-off function to pass to another function and get called a bunch of times, you
could use a macro to generate a single block of code that inlines the expression you could use a macro to generate a single block of code that inlines the expression you
want to use: want to use:
immediately parse [best of %items where %item-var has score %key-expr] as
parse [best of %items according to %key-var in %key-expr] as result of
result of <- {%best:nil, %best-key:nil}
<- {%best:nil, %best-key:nil} for %item-var in %items
for %key-var in %items %key <- %key-expr
%key <- %key-expr if: (%best is (nil)) or (%key > %best-key)
if: (%best is (nil)) or (%key > %best-key) <- {%best:%item-var, %best-key:%key}
<- {%best:%, %best-key:%key} return %best
return %best ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
# This results in generated code that is more efficient (no function calls in the say: best of [2,-3,4,-8] where %x has score (%x * %x)
inner loop)
say: best of [2,-3,4,-8] according to % in (% * %)
# In a functional programming language, you might do something like: # The first example generates code that looks like:
doubled = map(list, function(x) return 2 * x end)
to get a new list with every entry multiplied by 2, but it's *much* more readable to
do something like:
%nums <- [1,2,3,4,5]
%double-nums <- ((2 * %num) for %num in %nums)
# Nomsu comes with built-in list comprehensions, but the flexible macro system makes it A_best_of_1_according_to_2 = function(items, key_fn)
incredibly easy to make similar constructs. local best, best_key = nil, nil
immediately for _, item in ipairs(items) do
parse [%expr for %key in %list BACKWARDS!] as local key = key_fn(item)
result of if best == nil or key > best_key then
%result <- [] best, best_key = item, key
%N <- (length of %list) end
for %i = %key in %list end
%result.(%N - %i + 1) <- %expr return best
return %result end
print(A_best_of_1_according_to_2({2,-3,4,-8}, function(x)
return x * x
end))
%double-nums <- ((2 * %num) for %num in %nums BACKWARDS!) But the second example produces something more like:
say %double-nums
print((function()
local best, best_key = nil, nil
for _, x in ipairs({1,-2,3,-4}) do
local key = x * x
if best == nil or key > best_key then
best, best_key = x, key
end
end
return best
end)())
Notice that the second example has inlined the key function, so that in the inner loop,
the code only has to do a multiplication, instead of calling a function that does the
multiplication. In addition to being more efficient, it's also more powerful and
flexible for the language to allow you to define syntax that manipulates expressions.
For example, list comprehensions can (and are) easily defined in Nomsu as:
parse [%expr for %key-var in %list] as
result of
%result <- []
for %key-var in %list
add %expr to %result
return %result
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
%double-nums <- ((2 * %num) for %num in [1,2,3,4,5])
# Naturally, Nomsu's macros are hygienic, so even though "%result" is used as a local
variable in the macro, there won't be any namespace collisions or confusion if you
use a different variable named "%result" as any (or all) of the arguments to the macro.
%result <- [1,2,3,4]
%result <- ((2*%result) for %result in %result)
assume: %result = [2,4,6,8]
# You'll find that in a lot places where a functional programmer would think to use a
lambda function, what they actually want to do is some metaprogramming. If you embrace
the metaprogramming from the start, then you can do a lot more powerful stuff.
# For example, you can define a loop that operates on recursive datastructures, but still
has all of the normal imperative control structures:
compile [stack for %var] to
Lua value "stack\(%var as lua id)"
parse [recurse on %sub as %super] as
add %sub to (stack for %super)
parse [for %var in recursive %iterable %body] as
with {(stack for %var): [%iterable]}
repeat while: (length of (stack for %var)) > 0
%var <- (remove index 1 from (stack for %var))
with {%val: %expr}
if: %val != (nil)
add %val to %result
%body
=== next %var ===
=== stop %var ===
~~~~~~~~~~~~~~~~
# This action iterates over a recursive structure, but it's written imperatively and it
can return early without traversing the whole tree.
action [tree %tree contains %val]
for %subtree in recursive %tree
if: %subtree = %val
return: yes
if: (type of %subtree) = "table"
for % in %subtree: recurse on % as %subtree
return: no
%t <- [1,[2,[3,4,[]],5,[6]]]
say: tree %t contains 3
say: tree %t contains 99
action [linked list from %list]
%head <- (nil)
for %i in (length of %list) to 1 via -1
%head <- {value:%list.%i, next:%head}
return %head
for %node in recursive (linked list from [3,4,5,6,7])
say "NODE: \(%node.value)"
recurse on %node.next as %node

View File

@ -501,7 +501,7 @@ do
source = nil source = nil
end end
local lua_string = tostring(lua) local lua_string = tostring(lua)
local run_lua_fn, err = load(lua_string, tostring(source or lua.source), "t", self) local run_lua_fn, err = load(lua_string, nil and tostring(source or lua.source), "t", self)
if not run_lua_fn then if not run_lua_fn then
local line_numbered_lua = concat((function() local line_numbered_lua = concat((function()
local _accum_0 = { } local _accum_0 = { }

View File

@ -339,7 +339,7 @@ with NomsuCompiler
.run_lua = (lua, source=nil)=> .run_lua = (lua, source=nil)=>
lua_string = tostring(lua) lua_string = tostring(lua)
run_lua_fn, err = load(lua_string, tostring(source or lua.source), "t", self) run_lua_fn, err = load(lua_string, nil and tostring(source or lua.source), "t", self)
if not run_lua_fn if not run_lua_fn
line_numbered_lua = concat( line_numbered_lua = concat(
[format("%3d|%s",i,line) for i, line in ipairs get_lines\match(lua_string)], [format("%3d|%s",i,line) for i, line in ipairs get_lines\match(lua_string)],