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
|
type
SharedPtr*[T] = object
val: ptr tuple[atomicCounter: int, value: T]
proc `=destroy`*[T](p: var SharedPtr[T]) =
mixin `=destroy`
if p.val != nil:
let c = atomicDec(p.val[].atomicCounter)
if c == 0:
`=destroy`(p.val.value)
freeShared(p.val)
p.val = nil
proc `=`*[T](dest: var SharedPtr[T], src: SharedPtr[T]) {.inline.} =
if dest.val != src.val:
if dest.val != nil:
`=destroy`(dest)
if src.val != nil:
discard atomicInc(src.val[].atomicCounter)
dest.val = src.val
proc newSharedPtr*[T](val: sink T): SharedPtr[T] =
result.val = cast[type(result.val)](allocShared0(sizeof(result.val[])))
result.val.atomicCounter = 1
result.val.value = val
func get*[T](p: SharedPtr[T]): var T {.inline.} =
p.val.value
func isNil*[T](p: SharedPtr[T]): bool {.inline.} =
p.val == nil
proc cas*[T](p, old_val: var SharedPtr[T], new_val: SharedPtr[T]): bool {.inline.} =
if old_val.val == new_val.val:
result = true
else:
result = cas(p.val.addr, old_val.val, new_val.val)
if result:
`=destroy`(old_val)
if new_val.val != nil:
discard atomicInc(new_val.val[].atomicCounter)
|