mirror of
https://github.com/apple/swift.git
synced 2025-12-14 20:36:38 +01:00
* Migrate from `UnsafePointer<Void>` to `UnsafeRawPointer`. As proposed in SE-0107: UnsafeRawPointer. `void*` imports as `UnsafeMutableRawPointer`. `const void*` imports as `UnsafeRawPointer`. Occurrences of `UnsafePointer<Void>` are replaced with UnsafeRawPointer. * Migrate overlays from UnsafePointer<Void> to UnsafeRawPointer. This requires explicit memory binding in several places, particularly in NSData and CoreAudio. * Fix a bunch of test cases for Void->Raw migration. * qsort takes IUO values * Bridge `Unsafe[Mutable]RawPointer as `void [const] *`. * Parse #dsohandle as UnsafeMutableRawPointer * Update a bunch of test cases for Void->Raw migration. * Trivial fix for the SceneKit test case. * Add an UnsafeRawPointer self initializer. This is unfortunately necessary for assignment between types imported from C. * Tiny simplification of the initializer.
142 lines
4.3 KiB
Swift
142 lines
4.3 KiB
Swift
//===--- SwiftPrivatePthreadExtras.swift ----------------------------------===//
|
|
//
|
|
// This source file is part of the Swift.org open source project
|
|
//
|
|
// Copyright (c) 2014 - 2016 Apple Inc. and the Swift project authors
|
|
// Licensed under Apache License v2.0 with Runtime Library Exception
|
|
//
|
|
// See http://swift.org/LICENSE.txt for license information
|
|
// See http://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file contains wrappers for pthread APIs that are less painful to use
|
|
// than the C APIs.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#if os(OSX) || os(iOS) || os(watchOS) || os(tvOS)
|
|
import Darwin
|
|
#elseif os(Linux) || os(FreeBSD) || os(PS4) || os(Android)
|
|
import Glibc
|
|
#endif
|
|
|
|
/// An abstract base class to encapsulate the context necessary to invoke
|
|
/// a block from pthread_create.
|
|
internal class PthreadBlockContext {
|
|
/// Execute the block, and return an `UnsafeMutablePointer` to memory
|
|
/// allocated with `UnsafeMutablePointer.alloc` containing the result of the
|
|
/// block.
|
|
func run() -> UnsafeMutableRawPointer { fatalError("abstract") }
|
|
}
|
|
|
|
internal class PthreadBlockContextImpl<Argument, Result>: PthreadBlockContext {
|
|
let block: (Argument) -> Result
|
|
let arg: Argument
|
|
|
|
init(block: (Argument) -> Result, arg: Argument) {
|
|
self.block = block
|
|
self.arg = arg
|
|
super.init()
|
|
}
|
|
|
|
override func run() -> UnsafeMutableRawPointer {
|
|
let result = UnsafeMutablePointer<Result>.allocate(capacity: 1)
|
|
result.initialize(to: block(arg))
|
|
return UnsafeMutableRawPointer(result)
|
|
}
|
|
}
|
|
|
|
/// Entry point for `pthread_create` that invokes a block context.
|
|
internal func invokeBlockContext(
|
|
_ contextAsVoidPointer: UnsafeMutableRawPointer?
|
|
) -> UnsafeMutableRawPointer! {
|
|
// The context is passed in +1; we're responsible for releasing it.
|
|
let context = Unmanaged<PthreadBlockContext>
|
|
.fromOpaque(contextAsVoidPointer!)
|
|
.takeRetainedValue()
|
|
|
|
return context.run()
|
|
}
|
|
|
|
/// Block-based wrapper for `pthread_create`.
|
|
public func _stdlib_pthread_create_block<Argument, Result>(
|
|
_ attr: UnsafePointer<pthread_attr_t>?,
|
|
_ start_routine: (Argument) -> Result,
|
|
_ arg: Argument
|
|
) -> (CInt, pthread_t?) {
|
|
let context = PthreadBlockContextImpl(block: start_routine, arg: arg)
|
|
// We hand ownership off to `invokeBlockContext` through its void context
|
|
// argument.
|
|
let contextAsVoidPointer = Unmanaged.passRetained(context).toOpaque()
|
|
|
|
var threadID = _make_pthread_t()
|
|
let result = pthread_create(&threadID, attr,
|
|
{ invokeBlockContext($0) }, contextAsVoidPointer)
|
|
if result == 0 {
|
|
return (result, threadID)
|
|
} else {
|
|
return (result, nil)
|
|
}
|
|
}
|
|
|
|
#if os(Linux) || os(Android)
|
|
internal func _make_pthread_t() -> pthread_t {
|
|
return pthread_t()
|
|
}
|
|
#else
|
|
internal func _make_pthread_t() -> pthread_t? {
|
|
return nil
|
|
}
|
|
#endif
|
|
|
|
/// Block-based wrapper for `pthread_join`.
|
|
public func _stdlib_pthread_join<Result>(
|
|
_ thread: pthread_t,
|
|
_ resultType: Result.Type
|
|
) -> (CInt, Result?) {
|
|
var threadResultRawPtr: UnsafeMutableRawPointer? = nil
|
|
let result = pthread_join(thread, &threadResultRawPtr)
|
|
if result == 0 {
|
|
let threadResultPtr = threadResultRawPtr!.assumingMemoryBound(
|
|
to: Result.self)
|
|
let threadResult = threadResultPtr.pointee
|
|
threadResultPtr.deinitialize()
|
|
threadResultPtr.deallocate(capacity: 1)
|
|
return (result, threadResult)
|
|
} else {
|
|
return (result, nil)
|
|
}
|
|
}
|
|
|
|
public class _stdlib_Barrier {
|
|
var _pthreadBarrier: _stdlib_pthread_barrier_t
|
|
|
|
var _pthreadBarrierPtr: UnsafeMutablePointer<_stdlib_pthread_barrier_t> {
|
|
return UnsafeMutablePointer(_getUnsafePointerToStoredProperties(self))
|
|
}
|
|
|
|
public init(threadCount: Int) {
|
|
self._pthreadBarrier = _stdlib_pthread_barrier_t()
|
|
let ret = _stdlib_pthread_barrier_init(
|
|
_pthreadBarrierPtr, nil, CUnsignedInt(threadCount))
|
|
if ret != 0 {
|
|
fatalError("_stdlib_pthread_barrier_init() failed")
|
|
}
|
|
}
|
|
|
|
deinit {
|
|
let ret = _stdlib_pthread_barrier_destroy(_pthreadBarrierPtr)
|
|
if ret != 0 {
|
|
fatalError("_stdlib_pthread_barrier_destroy() failed")
|
|
}
|
|
}
|
|
|
|
public func wait() {
|
|
let ret = _stdlib_pthread_barrier_wait(_pthreadBarrierPtr)
|
|
if !(ret == 0 || ret == _stdlib_PTHREAD_BARRIER_SERIAL_THREAD) {
|
|
fatalError("_stdlib_pthread_barrier_wait() failed")
|
|
}
|
|
}
|
|
}
|