mirror of
https://github.com/nim-lang/Nim.git
synced 2025-12-30 09:54:49 +00:00
238 lines
7.0 KiB
Nim
Executable File
238 lines
7.0 KiB
Nim
Executable File
#
|
|
#
|
|
# The Nimrod Compiler
|
|
# (c) Copyright 2012 Andreas Rumpf
|
|
#
|
|
# See the file "copying.txt", included in this
|
|
# distribution, for details about the copyright.
|
|
#
|
|
|
|
## This file implements features required for IDE support.
|
|
|
|
import
|
|
lexer, idents, ast, astalgo, semdata, msgs, types, sigmatch, options,
|
|
renderer
|
|
|
|
const
|
|
sep = '\t'
|
|
sectionSuggest = "sug"
|
|
sectionDef = "def"
|
|
sectionContext = "con"
|
|
|
|
proc SymToStr(s: PSym, isLocal: bool, section: string): string =
|
|
result = section
|
|
result.add(sep)
|
|
result.add($s.kind)
|
|
result.add(sep)
|
|
if not isLocal:
|
|
if s.kind != skModule and s.owner != nil:
|
|
result.add(s.owner.name.s)
|
|
result.add('.')
|
|
result.add(s.name.s)
|
|
result.add(sep)
|
|
if s.typ != nil:
|
|
result.add(typeToString(s.typ))
|
|
result.add(sep)
|
|
result.add(toFilename(s.info))
|
|
result.add(sep)
|
|
result.add($ToLinenumber(s.info))
|
|
result.add(sep)
|
|
result.add($ToColumn(s.info))
|
|
|
|
proc filterSym(s: PSym): bool {.inline.} =
|
|
result = s.name.s[0] in lexer.SymChars
|
|
|
|
proc suggestField(s: PSym) =
|
|
if filterSym(s):
|
|
OutWriteln(SymToStr(s, isLocal=true, sectionSuggest))
|
|
|
|
template wholeSymTab(cond, section: expr) {.immediate.} =
|
|
for i in countdown(c.tab.tos-1, 0):
|
|
for it in items(c.tab.stack[i]):
|
|
if cond:
|
|
OutWriteln(SymToStr(it, isLocal = i > ModuleTablePos, section))
|
|
|
|
proc suggestSymList(list: PNode) =
|
|
for i in countup(0, sonsLen(list) - 1):
|
|
if list.sons[i].kind != nkSym: InternalError(list.info, "getSymFromList")
|
|
suggestField(list.sons[i].sym)
|
|
|
|
proc suggestObject(n: PNode) =
|
|
case n.kind
|
|
of nkRecList:
|
|
for i in countup(0, sonsLen(n)-1): suggestObject(n.sons[i])
|
|
of nkRecCase:
|
|
var L = sonsLen(n)
|
|
if L > 0:
|
|
suggestObject(n.sons[0])
|
|
for i in countup(1, L-1): suggestObject(lastSon(n.sons[i]))
|
|
of nkSym: suggestField(n.sym)
|
|
else: nil
|
|
|
|
proc nameFits(c: PContext, s: PSym, n: PNode): bool =
|
|
var op = n.sons[0]
|
|
if op.kind == nkSymChoice: op = op.sons[0]
|
|
var opr: PIdent
|
|
case op.kind
|
|
of nkSym: opr = op.sym.name
|
|
of nkIdent: opr = op.ident
|
|
else: return false
|
|
result = opr.id == s.name.id
|
|
|
|
proc argsFit(c: PContext, candidate: PSym, n, nOrig: PNode): bool =
|
|
case candidate.kind
|
|
of OverloadableSyms:
|
|
var m: TCandidate
|
|
initCandidate(m, candidate, nil)
|
|
sigmatch.partialMatch(c, n, nOrig, m)
|
|
result = m.state != csNoMatch
|
|
else:
|
|
result = false
|
|
|
|
proc suggestCall(c: PContext, n, nOrig: PNode) =
|
|
wholeSymTab(filterSym(it) and nameFits(c, it, n) and argsFit(c, it, n, nOrig),
|
|
sectionContext)
|
|
|
|
proc typeFits(c: PContext, s: PSym, firstArg: PType): bool {.inline.} =
|
|
if s.typ != nil and sonsLen(s.typ) > 1 and s.typ.sons[1] != nil:
|
|
result = sigmatch.argtypeMatches(c, s.typ.sons[1], firstArg)
|
|
|
|
proc suggestOperations(c: PContext, n: PNode, typ: PType) =
|
|
assert typ != nil
|
|
wholeSymTab(filterSym(it) and typeFits(c, it, typ), sectionSuggest)
|
|
|
|
proc suggestEverything(c: PContext, n: PNode) =
|
|
# do not produce too many symbols:
|
|
for i in countdown(c.tab.tos-1, 1):
|
|
for it in items(c.tab.stack[i]):
|
|
if filterSym(it):
|
|
OutWriteln(SymToStr(it, isLocal = i > ModuleTablePos, sectionSuggest))
|
|
|
|
proc suggestFieldAccess(c: PContext, n: PNode) =
|
|
# special code that deals with ``myObj.``. `n` is NOT the nkDotExpr-node, but
|
|
# ``myObj``.
|
|
var typ = n.Typ
|
|
if typ == nil:
|
|
# a module symbol has no type for example:
|
|
if n.kind == nkSym and n.sym.kind == skModule:
|
|
if n.sym == c.module:
|
|
# all symbols accessible, because we are in the current module:
|
|
for it in items(c.tab.stack[ModuleTablePos]):
|
|
if filterSym(it):
|
|
OutWriteln(SymToStr(it, isLocal=false, sectionSuggest))
|
|
else:
|
|
for it in items(n.sym.tab):
|
|
if filterSym(it):
|
|
OutWriteln(SymToStr(it, isLocal=false, sectionSuggest))
|
|
else:
|
|
# fallback:
|
|
suggestEverything(c, n)
|
|
elif typ.kind == tyEnum and n.kind == nkSym and n.sym.kind == skType:
|
|
# look up if the identifier belongs to the enum:
|
|
var t = typ
|
|
while t != nil:
|
|
suggestSymList(t.n)
|
|
t = t.sons[0]
|
|
suggestOperations(c, n, typ)
|
|
else:
|
|
typ = skipTypes(typ, {tyGenericInst, tyVar, tyPtr, tyRef})
|
|
if typ.kind == tyObject:
|
|
var t = typ
|
|
while true:
|
|
suggestObject(t.n)
|
|
if t.sons[0] == nil: break
|
|
t = skipTypes(t.sons[0], {tyGenericInst})
|
|
suggestOperations(c, n, typ)
|
|
elif typ.kind == tyTuple and typ.n != nil:
|
|
suggestSymList(typ.n)
|
|
suggestOperations(c, n, typ)
|
|
else:
|
|
suggestOperations(c, n, typ)
|
|
|
|
proc findClosestDot(n: PNode): PNode =
|
|
if n.kind == nkDotExpr and msgs.inCheckpoint(n.info) == cpExact:
|
|
result = n
|
|
else:
|
|
for i in 0.. <safeLen(n):
|
|
result = findClosestDot(n.sons[i])
|
|
if result != nil: return
|
|
|
|
const
|
|
CallNodes = {nkCall, nkInfix, nkPrefix, nkPostfix, nkCommand, nkCallStrLit,
|
|
nkMacroStmt}
|
|
|
|
proc findClosestCall(n: PNode): PNode =
|
|
if n.kind in callNodes and msgs.inCheckpoint(n.info) == cpExact:
|
|
result = n
|
|
else:
|
|
for i in 0.. <safeLen(n):
|
|
result = findClosestCall(n.sons[i])
|
|
if result != nil: return
|
|
|
|
proc findClosestSym(n: PNode): PNode =
|
|
if n.kind == nkSym and msgs.inCheckpoint(n.info) == cpExact:
|
|
result = n
|
|
elif n.kind notin {nkNone..nkNilLit}:
|
|
for i in 0.. <sonsLen(n):
|
|
result = findClosestSym(n.sons[i])
|
|
if result != nil: return
|
|
|
|
var recursiveCheck = 0
|
|
|
|
proc safeSemExpr(c: PContext, n: PNode): PNode =
|
|
try:
|
|
result = c.semExpr(c, n)
|
|
except ERecoverableError:
|
|
result = ast.emptyNode
|
|
|
|
proc fuzzySemCheck(c: PContext, n: PNode): PNode =
|
|
result = safeSemExpr(c, n)
|
|
if result == nil or result.kind == nkEmpty:
|
|
result = newNodeI(n.kind, n.info)
|
|
if n.kind notin {nkNone..nkNilLit}:
|
|
for i in 0 .. < sonsLen(n): result.addSon(fuzzySemCheck(c, n.sons[i]))
|
|
|
|
proc suggestExpr*(c: PContext, node: PNode) =
|
|
var cp = msgs.inCheckpoint(node.info)
|
|
if cp == cpNone: return
|
|
# HACK: This keeps semExpr() from coming here recursively:
|
|
if recursiveCheck > 0: return
|
|
inc(recursiveCheck)
|
|
|
|
if optSuggest in gGlobalOptions:
|
|
var n = findClosestDot(node)
|
|
if n == nil: n = node
|
|
else: cp = cpExact
|
|
|
|
if n.kind == nkDotExpr and cp == cpExact:
|
|
var obj = safeSemExpr(c, n.sons[0])
|
|
suggestFieldAccess(c, obj)
|
|
else:
|
|
suggestEverything(c, n)
|
|
|
|
if optContext in gGlobalOptions:
|
|
var n = findClosestCall(node)
|
|
if n == nil: n = node
|
|
else: cp = cpExact
|
|
|
|
if n.kind in CallNodes:
|
|
var a = copyNode(n)
|
|
var x = safeSemExpr(c, n.sons[0])
|
|
if x.kind == nkEmpty or x.typ == nil: x = n.sons[0]
|
|
addSon(a, x)
|
|
for i in 1..sonsLen(n)-1:
|
|
# use as many typed arguments as possible:
|
|
var x = safeSemExpr(c, n.sons[i])
|
|
if x.kind == nkEmpty or x.typ == nil: break
|
|
addSon(a, x)
|
|
suggestCall(c, a, n)
|
|
|
|
if optDef in gGlobalOptions:
|
|
var n = findClosestSym(fuzzySemCheck(c, node))
|
|
if n != nil: OutWriteln(SymToStr(n.sym, isLocal=false, sectionDef))
|
|
quit(0)
|
|
|
|
proc suggestStmt*(c: PContext, n: PNode) =
|
|
suggestExpr(c, n)
|
|
|