123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421 |
- #
- #
- # The Nim Compiler
- # (c) Copyright 2020 Andreas Rumpf
- #
- # See the file "copying.txt", included in this
- # distribution, for details about the copyright.
- #
- ## Packed AST representation, mostly based on a seq of nodes.
- ## For IC support. Far future: Rewrite the compiler passes to
- ## use this representation directly in all the transformations,
- ## it is superior.
- import hashes, tables, strtabs
- import bitabs
- import ".." / [ast, options]
- when defined(nimPreviewSlimSystem):
- import std/assertions
- type
- SymId* = distinct int32
- ModuleId* = distinct int32
- NodePos* = distinct int
- NodeId* = distinct int32
- PackedItemId* = object
- module*: LitId # 0 if it's this module
- item*: int32 # same as the in-memory representation
- const
- nilItemId* = PackedItemId(module: LitId(0), item: -1.int32)
- const
- emptyNodeId* = NodeId(-1)
- type
- PackedLineInfo* = object
- line*: uint16
- col*: int16
- file*: LitId
- PackedLib* = object
- kind*: TLibKind
- generated*: bool
- isOverriden*: bool
- name*: LitId
- path*: NodeId
- PackedSym* = object
- kind*: TSymKind
- name*: LitId
- typ*: PackedItemId
- flags*: TSymFlags
- magic*: TMagic
- info*: PackedLineInfo
- ast*: NodeId
- owner*: PackedItemId
- guard*: PackedItemId
- bitsize*: int
- alignment*: int # for alignment
- options*: TOptions
- position*: int
- offset*: int
- externalName*: LitId # instead of TLoc
- locFlags*: TLocFlags
- annex*: PackedLib
- when hasFFI:
- cname*: LitId
- constraint*: NodeId
- PackedType* = object
- kind*: TTypeKind
- callConv*: TCallingConvention
- #nodekind*: TNodeKind
- flags*: TTypeFlags
- types*: seq[PackedItemId]
- n*: NodeId
- #nodeflags*: TNodeFlags
- sym*: PackedItemId
- owner*: PackedItemId
- size*: BiggestInt
- align*: int16
- paddingAtEnd*: int16
- lockLevel*: TLockLevel # lock level as required for deadlock checking
- # not serialized: loc*: TLoc because it is backend-specific
- typeInst*: PackedItemId
- nonUniqueId*: int32
- PackedNode* = object # 28 bytes
- kind*: TNodeKind
- flags*: TNodeFlags
- operand*: int32 # for kind in {nkSym, nkSymDef}: SymId
- # for kind in {nkStrLit, nkIdent, nkNumberLit}: LitId
- # for kind in nkInt32Lit: direct value
- # for non-atom kinds: the number of nodes (for easy skipping)
- typeId*: PackedItemId
- info*: PackedLineInfo
- PackedTree* = object ## usually represents a full Nim module
- nodes*: seq[PackedNode]
- PackedInstantiation* = object
- key*, sym*: PackedItemId
- concreteTypes*: seq[PackedItemId]
- proc `==`*(a, b: SymId): bool {.borrow.}
- proc hash*(a: SymId): Hash {.borrow.}
- proc `==`*(a, b: NodePos): bool {.borrow.}
- #proc `==`*(a, b: PackedItemId): bool {.borrow.}
- proc `==`*(a, b: NodeId): bool {.borrow.}
- proc newTreeFrom*(old: PackedTree): PackedTree =
- result.nodes = @[]
- when false: result.sh = old.sh
- when false:
- proc declareSym*(tree: var PackedTree; kind: TSymKind;
- name: LitId; info: PackedLineInfo): SymId =
- result = SymId(tree.sh.syms.len)
- tree.sh.syms.add PackedSym(kind: kind, name: name, flags: {}, magic: mNone, info: info)
- proc litIdFromName*(tree: PackedTree; name: string): LitId =
- result = tree.sh.strings.getOrIncl(name)
- proc add*(tree: var PackedTree; kind: TNodeKind; token: string; info: PackedLineInfo) =
- tree.nodes.add PackedNode(kind: kind, info: info,
- operand: int32 getOrIncl(tree.sh.strings, token))
- proc add*(tree: var PackedTree; kind: TNodeKind; info: PackedLineInfo) =
- tree.nodes.add PackedNode(kind: kind, operand: 0, info: info)
- proc throwAwayLastNode*(tree: var PackedTree) =
- tree.nodes.setLen(tree.nodes.len-1)
- proc addIdent*(tree: var PackedTree; s: LitId; info: PackedLineInfo) =
- tree.nodes.add PackedNode(kind: nkIdent, operand: int32(s), info: info)
- proc addSym*(tree: var PackedTree; s: int32; info: PackedLineInfo) =
- tree.nodes.add PackedNode(kind: nkSym, operand: s, info: info)
- proc addModuleId*(tree: var PackedTree; s: ModuleId; info: PackedLineInfo) =
- tree.nodes.add PackedNode(kind: nkInt32Lit, operand: int32(s), info: info)
- proc addSymDef*(tree: var PackedTree; s: SymId; info: PackedLineInfo) =
- tree.nodes.add PackedNode(kind: nkSym, operand: int32(s), info: info)
- proc isAtom*(tree: PackedTree; pos: int): bool {.inline.} = tree.nodes[pos].kind <= nkNilLit
- proc copyTree*(dest: var PackedTree; tree: PackedTree; n: NodePos) =
- # and this is why the IR is superior. We can copy subtrees
- # via a linear scan.
- let pos = n.int
- let L = if isAtom(tree, pos): 1 else: tree.nodes[pos].operand
- let d = dest.nodes.len
- dest.nodes.setLen(d + L)
- for i in 0..<L:
- dest.nodes[d+i] = tree.nodes[pos+i]
- when false:
- proc copySym*(dest: var PackedTree; tree: PackedTree; s: SymId): SymId =
- result = SymId(dest.sh.syms.len)
- assert int(s) < tree.sh.syms.len
- let oldSym = tree.sh.syms[s.int]
- dest.sh.syms.add oldSym
- type
- PatchPos = distinct int
- when false:
- proc prepare*(tree: var PackedTree; kind: TNodeKind; info: PackedLineInfo): PatchPos =
- result = PatchPos tree.nodes.len
- tree.nodes.add PackedNode(kind: kind, operand: 0, info: info)
- proc prepare*(tree: var PackedTree; kind: TNodeKind; flags: TNodeFlags; typeId: PackedItemId; info: PackedLineInfo): PatchPos =
- result = PatchPos tree.nodes.len
- tree.nodes.add PackedNode(kind: kind, flags: flags, operand: 0, info: info,
- typeId: typeId)
- proc prepare*(dest: var PackedTree; source: PackedTree; sourcePos: NodePos): PatchPos =
- result = PatchPos dest.nodes.len
- dest.nodes.add source.nodes[sourcePos.int]
- proc patch*(tree: var PackedTree; pos: PatchPos) =
- let pos = pos.int
- assert tree.nodes[pos].kind > nkNilLit
- let distance = int32(tree.nodes.len - pos)
- tree.nodes[pos].operand = distance
- proc len*(tree: PackedTree): int {.inline.} = tree.nodes.len
- proc `[]`*(tree: PackedTree; i: int): lent PackedNode {.inline.} =
- tree.nodes[i]
- proc nextChild(tree: PackedTree; pos: var int) {.inline.} =
- if tree.nodes[pos].kind > nkNilLit:
- assert tree.nodes[pos].operand > 0
- inc pos, tree.nodes[pos].operand
- else:
- inc pos
- iterator sonsReadonly*(tree: PackedTree; n: NodePos): NodePos =
- var pos = n.int
- assert tree.nodes[pos].kind > nkNilLit
- let last = pos + tree.nodes[pos].operand
- inc pos
- while pos < last:
- yield NodePos pos
- nextChild tree, pos
- iterator sons*(dest: var PackedTree; tree: PackedTree; n: NodePos): NodePos =
- let patchPos = prepare(dest, tree, n)
- for x in sonsReadonly(tree, n): yield x
- patch dest, patchPos
- iterator isons*(dest: var PackedTree; tree: PackedTree;
- n: NodePos): (int, NodePos) =
- var i = 0
- for ch0 in sons(dest, tree, n):
- yield (i, ch0)
- inc i
- iterator sonsFrom1*(tree: PackedTree; n: NodePos): NodePos =
- var pos = n.int
- assert tree.nodes[pos].kind > nkNilLit
- let last = pos + tree.nodes[pos].operand
- inc pos
- if pos < last:
- nextChild tree, pos
- while pos < last:
- yield NodePos pos
- nextChild tree, pos
- iterator sonsWithoutLast2*(tree: PackedTree; n: NodePos): NodePos =
- var count = 0
- for child in sonsReadonly(tree, n):
- inc count
- var pos = n.int
- assert tree.nodes[pos].kind > nkNilLit
- let last = pos + tree.nodes[pos].operand
- inc pos
- while pos < last and count > 2:
- yield NodePos pos
- dec count
- nextChild tree, pos
- proc parentImpl(tree: PackedTree; n: NodePos): NodePos =
- # finding the parent of a node is rather easy:
- var pos = n.int - 1
- while pos >= 0 and (isAtom(tree, pos) or (pos + tree.nodes[pos].operand - 1 < n.int)):
- dec pos
- #assert pos >= 0, "node has no parent"
- result = NodePos(pos)
- template parent*(n: NodePos): NodePos = parentImpl(tree, n)
- proc hasXsons*(tree: PackedTree; n: NodePos; x: int): bool =
- var count = 0
- if tree.nodes[n.int].kind > nkNilLit:
- for child in sonsReadonly(tree, n): inc count
- result = count == x
- proc hasAtLeastXsons*(tree: PackedTree; n: NodePos; x: int): bool =
- if tree.nodes[n.int].kind > nkNilLit:
- var count = 0
- for child in sonsReadonly(tree, n):
- inc count
- if count >= x: return true
- return false
- proc firstSon*(tree: PackedTree; n: NodePos): NodePos {.inline.} =
- NodePos(n.int+1)
- proc kind*(tree: PackedTree; n: NodePos): TNodeKind {.inline.} =
- tree.nodes[n.int].kind
- proc litId*(tree: PackedTree; n: NodePos): LitId {.inline.} =
- LitId tree.nodes[n.int].operand
- proc info*(tree: PackedTree; n: NodePos): PackedLineInfo {.inline.} =
- tree.nodes[n.int].info
- template typ*(n: NodePos): PackedItemId =
- tree.nodes[n.int].typeId
- template flags*(n: NodePos): TNodeFlags =
- tree.nodes[n.int].flags
- template operand*(n: NodePos): int32 =
- tree.nodes[n.int].operand
- proc span*(tree: PackedTree; pos: int): int {.inline.} =
- if isAtom(tree, pos): 1 else: tree.nodes[pos].operand
- proc sons2*(tree: PackedTree; n: NodePos): (NodePos, NodePos) =
- assert(not isAtom(tree, n.int))
- let a = n.int+1
- let b = a + span(tree, a)
- result = (NodePos a, NodePos b)
- proc sons3*(tree: PackedTree; n: NodePos): (NodePos, NodePos, NodePos) =
- assert(not isAtom(tree, n.int))
- let a = n.int+1
- let b = a + span(tree, a)
- let c = b + span(tree, b)
- result = (NodePos a, NodePos b, NodePos c)
- proc ithSon*(tree: PackedTree; n: NodePos; i: int): NodePos =
- if tree.nodes[n.int].kind > nkNilLit:
- var count = 0
- for child in sonsReadonly(tree, n):
- if count == i: return child
- inc count
- assert false, "node has no i-th child"
- when false:
- proc `@`*(tree: PackedTree; lit: LitId): lent string {.inline.} =
- tree.sh.strings[lit]
- template kind*(n: NodePos): TNodeKind = tree.nodes[n.int].kind
- template info*(n: NodePos): PackedLineInfo = tree.nodes[n.int].info
- template litId*(n: NodePos): LitId = LitId tree.nodes[n.int].operand
- template symId*(n: NodePos): SymId = SymId tree.nodes[n.int].operand
- proc firstSon*(n: NodePos): NodePos {.inline.} = NodePos(n.int+1)
- when false:
- # xxx `nkStrLit` or `nkStrLit..nkTripleStrLit:` below?
- proc strLit*(tree: PackedTree; n: NodePos): lent string =
- assert n.kind == nkStrLit
- result = tree.sh.strings[LitId tree.nodes[n.int].operand]
- proc strVal*(tree: PackedTree; n: NodePos): string =
- assert n.kind == nkStrLit
- result = tree.sh.strings[LitId tree.nodes[n.int].operand]
- #result = cookedStrLit(raw)
- proc filenameVal*(tree: PackedTree; n: NodePos): string =
- case n.kind
- of nkStrLit:
- result = strVal(tree, n)
- of nkIdent:
- result = tree.sh.strings[n.litId]
- of nkSym:
- result = tree.sh.strings[tree.sh.syms[int n.symId].name]
- else:
- result = ""
- proc identAsStr*(tree: PackedTree; n: NodePos): lent string =
- assert n.kind == nkIdent
- result = tree.sh.strings[LitId tree.nodes[n.int].operand]
- const
- externIntLit* = {nkCharLit,
- nkIntLit,
- nkInt8Lit,
- nkInt16Lit,
- nkInt64Lit,
- nkUIntLit,
- nkUInt8Lit,
- nkUInt16Lit,
- nkUInt32Lit,
- nkUInt64Lit} # nkInt32Lit is missing by design!
- externSIntLit* = {nkIntLit, nkInt8Lit, nkInt16Lit, nkInt64Lit}
- externUIntLit* = {nkUIntLit, nkUInt8Lit, nkUInt16Lit, nkUInt32Lit, nkUInt64Lit}
- directIntLit* = nkInt32Lit
- when false:
- proc identIdImpl(tree: PackedTree; n: NodePos): LitId =
- if n.kind == nkIdent:
- result = n.litId
- elif n.kind == nkSym:
- result = tree.sh.syms[int n.symId].name
- else:
- result = LitId(0)
- template identId*(n: NodePos): LitId = identIdImpl(tree, n)
- template copyInto*(dest, n, body) =
- let patchPos = prepare(dest, tree, n)
- body
- patch dest, patchPos
- template copyIntoKind*(dest, kind, info, body) =
- let patchPos = prepare(dest, kind, info)
- body
- patch dest, patchPos
- when false:
- proc hasPragma*(tree: PackedTree; n: NodePos; pragma: string): bool =
- let litId = tree.sh.strings.getKeyId(pragma)
- if litId == LitId(0):
- return false
- assert n.kind == nkPragma
- for ch0 in sonsReadonly(tree, n):
- if ch0.kind == nkExprColonExpr:
- if ch0.firstSon.identId == litId:
- return true
- elif ch0.identId == litId:
- return true
- proc getNodeId*(tree: PackedTree): NodeId {.inline.} = NodeId tree.nodes.len
- when false:
- proc produceError*(dest: var PackedTree; tree: PackedTree; n: NodePos; msg: string) =
- let patchPos = prepare(dest, nkError, n.info)
- dest.add nkStrLit, msg, n.info
- copyTree(dest, tree, n)
- patch dest, patchPos
- iterator allNodes*(tree: PackedTree): NodePos =
- var p = 0
- while p < tree.len:
- yield NodePos(p)
- let s = span(tree, p)
- inc p, s
- proc toPackedItemId*(item: int32): PackedItemId {.inline.} =
- PackedItemId(module: LitId(0), item: item)
|