mirror of
https://github.com/apple/sourcekit-lsp.git
synced 2026-03-02 18:23:24 +01:00
The term *build system* predated our wide-spread adoption of BSP for communicating between SourceKit-LSP to the build system and was never really the correct term anyway – ie. a `JSONCompilationDatabaseBuildSystem` never really sounded right. We now have a correct term for the communication layer between SourceKit-LSP: A build server. Rename most occurrences of *build system* to *build server* to reflect this. There are unfortunately a couple lingering instances of *build system* that we can’t change, most notably: `fallbackBuildSystem` in the config file, the `workspace/waitForBuildSystemUpdates` BSP extension request and the `synchronize-for-build-system-updates` experimental feature.
164 lines
5.6 KiB
Swift
164 lines
5.6 KiB
Swift
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This source file is part of the Swift.org open source project
|
|
//
|
|
// Copyright (c) 2014 - 2024 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
package struct CompilerCommandLineOption {
|
|
/// Return value of `matches(argument:)`.
|
|
package enum Match: Equatable {
|
|
/// The `CompilerCommandLineOption` matched the command line argument. The next element in the command line is a
|
|
/// separate argument and should not be removed.
|
|
case removeOption
|
|
|
|
/// The `CompilerCommandLineOption` matched the command line argument. The next element in the command line is an
|
|
/// argument to this option and should be removed as well.
|
|
case removeOptionAndNextArgument
|
|
|
|
/// The `CompilerCommandLineOption` matched the command line argument. The previous element in the command line is
|
|
/// a prefix to this argument and should be removed if it matches `name`.
|
|
case removeOptionAndPreviousArgument(name: String)
|
|
}
|
|
|
|
package enum DashSpelling {
|
|
case singleDash
|
|
case doubleDash
|
|
}
|
|
|
|
package enum ArgumentStyles {
|
|
/// A command line option where arguments can be passed without a space such as `-MT/file.txt`.
|
|
case noSpace
|
|
/// A command line option where the argument is passed, separated by a space (eg. `--serialize-diagnostics /file.txt`)
|
|
case separatedBySpace
|
|
/// A command line option where the argument is passed after a `=`, eg. `-fbuild-session-file=`.
|
|
case separatedByEqualSign
|
|
}
|
|
|
|
/// The name of the option, without any preceeding `-` or `--`.
|
|
private let name: String
|
|
|
|
/// The name of the argument that prefixes this flag, without any preceeding `-` or `--` (eg. `Xfrontend`/`Xclang`).
|
|
private let frontendName: String?
|
|
|
|
/// Whether the option can be spelled with one or two dashes.
|
|
private let dashSpellings: [DashSpelling]
|
|
|
|
/// The ways that arguments can specified after the option. Empty if the option is a flag that doesn't take any
|
|
/// argument.
|
|
private let argumentStyles: [ArgumentStyles]
|
|
|
|
package static func flag(
|
|
_ name: String,
|
|
frontendName: String? = nil,
|
|
_ dashSpellings: [DashSpelling]
|
|
) -> CompilerCommandLineOption {
|
|
precondition(!dashSpellings.isEmpty)
|
|
return CompilerCommandLineOption(
|
|
name: name,
|
|
frontendName: frontendName,
|
|
dashSpellings: dashSpellings,
|
|
argumentStyles: []
|
|
)
|
|
}
|
|
|
|
package static func option(
|
|
_ name: String,
|
|
_ dashSpellings: [DashSpelling],
|
|
_ argumentStyles: [ArgumentStyles]
|
|
) -> CompilerCommandLineOption {
|
|
precondition(!dashSpellings.isEmpty)
|
|
precondition(!argumentStyles.isEmpty)
|
|
return CompilerCommandLineOption(
|
|
name: name,
|
|
frontendName: nil,
|
|
dashSpellings: dashSpellings,
|
|
argumentStyles: argumentStyles
|
|
)
|
|
}
|
|
|
|
package func matches(argument: String) -> Match? {
|
|
let match = matchesIgnoringFrontend(argument: argument)
|
|
guard let match, let frontendName else {
|
|
return match
|
|
}
|
|
|
|
switch match {
|
|
case .removeOption:
|
|
return .removeOptionAndPreviousArgument(name: frontendName)
|
|
default:
|
|
return match
|
|
}
|
|
}
|
|
|
|
private func matchesIgnoringFrontend(argument: String) -> Match? {
|
|
let argumentName: Substring
|
|
if argument.hasPrefix("--") {
|
|
if dashSpellings.contains(.doubleDash) {
|
|
argumentName = argument.dropFirst(2)
|
|
} else {
|
|
return nil
|
|
}
|
|
} else if argument.hasPrefix("-") {
|
|
if dashSpellings.contains(.singleDash) {
|
|
argumentName = argument.dropFirst(1)
|
|
} else {
|
|
return nil
|
|
}
|
|
} else {
|
|
return nil
|
|
}
|
|
guard argumentName.hasPrefix(self.name) else {
|
|
// Fast path in case the argument doesn't match.
|
|
return nil
|
|
}
|
|
|
|
// Examples:
|
|
// - self.name: "emit-module", argument: "-emit-module", then textAfterArgumentName: ""
|
|
// - self.name: "o", argument: "-o", then textAfterArgumentName: ""
|
|
// - self.name: "o", argument: "-output-file-map", then textAfterArgumentName: "utput-file-map"
|
|
// - self.name: "MT", argument: "-MT/path/to/depfile", then textAfterArgumentName: "/path/to/depfile"
|
|
// - self.name: "fbuild-session-file", argument: "-fbuild-session-file=/path/to/file", then textAfterArgumentName: "=/path/to/file"
|
|
let textAfterArgumentName: Substring = argumentName.dropFirst(self.name.count)
|
|
|
|
if argumentStyles.isEmpty {
|
|
if textAfterArgumentName.isEmpty {
|
|
return .removeOption
|
|
}
|
|
// The command line option is a flag but there is text remaining after the argument name. Thus the flag didn't
|
|
// match. Eg. self.name: "o" and argument: "-output-file-map"
|
|
return nil
|
|
}
|
|
|
|
for argumentStyle in argumentStyles {
|
|
switch argumentStyle {
|
|
case .noSpace where !textAfterArgumentName.isEmpty:
|
|
return .removeOption
|
|
case .separatedBySpace where textAfterArgumentName.isEmpty:
|
|
return .removeOptionAndNextArgument
|
|
case .separatedByEqualSign where textAfterArgumentName.hasPrefix("="):
|
|
return .removeOption
|
|
default:
|
|
break
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
}
|
|
|
|
extension Array<CompilerCommandLineOption> {
|
|
func firstMatch(for argument: String) -> CompilerCommandLineOption.Match? {
|
|
for optionToRemove in self {
|
|
if let match = optionToRemove.matches(argument: argument) {
|
|
return match
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
}
|