2018-04-11 20:05:12 -07:00
|
|
|
{:insert, :remove, :concat} = table
|
|
|
|
immutable = require 'immutable'
|
2018-04-18 15:28:46 -07:00
|
|
|
local Lua, Source
|
2018-04-11 21:07:13 -07:00
|
|
|
export LINE_STARTS
|
2018-04-11 20:05:12 -07:00
|
|
|
|
2018-04-18 15:28:46 -07:00
|
|
|
Source = immutable {"filename","start","stop"}, {
|
|
|
|
name:"Source"
|
2018-04-13 14:54:35 -07:00
|
|
|
__new: (filename, start, stop)=>
|
2018-04-18 15:28:46 -07:00
|
|
|
if stop then assert(start <= stop, "Invalid range: #{start}, #{stop}")
|
|
|
|
return filename, start, stop
|
|
|
|
__tostring: =>
|
|
|
|
if @stop
|
|
|
|
"Source(\"#{@filename}\", #{@start}, #{@stop})"
|
|
|
|
else
|
|
|
|
"Source(\"#{@filename}\", #{@start})"
|
2018-04-11 20:05:12 -07:00
|
|
|
__lt: (other)=>
|
2018-04-13 14:54:35 -07:00
|
|
|
assert(@filename == other.filename, "Cannot compare sources from different files")
|
2018-04-11 20:05:12 -07:00
|
|
|
return if @start == other.start
|
2018-04-18 15:28:46 -07:00
|
|
|
(@stop or @start) < (other.stop or other.start)
|
2018-04-11 20:05:12 -07:00
|
|
|
else @start < other.start
|
|
|
|
__le: (other)=>
|
2018-04-13 14:54:35 -07:00
|
|
|
assert(@filename == other.filename, "Cannot compare sources from different files")
|
2018-04-11 20:05:12 -07:00
|
|
|
return if @start == other.start
|
2018-04-18 15:28:46 -07:00
|
|
|
(@stop or @start) <= (other.stop or other.start)
|
2018-04-11 20:05:12 -07:00
|
|
|
else @start <= other.start
|
2018-04-18 15:28:46 -07:00
|
|
|
__add: (offset)=>
|
|
|
|
if type(self) == 'number'
|
|
|
|
offset, self = self, offset
|
|
|
|
else assert(type(offset) == 'number', "Cannot add Source and #{type(offset)}")
|
|
|
|
return Source(@filename, @start+offset, @stop)
|
|
|
|
sub: (start, stop)=>
|
|
|
|
if not @stop
|
|
|
|
assert(not stop, "cannot subscript non-range with range")
|
|
|
|
assert(start > 0, "cannot subscript non-range with negative index")
|
|
|
|
return Source(@filename, @start + (start or 0))
|
|
|
|
else
|
|
|
|
start or= 1
|
|
|
|
if start < 0 then start = @stop + start + 1
|
|
|
|
stop or= -1
|
|
|
|
if stop < 0 then stop = @stop + stop + 1
|
|
|
|
return Source(@filename, @start + start - 1, @start + stop - 1)
|
|
|
|
get_text: =>
|
|
|
|
FILE_CACHE[@filename]\sub(@start,@stop)
|
2018-04-11 20:05:12 -07:00
|
|
|
get_line_number: =>
|
|
|
|
-- TODO: do a binary search if this is actually slow, which I doubt
|
2018-04-17 14:18:23 -07:00
|
|
|
src = FILE_CACHE[@filename]
|
|
|
|
line_starts = LINE_STARTS[src]
|
2018-04-11 20:05:12 -07:00
|
|
|
start_line = 1
|
2018-04-17 14:18:23 -07:00
|
|
|
while (line_starts[start_line+1] or math.huge) <= @start
|
2018-04-11 20:05:12 -07:00
|
|
|
start_line += 1
|
|
|
|
stop_line = start_line
|
2018-04-17 14:18:23 -07:00
|
|
|
while (line_starts[stop_line+1] or math.huge) <= @stop
|
2018-04-11 20:05:12 -07:00
|
|
|
stop_line += 1
|
|
|
|
return start_line, stop_line
|
2018-04-13 14:54:35 -07:00
|
|
|
get_line: => "#{@filename}:#{@get_line_number!}"
|
2018-04-11 20:05:12 -07:00
|
|
|
get_line_range: =>
|
2018-04-12 20:39:17 -07:00
|
|
|
start_line, stop_line = @get_line_number!
|
2018-04-11 20:05:12 -07:00
|
|
|
return if stop_line == start_line
|
2018-04-13 14:54:35 -07:00
|
|
|
"#{@filename}:#{start_line}"
|
|
|
|
else "#{@filename}:#{start_line}-#{stop_line}"
|
2018-04-11 20:05:12 -07:00
|
|
|
}
|
|
|
|
|
2018-04-18 15:28:46 -07:00
|
|
|
class Code
|
2018-04-11 21:07:13 -07:00
|
|
|
new: (@source, ...)=>
|
2018-04-13 14:54:35 -07:00
|
|
|
if type(@source) == 'string'
|
|
|
|
filename,start,stop = @source\match("^(.-)[(%d+):(%d+)]$")
|
2018-04-18 15:28:46 -07:00
|
|
|
@source = Source(filename, tonumber(start), tonumber(stop))
|
2018-04-11 20:05:12 -07:00
|
|
|
@bits = {...}
|
2018-04-18 15:28:46 -07:00
|
|
|
|
|
|
|
clone: =>
|
|
|
|
cls = @__class
|
|
|
|
copy = cls(@source, unpack(@bits))
|
|
|
|
copy.is_value = @is_value
|
|
|
|
for k,v in pairs @free_vars
|
|
|
|
copy.free_vars[k] = v
|
|
|
|
return copy
|
|
|
|
|
|
|
|
__tostring: =>
|
|
|
|
buff = {}
|
|
|
|
for i,b in ipairs @bits
|
|
|
|
buff[#buff+1] = tostring(b)
|
|
|
|
ret = concat(buff, "")
|
|
|
|
return ret
|
|
|
|
|
|
|
|
__len: =>
|
|
|
|
len = 0
|
|
|
|
for b in *@bits
|
|
|
|
len += #b
|
|
|
|
return len
|
|
|
|
|
|
|
|
sub: (start,stop)=>
|
|
|
|
str = tostring(self)\sub(start,stop)
|
|
|
|
cls = @__class
|
|
|
|
return cls(@source\sub(start-@source.start+1,stop-@source.stop+1), str)
|
|
|
|
|
|
|
|
append: (...)=>
|
|
|
|
n = select("#",...)
|
|
|
|
bits = @bits
|
|
|
|
for i=1,n
|
|
|
|
bits[#bits+1] = select(i, ...)
|
|
|
|
|
|
|
|
prepend: (...)=>
|
|
|
|
n = select("#",...)
|
|
|
|
bits = @bits
|
|
|
|
for i=#bits+n,n+1,-1
|
|
|
|
bits[i] = bits[i-n]
|
|
|
|
for i=1,n
|
|
|
|
bits[i] = select(i, ...)
|
|
|
|
|
|
|
|
class Lua extends Code
|
|
|
|
new: (...)=>
|
|
|
|
super ...
|
2018-04-11 20:05:12 -07:00
|
|
|
@free_vars = {}
|
2018-04-17 14:18:23 -07:00
|
|
|
@is_value = false
|
2018-04-11 20:05:12 -07:00
|
|
|
|
2018-04-17 14:18:23 -07:00
|
|
|
@Value = (...)->
|
|
|
|
lua = Lua(...)
|
|
|
|
lua.is_value = true
|
|
|
|
return lua
|
|
|
|
|
|
|
|
add_free_vars: (...)=>
|
2018-04-11 20:05:12 -07:00
|
|
|
seen = {[v]:true for v in *@free_vars}
|
2018-04-17 14:18:23 -07:00
|
|
|
for i=1,select("#",...)
|
|
|
|
var = select(i, ...)
|
2018-04-11 20:05:12 -07:00
|
|
|
unless seen[var]
|
|
|
|
@free_vars[#@free_vars+1] = var
|
|
|
|
seen[var] = true
|
|
|
|
|
2018-04-17 14:18:23 -07:00
|
|
|
convert_to_statements: (prefix="", suffix=";")=>
|
|
|
|
unless @is_value
|
|
|
|
return
|
|
|
|
if prefix != ""
|
|
|
|
@prepend prefix
|
|
|
|
if suffix != ""
|
|
|
|
@append suffix
|
2018-04-11 20:05:12 -07:00
|
|
|
|
|
|
|
declare_locals: (skip={})=>
|
|
|
|
if next(skip) == 1
|
|
|
|
skip = {[s]:true for s in *skip}
|
2018-04-12 18:01:51 -07:00
|
|
|
if #@free_vars > 0
|
|
|
|
@prepend "local #{concat @free_vars, ", "};\n"
|
2018-04-11 20:05:12 -07:00
|
|
|
for var in *@free_vars do skip[var] = true
|
|
|
|
for bit in *@bits
|
|
|
|
if type(bit) == Lua
|
|
|
|
bit\declare_locals(skip)
|
|
|
|
|
|
|
|
__tostring: =>
|
|
|
|
buff = {}
|
2018-04-13 15:29:16 -07:00
|
|
|
for i,b in ipairs @bits
|
2018-04-11 21:07:13 -07:00
|
|
|
buff[#buff+1] = tostring(b)
|
2018-04-12 18:01:51 -07:00
|
|
|
ret = concat(buff, "")
|
|
|
|
return ret
|
2018-04-11 20:05:12 -07:00
|
|
|
|
|
|
|
__len: =>
|
|
|
|
len = 0
|
|
|
|
for b in *@bits
|
|
|
|
len += #b
|
|
|
|
return len
|
2018-04-17 14:18:23 -07:00
|
|
|
|
2018-04-11 20:05:12 -07:00
|
|
|
append: (...)=>
|
|
|
|
n = select("#",...)
|
|
|
|
bits = @bits
|
|
|
|
for i=1,n
|
2018-04-18 15:28:46 -07:00
|
|
|
bit = select(i, ...)
|
|
|
|
bits[#bits+1] = bit
|
|
|
|
if type(bit) != 'string' and not bit.is_value and #@bits > 0
|
|
|
|
bits[#bits+1] = "\n"
|
2018-04-11 20:05:12 -07:00
|
|
|
|
|
|
|
prepend: (...)=>
|
|
|
|
n = select("#",...)
|
|
|
|
bits = @bits
|
2018-04-18 15:28:46 -07:00
|
|
|
insert_index = 1
|
|
|
|
-- TODO: optimize?
|
2018-04-11 20:05:12 -07:00
|
|
|
for i=1,n
|
2018-04-18 15:28:46 -07:00
|
|
|
bit = select(i, ...)
|
|
|
|
insert bits, insert_index, bit
|
|
|
|
insert_index += 1
|
|
|
|
if type(bit) != 'string' and not bit.is_value and insert_index < #@bits + 1
|
|
|
|
insert bits, insert_index, "\n"
|
|
|
|
insert_index += 1
|
2018-04-11 20:05:12 -07:00
|
|
|
|
2018-04-18 15:28:46 -07:00
|
|
|
make_offset_table: =>
|
2018-04-11 20:05:12 -07:00
|
|
|
-- Return a mapping from output (lua) character number to input (nomsu) character number
|
2018-04-18 15:28:46 -07:00
|
|
|
lua_chunkname = tostring(@source)..".lua"
|
2018-04-11 20:05:12 -07:00
|
|
|
lua_str = tostring(self)
|
|
|
|
metadata = {
|
2018-04-13 14:54:35 -07:00
|
|
|
nomsu_filename:@source.filename
|
2018-04-11 20:05:12 -07:00
|
|
|
lua_filename:lua_chunkname, lua_file:lua_str
|
|
|
|
lua_to_nomsu: {}, nomsu_to_lua: {}
|
|
|
|
}
|
|
|
|
lua_offset = 1
|
|
|
|
walk = (lua)->
|
|
|
|
if type(lua) == 'string'
|
|
|
|
lua_offset += #lua
|
|
|
|
else
|
|
|
|
lua_start = lua_offset
|
2018-04-11 21:07:13 -07:00
|
|
|
for b in *lua.bits
|
2018-04-11 20:05:12 -07:00
|
|
|
walk b
|
|
|
|
lua_stop = lua_offset
|
2018-04-18 15:28:46 -07:00
|
|
|
nomsu_src, lua_src = lua.source, Source(lua_chunkname, lua_start, lua_stop)
|
2018-04-11 20:05:12 -07:00
|
|
|
metadata.lua_to_nomsu[lua_src] = nomsu_src
|
|
|
|
metadata.nomsu_to_lua[nomsu_src] = lua_src
|
|
|
|
walk self
|
|
|
|
return lua_str, metadata
|
|
|
|
|
|
|
|
parenthesize: =>
|
2018-04-17 14:18:23 -07:00
|
|
|
if @is_value
|
|
|
|
@prepend "("
|
|
|
|
@append ")"
|
|
|
|
else
|
|
|
|
error "Cannot parenthesize lua statements"
|
2018-04-11 20:05:12 -07:00
|
|
|
|
2018-04-18 15:28:46 -07:00
|
|
|
class Nomsu extends Code
|
|
|
|
__tostring: =>
|
|
|
|
buff = {}
|
|
|
|
for i,b in ipairs @bits
|
|
|
|
buff[#buff+1] = tostring(b)
|
|
|
|
ret = concat(buff, "")
|
|
|
|
return ret
|
|
|
|
|
|
|
|
__len: =>
|
|
|
|
len = 0
|
|
|
|
for b in *@bits
|
|
|
|
len += #b
|
|
|
|
return len
|
|
|
|
|
|
|
|
return {:Code, :Nomsu, :Lua, :Source}
|