123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340 |
- #
- #
- # The Nim Compiler
- # (c) Copyright 2023 Andreas Rumpf
- #
- # See the file "copying.txt", included in this
- # distribution, for details about the copyright.
- #
- ## Type system for NIR. Close to C's type system but without its quirks.
- import std / [assertions, hashes]
- import .. / ic / bitabs
- type
- NirTypeKind* = enum
- VoidTy, IntTy, UIntTy, FloatTy, BoolTy, CharTy, NameVal, IntVal,
- AnnotationVal,
- VarargsTy, # the `...` in a C prototype; also the last "atom"
- APtrTy, # pointer to aliasable memory
- UPtrTy, # pointer to unique/unaliasable memory
- AArrayPtrTy, # pointer to array of aliasable memory
- UArrayPtrTy, # pointer to array of unique/unaliasable memory
- ArrayTy,
- LastArrayTy, # array of unspecified size as a last field inside an object
- ObjectTy,
- UnionTy,
- ProcTy,
- ObjectDecl,
- UnionDecl,
- FieldDecl
- const
- TypeKindBits = 8'u32
- TypeKindMask = (1'u32 shl TypeKindBits) - 1'u32
- type
- TypeNode* = object # 4 bytes
- x: uint32
- template kind*(n: TypeNode): NirTypeKind = NirTypeKind(n.x and TypeKindMask)
- template operand(n: TypeNode): uint32 = (n.x shr TypeKindBits)
- template toX(k: NirTypeKind; operand: uint32): uint32 =
- uint32(k) or (operand shl TypeKindBits)
- template toX(k: NirTypeKind; operand: LitId): uint32 =
- uint32(k) or (operand.uint32 shl TypeKindBits)
- type
- TypeId* = distinct int
- proc `==`*(a, b: TypeId): bool {.borrow.}
- proc hash*(a: TypeId): Hash {.borrow.}
- type
- TypeGraph* = object
- nodes: seq[TypeNode]
- names: BiTable[string]
- numbers: BiTable[uint64]
- const
- VoidId* = TypeId 0
- Bool8Id* = TypeId 1
- Char8Id* = TypeId 2
- Int8Id* = TypeId 3
- Int16Id* = TypeId 4
- Int32Id* = TypeId 5
- Int64Id* = TypeId 6
- UInt8Id* = TypeId 7
- UInt16Id* = TypeId 8
- UInt32Id* = TypeId 9
- UInt64Id* = TypeId 10
- Float32Id* = TypeId 11
- Float64Id* = TypeId 12
- LastBuiltinId* = 12
- proc initTypeGraph*(): TypeGraph =
- result = TypeGraph(nodes: @[
- TypeNode(x: toX(VoidTy, 0'u32)),
- TypeNode(x: toX(BoolTy, 8'u32)),
- TypeNode(x: toX(CharTy, 8'u32)),
- TypeNode(x: toX(IntTy, 8'u32)),
- TypeNode(x: toX(IntTy, 16'u32)),
- TypeNode(x: toX(IntTy, 32'u32)),
- TypeNode(x: toX(IntTy, 64'u32)),
- TypeNode(x: toX(UIntTy, 8'u32)),
- TypeNode(x: toX(UIntTy, 16'u32)),
- TypeNode(x: toX(UIntTy, 32'u32)),
- TypeNode(x: toX(UIntTy, 64'u32)),
- TypeNode(x: toX(FloatTy, 32'u32)),
- TypeNode(x: toX(FloatTy, 64'u32))
- ])
- assert result.nodes.len == LastBuiltinId+1
- type
- TypePatchPos* = distinct int
- const
- InvalidTypePatchPos* = TypePatchPos(-1)
- LastAtomicValue = VarargsTy
- proc isValid(p: TypePatchPos): bool {.inline.} = p.int != -1
- proc prepare(tree: var TypeGraph; kind: NirTypeKind): TypePatchPos =
- result = TypePatchPos tree.nodes.len
- tree.nodes.add TypeNode(x: toX(kind, 1'u32))
- proc isAtom(tree: TypeGraph; pos: int): bool {.inline.} = tree.nodes[pos].kind <= LastAtomicValue
- proc isAtom(tree: TypeGraph; pos: TypeId): bool {.inline.} = tree.nodes[pos.int].kind <= LastAtomicValue
- proc patch(tree: var TypeGraph; pos: TypePatchPos) =
- let pos = pos.int
- let k = tree.nodes[pos].kind
- assert k > LastAtomicValue
- let distance = int32(tree.nodes.len - pos)
- assert distance > 0
- tree.nodes[pos].x = toX(k, cast[uint32](distance))
- proc len*(tree: TypeGraph): int {.inline.} = tree.nodes.len
- template rawSpan(n: TypeNode): int = int(operand(n))
- proc nextChild(tree: TypeGraph; pos: var int) {.inline.} =
- if tree.nodes[pos].kind > LastAtomicValue:
- assert tree.nodes[pos].operand > 0'u32
- inc pos, tree.nodes[pos].rawSpan
- else:
- inc pos
- iterator sons*(tree: TypeGraph; n: TypeId): TypeId =
- var pos = n.int
- assert tree.nodes[pos].kind > LastAtomicValue
- let last = pos + tree.nodes[pos].rawSpan
- inc pos
- while pos < last:
- yield TypeId pos
- nextChild tree, pos
- template `[]`*(t: TypeGraph; n: TypeId): TypeNode = t.nodes[n.int]
- proc elementType*(tree: TypeGraph; n: TypeId): TypeId {.inline.} =
- assert tree[n].kind in {APtrTy, UPtrTy, AArrayPtrTy, UArrayPtrTy, ArrayTy, LastArrayTy}
- result = TypeId(n.int+1)
- proc kind*(tree: TypeGraph; n: TypeId): NirTypeKind {.inline.} = tree[n].kind
- proc span(tree: TypeGraph; pos: int): int {.inline.} =
- if tree.nodes[pos].kind <= LastAtomicValue: 1 else: int(tree.nodes[pos].operand)
- proc sons2(tree: TypeGraph; n: TypeId): (TypeId, TypeId) =
- assert(not isAtom(tree, n.int))
- let a = n.int+1
- let b = a + span(tree, a)
- result = (TypeId a, TypeId b)
- proc sons3(tree: TypeGraph; n: TypeId): (TypeId, TypeId, TypeId) =
- assert(not isAtom(tree, n.int))
- let a = n.int+1
- let b = a + span(tree, a)
- let c = b + span(tree, b)
- result = (TypeId a, TypeId b, TypeId c)
- proc arrayLen*(tree: TypeGraph; n: TypeId): BiggestUInt =
- assert tree[n].kind == ArrayTy
- result = tree.numbers[LitId tree[n].operand]
- proc openType*(tree: var TypeGraph; kind: NirTypeKind): TypePatchPos =
- assert kind in {APtrTy, UPtrTy, AArrayPtrTy, UArrayPtrTy,
- ArrayTy, LastArrayTy, ProcTy, ObjectDecl, UnionDecl,
- FieldDecl}
- result = prepare(tree, kind)
- proc sealType*(tree: var TypeGraph; p: TypePatchPos): TypeId =
- # TODO: Search for an existing instance of this type in
- # order to reduce memory consumption.
- result = TypeId(p)
- patch tree, p
- proc nominalType*(tree: var TypeGraph; kind: NirTypeKind; name: string): TypeId =
- assert kind in {ObjectTy, UnionTy}
- result = TypeId tree.nodes.len
- tree.nodes.add TypeNode(x: toX(kind, tree.names.getOrIncl(name)))
- proc addNominalType*(tree: var TypeGraph; kind: NirTypeKind; name: string) =
- assert kind in {ObjectTy, UnionTy}
- tree.nodes.add TypeNode(x: toX(kind, tree.names.getOrIncl(name)))
- proc addVarargs*(tree: var TypeGraph) =
- tree.nodes.add TypeNode(x: toX(VarargsTy, 0'u32))
- proc getFloat128Type*(tree: var TypeGraph): TypeId =
- result = TypeId tree.nodes.len
- tree.nodes.add TypeNode(x: toX(FloatTy, 128'u32))
- proc addBuiltinType*(g: var TypeGraph; id: TypeId) =
- g.nodes.add g[id]
- template firstSon(n: TypeId): TypeId = TypeId(n.int+1)
- proc addType*(g: var TypeGraph; t: TypeId) =
- # We cannot simply copy `*Decl` nodes. We have to introduce `*Ty` nodes instead:
- if g[t].kind in {ObjectDecl, UnionDecl}:
- assert g[t.firstSon].kind == NameVal
- let name = LitId g[t.firstSon].operand
- if g[t].kind == ObjectDecl:
- g.nodes.add TypeNode(x: toX(ObjectTy, name))
- else:
- g.nodes.add TypeNode(x: toX(UnionTy, name))
- else:
- let pos = t.int
- let L = span(g, pos)
- let d = g.nodes.len
- g.nodes.setLen(d + L)
- assert L > 0
- for i in 0..<L:
- g.nodes[d+i] = g.nodes[pos+i]
- proc addArrayLen*(g: var TypeGraph; len: uint64) =
- g.nodes.add TypeNode(x: toX(IntVal, g.numbers.getOrIncl(len)))
- proc addName*(g: var TypeGraph; name: string) =
- g.nodes.add TypeNode(x: toX(NameVal, g.names.getOrIncl(name)))
- proc addAnnotation*(g: var TypeGraph; name: string) =
- g.nodes.add TypeNode(x: toX(NameVal, g.names.getOrIncl(name)))
- proc addField*(g: var TypeGraph; name: string; typ: TypeId) =
- let f = g.openType FieldDecl
- g.addType typ
- g.addName name
- discard sealType(g, f)
- proc toString*(dest: var string; g: TypeGraph; i: TypeId) =
- case g[i].kind
- of VoidTy: dest.add "void"
- of IntTy:
- dest.add "i"
- dest.addInt g[i].operand
- of UIntTy:
- dest.add "u"
- dest.addInt g[i].operand
- of FloatTy:
- dest.add "f"
- dest.addInt g[i].operand
- of BoolTy:
- dest.add "b"
- dest.addInt g[i].operand
- of CharTy:
- dest.add "c"
- dest.addInt g[i].operand
- of NameVal, AnnotationVal:
- dest.add g.names[LitId g[i].operand]
- of IntVal:
- dest.add $g.numbers[LitId g[i].operand]
- of VarargsTy:
- dest.add "..."
- of APtrTy:
- dest.add "aptr["
- toString(dest, g, g.elementType(i))
- dest.add "]"
- of UPtrTy:
- dest.add "uptr["
- toString(dest, g, g.elementType(i))
- dest.add "]"
- of AArrayPtrTy:
- dest.add "aArrayPtr["
- toString(dest, g, g.elementType(i))
- dest.add "]"
- of UArrayPtrTy:
- dest.add "uArrayPtr["
- toString(dest, g, g.elementType(i))
- dest.add "]"
- of ArrayTy:
- dest.add "Array["
- let (elems, len) = g.sons2(i)
- toString(dest, g, elems)
- dest.add ", "
- toString(dest, g, len)
- dest.add "]"
- of LastArrayTy:
- # array of unspecified size as a last field inside an object
- dest.add "LastArrayTy["
- toString(dest, g, g.elementType(i))
- dest.add "]"
- of ObjectTy:
- dest.add "object "
- dest.add g.names[LitId g[i].operand]
- of UnionTy:
- dest.add "union "
- dest.add g.names[LitId g[i].operand]
- of ProcTy:
- dest.add "proc["
- for t in sons(g, i): toString(dest, g, t)
- dest.add "]"
- of ObjectDecl:
- dest.add "object["
- for t in sons(g, i):
- toString(dest, g, t)
- dest.add '\n'
- dest.add "]"
- of UnionDecl:
- dest.add "union["
- for t in sons(g, i):
- toString(dest, g, t)
- dest.add '\n'
- dest.add "]"
- of FieldDecl:
- let (typ, name) = g.sons2(i)
- toString(dest, g, typ)
- dest.add ' '
- toString(dest, g, name)
- proc toString*(dest: var string; g: TypeGraph) =
- var i = 0
- while i < g.len:
- toString(dest, g, TypeId i)
- dest.add '\n'
- nextChild g, i
- proc `$`(g: TypeGraph): string =
- result = ""
- toString(result, g)
- when isMainModule:
- var g = initTypeGraph()
- let a = g.openType ArrayTy
- g.addBuiltinType Int8Id
- g.addArrayLen 5'u64
- let finalArrayType = sealType(g, a)
- let obj = g.openType ObjectDecl
- g.nodes.add TypeNode(x: toX(NameVal, g.names.getOrIncl("MyType")))
- g.addField "p", finalArrayType
- discard sealType(g, obj)
- echo g
|