summaryrefslogtreecommitdiff
path: root/script/vm
diff options
context:
space:
mode:
author最萌小汐 <sumneko@hotmail.com>2022-02-18 16:53:34 +0800
committer最萌小汐 <sumneko@hotmail.com>2022-02-18 16:53:34 +0800
commitcd40c6c9117dce31c3040271e79038e20376af6a (patch)
treed2eb4d6c90db556b62fda913dae73e0a4ebb545b /script/vm
parentf8a6a73eaa1aa7915b296870f9a3862b429c7e74 (diff)
downloadlua-language-server-cd40c6c9117dce31c3040271e79038e20376af6a.zip
update
Diffstat (limited to 'script/vm')
-rw-r--r--script/vm/getDef.lua85
-rw-r--r--script/vm/local-id.lua109
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