Added tentative support for luajit (depends on pure-lua LPEG

implementation).
This commit is contained in:
Bruce Hill 2018-05-09 13:34:33 -07:00
parent 126f51f955
commit 1a755ddee8
11 changed files with 143 additions and 111 deletions

View File

@ -163,7 +163,7 @@ do
if start or stop then if start or stop then
self.source = Source(filename, tonumber(start), tonumber(stop)) self.source = Source(filename, tonumber(start), tonumber(stop))
else else
self.source = Source(self.source, 1, #self + 1) self.source = Source(self.source, 1, #tostring(self) + 1)
end end
end end
assert(self.source == nil or Source:is_instance(self.source)) assert(self.source == nil or Source:is_instance(self.source))
@ -346,7 +346,7 @@ do
else else
walk(b, pos) walk(b, pos)
end end
pos = pos + #b pos = pos + #tostring(b)
end end
end end
walk(self, 1) walk(self, 1)

View File

@ -71,7 +71,7 @@ class Code
if start or stop if start or stop
@source = Source(filename, tonumber(start), tonumber(stop)) @source = Source(filename, tonumber(start), tonumber(stop))
else else
@source = Source(@source, 1, #self+1) @source = Source(@source, 1, #tostring(self)+1)
assert(@source == nil or Source\is_instance(@source)) assert(@source == nil or Source\is_instance(@source))
indent = 0 indent = 0
for i,b in ipairs @bits for i,b in ipairs @bits
@ -219,7 +219,7 @@ class Lua extends Code
nomsu_to_lua[lua.source.start] = pos nomsu_to_lua[lua.source.start] = pos
else else
walk b, pos walk b, pos
pos += #b pos += #tostring(b)
walk self, 1 walk self, 1
return { return {
nomsu_filename:@source.filename nomsu_filename:@source.filename

View File

@ -159,19 +159,6 @@ immediately
compile [nomsu] to: Lua value "nomsu" compile [nomsu] to: Lua value "nomsu"
compile [%var as lua identifier] to: Lua value "nomsu:var_to_lua_identifier(\(%var as lua expr))" compile [%var as lua identifier] to: Lua value "nomsu:var_to_lua_identifier(\(%var as lua expr))"
action [action %names metadata]
=lua "nomsu.action_metadata[ACTIONS[\%names]]"
# Get the source code for a function
action [help %action]
lua> ".."
local metadata = \(action %action metadata);
if not metadata then
print("Action not found: "..repr(\%action));
else
print(metadata.src or "<unknown source code>");
end
# Compiler tools # Compiler tools
immediately immediately
compile [run %code] to compile [run %code] to

View File

@ -163,7 +163,14 @@ immediately
# Unary operators # Unary operators
compile [- %] to: Lua value "(- \(% as lua expr))" compile [- %] to: Lua value "(- \(% as lua expr))"
compile [not %] to: Lua value "(not \(% as lua expr))" compile [not %] to: Lua value "(not \(% as lua expr))"
compile [length of %list] to: Lua value "(#\(%list as lua expr))" compile [length of %list] to
# A bit of a hack so that luajit works properly.
Lua value ".."
(function(l)
local mt = getmetatable(l);
if mt and mt.__len then return mt.__len(l) end
return #l
end)(\(%list as lua expr))
# Update operators # Update operators
immediately immediately

View File

@ -83,34 +83,36 @@ compile [define object %classname %class_body] to
%actions <- %line.value.2 %actions <- %line.value.2
%body <- %line.value.3 %body <- %line.value.3
lua> ".." lua> ".."
local stubs = {}; do
for i, action in ipairs(\%actions.value) do local stubs = {};
stubs[i] = nomsu:tree_to_named_stub(action); for i, action in ipairs(\%actions.value) do
end stubs[i] = nomsu:tree_to_named_stub(action);
local args = {};
for i,tok in ipairs(\%actions.value[1].value) do
if tok.type == "Var" then args[#args+1] = nomsu:var_to_lua_identifier(tok.value); end
end
local arg_set = {};
for i, arg in ipairs(args) do arg_set[arg] = true; end
local body_lua = nomsu:tree_to_lua(\%body);
body_lua:convert_to_statements();
body_lua:declare_locals();
local lua_fn_args = table.concat({"self", unpack(args)}, ", ");
local def_tree = nomsu.compilestack[#nomsu.compilestack];
local compiled_args = {};
for i, arg in ipairs(args) do
compiled_args[i] = "nomsu:tree_to_lua("..arg..")";
end
compiled_args = table.concat(compiled_args, "..', '..");
table.insert(\%methods, ([==[
%s[ %s] = function(%s)
%s
end end
]==]):format( local args = {};
\%class_identifier, repr(stubs[1]), lua_fn_args, for i,tok in ipairs(\%actions.value[1].value) do
body_lua)); if tok.type == "Var" then args[#args+1] = nomsu:var_to_lua_identifier(tok.value); end
end
local arg_set = {};
for i, arg in ipairs(args) do arg_set[arg] = true; end
local body_lua = nomsu:tree_to_lua(\%body);
body_lua:convert_to_statements();
body_lua:declare_locals();
local lua_fn_args = table.concat({"self", unpack(args)}, ", ");
local def_tree = nomsu.compilestack[#nomsu.compilestack];
local compiled_args = {};
for i, arg in ipairs(args) do
compiled_args[i] = "nomsu:tree_to_lua("..arg..")";
end
compiled_args = table.concat(compiled_args, "..', '..");
table.insert(\%methods, ([==[
%s[ %s] = function(%s)
%s
end
]==]):format(
\%class_identifier, repr(stubs[1]), lua_fn_args,
body_lua));
end
return return
Lua ".." Lua ".."

120
nomsu.lua
View File

@ -1,5 +1,35 @@
local re = require('re') if jit then
local lpeg = require('lpeg') package.path = "LPegLJ/src/?.lua;" .. tostring(package.path)
lpeg = require('lpeglj')
re = require('re')
bit32 = require('bit')
local _pairs, _ipairs = pairs, ipairs
pairs = function(x)
do
local mt = getmetatable(x)
if mt then
if mt.__pairs then
return mt.__pairs(x)
end
end
end
return _pairs(x)
end
ipairs = function(x)
do
local mt = getmetatable(x)
if mt then
if mt.__ipairs then
return mt.__ipairs(x)
end
end
end
return _ipairs(x)
end
else
re = require('re')
lpeg = require('lpeg')
end
lpeg.setmaxstack(10000) lpeg.setmaxstack(10000)
local P, R, V, S, Cg, C, Cp, B, Cmt local P, R, V, S, Cg, C, Cp, B, Cmt
P, R, V, S, Cg, C, Cp, B, Cmt = lpeg.P, lpeg.R, lpeg.V, lpeg.S, lpeg.Cg, lpeg.C, lpeg.Cp, lpeg.B, lpeg.Cmt P, R, V, S, Cg, C, Cp, B, Cmt = lpeg.P, lpeg.R, lpeg.V, lpeg.S, lpeg.Cg, lpeg.C, lpeg.Cp, lpeg.B, lpeg.Cmt
@ -41,7 +71,7 @@ FILE_CACHE = setmetatable({ }, {
if not (file) then if not (file) then
return nil return nil
end end
local contents = file:read("a") local contents = file:read("*a")
file:close() file:close()
self[filename] = contents self[filename] = contents
return contents return contents
@ -118,7 +148,7 @@ local NOMSU_DEFS
do do
local _with_0 = { } local _with_0 = { }
_with_0.Tuple = function(values) _with_0.Tuple = function(values)
return Tuple(table.unpack(values)) return Tuple(unpack(values))
end end
_with_0.DictEntry = function(k, v) _with_0.DictEntry = function(k, v)
return Types.DictEntry(k, v) return Types.DictEntry(k, v)
@ -343,7 +373,7 @@ do
if compile_fn == nil then if compile_fn == nil then
compile_fn = nil compile_fn = nil
end end
if #nomsu_code == 0 then if #tostring(nomsu_code) == 0 then
return nil return nil
end end
local tree = self:parse(nomsu_code) local tree = self:parse(nomsu_code)
@ -884,46 +914,35 @@ OPTIONS
break break
end end
local line = nil local line = nil
do _ = [=[ if metadata = nomsu.action_metadata[calling_fn.func]
local metadata = nomsu.action_metadata[calling_fn.func] filename, start, stop = metadata.source\match("([^:]*):([0-9]*),([0-9]*)")
if metadata then if filename
local filename, start, stop = metadata.source:match("([^:]*):([0-9]*),([0-9]*)") file = FILE_CACHE[filename]
if filename then line_no = 1
local file = FILE_CACHE[filename] for _ in file\sub(1,tonumber(start))\gmatch("\n") do line_no += 1
local line_no = 1 offending_statement = file\sub(tonumber(start),tonumber(stop))
for _ in file:sub(1, tonumber(start)):gmatch("\n") do if #offending_statement > 50
line_no = line_no + 1 offending_statement = offending_statement\sub(1,50).."..."
end offending_statement = colored.red(offending_statement)
local offending_statement = file:sub(tonumber(start), tonumber(stop)) line = colored.yellow(filename..":"..tostring(line_no).."\n "..offending_statement)
if #offending_statement > 50 then else
offending_statement = offending_statement:sub(1, 50) .. "..." line = colored.yellow(metadata.source)
end name = colored.bright(colored.yellow(metadata.aliases[1]))
offending_statement = colored.red(offending_statement)
line = colored.yellow(filename .. ":" .. tostring(line_no) .. "\n " .. offending_statement)
else else
line = colored.yellow(metadata.source) if calling_fn.istailcall and not name
end name = "<tail call>"
name = colored.bright(colored.yellow(metadata.aliases[1])) if calling_fn.short_src == "./nomsu.moon" and line_table
else char = line_table[calling_fn.currentline]
if calling_fn.istailcall and not name then line_num = 1
name = "<tail call>" for _ in nomsu_source\sub(1,char)\gmatch("\n") do line_num += 1
end line = colored.cyan("#{calling_fn.short_src}:#{line_num}")
if calling_fn.short_src == "./nomsu.moon" and line_table then name = colored.bright(colored.cyan(name or "???"))
local char = line_table[calling_fn.currentline] else
local line_num = 1 line = colored.blue("#{calling_fn.short_src}:#{calling_fn.currentline}")
for _ in nomsu_source:sub(1, char):gmatch("\n") do name = colored.bright(colored.blue(name or "???"))
line_num = line_num + 1 _from = colored.dim colored.white "|"
end io.stderr\write(("%32s %s %s\n")\format(name, _from, line))
line = colored.cyan(tostring(calling_fn.short_src) .. ":" .. tostring(line_num)) ]=]
name = colored.bright(colored.cyan(name or "???"))
else
line = colored.blue(tostring(calling_fn.short_src) .. ":" .. tostring(calling_fn.currentline))
name = colored.bright(colored.blue(name or "???"))
end
end
end
local _from = colored.dim(colored.white("|"))
io.stderr:write(("%32s %s %s\n"):format(name, _from, line))
_continue_0 = true _continue_0 = true
until true until true
if not _continue_0 then if not _continue_0 then
@ -1064,13 +1083,12 @@ OPTIONS
print_err_msg(error_message) print_err_msg(error_message)
return os.exit(false, true) return os.exit(false, true)
end end
do local ldt
local ldt = require('ldt') ok, ldt = pcall(require, 'ldt')
if ldt then if ok then
ldt.guard(run) ldt.guard(run)
else else
xpcall(run, err_hand) xpcall(run, err_hand)
end
end end
end end
return NomsuCompiler return NomsuCompiler

View File

@ -10,8 +10,30 @@
-- nomsu:run(your_nomsu_code) -- nomsu:run(your_nomsu_code)
-- Or from the command line: -- Or from the command line:
-- lua nomsu.lua [input_file [output_file or -]] -- lua nomsu.lua [input_file [output_file or -]]
re = require 're' export lpeg, re
lpeg = require 'lpeg' if jit
package.path = "LPegLJ/src/?.lua;#{package.path}"
lpeg = require 'lpeglj'
re = require 're'
export bit32
bit32 = require('bit')
_pairs, _ipairs = pairs, ipairs
export pairs, ipairs
pairs = (x)->
if mt = getmetatable(x)
if mt.__pairs
return mt.__pairs(x)
return _pairs(x)
ipairs = (x)->
if mt = getmetatable(x)
if mt.__ipairs
return mt.__ipairs(x)
return _ipairs(x)
else
re = require 're'
lpeg = require 'lpeg'
lpeg.setmaxstack 10000 lpeg.setmaxstack 10000
{:P,:R,:V,:S,:Cg,:C,:Cp,:B,:Cmt} = lpeg {:P,:R,:V,:S,:Cg,:C,:Cp,:B,:Cmt} = lpeg
utils = require 'utils' utils = require 'utils'
@ -44,7 +66,7 @@ FILE_CACHE = setmetatable {}, {
__index: (filename)=> __index: (filename)=>
file = io.open(filename) file = io.open(filename)
return nil unless file return nil unless file
contents = file\read("a") contents = file\read("*a")
file\close! file\close!
self[filename] = contents self[filename] = contents
return contents return contents
@ -111,7 +133,7 @@ Types = require "nomsu_tree"
NOMSU_DEFS = with {} NOMSU_DEFS = with {}
-- Newline supports either windows-style CR+LF or unix-style LF -- Newline supports either windows-style CR+LF or unix-style LF
.Tuple = (values)-> .Tuple = (values)->
return Tuple(table.unpack(values)) return Tuple(unpack(values))
.DictEntry = (k,v) -> Types.DictEntry(k,v) .DictEntry = (k,v) -> Types.DictEntry(k,v)
.nl = P("\r")^-1 * P("\n") .nl = P("\r")^-1 * P("\n")
.ws = S(" \t") .ws = S(" \t")
@ -308,7 +330,7 @@ class NomsuCompiler
_nomsu_chunk_counter = 0 _nomsu_chunk_counter = 0
run: (nomsu_code, compile_fn=nil)=> run: (nomsu_code, compile_fn=nil)=>
if #nomsu_code == 0 then return nil if #tostring(nomsu_code) == 0 then return nil
tree = @parse(nomsu_code) tree = @parse(nomsu_code)
assert tree, "Failed to parse: #{nomsu_code}" assert tree, "Failed to parse: #{nomsu_code}"
assert tree.type == "File", "Attempt to run non-file: #{tree.type}" assert tree.type == "File", "Attempt to run non-file: #{tree.type}"
@ -629,6 +651,7 @@ OPTIONS
name = calling_fn.name name = calling_fn.name
if name == "run_lua_fn" then continue if name == "run_lua_fn" then continue
line = nil line = nil
[=[
if metadata = nomsu.action_metadata[calling_fn.func] if metadata = nomsu.action_metadata[calling_fn.func]
filename, start, stop = metadata.source\match("([^:]*):([0-9]*),([0-9]*)") filename, start, stop = metadata.source\match("([^:]*):([0-9]*),([0-9]*)")
if filename if filename
@ -657,6 +680,7 @@ OPTIONS
name = colored.bright(colored.blue(name or "???")) name = colored.bright(colored.blue(name or "???"))
_from = colored.dim colored.white "|" _from = colored.dim colored.white "|"
io.stderr\write(("%32s %s %s\n")\format(name, _from, line)) io.stderr\write(("%32s %s %s\n")\format(name, _from, line))
]=]
io.stderr\flush! io.stderr\flush!
run = -> run = ->
@ -763,7 +787,8 @@ OPTIONS
--ProFi = require 'ProFi' --ProFi = require 'ProFi'
--ProFi\start() --ProFi\start()
if ldt = require('ldt') ok, ldt = pcall(require,'ldt')
if ok
ldt.guard run ldt.guard run
else xpcall(run, err_hand) else xpcall(run, err_hand)
--ProFi\stop() --ProFi\stop()

View File

@ -1,6 +1,4 @@
local utils = require('utils') local utils = require('utils')
local re = require('re')
local lpeg = require('lpeg')
local repr, stringify, min, max, equivalent, set, is_list, sum local repr, stringify, min, max, equivalent, set, is_list, sum
repr, stringify, min, max, equivalent, set, is_list, sum = utils.repr, utils.stringify, utils.min, utils.max, utils.equivalent, utils.set, utils.is_list, utils.sum repr, stringify, min, max, equivalent, set, is_list, sum = utils.repr, utils.stringify, utils.min, utils.max, utils.equivalent, utils.set, utils.is_list, utils.sum
local immutable = require('immutable') local immutable = require('immutable')

View File

@ -1,8 +1,6 @@
-- This file contains the datastructures used to represent parsed Nomsu syntax trees, -- This file contains the datastructures used to represent parsed Nomsu syntax trees,
-- as well as the logic for converting them to Lua code. -- as well as the logic for converting them to Lua code.
utils = require 'utils' utils = require 'utils'
re = require 're'
lpeg = require 'lpeg'
{:repr, :stringify, :min, :max, :equivalent, :set, :is_list, :sum} = utils {:repr, :stringify, :min, :max, :equivalent, :set, :is_list, :sum} = utils
immutable = require 'immutable' immutable = require 'immutable'
{:insert, :remove, :concat} = table {:insert, :remove, :concat} = table

View File

@ -1,8 +1,5 @@
--
-- A collection of helper utility functions -- A collection of helper utility functions
-- --
local lpeg, re = require("lpeg"), require("re")
local function is_list(t) local function is_list(t)
if type(t) ~= 'table' then if type(t) ~= 'table' then
return false return false

View File

@ -13,10 +13,10 @@ local function uuid()
-- Set the four most significant bits (bits 12 through 15) of the -- Set the four most significant bits (bits 12 through 15) of the
-- time_hi_and_version field to the 4-bit version number from -- time_hi_and_version field to the 4-bit version number from
-- Section 4.1.3. -- Section 4.1.3.
bytes[3] = bit32.bor(bytes[3], 0x4000) bytes[3] = bytes[3] + 0x4000
-- Set the two most significant bits (bits 6 and 7) of the -- Set the two most significant bits (bits 6 and 7) of the
-- clock_seq_hi_and_reserved to zero and one, respectively. -- clock_seq_hi_and_reserved to zero and one, respectively.
bytes[4] = bit32.bor(bytes[4], 0xC0) bytes[4] = bytes[4] + 0xC0
return ("%08x-%04x-%04x-%02x%02x-%6x%6x"):format(unpack(bytes)) return ("%08x-%04x-%04x-%02x%02x-%6x%6x"):format(unpack(bytes))
end end