Files
swift-mirror/test/1_stdlib/Runtime.swift
Graham Batty 83b4384fac Update test flags for linux failures and support.
Also removed the sdk 'feature' in favour of the more specific
objc_interop.

Swift SVN r24856
2015-01-30 21:31:48 +00:00

1416 lines
38 KiB
Swift

// RUN: %target-build-swift -parse-stdlib -Xfrontend -disable-access-control -module-name a %s -o %t.out
// RUN: %target-run %t.out
// XFAIL: linux
import Swift
import StdlibUnittest
import Foundation
import CoreGraphics
var nsObjectCanaryCount = 0
@objc class NSObjectCanary : NSObject {
override init() {
++nsObjectCanaryCount
}
deinit {
--nsObjectCanaryCount
}
}
struct NSObjectCanaryStruct {
var ref = NSObjectCanary()
}
var swiftObjectCanaryCount = 0
class SwiftObjectCanary {
init() {
++swiftObjectCanaryCount
}
deinit {
--swiftObjectCanaryCount
}
}
struct SwiftObjectCanaryStruct {
var ref = SwiftObjectCanary()
}
@objc class ClassA {
init(value: Int) {
self.value = value
}
var value: Int
}
struct NotBridgedValueType {
// Keep it pointer-sized.
var canaryRef = SwiftObjectCanary()
}
struct BridgedValueType : _ObjectiveCBridgeable {
init(value: Int) {
self.value = value
}
static func _getObjectiveCType() -> Any.Type {
return ClassA.self
}
func _bridgeToObjectiveC() -> ClassA {
return ClassA(value: value)
}
static func _isBridgedToObjectiveC() -> Bool {
return true
}
static func _forceBridgeFromObjectiveC(
x: ClassA,
inout result: BridgedValueType?
) {
assert(x.value % 2 == 0, "not bridged to Objective-C")
result = BridgedValueType(value: x.value)
}
static func _conditionallyBridgeFromObjectiveC(
x: ClassA,
inout result: BridgedValueType?
) -> Bool {
if x.value % 2 == 0 {
result = BridgedValueType(value: x.value)
return true
}
result = nil
return false
}
var value: Int
var canaryRef = SwiftObjectCanary()
}
struct BridgedLargeValueType : _ObjectiveCBridgeable {
init(value: Int) {
value0 = value
value1 = value
value2 = value
value3 = value
value4 = value
value5 = value
value6 = value
value7 = value
}
static func _getObjectiveCType() -> Any.Type {
return ClassA.self
}
func _bridgeToObjectiveC() -> ClassA {
assert(value == value0)
return ClassA(value: value0)
}
static func _isBridgedToObjectiveC() -> Bool {
return true
}
static func _forceBridgeFromObjectiveC(
x: ClassA,
inout result: BridgedLargeValueType?
) {
assert(x.value % 2 == 0, "not bridged to Objective-C")
result = BridgedLargeValueType(value: x.value)
}
static func _conditionallyBridgeFromObjectiveC(
x: ClassA,
inout result: BridgedLargeValueType?
) -> Bool {
if x.value % 2 == 0 {
result = BridgedLargeValueType(value: x.value)
return true
}
result = nil
return false
}
var value: Int {
let x = value0
assert(value0 == x && value1 == x && value2 == x && value3 == x &&
value4 == x && value5 == x && value6 == x && value7 == x)
return x
}
var (value0, value1, value2, value3): (Int, Int, Int, Int)
var (value4, value5, value6, value7): (Int, Int, Int, Int)
var canaryRef = SwiftObjectCanary()
}
struct ConditionallyBridgedValueType<T> : _ObjectiveCBridgeable {
init(value: Int) {
self.value = value
}
static func _getObjectiveCType() -> Any.Type {
return ClassA.self
}
func _bridgeToObjectiveC() -> ClassA {
return ClassA(value: value)
}
static func _forceBridgeFromObjectiveC(
x: ClassA,
inout result: ConditionallyBridgedValueType?
) {
assert(x.value % 2 == 0, "not bridged from Objective-C")
result = ConditionallyBridgedValueType(value: x.value)
}
static func _conditionallyBridgeFromObjectiveC(
x: ClassA,
inout result: ConditionallyBridgedValueType?
) -> Bool {
if x.value % 2 == 0 {
result = ConditionallyBridgedValueType(value: x.value)
return true
}
result = nil
return false
}
static func _isBridgedToObjectiveC() -> Bool {
return ((T.self as Any) as? String.Type) == nil
}
var value: Int
var canaryRef = SwiftObjectCanary()
}
class BridgedVerbatimRefType {
var value: Int = 42
var canaryRef = SwiftObjectCanary()
}
func withSwiftObjectCanary<T>(
createValue: () -> T,
check: (T) -> (),
file: String = __FILE__, line: UWord = __LINE__
) {
let stackTrace = SourceLocStack(SourceLoc(file, line))
swiftObjectCanaryCount = 0
if true {
var valueWithCanary = createValue()
expectEqual(1, swiftObjectCanaryCount, stackTrace: stackTrace)
check(valueWithCanary)
}
expectEqual(0, swiftObjectCanaryCount, stackTrace: stackTrace)
}
var Runtime = TestSuite("Runtime")
func _isClassOrObjCExistential_Opaque<T>(x: T.Type) -> Bool {
return _isClassOrObjCExistential(_opaqueIdentity(x))
}
Runtime.test("_isClassOrObjCExistential") {
expectTrue(_isClassOrObjCExistential(NSObjectCanary.self))
expectTrue(_isClassOrObjCExistential_Opaque(NSObjectCanary.self))
expectFalse(_isClassOrObjCExistential(NSObjectCanaryStruct.self))
expectFalse(_isClassOrObjCExistential_Opaque(NSObjectCanaryStruct.self))
expectTrue(_isClassOrObjCExistential(SwiftObjectCanary.self))
expectTrue(_isClassOrObjCExistential_Opaque(SwiftObjectCanary.self))
expectFalse(_isClassOrObjCExistential(SwiftObjectCanaryStruct.self))
expectFalse(_isClassOrObjCExistential_Opaque(SwiftObjectCanaryStruct.self))
typealias SwiftClosure = ()->()
expectFalse(_isClassOrObjCExistential(SwiftClosure.self))
expectFalse(_isClassOrObjCExistential_Opaque(SwiftClosure.self))
typealias ObjCClosure = @objc_block ()->()
expectTrue(_isClassOrObjCExistential(ObjCClosure.self))
expectTrue(_isClassOrObjCExistential_Opaque(ObjCClosure.self))
expectTrue(_isClassOrObjCExistential(CFArray.self))
expectTrue(_isClassOrObjCExistential_Opaque(CFArray.self))
}
Runtime.test("_canBeClass") {
expectEqual(1, _canBeClass(NSObjectCanary.self))
expectEqual(0, _canBeClass(NSObjectCanaryStruct.self))
expectEqual(1, _canBeClass(SwiftObjectCanary.self))
expectEqual(0, _canBeClass(SwiftObjectCanaryStruct.self))
typealias SwiftClosure = ()->()
expectEqual(0, _canBeClass(SwiftClosure.self))
typealias ObjCClosure = @objc_block ()->()
expectEqual(1, _canBeClass(ObjCClosure.self))
expectEqual(1, _canBeClass(CFArray.self))
}
Runtime.test("bridgeToObjectiveC") {
expectEmpty(_bridgeToObjectiveC(NotBridgedValueType()))
expectEqual(42, (_bridgeToObjectiveC(BridgedValueType(value: 42)) as! ClassA).value)
expectEqual(42, (_bridgeToObjectiveC(BridgedLargeValueType(value: 42)) as! ClassA).value)
expectEqual(42, (_bridgeToObjectiveC(ConditionallyBridgedValueType<Int>(value: 42)) as! ClassA).value)
expectEmpty(_bridgeToObjectiveC(ConditionallyBridgedValueType<String>(value: 42)))
var bridgedVerbatimRef = BridgedVerbatimRefType()
expectTrue(_bridgeToObjectiveC(bridgedVerbatimRef) === bridgedVerbatimRef)
}
Runtime.test("bridgeToObjectiveC/NoLeak") {
withSwiftObjectCanary(
{ NotBridgedValueType() },
{ expectEmpty(_bridgeToObjectiveC($0)) })
withSwiftObjectCanary(
{ BridgedValueType(value: 42) },
{ expectEqual(42, (_bridgeToObjectiveC($0) as! ClassA).value) })
withSwiftObjectCanary(
{ BridgedLargeValueType(value: 42) },
{ expectEqual(42, (_bridgeToObjectiveC($0) as! ClassA).value) })
withSwiftObjectCanary(
{ ConditionallyBridgedValueType<Int>(value: 42) },
{ expectEqual(42, (_bridgeToObjectiveC($0) as! ClassA).value) })
withSwiftObjectCanary(
{ ConditionallyBridgedValueType<String>(value: 42) },
{ expectEmpty(_bridgeToObjectiveC($0)) })
withSwiftObjectCanary(
{ BridgedVerbatimRefType() },
{ expectTrue(_bridgeToObjectiveC($0) === $0) })
}
Runtime.test("forceBridgeFromObjectiveC") {
// Bridge back using NotBridgedValueType.
expectEmpty(_conditionallyBridgeFromObjectiveC(
ClassA(value: 21), NotBridgedValueType.self))
expectEmpty(_conditionallyBridgeFromObjectiveC(
ClassA(value: 42), NotBridgedValueType.self))
expectEmpty(_conditionallyBridgeFromObjectiveC(
BridgedVerbatimRefType(), NotBridgedValueType.self))
// Bridge back using BridgedValueType.
expectEmpty(_conditionallyBridgeFromObjectiveC(
ClassA(value: 21), BridgedValueType.self))
expectEqual(42, _forceBridgeFromObjectiveC(
ClassA(value: 42), BridgedValueType.self).value)
expectEqual(42, _conditionallyBridgeFromObjectiveC(
ClassA(value: 42), BridgedValueType.self)!.value)
expectEmpty(_conditionallyBridgeFromObjectiveC(
BridgedVerbatimRefType(), BridgedValueType.self))
// Bridge back using BridgedLargeValueType.
expectEmpty(_conditionallyBridgeFromObjectiveC(
ClassA(value: 21), BridgedLargeValueType.self))
expectEqual(42, _forceBridgeFromObjectiveC(
ClassA(value: 42), BridgedLargeValueType.self).value)
expectEqual(42, _conditionallyBridgeFromObjectiveC(
ClassA(value: 42), BridgedLargeValueType.self)!.value)
expectEmpty(_conditionallyBridgeFromObjectiveC(
BridgedVerbatimRefType(), BridgedLargeValueType.self))
// Bridge back using BridgedVerbatimRefType.
expectEmpty(_conditionallyBridgeFromObjectiveC(
ClassA(value: 21), BridgedVerbatimRefType.self))
expectEmpty(_conditionallyBridgeFromObjectiveC(
ClassA(value: 42), BridgedVerbatimRefType.self))
var bridgedVerbatimRef = BridgedVerbatimRefType()
expectTrue(_forceBridgeFromObjectiveC(
bridgedVerbatimRef, BridgedVerbatimRefType.self) === bridgedVerbatimRef)
expectTrue(_conditionallyBridgeFromObjectiveC(
bridgedVerbatimRef, BridgedVerbatimRefType.self)! === bridgedVerbatimRef)
}
Runtime.test("isBridgedToObjectiveC") {
expectFalse(_isBridgedToObjectiveC(NotBridgedValueType))
expectTrue(_isBridgedToObjectiveC(BridgedValueType))
expectTrue(_isBridgedToObjectiveC(BridgedVerbatimRefType))
}
Runtime.test("isBridgedVerbatimToObjectiveC") {
expectFalse(_isBridgedVerbatimToObjectiveC(NotBridgedValueType))
expectFalse(_isBridgedVerbatimToObjectiveC(BridgedValueType))
expectTrue(_isBridgedVerbatimToObjectiveC(BridgedVerbatimRefType))
}
//===---------------------------------------------------------------------===//
// The protocol should be defined in the standard library, otherwise the cast
// does not work.
typealias P1 = BooleanType
typealias P2 = Printable
protocol Q1 {}
// A small struct that can be stored inline in an opaque buffer.
struct StructConformsToP1 : BooleanType, Q1 {
var boolValue: Bool {
return true
}
}
// A small struct that can be stored inline in an opaque buffer.
struct Struct2ConformsToP1<T : BooleanType> : BooleanType, Q1 {
init(_ value: T) {
self.value = value
}
var boolValue: Bool {
return value.boolValue
}
var value: T
}
// A large struct that can not be stored inline in an opaque buffer.
struct Struct3ConformsToP2 : Printable, Q1 {
var a: UInt64 = 10
var b: UInt64 = 20
var c: UInt64 = 30
var d: UInt64 = 40
var description: String {
// Don't rely on string interpolation, it uses the casts that we are trying
// to test.
var result = ""
result += _uint64ToString(a) + " "
result += _uint64ToString(b) + " "
result += _uint64ToString(c) + " "
result += _uint64ToString(d)
return result
}
}
// A large struct that can not be stored inline in an opaque buffer.
struct Struct4ConformsToP2<T : Printable> : Printable, Q1 {
var value: T
var e: UInt64 = 50
var f: UInt64 = 60
var g: UInt64 = 70
var h: UInt64 = 80
init(_ value: T) {
self.value = value
}
var description: String {
// Don't rely on string interpolation, it uses the casts that we are trying
// to test.
var result = value.description + " "
result += _uint64ToString(e) + " "
result += _uint64ToString(f) + " "
result += _uint64ToString(g) + " "
result += _uint64ToString(h)
return result
}
}
struct StructDoesNotConformToP1 : Q1 {}
class ClassConformsToP1 : BooleanType, Q1 {
var boolValue: Bool {
return true
}
}
class Class2ConformsToP1<T : BooleanType> : BooleanType, Q1 {
init(_ value: T) {
self.value = [ value ]
}
var boolValue: Bool {
return value[0].boolValue
}
// FIXME: should be "var value: T", but we don't support it now.
var value: Array<T>
}
class ClassDoesNotConformToP1 : Q1 {}
Runtime.test("dynamicCasting with as") {
var someP1Value = StructConformsToP1()
var someP1Value2 = Struct2ConformsToP1(true)
var someNotP1Value = StructDoesNotConformToP1()
var someP2Value = Struct3ConformsToP2()
var someP2Value2 = Struct4ConformsToP2(Struct3ConformsToP2())
var someP1Ref = ClassConformsToP1()
var someP1Ref2 = Class2ConformsToP1(true)
var someNotP1Ref = ClassDoesNotConformToP1()
expectTrue(_stdlib_conformsToProtocol(someP1Value, P1.self))
expectTrue(_stdlib_conformsToProtocol(someP1Value2, P1.self))
expectFalse(_stdlib_conformsToProtocol(someNotP1Value, P1.self))
expectTrue(_stdlib_conformsToProtocol(someP2Value, P2.self))
expectTrue(_stdlib_conformsToProtocol(someP2Value2, P2.self))
expectTrue(_stdlib_conformsToProtocol(someP1Ref, P1.self))
expectTrue(_stdlib_conformsToProtocol(someP1Ref2, P1.self))
expectFalse(_stdlib_conformsToProtocol(someNotP1Ref, P1.self))
expectTrue(_stdlib_conformsToProtocol(someP1Value as P1, P1.self))
expectTrue(_stdlib_conformsToProtocol(someP1Value2 as P1, P1.self))
expectTrue(_stdlib_conformsToProtocol(someP2Value as P2, P2.self))
expectTrue(_stdlib_conformsToProtocol(someP2Value2 as P2, P2.self))
expectTrue(_stdlib_conformsToProtocol(someP1Ref as P1, P1.self))
expectTrue(_stdlib_conformsToProtocol(someP1Value as Q1, P1.self))
expectTrue(_stdlib_conformsToProtocol(someP1Value2 as Q1, P1.self))
expectFalse(_stdlib_conformsToProtocol(someNotP1Value as Q1, P1.self))
expectTrue(_stdlib_conformsToProtocol(someP2Value as Q1, P2.self))
expectTrue(_stdlib_conformsToProtocol(someP2Value2 as Q1, P2.self))
expectTrue(_stdlib_conformsToProtocol(someP1Ref as Q1, P1.self))
expectTrue(_stdlib_conformsToProtocol(someP1Ref2 as Q1, P1.self))
expectFalse(_stdlib_conformsToProtocol(someNotP1Ref as Q1, P1.self))
expectTrue(_stdlib_conformsToProtocol(someP1Value as Any, P1.self))
expectTrue(_stdlib_conformsToProtocol(someP1Value2 as Any, P1.self))
expectFalse(_stdlib_conformsToProtocol(someNotP1Value as Any, P1.self))
expectTrue(_stdlib_conformsToProtocol(someP2Value as Any, P2.self))
expectTrue(_stdlib_conformsToProtocol(someP2Value2 as Any, P2.self))
expectTrue(_stdlib_conformsToProtocol(someP1Ref as Any, P1.self))
expectTrue(_stdlib_conformsToProtocol(someP1Ref2 as Any, P1.self))
expectFalse(_stdlib_conformsToProtocol(someNotP1Ref as Any, P1.self))
expectTrue(_stdlib_conformsToProtocol(someP1Ref as AnyObject, P1.self))
expectTrue(_stdlib_conformsToProtocol(someP1Ref2 as AnyObject, P1.self))
expectFalse(_stdlib_conformsToProtocol(someNotP1Ref as AnyObject, P1.self))
expectTrue((someP1Value as P1).boolValue)
expectTrue((someP1Value2 as P1).boolValue)
expectEqual("10 20 30 40", (someP2Value as P2).description)
expectEqual("10 20 30 40 50 60 70 80", (someP2Value2 as P2).description)
expectTrue((someP1Ref as P1).boolValue)
expectTrue((someP1Ref2 as P1).boolValue)
expectTrue(((someP1Value as Q1) as! P1).boolValue)
expectTrue(((someP1Value2 as Q1) as! P1).boolValue)
expectEqual("10 20 30 40", ((someP2Value as Q1) as! P2).description)
expectEqual("10 20 30 40 50 60 70 80",
((someP2Value2 as Q1) as! P2).description)
expectTrue(((someP1Ref as Q1) as! P1).boolValue)
expectTrue(((someP1Ref2 as Q1) as! P1).boolValue)
expectTrue(((someP1Value as Any) as! P1).boolValue)
expectTrue(((someP1Value2 as Any) as! P1).boolValue)
expectEqual("10 20 30 40", ((someP2Value as Any) as! P2).description)
expectEqual("10 20 30 40 50 60 70 80",
((someP2Value2 as Any) as! P2).description)
expectTrue(((someP1Ref as Any) as! P1).boolValue)
expectTrue(((someP1Ref2 as Any) as! P1).boolValue)
expectTrue(((someP1Ref as AnyObject) as! P1).boolValue)
expectEmpty((someNotP1Value as? P1))
expectEmpty((someNotP1Ref as? P1))
expectTrue(((someP1Value as Q1) as? P1)!.boolValue)
expectTrue(((someP1Value2 as Q1) as? P1)!.boolValue)
expectEmpty(((someNotP1Value as Q1) as? P1))
expectEqual("10 20 30 40", ((someP2Value as Q1) as? P2)!.description)
expectEqual("10 20 30 40 50 60 70 80",
((someP2Value2 as Q1) as? P2)!.description)
expectTrue(((someP1Ref as Q1) as? P1)!.boolValue)
expectTrue(((someP1Ref2 as Q1) as? P1)!.boolValue)
expectEmpty(((someNotP1Ref as Q1) as? P1))
expectTrue(((someP1Value as Any) as? P1)!.boolValue)
expectTrue(((someP1Value2 as Any) as? P1)!.boolValue)
expectEmpty(((someNotP1Value as Any) as? P1))
expectEqual("10 20 30 40", ((someP2Value as Any) as? P2)!.description)
expectEqual("10 20 30 40 50 60 70 80",
((someP2Value2 as Any) as? P2)!.description)
expectTrue(((someP1Ref as Any) as? P1)!.boolValue)
expectTrue(((someP1Ref2 as Any) as? P1)!.boolValue)
expectEmpty(((someNotP1Ref as Any) as? P1))
expectTrue(((someP1Ref as AnyObject) as? P1)!.boolValue)
expectTrue(((someP1Ref2 as AnyObject) as? P1)!.boolValue)
expectEmpty(((someNotP1Ref as AnyObject) as? P1))
}
extension Int {
class ExtensionClassConformsToP2 : P2 {
var description: String { return "abc" }
}
private class PrivateExtensionClassConformsToP2 : P2 {
var description: String { return "def" }
}
}
Runtime.test("dynamicCastToExistential1 with cross-module extensions") {
let internalObj = Int.ExtensionClassConformsToP2()
let privateObj = Int.PrivateExtensionClassConformsToP2()
expectTrue(_stdlib_conformsToProtocol(internalObj, P2.self))
expectTrue(_stdlib_conformsToProtocol(privateObj, P2.self))
}
class SomeClass {}
@objc class SomeObjCClass {}
class SomeNSObjectSubclass : NSObject {}
struct SomeStruct {}
enum SomeEnum {
case A
init() { self = .A }
}
Runtime.test("getDemangledTypeName") {
expectEqual("a.SomeClass", _stdlib_getDemangledTypeName(SomeClass()))
expectEqual("a.SomeObjCClass", _stdlib_getDemangledTypeName(SomeObjCClass()))
expectEqual("a.SomeNSObjectSubclass", _stdlib_getDemangledTypeName(SomeNSObjectSubclass()))
expectEqual("NSObject", _stdlib_getDemangledTypeName(NSObject()))
expectEqual("a.SomeStruct", _stdlib_getDemangledTypeName(SomeStruct()))
expectEqual("a.SomeEnum", _stdlib_getDemangledTypeName(SomeEnum()))
var a: Any = SomeClass()
expectEqual("a.SomeClass", _stdlib_getDemangledTypeName(a))
a = SomeObjCClass()
expectEqual("a.SomeObjCClass", _stdlib_getDemangledTypeName(a))
a = SomeNSObjectSubclass()
expectEqual("a.SomeNSObjectSubclass", _stdlib_getDemangledTypeName(a))
a = NSObject()
expectEqual("NSObject", _stdlib_getDemangledTypeName(a))
a = SomeStruct()
expectEqual("a.SomeStruct", _stdlib_getDemangledTypeName(a))
a = SomeEnum()
expectEqual("a.SomeEnum", _stdlib_getDemangledTypeName(a))
}
Runtime.test("demangleName") {
expectEqual("", _stdlib_demangleName(""))
expectEqual("abc", _stdlib_demangleName("abc"))
expectEqual("\0", _stdlib_demangleName("\0"))
expectEqual("Swift.Double", _stdlib_demangleName("_TtSd"))
expectEqual("x.a : x.Foo<x.Foo<x.Foo<Swift.Int, Swift.Int>, x.Foo<Swift.Int, Swift.Int>>, x.Foo<x.Foo<Swift.Int, Swift.Int>, x.Foo<Swift.Int, Swift.Int>>>",
_stdlib_demangleName("_Tv1x1aGCS_3FooGS0_GS0_SiSi_GS0_SiSi__GS0_GS0_SiSi_GS0_SiSi___"))
}
Runtime.test("_stdlib_atomicCompareExchangeStrongPtr") {
typealias IntPtr = UnsafeMutablePointer<Int>
var origP1 = IntPtr(bitPattern: 0x10101010)
var origP2 = IntPtr(bitPattern: 0x20202020)
var origP3 = IntPtr(bitPattern: 0x30303030)
if true {
var object = origP1
var expected = origP1
let r = _stdlib_atomicCompareExchangeStrongPtr(
object: &object, expected: &expected, desired: origP2)
expectTrue(r)
expectEqual(origP2, object)
expectEqual(origP1, expected)
}
if true {
var object = origP1
var expected = origP2
let r = _stdlib_atomicCompareExchangeStrongPtr(
object: &object, expected: &expected, desired: origP3)
expectFalse(r)
expectEqual(origP1, object)
expectEqual(origP1, expected)
}
struct FooStruct {
var i: Int
var object: IntPtr
var expected: IntPtr
init(_ object: IntPtr, _ expected: IntPtr) {
self.i = 0
self.object = object
self.expected = expected
}
}
if true {
var foo = FooStruct(origP1, origP1)
let r = _stdlib_atomicCompareExchangeStrongPtr(
object: &foo.object, expected: &foo.expected, desired: origP2)
expectTrue(r)
expectEqual(origP2, foo.object)
expectEqual(origP1, foo.expected)
}
if true {
var foo = FooStruct(origP1, origP2)
let r = _stdlib_atomicCompareExchangeStrongPtr(
object: &foo.object, expected: &foo.expected, desired: origP3)
expectFalse(r)
expectEqual(origP1, foo.object)
expectEqual(origP1, foo.expected)
}
}
var RuntimeFoundationWrappers = TestSuite("RuntimeFoundationWrappers")
RuntimeFoundationWrappers.test("_stdlib_NSObject_isEqual/NoLeak") {
nsObjectCanaryCount = 0
if true {
let a = NSObjectCanary()
let b = NSObjectCanary()
expectEqual(2, nsObjectCanaryCount)
_stdlib_NSObject_isEqual(a, b)
}
expectEqual(0, nsObjectCanaryCount)
}
var nsStringCanaryCount = 0
@objc class NSStringCanary : NSString {
override init() {
++nsStringCanaryCount
super.init()
}
required init(coder: NSCoder) {
fatalError("don't call this initializer")
}
deinit {
--nsStringCanaryCount
}
@objc override var length: Int {
return 0
}
@objc override func characterAtIndex(index: Int) -> unichar {
fatalError("out-of-bounds access")
}
}
RuntimeFoundationWrappers.test(
"_stdlib_compareNSStringDeterministicUnicodeCollation/NoLeak"
) {
nsStringCanaryCount = 0
if true {
let a = NSStringCanary()
let b = NSStringCanary()
expectEqual(2, nsStringCanaryCount)
_stdlib_compareNSStringDeterministicUnicodeCollation(a, b)
}
expectEqual(0, nsStringCanaryCount)
}
RuntimeFoundationWrappers.test("_stdlib_NSStringNFDHashValue/NoLeak") {
nsStringCanaryCount = 0
if true {
let a = NSStringCanary()
expectEqual(1, nsStringCanaryCount)
_stdlib_NSStringNFDHashValue(a)
}
expectEqual(0, nsStringCanaryCount)
}
RuntimeFoundationWrappers.test("_stdlib_NSStringASCIIHashValue/NoLeak") {
nsStringCanaryCount = 0
if true {
let a = NSStringCanary()
expectEqual(1, nsStringCanaryCount)
_stdlib_NSStringASCIIHashValue(a)
}
expectEqual(0, nsStringCanaryCount)
}
RuntimeFoundationWrappers.test("_stdlib_NSStringHasPrefixNFD/NoLeak") {
nsStringCanaryCount = 0
if true {
let a = NSStringCanary()
let b = NSStringCanary()
expectEqual(2, nsStringCanaryCount)
_stdlib_NSStringHasPrefixNFD(a, b)
}
expectEqual(0, nsStringCanaryCount)
}
RuntimeFoundationWrappers.test("_stdlib_NSStringHasSuffixNFD/NoLeak") {
nsStringCanaryCount = 0
if true {
let a = NSStringCanary()
let b = NSStringCanary()
expectEqual(2, nsStringCanaryCount)
_stdlib_NSStringHasSuffixNFD(a, b)
}
expectEqual(0, nsStringCanaryCount)
}
RuntimeFoundationWrappers.test("_stdlib_NSStringLowercaseString/NoLeak") {
nsStringCanaryCount = 0
if true {
let a = NSStringCanary()
expectEqual(1, nsStringCanaryCount)
_stdlib_NSStringLowercaseString(a)
}
expectEqual(0, nsStringCanaryCount)
}
RuntimeFoundationWrappers.test("_stdlib_NSStringUppercaseString/NoLeak") {
nsStringCanaryCount = 0
if true {
let a = NSStringCanary()
expectEqual(1, nsStringCanaryCount)
_stdlib_NSStringUppercaseString(a)
}
expectEqual(0, nsStringCanaryCount)
}
RuntimeFoundationWrappers.test("_stdlib_CFStringCreateCopy/NoLeak") {
nsStringCanaryCount = 0
if true {
let a = NSStringCanary()
expectEqual(1, nsStringCanaryCount)
_stdlib_CFStringCreateCopy(a)
}
expectEqual(0, nsStringCanaryCount)
}
RuntimeFoundationWrappers.test("_stdlib_CFStringGetLength/NoLeak") {
nsStringCanaryCount = 0
if true {
let a = NSStringCanary()
expectEqual(1, nsStringCanaryCount)
_stdlib_CFStringGetLength(a)
}
expectEqual(0, nsStringCanaryCount)
}
RuntimeFoundationWrappers.test("_stdlib_CFStringGetCharactersPtr/NoLeak") {
nsStringCanaryCount = 0
if true {
let a = NSStringCanary()
expectEqual(1, nsStringCanaryCount)
_stdlib_CFStringGetCharactersPtr(a)
}
expectEqual(0, nsStringCanaryCount)
}
var Reflection = TestSuite("Reflection")
Reflection.test("dumpToAStream") {
var output = ""
dump([ 42, 4242 ], &output)
expectEqual("▿ 2 elements\n - [0]: 42\n - [1]: 4242\n", output)
}
// A struct type that gets destructured by the default mirror.
struct Matte {
let s: String
init (_ s: String) {
self.s = s
}
}
Reflection.test("StructMirror") {
if true {
var output = ""
dump(Matte("123"), &output)
expectEqual("▿ a.Matte\n - s: 123\n", output)
}
if true {
// Build a String around an interpolation as a way of smoke-testing that
// the internal MirrorType implementation gets memory management right.
var output = ""
dump(Matte("\(456)"), &output)
expectEqual("▿ a.Matte\n - s: 456\n", output)
}
// Structs have no identity and thus no object identifier
expectEmpty(reflect(Matte("")).objectIdentifier)
// The default mirror provides no quick look object
expectEmpty(reflect(Matte("")).quickLookObject)
expectEqual(.Struct, reflect(Matte("")).disposition)
}
/// A type that provides its own mirror.
struct BrilliantMirror : MirrorType {
let _value: Brilliant
init (_ _value: Brilliant) {
self._value = _value
}
var value: Any {
return _value
}
var valueType: Any.Type {
return value.dynamicType
}
var objectIdentifier: ObjectIdentifier? {
return ObjectIdentifier(_value)
}
var count: Int {
return 3
}
subscript(i: Int) -> (String, MirrorType) {
switch i {
case 0:
return ("first", reflect(_value.first))
case 1:
return ("second", reflect(_value.second))
case 2:
return ("self", self)
case _:
_preconditionFailure("child index out of bounds")
}
}
var summary: String {
return "Brilliant(\(_value.first), \(_value.second))"
}
var quickLookObject: QuickLookObject? {
return nil
}
var disposition: MirrorDisposition {
return .Container
}
}
class Brilliant : Reflectable {
let first: Int
let second: String
init(_ fst: Int, _ snd: String) {
self.first = fst
self.second = snd
}
func getMirror() -> MirrorType {
return BrilliantMirror(self)
}
}
/// Subclasses inherit their parents' custom mirrors.
class Irradiant : Brilliant {
init() {
super.init(400, "")
}
}
Reflection.test("CustomMirror") {
if true {
var output = ""
dump(Brilliant(123, "four five six"), &output)
let expected =
"▿ Brilliant(123, four five six) #0\n" +
" - first: 123\n" +
" - second: four five six\n" +
" ▿ self: Brilliant(123, four five six) #0\n"
expectEqual(expected, output)
}
if true {
var output = ""
dump(Brilliant(123, "four five six"), &output, maxDepth: 0)
expectEqual("▹ Brilliant(123, four five six) #0\n", output)
}
if true {
var output = ""
dump(Brilliant(123, "four five six"), &output, maxItems: 3)
let expected =
"▿ Brilliant(123, four five six) #0\n" +
" - first: 123\n" +
" - second: four five six\n" +
" (1 more child)\n"
expectEqual(expected, output)
}
if true {
var output = ""
dump(Brilliant(123, "four five six"), &output, maxItems: 2)
let expected =
"▿ Brilliant(123, four five six) #0\n" +
" - first: 123\n" +
" (2 more children)\n"
expectEqual(expected, output)
}
if true {
var output = ""
dump(Brilliant(123, "four five six"), &output, maxItems: 1)
let expected =
"▿ Brilliant(123, four five six) #0\n" +
" (3 children)\n"
expectEqual(expected, output)
}
expectEqual(.Container, reflect(Brilliant(123, "four five six")).disposition)
if true {
// Check that object identifiers are unique to class instances.
let a = Brilliant(1, "")
let b = Brilliant(2, "")
let c = Brilliant(3, "")
// Equatable
checkEquatable(true, ObjectIdentifier(a), ObjectIdentifier(a))
checkEquatable(false, ObjectIdentifier(a), ObjectIdentifier(b))
// Comparable
func isComparable<X : Comparable>(x: X) {}
isComparable(ObjectIdentifier(a))
// Check the ObjectIdentifier created is stable
expectTrue(
(ObjectIdentifier(a) < ObjectIdentifier(b))
!= (ObjectIdentifier(a) > ObjectIdentifier(b)))
expectFalse(
ObjectIdentifier(a) >= ObjectIdentifier(b)
&& ObjectIdentifier(a) <= ObjectIdentifier(b))
// Check ordering is transitive
expectEqual(
sorted([ObjectIdentifier(a), ObjectIdentifier(b), ObjectIdentifier(c)]),
sorted([ObjectIdentifier(c), ObjectIdentifier(b), ObjectIdentifier(a)]))
}
}
Reflection.test("CustomMirrorIsInherited") {
if true {
var output = ""
dump(Irradiant(), &output)
let expected =
"▿ Brilliant(400, ) #0\n" +
" - first: 400\n" +
" - second: \n" +
" ▿ self: Brilliant(400, ) #0\n"
expectEqual(expected, output)
}
}
protocol SomeNativeProto {}
extension Int: SomeNativeProto {}
@objc protocol SomeObjCProto {}
extension SomeClass: SomeObjCProto {}
Reflection.test("MetatypeMirror") {
if true {
var output = ""
let concreteMetatype = Int.self
dump(concreteMetatype, &output)
let expectedInt = "- Swift.Int #0\n"
expectEqual(expectedInt, output)
let anyMetatype: Any.Type = Int.self
output = ""
dump(anyMetatype, &output)
expectEqual(expectedInt, output)
let nativeProtocolMetatype: SomeNativeProto.Type = Int.self
output = ""
dump(nativeProtocolMetatype, &output)
expectEqual(expectedInt, output)
expectEqual(reflect(concreteMetatype).objectIdentifier!,
reflect(anyMetatype).objectIdentifier!)
expectEqual(reflect(concreteMetatype).objectIdentifier!,
reflect(nativeProtocolMetatype).objectIdentifier!)
let concreteClassMetatype = SomeClass.self
let expectedSomeClass = "- a.SomeClass #0\n"
output = ""
dump(concreteClassMetatype, &output)
expectEqual(expectedSomeClass, output)
let objcProtocolMetatype: SomeObjCProto.Type = SomeClass.self
output = ""
dump(objcProtocolMetatype, &output)
expectEqual(expectedSomeClass, output)
expectEqual(reflect(concreteClassMetatype).objectIdentifier!,
reflect(objcProtocolMetatype).objectIdentifier!)
}
}
Reflection.test("TupleMirror") {
if true {
var output = ""
let tuple = (Brilliant(384, "seven six eight"), Matte("nine"))
dump(tuple, &output)
let expected =
"▿ (2 elements)\n" +
" ▿ .0: Brilliant(384, seven six eight) #0\n" +
" - first: 384\n" +
" - second: seven six eight\n" +
" ▿ self: Brilliant(384, seven six eight) #0\n" +
" ▿ .1: a.Matte\n" +
" - s: nine\n"
expectEqual(expected, output)
expectEmpty(reflect(tuple).quickLookObject)
expectEqual(.Tuple, reflect(tuple).disposition)
}
if true {
// A tuple of stdlib types with mirrors.
var output = ""
let tuple = (1, 2.5, false, "three")
dump(tuple, &output)
let expected =
"▿ (4 elements)\n" +
" - .0: 1\n" +
" - .1: 2.5\n" +
" - .2: false\n" +
" - .3: three\n"
expectEqual(expected, output)
}
if true {
// A nested tuple.
var output = ""
let tuple = (1, ("Hello", "World"))
dump(tuple, &output)
let expected =
"▿ (2 elements)\n" +
" - .0: 1\n" +
" ▿ .1: (2 elements)\n" +
" - .0: Hello\n" +
" - .1: World\n"
expectEqual(expected, output)
}
}
class DullClass {}
Reflection.test("ObjectIdentity") {
// Check that the primitive MirrorType implementation produces appropriately
// unique identifiers for class instances.
let x = DullClass()
let y = DullClass()
let o = NSObject()
let p = NSObject()
checkEquatable(
true, reflect(x).objectIdentifier!, reflect(x).objectIdentifier!)
checkEquatable(
false, reflect(x).objectIdentifier!, reflect(y).objectIdentifier!)
checkEquatable(
true, reflect(o).objectIdentifier!, reflect(o).objectIdentifier!)
checkEquatable(
false, reflect(o).objectIdentifier!, reflect(p).objectIdentifier!)
checkEquatable(
false, reflect(o).objectIdentifier!, reflect(y).objectIdentifier!)
expectEmpty(reflect(x).quickLookObject)
expectEqual(.Class, reflect(x).disposition)
}
Reflection.test("String.UTF8View/Mirror") {
// U+0061 LATIN SMALL LETTER A
// U+304B HIRAGANA LETTER KA
// U+3099 COMBINING KATAKANA-HIRAGANA VOICED SOUND MARK
var output = ""
dump("\u{61}\u{304b}\u{3099}".utf8, &output)
let expected =
"\u{61}\u{304b}\u{3099}\n" +
" - [0]: 97\n" +
" - [1]: 227\n" +
" - [2]: 129\n" +
" - [3]: 139\n" +
" - [4]: 227\n" +
" - [5]: 130\n" +
" - [6]: 153\n"
expectEqual(expected, output)
}
Reflection.test("String.UTF16View/Mirror") {
// U+0061 LATIN SMALL LETTER A
// U+304B HIRAGANA LETTER KA
// U+3099 COMBINING KATAKANA-HIRAGANA VOICED SOUND MARK
// U+1F425 FRONT-FACING BABY CHICK
var output = ""
dump("\u{61}\u{304b}\u{3099}\u{1f425}".utf16, &output)
let expected =
"\u{61}\u{304b}\u{3099}\u{1f425}\n" +
" - [0]: 97\n" +
" - [1]: 12363\n" +
" - [2]: 12441\n" +
" - [3]: 55357\n" +
" - [4]: 56357\n"
expectEqual(expected, output)
}
Reflection.test("String.UnicodeScalarView/Mirror") {
// U+0061 LATIN SMALL LETTER A
// U+304B HIRAGANA LETTER KA
// U+3099 COMBINING KATAKANA-HIRAGANA VOICED SOUND MARK
// U+1F425 FRONT-FACING BABY CHICK
var output = ""
dump("\u{61}\u{304b}\u{3099}\u{1f425}".unicodeScalars, &output)
let expected =
"\u{61}\u{304b}\u{3099}\u{1f425}\n" +
" - [0]: \u{61}\n" +
" - [1]: \u{304b}\n" +
" - [2]: \u{3099}\n" +
" - [3]: \u{1f425}\n"
expectEqual(expected, output)
}
Reflection.test("CGPoint") {
var output = ""
dump(CGPoint(x: 1.25, y: 2.75), &output)
let expected =
"▿ (1.25, 2.75)\n" +
" - x: 1.25\n" +
" - y: 2.75\n"
expectEqual(expected, output)
}
Reflection.test("CGSize") {
var output = ""
dump(CGSize(width: 1.25, height: 2.75), &output)
let expected =
"▿ (1.25, 2.75)\n" +
" - width: 1.25\n" +
" - height: 2.75\n"
expectEqual(expected, output)
}
Reflection.test("CGRect") {
var output = ""
dump(
CGRect(
origin: CGPoint(x: 1.25, y: 2.25),
size: CGSize(width: 10.25, height: 11.75)),
&output)
let expected =
"▿ (1.25, 2.25, 10.25, 11.75)\n" +
" ▿ origin: (1.25, 2.25)\n" +
" - x: 1.25\n" +
" - y: 2.25\n" +
" ▿ size: (10.25, 11.75)\n" +
" - width: 10.25\n" +
" - height: 11.75\n"
expectEqual(expected, output)
}
Reflection.test("Unmanaged/nil") {
var output = ""
var optionalURL: Unmanaged<CFURL>? = nil
dump(optionalURL, &output)
let expected = "- nil\n"
expectEqual(expected, output)
}
Reflection.test("Unmanaged/not-nil") {
var output = ""
var optionalURL: Unmanaged<CFURL>? =
Unmanaged.passRetained(CFURLCreateWithString(nil, "http://llvm.org/", nil))
dump(optionalURL, &output)
let expected =
"▿ Swift.Unmanaged<ObjectiveC.CFURL>\n" +
" ▿ Some: Swift.Unmanaged<ObjectiveC.CFURL>\n" +
" - _value: http://llvm.org/ #0\n"
expectEqual(expected, output)
optionalURL!.release()
}
Reflection.test("TupleMirror/NoLeak") {
if true {
nsObjectCanaryCount = 0
if true {
var tuple = (1, NSObjectCanary())
expectEqual(1, nsObjectCanaryCount)
var output = ""
dump(tuple, &output)
}
expectEqual(0, nsObjectCanaryCount)
}
if true {
nsObjectCanaryCount = 0
if true {
var tuple = (1, NSObjectCanaryStruct())
expectEqual(1, nsObjectCanaryCount)
var output = ""
dump(tuple, &output)
}
expectEqual(0, nsObjectCanaryCount)
}
if true {
swiftObjectCanaryCount = 0
if true {
var tuple = (1, SwiftObjectCanary())
expectEqual(1, swiftObjectCanaryCount)
var output = ""
dump(tuple, &output)
}
expectEqual(0, swiftObjectCanaryCount)
}
if true {
swiftObjectCanaryCount = 0
if true {
var tuple = (1, SwiftObjectCanaryStruct())
expectEqual(1, swiftObjectCanaryCount)
var output = ""
dump(tuple, &output)
}
expectEqual(0, swiftObjectCanaryCount)
}
}
// A struct type and class type whose NominalTypeDescriptor.FieldNames
// data is exactly eight bytes long. FieldNames data of exactly
// 4 or 8 or 16 bytes was once miscompiled on arm64.
struct EightByteFieldNamesStruct {
let abcdef = 42
}
class EightByteFieldNamesClass {
let abcdef = 42
}
Reflection.test("FieldNamesBug") {
if true {
let expected =
"▿ a.EightByteFieldNamesStruct\n" +
" - abcdef: 42\n"
var output = ""
dump(EightByteFieldNamesStruct(), &output)
expectEqual(expected, output)
}
if true {
let expected =
"▿ a.EightByteFieldNamesClass #0\n" +
" - abcdef: 42\n"
var output = ""
dump(EightByteFieldNamesClass(), &output)
expectEqual(expected, output)
}
}
var BitTwiddlingTestSuite = TestSuite("BitTwiddling")
func computeCountLeadingZeroes(var x: Int64) -> Int64 {
var r: Int64 = 64
while x != 0 {
x >>= 1
r--
}
return r
}
BitTwiddlingTestSuite.test("_countLeadingZeros") {
for i in Int64(0)..<1000 {
expectEqual(computeCountLeadingZeroes(i), _countLeadingZeros(i))
}
expectEqual(0, _countLeadingZeros(Int64.min))
}
BitTwiddlingTestSuite.test("_isPowerOf2/Int") {
func asInt(a: Int) -> Int { return a }
expectFalse(_isPowerOf2(asInt(-1025)))
expectFalse(_isPowerOf2(asInt(-1024)))
expectFalse(_isPowerOf2(asInt(-1023)))
expectFalse(_isPowerOf2(asInt(-4)))
expectFalse(_isPowerOf2(asInt(-3)))
expectFalse(_isPowerOf2(asInt(-2)))
expectFalse(_isPowerOf2(asInt(-1)))
expectFalse(_isPowerOf2(asInt(0)))
expectTrue(_isPowerOf2(asInt(1)))
expectTrue(_isPowerOf2(asInt(2)))
expectFalse(_isPowerOf2(asInt(3)))
expectTrue(_isPowerOf2(asInt(1024)))
#if arch(i386) || arch(arm)
// Not applicable to 32-bit architectures.
#elseif arch(x86_64) || arch(arm64)
expectTrue(_isPowerOf2(asInt(0x8000_0000)))
#else
fatalError("implement")
#endif
expectFalse(_isPowerOf2(Int.min))
expectFalse(_isPowerOf2(Int.max))
}
BitTwiddlingTestSuite.test("_isPowerOf2/UInt") {
func asUInt(a: UInt) -> UInt { return a }
expectFalse(_isPowerOf2(asUInt(0)))
expectTrue(_isPowerOf2(asUInt(1)))
expectTrue(_isPowerOf2(asUInt(2)))
expectFalse(_isPowerOf2(asUInt(3)))
expectTrue(_isPowerOf2(asUInt(1024)))
expectTrue(_isPowerOf2(asUInt(0x8000_0000)))
expectFalse(_isPowerOf2(UInt.max))
}
BitTwiddlingTestSuite.test("_floorLog2") {
expectEqual(_floorLog2(1), 0)
expectEqual(_floorLog2(8), 3)
expectEqual(_floorLog2(15), 3)
expectEqual(_floorLog2(Int64.max), 62) // 63 minus 1 for sign bit.
}
runAllTests()