From 9cae97d4433181150742b2bc946898c321d5bf08 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=9C=80=E8=90=8C=E5=B0=8F=E6=B1=90?= Date: Wed, 12 Dec 2018 16:17:12 +0800 Subject: =?UTF-8?q?=E5=88=A0=E6=8E=89=E6=B2=A1=E7=94=A8=E7=9A=84=E4=BB=A3?= =?UTF-8?q?=E7=A0=81?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- server/src/matcher/compile.lua | 683 ------------------------------------ server/src/matcher/init.lua | 1 - server/test/main.lua | 2 +- server/test/type_inference/init.lua | 12 +- 4 files changed, 4 insertions(+), 694 deletions(-) delete mode 100644 server/src/matcher/compile.lua diff --git a/server/src/matcher/compile.lua b/server/src/matcher/compile.lua deleted file mode 100644 index 53da49e0..00000000 --- a/server/src/matcher/compile.lua +++ /dev/null @@ -1,683 +0,0 @@ -local env = require 'matcher.env' -local mt = {} -mt.__index = mt - -function mt:getVar(key, source) - if key == nil then - return nil - end - local var = self.env.var[key] - or self:getField(self.env.var._ENV, key, source) -- 这里不需要用getVar来递归获取_ENV - if not var and source then - var = self:addField(self:getVar '_ENV', key, source) - end - if var and var.meta then - var = var.meta - end - return var -end - -function mt:addInfo(obj, type, source) - if not obj then - return nil - end - obj[#obj+1] = { - type = type, - source = source, - } - return obj -end - -function mt:createVar(type, key, source) - local var = { - type = type, - key = key, - source = source, - childs = {}, - } - self.results.vars[#self.results.vars+1] = var - return var -end - -function mt:createLabel(key) - local lbl = { - key = key, - type = 'label', - } - self.results.labels[#self.results.labels+1] = lbl - return lbl -end - -function mt:createDots() - local dots = { - type = 'dots', - } - self.results.dots[#self.results.dots+1] = dots - return dots -end - -function mt:createLib(name) - local lib = { - name = name, - type = 'lib', - childs = {}, - } - return lib -end - -function mt:createLocal(key, source, var) - if key == nil then - return nil - end - if not var then - var = self:createVar('local', key, source) - self:addInfo(var, 'local', source) - end - local old = self.env.var[key] - if old then - local shadow = old.shadow - if not shadow then - shadow = { old } - old.shadow = shadow - end - var.shadow = shadow - shadow[#shadow+1] = var - end - self.env.var[key] = var - return var -end - -function mt:addField(parent, key, source) - if parent == nil or key == nil then - return nil - end - assert(source) - local var = parent.childs[key] - if not var then - var = self:createVar('field', key, source) - parent.childs[key] = var - var.parent = parent - end - return var -end - -function mt:getField(parent, key, source) - if parent == nil or key == nil then - return nil - end - local var - if parent.childs then - var = parent.childs[key] - end - if not var and source then - var = self:addField(parent, key, source) - end - return var -end - -function mt:isGlobal(var) - if var.type ~= 'field' then - return false - end - if not var.parent then - return false - end - return var.parent.key == '_ENV' or var.parent.key == '_G' -end - -function mt:getApi(func) - if not func then - return nil - end - func = func.value or func - if self:isGlobal(func) then - return func.key - end - return nil -end - -function mt:checkName(name) - local var = self:getVar(name[1], name) - self:addInfo(var, 'get', name) - return var -end - -function mt:checkDots(source) - local dots = self.env.dots - if not dots then - return - end - self:addInfo(dots, 'get', source) -end - -function mt:fixCallAsSetMetaTable(results) - local obj = results[1] - local metatable = results[2] - if metatable then - local index = self:getField(metatable, '__index') - if obj then - self:setMeta(obj, index) - return obj - else - return index - end - else - return obj - end -end - -function mt:fixCallAsRequire(results) - local libname = results[1] - if not libname then - return - end - libname = libname.value or libname - if libname.type ~= 'string' then - return - end - return self:createLib(libname.string) -end - -function mt:searchCall(call, func, lastobj) - local results = {} - for i, exp in ipairs(call) do - results[i] = self:searchExp(exp) - end - - if lastobj then - table.insert(self.results.calls, { - call = call, - func = func, - lastobj = lastobj, - results = results, - }) - end - - local api = self:getApi(func) - - if api == 'setmetatable' then - return self:fixCallAsSetMetaTable(results) - end - - if api == 'require' then - return self:fixCallAsRequire(results) - end - - return nil -end - -function mt:searchSimple(simple) - local name = simple[1] - local var - if name.type == 'name' then - var = self:getVar(name[1], name) - end - if name.type == 'string' then - var = self:getString(name) - end - self:searchExp(simple[1]) - for i = 2, #simple do - local obj = simple[i] - local tp = obj.type - if tp == 'call' then - var = self:searchCall(obj, var, simple[i-1]) - elseif tp == ':' then - elseif tp == 'name' then - if obj.index then - self:checkName(obj) - var = nil - else - var = self:getField(var, obj[1], obj) - self:addInfo(var, 'get', obj) - end - elseif obj.index and (obj.type == 'string' or obj.type == 'number' or obj.type == 'boolean') then - var = self:getField(var, obj[1], obj) - self:addInfo(var, 'get', obj) - else - self:searchExp(obj) - var = nil - end - end - return var -end - -function mt:searchBinary(exp) - return { - type = 'binary', - op = exp.op, - [1] = self:searchExp(exp[1]), - [2] = self:searchExp(exp[2]), - } -end - -function mt:searchUnary(exp) - return { - type = 'unary', - op = exp.op, - [1] = self:searchExp(exp[1]), - } -end - -function mt:searchTable(exp) - local tbl = { - type = 'table', - valuetype = 'table', - childs = {}, - } - for _, obj in ipairs(exp) do - if obj.type == 'pair' then - local key, value = obj[1], obj[2] - local res = self:searchExp(value) - local var = self:addField(tbl, key[1], key) - self:setValue(var, res) - self:addInfo(var, 'set', key) - else - self:searchExp(obj) - end - end - return tbl -end - -function mt:getString(exp) - return { - type = 'string', - string = exp[1], - valuetype = 'string', - childs = {}, - } -end - -function mt:getBoolean(exp) - return { - type = 'boolean', - boolean = exp[1], - valuetype = 'boolean', - } -end - -function mt:getNumber(exp) - return { - type = 'number', - number = exp[1], - valuetype = 'number', - } -end - -function mt:searchExp(exp) - local tp = exp.type - if tp == 'nil' then - elseif tp == 'string' then - return self:getString(exp) - elseif tp == 'boolean' then - return self:getBoolean(exp) - elseif tp == 'number' then - return self:getNumber(exp) - elseif tp == 'name' then - return self:checkName(exp) - elseif tp == 'simple' then - return self:searchSimple(exp) - elseif tp == 'binary' then - return self:searchBinary(exp) - elseif tp == 'unary' then - return self:searchUnary(exp) - elseif tp == '...' then - self:checkDots(exp) - elseif tp == 'function' then - return self:searchFunction(exp) - elseif tp == 'table' then - return self:searchTable(exp) - end - return nil -end - -function mt:searchReturn(action) - for _, exp in ipairs(action) do - self:searchExp(exp) - end -end - -function mt:setValue(var, value) - if not var or not value then - return - end - var.value = value.value or value - local group = var.group or value.group - if not group then - group = {} - end - if not group[var] then - var.group = group - group[var] = var - end - if not group[value] then - value.group = group - group[value] = value - end - if value.childs then - var.childs = value.childs - for _, child in pairs(value.childs) do - child.parent = var - end - end -end - -function mt:setMeta(var, meta) - if not var or not meta then - return - end - var.childs.meta = meta - meta.childs.meta = var -end - -function mt:markSimple(simple) - local name = simple[1] - local var - if name.type == 'name' then - var = self:getVar(name[1], name) - end - if name.type == 'string' then - var = self:getString(name) - end - self:searchExp(simple[1]) - for i = 2, #simple do - local obj = simple[i] - local tp = obj.type - if tp == 'call' then - var = self:searchCall(obj, var, simple[i-1]) - elseif tp == ':' then - var = self:createLocal('self', simple[i-1], self:getVar(simple[i-1][1])) - var.disableRename = true - elseif tp == 'name' then - if obj.index then - self:checkName(obj) - var = nil - else - if i == #simple then - var = self:addField(var, obj[1], obj) - self:addInfo(var, 'set', obj) - else - var = self:getField(var, obj[1], obj) - self:addInfo(var, 'get', obj) - end - end - elseif obj.index and (obj.type == 'string' or obj.type == 'number' or obj.type == 'boolean') then - if i == #simple then - var = self:addField(var, obj[1], obj) - self:addInfo(var, 'set', obj) - else - var = self:getField(var, obj[1], obj) - self:addInfo(var, 'get', obj) - end - else - self:searchExp(obj) - var = nil - end - end - return var -end - -function mt:markSet(simple, value) - if simple.type == 'name' then - local var = self:getVar(simple[1], simple) - self:addInfo(var, 'set', simple) - self:setValue(var, value) - return var - else - local var = self:markSimple(simple) - self:setValue(var, value) - return var - end -end - -function mt:markLocal(name, value) - if name.type == 'name' then - local str = name[1] - -- 创建一个局部变量 - local var = self:createLocal(str, name) - self:setValue(var, value) - return var - elseif name.type == '...' then - local dots = self:createDots() - self:addInfo(dots, 'local', name) - self.env.dots = dots - return dots - elseif name.type == ':' then - -- 创建一个局部变量 - local var = self:createLocal('self', name) - return var - end -end - -function mt:forList(list, callback) - if not list then - return - end - if list.type == 'list' then - for i = 1, #list do - callback(list[i]) - end - else - callback(list) - end -end - -function mt:markSets(action) - local keys = action[1] - local values = action[2] - local results = {} - -- 要先计算赋值 - local i = 0 - self:forList(values, function (value) - i = i + 1 - results[i] = self:searchExp(value) - end) - local i = 0 - self:forList(keys, function (key) - i = i + 1 - self:markSet(key, results[i]) - end) -end - -function mt:markLocals(action) - local keys = action[1] - local values = action[2] - local results = {} - -- 要先计算赋值 - local i = 0 - self:forList(values, function (value) - i = i + 1 - results[i] = self:searchExp(value) - end) - local i = 0 - self:forList(keys, function (key) - i = i + 1 - self:markLocal(key, results[i]) - end) -end - -function mt:searchIfs(action) - for _, block in ipairs(action) do - self.env:push() - if block.filter then - self:searchExp(block.filter) - end - self:searchActions(block) - self.env:pop() - end -end - -function mt:searchLoop(action) - self.env:push() - self:markLocal(action.arg) - self:searchExp(action.min) - self:searchExp(action.max) - if action.step then - self:searchExp(action.step) - end - self:searchActions(action) - self.env:pop() -end - -function mt:searchIn(action) - self:forList(action.exp, function (exp) - self:searchExp(exp) - end) - self.env:push() - self:forList(action.arg, function (arg) - self:markLocal(arg) - end) - self:searchActions(action) - self.env:pop() -end - -function mt:searchDo(action) - self.env:push() - self:searchActions(action) - self.env:pop() -end - -function mt:searchWhile(action) - self:searchExp(action.filter) - self.env:push() - self:searchActions(action) - self.env:pop() -end - -function mt:searchRepeat(action) - self.env:push() - self:searchActions(action) - self:searchExp(action.filter) - self.env:pop() -end - -function mt:searchFunction(func) - local obj = { - type = 'function', - valuetype = 'function', - args = {}, - actions = {}, - } - self.env:push() - self.env:cut 'dots' - self.env.label = {} - if func.name then - obj.name = self:markSet(func.name, obj) - end - self:forList(func.arg, function (arg) - obj.args[#obj.args+1] = self:markLocal(arg) - end) - for _, action in ipairs(func) do - obj.actions[#obj.actions+1] = self:searchAction(action) - end - self.env:pop() - return obj -end - -function mt:searchLocalFunction(func) - local obj = { - type = 'function', - valuetype = 'function', - args = {}, - actions = {}, - } - if func.name then - obj.name = self:markLocal(func.name, obj) - end - self.env:push() - self:forList(func.arg, function (arg) - obj.args[#obj.args+1] = self:markLocal(arg) - end) - for _, action in ipairs(func) do - obj.actions[#obj.actions+1] = self:searchAction(action) - end - self.env:pop() - return obj -end - -function mt:markLabel(label) - local str = label[1] - if not self.env.label[str] then - self.env.label[str] = self:createLabel(str) - end - self:addInfo(self.env.label[str], 'set', label) -end - -function mt:searchGoTo(obj) - local str = obj[1] - if not self.env.label[str] then - self.env.label[str] = self:createLabel(str) - end - self:addInfo(self.env.label[str], 'goto', obj) -end - -function mt:searchAction(action) - local tp = action.type - if tp == 'do' then - self:searchDo(action) - elseif tp == 'break' then - elseif tp == 'return' then - self:searchReturn(action) - elseif tp == 'label' then - self:markLabel(action) - elseif tp == 'goto' then - self:searchGoTo(action) - elseif tp == 'set' then - self:markSets(action) - elseif tp == 'local' then - self:markLocals(action) - elseif tp == 'simple' then - self:searchSimple(action) - elseif tp == 'if' then - self:searchIfs(action) - elseif tp == 'loop' then - self:searchLoop(action) - elseif tp == 'in' then - self:searchIn(action) - elseif tp == 'while' then - self:searchWhile(action) - elseif tp == 'repeat' then - self:searchRepeat(action) - elseif tp == 'function' then - self:searchFunction(action) - elseif tp == 'localfunction' then - self:searchLocalFunction(action) - end -end - -function mt:searchActions(actions) - for _, action in ipairs(actions) do - self:searchAction(action) - end - return nil -end - -local function compile(ast) - local searcher = setmetatable({ - env = env { - var = {}, - usable = {} - }, - results = { - labels = {}, - vars = {}, - dots = {}, - calls = {}, - } - }, mt) - searcher.env.label = {} - searcher:createLocal('_ENV', { start = -1, finish = -1 }) - searcher:searchActions(ast) - - return searcher.results -end - -return function (ast) - if not ast then - return nil - end - local suc, res = xpcall(compile, log.error, ast) - if not suc then - return nil - end - return res -end diff --git a/server/src/matcher/init.lua b/server/src/matcher/init.lua index a0a0d32c..9174d839 100644 --- a/server/src/matcher/init.lua +++ b/server/src/matcher/init.lua @@ -5,7 +5,6 @@ local api = { rename = require 'matcher.rename', hover = require 'matcher.hover', diagnostics = require 'matcher.diagnostics', - compile = require 'matcher.compile', findResult = require 'matcher.find_result', findLib = require 'matcher.find_lib', vm = require 'matcher.vm', diff --git a/server/test/main.lua b/server/test/main.lua index 690c8d3c..67e2da0a 100644 --- a/server/test/main.lua +++ b/server/test/main.lua @@ -24,9 +24,9 @@ local function main() end test 'vm' - --test 'type_inference' test 'definition' test 'diagnostics' + test 'type_inference' test 'find_lib' print('测试完成') diff --git a/server/test/type_inference/init.lua b/server/test/type_inference/init.lua index 59993b14..dc1a4099 100644 --- a/server/test/type_inference/init.lua +++ b/server/test/type_inference/init.lua @@ -10,16 +10,10 @@ function TEST(res) local pos = (start + finish) // 2 + 1 local new_script = script:gsub('<[!?]', ' '):gsub('[!?]>', ' ') local ast = parser:ast(new_script) - assert(ast) - local results = matcher.compile(ast) - assert(results) - matcher.typeInference(results) - local result = matcher.findResult(results, pos) + local vm = matcher.vm(ast) + assert(vm) + local result = matcher.findResult(vm.results, pos) assert(result) - local var = result.var - assert(var) - assert(var.group) - assert(var.group.type == res) end end -- cgit v1.2.3