Files
swift-mirror/stdlib/public/Cxx/CxxRandomAccessCollection.swift
Egor Zhdan 0ab681514b [cxx-interop] Add CxxMutableRandomAccessCollection protocol
This conforms mutable C++ container types, such as `std::vector`, to `MutableCollection` via a new overlay protocol `CxxMutableRandomAccessCollection`.

rdar://134531554
2024-08-28 12:43:42 +01:00

88 lines
2.8 KiB
Swift

//===----------------------------------------------------------------------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2022 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
//
//===----------------------------------------------------------------------===//
public protocol CxxRandomAccessCollection<Element>: CxxSequence, RandomAccessCollection {
override associatedtype Element
override associatedtype RawIterator: UnsafeCxxRandomAccessIterator
where RawIterator.Pointee == Element
override associatedtype Iterator = CxxIterator<Self>
override associatedtype Index = Int
override associatedtype Indices = Range<Int>
override associatedtype SubSequence = Slice<Self>
}
extension CxxRandomAccessCollection {
@inlinable
public var startIndex: Int {
return 0
}
@inlinable
public var endIndex: Int {
return count
}
@inlinable
public var count: Int {
return Int(__endUnsafe() - __beginUnsafe())
}
@inlinable
@inline(__always)
internal func _getRawIterator(at index: Int) -> RawIterator {
var rawIterator = self.__beginUnsafe()
rawIterator += RawIterator.Distance(index)
precondition(self.__endUnsafe() - rawIterator > 0,
"C++ iterator access out of bounds")
return rawIterator
}
/// A C++ implementation of the subscript might be more performant. This
/// overload should only be used if the C++ type does not define `operator[]`.
@inlinable
public subscript(_ index: Int) -> Element {
_read {
yield self._getRawIterator(at: index).pointee
}
}
}
public protocol CxxMutableRandomAccessCollection<Element>:
CxxRandomAccessCollection, MutableCollection {
associatedtype RawMutableIterator: UnsafeCxxMutableRandomAccessIterator
where RawMutableIterator.Pointee == Element
/// Do not implement this function manually in Swift.
mutating func __beginMutatingUnsafe() -> RawMutableIterator
/// Do not implement this function manually in Swift.
mutating func __endMutatingUnsafe() -> RawMutableIterator
}
extension CxxMutableRandomAccessCollection {
/// A C++ implementation of the subscript might be more performant. This
/// overload should only be used if the C++ type does not define `operator[]`.
@inlinable
public subscript(_ index: Int) -> Element {
_read {
yield self._getRawIterator(at: index).pointee
}
_modify {
var rawIterator = self.__beginMutatingUnsafe()
rawIterator += RawMutableIterator.Distance(index)
precondition(self.__endMutatingUnsafe() - rawIterator > 0,
"C++ iterator access out of bounds")
yield &rawIterator.pointee
}
}
}