arc.nim 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237
  1. #
  2. #
  3. # Nim's Runtime Library
  4. # (c) Copyright 2019 Andreas Rumpf
  5. #
  6. # See the file "copying.txt", included in this
  7. # distribution, for details about the copyright.
  8. #
  9. #[
  10. In this new runtime we simplify the object layouts a bit: The runtime type
  11. information is only accessed for the objects that have it and it's always
  12. at offset 0 then. The ``ref`` object header is independent from the
  13. runtime type and only contains a reference count.
  14. Object subtyping is checked via the generated 'name'. This should have
  15. comparable overhead to the old pointer chasing approach but has the benefit
  16. that it works across DLL boundaries.
  17. The generated name is a concatenation of the object names in the hierarchy
  18. so that a subtype check becomes a substring check. For example::
  19. type
  20. ObjectA = object of RootObj
  21. ObjectB = object of ObjectA
  22. ObjectA's ``name`` is "|ObjectA|RootObj|".
  23. ObjectB's ``name`` is "|ObjectB|ObjectA|RootObj|".
  24. Now to check for ``x of ObjectB`` we need to check
  25. for ``x.typ.name.hasSubstring("|ObjectB|")``. In the actual implementation,
  26. however, we could also use a
  27. hash of ``package & "." & module & "." & name`` to save space.
  28. ]#
  29. when defined(gcOrc):
  30. const
  31. rcIncrement = 0b10000 # so that lowest 4 bits are not touched
  32. rcMask = 0b1111
  33. rcShift = 4 # shift by rcShift to get the reference counter
  34. else:
  35. const
  36. rcIncrement = 0b1000 # so that lowest 3 bits are not touched
  37. rcMask = 0b111
  38. rcShift = 3 # shift by rcShift to get the reference counter
  39. type
  40. RefHeader = object
  41. rc: int # the object header is now a single RC field.
  42. # we could remove it in non-debug builds for the 'owned ref'
  43. # design but this seems unwise.
  44. when defined(gcOrc):
  45. rootIdx: int # thanks to this we can delete potential cycle roots
  46. # in O(1) without doubly linked lists
  47. when defined(nimArcDebug) or defined(nimArcIds):
  48. refId: int
  49. Cell = ptr RefHeader
  50. template head(p: pointer): Cell =
  51. cast[Cell](cast[int](p) -% sizeof(RefHeader))
  52. const
  53. traceCollector = defined(traceArc)
  54. when defined(nimArcDebug):
  55. include cellsets
  56. const traceId = 20 # 1037
  57. var gRefId: int
  58. var freedCells: CellSet
  59. elif defined(nimArcIds):
  60. var gRefId: int
  61. proc nimNewObj(size, alignment: int): pointer {.compilerRtl.} =
  62. let hdrSize = align(sizeof(RefHeader), alignment)
  63. let s = size + hdrSize
  64. when defined(nimscript):
  65. discard
  66. else:
  67. result = alignedAlloc0(s, alignment) +! hdrSize
  68. when defined(nimArcDebug) or defined(nimArcIds):
  69. head(result).refId = gRefId
  70. atomicInc gRefId
  71. if head(result).refId == traceId:
  72. writeStackTrace()
  73. cfprintf(cstderr, "[nimNewObj] %p %ld\n", result, head(result).rc shr rcShift)
  74. when traceCollector:
  75. cprintf("[Allocated] %p result: %p\n", result -! sizeof(RefHeader), result)
  76. proc nimNewObjUninit(size, alignment: int): pointer {.compilerRtl.} =
  77. # Same as 'newNewObj' but do not initialize the memory to zero.
  78. # The codegen proved for us that this is not necessary.
  79. let hdrSize = align(sizeof(RefHeader), alignment)
  80. let s = size + hdrSize
  81. when defined(nimscript):
  82. discard
  83. else:
  84. result = cast[ptr RefHeader](alignedAlloc(s, alignment) +! hdrSize)
  85. head(result).rc = 0
  86. when defined(gcOrc):
  87. head(result).rootIdx = 0
  88. when defined(nimArcDebug):
  89. head(result).refId = gRefId
  90. atomicInc gRefId
  91. if head(result).refId == traceId:
  92. writeStackTrace()
  93. cfprintf(cstderr, "[nimNewObjUninit] %p %ld\n", result, head(result).rc shr rcShift)
  94. when traceCollector:
  95. cprintf("[Allocated] %p result: %p\n", result -! sizeof(RefHeader), result)
  96. proc nimDecWeakRef(p: pointer) {.compilerRtl, inl.} =
  97. dec head(p).rc, rcIncrement
  98. proc nimIncRef(p: pointer) {.compilerRtl, inl.} =
  99. when defined(nimArcDebug):
  100. if head(p).refId == traceId:
  101. writeStackTrace()
  102. cfprintf(cstderr, "[IncRef] %p %ld\n", p, head(p).rc shr rcShift)
  103. inc head(p).rc, rcIncrement
  104. when traceCollector:
  105. cprintf("[INCREF] %p\n", head(p))
  106. proc unsureAsgnRef(dest: ptr pointer, src: pointer) {.inline.} =
  107. # This is only used by the old RTTI mechanism and we know
  108. # that 'dest[]' is nil and needs no destruction. Which is really handy
  109. # as we cannot destroy the object reliably if it's an object of unknown
  110. # compile-time type.
  111. dest[] = src
  112. if src != nil: nimIncRef src
  113. when not defined(nimscript) and defined(nimArcDebug):
  114. proc deallocatedRefId*(p: pointer): int =
  115. ## Returns the ref's ID if the ref was already deallocated. This
  116. ## is a memory corruption check. Returns 0 if there is no error.
  117. let c = head(p)
  118. if freedCells.data != nil and freedCells.contains(c):
  119. result = c.refId
  120. else:
  121. result = 0
  122. proc nimRawDispose(p: pointer, alignment: int) {.compilerRtl.} =
  123. when not defined(nimscript):
  124. when traceCollector:
  125. cprintf("[Freed] %p\n", p -! sizeof(RefHeader))
  126. when defined(nimOwnedEnabled):
  127. if head(p).rc >= rcIncrement:
  128. cstderr.rawWrite "[FATAL] dangling references exist\n"
  129. quit 1
  130. when defined(nimArcDebug):
  131. # we do NOT really free the memory here in order to reliably detect use-after-frees
  132. if freedCells.data == nil: init(freedCells)
  133. freedCells.incl head(p)
  134. else:
  135. let hdrSize = align(sizeof(RefHeader), alignment)
  136. alignedDealloc(p -! hdrSize, alignment)
  137. template dispose*[T](x: owned(ref T)) = nimRawDispose(cast[pointer](x), T.alignOf)
  138. #proc dispose*(x: pointer) = nimRawDispose(x)
  139. proc nimDestroyAndDispose(p: pointer) {.compilerRtl, raises: [].} =
  140. let rti = cast[ptr PNimTypeV2](p)
  141. if rti.destructor != nil:
  142. cast[DestructorProc](rti.destructor)(p)
  143. when false:
  144. cstderr.rawWrite cast[ptr PNimTypeV2](p)[].name
  145. cstderr.rawWrite "\n"
  146. if d == nil:
  147. cstderr.rawWrite "bah, nil\n"
  148. else:
  149. cstderr.rawWrite "has destructor!\n"
  150. nimRawDispose(p, rti.align)
  151. when defined(gcOrc):
  152. when defined(nimThinout):
  153. include cyclebreaker
  154. else:
  155. include orc
  156. #include cyclecollector
  157. proc nimDecRefIsLast(p: pointer): bool {.compilerRtl, inl.} =
  158. if p != nil:
  159. var cell = head(p)
  160. when defined(nimArcDebug):
  161. if cell.refId == traceId:
  162. writeStackTrace()
  163. cfprintf(cstderr, "[DecRef] %p %ld\n", p, cell.rc shr rcShift)
  164. if (cell.rc and not rcMask) == 0:
  165. result = true
  166. when traceCollector:
  167. cprintf("[ABOUT TO DESTROY] %p\n", cell)
  168. else:
  169. dec cell.rc, rcIncrement
  170. # According to Lins it's correct to do nothing else here.
  171. when traceCollector:
  172. cprintf("[DeCREF] %p\n", cell)
  173. proc GC_unref*[T](x: ref T) =
  174. ## New runtime only supports this operation for 'ref T'.
  175. if nimDecRefIsLast(cast[pointer](x)):
  176. # XXX this does NOT work for virtual destructors!
  177. `=destroy`(x[])
  178. nimRawDispose(cast[pointer](x), T.alignOf)
  179. proc GC_ref*[T](x: ref T) =
  180. ## New runtime only supports this operation for 'ref T'.
  181. if x != nil: nimIncRef(cast[pointer](x))
  182. when not defined(gcOrc):
  183. template GC_fullCollect* =
  184. ## Forces a full garbage collection pass. With ``--gc:arc`` a nop.
  185. discard
  186. template setupForeignThreadGc* =
  187. ## With ``--gc:arc`` a nop.
  188. discard
  189. template tearDownForeignThreadGc* =
  190. ## With ``--gc:arc`` a nop.
  191. discard
  192. proc isObj(obj: PNimTypeV2, subclass: cstring): bool {.compilerRtl, inl.} =
  193. proc strstr(s, sub: cstring): cstring {.header: "<string.h>", importc.}
  194. result = strstr(obj.name, subclass) != nil
  195. proc chckObj(obj: PNimTypeV2, subclass: cstring) {.compilerRtl.} =
  196. # checks if obj is of type subclass:
  197. if not isObj(obj, subclass): sysFatal(ObjectConversionDefect, "invalid object conversion")