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-08-28 15:08:00 -07:00
|
|
|
unpack or= table.unpack
|
2018-06-18 15:44:29 -07:00
|
|
|
local LuaCode, NomsuCode, Source
|
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-08-28 15:08:00 -07:00
|
|
|
__tostring: => "@#{@filename}[#{@start}#{@stop and ':'..@stop or ''}]"
|
|
|
|
|
2018-09-18 19:48:58 -07:00
|
|
|
as_lua: => "Source(#{@filename\as_lua!}, #{@start}#{@stop and ', '..@stop or ''})"
|
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-11-09 16:40:36 -08:00
|
|
|
new: (...)=>
|
2018-07-10 17:34:39 -07:00
|
|
|
@bits = {}
|
2018-11-29 14:57:22 -08:00
|
|
|
@add(...)
|
2018-07-23 15:24:43 -07:00
|
|
|
|
2018-11-09 16:40:36 -08:00
|
|
|
@from: (source, ...)=>
|
|
|
|
inst = self(...)
|
|
|
|
if type(source) == 'string'
|
|
|
|
source = Source\from_string(source)
|
|
|
|
inst.source = source
|
|
|
|
return inst
|
|
|
|
|
2018-11-08 15:23:22 -08:00
|
|
|
@is_instance: (x)=> type(x) == 'table' and x.__class == @
|
|
|
|
|
2018-11-02 14:38:24 -07:00
|
|
|
text: =>
|
2018-07-23 15:24:43 -07:00
|
|
|
if @__str == nil
|
|
|
|
buff, indent = {}, 0
|
|
|
|
{:match, :gsub, :rep} = string
|
|
|
|
for i,b in ipairs @bits
|
|
|
|
if type(b) == 'string'
|
|
|
|
if spaces = match(b, "\n([ ]*)[^\n]*$")
|
|
|
|
indent = #spaces
|
|
|
|
else
|
2018-11-02 14:38:24 -07:00
|
|
|
b = b\text!
|
2018-07-23 15:24:43 -07:00
|
|
|
if indent > 0
|
|
|
|
b = gsub(b, "\n", "\n"..rep(" ", indent))
|
|
|
|
buff[#buff+1] = b
|
|
|
|
@__str = concat(buff, "")
|
|
|
|
return @__str
|
|
|
|
|
2019-03-09 15:52:23 -08:00
|
|
|
last: (n)=>
|
|
|
|
if @__str
|
|
|
|
return @__str\sub(-n, -1)
|
|
|
|
last = ""
|
|
|
|
for i=#@bits,1,-1
|
|
|
|
b = @bits[i]
|
|
|
|
last = (type(b) == 'string' and b\sub(-(n-#last)) or b\last(n-#last))..last
|
|
|
|
break if #last == n
|
|
|
|
return last
|
|
|
|
|
|
|
|
first: (n)=>
|
|
|
|
if @__str
|
|
|
|
return @__str\sub(1,n)
|
|
|
|
first = ""
|
|
|
|
for b in *@bits
|
|
|
|
first ..= type(b) == 'string' and b\sub(1,n-#first+1) or b\first(n-#first+1)
|
|
|
|
break if #first == n
|
|
|
|
return first
|
|
|
|
|
2018-11-02 14:38:24 -07:00
|
|
|
__tostring: => @text!
|
2018-09-28 22:15:06 -07:00
|
|
|
|
2018-09-18 19:48:58 -07:00
|
|
|
as_lua: =>
|
2018-11-09 16:40:36 -08:00
|
|
|
if @source
|
|
|
|
"#{@__class.__name}:from(#{concat {tostring(@source)\as_lua!, unpack([b\as_lua! for b in *@bits])}, ", "})"
|
|
|
|
else
|
|
|
|
"#{@__class.__name}(#{concat [b\as_lua! for b in *@bits], ", "})"
|
2018-08-28 15:08:00 -07:00
|
|
|
|
2019-03-09 15:52:23 -08:00
|
|
|
__len: =>
|
|
|
|
if @__str
|
|
|
|
return #@__str
|
|
|
|
len = 0
|
|
|
|
for b in *@bits
|
|
|
|
len += #b
|
|
|
|
return len
|
2018-08-28 15:08:00 -07:00
|
|
|
|
2018-11-02 14:38:24 -07:00
|
|
|
match: (...)=> @text!\match(...)
|
2018-08-28 15:08:00 -07:00
|
|
|
|
2018-11-02 14:38:24 -07:00
|
|
|
gmatch: (...)=> @text!\gmatch(...)
|
2018-07-19 20:41:31 -07:00
|
|
|
|
|
|
|
dirty: =>
|
|
|
|
@__str = nil
|
|
|
|
@_trailing_line_len = nil
|
2018-07-20 17:51:21 -07:00
|
|
|
-- Multi-line only goes from false->true, since there is no API for removing bits
|
|
|
|
@_is_multiline = nil if @_is_multiline == false
|
2018-04-24 20:16:46 -07:00
|
|
|
|
2018-11-29 14:57:22 -08:00
|
|
|
add: (...)=>
|
2018-04-18 15:28:46 -07:00
|
|
|
n = select("#",...)
|
2018-05-24 16:13:23 -07:00
|
|
|
match = string.match
|
2018-07-17 16:13:35 -07:00
|
|
|
bits = @bits
|
2018-04-18 15:28:46 -07:00
|
|
|
for i=1,n
|
2018-04-24 20:16:46 -07:00
|
|
|
b = select(i, ...)
|
2018-07-10 17:34:39 -07:00
|
|
|
assert(b, "code bit is nil")
|
2018-07-13 14:30:32 -07:00
|
|
|
assert(not Source\is_instance(b), "code bit is a Source")
|
2018-06-28 14:12:24 -07:00
|
|
|
if b == '' then continue
|
2018-08-27 13:38:58 -07:00
|
|
|
bits[#bits+1] = b
|
2018-07-19 20:41:31 -07:00
|
|
|
@dirty!
|
2018-07-17 16:13:35 -07:00
|
|
|
|
|
|
|
trailing_line_len: =>
|
|
|
|
if @_trailing_line_len == nil
|
2018-11-02 14:38:24 -07:00
|
|
|
@_trailing_line_len = #@text!\match("[^\n]*$")
|
2018-07-17 16:13:35 -07:00
|
|
|
return @_trailing_line_len
|
2018-07-19 20:41:31 -07:00
|
|
|
|
|
|
|
is_multiline: =>
|
|
|
|
if @_is_multiline == nil
|
|
|
|
match = string.match
|
|
|
|
@_is_multiline = false
|
|
|
|
for b in *@bits
|
2018-07-19 21:09:44 -07:00
|
|
|
if type(b) == 'string'
|
|
|
|
if match(b, '\n')
|
|
|
|
@_is_multiline = true
|
|
|
|
break
|
|
|
|
elseif b\is_multiline!
|
2018-07-19 20:41:31 -07:00
|
|
|
@_is_multiline = true
|
|
|
|
break
|
|
|
|
return @_is_multiline
|
|
|
|
|
2018-11-29 14:57:22 -08:00
|
|
|
concat_add: (values, joiner, wrapping_joiner)=>
|
2018-06-19 00:44:17 -07:00
|
|
|
wrapping_joiner or= joiner
|
2018-06-14 21:59:25 -07:00
|
|
|
match = string.match
|
2018-07-17 16:13:35 -07:00
|
|
|
bits = @bits
|
2018-06-19 00:44:17 -07:00
|
|
|
line_len = 0
|
2018-06-14 21:59:25 -07:00
|
|
|
for i=1,#values
|
|
|
|
b = values[i]
|
|
|
|
if i > 1
|
2018-06-19 00:44:17 -07:00
|
|
|
if line_len > 80
|
|
|
|
bits[#bits+1] = wrapping_joiner
|
|
|
|
line_len = 0
|
|
|
|
else
|
|
|
|
bits[#bits+1] = joiner
|
2018-06-14 21:59:25 -07:00
|
|
|
bits[#bits+1] = b
|
2018-11-17 14:38:05 -08:00
|
|
|
b.dirty = error if type(b) != 'string'
|
2019-03-09 15:52:23 -08:00
|
|
|
line = b\match("\n([^\n]*)$")
|
2018-07-17 16:13:35 -07:00
|
|
|
if line
|
2018-06-19 00:44:17 -07:00
|
|
|
line_len = #line
|
|
|
|
else
|
|
|
|
line_len += #b
|
2018-07-19 20:41:31 -07:00
|
|
|
@dirty!
|
2018-04-18 15:28:46 -07:00
|
|
|
|
|
|
|
prepend: (...)=>
|
|
|
|
n = select("#",...)
|
2018-07-17 16:13:35 -07:00
|
|
|
bits = @bits
|
2018-04-18 15:28:46 -07:00
|
|
|
for i=#bits+n,n+1,-1
|
|
|
|
bits[i] = bits[i-n]
|
|
|
|
for i=1,n
|
2018-07-20 17:51:21 -07:00
|
|
|
b = select(i, ...)
|
2018-11-17 14:38:05 -08:00
|
|
|
b.dirty = error if type(b) != 'string'
|
2018-08-27 13:38:58 -07:00
|
|
|
bits[i] = b
|
2018-07-19 20:41:31 -07:00
|
|
|
@dirty!
|
2018-04-18 15:28:46 -07:00
|
|
|
|
2018-07-23 15:24:43 -07:00
|
|
|
parenthesize: =>
|
|
|
|
@prepend "("
|
2018-11-29 14:57:22 -08:00
|
|
|
@add ")"
|
2018-07-23 15:24:43 -07:00
|
|
|
|
2018-06-18 15:44:29 -07:00
|
|
|
class LuaCode extends Code
|
2018-07-23 15:24:43 -07:00
|
|
|
__tostring: Code.__tostring
|
2018-09-18 19:48:58 -07:00
|
|
|
as_lua: Code.as_lua
|
2018-07-23 15:24:43 -07:00
|
|
|
__len: Code.__len
|
2018-04-18 15:28:46 -07:00
|
|
|
new: (...)=>
|
|
|
|
super ...
|
2018-04-11 20:05:12 -07:00
|
|
|
@free_vars = {}
|
|
|
|
|
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-06-18 18:10:59 -07:00
|
|
|
assert type(var) == 'string'
|
2018-04-11 20:05:12 -07:00
|
|
|
unless seen[var]
|
|
|
|
@free_vars[#@free_vars+1] = var
|
|
|
|
seen[var] = true
|
2018-07-19 20:41:31 -07:00
|
|
|
@dirty!
|
2018-04-19 17:23:44 -07:00
|
|
|
|
2019-01-01 15:05:58 -08:00
|
|
|
remove_free_vars: (vars=nil)=>
|
|
|
|
vars or= @get_free_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-18 18:10:59 -07:00
|
|
|
assert type(var) == 'string'
|
|
|
|
removals[var] = 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-18 18:10:59 -07:00
|
|
|
free_var = lua.free_vars[i]
|
|
|
|
if removals[free_var]
|
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-07-19 20:41:31 -07:00
|
|
|
@dirty!
|
2018-04-11 20:05:12 -07:00
|
|
|
|
2019-01-01 15:05:58 -08:00
|
|
|
get_free_vars: =>
|
|
|
|
vars, seen = {}, {}
|
|
|
|
gather_from = =>
|
|
|
|
for var in *@free_vars
|
|
|
|
unless seen[var]
|
|
|
|
seen[var] = true
|
|
|
|
vars[#vars+1] = var
|
|
|
|
for bit in *@bits
|
|
|
|
unless type(bit) == 'string'
|
|
|
|
gather_from bit
|
|
|
|
gather_from self
|
|
|
|
return vars
|
|
|
|
|
2018-04-19 17:23:44 -07:00
|
|
|
declare_locals: (to_declare=nil)=>
|
2019-01-01 15:05:58 -08:00
|
|
|
to_declare or= @get_free_vars!
|
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-18 18:10:59 -07:00
|
|
|
@prepend "local #{concat to_declare, ", "};\n"
|
2018-05-04 13:49:09 -07:00
|
|
|
return to_declare
|
2018-04-11 20:05:12 -07:00
|
|
|
|
2018-04-18 15:28:46 -07:00
|
|
|
make_offset_table: =>
|
2018-11-09 16:40:36 -08:00
|
|
|
assert @source, "This code doesn't have a source"
|
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-09-28 22:15:06 -07:00
|
|
|
pos += #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
|
2018-11-06 15:13:55 -08:00
|
|
|
lua_filename:tostring(@source)..".lua", lua_file:@text!
|
2018-04-20 16:23:53 -07:00
|
|
|
:lua_to_nomsu, :nomsu_to_lua
|
|
|
|
}
|
2018-04-11 20:05:12 -07:00
|
|
|
|
|
|
|
parenthesize: =>
|
2018-04-25 16:04:46 -07:00
|
|
|
@prepend "("
|
2018-11-29 14:57:22 -08:00
|
|
|
@add ")"
|
2018-04-25 16:04:46 -07:00
|
|
|
|
2018-07-23 15:24:43 -07:00
|
|
|
class NomsuCode extends Code
|
|
|
|
__tostring: Code.__tostring
|
2018-09-18 19:48:58 -07:00
|
|
|
as_lua: Code.as_lua
|
2018-07-23 15:24:43 -07:00
|
|
|
__len: Code.__len
|
|
|
|
|
2018-11-29 14:57:22 -08:00
|
|
|
Code.__base.add_1_joined_with = assert Code.__base.concat_add
|
|
|
|
Code.__base.add = assert Code.__base.add
|
2018-08-29 15:02:36 -07:00
|
|
|
|
2018-06-18 15:44:29 -07:00
|
|
|
return {:Code, :NomsuCode, :LuaCode, :Source}
|