2018-06-05 03:39:44 -07:00
|
|
|
-- This file contains objects that are used to track code positions and incrementally
|
|
|
|
-- build up generated code, while keeping track of where it came from, and managing
|
|
|
|
-- indentation levels.
|
2018-04-11 20:05:12 -07:00
|
|
|
{:insert, :remove, :concat} = table
|
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-06-12 15:12:27 -07:00
|
|
|
class Source
|
|
|
|
new: (@filename, @start, @stop)=>
|
|
|
|
|
|
|
|
@from_string: (str)=>
|
2018-05-30 17:20:22 -07:00
|
|
|
filename,start,stop = str\match("^@(.-)%[(%d+):(%d+)%]$")
|
2018-05-26 15:58:32 -07:00
|
|
|
unless filename
|
2018-05-30 17:20:22 -07:00
|
|
|
filename,start = str\match("^@(.-)%[(%d+)%]$")
|
2018-06-12 15:12:27 -07:00
|
|
|
return @(filename or str, tonumber(start or 1), tonumber(stop))
|
|
|
|
|
|
|
|
@is_instance: (x)=> type(x) == 'table' and x.__class == @
|
|
|
|
|
2018-04-18 15:28:46 -07:00
|
|
|
__tostring: =>
|
|
|
|
if @stop
|
2018-05-30 17:20:22 -07:00
|
|
|
"@#{@filename}[#{@start}:#{@stop}]"
|
2018-04-18 15:28:46 -07:00
|
|
|
else
|
2018-05-30 17:20:22 -07:00
|
|
|
"@#{@filename}[#{@start}]"
|
2018-06-12 15:12:27 -07:00
|
|
|
|
|
|
|
__eq: (other)=>
|
|
|
|
getmetatable(@) == getmetatable(other) and @filename == other.filename and @start == other.start and @stop == other.stop
|
|
|
|
|
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
|
2018-06-12 15:12:27 -07:00
|
|
|
|
2018-04-11 20:05:12 -07:00
|
|
|
__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-06-12 15:12:27 -07:00
|
|
|
|
2018-04-18 15:28:46 -07:00
|
|
|
__add: (offset)=>
|
|
|
|
if type(self) == 'number'
|
|
|
|
offset, self = self, offset
|
2018-05-29 11:13:58 -07:00
|
|
|
else if type(offset) != 'number' then error("Cannot add Source and #{type(offset)}")
|
2018-04-18 15:28:46 -07:00
|
|
|
return Source(@filename, @start+offset, @stop)
|
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-05-24 16:13:23 -07:00
|
|
|
@bits, @indents, @current_indent = {}, {}, 0
|
|
|
|
@append(...)
|
2018-04-13 14:54:35 -07:00
|
|
|
if type(@source) == 'string'
|
2018-05-26 15:58:32 -07:00
|
|
|
@source = Source\from_string(@source)
|
2018-05-29 16:14:53 -07:00
|
|
|
assert(@source and Source\is_instance(@source))
|
2018-04-24 20:16:46 -07:00
|
|
|
|
2018-04-18 15:28:46 -07:00
|
|
|
append: (...)=>
|
|
|
|
n = select("#",...)
|
2018-05-24 16:13:23 -07:00
|
|
|
bits, indents = @bits, @indents
|
|
|
|
match = string.match
|
2018-04-18 15:28:46 -07:00
|
|
|
for i=1,n
|
2018-04-24 20:16:46 -07:00
|
|
|
b = select(i, ...)
|
2018-06-12 20:06:33 -07:00
|
|
|
assert(b)
|
2018-04-24 20:16:46 -07:00
|
|
|
bits[#bits+1] = b
|
|
|
|
if type(b) == 'string'
|
2018-05-24 16:13:23 -07:00
|
|
|
if spaces = match(b, "\n([ ]*)[^\n]*$")
|
2018-04-24 20:16:46 -07:00
|
|
|
@current_indent = #spaces
|
|
|
|
elseif @current_indent != 0
|
2018-05-24 16:13:23 -07:00
|
|
|
indents[#bits] = @current_indent
|
2018-04-24 20:16:46 -07:00
|
|
|
@__str = nil
|
2018-06-14 21:59:25 -07:00
|
|
|
|
|
|
|
concat_append: (values, joiner)=>
|
|
|
|
bits, indents = @bits, @indents
|
|
|
|
match = string.match
|
|
|
|
for i=1,#values
|
|
|
|
b = values[i]
|
|
|
|
assert(b)
|
|
|
|
if i > 1
|
|
|
|
bits[#bits+1] = joiner
|
|
|
|
bits[#bits+1] = b
|
|
|
|
if type(b) == 'string'
|
|
|
|
if spaces = match(b, "\n([ ]*)[^\n]*$")
|
|
|
|
@current_indent = #spaces
|
|
|
|
elseif @current_indent != 0
|
|
|
|
indents[#bits] = @current_indent
|
|
|
|
@__str = nil
|
2018-04-18 15:28:46 -07:00
|
|
|
|
|
|
|
prepend: (...)=>
|
|
|
|
n = select("#",...)
|
2018-04-24 20:16:46 -07:00
|
|
|
bits, indents = @bits, @indents
|
2018-04-18 15:28:46 -07:00
|
|
|
for i=#bits+n,n+1,-1
|
|
|
|
bits[i] = bits[i-n]
|
|
|
|
for i=1,n
|
|
|
|
bits[i] = select(i, ...)
|
2018-04-24 20:16:46 -07:00
|
|
|
@current_indent = 0
|
|
|
|
for i,b in ipairs(bits)
|
|
|
|
if type(b) == 'string'
|
|
|
|
if spaces = b\match("\n([ ]*)[^\n]*$")
|
|
|
|
@current_indent = #spaces
|
|
|
|
elseif @current_indent != 0
|
|
|
|
indents[i] = @current_indent
|
|
|
|
else indents[i] = nil
|
|
|
|
@__str = nil
|
2018-04-18 15:28:46 -07:00
|
|
|
|
|
|
|
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-20 14:33:49 -07:00
|
|
|
@__str = nil
|
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
|
|
|
|
|
2018-05-04 13:49:09 -07:00
|
|
|
add_free_vars: (vars)=>
|
2018-05-16 19:08:16 -07:00
|
|
|
return unless #vars > 0
|
2018-04-11 20:05:12 -07:00
|
|
|
seen = {[v]:true for v in *@free_vars}
|
2018-05-04 13:49:09 -07:00
|
|
|
for var in *vars
|
2018-04-11 20:05:12 -07:00
|
|
|
unless seen[var]
|
|
|
|
@free_vars[#@free_vars+1] = var
|
|
|
|
seen[var] = true
|
2018-04-20 14:33:49 -07:00
|
|
|
@__str = nil
|
2018-04-19 17:23:44 -07:00
|
|
|
|
2018-05-04 13:49:09 -07:00
|
|
|
remove_free_vars: (vars)=>
|
2018-05-16 19:08:16 -07:00
|
|
|
return unless #vars > 0
|
2018-04-19 17:23:44 -07:00
|
|
|
removals = {}
|
2018-05-04 13:49:09 -07:00
|
|
|
for var in *vars
|
2018-06-12 18:04:18 -07:00
|
|
|
removals[var[1]] = true
|
2018-04-20 16:23:53 -07:00
|
|
|
|
|
|
|
stack = {self}
|
|
|
|
while #stack > 0
|
|
|
|
lua, stack[#stack] = stack[#stack], nil
|
|
|
|
for i=#lua.free_vars,1,-1
|
2018-06-12 18:04:18 -07:00
|
|
|
if removals[lua.free_vars[i][1]]
|
2018-04-20 16:23:53 -07:00
|
|
|
remove lua.free_vars, i
|
|
|
|
for b in *lua.bits
|
2018-04-19 17:23:44 -07:00
|
|
|
if type(b) != 'string'
|
2018-04-20 16:23:53 -07:00
|
|
|
stack[#stack+1] = b
|
2018-04-20 14:33:49 -07:00
|
|
|
@__str = nil
|
2018-04-11 20:05:12 -07:00
|
|
|
|
2018-05-15 14:53:37 -07:00
|
|
|
as_statements: (prefix="", suffix=";")=>
|
2018-04-17 14:18:23 -07:00
|
|
|
unless @is_value
|
2018-05-15 14:53:37 -07:00
|
|
|
return self
|
|
|
|
statements = Lua(@source)
|
2018-04-17 14:18:23 -07:00
|
|
|
if prefix != ""
|
2018-05-15 14:53:37 -07:00
|
|
|
statements\append prefix
|
|
|
|
statements\append self
|
2018-04-17 14:18:23 -07:00
|
|
|
if suffix != ""
|
2018-05-15 14:53:37 -07:00
|
|
|
statements\append suffix
|
|
|
|
return statements
|
2018-04-11 20:05:12 -07:00
|
|
|
|
2018-04-19 17:23:44 -07:00
|
|
|
declare_locals: (to_declare=nil)=>
|
|
|
|
if to_declare == nil
|
|
|
|
to_declare, seen = {}, {}
|
|
|
|
gather_from = =>
|
|
|
|
for var in *@free_vars
|
|
|
|
unless seen[var]
|
|
|
|
seen[var] = true
|
|
|
|
to_declare[#to_declare+1] = var
|
|
|
|
for bit in *@bits
|
|
|
|
if bit.__class == Lua
|
|
|
|
gather_from bit
|
|
|
|
gather_from self
|
2018-04-18 17:41:40 -07:00
|
|
|
if #to_declare > 0
|
2018-05-04 13:49:09 -07:00
|
|
|
@remove_free_vars to_declare
|
2018-06-12 20:06:33 -07:00
|
|
|
@prepend "local #{concat [type(v) == 'string' and v or string.as_lua_id(v[1]) for v in *to_declare], ", "};\n"
|
2018-05-04 13:49:09 -07:00
|
|
|
return to_declare
|
2018-04-11 20:05:12 -07:00
|
|
|
|
|
|
|
__tostring: =>
|
2018-04-20 14:33:49 -07:00
|
|
|
if @__str == nil
|
2018-04-24 20:16:46 -07:00
|
|
|
buff, indents = {}, @indents
|
2018-04-20 14:33:49 -07:00
|
|
|
for i,b in ipairs @bits
|
2018-04-24 20:16:46 -07:00
|
|
|
b = tostring(b)
|
|
|
|
if indents[i]
|
|
|
|
b = b\gsub("\n", "\n"..((" ")\rep(indents[i])))
|
|
|
|
buff[#buff+1] = b
|
2018-04-20 14:33:49 -07:00
|
|
|
@__str = concat(buff, "")
|
|
|
|
return @__str
|
2018-04-11 20:05:12 -07:00
|
|
|
|
|
|
|
__len: =>
|
2018-04-24 20:16:46 -07:00
|
|
|
#tostring(self)
|
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-20 16:23:53 -07:00
|
|
|
lua_to_nomsu, nomsu_to_lua = {}, {}
|
2018-04-20 14:33:49 -07:00
|
|
|
walk = (lua, pos)->
|
2018-04-18 17:41:40 -07:00
|
|
|
for b in *lua.bits
|
|
|
|
if type(b) == 'string'
|
2018-04-20 16:23:53 -07:00
|
|
|
if lua.source
|
|
|
|
lua_to_nomsu[pos] = lua.source.start
|
|
|
|
nomsu_to_lua[lua.source.start] = pos
|
2018-04-18 17:41:40 -07:00
|
|
|
else
|
2018-04-20 16:23:53 -07:00
|
|
|
walk b, pos
|
2018-05-09 13:34:33 -07:00
|
|
|
pos += #tostring(b)
|
2018-04-20 14:33:49 -07:00
|
|
|
walk self, 1
|
2018-04-20 16:23:53 -07:00
|
|
|
return {
|
|
|
|
nomsu_filename:@source.filename
|
|
|
|
lua_filename:tostring(@source)..".lua", lua_file:@stringify!
|
|
|
|
:lua_to_nomsu, :nomsu_to_lua
|
|
|
|
}
|
2018-04-11 20:05:12 -07:00
|
|
|
|
|
|
|
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: =>
|
2018-04-24 20:16:46 -07:00
|
|
|
if @__str == nil
|
|
|
|
buff, indents = {}, @indents
|
|
|
|
for i,b in ipairs @bits
|
|
|
|
b = tostring(b)
|
|
|
|
if indents[i]
|
|
|
|
b = b\gsub("\n", "\n"..((" ")\rep(indents[i])))
|
|
|
|
buff[#buff+1] = b
|
|
|
|
@__str = concat(buff, "")
|
|
|
|
return @__str
|
2018-04-18 15:28:46 -07:00
|
|
|
|
|
|
|
__len: =>
|
2018-04-24 20:16:46 -07:00
|
|
|
#tostring(self)
|
2018-04-18 15:28:46 -07:00
|
|
|
|
2018-04-25 16:04:46 -07:00
|
|
|
parenthesize: =>
|
|
|
|
@prepend "("
|
|
|
|
@append ")"
|
|
|
|
|
2018-04-18 15:28:46 -07:00
|
|
|
return {:Code, :Nomsu, :Lua, :Source}
|