Moved timeout code into compiler and out of a lib.

This commit is contained in:
Bruce Hill 2017-10-09 04:31:41 -07:00
parent e737333d27
commit 9e1aaf2d5c
3 changed files with 25 additions and 14 deletions

View File

@ -1,8 +0,0 @@
require "lib/core.nom"
rule [cap at %num instructions] =:
lua code ".."
|print("capped at", \(%num))
|local f = function()
| nomsu:error("Execution quota exceeded. Your code took too long.")
|end
|debug.sethook(f, "", \(%num))

View File

@ -332,7 +332,17 @@ do
end end
return tree return tree
end, end,
run = function(self, src, filename) run = function(self, src, filename, max_operations)
if max_operations == nil then
max_operations = nil
end
if max_operations then
local timeout
timeout = function()
return self:error("Execution quota exceeded. Your code took too long.")
end
debug.sethook(timeout, "", max_operations)
end
local tree = self:parse(src, filename) local tree = self:parse(src, filename)
assert(tree, "Tree failed to compile: " .. tostring(src)) assert(tree, "Tree failed to compile: " .. tostring(src))
assert(tree.type == "File", "Attempt to run non-file: " .. tostring(tree.type)) assert(tree.type == "File", "Attempt to run non-file: " .. tostring(tree.type))
@ -388,6 +398,9 @@ do
%s %s
return ret; return ret;
end);]]):format(concat(buffer, "\n")) end);]]):format(concat(buffer, "\n"))
if max_operations then
debug.sethook()
end
return return_value, lua_code return return_value, lua_code
end, end,
tree_to_value = function(self, tree, vars) tree_to_value = function(self, tree, vars)
@ -896,7 +909,7 @@ if arg then
end end
end end
if not args.input or args.flags["-i"] then if not args.input or args.flags["-i"] then
c:run('require "lib/core.nom"') c:run('require "lib/core.nom"', "stdin")
while true do while true do
local buff = "" local buff = ""
while true do while true do
@ -911,7 +924,7 @@ if arg then
break break
end end
local ok, ret = pcall(function() local ok, ret = pcall(function()
return c:run(buff) return c:run(buff, "stdin")
end) end)
if ok and ret ~= nil then if ok and ret ~= nil then
print("= " .. repr(ret)) print("= " .. repr(ret))

View File

@ -274,7 +274,11 @@ class NomsuCompiler
@print_tree tree, " " @print_tree tree, " "
return tree return tree
run: (src, filename)=> run: (src, filename, max_operations=nil)=>
if max_operations
timeout = ->
@error "Execution quota exceeded. Your code took too long."
debug.sethook timeout, "", max_operations
tree = @parse(src, filename) tree = @parse(src, filename)
assert tree, "Tree failed to compile: #{src}" assert tree, "Tree failed to compile: #{src}"
assert tree.type == "File", "Attempt to run non-file: #{tree.type}" assert tree.type == "File", "Attempt to run non-file: #{tree.type}"
@ -321,6 +325,8 @@ class NomsuCompiler
%s %s
return ret; return ret;
end);]])\format(concat(buffer, "\n")) end);]])\format(concat(buffer, "\n"))
if max_operations
debug.sethook!
return return_value, lua_code return return_value, lua_code
tree_to_value: (tree, vars)=> tree_to_value: (tree, vars)=>
@ -635,7 +641,7 @@ if arg
if not args.input or args.flags["-i"] if not args.input or args.flags["-i"]
-- REPL -- REPL
c\run('require "lib/core.nom"') c\run('require "lib/core.nom"', "stdin")
while true while true
buff = "" buff = ""
while true while true
@ -646,7 +652,7 @@ if arg
buff ..= line buff ..= line
if #buff == 0 if #buff == 0
break break
ok, ret = pcall(-> c\run(buff)) ok, ret = pcall(-> c\run(buff, "stdin"))
if ok and ret != nil if ok and ret != nil
print "= "..repr(ret) print "= "..repr(ret)