summary refs log tree commit diff stats
path: root/tests/iter/t22619.nim
blob: 6a98391f30bbe2de75a69a1da8d3cc143481bd0b (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
# bug #22619

when false: # todo fixme
  block:
    type
      Resource = object
        value: int
  
      Object = object
        r {.cursor.}: Resource
        s {.cursor.}: seq[Resource]
  
    var numDestroy = 0
  
    proc `=copy`(x: var Resource, y: Resource) {.error.} # disallow full copies
    proc `=destroy`(x: Resource) =
      inc numDestroy
  
    proc test() =
      # perform the test in procedure so that globals aren't used (their different
      # semantics with regards to destruction would interfere)
      var
        r = Resource(value: 1) # initialize a resource
        s = @[Resource(value: 2)]
  
      # make sure no copy is required in the initializer expression:
      var o = Object(r: r, s: s)
  
      # copying the object doesn't perform a full copy of the cursor fields:
      var o2 = o
      discard addr(o2) # prevent `o2` from being turned into a cursor
  
      # check that the fields were shallow-copied:
      doAssert o2.r.value == 1
      doAssert o2.s[0].value == 2
  
      # make sure no copy is required with normal field assignments:
      o.r = r
      o.s = s
  
  
      # when `o` and `o2` are destroyed, their destructor must not be called on
      # their fields
  
    test()
  
    # one call for the `r` local and one for the object in `s`
    doAssert numDestroy == 2

block:
  type Value = distinct int

  var numDestroy = 0

  when defined(gcRefc):
    proc `=destroy`(x: var Value) =
      inc numDestroy
  else:
    proc `=destroy`(x: Value) =
      inc numDestroy

  iterator iter(s: seq[Value]): int {.closure.} =
    # because it is used across yields, `s2` is lifted into the iterator's
    # environment. Since non-ref cursors in object didn't have their hooks
    # disabled inside the environments lifted hooks, this led to double
    # frees
    var s2 {.cursor.} = s
    var i = 0
    let L = s2.len
    while i < L:
      yield s2[i].int
      inc i

  proc test() =
    var s = @[Value(1), Value(2)]
    let cl = iter
    # make sure resuming the iterator works:
    doAssert cl(s) == 1
    doAssert cl(s) == 2
    doAssert cl(s) == 0

  test()
  doAssert numDestroy == 2