Sources/Concurrency/AtomicReference.swift (53 lines of code) (raw):

// // Copyright (c) 2018. Uber Technologies // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. // import Foundation import libkern import ObjCBridges /// A concurrency utility class that supports locking-free synchronization on mutating an object /// reference. Unlike using a lock, concurrent read and write accesses to this class is allowed. At /// the same time, concurrent operations using the atomic functions provided by this class ensures /// synchronization correctness without the higher cost of locking. public class AtomicReference<ValueType> { /// The value that guarantees atomic read and write-through memory behavior. public var value: ValueType { get { // Create a memory barrier to ensure the entire memory stack is in sync so we // can safely retrieve the value. This guarantees the initial value is in sync. atomic_thread_fence(memory_order_seq_cst) return wrappedValue } set { while true { let oldValue = self.value if self.compareAndSet(expect: oldValue, newValue: newValue) { break } } } } /// Initializer. /// /// - parameter initialValue: The initial value. public init(initialValue: ValueType) { wrappedValue = initialValue pointer.pointee = unsafePassUnretainedPointer(value: wrappedValue) } /// Atomically sets the new value, if the current value's memory pointer equals the /// expected value's memory pointer. /// /// - parameter expect: The expected value to compare against. /// - parameter newValue: The new value to set to if the comparison succeeds. /// - returns: true if the comparison succeeded and the value is set. false otherwise. public func compareAndSet(expect: ValueType, newValue: ValueType) -> Bool { let expectPointer = unsafePassUnretainedPointer(value: expect) let newValuePointer = unsafePassUnretainedPointer(value: newValue) if AtomicBridges.comparePointer(pointer, withExpectedPointer: expectPointer, andSwapPointer: newValuePointer) { // If pointer swap succeeded, a memory berrier is created, so we can safely write the new // value. wrappedValue = newValue return true } else { return false } } /// Atomically sets to the given new value and returns the old value. /// /// - parameter newValue: The new value to set to. /// - returns: The old value. public func getAndSet(newValue: ValueType) -> ValueType { while true { let oldValue = self.value if compareAndSet(expect: oldValue, newValue: newValue) { return oldValue } } } // MARK: - Private private let pointer: UnsafeMutablePointer<UnsafeMutableRawPointer?> = UnsafeMutablePointer.allocate(capacity: 1) private var wrappedValue: ValueType private func unsafePassUnretainedPointer(value: ValueType) -> UnsafeMutableRawPointer { return UnsafeMutableRawPointer(Unmanaged.passUnretained(value as AnyObject).toOpaque()) } deinit { #if swift(>=4.1) pointer.deallocate() #else pointer.deallocate(capacity: 1) #endif } }