Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Enable Windows linker discovery #242

Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
46 changes: 25 additions & 21 deletions Sources/SWBCore/Core.swift
Original file line number Diff line number Diff line change
Expand Up @@ -98,6 +98,8 @@ public final class Core: Sendable {

await core.initializeSpecRegistry()

await core.initializePlatformRegistry()

await core.initializeToolchainRegistry()

// Force loading SDKs.
Expand Down Expand Up @@ -315,27 +317,10 @@ public final class Core: Sendable {
@_spi(Testing) public var toolchainPaths: [(Path, strict: Bool)]

/// The platform registry.
public lazy var platformRegistry: PlatformRegistry = {
// FIXME: We should support building the platforms (with symlinks) locally (for `inferiorProductsPath`).

// Search the default location first (unless directed not to), then search any extra locations we've been passed.
var searchPaths: [Path]
let fs = localFS
if let onlySearchAdditionalPlatformPaths = getEnvironmentVariable("XCODE_ONLY_EXTRA_PLATFORM_FOLDERS"), onlySearchAdditionalPlatformPaths.boolValue {
searchPaths = []
}
else {
let platformsDir = self.developerPath.join("Platforms")
searchPaths = [platformsDir]
}
if let additionalPlatformSearchPaths = getEnvironmentVariable("XCODE_EXTRA_PLATFORM_FOLDERS") {
for searchPath in additionalPlatformSearchPaths.split(separator: ":") {
searchPaths.append(Path(searchPath))
}
}
searchPaths += UserDefaults.additionalPlatformSearchPaths
return PlatformRegistry(delegate: self.registryDelegate, searchPaths: searchPaths, hostOperatingSystem: hostOperatingSystem, fs: fs)
}()
let _platformRegistry: UnsafeDelayedInitializationSendableWrapper<PlatformRegistry> = .init()
public var platformRegistry: PlatformRegistry {
_platformRegistry.value
}

@PluginExtensionSystemActor public var loadedPluginPaths: [Path] {
pluginManager.pluginsByIdentifier.values.map(\.path)
Expand Down Expand Up @@ -397,6 +382,25 @@ public final class Core: Sendable {

private var _specRegistry: SpecRegistry?

private func initializePlatformRegistry() async {
var searchPaths: [Path]
let fs = localFS
if let onlySearchAdditionalPlatformPaths = getEnvironmentVariable("XCODE_ONLY_EXTRA_PLATFORM_FOLDERS"), onlySearchAdditionalPlatformPaths.boolValue {
searchPaths = []
} else {
let platformsDir = self.developerPath.join("Platforms")
searchPaths = [platformsDir]
}
if let additionalPlatformSearchPaths = getEnvironmentVariable("XCODE_EXTRA_PLATFORM_FOLDERS") {
for searchPath in additionalPlatformSearchPaths.split(separator: Path.pathEnvironmentSeparator) {
searchPaths.append(Path(searchPath))
}
}
searchPaths += UserDefaults.additionalPlatformSearchPaths
_platformRegistry.initialize(to: await PlatformRegistry(delegate: self.registryDelegate, searchPaths: searchPaths, hostOperatingSystem: hostOperatingSystem, fs: fs))
}


private func initializeToolchainRegistry() async {
self.toolchainRegistry = await ToolchainRegistry(delegate: self.registryDelegate, searchPaths: self.toolchainPaths, fs: localFS, hostOperatingSystem: hostOperatingSystem)
}
Expand Down
4 changes: 2 additions & 2 deletions Sources/SWBCore/Extensions/PlatformInfoExtension.swift
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,7 @@ public protocol PlatformInfoExtension: Sendable {

func additionalKnownTestLibraryPathSuffixes() -> [Path]

func additionalPlatformExecutableSearchPaths(platformName: String, platformPath: Path) -> [Path]
func additionalPlatformExecutableSearchPaths(platformName: String, platformPath: Path, fs: any FSProxy) async -> [Path]

func additionalToolchainExecutableSearchPaths(toolchainIdentifier: String, toolchainPath: Path) -> [Path]

Expand All @@ -56,7 +56,7 @@ extension PlatformInfoExtension {
[]
}

public func additionalPlatformExecutableSearchPaths(platformName: String, platformPath: Path) -> [Path] {
public func additionalPlatformExecutableSearchPaths(platformName: String, platformPath: Path, fs: any FSProxy) async -> [Path] {
[]
}

Expand Down
32 changes: 17 additions & 15 deletions Sources/SWBCore/PlatformRegistry.swift
Original file line number Diff line number Diff line change
Expand Up @@ -324,23 +324,23 @@ public final class PlatformRegistry {
})
}

@_spi(Testing) public init(delegate: any PlatformRegistryDelegate, searchPaths: [Path], hostOperatingSystem: OperatingSystem, fs: any FSProxy) {
@_spi(Testing) public init(delegate: any PlatformRegistryDelegate, searchPaths: [Path], hostOperatingSystem: OperatingSystem, fs: any FSProxy) async {
self.delegate = delegate

for path in searchPaths {
registerPlatformsInDirectory(path, fs)
await registerPlatformsInDirectory(path, fs)
}

do {
if hostOperatingSystem.createFallbackSystemToolchain {
try registerFallbackSystemPlatform(operatingSystem: hostOperatingSystem, fs: fs)
try await registerFallbackSystemPlatform(operatingSystem: hostOperatingSystem, fs: fs)
}
} catch {
delegate.error(error)
}

@preconcurrency @PluginExtensionSystemActor func platformInfoExtensions() -> [any PlatformInfoExtensionPoint.ExtensionProtocol] {
delegate.pluginManager.extensions(of: PlatformInfoExtensionPoint.self)
@preconcurrency @PluginExtensionSystemActor func platformInfoExtensions() async -> [any PlatformInfoExtensionPoint.ExtensionProtocol] {
return await delegate.pluginManager.extensions(of: PlatformInfoExtensionPoint.self)
}

struct Context: PlatformInfoExtensionAdditionalPlatformsContext {
Expand All @@ -349,19 +349,20 @@ public final class PlatformRegistry {
var fs: any FSProxy
}

for platformExtension in platformInfoExtensions() {
for platformExtension in await platformInfoExtensions() {
do {
for (path, data) in try platformExtension.additionalPlatforms(context: Context(hostOperatingSystem: hostOperatingSystem, developerPath: delegate.developerPath, fs: fs)) {
registerPlatform(path, .plDict(data), fs)
await registerPlatform(path, .plDict(data), fs)
}
} catch {
delegate.error(error)

}
}
}

private func registerFallbackSystemPlatform(operatingSystem: OperatingSystem, fs: any FSProxy) throws {
try registerPlatform(Path("/"), .plDict(fallbackSystemPlatformSettings(operatingSystem: operatingSystem)), fs)
private func registerFallbackSystemPlatform(operatingSystem: OperatingSystem, fs: any FSProxy) async throws {
try await registerPlatform(Path("/"), .plDict(fallbackSystemPlatformSettings(operatingSystem: operatingSystem)), fs)
}

private func fallbackSystemPlatformSettings(operatingSystem: OperatingSystem) throws -> [String: PropertyListItem] {
Expand Down Expand Up @@ -428,7 +429,7 @@ public final class PlatformRegistry {
}

/// Register all platforms in the given directory.
private func registerPlatformsInDirectory(_ path: Path, _ fs: any FSProxy) {
private func registerPlatformsInDirectory(_ path: Path, _ fs: any FSProxy) async {
for item in (try? localFS.listdir(path))?.sorted(by: <) ?? [] {
let itemPath = path.join(item)

Expand All @@ -446,14 +447,14 @@ public final class PlatformRegistry {
// Silently skip loading the platform if it does not have an Info.plist at all. (We will still error below if it has an Info.plist which is malformed.)
continue
}
registerPlatform(itemPath, infoPlist, fs)
await registerPlatform(itemPath, infoPlist, fs)
} catch let err {
delegate.error(itemPath, "unable to load platform: 'Info.plist' was malformed: \(err)")
}
}
}

private func registerPlatform(_ path: Path, _ data: PropertyListItem, _ fs: any FSProxy) {
private func registerPlatform(_ path: Path, _ data: PropertyListItem, _ fs: any FSProxy) async {
// The data should always be a dictionary.
guard case .plDict(let items) = data else {
delegate.error(path, "unexpected platform data")
Expand Down Expand Up @@ -617,7 +618,7 @@ public final class PlatformRegistry {
delegate.pluginManager.extensions(of: PlatformInfoExtensionPoint.self)
}

for platformExtension in platformInfoExtensions() {
for platformExtension in await platformInfoExtensions() {
if let value = platformExtension.preferredArchValue(for: name) {
preferredArchValue = value
}
Expand All @@ -631,10 +632,11 @@ public final class PlatformRegistry {
path.join("Developer").join("SDKs")
]

for platformExtension in platformInfoExtensions() {
executableSearchPaths.append(contentsOf: platformExtension.additionalPlatformExecutableSearchPaths(platformName: name, platformPath: path))
for platformExtension in await platformInfoExtensions() {
await executableSearchPaths.append(contentsOf: platformExtension.additionalPlatformExecutableSearchPaths(platformName: name, platformPath: path, fs: localFS))

platformExtension.adjustPlatformSDKSearchPaths(platformName: name, platformPath: path, sdkSearchPaths: &sdkSearchPaths)

}

executableSearchPaths.append(contentsOf: [
Expand Down
4 changes: 4 additions & 0 deletions Sources/SWBCore/Settings/BuiltinMacros.swift
Original file line number Diff line number Diff line change
Expand Up @@ -803,6 +803,8 @@ public final class BuiltinMacros {
public static let LD_RUNPATH_SEARCH_PATHS = BuiltinMacros.declareStringListMacro("LD_RUNPATH_SEARCH_PATHS")
public static let LD_SDK_IMPORTS_FILE = BuiltinMacros.declarePathMacro("LD_SDK_IMPORTS_FILE")
public static let LD_WARN_UNUSED_DYLIBS = BuiltinMacros.declareBooleanMacro("LD_WARN_UNUSED_DYLIBS")
public static let _LD_MULTIARCH = BuiltinMacros.declareBooleanMacro("_LD_MULTIARCH")
public static let _LD_MULTIARCH_PREFIX_MAP = BuiltinMacros.declareStringListMacro("_LD_MULTIARCH_PREFIX_MAP")
public static let LEX = BuiltinMacros.declarePathMacro("LEX")
public static let LEXFLAGS = BuiltinMacros.declareStringListMacro("LEXFLAGS")
public static let LIBRARIAN = BuiltinMacros.declareStringMacro("LIBRARIAN")
Expand Down Expand Up @@ -1858,6 +1860,8 @@ public final class BuiltinMacros {
LD_RUNPATH_SEARCH_PATHS,
LD_SDK_IMPORTS_FILE,
LD_WARN_UNUSED_DYLIBS,
_LD_MULTIARCH,
_LD_MULTIARCH_PREFIX_MAP,
LEGACY_DEVELOPER_DIR,
LEX,
LEXFLAGS,
Expand Down
52 changes: 46 additions & 6 deletions Sources/SWBCore/SpecImplementations/Tools/LinkerTools.swift
Original file line number Diff line number Diff line change
Expand Up @@ -1273,7 +1273,6 @@ public final class LdLinkerSpec : GenericLinkerSpec, SpecIdentifierType, @unchec
}

override public func discoveredCommandLineToolSpecInfo(_ producer: any CommandProducer, _ scope: MacroEvaluationScope, _ delegate: any CoreClientTargetDiagnosticProducingDelegate) async -> (any DiscoveredCommandLineToolSpecInfo)? {
let alternateLinker = scope.evaluate(BuiltinMacros.ALTERNATE_LINKER)
// The ALTERNATE_LINKER is the 'name' of the linker not the executable name, clang will find the linker binary based on name passed via -fuse-ld, but we need to discover
// its properties by executing the actual binary. There is a common filename when the linker is not "ld" across all platforms using "ld.<ALTERNAME_LINKER>(.exe)"
// macOS (Xcode SDK)
Expand Down Expand Up @@ -1309,15 +1308,47 @@ public final class LdLinkerSpec : GenericLinkerSpec, SpecIdentifierType, @unchec
//
// Note: On Linux you cannot invoke the llvm linker by the direct name for determining the version,
// you need to use ld.<ALTERNATE_LINKER>
var linkerPath = Path("ld")
if alternateLinker != "" && alternateLinker != "ld" {
let alternateLinker = scope.evaluate(BuiltinMacros.ALTERNATE_LINKER)
let isLinkerMultiarch = scope.evaluate(BuiltinMacros._LD_MULTIARCH)

var linkerPath = producer.hostOperatingSystem == .windows ? Path("ld.lld") : Path("ld")
if alternateLinker != "" && alternateLinker != "ld" && alternateLinker != "link" {
linkerPath = Path(producer.hostOperatingSystem.imageFormat.executableName(basename: "ld.\(alternateLinker)"))
} else if alternateLinker != "" {
linkerPath = Path(alternateLinker)
}
// If the linker does not support multiple architectures update the path to include a subfolder based on the prefix map
// to find the architecture specific executable.
if !isLinkerMultiarch {
let archMap = scope.evaluate(BuiltinMacros._LD_MULTIARCH_PREFIX_MAP)
let archMappings = archMap.reduce(into: [String: String]()) { mappings, map in
let (arch, prefixDir) = map.split(":")
if !arch.isEmpty && !prefixDir.isEmpty {
return mappings[arch] = prefixDir
}
}
if archMappings.isEmpty {
delegate.error("_LD_MULTIARCH is 'false', but no prefix mappings are present in _LD_MULTIARCH_PREFIX_MAP")
return nil
}
// Linkers that don't support multiple architectures cannot support universal binaries, so ARCHS will
// contain the target architecture and can only be a single value.
guard let arch = scope.evaluate(BuiltinMacros.ARCHS).only else {
delegate.error("_LD_MULTIARCH is 'false', but multiple ARCHS have been given, this is invalid")
return nil
}
if let prefix = archMappings[arch] {
// Add in the target architecture prefix directory to path for search.
linkerPath = Path(prefix).join(linkerPath)
} else {
delegate.error("Could not find prefix mapping for \(arch) in _LD_MULTIARCH_PREFIX_MAP")
return nil
}
}
// Create the cache key. This is just the path to the linker we would invoke if we were invoking the linker directly.
guard let toolPath = producer.executableSearchPaths.lookup(linkerPath) else {
guard let toolPath = producer.executableSearchPaths.findExecutable(operatingSystem: producer.hostOperatingSystem, basename: linkerPath.str) else {
return nil
}

// Create the cache key. This is just the path to the linker we would invoke if we were invoking the linker directly.
return await discoveredLinkerToolsInfo(producer, delegate, at: toolPath)
}
}
Expand Down Expand Up @@ -1638,10 +1669,19 @@ public func discoveredLinkerToolsInfo(_ producer: any CommandProducer, _ delegat
#/GNU gold \(GNU Binutils.*\) (?<version>[\d.]+)/#, // Ubuntu "GNU gold (GNU Binutils for Ubuntu 2.38) 1.16", Debian "GNU gold (GNU Binutils for Debian 2.40) 1.16"
#/GNU gold \(version .*\) (?<version>[\d.]+)/#, // Fedora "GNU gold (version 2.40-14.fc39) 1.16", RHEL "GNU gold (version 2.35.2-54.el9) 1.16", Amazon "GNU gold (version 2.29.1-31.amzn2.0.1) 1.14"
]

if let match = try goLD.compactMap({ try $0.firstMatch(in: String(decoding: executionResult.stdout, as: UTF8.self)) }).first {
return DiscoveredLdLinkerToolSpecInfo(linker: .gold, toolPath: toolPath, toolVersion: try Version(String(match.output.version)), architectures: Set())
}

// link.exe has no option to simply dump the version, running, the program will no arguments or an invalid one will dump a header that contains the version.
let linkExe = [
#/Microsoft \(R\) Incremental Linker Version (?<version>[\d.]+)/#
]
if let match = try linkExe.compactMap({ try $0.firstMatch(in: String(decoding: executionResult.stdout, as: UTF8.self)) }).first {
return DiscoveredLdLinkerToolSpecInfo(linker: .linkExe, toolPath: toolPath, toolVersion: try Version(String(match.output.version)), architectures: Set())
}

struct LDVersionDetails: Decodable {
let version: Version
let architectures: Set<String>
Expand Down
5 changes: 5 additions & 0 deletions Sources/SWBCore/Specs/CoreBuildSystem.xcspec
Original file line number Diff line number Diff line change
Expand Up @@ -1148,6 +1148,11 @@ When `GENERATE_INFOPLIST_FILE` is enabled, sets the value of the [CFBundleIdenti
DefaultValue = NO;
Category = "Linking - Warnings";
},
{
Name = "_LD_MULTIARCH";
Type = Boolean;
DefaultValue = YES;
},
{
Name = "LIBRARY_SEARCH_PATHS";
Type = PathList;
Expand Down
Loading