assign.nim 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244
  1. #
  2. #
  3. # Nim's Runtime Library
  4. # (c) Copyright 2012 Andreas Rumpf
  5. #
  6. # See the file "copying.txt", included in this
  7. # distribution, for details about the copyright.
  8. #
  9. proc genericResetAux(dest: pointer, n: ptr TNimNode) {.benign.}
  10. proc genericAssignAux(dest, src: pointer, mt: PNimType, shallow: bool) {.benign.}
  11. proc genericAssignAux(dest, src: pointer, n: ptr TNimNode,
  12. shallow: bool) {.benign.} =
  13. var
  14. d = cast[ByteAddress](dest)
  15. s = cast[ByteAddress](src)
  16. case n.kind
  17. of nkSlot:
  18. genericAssignAux(cast[pointer](d +% n.offset),
  19. cast[pointer](s +% n.offset), n.typ, shallow)
  20. of nkList:
  21. for i in 0..n.len-1:
  22. genericAssignAux(dest, src, n.sons[i], shallow)
  23. of nkCase:
  24. var dd = selectBranch(dest, n)
  25. var m = selectBranch(src, n)
  26. # reset if different branches are in use; note different branches also
  27. # imply that's not self-assignment (``x = x``)!
  28. if m != dd and dd != nil:
  29. genericResetAux(dest, dd)
  30. copyMem(cast[pointer](d +% n.offset), cast[pointer](s +% n.offset),
  31. n.typ.size)
  32. if m != nil:
  33. genericAssignAux(dest, src, m, shallow)
  34. of nkNone: sysAssert(false, "genericAssignAux")
  35. #else:
  36. # echo "ugh memory corruption! ", n.kind
  37. # quit 1
  38. proc genericAssignAux(dest, src: pointer, mt: PNimType, shallow: bool) =
  39. var
  40. d = cast[ByteAddress](dest)
  41. s = cast[ByteAddress](src)
  42. sysAssert(mt != nil, "genericAssignAux 2")
  43. case mt.kind
  44. of tyString:
  45. var x = cast[PPointer](dest)
  46. var s2 = cast[PPointer](s)[]
  47. if s2 == nil or shallow or (
  48. cast[PGenericSeq](s2).reserved and seqShallowFlag) != 0:
  49. unsureAsgnRef(x, s2)
  50. else:
  51. unsureAsgnRef(x, copyString(cast[NimString](s2)))
  52. of tySequence:
  53. var s2 = cast[PPointer](src)[]
  54. var seq = cast[PGenericSeq](s2)
  55. var x = cast[PPointer](dest)
  56. if s2 == nil or shallow or (seq.reserved and seqShallowFlag) != 0:
  57. # this can happen! nil sequences are allowed
  58. unsureAsgnRef(x, s2)
  59. return
  60. sysAssert(dest != nil, "genericAssignAux 3")
  61. if ntfNoRefs in mt.base.flags:
  62. var ss = nimNewSeqOfCap(mt, seq.len)
  63. cast[PGenericSeq](ss).len = seq.len
  64. unsureAsgnRef(x, ss)
  65. var dst = cast[ByteAddress](cast[PPointer](dest)[])
  66. copyMem(cast[pointer](dst +% GenericSeqSize),
  67. cast[pointer](cast[ByteAddress](s2) +% GenericSeqSize),
  68. seq.len * mt.base.size)
  69. else:
  70. unsureAsgnRef(x, newSeq(mt, seq.len))
  71. var dst = cast[ByteAddress](cast[PPointer](dest)[])
  72. for i in 0..seq.len-1:
  73. genericAssignAux(
  74. cast[pointer](dst +% i *% mt.base.size +% GenericSeqSize),
  75. cast[pointer](cast[ByteAddress](s2) +% i *% mt.base.size +%
  76. GenericSeqSize),
  77. mt.base, shallow)
  78. of tyObject:
  79. var it = mt.base
  80. # don't use recursion here on the PNimType because the subtype
  81. # check should only be done at the very end:
  82. while it != nil:
  83. genericAssignAux(dest, src, it.node, shallow)
  84. it = it.base
  85. genericAssignAux(dest, src, mt.node, shallow)
  86. # we need to copy m_type field for tyObject, as it could be empty for
  87. # sequence reallocations:
  88. var pint = cast[ptr PNimType](dest)
  89. # We need to copy the *static* type not the dynamic type:
  90. # if p of TB:
  91. # var tbObj = TB(p)
  92. # tbObj of TC # needs to be false!
  93. #c_fprintf(stdout, "%s %s\n", pint[].name, mt.name)
  94. chckObjAsgn(cast[ptr PNimType](src)[], mt)
  95. pint[] = mt # cast[ptr PNimType](src)[]
  96. of tyTuple:
  97. genericAssignAux(dest, src, mt.node, shallow)
  98. of tyArray, tyArrayConstr:
  99. for i in 0..(mt.size div mt.base.size)-1:
  100. genericAssignAux(cast[pointer](d +% i *% mt.base.size),
  101. cast[pointer](s +% i *% mt.base.size), mt.base, shallow)
  102. of tyRef:
  103. unsureAsgnRef(cast[PPointer](dest), cast[PPointer](s)[])
  104. else:
  105. copyMem(dest, src, mt.size) # copy raw bits
  106. proc genericAssign(dest, src: pointer, mt: PNimType) {.compilerProc.} =
  107. genericAssignAux(dest, src, mt, false)
  108. proc genericShallowAssign(dest, src: pointer, mt: PNimType) {.compilerProc.} =
  109. genericAssignAux(dest, src, mt, true)
  110. when false:
  111. proc debugNimType(t: PNimType) =
  112. if t.isNil:
  113. cprintf("nil!")
  114. return
  115. var k: cstring
  116. case t.kind
  117. of tyBool: k = "bool"
  118. of tyChar: k = "char"
  119. of tyEnum: k = "enum"
  120. of tyArray: k = "array"
  121. of tyObject: k = "object"
  122. of tyTuple: k = "tuple"
  123. of tyRange: k = "range"
  124. of tyPtr: k = "ptr"
  125. of tyRef: k = "ref"
  126. of tyVar: k = "var"
  127. of tySequence: k = "seq"
  128. of tyProc: k = "proc"
  129. of tyPointer: k = "range"
  130. of tyOpenArray: k = "openarray"
  131. of tyString: k = "string"
  132. of tyCString: k = "cstring"
  133. of tyInt: k = "int"
  134. of tyInt32: k = "int32"
  135. else: k = "other"
  136. cprintf("%s %ld\n", k, t.size)
  137. debugNimType(t.base)
  138. proc genericSeqAssign(dest, src: pointer, mt: PNimType) {.compilerProc.} =
  139. var src = src # ugly, but I like to stress the parser sometimes :-)
  140. genericAssign(dest, addr(src), mt)
  141. proc genericAssignOpenArray(dest, src: pointer, len: int,
  142. mt: PNimType) {.compilerproc.} =
  143. var
  144. d = cast[ByteAddress](dest)
  145. s = cast[ByteAddress](src)
  146. for i in 0..len-1:
  147. genericAssign(cast[pointer](d +% i *% mt.base.size),
  148. cast[pointer](s +% i *% mt.base.size), mt.base)
  149. proc objectInit(dest: pointer, typ: PNimType) {.compilerProc, benign.}
  150. proc objectInitAux(dest: pointer, n: ptr TNimNode) {.benign.} =
  151. var d = cast[ByteAddress](dest)
  152. case n.kind
  153. of nkNone: sysAssert(false, "objectInitAux")
  154. of nkSlot: objectInit(cast[pointer](d +% n.offset), n.typ)
  155. of nkList:
  156. for i in 0..n.len-1:
  157. objectInitAux(dest, n.sons[i])
  158. of nkCase:
  159. var m = selectBranch(dest, n)
  160. if m != nil: objectInitAux(dest, m)
  161. proc objectInit(dest: pointer, typ: PNimType) =
  162. # the generic init proc that takes care of initialization of complex
  163. # objects on the stack or heap
  164. var d = cast[ByteAddress](dest)
  165. case typ.kind
  166. of tyObject:
  167. # iterate over any structural type
  168. # here we have to init the type field:
  169. var pint = cast[ptr PNimType](dest)
  170. pint[] = typ
  171. objectInitAux(dest, typ.node)
  172. of tyTuple:
  173. objectInitAux(dest, typ.node)
  174. of tyArray, tyArrayConstr:
  175. for i in 0..(typ.size div typ.base.size)-1:
  176. objectInit(cast[pointer](d +% i * typ.base.size), typ.base)
  177. else: discard # nothing to do
  178. # ---------------------- assign zero -----------------------------------------
  179. proc genericReset(dest: pointer, mt: PNimType) {.compilerProc, benign.}
  180. proc genericResetAux(dest: pointer, n: ptr TNimNode) =
  181. var d = cast[ByteAddress](dest)
  182. case n.kind
  183. of nkNone: sysAssert(false, "genericResetAux")
  184. of nkSlot: genericReset(cast[pointer](d +% n.offset), n.typ)
  185. of nkList:
  186. for i in 0..n.len-1: genericResetAux(dest, n.sons[i])
  187. of nkCase:
  188. var m = selectBranch(dest, n)
  189. if m != nil: genericResetAux(dest, m)
  190. zeroMem(cast[pointer](d +% n.offset), n.typ.size)
  191. proc genericReset(dest: pointer, mt: PNimType) =
  192. var d = cast[ByteAddress](dest)
  193. sysAssert(mt != nil, "genericReset 2")
  194. case mt.kind
  195. of tyString, tyRef, tySequence:
  196. unsureAsgnRef(cast[PPointer](dest), nil)
  197. of tyTuple:
  198. genericResetAux(dest, mt.node)
  199. of tyObject:
  200. genericResetAux(dest, mt.node)
  201. # also reset the type field for tyObject, for correct branch switching!
  202. var pint = cast[ptr PNimType](dest)
  203. pint[] = nil
  204. of tyArray, tyArrayConstr:
  205. for i in 0..(mt.size div mt.base.size)-1:
  206. genericReset(cast[pointer](d +% i *% mt.base.size), mt.base)
  207. else:
  208. zeroMem(dest, mt.size) # set raw bits to zero
  209. proc selectBranch(discVal, L: int,
  210. a: ptr array[0x7fff, ptr TNimNode]): ptr TNimNode =
  211. result = a[L] # a[L] contains the ``else`` part (but may be nil)
  212. if discVal <% L:
  213. let x = a[discVal]
  214. if x != nil: result = x
  215. proc FieldDiscriminantCheck(oldDiscVal, newDiscVal: int,
  216. a: ptr array[0x7fff, ptr TNimNode],
  217. L: int) {.compilerProc.} =
  218. let oldBranch = selectBranch(oldDiscVal, L, a)
  219. let newBranch = selectBranch(newDiscVal, L, a)
  220. when defined(nimOldCaseObjects):
  221. if newBranch != oldBranch and oldDiscVal != 0:
  222. sysFatal(FieldError, "assignment to discriminant changes object branch")
  223. else:
  224. if newBranch != oldBranch:
  225. if oldDiscVal != 0:
  226. sysFatal(FieldError, "assignment to discriminant changes object branch")
  227. else:
  228. sysFatal(FieldError, "assignment to discriminant changes object branch; compile with -d:nimOldCaseObjects for a transition period")