diff options
author | Araq <rumpf_a@web.de> | 2012-10-12 16:49:04 -0700 |
---|---|---|
committer | Araq <rumpf_a@web.de> | 2012-10-12 16:49:04 -0700 |
commit | a9da8eba06f5fb5992b2a537a8872d5e64ee0cdb (patch) | |
tree | 6ee94b7981ad614ed3f255dfb25ed6e42fa0d6b1 | |
parent | 2193460ea6ecd9b2ee3858fffcadd2635f828f9a (diff) | |
parent | bfd3ac4ee7471dbe457d860fa4773e817a3c1543 (diff) | |
download | Nim-a9da8eba06f5fb5992b2a537a8872d5e64ee0cdb.tar.gz |
Merge pull request #221 from exhu/cas
Added CAS to system/atomics.nim
-rwxr-xr-x | lib/system/atomics.nim | 34 |
1 files changed, 34 insertions, 0 deletions
diff --git a/lib/system/atomics.nim b/lib/system/atomics.nim index 127a8637f..623f8d0d2 100755 --- a/lib/system/atomics.nim +++ b/lib/system/atomics.nim @@ -40,3 +40,37 @@ proc atomicDec(memLoc: var int, x: int = 1): int = dec(memLoc, x) result = memLoc + +# atomic compare and swap (CAS) funcitons to implement lock-free algorithms + +when (defined(gcc) or defined(llvm_gcc)) and hasThreadSupport: + proc compareAndSwap*[T: ptr|ref|pointer](mem: var T, expected: T, newValue: T): bool {.nodecl, + importc: " __sync_bool_compare_and_swap".} + ## Returns true if successfully set value at mem to newValue when value + ## at mem == expected + +elif defined(windows) and hasThreadSupport: + proc InterlockedCompareExchangePointer(mem: ptr pointer, + newValue: pointer, comparand: pointer) : pointer {.nodecl, + importc: "InterlockedCompareExchangePointer", header:"windows.h".} + + + proc compareAndSwap*[T: ptr|ref|pointer](mem: var T, + expected: T, newValue: T): bool {.inline.}= + ## Returns true if successfully set value at mem to newValue when value + ## at mem == expected + return InterlockedCompareExchangePointer(addr(mem), + newValue, expected) == expected + +elif not hasThreadSupport: + proc compareAndSwap*[T: ptr|ref|pointer](mem: var T, + expected: T, newValue: T): bool {.inline.} = + ## Returns true if successfully set value at mem to newValue when value + ## at mem == expected + var oldval = mem + if oldval == expected: + mem = newValue + return true + return false + + |