diff options
author | 最萌小汐 <sumneko@hotmail.com> | 2022-11-18 17:47:11 +0800 |
---|---|---|
committer | 最萌小汐 <sumneko@hotmail.com> | 2022-11-18 17:47:11 +0800 |
commit | b137f3faf9225e4e4279e0aefc1d98bfb6aa4724 (patch) | |
tree | e0044497742e14bddb0f6f4e54163319a040cd86 /script/vm | |
parent | e13503e871ade8422e5a44d5c582f05769fe8dde (diff) | |
download | lua-language-server-b137f3faf9225e4e4279e0aefc1d98bfb6aa4724.zip |
resolve dead lock
Diffstat (limited to 'script/vm')
-rw-r--r-- | script/vm/compiler.lua | 64 | ||||
-rw-r--r-- | script/vm/runner.lua | 112 |
2 files changed, 123 insertions, 53 deletions
diff --git a/script/vm/compiler.lua b/script/vm/compiler.lua index ef0f7157..70a46677 100644 --- a/script/vm/compiler.lua +++ b/script/vm/compiler.lua @@ -976,7 +976,7 @@ end ---@param source parser.object local function compileLocal(source) - vm.setNode(source, source) + local myNode = vm.setNode(source, source) local hasMarkDoc if source.bindDocs then @@ -988,7 +988,7 @@ local function compileLocal(source) if selfNode then hasMarkParam = true vm.setNode(source, vm.compileNode(selfNode)) - vm.getNode(source):remove 'function' + myNode:remove 'function' end end local hasMarkValue @@ -1062,7 +1062,7 @@ local function compileLocal(source) end end - vm.getNode(source):setData('hasDefined', hasMarkDoc or hasMarkParam or hasMarkValue) + myNode:setData('hasDefined', hasMarkDoc or hasMarkParam or hasMarkValue) end ---@param source parser.object @@ -1192,15 +1192,40 @@ local compilerSwitch = util.switch() ---@async ---@param source parser.object : call(function (source) - compileLocal(source) - local refs = source.ref - if not refs then - return - end - - local hasMark = vm.getNode(source):getData 'hasDefined' - - vm.launchRunner(source, function (src, node) + vm.launchRunner(source, function () + local myNode = vm.getNode(source) + ---@cast myNode -? + myNode:setData('resolving', true) + + if source.ref then + for _, ref in ipairs(source.ref) do + if ref.type == 'getlocal' + or ref.type == 'setlocal' then + vm.setNode(ref, myNode, true) + end + end + end + compileLocal(source) + + myNode:setData('hasResolved', true) + end, function () + local myNode = vm.getNode(source) + ---@cast myNode -? + myNode:setData('resolving', nil) + local hasMark = vm.getNode(source):getData 'hasDefined' + if source.ref and not hasMark then + local parentFunc = guide.getParentFunction(source) + for _, ref in ipairs(source.ref) do + if ref.type == 'setlocal' + and guide.getParentFunction(ref) == parentFunc then + local refNode = vm.getNode(ref) + if refNode then + vm.setNode(source, refNode) + end + end + end + end + end, function (src, node) if src.type == 'setlocal' then if src.bindDocs then for _, doc in ipairs(src.bindDocs) do @@ -1212,7 +1237,7 @@ local compilerSwitch = util.switch() end if src.value then if src.value.type == 'table' then - vm.setNode(src, vm.createNode(src.value)) + vm.setNode(src, vm.createNode(src.value), true) vm.setNode(src, node:copy():asTable()) else vm.setNode(src, vm.compileNode(src.value), true) @@ -1231,18 +1256,7 @@ local compilerSwitch = util.switch() end end) - if not hasMark then - local parentFunc = guide.getParentFunction(source) - for _, ref in ipairs(source.ref) do - if ref.type == 'setlocal' - and guide.getParentFunction(ref) == parentFunc then - local refNode = vm.getNode(ref) - if refNode then - vm.setNode(source, refNode) - end - end - end - end + vm.waitResolveRunner(source) end) : case 'setlocal' : call(function (source) diff --git a/script/vm/runner.lua b/script/vm/runner.lua index 341126c6..13370ac7 100644 --- a/script/vm/runner.lua +++ b/script/vm/runner.lua @@ -345,7 +345,7 @@ function mt:lookIntoBlock(block, topNode) return topNode end ----@alias runner.info { source?: parser.object, loc: parser.object } +---@alias runner.info { target?: parser.object, loc: parser.object } ---@type thread? local masterRunner = nil @@ -364,25 +364,55 @@ local runnerList = nil ---@param info runner.info local function waitResolve(info) while true do - if not info.source then + if not info.target then break end - if info.source.node == info.loc then + if info.target.node == info.loc then break end - local node = vm.getNode(info.source) + local node = vm.getNode(info.target) if node and node:getData('hasResolved') then break end coroutine.yield() end - info.source = nil + info.target = nil +end + +local function resolveDeadLock() + if not runnerList then + return + end + + ---@type runner.info[] + local infos = {} + for runner in runnerList:pairs() do + local info = runnerInfo[runner] + infos[#infos+1] = info + end + + table.sort(infos, function (a, b) + local uriA = guide.getUri(a.loc) + local uriB = guide.getUri(b.loc) + if uriA ~= uriB then + return uriA < uriB + end + return a.loc.start < b.loc.start + end) + + local firstTarget = infos[1].target + ---@cast firstTarget -? + local firstNode = vm.setNode(firstTarget, vm.getNode(firstTarget):copy(), true) + firstNode:setData('hasResolved', true) + firstNode:setData('resolvedByDeadLock', true) end ---@async ---@param loc parser.object +---@param start fun() +---@param finish fun() ---@param callback vm.runner.callback -function vm.launchRunner(loc, callback) +function vm.launchRunner(loc, start, finish, callback) local locNode = vm.getNode(loc) if not locNode then return @@ -393,10 +423,10 @@ function vm.launchRunner(loc, callback) if not runnerList or runnerList:getSize() == 0 then return end - local allWaiting = true + local deadLock = true for runner in runnerList:pairs() do local info = runnerInfo[runner] - local waitingSource = info.source + local waitingSource = info.target if coroutine.status(runner) == 'suspended' then local suc, err = coroutine.resume(runner) if not suc then @@ -404,25 +434,33 @@ function vm.launchRunner(loc, callback) end else runnerList:pop(runner) + deadLock = false end - if not waitingSource or waitingSource ~= info.source then - allWaiting = false + if not waitingSource or waitingSource ~= info.target then + deadLock = false end end if runnerList:getSize() == 0 then return end - if allWaiting or i == 10000 then + if deadLock then + resolveDeadLock() + end + if i == 10000 then local lines = {} lines[#lines+1] = 'Dead lock:' - lines[#lines+1] = guide.getUri(loc) for runner in runnerList:pairs() do local info = runnerInfo[runner] - lines[#lines+1] = string.format('Runner `%s` at %d waiting for `%s` at %d' - , loc[1] - , loc.start - , info.source and info.source[1] or '' - , info.source and info.source.start or 0 + lines[#lines+1] = '===============' + lines[#lines+1] = string.format('Runner `%s` at %d(%s)' + , info.loc[1] + , info.loc.start + , guide.getUri(info.loc) + ) + lines[#lines+1] = string.format('Waiting `%s` at %d(%s)' + , info.target[1] + , info.target.start + , guide.getUri(info.target) ) end local msg = table.concat(lines, '\n') @@ -432,8 +470,14 @@ function vm.launchRunner(loc, callback) end local function launch() + start() + if not loc.ref then + finish() + return + end local main = guide.getParentBlock(loc) if not main then + finish() return end local self = setmetatable({ @@ -451,6 +495,8 @@ function vm.launchRunner(loc, callback) self:lookIntoBlock(main, locNode:copy()) locNode:setData('runner', nil) + + finish() end local co = coroutine.create(launch) @@ -474,25 +520,35 @@ end ---@async ---@param source parser.object function vm.waitResolveRunner(source) - local running = coroutine.running() - if not masterRunner or running == masterRunner then + local myNode = vm.getNode(source) + if myNode and myNode:getData('hasResolved') then return end - local loc = source.node - local locNode = vm.getNode(loc) - if not locNode then - return - end - local runner = locNode:getData('runner') - if not runner or runner == running then + local running = coroutine.running() + if not masterRunner or running == masterRunner then return end local info = runnerInfo[running] - if info.loc == loc then + + local targetLoc + if source.type == 'getlocal' then + targetLoc = source.node + elseif source.type == 'local' + or source.type == 'self' then + targetLoc = source + info.target = info.target or source + else + error('Unknown source type: ' .. source.type) + end + + local targetNode = vm.getNode(targetLoc) + if not targetNode then + -- Wait for compiling local by `compiler` return end + waitResolve(info) end @@ -505,5 +561,5 @@ function vm.storeWaitingRunner(source) local running = coroutine.running() local info = runnerInfo[running] - info.source = source + info.target = source end |