diff options
-rw-r--r-- | script/core/noder.lua | 227 | ||||
-rw-r--r-- | script/parser/guide.lua | 8 | ||||
-rw-r--r-- | script/utility.lua | 27 | ||||
-rw-r--r-- | test.lua | 2 |
4 files changed, 194 insertions, 70 deletions
diff --git a/script/core/noder.lua b/script/core/noder.lua index 0ae48965..9a641e3b 100644 --- a/script/core/noder.lua +++ b/script/core/noder.lua @@ -92,37 +92,47 @@ local function getMethodNode(source) end end ----获取语法树单元的key ----@param source parser.guide.object ----@return string? key ----@return parser.guide.object? node -local function getKey(source) - if source.type == 'local' then +local getKey +local getKeyMap = util.switch() + : case 'local' + : call(function (source) if source.parent.type == 'funcargs' then return 'p:' .. source.start, nil end return 'l:' .. source.start, nil - elseif source.type == 'setlocal' - or source.type == 'getlocal' then + end) + : case 'setlocal' + : case 'getlocal' + : call(function (source) return getKey(source.node) - elseif source.type == 'setglobal' - or source.type == 'getglobal' then + end) + : case 'setglobal' + : case 'getglobal' + : call(function (source) local node = source.node if node.tag == '_ENV' then return ('%q'):format(source[1] or ''), nil else return ('%q'):format(source[1] or ''), node end - elseif source.type == 'getfield' - or source.type == 'setfield' then + end) + : case 'getfield' + : case 'setfield' + : call(function (source) return ('%q'):format(source.field and source.field[1] or ''), source.node - elseif source.type == 'tablefield' then + end) + : case 'tablefield' + : call(function (source) return ('%q'):format(source.field and source.field[1] or ''), source.parent - elseif source.type == 'getmethod' - or source.type == 'setmethod' then + end) + : case 'getmethod' + : case 'setmethod' + : call(function (source) return ('%q'):format(source.method and source.method[1] or ''), source.node - elseif source.type == 'setindex' - or source.type == 'getindex' then + end) + : case 'setindex' + : case 'getindex' + : call(function (source) local index = source.index if not index then return INDEX_CHAR, source.node @@ -135,7 +145,9 @@ local function getKey(source) else return INDEX_CHAR, source.node end - elseif source.type == 'tableindex' then + end) + : case 'tableindex' + : call(function (source) local index = source.index if not index then return ANY_FIELD_CHAR, source.parent @@ -149,38 +161,66 @@ local function getKey(source) and index.type ~= 'table' then return ANY_FIELD_CHAR, source.parent end - elseif source.type == 'tableexp' then + end) + : case 'tableexp' + : call(function (source) return tostring(source.tindex), source.parent - elseif source.type == 'table' then + end) + : case 'table' + : call(function (source) return 't:' .. source.start, nil - elseif source.type == 'label' then + end) + : case 'label' + : call(function (source) return 'l:' .. source.start, nil - elseif source.type == 'goto' then + end) + : case 'goto' + : call(function (source) if source.node then return 'l:' .. source.node.start, nil end return nil, nil - elseif source.type == 'function' then + end) + : case 'function' + : call(function (source) return 'f:' .. source.start, nil - elseif source.type == 'string' then + end) + : case 'string' + : call(function (source) return 'str:', nil - elseif source.type == 'integer' then - return 'int:' - elseif source.type == 'number' then - return 'num:' - elseif source.type == 'boolean' then - return 'bool:' - elseif source.type == 'nil' then + end) + : case 'integer' + : call(function (source) + return 'int:', nil + end) + : case 'number' + : call(function (source) + return 'num:', nil + end) + : case 'boolean' + : call(function (source) + return 'bool:', nil + end) + : case 'nil' + : call(function (source) return 'nil:', nil - elseif source.type == '...' then + end) + : case '...' + : call(function (source) return 'va:' .. source.start, nil - elseif source.type == 'varargs' then + end) + : case 'varargs' + : call(function (source) if source.node then return 'va:' .. source.node.start, nil end - elseif source.type == 'select' then + end) + : case 'select' + : call(function (source) return ('s:%d%s%d'):format(source.start, RETURN_INDEX, source.sindex) - elseif source.type == 'call' then + end) + : case 'call' + : call(function (source) local node = source.node if node.special == 'rawget' or node.special == 'rawset' then @@ -198,49 +238,83 @@ local function getKey(source) end end return 'c:' .. source.finish, nil - elseif source.type == 'doc.class.name' - or source.type == 'doc.alias.name' - or source.type == 'doc.extends.name' then + end) + : case 'doc.class.name' + : case 'doc.alias.name' + : case 'doc.extends.name' + : call(function (source) local name = source[1] return 'dn:' .. name, nil - elseif source.type == 'doc.type.name' then + end) + : case 'doc.type.name' + : call(function (source) local name = source[1] if source.typeGeneric then return 'dg:' .. source.typeGeneric[name][1].start, nil else return 'dn:' .. name, nil end - elseif source.type == 'doc.see.name' then + end) + : case 'doc.see.name' + : call(function (source) local name = source[1] return 'dsn:' .. name, nil - elseif source.type == 'doc.class' then + end) + : case 'doc.class' + : call(function (source) return 'dc:' .. source.start - elseif source.type == 'doc.type' then + end) + : case 'doc.type' + : call(function (source) return 'dt:' .. source.start - elseif source.type == 'doc.param' then + end) + : case 'doc.param' + : call(function (source) return 'dp:' .. source.start - elseif source.type == 'doc.vararg' then + end) + : case 'doc.vararg' + : call(function (source) return 'dv:' .. source.start - elseif source.type == 'doc.field.name' then + end) + : case 'doc.field.name' + : call(function (source) return 'dfn:' .. source.start - elseif source.type == 'doc.type.enum' - or source.type == 'doc.resume' then + end) + : case 'doc.type.enum' + : case 'doc.resume' + : call(function (source) return 'de:' .. source.start - elseif source.type == 'doc.type.table' then + end) + : case 'doc.type.table' + : call(function (source) return 'dtable:' .. source.start - elseif source.type == 'doc.type.ltable' then + end) + : case 'doc.type.ltable' + : call(function (source) return 'dltable:' .. source.start - elseif source.type == 'doc.type.field' then + end) + : case 'doc.type.field' + : call(function (source) return 'dfield:' .. source.start - elseif source.type == 'doc.type.array' then + end) + : case 'doc.type.array' + : call(function (source) return 'darray:' .. source.finish - elseif source.type == 'doc.type.function' then + end) + : case 'doc.type.function' + : call(function (source) return 'dfun:' .. source.start, nil - elseif source.type == 'doc.see.field' then + end) + : case 'doc.see.field' + : call(function (source) return ('%q'):format(source[1]), source.parent.name - elseif source.type == 'generic.closure' then + end) + : case 'generic.closure' + : call(function (source) return 'gc:' .. source.call.start, nil - elseif source.type == 'generic.value' then + end) + : case 'generic.value' + : call(function (source) local tail = '' if guide.getUri(source.closure.call) ~= guide.getUri(source.proto) then tail = URI_CHAR .. guide.getUri(source.closure.call) @@ -250,8 +324,19 @@ local function getKey(source) getKey(source.proto), tail ) + end) + : getMap() + +---获取语法树单元的key +---@param source parser.guide.object +---@return string? key +---@return parser.guide.object? node +function getKey(source) + local f = getKeyMap[source.type] + if f then + return f(source) end - return nil, nil + return nil end local function getNodeKey(source) @@ -286,6 +371,8 @@ local function getID(source) source._id = false return nil end + tracy.ZoneBeginN 'getID' + local _ <close> = tracy.ZoneEnd local current = source local index = 0 while true do @@ -416,19 +503,19 @@ local function getDocStateWithoutCrossFunction(obj) error('guide.getDocState overstack') end +local dontPushSourceMap = util.arrayToHash { + 'str:', 'nil:', 'num:', 'int:', 'bool:' +} + ---添加关联单元 ---@param noders noders ---@param source parser.guide.object function m.pushSource(noders, source, id) - id = id or m.getID(source) + id = id or getID(source) if not id then return end - if id == 'str:' - or id == 'nil:' - or id == 'num:' - or id == 'int:' - or id == 'bool:' then + if dontPushSourceMap[id] then return end local node = getNode(noders, id) @@ -436,10 +523,12 @@ function m.pushSource(noders, source, id) node.source = source return end - if not node.sources then - node.sources = {} + local sources = node.sources + if not sources then + sources = {} + node.sources = sources end - node.sources[#node.sources+1] = source + sources[#sources+1] = source end local DUMMY_FUNCTION = function () end @@ -1302,12 +1391,18 @@ function m.compileNodes(source) if next(noders) then return noders end + tracy.ZoneBeginN 'compileNodes' log.debug('compileNodes:', guide.getUri(root)) collector.dropUri(guide.getUri(root)) guide.eachSource(root, function (src) + tracy.ZoneBeginN 'pushSource' m.pushSource(noders, src) + tracy.ZoneEnd() + tracy.ZoneBeginN 'compileNode' m.compileNode(noders, src) + tracy.ZoneEnd() end) + tracy.ZoneEnd() log.debug('compileNodes finish:', guide.getUri(root)) return noders end @@ -1325,4 +1420,6 @@ files.watch(function (ev, uri) end end) +require 'tracy'.enable() + return m diff --git a/script/parser/guide.lua b/script/parser/guide.lua index fe0c62f1..998163aa 100644 --- a/script/parser/guide.lua +++ b/script/parser/guide.lua @@ -580,7 +580,7 @@ function m.isBetweenRange(source, tStart, tFinish) end --- 添加child -function m.addChilds(list, obj) +local function addChilds(list, obj) local tp = obj.type if not tp then return @@ -612,7 +612,7 @@ function m.eachSourceContain(ast, offset, callback) return res end end - m.addChilds(list, obj) + addChilds(list, obj) end end end @@ -638,7 +638,7 @@ function m.eachSourceBetween(ast, start, finish, callback) return res end end - m.addChilds(list, obj) + addChilds(list, obj) end end end @@ -705,7 +705,7 @@ function m.eachSource(ast, callback) index = index + 1 if not mark[obj] then mark[obj] = true - m.addChilds(cache, obj) + addChilds(cache, obj) end end end diff --git a/script/utility.lua b/script/utility.lua index ed11c28b..d1a539a5 100644 --- a/script/utility.lua +++ b/script/utility.lua @@ -21,6 +21,7 @@ local mathHuge = math.huge local inf = 1 / 0 local nan = 0 / 0 local utf8 = utf8 +local error = error _ENV = nil @@ -655,4 +656,30 @@ function m.arrayToHash(l) return t end +function m.switch() + local map = {} + local cachedCases = {} + local obj = { + case = function (self, name) + cachedCases[#cachedCases+1] = name + return self + end, + call = function (self, callback) + for i = 1, #cachedCases do + local name = cachedCases[i] + cachedCases[i] = nil + if map[name] then + error('Repeated fields:' .. tostring(name)) + end + map[name] = callback + end + return self + end, + getMap = function (self) + return map + end + } + return obj +end + return m @@ -91,7 +91,7 @@ local function main() --config.Lua.intelliSense.searchDepth = 5 --loadDocMetas() - --test 'full' + test 'full' require 'bee.platform'.OS = 'Windows' testAll() |