Files
swift-mirror/stdlib/core/LifetimeManager.swift
2014-07-07 10:13:44 +00:00

111 lines
3.2 KiB
Swift

//===----------------------------------------------------------------------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2015 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
//
//===----------------------------------------------------------------------===//
/// An instance of this struct keeps the references registered with it
/// at +1 reference count until the call to `release()`.
///
/// It is absolutely necessary to call `release()`. Forgetting to call
/// `release()` will not cause a memory leak. Instead, the managed objects will be
/// released earlier than expected.
///
/// This class can be used to extend lifetime of objects to pass UnsafePointers
/// to them to C APIs.
@internal class LifetimeManager {
var _managedRefs : [Builtin.NativeObject]
var _releaseCalled : Bool
init() {
_managedRefs = []
_releaseCalled = false
}
deinit {
if !_releaseCalled {
_fatalError("release() should have been called")
}
}
func put(objPtr: Builtin.NativeObject) {
_managedRefs.append(objPtr)
}
// FIXME: Need class constraints for this to work properly.
// func put<T>(obj: T) {
// put(Builtin.castToNativeObject(obj))
// }
/// Call this function to end the forced lifetime extension.
func release() {
_fixLifetime(_managedRefs._owner)
_releaseCalled = true
}
}
/// Evaluate `f()` and return its result, ensuring that `x` is not
/// destroyed before f returns.
@public func withExtendedLifetime<T, Result>(
x: T, f: ()->Result
) -> Result {
let result = f()
_fixLifetime(x)
return result
}
/// Evaluate `f(x)` and return its result, ensuring that `x` is not
/// destroyed before f returns.
@public func withExtendedLifetime<T, Result>(
x: T, f: (T)->Result
) -> Result {
let result = f(x)
_fixLifetime(x)
return result
}
extension String {
/// Invoke `f` on the contents of this string, represented as
/// a nul-terminated array of char, ensuring that the array's
/// lifetime extends through the execution of `f`.
@public func withCString<Result>(
f: (ConstUnsafePointer<Int8>)->Result
) -> Result {
return self.nulTerminatedUTF8.withUnsafePointerToElements {
f(ConstUnsafePointer($0))
}
}
/// Invoke `f` on the contents of this string, represented as
/// a nul-terminated array of char, ensuring that the array's
/// lifetime extends through the execution of `f`.
@public func withCString<Result>(
f: (UnsafePointer<CChar>)->Result
) -> Result {
// FIXME: This interface isn't const-correct; only the UnsafePointer variant
// above should be available.
return self.nulTerminatedUTF8.withUnsafePointerToElements {
f(UnsafePointer($0))
}
}
}
// This function should be opaque to the optimizer.
// BLOCKED: <rdar://problem/16464507> This function will be unnecessary when
// fix_lifetime is honored by the ARC optimizer.
@asmname("swift_keepAlive") @internal
func _swift_keepAlive<T>(inout _: T)
@transparent @public
func _fixLifetime<T>(var x: T) {
_swift_keepAlive(&x)
}