mirror of
https://github.com/apple/swift.git
synced 2025-12-14 20:36:38 +01:00
Move the backtracing code into a new Runtime module. This means renaming the Swift Runtime's CMake target because otherwise there will be a name clash. rdar://124913332
122 lines
3.4 KiB
Swift
122 lines
3.4 KiB
Swift
//===--- CachingMemoryReader.swift ----------------------------*- swift -*-===//
|
|
//
|
|
// This source file is part of the Swift.org open source project
|
|
//
|
|
// Copyright (c) 2023 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// Wraps a MemoryReader in a layer that caches memory pages.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
import Swift
|
|
|
|
// The size of the pages in the page cache (must be a power of 2)
|
|
fileprivate let pageSize = 4096
|
|
fileprivate let pageMask = pageSize - 1
|
|
|
|
// The largest chunk we will try to cache data for
|
|
fileprivate let maxCachedSize = pageSize * 8
|
|
|
|
@_spi(MemoryReaders)
|
|
public class CachingMemoryReader<Reader: MemoryReader>: MemoryReader {
|
|
private var reader: Reader
|
|
private var cache: [Address:UnsafeRawBufferPointer]
|
|
|
|
public init(for reader: Reader) {
|
|
self.reader = reader
|
|
self.cache = [:]
|
|
}
|
|
|
|
deinit {
|
|
for (_, page) in cache {
|
|
page.deallocate()
|
|
}
|
|
}
|
|
|
|
func getPage(at address: Address) throws -> UnsafeRawBufferPointer {
|
|
precondition((address & Address(pageMask)) == 0)
|
|
|
|
if let page = cache[address] {
|
|
return page
|
|
}
|
|
|
|
let page = UnsafeMutableRawBufferPointer.allocate(byteCount: pageSize,
|
|
alignment: pageSize)
|
|
try reader.fetch(from: address, into: page)
|
|
|
|
let result = UnsafeRawBufferPointer(page)
|
|
|
|
cache[address] = result
|
|
|
|
return result
|
|
}
|
|
|
|
public func fetch(from address: Address,
|
|
into buffer: UnsafeMutableRawBufferPointer) throws {
|
|
guard buffer.count <= maxCachedSize else {
|
|
try reader.fetch(from: address, into: buffer)
|
|
return
|
|
}
|
|
|
|
var pageAddress = address & ~Address(pageMask)
|
|
var done = 0
|
|
var offset = Int(address - pageAddress)
|
|
var remaining = buffer.count
|
|
|
|
while remaining > 0 {
|
|
let page = try getPage(at: pageAddress)
|
|
let maxBytes = pageSize - offset
|
|
let chunk = min(remaining, maxBytes)
|
|
|
|
buffer[done..<done+chunk].copyBytes(from: page[offset..<offset+chunk])
|
|
|
|
offset = 0
|
|
done += chunk
|
|
remaining -= chunk
|
|
pageAddress += Address(pageSize)
|
|
}
|
|
}
|
|
}
|
|
|
|
#if os(Linux)
|
|
@_spi(MemoryReaders)
|
|
public typealias MemserverMemoryReader
|
|
= CachingMemoryReader<UncachedMemserverMemoryReader>
|
|
|
|
extension CachingMemoryReader where Reader == UncachedMemserverMemoryReader {
|
|
convenience public init(fd: CInt) {
|
|
self.init(for: UncachedMemserverMemoryReader(fd: fd))
|
|
}
|
|
}
|
|
#endif
|
|
|
|
@_spi(MemoryReaders)
|
|
public typealias RemoteMemoryReader = CachingMemoryReader<UncachedRemoteMemoryReader>
|
|
|
|
extension CachingMemoryReader where Reader == UncachedRemoteMemoryReader {
|
|
#if os(macOS)
|
|
convenience public init(task: Any) {
|
|
self.init(for: UncachedRemoteMemoryReader(task: task))
|
|
}
|
|
#elseif os(Linux)
|
|
convenience public init(pid: Any) {
|
|
self.init(for: UncachedRemoteMemoryReader(pid: pid))
|
|
}
|
|
#endif
|
|
}
|
|
|
|
@_spi(MemoryReaders)
|
|
public typealias LocalMemoryReader = CachingMemoryReader<UncachedLocalMemoryReader>
|
|
|
|
extension CachingMemoryReader where Reader == UncachedLocalMemoryReader {
|
|
convenience public init() {
|
|
self.init(for: UncachedLocalMemoryReader())
|
|
}
|
|
}
|