Files
swift-mirror/SwiftCompilerSources/Sources/AST/DiagnosticEngine.swift
Alex Lorenz 220e6faad4 Revert "[cxx-interop][SwiftCompilerSources] Use swift::DiagnosticEngine instead of BridgedDiagnosticEngine"
This reverts commit e9dedf3c27.

The revert is required as foreign reference types are available for SwiftStdlib 5.8 and above, but the Swift compiler
sources back deploy to older stdlibs as well.
2023-04-11 15:28:14 -07:00

123 lines
4.1 KiB
Swift

//===--- DiagnosticEngine.swift -------------------------------------------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 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
//
//===----------------------------------------------------------------------===//
import ASTBridging
import Basic
public typealias DiagID = BridgedDiagID
public protocol DiagnosticArgument {
func _withBridgedDiagnosticArgument(_ fn: (swift.DiagnosticArgument) -> Void)
}
extension String: DiagnosticArgument {
public func _withBridgedDiagnosticArgument(_ fn: (swift.DiagnosticArgument) -> Void) {
_withStringRef { fn(swift.DiagnosticArgument($0)) }
}
}
extension Int: DiagnosticArgument {
public func _withBridgedDiagnosticArgument(_ fn: (swift.DiagnosticArgument) -> Void) {
fn(swift.DiagnosticArgument(Int32(self)))
}
}
public struct DiagnosticFixIt {
public let start: SourceLoc
public let byteLength: Int
public let text: String
public init(start: SourceLoc, byteLength: Int, replacement text: String) {
self.start = start
self.byteLength = byteLength
self.text = text
}
func withBridgedDiagnosticFixIt(_ fn: (swift.DiagnosticInfo.FixIt) -> Void) {
text._withStringRef { bridgedTextRef in
let bridgedDiagnosticFixIt = swift.DiagnosticInfo.FixIt(
swift.CharSourceRange(start.bridged, UInt32(byteLength)),
bridgedTextRef,
ArrayRefOfDiagnosticArgument())
fn(bridgedDiagnosticFixIt)
}
}
}
public struct DiagnosticEngine {
private let bridged: BridgedDiagnosticEngine
public init(bridged: BridgedDiagnosticEngine) {
self.bridged = bridged
}
public init?(bridged: BridgedOptionalDiagnosticEngine) {
guard let object = bridged.object else {
return nil
}
self.bridged = BridgedDiagnosticEngine(object: object)
}
public func diagnose(_ position: SourceLoc?,
_ id: DiagID,
_ args: [DiagnosticArgument],
highlight: CharSourceRange? = nil,
fixIts: [DiagnosticFixIt] = []) {
let bridgedSourceLoc: swift.SourceLoc = position.bridged
let bridgedHighlightRange: swift.CharSourceRange = highlight.bridged
var bridgedArgs: [swift.DiagnosticArgument] = []
var bridgedFixIts: [swift.DiagnosticInfo.FixIt] = []
// Build a higher-order function to wrap every 'withBridgedXXX { ... }'
// calls, so we don't escape anything from the closure. 'bridgedArgs' and
// 'bridgedFixIts' are temporary storage to store bridged values. So they
// should not be used after the closure is executed.
var closure: () -> Void = {
bridgedArgs.withBridgedArrayRef { bridgedArgsRef in
bridgedFixIts.withBridgedArrayRef { bridgedFixItsRef in
DiagnosticEngine_diagnose(bridged, bridgedSourceLoc,
id, bridgedArgsRef,
bridgedHighlightRange, bridgedFixItsRef)
}
}
}
// 'reversed()' because the closure should be wrapped in that order.
for arg in args.reversed() {
closure = { [closure, arg] in
arg._withBridgedDiagnosticArgument { bridgedArg in
bridgedArgs.append(bridgedArg)
closure()
}
}
}
// 'reversed()' because the closure should be wrapped in that order.
for fixIt in fixIts.reversed() {
closure = { [closure, fixIt] in
fixIt.withBridgedDiagnosticFixIt { bridgedFixIt in
bridgedFixIts.append(bridgedFixIt)
closure()
}
}
}
closure()
}
public func diagnose(_ position: SourceLoc?,
_ id: DiagID,
_ args: DiagnosticArgument...,
highlight: CharSourceRange? = nil,
fixIts: DiagnosticFixIt...) {
diagnose(position, id, args, highlight: highlight, fixIts: fixIts)
}
}