Some cleanup and fixes. Simplifying a lot of code, and extending the
flexibility of scoping. Redesigned Object system too.
This commit is contained in:
parent
49f1eb3d08
commit
b12744d831
@ -359,51 +359,6 @@ immediately
|
|||||||
end --when % = ?
|
end --when % = ?
|
||||||
return %code
|
return %code
|
||||||
|
|
||||||
# Try/except
|
|
||||||
immediately
|
|
||||||
compile [..]
|
|
||||||
try %action and if it succeeds %success or if it barfs %msg %fallback
|
|
||||||
try %action and if it barfs %msg %fallback or if it succeeds %success
|
|
||||||
..to
|
|
||||||
Lua ".."
|
|
||||||
do
|
|
||||||
local fell_through = false
|
|
||||||
local err, erred = nil, false
|
|
||||||
local ok, ret = xpcall(function()
|
|
||||||
\(%action as lua statements)
|
|
||||||
fell_through = true
|
|
||||||
end, function(\(%msg as lua expr))
|
|
||||||
local ok, ret = pcall(function()
|
|
||||||
\(%fallback as lua statements)
|
|
||||||
end)
|
|
||||||
if not ok then err, erred = ret, true end
|
|
||||||
end)
|
|
||||||
if ok then
|
|
||||||
\(%success as lua statements)
|
|
||||||
if not fell_through then
|
|
||||||
return ret
|
|
||||||
end
|
|
||||||
elseif erred then
|
|
||||||
error(err, 0)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
immediately
|
|
||||||
parse [..]
|
|
||||||
try %action and if it succeeds %success or if it barfs %fallback
|
|
||||||
try %action and if it barfs %fallback or if it succeeds %success
|
|
||||||
..as: try %action and if it succeeds %success or if it barfs (=lua "") %fallback
|
|
||||||
immediately
|
|
||||||
parse [try %action] as
|
|
||||||
try %action and if it succeeds: do nothing
|
|
||||||
..or if it barfs: do nothing
|
|
||||||
parse [try %action and if it barfs %fallback] as
|
|
||||||
try %action and if it succeeds: do nothing
|
|
||||||
..or if it barfs %fallback
|
|
||||||
parse [try %action and if it barfs %msg %fallback] as
|
|
||||||
try %action and if it succeeds: do nothing
|
|
||||||
..or if it barfs %msg %fallback
|
|
||||||
parse [try %action and if it succeeds %success] as
|
|
||||||
try %action and if it succeeds %success or if it barfs: do nothing
|
|
||||||
|
|
||||||
# Do/finally
|
# Do/finally
|
||||||
immediately
|
immediately
|
||||||
@ -433,17 +388,7 @@ immediately
|
|||||||
declare locals in %body
|
declare locals in %body
|
||||||
return
|
return
|
||||||
Lua value ".."
|
Lua value ".."
|
||||||
((function()
|
(function()
|
||||||
\%body
|
\%body
|
||||||
end)())
|
end)()
|
||||||
|
|
||||||
# Coroutines:
|
|
||||||
immediately
|
|
||||||
compile [values %body, coroutine %body, generator %body] to
|
|
||||||
Lua value ".."
|
|
||||||
(function()
|
|
||||||
\(%body as lua statements)
|
|
||||||
end)
|
|
||||||
compile [->] to: Lua "coroutine.yield(true);"
|
|
||||||
compile [-> %] to: Lua "coroutine.yield(true, \(% as lua expr));"
|
|
||||||
compile [-> %k = %v] to: Lua "coroutine.yield(\(%k as lua expr), \(%v as lua expr));"
|
|
||||||
|
17
core/coroutines.nom
Normal file
17
core/coroutines.nom
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
#
|
||||||
|
This file defines the code that creates and manipulates coroutines
|
||||||
|
|
||||||
|
use "core/metaprogramming.nom"
|
||||||
|
|
||||||
|
compile [coroutine %body, generator %body] to
|
||||||
|
Lua value ".."
|
||||||
|
(function()
|
||||||
|
\(%body as lua statements)
|
||||||
|
end)
|
||||||
|
compile [->] to: Lua value "coroutine.yield(true)"
|
||||||
|
compile [-> %] to: Lua value "coroutine.yield(true, \(% as lua expr))"
|
||||||
|
compile [for % in coroutine %co %body] to
|
||||||
|
Lua ".."
|
||||||
|
for junk,\(% as lua expr) in coroutine.wrap(\(%co as lua expr)) do
|
||||||
|
\(%body as lua statements)
|
||||||
|
end
|
@ -20,3 +20,49 @@ compile [assume %condition or barf %message] to
|
|||||||
if not \(%condition as lua expr) then
|
if not \(%condition as lua expr) then
|
||||||
error(\(%message as lua expr), 0);
|
error(\(%message as lua expr), 0);
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# Try/except
|
||||||
|
immediately
|
||||||
|
compile [..]
|
||||||
|
try %action and if it succeeds %success or if it barfs %msg %fallback
|
||||||
|
try %action and if it barfs %msg %fallback or if it succeeds %success
|
||||||
|
..to
|
||||||
|
Lua ".."
|
||||||
|
do
|
||||||
|
local fell_through = false
|
||||||
|
local err, erred = nil, false
|
||||||
|
local ok, ret = xpcall(function()
|
||||||
|
\(%action as lua statements)
|
||||||
|
fell_through = true
|
||||||
|
end, function(\(%msg as lua expr))
|
||||||
|
local ok, ret = pcall(function()
|
||||||
|
\(%fallback as lua statements)
|
||||||
|
end)
|
||||||
|
if not ok then err, erred = ret, true end
|
||||||
|
end)
|
||||||
|
if ok then
|
||||||
|
\(%success as lua statements)
|
||||||
|
if not fell_through then
|
||||||
|
return ret
|
||||||
|
end
|
||||||
|
elseif erred then
|
||||||
|
error(err, 0)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
immediately
|
||||||
|
parse [..]
|
||||||
|
try %action and if it succeeds %success or if it barfs %fallback
|
||||||
|
try %action and if it barfs %fallback or if it succeeds %success
|
||||||
|
..as: try %action and if it succeeds %success or if it barfs (=lua "") %fallback
|
||||||
|
immediately
|
||||||
|
parse [try %action] as
|
||||||
|
try %action and if it succeeds: do nothing
|
||||||
|
..or if it barfs: do nothing
|
||||||
|
parse [try %action and if it barfs %fallback] as
|
||||||
|
try %action and if it succeeds: do nothing
|
||||||
|
..or if it barfs %fallback
|
||||||
|
parse [try %action and if it barfs %msg %fallback] as
|
||||||
|
try %action and if it succeeds: do nothing
|
||||||
|
..or if it barfs %msg %fallback
|
||||||
|
parse [try %action and if it succeeds %success] as
|
||||||
|
try %action and if it succeeds %success or if it barfs: do nothing
|
||||||
|
@ -117,8 +117,12 @@ immediately
|
|||||||
action [%tree as lua return]
|
action [%tree as lua return]
|
||||||
=lua "nomsu:tree_to_lua(\%tree):as_statements('return ')"
|
=lua "nomsu:tree_to_lua(\%tree):as_statements('return ')"
|
||||||
|
|
||||||
action [%var as lua identifier]
|
action [%var as lua identifier, %var as lua id]
|
||||||
=lua "type(\%var) == 'string' and string.as_lua_id(\%var) or nomsu:tree_to_lua(\%var)"
|
lua> ".."
|
||||||
|
if type(\%var) == 'string' then return string.as_lua_id(\%var)
|
||||||
|
elseif \%var.type == 'Var' then return string.as_lua_id(\%var[1])
|
||||||
|
elseif \%var.type == 'Action' then return "A"..string.as_lua_id(\%var.stub)
|
||||||
|
end
|
||||||
|
|
||||||
immediately
|
immediately
|
||||||
compile [%tree with %t -> %replacement] to
|
compile [%tree with %t -> %replacement] to
|
||||||
|
@ -40,7 +40,7 @@ immediately
|
|||||||
|
|
||||||
immediately
|
immediately
|
||||||
# Simultaneous mutli-assignments like: x,y,z = 1,x,3;
|
# Simultaneous mutli-assignments like: x,y,z = 1,x,3;
|
||||||
compile [<- %assignments] to
|
compile [<- %assignments, assign %assignments] to
|
||||||
assume (%assignments.type is "Dict") or barf ".."
|
assume (%assignments.type is "Dict") or barf ".."
|
||||||
Expected a Dict for the assignments part of '<- %' statement, not \%assignments
|
Expected a Dict for the assignments part of '<- %' statement, not \%assignments
|
||||||
lua> ".."
|
lua> ".."
|
||||||
|
@ -1,54 +1,30 @@
|
|||||||
use "core/metaprogramming.nom"
|
use "core/metaprogramming.nom"
|
||||||
use "core/text.nom"
|
|
||||||
use "core/operators.nom"
|
use "core/operators.nom"
|
||||||
use "core/collections.nom"
|
use "core/collections.nom"
|
||||||
use "core/control_flow.nom"
|
use "core/control_flow.nom"
|
||||||
|
|
||||||
compile [using %definitions %body, using %definitions do %body] to
|
compile [with local %locals %body, with local %locals do %body] to
|
||||||
%setup_lua <-
|
%body_lua <- (%body as lua statements)
|
||||||
Lua ".."
|
when %locals.type = ?
|
||||||
local fell_through = false
|
* "Dict"
|
||||||
local ok, ret = pcall(function()
|
%body_lua <-
|
||||||
\(%definitions as lua statements)
|
Lua ".."
|
||||||
fell_through = true
|
\(=lua "A_assign_1(\%locals, \%locals)")
|
||||||
end)
|
\%body_lua
|
||||||
%body_lua <-
|
declare locals
|
||||||
Lua ".."
|
(%.1 as lua id) for % in %locals
|
||||||
local fell_through = false
|
.. in %body_lua
|
||||||
local ok, ret = pcall(function()
|
* "List"
|
||||||
\(%body as lua statements)
|
declare locals
|
||||||
fell_through = true
|
(% as lua id) for % in %locals
|
||||||
end)
|
.. in %body_lua
|
||||||
remove free vars (declare locals in %setup_lua) from %body_lua
|
* "Var"
|
||||||
%lua <-
|
* "Action"
|
||||||
|
declare locals [%locals as lua id] in %body_lua
|
||||||
|
* else
|
||||||
|
barf "Unexpected local: \(%locals as nomsu)"
|
||||||
|
return
|
||||||
Lua ".."
|
Lua ".."
|
||||||
do
|
do
|
||||||
local old_actions, old_compile_actions, old_arg_orders = ACTIONS, COMPILE_ACTIONS, ARG_ORDERS
|
|
||||||
ACTIONS = setmetatable({}, {__index=old_actions})
|
|
||||||
COMPILE_ACTIONS = setmetatable({}, {__index=old_compile_actions})
|
|
||||||
ARG_ORDERS = setmetatable({}, {__index=old_arg_orders})
|
|
||||||
\%setup_lua
|
|
||||||
if not ok then
|
|
||||||
ACTIONS, COMPILE_ACTIONS, ARG_ORDERS = old_actions, old_compile_actions, old_arg_orders
|
|
||||||
error(ret, 0)
|
|
||||||
end
|
|
||||||
if not fell_through then
|
|
||||||
ACTIONS, COMPILE_ACTIONS, ARG_ORDERS = old_actions, old_compile_actions, old_arg_orders
|
|
||||||
return ret
|
|
||||||
end
|
|
||||||
getmetatable(ACTIONS).__newindex = old_actions
|
|
||||||
getmetatable(COMPILE_ACTIONS).__newindex = old_compile_actions
|
|
||||||
getmetatable(ARG_ORDERS).__newindex = old_arg_orders
|
|
||||||
\%body_lua
|
\%body_lua
|
||||||
ACTIONS, COMPILE_ACTIONS, ARG_ORDERS = old_actions, old_compile_actions, old_arg_orders
|
|
||||||
if not ok then
|
|
||||||
error(ret, 0)
|
|
||||||
end
|
|
||||||
if not fell_through then
|
|
||||||
return ret
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
declare locals in %lua
|
|
||||||
return %lua
|
|
||||||
|
|
||||||
parse [using %] as: using % (do nothing)
|
|
||||||
|
@ -1,7 +1,5 @@
|
|||||||
use "core"
|
use "core"
|
||||||
|
|
||||||
# TODO: make codegen less verbose
|
|
||||||
|
|
||||||
lua> "CLASSES = {}"
|
lua> "CLASSES = {}"
|
||||||
|
|
||||||
immediately
|
immediately
|
||||||
@ -11,50 +9,41 @@ action [new %classname %inst]
|
|||||||
immediately
|
immediately
|
||||||
parse [new %classname] as: new %classname {}
|
parse [new %classname] as: new %classname {}
|
||||||
|
|
||||||
compile [as %instance %body] to
|
immediately
|
||||||
Lua ".."
|
compile [call method %method] to: Lua value "self:\(%method as lua expr)"
|
||||||
do
|
|
||||||
local self = \(%instance as lua expr)
|
parse [as %instance %body] as
|
||||||
local old_self = self.class:set_self(self)
|
result of
|
||||||
old_actions, ACTIONS = ACTIONS, self.class.ACTIONS
|
%old_self <- (me)
|
||||||
old_compile_actions, COMPILE_ACTIONS = COMPILE_ACTIONS, self.class.COMPILE_ACTIONS
|
(me) <- %instance
|
||||||
old_arg_orders, ARG_ORDERS = ARG_ORDERS, self.class.ARG_ORDERS
|
try
|
||||||
local fell_through = false
|
%body
|
||||||
local ok, ret = pcall(function()
|
..and if it barfs %msg
|
||||||
\(%body as lua statements)
|
(me) <- %old_self
|
||||||
fell_through = true
|
barf %msg
|
||||||
end)
|
..or if it succeeds
|
||||||
self.class:set_self(old_self)
|
(me) <- %old_self
|
||||||
ACTIONS = old_actions
|
|
||||||
COMPILE_ACTIONS = old_compile_actions
|
compile [object %classname %class_body] to
|
||||||
ARG_ORDERS = old_arg_orders
|
%lua <-
|
||||||
if not ok then error(ret) end
|
Lua ".."
|
||||||
if not fell_through then return ret end
|
do
|
||||||
|
local class = {name=\(%classname as lua expr)}
|
||||||
|
setmetatable(class, {__tostring=function(cls) return cls.name end})
|
||||||
|
CLASSES[class.name] = class
|
||||||
|
class.__index = class
|
||||||
|
class.class = class
|
||||||
|
|
||||||
|
if: %class_body.type != "Block"
|
||||||
|
%class_body <- [%class_body]
|
||||||
|
for %statement in %class_body
|
||||||
|
assume: (%statement.type is "Action") and (%statement.stub is "action % %")
|
||||||
|
to %lua write "\n class."
|
||||||
|
to %lua write (%statement as lua)
|
||||||
|
|
||||||
|
to %lua write ".."
|
||||||
|
|
||||||
|
class.__tostring = class["A"..string.as_lua_id("as text")]
|
||||||
end
|
end
|
||||||
|
|
||||||
parse [object %classname %class_body] as
|
return %lua
|
||||||
using
|
|
||||||
%cls <- {..}
|
|
||||||
name:%classname
|
|
||||||
ACTIONS:=lua "ACTIONS", COMPILE_ACTIONS:=lua "COMPILE_ACTIONS"
|
|
||||||
ARG_ORDERS:=lua "ARG_ORDERS"
|
|
||||||
(=lua "CLASSES").%classname <- %cls
|
|
||||||
lua> ".."
|
|
||||||
setmetatable(\%cls, {__tostring=function() return \%classname end})
|
|
||||||
local self = nil
|
|
||||||
\%cls.set_self = function(_, inst)
|
|
||||||
local old_self = self
|
|
||||||
self = inst
|
|
||||||
return old_self
|
|
||||||
end
|
|
||||||
\%cls.__index = \%cls
|
|
||||||
\%cls.class = \%cls
|
|
||||||
%class_body
|
|
||||||
run ".."
|
|
||||||
action [new \%classname %inst]
|
|
||||||
say "NEWING"
|
|
||||||
return: =lua "setmetatable(\\%inst, \\%cls)"
|
|
||||||
lua> ".."
|
|
||||||
if ACTIONS["as text"] then
|
|
||||||
\%cls.__tostring = ACTIONS["as text"]
|
|
||||||
end
|
|
||||||
|
10
nomsu.lua
10
nomsu.lua
@ -1305,6 +1305,16 @@ do
|
|||||||
self.environment.LOADED = { }
|
self.environment.LOADED = { }
|
||||||
self.environment.AST = AST
|
self.environment.AST = AST
|
||||||
self.environment._ENV = self.environment
|
self.environment._ENV = self.environment
|
||||||
|
setmetatable(self.environment, {
|
||||||
|
__index = function(self, k)
|
||||||
|
do
|
||||||
|
local _self = rawget(self, "self")
|
||||||
|
if _self then
|
||||||
|
return _self[k]
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
})
|
||||||
return self:initialize_core()
|
return self:initialize_core()
|
||||||
end,
|
end,
|
||||||
__base = _base_0,
|
__base = _base_0,
|
||||||
|
@ -299,6 +299,10 @@ class NomsuCompiler
|
|||||||
@environment.LOADED = {}
|
@environment.LOADED = {}
|
||||||
@environment.AST = AST
|
@environment.AST = AST
|
||||||
@environment._ENV = @environment
|
@environment._ENV = @environment
|
||||||
|
setmetatable @environment,
|
||||||
|
__index: (k)=>
|
||||||
|
if _self = rawget(@, "self")
|
||||||
|
return _self[k]
|
||||||
@initialize_core!
|
@initialize_core!
|
||||||
|
|
||||||
parse: (nomsu_code)=>
|
parse: (nomsu_code)=>
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
#..
|
#..
|
||||||
Tests for the stuff defined in lib/control_flow.nom
|
Tests for the stuff defined in core/control_flow.nom
|
||||||
|
|
||||||
use "core"
|
use "core"
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
#
|
#
|
||||||
Tests for the stuff defined in lib/control_flow.nom
|
Tests for the stuff defined in core/control_flow.nom
|
||||||
|
|
||||||
use "core"
|
use "core"
|
||||||
|
|
||||||
@ -193,15 +193,3 @@ assume
|
|||||||
return %n
|
return %n
|
||||||
..= 6
|
..= 6
|
||||||
|
|
||||||
#
|
|
||||||
%nums <- []
|
|
||||||
for % in
|
|
||||||
values
|
|
||||||
-> 4
|
|
||||||
-> 5
|
|
||||||
-> 6
|
|
||||||
..
|
|
||||||
add % to %nums
|
|
||||||
|
|
||||||
assume (%nums = [4,5,6]) or barf "Coroutine iteration failed"
|
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
#..
|
#..
|
||||||
Tests for the stuff defined in lib/control_flow.nom
|
Tests for the stuff defined in core/control_flow.nom
|
||||||
|
|
||||||
use "core"
|
use "core"
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
#..
|
#..
|
||||||
Tests for the stuff defined in lib/metaprogramming.nom
|
Tests for the stuff defined in core/metaprogramming.nom
|
||||||
|
|
||||||
use "core"
|
use "core"
|
||||||
|
|
||||||
|
@ -1,3 +1,6 @@
|
|||||||
|
#
|
||||||
|
Tests for the object model defined in lib/object.nom
|
||||||
|
|
||||||
use "core"
|
use "core"
|
||||||
use "lib/object.nom"
|
use "lib/object.nom"
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
#..
|
#..
|
||||||
Tests for the stuff defined in lib/operators.nom
|
Tests for the stuff defined in core/operators.nom
|
||||||
|
|
||||||
use "core"
|
use "core"
|
||||||
|
|
||||||
|
@ -1,25 +1,16 @@
|
|||||||
use "core"
|
use "core"
|
||||||
|
|
||||||
using
|
%x <- "outer"
|
||||||
%x <- 99
|
with local %x
|
||||||
..do
|
%x <- "inner"
|
||||||
assume: %x = 99
|
assume: %x = "inner"
|
||||||
|
assume: %x = "outer"
|
||||||
|
|
||||||
using
|
action [foo] "outer foo"
|
||||||
action [foo]
|
with local (foo)
|
||||||
return 99
|
action [foo] "inner foo"
|
||||||
|
assume: (foo) = "inner foo"
|
||||||
|
assume: (foo) = "outer foo"
|
||||||
|
|
||||||
..do
|
|
||||||
assume: (foo) = 99
|
|
||||||
|
|
||||||
action [baz]
|
|
||||||
return "outer"
|
|
||||||
|
|
||||||
#
|
|
||||||
do
|
|
||||||
local action [baz]
|
|
||||||
return "inner"
|
|
||||||
assume: (baz) = "inner"
|
|
||||||
assume: (baz) = "outer"
|
|
||||||
|
|
||||||
say "Scopes test passed."
|
say "Scopes test passed."
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
#..
|
#..
|
||||||
Tests for the stuff defined in lib/text.nom
|
Tests for the stuff defined in core/text.nom
|
||||||
|
|
||||||
use "core"
|
use "core"
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user