From 1e0301b312405f93c69a5f005f70e23e48611e9d Mon Sep 17 00:00:00 2001 From: Alex Lorenz Date: Mon, 15 Jul 2024 15:03:26 -0700 Subject: [PATCH 1/3] Revert "Merge pull request #1662 from swiftlang/revert-1560-android" This reverts commit ead33028e3d878a5002cb9d5abc9fd4951d4f81c, reversing changes made to 270bf6c2749bbad6724bd5fda953ee2383389cf8. --- .../GenericUnixToolchain+LinkerSupport.swift | 31 +++++++++---- .../Toolchains/GenericUnixToolchain.swift | 46 +++++++++++++++++++ 2 files changed, 69 insertions(+), 8 deletions(-) diff --git a/Sources/SwiftDriver/Jobs/GenericUnixToolchain+LinkerSupport.swift b/Sources/SwiftDriver/Jobs/GenericUnixToolchain+LinkerSupport.swift index e17d8a88a..d364a1451 100644 --- a/Sources/SwiftDriver/Jobs/GenericUnixToolchain+LinkerSupport.swift +++ b/Sources/SwiftDriver/Jobs/GenericUnixToolchain+LinkerSupport.swift @@ -147,14 +147,24 @@ extension GenericUnixToolchain { } } + if targetInfo.sdkPath != nil { + for libpath in targetInfo.runtimeLibraryImportPaths { + commandLine.appendFlag(.L) + commandLine.appendPath(VirtualPath.lookup(libpath.path)) + } + } + if !isEmbeddedEnabled && !parsedOptions.hasArgument(.nostartfiles) { - let swiftrtPath = VirtualPath.lookup(targetInfo.runtimeResourcePath.path) - .appending( - components: targetTriple.platformName() ?? "", - String(majorArchitectureName(for: targetTriple)), - "swiftrt.o" - ) - commandLine.appendPath(swiftrtPath) + let rsrc: VirtualPath + // Prefer the swiftrt.o runtime file from the SDK if it's specified. + if let sdk = targetInfo.sdkPath.flatMap({ VirtualPath.lookup($0.path) }) { + rsrc = sdk.appending(components: "usr", "lib", "swift") + } else { + rsrc = VirtualPath.lookup(targetInfo.runtimeResourcePath.path) + } + let platform: String = targetTriple.platformName() ?? "" + let architecture: String = majorArchitectureName(for: targetTriple) + commandLine.appendPath(rsrc.appending(components: platform, architecture, "swiftrt.o")) } // If we are linking statically, we need to add all @@ -194,7 +204,12 @@ extension GenericUnixToolchain { commandLine.appendPath(try VirtualPath(path: opt.argument.asSingle)) } - if let path = targetInfo.sdkPath?.path { + if targetTriple.environment == .android { + if let sysroot = try getAndroidNDKSysrootPath() { + commandLine.appendFlag("--sysroot") + commandLine.appendPath(sysroot) + } + } else if let path = targetInfo.sdkPath?.path { commandLine.appendFlag("--sysroot") commandLine.appendPath(VirtualPath.lookup(path)) } diff --git a/Sources/SwiftDriver/Toolchains/GenericUnixToolchain.swift b/Sources/SwiftDriver/Toolchains/GenericUnixToolchain.swift index 7abe9d771..1b2d96808 100644 --- a/Sources/SwiftDriver/Toolchains/GenericUnixToolchain.swift +++ b/Sources/SwiftDriver/Toolchains/GenericUnixToolchain.swift @@ -117,4 +117,50 @@ public final class GenericUnixToolchain: Toolchain { let environment = (targetTriple.environment == .android) ? "-android" : "" return "libclang_rt.\(sanitizer.libraryName)-\(targetTriple.archName)\(environment).a" } + + private func getAndroidNDKHostOSSuffix() -> String? { +#if os(Windows) + "windows" +#elseif os(Linux) + "linux" +#elseif os(macOS) + "darwin" +#else + // The NDK is only available on macOS, linux and windows hosts. + nil +#endif + } + + func getAndroidNDKSysrootPath() throws -> AbsolutePath? { +#if arch(x86_64) + // The NDK's sysroot should be specified in the environment. + guard let ndk = env["ANDROID_NDK_ROOT"], + let osSuffix = getAndroidNDKHostOSSuffix() else { + return nil + } + var sysroot: AbsolutePath = + try AbsolutePath(validating: ndk) + .appending(components: "toolchains", "llvm", "prebuilt") + .appending(component: "\(osSuffix)-x86_64") + .appending(component: "sysroot") + return sysroot +#else + // The NDK is only available on an x86_64 host. + return nil +#endif + } + + public func addPlatformSpecificCommonFrontendOptions( + commandLine: inout [Job.ArgTemplate], + inputs: inout [TypedVirtualPath], + frontendTargetInfo: FrontendTargetInfo, + driver: inout Driver + ) throws { + if driver.targetTriple.environment == .android { + if let sysroot = try getAndroidNDKSysrootPath() { + commandLine.appendFlag("-sysroot") + commandLine.appendFlag(sysroot.pathString) + } + } + } } From 0ebe97529b738b5e01d65f48515217aad05080c8 Mon Sep 17 00:00:00 2001 From: Alex Lorenz Date: Mon, 15 Jul 2024 19:48:16 -0700 Subject: [PATCH 2/3] fix driver unittests --- Tests/SwiftDriverTests/SwiftDriverTests.swift | 24 ++++++++++++++++++- 1 file changed, 23 insertions(+), 1 deletion(-) diff --git a/Tests/SwiftDriverTests/SwiftDriverTests.swift b/Tests/SwiftDriverTests/SwiftDriverTests.swift index cdb9d19c4..10ae8495c 100644 --- a/Tests/SwiftDriverTests/SwiftDriverTests.swift +++ b/Tests/SwiftDriverTests/SwiftDriverTests.swift @@ -7144,10 +7144,14 @@ final class SwiftDriverTests: XCTestCase { func testRelativeResourceDir() throws { do { + // Reset the environment to avoid 'SDKROOT' influencing the + // linux driver paths and taking the priority over the resource directory. + var env = ProcessEnv.vars + env["SDKROOT"] = nil var driver = try Driver(args: ["swiftc", "-target", "x86_64-unknown-linux", "-lto=llvm-thin", "foo.swift", - "-resource-dir", "resource/dir"]) + "-resource-dir", "resource/dir"], env: env) let plannedJobs = try driver.planBuild().removingAutolinkExtractJobs() let compileJob = plannedJobs[0] XCTAssertEqual(compileJob.kind, .compile) @@ -7163,6 +7167,24 @@ final class SwiftDriverTests: XCTestCase { } } + func testSDKDirLinuxPrioritizedOverRelativeResourceDirForLinkingSwiftRT() throws { + do { + let sdkRoot = try testInputsPath.appending(component: "mock-sdk.sdk") + var env = ProcessEnv.vars + env["SDKROOT"] = sdkRoot.pathString + var driver = try Driver(args: ["swiftc", + "-target", "x86_64-unknown-linux", "-lto=llvm-thin", + "foo.swift", + "-resource-dir", "resource/dir"], env: env) + let plannedJobs = try driver.planBuild().removingAutolinkExtractJobs() + let compileJob = plannedJobs[0] + XCTAssertEqual(compileJob.kind, .compile) + let linkJob = plannedJobs[1] + XCTAssertEqual(linkJob.kind, .link) + XCTAssertTrue(linkJob.commandLine.contains(try toPathOption(sdkRoot.pathString + "/usr/lib/swift/linux/x86_64/swiftrt.o"))) + } + } + func testSanitizerArgsForTargets() throws { let targets = ["x86_64-unknown-freebsd", "x86_64-unknown-linux", "x86_64-apple-macosx10.9"] try targets.forEach { From b54d4a2d0a62b51d93956aef6703c78b8fa9debe Mon Sep 17 00:00:00 2001 From: Alex Lorenz Date: Tue, 16 Jul 2024 08:58:47 -0700 Subject: [PATCH 3/3] review fix --- .../SwiftDriver/Jobs/GenericUnixToolchain+LinkerSupport.swift | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Sources/SwiftDriver/Jobs/GenericUnixToolchain+LinkerSupport.swift b/Sources/SwiftDriver/Jobs/GenericUnixToolchain+LinkerSupport.swift index d364a1451..617021560 100644 --- a/Sources/SwiftDriver/Jobs/GenericUnixToolchain+LinkerSupport.swift +++ b/Sources/SwiftDriver/Jobs/GenericUnixToolchain+LinkerSupport.swift @@ -157,8 +157,8 @@ extension GenericUnixToolchain { if !isEmbeddedEnabled && !parsedOptions.hasArgument(.nostartfiles) { let rsrc: VirtualPath // Prefer the swiftrt.o runtime file from the SDK if it's specified. - if let sdk = targetInfo.sdkPath.flatMap({ VirtualPath.lookup($0.path) }) { - rsrc = sdk.appending(components: "usr", "lib", "swift") + if let sdk = targetInfo.sdkPath { + rsrc = VirtualPath.lookup(sdk.path).appending(components: "usr", "lib", "swift") } else { rsrc = VirtualPath.lookup(targetInfo.runtimeResourcePath.path) }