mirror of
https://github.com/apple/swift.git
synced 2025-12-14 20:36:38 +01:00
173 lines
4.3 KiB
Swift
173 lines
4.3 KiB
Swift
//===--- ArrayCore.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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
// RUN: %target-run-stdlib-swift | FileCheck %s
|
|
// XFAIL: linux
|
|
import Swift
|
|
|
|
//===--- class Tracked ----------------------------------------------------===//
|
|
// Instead of testing with Int elements, we use this wrapper class
|
|
// that can help us track allocations and find issues with object
|
|
// lifetime inside Array implementations.
|
|
var trackedCount = 0
|
|
var nextTrackedSerialNumber = 0
|
|
|
|
final class Tracked : ForwardIndexType, Printable {
|
|
required init(_ value: Int) {
|
|
++trackedCount
|
|
serialNumber = ++nextTrackedSerialNumber
|
|
self.value = value
|
|
}
|
|
|
|
deinit {
|
|
assert(serialNumber > 0, "double destruction!")
|
|
--trackedCount
|
|
serialNumber = -serialNumber
|
|
}
|
|
|
|
var description: String {
|
|
assert(serialNumber > 0, "dead Tracked!")
|
|
return value.description
|
|
}
|
|
|
|
func successor() -> Self {
|
|
return self.dynamicType(self.value.successor())
|
|
}
|
|
|
|
var value: Int
|
|
var serialNumber: Int
|
|
}
|
|
|
|
func == (x: Tracked, y: Tracked) -> Bool {
|
|
return x.value == y.value
|
|
}
|
|
|
|
//===--- struct MrMcRange -------------------------------------------------===//
|
|
// A wrapper around Range<Tracked> that allows us to detect when it is
|
|
// being treated as a CollectionType rather than merely a SequenceType, which
|
|
// helps us to prove that an optimization is being used. In
|
|
// particular, when constructing a _ContiguousArrayBuffer from a
|
|
// CollectionType, the necessary storage should be pre-allocated.
|
|
struct MrMcRange : CollectionType {
|
|
typealias Base = Range<Int>
|
|
|
|
init(_ base: Base) {
|
|
self.base = base
|
|
}
|
|
|
|
func generate() -> IndexingGenerator<MrMcRange> {
|
|
return IndexingGenerator(self)
|
|
}
|
|
|
|
var startIndex: Int {
|
|
println("using collection API")
|
|
return base.startIndex
|
|
}
|
|
|
|
var endIndex: Int {
|
|
return base.endIndex
|
|
}
|
|
|
|
subscript(i: Int) -> Tracked {
|
|
return Tracked(i)
|
|
}
|
|
|
|
var base: Base
|
|
}
|
|
|
|
//===--- struct MrMcArray<T> ----------------------------------------------===//
|
|
// A faux ArrayType that allows us to detect that, rather than being
|
|
// treated as an arbitrary CollectionType when converting to a
|
|
// _ContiguousArrayBuffer, it is first asked for its underlying
|
|
// _ContiguousArrayBuffer.
|
|
struct MrMcArray<T> : CollectionType, __ArrayType {
|
|
typealias _Buffer = _ContiguousArrayBuffer<T>
|
|
|
|
init(_ buffer: _Buffer) {
|
|
self._buffer = buffer
|
|
}
|
|
|
|
var count: Int {
|
|
return _buffer.count
|
|
}
|
|
|
|
typealias Generator = IndexingGenerator<MrMcArray>
|
|
func generate() -> Generator {
|
|
return IndexingGenerator(self)
|
|
}
|
|
|
|
var startIndex: Int {
|
|
return 0
|
|
}
|
|
|
|
var endIndex: Int {
|
|
return _buffer.count
|
|
}
|
|
|
|
subscript(i: Int) -> T {
|
|
return _buffer[i]
|
|
}
|
|
|
|
func _doCopyToNativeArrayBuffer() -> _ContiguousArrayBuffer<T> {
|
|
return _buffer
|
|
}
|
|
|
|
var _buffer: _Buffer
|
|
}
|
|
|
|
func printSequence<T: SequenceType>(x: T) {
|
|
print("<")
|
|
var prefix = ""
|
|
for a in x {
|
|
print(prefix)
|
|
print(a)
|
|
prefix = " "
|
|
}
|
|
println(">")
|
|
}
|
|
|
|
|
|
|
|
// CHECK: testing...
|
|
println("testing...")
|
|
|
|
func test() {
|
|
//===--- Sequences can be converted -------------------------------------===//
|
|
|
|
let n0 = ((Tracked(10)..<Tracked(27)).generate())~>_copyToNativeArrayBuffer()
|
|
// CHECK-NEXT: <10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26>
|
|
printSequence(n0)
|
|
|
|
//===--- Collections get measured ---------------------------------------===//
|
|
|
|
// CHECK-NEXT: using collection API
|
|
let n1 = MrMcRange(3..<23)~>_copyToNativeArrayBuffer()
|
|
// CHECK: <3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22>
|
|
printSequence(n1)
|
|
|
|
//===--- _ArrayTypes get asked for a NativeBuffer -----------------------===//
|
|
|
|
let a0 = MrMcArray<Tracked>(n1)
|
|
|
|
let cp = _copyToNativeArrayBuffer()
|
|
let n2 = a0~>cp
|
|
|
|
// CHECK-NEXT: true
|
|
println(n1.identity == n2.identity)
|
|
}
|
|
test()
|
|
|
|
// CHECK-NEXT: trackedCount = 0
|
|
println("trackedCount = \(trackedCount)")
|
|
|
|
// CHECK-NEXT: done.
|
|
println("done.")
|