cgmeth.nim 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299
  1. #
  2. #
  3. # The Nim Compiler
  4. # (c) Copyright 2013 Andreas Rumpf
  5. #
  6. # See the file "copying.txt", included in this
  7. # distribution, for details about the copyright.
  8. #
  9. ## This module implements code generation for methods.
  10. import
  11. intsets, options, ast, msgs, idents, renderer, types, magicsys,
  12. sempass2, strutils, modulegraphs, lineinfos
  13. proc genConv(n: PNode, d: PType, downcast: bool; conf: ConfigRef): PNode =
  14. var dest = skipTypes(d, abstractPtrs)
  15. var source = skipTypes(n.typ, abstractPtrs)
  16. if (source.kind == tyObject) and (dest.kind == tyObject):
  17. var diff = inheritanceDiff(dest, source)
  18. if diff == high(int):
  19. # no subtype relation, nothing to do
  20. result = n
  21. elif diff < 0:
  22. result = newNodeIT(nkObjUpConv, n.info, d)
  23. result.add n
  24. if downcast: internalError(conf, n.info, "cgmeth.genConv: no upcast allowed")
  25. elif diff > 0:
  26. result = newNodeIT(nkObjDownConv, n.info, d)
  27. result.add n
  28. if not downcast:
  29. internalError(conf, n.info, "cgmeth.genConv: no downcast allowed")
  30. else:
  31. result = n
  32. else:
  33. result = n
  34. proc getDispatcher*(s: PSym): PSym =
  35. ## can return nil if is has no dispatcher.
  36. if dispatcherPos < s.ast.len:
  37. result = s.ast[dispatcherPos].sym
  38. doAssert sfDispatcher in result.flags
  39. proc methodCall*(n: PNode; conf: ConfigRef): PNode =
  40. result = n
  41. # replace ordinary method by dispatcher method:
  42. let disp = getDispatcher(result[0].sym)
  43. if disp != nil:
  44. result[0].typ = disp.typ
  45. result[0].sym = disp
  46. # change the arguments to up/downcasts to fit the dispatcher's parameters:
  47. for i in 1..<result.len:
  48. result[i] = genConv(result[i], disp.typ[i], true, conf)
  49. else:
  50. localError(conf, n.info, "'" & $result[0] & "' lacks a dispatcher")
  51. type
  52. MethodResult = enum No, Invalid, Yes
  53. proc sameMethodBucket(a, b: PSym; multiMethods: bool): MethodResult =
  54. if a.name.id != b.name.id: return
  55. if a.typ.len != b.typ.len:
  56. return
  57. for i in 1..<a.typ.len:
  58. var aa = a.typ[i]
  59. var bb = b.typ[i]
  60. while true:
  61. aa = skipTypes(aa, {tyGenericInst, tyAlias})
  62. bb = skipTypes(bb, {tyGenericInst, tyAlias})
  63. if aa.kind == bb.kind and aa.kind in {tyVar, tyPtr, tyRef, tyLent, tySink}:
  64. aa = aa.lastSon
  65. bb = bb.lastSon
  66. else:
  67. break
  68. if sameType(a.typ[i], b.typ[i]):
  69. if aa.kind == tyObject and result != Invalid:
  70. result = Yes
  71. elif aa.kind == tyObject and bb.kind == tyObject and (i == 1 or multiMethods):
  72. let diff = inheritanceDiff(bb, aa)
  73. if diff < 0:
  74. if result != Invalid:
  75. result = Yes
  76. else:
  77. return No
  78. elif diff != high(int) and sfFromGeneric notin (a.flags+b.flags):
  79. result = Invalid
  80. else:
  81. return No
  82. else:
  83. return No
  84. if result == Yes:
  85. # check for return type:
  86. if not sameTypeOrNil(a.typ[0], b.typ[0]):
  87. if b.typ[0] != nil and b.typ[0].kind == tyUntyped:
  88. # infer 'auto' from the base to make it consistent:
  89. b.typ[0] = a.typ[0]
  90. else:
  91. return No
  92. proc attachDispatcher(s: PSym, dispatcher: PNode) =
  93. if dispatcherPos < s.ast.len:
  94. # we've added a dispatcher already, so overwrite it
  95. s.ast[dispatcherPos] = dispatcher
  96. else:
  97. setLen(s.ast.sons, dispatcherPos+1)
  98. if s.ast[resultPos] == nil:
  99. s.ast[resultPos] = newNodeI(nkEmpty, s.info)
  100. s.ast[dispatcherPos] = dispatcher
  101. proc createDispatcher(s: PSym; g: ModuleGraph; idgen: IdGenerator): PSym =
  102. var disp = copySym(s, nextSymId(idgen))
  103. incl(disp.flags, sfDispatcher)
  104. excl(disp.flags, sfExported)
  105. let old = disp.typ
  106. disp.typ = copyType(disp.typ, nextTypeId(idgen), disp.typ.owner)
  107. copyTypeProps(g, idgen.module, disp.typ, old)
  108. # we can't inline the dispatcher itself (for now):
  109. if disp.typ.callConv == ccInline: disp.typ.callConv = ccNimCall
  110. disp.ast = copyTree(s.ast)
  111. disp.ast[bodyPos] = newNodeI(nkEmpty, s.info)
  112. disp.loc.r = nil
  113. if s.typ[0] != nil:
  114. if disp.ast.len > resultPos:
  115. disp.ast[resultPos].sym = copySym(s.ast[resultPos].sym, nextSymId(idgen))
  116. else:
  117. # We've encountered a method prototype without a filled-in
  118. # resultPos slot. We put a placeholder in there that will
  119. # be updated in fixupDispatcher().
  120. disp.ast.add newNodeI(nkEmpty, s.info)
  121. attachDispatcher(s, newSymNode(disp))
  122. # attach to itself to prevent bugs:
  123. attachDispatcher(disp, newSymNode(disp))
  124. return disp
  125. proc fixupDispatcher(meth, disp: PSym; conf: ConfigRef) =
  126. # We may have constructed the dispatcher from a method prototype
  127. # and need to augment the incomplete dispatcher with information
  128. # from later definitions, particularly the resultPos slot. Also,
  129. # the lock level of the dispatcher needs to be updated/checked
  130. # against that of the method.
  131. if disp.ast.len > resultPos and meth.ast.len > resultPos and
  132. disp.ast[resultPos].kind == nkEmpty:
  133. disp.ast[resultPos] = copyTree(meth.ast[resultPos])
  134. # The following code works only with lock levels, so we disable
  135. # it when they're not available.
  136. when declared(TLockLevel):
  137. proc `<`(a, b: TLockLevel): bool {.borrow.}
  138. proc `==`(a, b: TLockLevel): bool {.borrow.}
  139. if disp.typ.lockLevel == UnspecifiedLockLevel:
  140. disp.typ.lockLevel = meth.typ.lockLevel
  141. elif meth.typ.lockLevel != UnspecifiedLockLevel and
  142. meth.typ.lockLevel != disp.typ.lockLevel:
  143. message(conf, meth.info, warnLockLevel,
  144. "method has lock level $1, but another method has $2" %
  145. [$meth.typ.lockLevel, $disp.typ.lockLevel])
  146. # XXX The following code silences a duplicate warning in
  147. # checkMethodeffects() in sempass2.nim for now.
  148. if disp.typ.lockLevel < meth.typ.lockLevel:
  149. disp.typ.lockLevel = meth.typ.lockLevel
  150. proc methodDef*(g: ModuleGraph; idgen: IdGenerator; s: PSym) =
  151. var witness: PSym
  152. for i in 0..<g.methods.len:
  153. let disp = g.methods[i].dispatcher
  154. case sameMethodBucket(disp, s, multimethods = optMultiMethods in g.config.globalOptions)
  155. of Yes:
  156. g.methods[i].methods.add(s)
  157. attachDispatcher(s, disp.ast[dispatcherPos])
  158. fixupDispatcher(s, disp, g.config)
  159. #echo "fixup ", disp.name.s, " ", disp.id
  160. when useEffectSystem: checkMethodEffects(g, disp, s)
  161. if {sfBase, sfFromGeneric} * s.flags == {sfBase} and
  162. g.methods[i].methods[0] != s:
  163. # already exists due to forwarding definition?
  164. localError(g.config, s.info, "method is not a base")
  165. return
  166. of No: discard
  167. of Invalid:
  168. if witness.isNil: witness = g.methods[i].methods[0]
  169. # create a new dispatcher:
  170. g.methods.add((methods: @[s], dispatcher: createDispatcher(s, g, idgen)))
  171. #echo "adding ", s.info
  172. if witness != nil:
  173. localError(g.config, s.info, "invalid declaration order; cannot attach '" & s.name.s &
  174. "' to method defined here: " & g.config$witness.info)
  175. elif sfBase notin s.flags:
  176. message(g.config, s.info, warnUseBase)
  177. proc relevantCol(methods: seq[PSym], col: int): bool =
  178. # returns true iff the position is relevant
  179. var t = methods[0].typ[col].skipTypes(skipPtrs)
  180. if t.kind == tyObject:
  181. for i in 1..high(methods):
  182. let t2 = skipTypes(methods[i].typ[col], skipPtrs)
  183. if not sameType(t2, t):
  184. return true
  185. proc cmpSignatures(a, b: PSym, relevantCols: IntSet): int =
  186. for col in 1..<a.typ.len:
  187. if contains(relevantCols, col):
  188. var aa = skipTypes(a.typ[col], skipPtrs)
  189. var bb = skipTypes(b.typ[col], skipPtrs)
  190. var d = inheritanceDiff(aa, bb)
  191. if (d != high(int)) and d != 0:
  192. return d
  193. proc sortBucket(a: var seq[PSym], relevantCols: IntSet) =
  194. # we use shellsort here; fast and simple
  195. var n = a.len
  196. var h = 1
  197. while true:
  198. h = 3 * h + 1
  199. if h > n: break
  200. while true:
  201. h = h div 3
  202. for i in h..<n:
  203. var v = a[i]
  204. var j = i
  205. while cmpSignatures(a[j - h], v, relevantCols) >= 0:
  206. a[j] = a[j - h]
  207. j = j - h
  208. if j < h: break
  209. a[j] = v
  210. if h == 1: break
  211. proc genDispatcher(g: ModuleGraph; methods: seq[PSym], relevantCols: IntSet): PSym =
  212. var base = methods[0].ast[dispatcherPos].sym
  213. result = base
  214. var paramLen = base.typ.len
  215. var nilchecks = newNodeI(nkStmtList, base.info)
  216. var disp = newNodeI(nkIfStmt, base.info)
  217. var ands = getSysMagic(g, unknownLineInfo, "and", mAnd)
  218. var iss = getSysMagic(g, unknownLineInfo, "of", mOf)
  219. let boolType = getSysType(g, unknownLineInfo, tyBool)
  220. for col in 1..<paramLen:
  221. if contains(relevantCols, col):
  222. let param = base.typ.n[col].sym
  223. if param.typ.skipTypes(abstractInst).kind in {tyRef, tyPtr}:
  224. nilchecks.add newTree(nkCall,
  225. newSymNode(getCompilerProc(g, "chckNilDisp")), newSymNode(param))
  226. for meth in 0..high(methods):
  227. var curr = methods[meth] # generate condition:
  228. var cond: PNode = nil
  229. for col in 1..<paramLen:
  230. if contains(relevantCols, col):
  231. var isn = newNodeIT(nkCall, base.info, boolType)
  232. isn.add newSymNode(iss)
  233. let param = base.typ.n[col].sym
  234. isn.add newSymNode(param)
  235. isn.add newNodeIT(nkType, base.info, curr.typ[col])
  236. if cond != nil:
  237. var a = newNodeIT(nkCall, base.info, boolType)
  238. a.add newSymNode(ands)
  239. a.add cond
  240. a.add isn
  241. cond = a
  242. else:
  243. cond = isn
  244. let retTyp = base.typ[0]
  245. let call = newNodeIT(nkCall, base.info, retTyp)
  246. call.add newSymNode(curr)
  247. for col in 1..<paramLen:
  248. call.add genConv(newSymNode(base.typ.n[col].sym),
  249. curr.typ[col], false, g.config)
  250. var ret: PNode
  251. if retTyp != nil:
  252. var a = newNodeI(nkFastAsgn, base.info)
  253. a.add newSymNode(base.ast[resultPos].sym)
  254. a.add call
  255. ret = newNodeI(nkReturnStmt, base.info)
  256. ret.add a
  257. else:
  258. ret = call
  259. if cond != nil:
  260. var a = newNodeI(nkElifBranch, base.info)
  261. a.add cond
  262. a.add ret
  263. disp.add a
  264. else:
  265. disp = ret
  266. nilchecks.add disp
  267. nilchecks.flags.incl nfTransf # should not be further transformed
  268. result.ast[bodyPos] = nilchecks
  269. proc generateMethodDispatchers*(g: ModuleGraph): PNode =
  270. result = newNode(nkStmtList)
  271. for bucket in 0..<g.methods.len:
  272. var relevantCols = initIntSet()
  273. for col in 1..<g.methods[bucket].methods[0].typ.len:
  274. if relevantCol(g.methods[bucket].methods, col): incl(relevantCols, col)
  275. if optMultiMethods notin g.config.globalOptions:
  276. # if multi-methods are not enabled, we are interested only in the first field
  277. break
  278. sortBucket(g.methods[bucket].methods, relevantCols)
  279. result.add newSymNode(genDispatcher(g, g.methods[bucket].methods, relevantCols))