diff options
author | 最萌小汐 <sumneko@hotmail.com> | 2022-02-18 16:53:34 +0800 |
---|---|---|
committer | 最萌小汐 <sumneko@hotmail.com> | 2022-02-18 16:53:34 +0800 |
commit | cd40c6c9117dce31c3040271e79038e20376af6a (patch) | |
tree | d2eb4d6c90db556b62fda913dae73e0a4ebb545b /script/vm | |
parent | f8a6a73eaa1aa7915b296870f9a3862b429c7e74 (diff) | |
download | lua-language-server-cd40c6c9117dce31c3040271e79038e20376af6a.zip |
update
Diffstat (limited to 'script/vm')
-rw-r--r-- | script/vm/getDef.lua | 85 | ||||
-rw-r--r-- | script/vm/local-id.lua | 109 |
2 files changed, 152 insertions, 42 deletions
diff --git a/script/vm/getDef.lua b/script/vm/getDef.lua index 3cb33e69..a303f5f7 100644 --- a/script/vm/getDef.lua +++ b/script/vm/getDef.lua @@ -8,72 +8,72 @@ local globalNode = require 'vm.global-node' local simpleMap -local function searchGetLocal(source, node, results) +local function searchGetLocal(source, node, pushResult) local key = guide.getKeyName(source) for _, ref in ipairs(node.node.ref) do if ref.type == 'getlocal' and guide.isSet(ref.next) and guide.getKeyName(ref.next) == key then - results[#results+1] = ref.next + pushResult(ref.next) end end end simpleMap = util.switch() : case 'local' - : call(function (source, results) - results[#results+1] = source + : call(function (source, pushResult) + pushResult(source) if source.ref then for _, ref in ipairs(source.ref) do if ref.type == 'setlocal' then - results[#results+1] = ref + pushResult(ref) end end end if source.dummy then for _, res in ipairs(vm.getDefs(source.method.node)) do - results[#results+1] = res + pushResult(res) end end end) : case 'getlocal' : case 'setlocal' - : call(function (source, results) - simpleMap['local'](source.node, results) + : call(function (source, pushResult) + simpleMap['local'](source.node, pushResult) end) : case 'field' - : call(function (source, results) + : call(function (source, pushResult) local parent = source.parent - simpleMap[parent.type](parent, results) + simpleMap[parent.type](parent, pushResult) end) : case 'setfield' : case 'getfield' - : call(function (source, results) + : call(function (source, pushResult) local node = source.node if node.type == 'getlocal' then - searchGetLocal(source, node, results) + searchGetLocal(source, node, pushResult) return end end) : case 'getindex' : case 'setindex' - : call(function (source, results) + : call(function (source, pushResult) local node = source.node if node.type == 'getlocal' then - searchGetLocal(source, node, results) + searchGetLocal(source, node, pushResult) end end) : getMap() local searchFieldMap = util.switch() : case 'table' - : call(function (node, key, results) + : call(function (node, key, pushResult) for _, field in ipairs(node) do if field.type == 'tablefield' or field.type == 'tableindex' then if guide.getKeyName(field) == key then - results[#results+1] = field + pushResult(field) end end end @@ -82,57 +82,64 @@ local searchFieldMap = util.switch() local nodeMap;nodeMap = util.switch() : case 'field' - : call(function (source, results) + : call(function (source, pushResult) local parent = source.parent - nodeMap[parent.type](parent, results) + nodeMap[parent.type](parent, pushResult) end) : case 'getfield' : case 'setfield' - : call(function (source, results) + : call(function (source, pushResult) local node = compiler.compileNode(guide.getUri(source.node), source.node) if not node then return end if searchFieldMap[node.type] then - searchFieldMap[node.type](node, guide.getKeyName(source), results) + searchFieldMap[node.type](node, guide.getKeyName(source), pushResult) end end) : getMap() ----@param source parser.object ----@param results parser.object[] -local function searchBySimple(source, results) + ---@param source parser.object + ---@param pushResult fun(src: parser.object) +local function searchBySimple(source, pushResult) local simple = simpleMap[source.type] if simple then - simple(source, results) + simple(source, pushResult) end end ---@param source parser.object ----@param results parser.object[] -local function searchByGlobal(source, results) +---@param pushResult fun(src: parser.object) +local function searchByGlobal(source, pushResult) local global = globalNode.getNode(source) if not global then return end for _, src in ipairs(global:getSets()) do - results[#results+1] = src + pushResult(src) end end -local function searchByID(source, results) +---@param source parser.object +---@param pushResult fun(src: parser.object) +local function searchByID(source, pushResult) local idSources = localID.getSources(source) if not idSources then return end + for _, src in ipairs(idSources) do + if guide.isSet(src) then + pushResult(src) + end + end end ---@param source parser.object ----@param results parser.object[] -local function searchByNode(source, results) +---@param pushResult fun(src: parser.object) +local function searchByNode(source, pushResult) local node = nodeMap[source.type] if node then - node(source, results) + node(source, pushResult) end end @@ -140,11 +147,19 @@ end ---@return parser.object[] function vm.getDefs(source) local results = {} + local mark = {} + + local function pushResult(src) + if not mark[src] then + mark[src] = true + results[#results+1] = src + end + end - searchBySimple(source, results) - searchByGlobal(source, results) - --searchByID(source, results) - searchByNode(source, results) + searchBySimple(source, pushResult) + searchByGlobal(source, pushResult) + searchByID(source, pushResult) + searchByNode(source, pushResult) return results end diff --git a/script/vm/local-id.lua b/script/vm/local-id.lua index f19e85e6..8487d96a 100644 --- a/script/vm/local-id.lua +++ b/script/vm/local-id.lua @@ -1,21 +1,116 @@ -local util = require 'utility' +local util = require 'utility' +local guide = require 'parser.guide' ---@class parser.object ---@field _localID string +---@field _localIDs table<string, parser.object[]> + +---@class vm.local-id +local m = {} + +m.ID_SPLITE = '\x1F' local compileMap = util.switch() + : case 'local' + : call(function (source) + if not source.ref then + return + end + for _, ref in ipairs(source.ref) do + m.compileLocalID(ref) + end + end) + : case 'getlocal' + : call(function (source) + source._localID = ('%d'):format(source.node.start) + m.compileLocalID(source.next) + end) + : case 'getfield' + : case 'setfield' + : call(function (source) + local parentID = source.node._localID + if not parentID then + return + end + source._localID = parentID .. m.ID_SPLITE .. guide.getKeyName(source) + source.field._localID = source._localID + if source.type == 'getfield' then + m.compileLocalID(source.next) + end + end) : getMap() -local m = {} +local leftMap = util.switch() + : case 'field' + : call(function (source) + return m.getLocal(source.parent) + end) + : case 'getfield' + : case 'setfield' + : call(function (source) + return m.getLocal(source.node) + end) + : case 'getlocal' + : call(function (source) + return source.node + end) + : getMap() -m.ID_SPLITE = '\x1F' +---@param source parser.object +---@return parser.object? +function m.getLocal(source) + local getLeft = leftMap[source.type] + if getLeft then + return getLeft(source) + end + return nil +end -function m.getID(source) +function m.compileLocalID(source) + if not source then + return + end + source._localID = false local compiler = compileMap[source.type] - if compiler then - return compiler(source) + if not compiler then + return + end + compiler(source) + local root = guide.getRoot(source) + if not root._localIDs then + root._localIDs = util.multiTable(2) + end + local sources = root._localIDs[source._localID] + sources[#sources+1] = source +end + +---@param source parser.object +---@return string|boolean +function m.getID(source) + if source._localID ~= nil then + return source._localID + end + source._localID = false + local loc = m.getLocal(source) + if not loc then + return source._localID + end + m.compileLocalID(loc) + return source._localID +end + +---@param source parser.object +---@return parser.object[]? +function m.getSources(source) + local id = m.getID(source) + if not id then + return nil + end + local root = guide.getRoot(source) + if not root._localIDs then + return nil end - return false + return root._localIDs[id] end return m |