destructors: preparations for upcoming changes

This commit is contained in:
Araq
2017-10-17 12:46:49 +02:00
parent 7ee825a6e5
commit 45d74f4081
8 changed files with 27 additions and 8 deletions

View File

@@ -1663,3 +1663,5 @@ when false:
if n.isNil: return true
for i in 0 ..< n.safeLen:
if n[i].containsNil: return true
template hasDestructor*(t: PType): bool = tfHasAsgn in t.flags

View File

@@ -95,8 +95,6 @@ import
intsets, ast, astalgo, msgs, renderer, magicsys, types, idents, trees,
strutils, options, dfa, lowerings
template hasDestructor(t: PType): bool = tfHasAsgn in t.flags
const
InterestingSyms = {skVar, skResult, skLet}
@@ -253,6 +251,7 @@ proc p(n: PNode; c: var Con): PNode =
result = copyNode(n)
recurse(n, result)
of nkAsgn, nkFastAsgn:
# XXX if special, call special operator
if n[0].kind == nkSym and interestingSym(n[0].sym):
result = moveOrCopy(n[0], n[1], c)
else:

View File

@@ -109,7 +109,7 @@ proc considerOverloadedOp(c: var TLiftCtx; t: PType; body, x, y: PNode): bool =
styleCheckUse(c.info, op)
body.add newOpCall(op, x)
result = true
of attachedAsgn:
of attachedAsgn, attachedSink:
if tfHasAsgn in t.flags:
var op: PSym
if sameType(t, c.asgnForType):
@@ -285,3 +285,13 @@ proc getAsgnOrLiftBody(c: PContext; typ: PType; info: TLineInfo): PSym =
proc overloadedAsgn(c: PContext; dest, src: PNode): PNode =
let a = getAsgnOrLiftBody(c, dest.typ, dest.info)
result = newAsgnCall(c, a, dest, src)
proc liftTypeBoundOps*(c: PContext; typ: PType) =
## In the semantic pass this is called in strategic places
## to ensure we lift assignment, destructors and moves properly.
## Since this is done in the sem* routines generics already have
## been resolved for us and do not complicate the logic any further.
## We have to ensure that the 'tfHasDesctructor' flags bubbles up
## in the generic instantiations though.
## The later 'destroyer' pass depends on it.
if not newDestructors or not hasDestructor(typ): return

View File

@@ -70,6 +70,7 @@ type
TTypeAttachedOp* = enum
attachedAsgn,
attachedSink,
attachedDeepCopy,
attachedDestructor

View File

@@ -670,6 +670,7 @@ proc afterCallActions(c: PContext; n, orig: PNode, flags: TExprFlags): PNode =
analyseIfAddressTakenInCall(c, result)
if callee.magic != mNone:
result = magicsAfterOverloadResolution(c, result, flags)
if result.typ != nil: liftTypeBoundOps(c, result.typ)
if c.matchedConcept == nil:
result = evalAtCompileTime(c, result)
@@ -1390,6 +1391,8 @@ proc semAsgn(c: PContext, n: PNode; mode=asgnNormal): PNode =
if tfHasAsgn in lhs.typ.flags and not lhsIsResult and
mode != noOverloadedAsgn:
return overloadedAsgn(c, lhs, n.sons[1])
else:
liftTypeBoundOps(c, lhs.typ)
fixAbstractType(c, n)
asgnToResultVar(c, n, n.sons[0], n.sons[1])

View File

@@ -553,6 +553,7 @@ proc semVarOrLet(c: PContext, n: PNode, symkind: TSymKind): PNode =
# this can only happen for errornous var statements:
if typ == nil: continue
typeAllowedCheck(a.info, typ, symkind)
liftTypeBoundOps(c, typ)
var tup = skipTypes(typ, {tyGenericInst, tyAlias})
if a.kind == nkVarTuple:
if tup.kind != tyTuple:

View File

@@ -363,11 +363,7 @@ proc handleGenericInvocation(cl: var TReplTypeVars, t: PType): PType =
# '=' needs to be instantiated for generics when the type is constructed:
newbody.field = cl.c.instTypeBoundOp(cl.c, opr, result, cl.info,
attachedAsgn, 1)
# we need to produce the destructor first here because generated '='
# and '=sink' operators can rely on it:
if newDestructors: typeBound(destructor)
typeBound(assignment)
typeBound(sink)
if not newDestructors: typeBound(assignment)
let methods = skipTypes(bbody, abstractPtrs).methods
for col, meth in items(methods):
# we instantiate the known methods belonging to that type, this causes

View File

@@ -292,6 +292,13 @@ when defined(nimArrIdx):
proc `[]=`*[I: Ordinal;T,S](a: T; i: I;
x: S) {.noSideEffect, magic: "ArrPut".}
proc `=`*[T](dest: var T; src: T) {.noSideEffect, magic: "Asgn".}
when defined(nimNewRuntime):
template `=destroy`*[T](x: var T) =
## generic `destructor`:idx: implementation that can be overriden.
discard
template `=sink`*[T](x: var T; y: T) =
## generic `sink`:idx: implementation that can be overriden.
shallowCopy(x, y)
type
Slice*[T] = object ## builtin slice type