packed_ast.nim 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421
  1. #
  2. #
  3. # The Nim Compiler
  4. # (c) Copyright 2020 Andreas Rumpf
  5. #
  6. # See the file "copying.txt", included in this
  7. # distribution, for details about the copyright.
  8. #
  9. ## Packed AST representation, mostly based on a seq of nodes.
  10. ## For IC support. Far future: Rewrite the compiler passes to
  11. ## use this representation directly in all the transformations,
  12. ## it is superior.
  13. import hashes, tables, strtabs
  14. import bitabs
  15. import ".." / [ast, options]
  16. when defined(nimPreviewSlimSystem):
  17. import std/assertions
  18. type
  19. SymId* = distinct int32
  20. ModuleId* = distinct int32
  21. NodePos* = distinct int
  22. NodeId* = distinct int32
  23. PackedItemId* = object
  24. module*: LitId # 0 if it's this module
  25. item*: int32 # same as the in-memory representation
  26. const
  27. nilItemId* = PackedItemId(module: LitId(0), item: -1.int32)
  28. const
  29. emptyNodeId* = NodeId(-1)
  30. type
  31. PackedLineInfo* = object
  32. line*: uint16
  33. col*: int16
  34. file*: LitId
  35. PackedLib* = object
  36. kind*: TLibKind
  37. generated*: bool
  38. isOverriden*: bool
  39. name*: LitId
  40. path*: NodeId
  41. PackedSym* = object
  42. kind*: TSymKind
  43. name*: LitId
  44. typ*: PackedItemId
  45. flags*: TSymFlags
  46. magic*: TMagic
  47. info*: PackedLineInfo
  48. ast*: NodeId
  49. owner*: PackedItemId
  50. guard*: PackedItemId
  51. bitsize*: int
  52. alignment*: int # for alignment
  53. options*: TOptions
  54. position*: int
  55. offset*: int
  56. externalName*: LitId # instead of TLoc
  57. locFlags*: TLocFlags
  58. annex*: PackedLib
  59. when hasFFI:
  60. cname*: LitId
  61. constraint*: NodeId
  62. PackedType* = object
  63. kind*: TTypeKind
  64. callConv*: TCallingConvention
  65. #nodekind*: TNodeKind
  66. flags*: TTypeFlags
  67. types*: seq[PackedItemId]
  68. n*: NodeId
  69. #nodeflags*: TNodeFlags
  70. sym*: PackedItemId
  71. owner*: PackedItemId
  72. size*: BiggestInt
  73. align*: int16
  74. paddingAtEnd*: int16
  75. lockLevel*: TLockLevel # lock level as required for deadlock checking
  76. # not serialized: loc*: TLoc because it is backend-specific
  77. typeInst*: PackedItemId
  78. nonUniqueId*: int32
  79. PackedNode* = object # 28 bytes
  80. kind*: TNodeKind
  81. flags*: TNodeFlags
  82. operand*: int32 # for kind in {nkSym, nkSymDef}: SymId
  83. # for kind in {nkStrLit, nkIdent, nkNumberLit}: LitId
  84. # for kind in nkInt32Lit: direct value
  85. # for non-atom kinds: the number of nodes (for easy skipping)
  86. typeId*: PackedItemId
  87. info*: PackedLineInfo
  88. PackedTree* = object ## usually represents a full Nim module
  89. nodes*: seq[PackedNode]
  90. PackedInstantiation* = object
  91. key*, sym*: PackedItemId
  92. concreteTypes*: seq[PackedItemId]
  93. proc `==`*(a, b: SymId): bool {.borrow.}
  94. proc hash*(a: SymId): Hash {.borrow.}
  95. proc `==`*(a, b: NodePos): bool {.borrow.}
  96. #proc `==`*(a, b: PackedItemId): bool {.borrow.}
  97. proc `==`*(a, b: NodeId): bool {.borrow.}
  98. proc newTreeFrom*(old: PackedTree): PackedTree =
  99. result.nodes = @[]
  100. when false: result.sh = old.sh
  101. when false:
  102. proc declareSym*(tree: var PackedTree; kind: TSymKind;
  103. name: LitId; info: PackedLineInfo): SymId =
  104. result = SymId(tree.sh.syms.len)
  105. tree.sh.syms.add PackedSym(kind: kind, name: name, flags: {}, magic: mNone, info: info)
  106. proc litIdFromName*(tree: PackedTree; name: string): LitId =
  107. result = tree.sh.strings.getOrIncl(name)
  108. proc add*(tree: var PackedTree; kind: TNodeKind; token: string; info: PackedLineInfo) =
  109. tree.nodes.add PackedNode(kind: kind, info: info,
  110. operand: int32 getOrIncl(tree.sh.strings, token))
  111. proc add*(tree: var PackedTree; kind: TNodeKind; info: PackedLineInfo) =
  112. tree.nodes.add PackedNode(kind: kind, operand: 0, info: info)
  113. proc throwAwayLastNode*(tree: var PackedTree) =
  114. tree.nodes.setLen(tree.nodes.len-1)
  115. proc addIdent*(tree: var PackedTree; s: LitId; info: PackedLineInfo) =
  116. tree.nodes.add PackedNode(kind: nkIdent, operand: int32(s), info: info)
  117. proc addSym*(tree: var PackedTree; s: int32; info: PackedLineInfo) =
  118. tree.nodes.add PackedNode(kind: nkSym, operand: s, info: info)
  119. proc addModuleId*(tree: var PackedTree; s: ModuleId; info: PackedLineInfo) =
  120. tree.nodes.add PackedNode(kind: nkInt32Lit, operand: int32(s), info: info)
  121. proc addSymDef*(tree: var PackedTree; s: SymId; info: PackedLineInfo) =
  122. tree.nodes.add PackedNode(kind: nkSym, operand: int32(s), info: info)
  123. proc isAtom*(tree: PackedTree; pos: int): bool {.inline.} = tree.nodes[pos].kind <= nkNilLit
  124. proc copyTree*(dest: var PackedTree; tree: PackedTree; n: NodePos) =
  125. # and this is why the IR is superior. We can copy subtrees
  126. # via a linear scan.
  127. let pos = n.int
  128. let L = if isAtom(tree, pos): 1 else: tree.nodes[pos].operand
  129. let d = dest.nodes.len
  130. dest.nodes.setLen(d + L)
  131. for i in 0..<L:
  132. dest.nodes[d+i] = tree.nodes[pos+i]
  133. when false:
  134. proc copySym*(dest: var PackedTree; tree: PackedTree; s: SymId): SymId =
  135. result = SymId(dest.sh.syms.len)
  136. assert int(s) < tree.sh.syms.len
  137. let oldSym = tree.sh.syms[s.int]
  138. dest.sh.syms.add oldSym
  139. type
  140. PatchPos = distinct int
  141. when false:
  142. proc prepare*(tree: var PackedTree; kind: TNodeKind; info: PackedLineInfo): PatchPos =
  143. result = PatchPos tree.nodes.len
  144. tree.nodes.add PackedNode(kind: kind, operand: 0, info: info)
  145. proc prepare*(tree: var PackedTree; kind: TNodeKind; flags: TNodeFlags; typeId: PackedItemId; info: PackedLineInfo): PatchPos =
  146. result = PatchPos tree.nodes.len
  147. tree.nodes.add PackedNode(kind: kind, flags: flags, operand: 0, info: info,
  148. typeId: typeId)
  149. proc prepare*(dest: var PackedTree; source: PackedTree; sourcePos: NodePos): PatchPos =
  150. result = PatchPos dest.nodes.len
  151. dest.nodes.add source.nodes[sourcePos.int]
  152. proc patch*(tree: var PackedTree; pos: PatchPos) =
  153. let pos = pos.int
  154. assert tree.nodes[pos].kind > nkNilLit
  155. let distance = int32(tree.nodes.len - pos)
  156. tree.nodes[pos].operand = distance
  157. proc len*(tree: PackedTree): int {.inline.} = tree.nodes.len
  158. proc `[]`*(tree: PackedTree; i: int): lent PackedNode {.inline.} =
  159. tree.nodes[i]
  160. proc nextChild(tree: PackedTree; pos: var int) {.inline.} =
  161. if tree.nodes[pos].kind > nkNilLit:
  162. assert tree.nodes[pos].operand > 0
  163. inc pos, tree.nodes[pos].operand
  164. else:
  165. inc pos
  166. iterator sonsReadonly*(tree: PackedTree; n: NodePos): NodePos =
  167. var pos = n.int
  168. assert tree.nodes[pos].kind > nkNilLit
  169. let last = pos + tree.nodes[pos].operand
  170. inc pos
  171. while pos < last:
  172. yield NodePos pos
  173. nextChild tree, pos
  174. iterator sons*(dest: var PackedTree; tree: PackedTree; n: NodePos): NodePos =
  175. let patchPos = prepare(dest, tree, n)
  176. for x in sonsReadonly(tree, n): yield x
  177. patch dest, patchPos
  178. iterator isons*(dest: var PackedTree; tree: PackedTree;
  179. n: NodePos): (int, NodePos) =
  180. var i = 0
  181. for ch0 in sons(dest, tree, n):
  182. yield (i, ch0)
  183. inc i
  184. iterator sonsFrom1*(tree: PackedTree; n: NodePos): NodePos =
  185. var pos = n.int
  186. assert tree.nodes[pos].kind > nkNilLit
  187. let last = pos + tree.nodes[pos].operand
  188. inc pos
  189. if pos < last:
  190. nextChild tree, pos
  191. while pos < last:
  192. yield NodePos pos
  193. nextChild tree, pos
  194. iterator sonsWithoutLast2*(tree: PackedTree; n: NodePos): NodePos =
  195. var count = 0
  196. for child in sonsReadonly(tree, n):
  197. inc count
  198. var pos = n.int
  199. assert tree.nodes[pos].kind > nkNilLit
  200. let last = pos + tree.nodes[pos].operand
  201. inc pos
  202. while pos < last and count > 2:
  203. yield NodePos pos
  204. dec count
  205. nextChild tree, pos
  206. proc parentImpl(tree: PackedTree; n: NodePos): NodePos =
  207. # finding the parent of a node is rather easy:
  208. var pos = n.int - 1
  209. while pos >= 0 and (isAtom(tree, pos) or (pos + tree.nodes[pos].operand - 1 < n.int)):
  210. dec pos
  211. #assert pos >= 0, "node has no parent"
  212. result = NodePos(pos)
  213. template parent*(n: NodePos): NodePos = parentImpl(tree, n)
  214. proc hasXsons*(tree: PackedTree; n: NodePos; x: int): bool =
  215. var count = 0
  216. if tree.nodes[n.int].kind > nkNilLit:
  217. for child in sonsReadonly(tree, n): inc count
  218. result = count == x
  219. proc hasAtLeastXsons*(tree: PackedTree; n: NodePos; x: int): bool =
  220. if tree.nodes[n.int].kind > nkNilLit:
  221. var count = 0
  222. for child in sonsReadonly(tree, n):
  223. inc count
  224. if count >= x: return true
  225. return false
  226. proc firstSon*(tree: PackedTree; n: NodePos): NodePos {.inline.} =
  227. NodePos(n.int+1)
  228. proc kind*(tree: PackedTree; n: NodePos): TNodeKind {.inline.} =
  229. tree.nodes[n.int].kind
  230. proc litId*(tree: PackedTree; n: NodePos): LitId {.inline.} =
  231. LitId tree.nodes[n.int].operand
  232. proc info*(tree: PackedTree; n: NodePos): PackedLineInfo {.inline.} =
  233. tree.nodes[n.int].info
  234. template typ*(n: NodePos): PackedItemId =
  235. tree.nodes[n.int].typeId
  236. template flags*(n: NodePos): TNodeFlags =
  237. tree.nodes[n.int].flags
  238. template operand*(n: NodePos): int32 =
  239. tree.nodes[n.int].operand
  240. proc span*(tree: PackedTree; pos: int): int {.inline.} =
  241. if isAtom(tree, pos): 1 else: tree.nodes[pos].operand
  242. proc sons2*(tree: PackedTree; n: NodePos): (NodePos, NodePos) =
  243. assert(not isAtom(tree, n.int))
  244. let a = n.int+1
  245. let b = a + span(tree, a)
  246. result = (NodePos a, NodePos b)
  247. proc sons3*(tree: PackedTree; n: NodePos): (NodePos, NodePos, NodePos) =
  248. assert(not isAtom(tree, n.int))
  249. let a = n.int+1
  250. let b = a + span(tree, a)
  251. let c = b + span(tree, b)
  252. result = (NodePos a, NodePos b, NodePos c)
  253. proc ithSon*(tree: PackedTree; n: NodePos; i: int): NodePos =
  254. if tree.nodes[n.int].kind > nkNilLit:
  255. var count = 0
  256. for child in sonsReadonly(tree, n):
  257. if count == i: return child
  258. inc count
  259. assert false, "node has no i-th child"
  260. when false:
  261. proc `@`*(tree: PackedTree; lit: LitId): lent string {.inline.} =
  262. tree.sh.strings[lit]
  263. template kind*(n: NodePos): TNodeKind = tree.nodes[n.int].kind
  264. template info*(n: NodePos): PackedLineInfo = tree.nodes[n.int].info
  265. template litId*(n: NodePos): LitId = LitId tree.nodes[n.int].operand
  266. template symId*(n: NodePos): SymId = SymId tree.nodes[n.int].operand
  267. proc firstSon*(n: NodePos): NodePos {.inline.} = NodePos(n.int+1)
  268. when false:
  269. # xxx `nkStrLit` or `nkStrLit..nkTripleStrLit:` below?
  270. proc strLit*(tree: PackedTree; n: NodePos): lent string =
  271. assert n.kind == nkStrLit
  272. result = tree.sh.strings[LitId tree.nodes[n.int].operand]
  273. proc strVal*(tree: PackedTree; n: NodePos): string =
  274. assert n.kind == nkStrLit
  275. result = tree.sh.strings[LitId tree.nodes[n.int].operand]
  276. #result = cookedStrLit(raw)
  277. proc filenameVal*(tree: PackedTree; n: NodePos): string =
  278. case n.kind
  279. of nkStrLit:
  280. result = strVal(tree, n)
  281. of nkIdent:
  282. result = tree.sh.strings[n.litId]
  283. of nkSym:
  284. result = tree.sh.strings[tree.sh.syms[int n.symId].name]
  285. else:
  286. result = ""
  287. proc identAsStr*(tree: PackedTree; n: NodePos): lent string =
  288. assert n.kind == nkIdent
  289. result = tree.sh.strings[LitId tree.nodes[n.int].operand]
  290. const
  291. externIntLit* = {nkCharLit,
  292. nkIntLit,
  293. nkInt8Lit,
  294. nkInt16Lit,
  295. nkInt64Lit,
  296. nkUIntLit,
  297. nkUInt8Lit,
  298. nkUInt16Lit,
  299. nkUInt32Lit,
  300. nkUInt64Lit} # nkInt32Lit is missing by design!
  301. externSIntLit* = {nkIntLit, nkInt8Lit, nkInt16Lit, nkInt64Lit}
  302. externUIntLit* = {nkUIntLit, nkUInt8Lit, nkUInt16Lit, nkUInt32Lit, nkUInt64Lit}
  303. directIntLit* = nkInt32Lit
  304. when false:
  305. proc identIdImpl(tree: PackedTree; n: NodePos): LitId =
  306. if n.kind == nkIdent:
  307. result = n.litId
  308. elif n.kind == nkSym:
  309. result = tree.sh.syms[int n.symId].name
  310. else:
  311. result = LitId(0)
  312. template identId*(n: NodePos): LitId = identIdImpl(tree, n)
  313. template copyInto*(dest, n, body) =
  314. let patchPos = prepare(dest, tree, n)
  315. body
  316. patch dest, patchPos
  317. template copyIntoKind*(dest, kind, info, body) =
  318. let patchPos = prepare(dest, kind, info)
  319. body
  320. patch dest, patchPos
  321. when false:
  322. proc hasPragma*(tree: PackedTree; n: NodePos; pragma: string): bool =
  323. let litId = tree.sh.strings.getKeyId(pragma)
  324. if litId == LitId(0):
  325. return false
  326. assert n.kind == nkPragma
  327. for ch0 in sonsReadonly(tree, n):
  328. if ch0.kind == nkExprColonExpr:
  329. if ch0.firstSon.identId == litId:
  330. return true
  331. elif ch0.identId == litId:
  332. return true
  333. proc getNodeId*(tree: PackedTree): NodeId {.inline.} = NodeId tree.nodes.len
  334. when false:
  335. proc produceError*(dest: var PackedTree; tree: PackedTree; n: NodePos; msg: string) =
  336. let patchPos = prepare(dest, nkError, n.info)
  337. dest.add nkStrLit, msg, n.info
  338. copyTree(dest, tree, n)
  339. patch dest, patchPos
  340. iterator allNodes*(tree: PackedTree): NodePos =
  341. var p = 0
  342. while p < tree.len:
  343. yield NodePos(p)
  344. let s = span(tree, p)
  345. inc p, s
  346. proc toPackedItemId*(item: int32): PackedItemId {.inline.} =
  347. PackedItemId(module: LitId(0), item: item)