Files
swift-mirror/stdlib/public/Concurrency/ExecutorAssertions.swift
2025-02-26 14:28:24 -08:00

384 lines
16 KiB
Swift

//===----------------------------------------------------------------------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 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 Swift
import SwiftShims
#if !SWIFT_STDLIB_TASK_TO_THREAD_MODEL_CONCURRENCY
// ==== -----------------------------------------------------------------------
// MARK: Precondition executors
@available(SwiftStdlib 5.1, *)
extension SerialExecutor {
/// Stops program execution if the current task is not executing on this
/// serial executor.
///
/// This function's effect varies depending on the build flag used:
///
/// * In playgrounds and `-Onone` builds (the default for Xcode's Debug
/// configuration), stops program execution in a debuggable state after
/// printing `message`.
///
/// * In `-O` builds (the default for Xcode's Release configuration), stops
/// program execution.
///
/// - Note: Because this check is performed against the actor's serial executor,
/// if another actor uses the same serial executor--by using
/// that actor's serial executor as its own ``Actor/unownedExecutor``--this
/// check will succeed. From a concurrency safety perspective, the
/// serial executor guarantees mutual exclusion of those two actors.
///
/// - Parameters:
/// - message: The message to print if the assertion fails.
/// - file: The file name to print if the assertion fails. The default value is
/// the file where this method was called.
/// - line: The line number to print if the assertion fails The default value is
/// the line where this method was called.
@available(SwiftStdlib 5.1, *)
#if !$Embedded
@backDeployed(before: SwiftStdlib 5.9)
#endif
@_unavailableInEmbedded
public func preconditionIsolated(
_ message: @autoclosure () -> String = String(),
file: StaticString = #fileID, line: UInt = #line
) {
guard _isDebugAssertConfiguration() || _isReleaseAssertConfiguration() else {
return
}
let expectationCheck = unsafe _taskIsCurrentExecutor(self.asUnownedSerialExecutor().executor)
/// TODO: implement the logic in-place perhaps rather than delegating to precondition()?
precondition(expectationCheck,
// TODO: offer information which executor we actually got
"Incorrect actor executor assumption; Expected '\(self)' executor. \(message())",
file: file, line: line) // short-cut so we get the exact same failure reporting semantics
}
}
@available(SwiftStdlib 5.1, *)
extension Actor {
/// Stops program execution if the current task is not executing on this
/// actor's serial executor.
///
/// This function's effect varies depending on the build flag used:
///
/// * In playgrounds and `-Onone` builds (the default for Xcode's Debug
/// configuration), stops program execution in a debuggable state after
/// printing `message`.
///
/// * In `-O` builds (the default for Xcode's Release configuration), stops
/// program execution.
///
/// - Note: This check is performed against the actor's serial executor,
/// meaning that / if another actor uses the same serial executor--by using
/// that actor's serial executor as its own ``Actor/unownedExecutor``--this
/// check will succeed , as from a concurrency safety perspective, the
/// serial executor guarantees mutual exclusion of those two actors.
///
/// - Parameters:
/// - message: The message to print if the assertion fails.
/// - file: The file name to print if the assertion fails. The default is
/// where this method was called.
/// - line: The line number to print if the assertion fails The default is
/// where this method was called.
@available(SwiftStdlib 5.1, *)
#if !$Embedded
@backDeployed(before: SwiftStdlib 5.9)
#endif
@_unavailableInEmbedded
public nonisolated func preconditionIsolated(
_ message: @autoclosure () -> String = String(),
file: StaticString = #fileID, line: UInt = #line
) {
guard _isDebugAssertConfiguration() || _isReleaseAssertConfiguration() else {
return
}
// NOTE: This method will CRASH in new runtime versions,
// if it would have previously returned `false`.
// It will call through to SerialExecutor.checkIsolated` as a last resort.
let expectationCheck = unsafe _taskIsCurrentExecutor(self.unownedExecutor.executor)
precondition(expectationCheck,
unsafe "Incorrect actor executor assumption; Expected '\(self.unownedExecutor)' executor. \(message())",
file: file, line: line)
}
}
@available(SwiftStdlib 5.1, *)
extension GlobalActor {
/// Stops program execution if the current task is not executing on this
/// actor's serial executor.
///
/// This function's effect varies depending on the build flag used:
///
/// * In playgrounds and `-Onone` builds (the default for Xcode's Debug
/// configuration), stops program execution in a debuggable state after
/// printing `message`.
///
/// * In `-O` builds (the default for Xcode's Release configuration), stops
/// program execution.
///
/// - Note: This check is performed against the actor's serial executor,
/// meaning that / if another actor uses the same serial executor--by using
/// that actor's serial executor as its own ``Actor/unownedExecutor``--this
/// check will succeed , as from a concurrency safety perspective, the
/// serial executor guarantees mutual exclusion of those two actors.
///
/// - Parameters:
/// - message: The message to print if the assertion fails.
/// - file: The file name to print if the assertion fails. The default is
/// where this method was called.
/// - line: The line number to print if the assertion fails The default is
/// where this method was called.
@available(SwiftStdlib 5.1, *)
#if !$Embedded
@backDeployed(before: SwiftStdlib 5.9)
#endif
@_unavailableInEmbedded
public static func preconditionIsolated(
_ message: @autoclosure () -> String = String(),
file: StaticString = #fileID, line: UInt = #line
) {
Self.shared.preconditionIsolated(message(), file: file, line: line)
}
}
// ==== -----------------------------------------------------------------------
// MARK: Assert executors
@available(SwiftStdlib 5.1, *)
extension SerialExecutor {
/// Stops program execution if the current task is not executing on this
/// serial executor.
///
/// This function's effect varies depending on the build flag used:
///
/// * In playgrounds and `-Onone` builds (the default for Xcode's Debug
/// configuration), stops program execution in a debuggable state after
/// printing `message`.
///
/// * In `-O` builds (the default for Xcode's Release configuration),
/// the isolation check is not performed and there are no effects.
///
/// - Note: This check is performed against the actor's serial executor,
/// meaning that / if another actor uses the same serial executor--by using
/// that actor's serial executor as its own ``Actor/unownedExecutor``--this
/// check will succeed , as from a concurrency safety perspective, the
/// serial executor guarantees mutual exclusion of those two actors.
///
/// - Parameters:
/// - message: The message to print if the assertion fails.
/// - file: The file name to print if the assertion fails. The default is
/// where this method was called.
/// - line: The line number to print if the assertion fails The default is
/// where this method was called.
@available(SwiftStdlib 5.1, *)
#if !$Embedded
@backDeployed(before: SwiftStdlib 5.9)
#endif
@_unavailableInEmbedded
public func assertIsolated(
_ message: @autoclosure () -> String = String(),
file: StaticString = #fileID, line: UInt = #line
) {
guard _isDebugAssertConfiguration() else {
return
}
guard unsafe _taskIsCurrentExecutor(self.asUnownedSerialExecutor().executor) else {
// TODO: offer information which executor we actually got
let msg = "Incorrect actor executor assumption; Expected '\(self)' executor. \(message())"
/// TODO: implement the logic in-place perhaps rather than delegating to precondition()?
assertionFailure(msg, file: file, line: line)
return
}
}
}
@available(SwiftStdlib 5.1, *)
extension Actor {
/// Stops program execution if the current task is not executing on this
/// actor's serial executor.
///
/// This function's effect varies depending on the build flag used:
///
/// * In playgrounds and `-Onone` builds (the default for Xcode's Debug
/// configuration), stops program execution in a debuggable state after
/// printing `message`.
///
/// * In `-O` builds (the default for Xcode's Release configuration),
/// the isolation check is not performed and there are no effects.
///
/// - Note: This check is performed against the actor's serial executor,
/// meaning that / if another actor uses the same serial executor--by using
/// that actor's serial executor as its own ``Actor/unownedExecutor``--this
/// check will succeed , as from a concurrency safety perspective, the
/// serial executor guarantees mutual exclusion of those two actors.
///
/// - Parameters:
/// - message: The message to print if the assertion fails.
/// - file: The file name to print if the assertion fails. The default is
/// where this method was called.
/// - line: The line number to print if the assertion fails The default is
/// where this method was called.
@available(SwiftStdlib 5.1, *)
#if !$Embedded
@backDeployed(before: SwiftStdlib 5.9)
#endif
@_unavailableInEmbedded
public nonisolated func assertIsolated(
_ message: @autoclosure () -> String = String(),
file: StaticString = #fileID, line: UInt = #line
) {
guard _isDebugAssertConfiguration() else {
return
}
guard unsafe _taskIsCurrentExecutor(self.unownedExecutor.executor) else {
let msg = unsafe "Incorrect actor executor assumption; Expected '\(self.unownedExecutor)' executor. \(message())"
/// TODO: implement the logic in-place perhaps rather than delegating to precondition()?
assertionFailure(msg, file: file, line: line) // short-cut so we get the exact same failure reporting semantics
return
}
}
}
@available(SwiftStdlib 5.1, *)
extension GlobalActor {
/// Stops program execution if the current task is not executing on this
/// actor's serial executor.
///
/// This function's effect varies depending on the build flag used:
///
/// * In playgrounds and `-Onone` builds (the default for Xcode's Debug
/// configuration), stops program execution in a debuggable state after
/// printing `message`.
///
/// * In `-O` builds (the default for Xcode's Release configuration),
/// the isolation check is not performed and there are no effects.
///
/// - Note: This check is performed against the actor's serial executor,
/// meaning that / if another actor uses the same serial executor--by using
/// that actor's serial executor as its own ``Actor/unownedExecutor``--this
/// check will succeed , as from a concurrency safety perspective, the
/// serial executor guarantees mutual exclusion of those two actors.
///
/// - Parameters:
/// - message: The message to print if the assertion fails.
/// - file: The file name to print if the assertion fails. The default is
/// where this method was called.
/// - line: The line number to print if the assertion fails The default is
/// where this method was called.
@available(SwiftStdlib 5.1, *)
#if !$Embedded
@backDeployed(before: SwiftStdlib 5.9)
#endif
@_unavailableInEmbedded
public static func assertIsolated(
_ message: @autoclosure () -> String = String(),
file: StaticString = #fileID, line: UInt = #line
) {
Self.shared.assertIsolated(message(), file: file, line: line)
}
}
// ==== -----------------------------------------------------------------------
// MARK: Assume Executor
@available(SwiftStdlib 5.1, *)
extension Actor {
/// Assume that the current task is executing on this actor's serial executor,
/// or stop program execution otherwise.
///
/// You call this method to *assume and verify* that the currently
/// executing synchronous function is actually executing on the serial
/// executor of this actor.
///
/// If that is the case, the operation is invoked with an `isolated` version
/// of the actor, allowing synchronous access to actor local state without
/// hopping through asynchronous boundaries.
///
/// If the current context is not running on the actor's serial executor, or
/// if the actor is a reference to a remote actor, this method will crash
/// with a fatal error (similar to ``preconditionIsolated()``).
///
/// Note that this check is performed against the passed in actor's serial
/// executor, meaning that if another actor uses the same serial executor--by
/// using that actor's ``Actor/unownedExecutor`` as its own
/// ``Actor/unownedExecutor``--this check will succeed, as from a concurrency
/// safety perspective, the serial executor guarantees mutual exclusion of
/// those two actors.
///
/// This method can only be used from synchronous functions, as asynchronous
/// functions should instead perform a normal method call to the actor, which
/// will hop task execution to the target actor if necessary.
///
/// - Note: This check is performed against the actor's serial executor,
/// meaning that / if another actor uses the same serial executor--by using
/// another actor's executor as its own ``Actor/unownedExecutor``
/// --this check will succeed , as from a concurrency safety perspective,
/// the serial executor guarantees mutual exclusion of those two actors.
///
/// - Parameters:
/// - operation: the operation that will be executed if the current context
/// is executing on the actors serial executor.
/// - file: The file name to print if the assertion fails. The default is
/// where this method was called.
/// - line: The line number to print if the assertion fails The default is
/// where this method was called.
/// - Returns: the return value of the `operation`
/// - Throws: rethrows the `Error` thrown by the operation if it threw
@available(SwiftStdlib 5.1, *)
@_alwaysEmitIntoClient
@_unavailableFromAsync(message: "express the closure as an explicit function declared on the specified 'actor' instead")
@_unavailableInEmbedded
public nonisolated func assumeIsolated<T : Sendable>(
_ operation: (isolated Self) throws -> T,
file: StaticString = #fileID, line: UInt = #line
) rethrows -> T {
typealias YesActor = (isolated Self) throws -> T
typealias NoActor = (Self) throws -> T
/// This is guaranteed to be fatal if the check fails,
/// as this is our "safe" version of this API.
let executor: Builtin.Executor = unsafe self.unownedExecutor.executor
guard _taskIsCurrentExecutor(executor) else {
// TODO: offer information which executor we actually got
fatalError("Incorrect actor executor assumption; Expected same executor as \(self).", file: file, line: line)
}
// To do the unsafe cast, we have to pretend it's @escaping.
return try withoutActuallyEscaping(operation) {
(_ fn: @escaping YesActor) throws -> T in
let rawFn = unsafe unsafeBitCast(fn, to: NoActor.self)
return try rawFn(self)
}
}
@available(SwiftStdlib 5.9, *)
@usableFromInline
@_unavailableInEmbedded
@_silgen_name("$sScAsE14assumeIsolated_4file4lineqd__qd__xYiKXE_s12StaticStringVSutKlF")
internal nonisolated func __abi__assumeIsolated<T : Sendable>(
_ operation: (isolated Self) throws -> T,
_ file: StaticString, _ line: UInt
) rethrows -> T {
try assumeIsolated(operation, file: file, line: line)
}
}
#endif // not SWIFT_STDLIB_TASK_TO_THREAD_MODEL_CONCURRENCY