mirror of
https://github.com/apple/sourcekit-lsp.git
synced 2026-03-02 18:23:24 +01:00
Replaces DispatchQueue with ThreadSafeBox, relaxes T: Sendable constraints, and removes @unchecked Sendable from Document.
66 lines
1.8 KiB
Swift
66 lines
1.8 KiB
Swift
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This source file is part of the Swift.org open source project
|
|
//
|
|
// Copyright (c) 2014 - 2023 Apple Inc. and the Swift project authors
|
|
// Licensed under Apache License v2.0 with Runtime Library Exception
|
|
//
|
|
// See https://swift.org/LICENSE.txt for license information
|
|
// See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
import Foundation
|
|
|
|
/// A thread safe container that contains a value of type `T`.
|
|
///
|
|
/// - Note: Unchecked sendable conformance because value is guarded by a lock.
|
|
package final class ThreadSafeBox<T>: Sendable {
|
|
/// Lock guarding `_value`.
|
|
private let lock = NSLock()
|
|
|
|
private nonisolated(unsafe) var _value: T
|
|
|
|
package init(initialValue: T) {
|
|
_value = initialValue
|
|
}
|
|
|
|
/// Restrict the result of the body to `Sendable` so callers can safely use
|
|
/// the returned value outside the lock without requiring `T: Sendable`.
|
|
package func withLock<Result: Sendable>(_ body: (inout T) throws -> Result) rethrows -> Result {
|
|
return try lock.withLock {
|
|
return try body(&_value)
|
|
}
|
|
}
|
|
}
|
|
|
|
extension ThreadSafeBox where T: Sendable {
|
|
package var value: T {
|
|
get {
|
|
return lock.withLock {
|
|
return _value
|
|
}
|
|
}
|
|
set {
|
|
lock.withLock {
|
|
_value = newValue
|
|
}
|
|
}
|
|
_modify {
|
|
lock.lock()
|
|
defer { lock.unlock() }
|
|
yield &_value
|
|
}
|
|
}
|
|
|
|
/// If the value in the box is an optional, return it and reset it to `nil`
|
|
/// in an atomic operation.
|
|
package func takeValue<U>() -> T where U? == T {
|
|
lock.withLock {
|
|
guard let value = self._value else { return nil }
|
|
self._value = nil
|
|
return value
|
|
}
|
|
}
|
|
}
|