nirtypes.nim 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340
  1. #
  2. #
  3. # The Nim Compiler
  4. # (c) Copyright 2023 Andreas Rumpf
  5. #
  6. # See the file "copying.txt", included in this
  7. # distribution, for details about the copyright.
  8. #
  9. ## Type system for NIR. Close to C's type system but without its quirks.
  10. import std / [assertions, hashes]
  11. import .. / ic / bitabs
  12. type
  13. NirTypeKind* = enum
  14. VoidTy, IntTy, UIntTy, FloatTy, BoolTy, CharTy, NameVal, IntVal,
  15. AnnotationVal,
  16. VarargsTy, # the `...` in a C prototype; also the last "atom"
  17. APtrTy, # pointer to aliasable memory
  18. UPtrTy, # pointer to unique/unaliasable memory
  19. AArrayPtrTy, # pointer to array of aliasable memory
  20. UArrayPtrTy, # pointer to array of unique/unaliasable memory
  21. ArrayTy,
  22. LastArrayTy, # array of unspecified size as a last field inside an object
  23. ObjectTy,
  24. UnionTy,
  25. ProcTy,
  26. ObjectDecl,
  27. UnionDecl,
  28. FieldDecl
  29. const
  30. TypeKindBits = 8'u32
  31. TypeKindMask = (1'u32 shl TypeKindBits) - 1'u32
  32. type
  33. TypeNode* = object # 4 bytes
  34. x: uint32
  35. template kind*(n: TypeNode): NirTypeKind = NirTypeKind(n.x and TypeKindMask)
  36. template operand(n: TypeNode): uint32 = (n.x shr TypeKindBits)
  37. template toX(k: NirTypeKind; operand: uint32): uint32 =
  38. uint32(k) or (operand shl TypeKindBits)
  39. template toX(k: NirTypeKind; operand: LitId): uint32 =
  40. uint32(k) or (operand.uint32 shl TypeKindBits)
  41. type
  42. TypeId* = distinct int
  43. proc `==`*(a, b: TypeId): bool {.borrow.}
  44. proc hash*(a: TypeId): Hash {.borrow.}
  45. type
  46. TypeGraph* = object
  47. nodes: seq[TypeNode]
  48. names: BiTable[string]
  49. numbers: BiTable[uint64]
  50. const
  51. VoidId* = TypeId 0
  52. Bool8Id* = TypeId 1
  53. Char8Id* = TypeId 2
  54. Int8Id* = TypeId 3
  55. Int16Id* = TypeId 4
  56. Int32Id* = TypeId 5
  57. Int64Id* = TypeId 6
  58. UInt8Id* = TypeId 7
  59. UInt16Id* = TypeId 8
  60. UInt32Id* = TypeId 9
  61. UInt64Id* = TypeId 10
  62. Float32Id* = TypeId 11
  63. Float64Id* = TypeId 12
  64. LastBuiltinId* = 12
  65. proc initTypeGraph*(): TypeGraph =
  66. result = TypeGraph(nodes: @[
  67. TypeNode(x: toX(VoidTy, 0'u32)),
  68. TypeNode(x: toX(BoolTy, 8'u32)),
  69. TypeNode(x: toX(CharTy, 8'u32)),
  70. TypeNode(x: toX(IntTy, 8'u32)),
  71. TypeNode(x: toX(IntTy, 16'u32)),
  72. TypeNode(x: toX(IntTy, 32'u32)),
  73. TypeNode(x: toX(IntTy, 64'u32)),
  74. TypeNode(x: toX(UIntTy, 8'u32)),
  75. TypeNode(x: toX(UIntTy, 16'u32)),
  76. TypeNode(x: toX(UIntTy, 32'u32)),
  77. TypeNode(x: toX(UIntTy, 64'u32)),
  78. TypeNode(x: toX(FloatTy, 32'u32)),
  79. TypeNode(x: toX(FloatTy, 64'u32))
  80. ])
  81. assert result.nodes.len == LastBuiltinId+1
  82. type
  83. TypePatchPos* = distinct int
  84. const
  85. InvalidTypePatchPos* = TypePatchPos(-1)
  86. LastAtomicValue = VarargsTy
  87. proc isValid(p: TypePatchPos): bool {.inline.} = p.int != -1
  88. proc prepare(tree: var TypeGraph; kind: NirTypeKind): TypePatchPos =
  89. result = TypePatchPos tree.nodes.len
  90. tree.nodes.add TypeNode(x: toX(kind, 1'u32))
  91. proc isAtom(tree: TypeGraph; pos: int): bool {.inline.} = tree.nodes[pos].kind <= LastAtomicValue
  92. proc isAtom(tree: TypeGraph; pos: TypeId): bool {.inline.} = tree.nodes[pos.int].kind <= LastAtomicValue
  93. proc patch(tree: var TypeGraph; pos: TypePatchPos) =
  94. let pos = pos.int
  95. let k = tree.nodes[pos].kind
  96. assert k > LastAtomicValue
  97. let distance = int32(tree.nodes.len - pos)
  98. assert distance > 0
  99. tree.nodes[pos].x = toX(k, cast[uint32](distance))
  100. proc len*(tree: TypeGraph): int {.inline.} = tree.nodes.len
  101. template rawSpan(n: TypeNode): int = int(operand(n))
  102. proc nextChild(tree: TypeGraph; pos: var int) {.inline.} =
  103. if tree.nodes[pos].kind > LastAtomicValue:
  104. assert tree.nodes[pos].operand > 0'u32
  105. inc pos, tree.nodes[pos].rawSpan
  106. else:
  107. inc pos
  108. iterator sons*(tree: TypeGraph; n: TypeId): TypeId =
  109. var pos = n.int
  110. assert tree.nodes[pos].kind > LastAtomicValue
  111. let last = pos + tree.nodes[pos].rawSpan
  112. inc pos
  113. while pos < last:
  114. yield TypeId pos
  115. nextChild tree, pos
  116. template `[]`*(t: TypeGraph; n: TypeId): TypeNode = t.nodes[n.int]
  117. proc elementType*(tree: TypeGraph; n: TypeId): TypeId {.inline.} =
  118. assert tree[n].kind in {APtrTy, UPtrTy, AArrayPtrTy, UArrayPtrTy, ArrayTy, LastArrayTy}
  119. result = TypeId(n.int+1)
  120. proc kind*(tree: TypeGraph; n: TypeId): NirTypeKind {.inline.} = tree[n].kind
  121. proc span(tree: TypeGraph; pos: int): int {.inline.} =
  122. if tree.nodes[pos].kind <= LastAtomicValue: 1 else: int(tree.nodes[pos].operand)
  123. proc sons2(tree: TypeGraph; n: TypeId): (TypeId, TypeId) =
  124. assert(not isAtom(tree, n.int))
  125. let a = n.int+1
  126. let b = a + span(tree, a)
  127. result = (TypeId a, TypeId b)
  128. proc sons3(tree: TypeGraph; n: TypeId): (TypeId, TypeId, TypeId) =
  129. assert(not isAtom(tree, n.int))
  130. let a = n.int+1
  131. let b = a + span(tree, a)
  132. let c = b + span(tree, b)
  133. result = (TypeId a, TypeId b, TypeId c)
  134. proc arrayLen*(tree: TypeGraph; n: TypeId): BiggestUInt =
  135. assert tree[n].kind == ArrayTy
  136. result = tree.numbers[LitId tree[n].operand]
  137. proc openType*(tree: var TypeGraph; kind: NirTypeKind): TypePatchPos =
  138. assert kind in {APtrTy, UPtrTy, AArrayPtrTy, UArrayPtrTy,
  139. ArrayTy, LastArrayTy, ProcTy, ObjectDecl, UnionDecl,
  140. FieldDecl}
  141. result = prepare(tree, kind)
  142. proc sealType*(tree: var TypeGraph; p: TypePatchPos): TypeId =
  143. # TODO: Search for an existing instance of this type in
  144. # order to reduce memory consumption.
  145. result = TypeId(p)
  146. patch tree, p
  147. proc nominalType*(tree: var TypeGraph; kind: NirTypeKind; name: string): TypeId =
  148. assert kind in {ObjectTy, UnionTy}
  149. result = TypeId tree.nodes.len
  150. tree.nodes.add TypeNode(x: toX(kind, tree.names.getOrIncl(name)))
  151. proc addNominalType*(tree: var TypeGraph; kind: NirTypeKind; name: string) =
  152. assert kind in {ObjectTy, UnionTy}
  153. tree.nodes.add TypeNode(x: toX(kind, tree.names.getOrIncl(name)))
  154. proc addVarargs*(tree: var TypeGraph) =
  155. tree.nodes.add TypeNode(x: toX(VarargsTy, 0'u32))
  156. proc getFloat128Type*(tree: var TypeGraph): TypeId =
  157. result = TypeId tree.nodes.len
  158. tree.nodes.add TypeNode(x: toX(FloatTy, 128'u32))
  159. proc addBuiltinType*(g: var TypeGraph; id: TypeId) =
  160. g.nodes.add g[id]
  161. template firstSon(n: TypeId): TypeId = TypeId(n.int+1)
  162. proc addType*(g: var TypeGraph; t: TypeId) =
  163. # We cannot simply copy `*Decl` nodes. We have to introduce `*Ty` nodes instead:
  164. if g[t].kind in {ObjectDecl, UnionDecl}:
  165. assert g[t.firstSon].kind == NameVal
  166. let name = LitId g[t.firstSon].operand
  167. if g[t].kind == ObjectDecl:
  168. g.nodes.add TypeNode(x: toX(ObjectTy, name))
  169. else:
  170. g.nodes.add TypeNode(x: toX(UnionTy, name))
  171. else:
  172. let pos = t.int
  173. let L = span(g, pos)
  174. let d = g.nodes.len
  175. g.nodes.setLen(d + L)
  176. assert L > 0
  177. for i in 0..<L:
  178. g.nodes[d+i] = g.nodes[pos+i]
  179. proc addArrayLen*(g: var TypeGraph; len: uint64) =
  180. g.nodes.add TypeNode(x: toX(IntVal, g.numbers.getOrIncl(len)))
  181. proc addName*(g: var TypeGraph; name: string) =
  182. g.nodes.add TypeNode(x: toX(NameVal, g.names.getOrIncl(name)))
  183. proc addAnnotation*(g: var TypeGraph; name: string) =
  184. g.nodes.add TypeNode(x: toX(NameVal, g.names.getOrIncl(name)))
  185. proc addField*(g: var TypeGraph; name: string; typ: TypeId) =
  186. let f = g.openType FieldDecl
  187. g.addType typ
  188. g.addName name
  189. discard sealType(g, f)
  190. proc toString*(dest: var string; g: TypeGraph; i: TypeId) =
  191. case g[i].kind
  192. of VoidTy: dest.add "void"
  193. of IntTy:
  194. dest.add "i"
  195. dest.addInt g[i].operand
  196. of UIntTy:
  197. dest.add "u"
  198. dest.addInt g[i].operand
  199. of FloatTy:
  200. dest.add "f"
  201. dest.addInt g[i].operand
  202. of BoolTy:
  203. dest.add "b"
  204. dest.addInt g[i].operand
  205. of CharTy:
  206. dest.add "c"
  207. dest.addInt g[i].operand
  208. of NameVal, AnnotationVal:
  209. dest.add g.names[LitId g[i].operand]
  210. of IntVal:
  211. dest.add $g.numbers[LitId g[i].operand]
  212. of VarargsTy:
  213. dest.add "..."
  214. of APtrTy:
  215. dest.add "aptr["
  216. toString(dest, g, g.elementType(i))
  217. dest.add "]"
  218. of UPtrTy:
  219. dest.add "uptr["
  220. toString(dest, g, g.elementType(i))
  221. dest.add "]"
  222. of AArrayPtrTy:
  223. dest.add "aArrayPtr["
  224. toString(dest, g, g.elementType(i))
  225. dest.add "]"
  226. of UArrayPtrTy:
  227. dest.add "uArrayPtr["
  228. toString(dest, g, g.elementType(i))
  229. dest.add "]"
  230. of ArrayTy:
  231. dest.add "Array["
  232. let (elems, len) = g.sons2(i)
  233. toString(dest, g, elems)
  234. dest.add ", "
  235. toString(dest, g, len)
  236. dest.add "]"
  237. of LastArrayTy:
  238. # array of unspecified size as a last field inside an object
  239. dest.add "LastArrayTy["
  240. toString(dest, g, g.elementType(i))
  241. dest.add "]"
  242. of ObjectTy:
  243. dest.add "object "
  244. dest.add g.names[LitId g[i].operand]
  245. of UnionTy:
  246. dest.add "union "
  247. dest.add g.names[LitId g[i].operand]
  248. of ProcTy:
  249. dest.add "proc["
  250. for t in sons(g, i): toString(dest, g, t)
  251. dest.add "]"
  252. of ObjectDecl:
  253. dest.add "object["
  254. for t in sons(g, i):
  255. toString(dest, g, t)
  256. dest.add '\n'
  257. dest.add "]"
  258. of UnionDecl:
  259. dest.add "union["
  260. for t in sons(g, i):
  261. toString(dest, g, t)
  262. dest.add '\n'
  263. dest.add "]"
  264. of FieldDecl:
  265. let (typ, name) = g.sons2(i)
  266. toString(dest, g, typ)
  267. dest.add ' '
  268. toString(dest, g, name)
  269. proc toString*(dest: var string; g: TypeGraph) =
  270. var i = 0
  271. while i < g.len:
  272. toString(dest, g, TypeId i)
  273. dest.add '\n'
  274. nextChild g, i
  275. proc `$`(g: TypeGraph): string =
  276. result = ""
  277. toString(result, g)
  278. when isMainModule:
  279. var g = initTypeGraph()
  280. let a = g.openType ArrayTy
  281. g.addBuiltinType Int8Id
  282. g.addArrayLen 5'u64
  283. let finalArrayType = sealType(g, a)
  284. let obj = g.openType ObjectDecl
  285. g.nodes.add TypeNode(x: toX(NameVal, g.names.getOrIncl("MyType")))
  286. g.addField "p", finalArrayType
  287. discard sealType(g, obj)
  288. echo g