diff options
Diffstat (limited to 'compiler/vmdeps.nim')
-rw-r--r-- | compiler/vmdeps.nim | 334 |
1 files changed, 334 insertions, 0 deletions
diff --git a/compiler/vmdeps.nim b/compiler/vmdeps.nim new file mode 100644 index 000000000..294aaaa79 --- /dev/null +++ b/compiler/vmdeps.nim @@ -0,0 +1,334 @@ +# +# +# The Nim Compiler +# (c) Copyright 2015 Andreas Rumpf +# +# See the file "copying.txt", included in this +# distribution, for details about the copyright. +# + +import ast, types, msgs, options, idents, lineinfos +from pathutils import AbsoluteFile + +import std/os + +when defined(nimPreviewSlimSystem): + import std/syncio + +proc opSlurp*(file: string, info: TLineInfo, module: PSym; conf: ConfigRef): string = + try: + var filename = parentDir(toFullPath(conf, info)) / file + if not fileExists(filename): + filename = findFile(conf, file).string + result = readFile(filename) + # we produce a fake include statement for every slurped filename, so that + # the module dependencies are accurate: + discard conf.fileInfoIdx(AbsoluteFile filename) + appendToModule(module, newTreeI(nkIncludeStmt, info, newStrNode(nkStrLit, filename))) + except IOError: + localError(conf, info, "cannot open file: " & file) + result = "" + +proc atomicTypeX(cache: IdentCache; name: string; m: TMagic; t: PType; info: TLineInfo; + idgen: IdGenerator): PNode = + let sym = newSym(skType, getIdent(cache, name), idgen, t.owner, info) + sym.magic = m + sym.typ = t + result = newSymNode(sym) + result.typ = t + +proc atomicTypeX(s: PSym; info: TLineInfo): PNode = + result = newSymNode(s) + result.info = info + +proc mapTypeToAstX(cache: IdentCache; t: PType; info: TLineInfo; idgen: IdGenerator; + inst=false; allowRecursionX=false): PNode + +proc mapTypeToBracketX(cache: IdentCache; name: string; m: TMagic; t: PType; info: TLineInfo; + idgen: IdGenerator; + inst=false): PNode = + result = newNodeIT(nkBracketExpr, if t.n.isNil: info else: t.n.info, t) + result.add atomicTypeX(cache, name, m, t, info, idgen) + for a in t.kids: + if a == nil: + let voidt = atomicTypeX(cache, "void", mVoid, t, info, idgen) + voidt.typ = newType(tyVoid, idgen, t.owner) + result.add voidt + else: + result.add mapTypeToAstX(cache, a, info, idgen, inst) + +proc objectNode(cache: IdentCache; n: PNode; idgen: IdGenerator): PNode = + if n.kind == nkSym: + result = newNodeI(nkIdentDefs, n.info) + result.add n # name + result.add mapTypeToAstX(cache, n.sym.typ, n.info, idgen, true, false) # type + result.add newNodeI(nkEmpty, n.info) # no assigned value + else: + result = copyNode(n) + for i in 0..<n.safeLen: + result.add objectNode(cache, n[i], idgen) + +proc mapTypeToAstX(cache: IdentCache; t: PType; info: TLineInfo; + idgen: IdGenerator; + inst=false; allowRecursionX=false): PNode = + var allowRecursion = allowRecursionX + template atomicType(name, m): untyped = atomicTypeX(cache, name, m, t, info, idgen) + template atomicType(s): untyped = atomicTypeX(s, info) + template mapTypeToAst(t, info): untyped = mapTypeToAstX(cache, t, info, idgen, inst) + template mapTypeToAstR(t, info): untyped = mapTypeToAstX(cache, t, info, idgen, inst, true) + template mapTypeToAst(t, i, info): untyped = + if i<t.len and t[i]!=nil: mapTypeToAstX(cache, t[i], info, idgen, inst) + else: newNodeI(nkEmpty, info) + template mapTypeToBracket(name, m, t, info): untyped = + mapTypeToBracketX(cache, name, m, t, info, idgen, inst) + template newNodeX(kind): untyped = + newNodeIT(kind, if t.n.isNil: info else: t.n.info, t) + template newIdentDefs(n,t): untyped = + var id = newNodeX(nkIdentDefs) + id.add n # name + id.add mapTypeToAst(t, info) # type + id.add newNodeI(nkEmpty, info) # no assigned value + id + template newIdentDefs(s): untyped = newIdentDefs(s, s.typ) + + if inst and not allowRecursion and t.sym != nil: + # getTypeInst behavior: return symbol + return atomicType(t.sym) + + case t.kind + of tyNone: result = atomicType("none", mNone) + of tyBool: result = atomicType("bool", mBool) + of tyChar: result = atomicType("char", mChar) + of tyNil: result = atomicType("nil", mNil) + of tyUntyped: result = atomicType("untyped", mExpr) + of tyTyped: result = atomicType("typed", mStmt) + of tyVoid: result = atomicType("void", mVoid) + of tyEmpty: result = atomicType("empty", mNone) + of tyUncheckedArray: + result = newNodeIT(nkBracketExpr, if t.n.isNil: info else: t.n.info, t) + result.add atomicType("UncheckedArray", mUncheckedArray) + result.add mapTypeToAst(t.elementType, info) + of tyArray: + result = newNodeIT(nkBracketExpr, if t.n.isNil: info else: t.n.info, t) + result.add atomicType("array", mArray) + if inst and t.indexType.kind == tyRange: + var rng = newNodeX(nkInfix) + rng.add newIdentNode(getIdent(cache, ".."), info) + rng.add t.indexType.n[0].copyTree + rng.add t.indexType.n[1].copyTree + result.add rng + else: + result.add mapTypeToAst(t.indexType, info) + result.add mapTypeToAst(t.elementType, info) + of tyTypeDesc: + if t.base != nil: + result = newNodeIT(nkBracketExpr, if t.n.isNil: info else: t.n.info, t) + result.add atomicType("typeDesc", mTypeDesc) + result.add mapTypeToAst(t.base, info) + else: + result = atomicType("typeDesc", mTypeDesc) + of tyGenericInvocation: + result = newNodeIT(nkBracketExpr, if t.n.isNil: info else: t.n.info, t) + for a in t.kids: + result.add mapTypeToAst(a, info) + of tyGenericInst: + if inst: + if allowRecursion: + result = mapTypeToAstR(t.skipModifier, info) + # keep original type info for getType calls on the output node: + result.typ = t + else: + result = newNodeX(nkBracketExpr) + #result.add mapTypeToAst(t.last, info) + result.add mapTypeToAst(t.genericHead, info) + for _, a in t.genericInstParams: + result.add mapTypeToAst(a, info) + else: + result = mapTypeToAstX(cache, t.skipModifier, info, idgen, inst, allowRecursion) + # keep original type info for getType calls on the output node: + result.typ = t + of tyGenericBody: + if inst: + result = mapTypeToAstR(t.typeBodyImpl, info) + else: + result = mapTypeToAst(t.typeBodyImpl, info) + of tyAlias: + result = mapTypeToAstX(cache, t.skipModifier, info, idgen, inst, allowRecursion) + of tyOrdinal: + result = mapTypeToAst(t.skipModifier, info) + of tyDistinct: + if inst: + result = newNodeX(nkDistinctTy) + result.add mapTypeToAst(t.skipModifier, info) + else: + if allowRecursion or t.sym == nil: + result = mapTypeToBracket("distinct", mDistinct, t, info) + else: + result = atomicType(t.sym) + of tyGenericParam, tyForward: + result = atomicType(t.sym) + of tyObject: + if inst: + result = newNodeX(nkObjectTy) + var objectDef = t.sym.ast[2] + if objectDef.kind == nkRefTy: + objectDef = objectDef[0] + result.add objectDef[0].copyTree # copy object pragmas + if t.baseClass == nil: + result.add newNodeI(nkEmpty, info) + else: # handle parent object + var nn = newNodeX(nkOfInherit) + nn.add mapTypeToAst(t.baseClass, info) + result.add nn + if t.n.len > 0: + result.add objectNode(cache, t.n, idgen) + else: + result.add newNodeI(nkEmpty, info) + else: + if allowRecursion or t.sym == nil: + result = newNodeIT(nkObjectTy, if t.n.isNil: info else: t.n.info, t) + result.add newNodeI(nkEmpty, info) + if t.baseClass == nil: + result.add newNodeI(nkEmpty, info) + else: + result.add mapTypeToAst(t.baseClass, info) + result.add copyTree(t.n) + else: + result = atomicType(t.sym) + of tyEnum: + result = newNodeIT(nkEnumTy, if t.n.isNil: info else: t.n.info, t) + result.add newNodeI(nkEmpty, info) # pragma node, currently always empty for enum + for c in t.n.sons: + result.add copyTree(c) + of tyTuple: + if inst: + # only named tuples have a node, unnamed tuples don't + if t.n.isNil: + result = newNodeX(nkTupleConstr) + for subType in t.kids: + result.add mapTypeToAst(subType, info) + else: + result = newNodeX(nkTupleTy) + for s in t.n.sons: + result.add newIdentDefs(s) + else: + result = mapTypeToBracket("tuple", mTuple, t, info) + of tySet: result = mapTypeToBracket("set", mSet, t, info) + of tyPtr: + if inst: + result = newNodeX(nkPtrTy) + result.add mapTypeToAst(t.elementType, info) + else: + result = mapTypeToBracket("ptr", mPtr, t, info) + of tyRef: + if inst: + result = newNodeX(nkRefTy) + result.add mapTypeToAst(t.elementType, info) + else: + result = mapTypeToBracket("ref", mRef, t, info) + of tyVar: + if inst: + result = newNodeX(nkVarTy) + result.add mapTypeToAst(t.elementType, info) + else: + result = mapTypeToBracket("var", mVar, t, info) + of tyLent: result = mapTypeToBracket("lent", mBuiltinType, t, info) + of tySink: result = mapTypeToBracket("sink", mBuiltinType, t, info) + of tySequence: result = mapTypeToBracket("seq", mSeq, t, info) + of tyProc: + if inst: + result = newNodeX(if tfIterator in t.flags: nkIteratorTy else: nkProcTy) + var fp = newNodeX(nkFormalParams) + if t.returnType == nil: + fp.add newNodeI(nkEmpty, info) + else: + fp.add mapTypeToAst(t.returnType, t.n[0].info) + for i in FirstParamAt..<t.kidsLen: + fp.add newIdentDefs(t.n[i], t[i]) + result.add fp + var prag = + if t.n[0].len > 0: + t.n[0][pragmasEffects].copyTree + else: + newNodeI(nkEmpty, info) + if t.callConv != ccClosure or tfExplicitCallConv in t.flags: + if prag.kind == nkEmpty: prag = newNodeI(nkPragma, info) + prag.add newIdentNode(getIdent(cache, $t.callConv), info) + result.add prag + else: + result = mapTypeToBracket("proc", mNone, t, info) + of tyOpenArray: result = mapTypeToBracket("openArray", mOpenArray, t, info) + of tyRange: + result = newNodeIT(nkBracketExpr, if t.n.isNil: info else: t.n.info, t) + result.add atomicType("range", mRange) + if inst and t.n.len == 2: + let rng = newNodeX(nkInfix) + rng.add newIdentNode(getIdent(cache, ".."), info) + rng.add t.n[0].copyTree + rng.add t.n[1].copyTree + result.add rng + else: + result.add t.n[0].copyTree + if t.n.len > 1: + result.add t.n[1].copyTree + of tyPointer: result = atomicType("pointer", mPointer) + of tyString: result = atomicType("string", mString) + of tyCstring: result = atomicType("cstring", mCstring) + of tyInt: result = atomicType("int", mInt) + of tyInt8: result = atomicType("int8", mInt8) + of tyInt16: result = atomicType("int16", mInt16) + of tyInt32: result = atomicType("int32", mInt32) + of tyInt64: result = atomicType("int64", mInt64) + of tyFloat: result = atomicType("float", mFloat) + of tyFloat32: result = atomicType("float32", mFloat32) + of tyFloat64: result = atomicType("float64", mFloat64) + of tyFloat128: result = atomicType("float128", mFloat128) + of tyUInt: result = atomicType("uint", mUInt) + of tyUInt8: result = atomicType("uint8", mUInt8) + of tyUInt16: result = atomicType("uint16", mUInt16) + of tyUInt32: result = atomicType("uint32", mUInt32) + of tyUInt64: result = atomicType("uint64", mUInt64) + of tyVarargs: result = mapTypeToBracket("varargs", mVarargs, t, info) + of tyError: result = atomicType("error", mNone) + of tyBuiltInTypeClass: + result = mapTypeToBracket("builtinTypeClass", mNone, t, info) + of tyUserTypeClass, tyUserTypeClassInst: + if t.isResolvedUserTypeClass: + result = mapTypeToAst(t.last, info) + else: + result = mapTypeToBracket("concept", mNone, t, info) + result.add t.n.copyTree + of tyCompositeTypeClass: + result = mapTypeToBracket("compositeTypeClass", mNone, t, info) + of tyAnd: result = mapTypeToBracket("and", mAnd, t, info) + of tyOr: result = mapTypeToBracket("or", mOr, t, info) + of tyNot: result = mapTypeToBracket("not", mNot, t, info) + of tyIterable: result = mapTypeToBracket("iterable", mIterableType, t, info) + of tyAnything: result = atomicType("anything", mNone) + of tyInferred: result = mapTypeToAstX(cache, t.skipModifier, info, idgen, inst, allowRecursion) + of tyStatic, tyFromExpr: + if inst: + if t.n != nil: result = t.n.copyTree + else: result = atomicType("void", mVoid) + else: + result = newNodeIT(nkBracketExpr, if t.n.isNil: info else: t.n.info, t) + result.add atomicType("static", mNone) + if t.n != nil: + result.add t.n.copyTree + of tyOwned: result = mapTypeToBracket("owned", mBuiltinType, t, info) + of tyConcept: + result = mapTypeToBracket("concept", mNone, t, info) + result.add t.n.copyTree + +proc opMapTypeToAst*(cache: IdentCache; t: PType; info: TLineInfo; idgen: IdGenerator): PNode = + result = mapTypeToAstX(cache, t, info, idgen, inst=false, allowRecursionX=true) + +# the "Inst" version includes generic parameters in the resulting type tree +# and also tries to look like the corresponding Nim type declaration +proc opMapTypeInstToAst*(cache: IdentCache; t: PType; info: TLineInfo; idgen: IdGenerator): PNode = + result = mapTypeToAstX(cache, t, info, idgen, inst=true, allowRecursionX=false) + +# the "Impl" version includes generic parameters in the resulting type tree +# and also tries to look like the corresponding Nim type implementation +proc opMapTypeImplToAst*(cache: IdentCache; t: PType; info: TLineInfo; idgen: IdGenerator): PNode = + result = mapTypeToAstX(cache, t, info, idgen, inst=true, allowRecursionX=true) |