Files
swift-mirror/utils/swift-xcodegen/Sources/SwiftXcodeGen/BuildArgs/ClangBuildArgsProvider.swift
Hamish Knight 56178cd97c [xcodegen] Allow absolute path for output in compile commands
Looks like in newer versions of CMake this can be an absolute path.
2025-08-12 15:04:28 +01:00

102 lines
3.8 KiB
Swift

//===--- ClangBuildArgsProvider.swift -------------------------------------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 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
//
//===----------------------------------------------------------------------===//
import Foundation
struct ClangBuildArgsProvider {
private var args = CommandArgTree()
private var outputs: [RelativePath: AbsolutePath] = [:]
init(for buildDir: RepoBuildDir) throws {
let buildDirPath = buildDir.path
let repoPath = buildDir.repoPath
// TODO: Should we get Clang build args from the build.ninja? We're already
// parsing that to get the Swift targets, seems unfortunate to have 2
// sources of truth.
let fileName = buildDirPath.appending("compile_commands.json")
guard fileName.exists else {
throw XcodeGenError.pathNotFound(fileName)
}
log.debug("[*] Reading Clang build args from '\(fileName)'")
let parsed = try JSONDecoder().decode(
CompileCommands.self, from: try fileName.read()
)
// Gather the candidates for each file to get build arguments for. We may
// have multiple outputs, in which case, pick the first one that exists.
var commandsToAdd: [RelativePath:
(output: AbsolutePath?, args: [Command.Argument])] = [:]
for command in parsed {
guard command.command.executable.knownCommand == .clang,
command.file.exists,
let relFilePath = command.file.realPath.removingPrefix(repoPath)
else {
continue
}
let output = command.output.map { $0.absolute(in: command.directory) }
if let existing = commandsToAdd[relFilePath],
let existingOutput = existing.output,
output == nil || existingOutput.exists || !output!.exists {
continue
}
commandsToAdd[relFilePath] = (output, command.command.args)
}
for (path, (output, commandArgs)) in commandsToAdd {
let commandArgs = commandArgs.filter { arg in
// Only include arguments that have known flags.
// Ignore `-fdiagnostics-color`, we don't want ANSI escape sequences
// in Xcode build logs.
guard let flag = arg.flag, flag != .fDiagnosticsColor else {
return false
}
return true
}
args.insert(commandArgs, for: path)
outputs[path] = output
}
}
/// Retrieve the arguments at a given path, including those in the parent.
func getArgs(for path: RelativePath) -> BuildArgs {
// FIXME: We ought to get the command from the arg tree.
.init(for: .clang, args: args.getArgs(for: path))
}
/// Retrieve the arguments at a given path, excluding those already covered
/// by a parent.
func getUniqueArgs(
for path: RelativePath, parent: RelativePath, infer: Bool = false
) -> BuildArgs {
var fileArgs: [Command.Argument] = []
if hasBuildArgs(for: path) {
fileArgs = args.getUniqueArgs(for: path, parent: parent)
} else if infer {
// If we can infer arguments, walk up to the nearest parent with args.
if let component = path.stackedComponents
.reversed().dropFirst().first(where: hasBuildArgs) {
fileArgs = args.getUniqueArgs(for: component, parent: parent)
}
}
// FIXME: We ought to get the command from the arg tree.
return .init(for: .clang, args: fileArgs)
}
/// Whether the given file has build arguments.
func hasBuildArgs(for path: RelativePath) -> Bool {
!args.getArgs(for: path).isEmpty
}
func isObjectFilePresent(for path: RelativePath) -> Bool {
outputs[path]?.exists == true
}
}