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

[SwiftSyntax] Add #_objectFileFormat compilation conditional #3027

Merged
merged 5 commits into from
Mar 27, 2025
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
37 changes: 37 additions & 0 deletions Sources/SwiftIfConfig/BuildConfiguration.swift
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,17 @@ public enum CanImportVersion {
case underlyingVersion(VersionTuple)
}

enum BuildConfigurationError: Error, CustomStringConvertible {
case experimentalFeature(name: String)

var description: String {
switch self {
case .experimentalFeature(let name):
return "'name' is an experimental feature"
}
}
}

/// Captures information about the build configuration that can be
/// queried in a `#if` expression, including OS, compiler version,
/// enabled language features, and available modules.
Expand Down Expand Up @@ -214,6 +225,22 @@ public protocol BuildConfiguration {
/// pointer authentication scheme.
func isActiveTargetPointerAuthentication(name: String) throws -> Bool

/// Determine whether the given name is the active target object file format (e.g., ELF).
///
/// The target object file format can only be queried by an experimental
/// syntax `_objectFileFormat(<name>)`, e.g.,
///
/// ```swift
/// #if _objectFileFormat(ELF)
/// // Special logic for ELF object file formats
/// #endif
/// ```
/// - Parameters:
/// - name: The name of the object file format.
/// - Returns: Whether the target object file format matches the given name.
@_spi(ExperimentalLanguageFeatures)
func isActiveTargetObjectFileFormat(name: String) throws -> Bool
Copy link
Member

@rintaro rintaro Mar 24, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Could you add a default implementation returning false so that we can land this swift-syntax PR independently before swift repo counterpart?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Added!

Copy link
Member

@rintaro rintaro Mar 25, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The test failure is because the default implementation is not public 🙏

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We should also make this @_spi(ExperimentalLanguageFeatures) so that clients can’t call it as public API and start relying on it.


/// The bit width of a data pointer for the target architecture.
///
/// The target's pointer bit width (which also corresponds to the number of
Expand Down Expand Up @@ -276,3 +303,13 @@ public protocol BuildConfiguration {
/// #endif
var compilerVersion: VersionTuple { get }
}

/// Default implementation of BuildConfiguration, to avoid a revlock with the
/// swift repo, and breaking clients with the new addition to the protocol.
extension BuildConfiguration {
/// FIXME: This should be @_spi(ExperimentalLanguageFeatures) but cannot due
/// to rdar://147943518, https://github.com/swiftlang/swift/issues/80313
public func isActiveTargetObjectFileFormat(name: String) throws -> Bool {
throw BuildConfigurationError.experimentalFeature(name: "_objectFileFormat")
}
}
3 changes: 2 additions & 1 deletion Sources/SwiftIfConfig/IfConfigDiagnostic.swift
Original file line number Diff line number Diff line change
Expand Up @@ -159,7 +159,8 @@ extension IfConfigDiagnostic: DiagnosticMessage {
var severity: SwiftDiagnostics.DiagnosticSeverity {
switch self {
case .compilerVersionSecondComponentNotWildcard, .ignoredTrailingComponents,
.likelySimulatorPlatform, .likelyTargetOS, .endiannessDoesNotMatch, .macabiIsMacCatalyst:
.likelySimulatorPlatform, .likelyTargetOS, .endiannessDoesNotMatch,
.macabiIsMacCatalyst:
return .warning
default: return .error
}
Expand Down
7 changes: 7 additions & 0 deletions Sources/SwiftIfConfig/IfConfigEvaluation.swift
Original file line number Diff line number Diff line change
Expand Up @@ -307,6 +307,9 @@ func evaluateIfConfig(
case .targetEnvironment:
return doSingleIdentifierArgumentCheck(configuration.isActiveTargetEnvironment, role: "environment")

case ._objectFileFormat:
return doSingleIdentifierArgumentCheck(configuration.isActiveTargetObjectFileFormat, role: "object file format")

case ._runtime:
return doSingleIdentifierArgumentCheck(configuration.isActiveTargetRuntime, role: "runtime")

Expand Down Expand Up @@ -818,6 +821,10 @@ private struct CanImportSuppressingBuildConfiguration<Other: BuildConfiguration>
return try other.isActiveTargetPointerAuthentication(name: name)
}

func isActiveTargetObjectFileFormat(name: String) throws -> Bool {
return try other.isActiveTargetObjectFileFormat(name: name)
}

var targetPointerBitWidth: Int { return other.targetPointerBitWidth }

var targetAtomicBitWidths: [Int] { return other.targetAtomicBitWidths }
Expand Down
5 changes: 4 additions & 1 deletion Sources/SwiftIfConfig/IfConfigFunctions.swift
Original file line number Diff line number Diff line change
Expand Up @@ -49,6 +49,9 @@ enum IfConfigFunctions: String {
/// A check for the target bit width of a pointer (e.g., _64)
case _pointerBitWidth

/// A check for the target object file format (e.g., ELF)
case _objectFileFormat

/// A check for the target runtime paired with the Swift runtime (e.g., _ObjC)
/// via `_runtime(<name>)`.
case _runtime
Expand All @@ -69,7 +72,7 @@ enum IfConfigFunctions: String {
return true

case .hasAttribute, .hasFeature, .canImport, .os, .arch, .targetEnvironment,
._hasAtomicBitWidth, ._endian, ._pointerBitWidth, ._runtime, ._ptrauth, .defined:
._hasAtomicBitWidth, ._endian, ._pointerBitWidth, ._objectFileFormat, ._runtime, ._ptrauth, .defined:
return false
}
}
Expand Down
2 changes: 2 additions & 0 deletions Tests/SwiftIfConfigTest/EvaluateTests.swift
Original file line number Diff line number Diff line change
Expand Up @@ -208,6 +208,8 @@ public class EvaluateTests: XCTestCase {
assertIfConfig("_pointerBitWidth(_32)", .inactive)
assertIfConfig("_hasAtomicBitWidth(_64)", .active)
assertIfConfig("_hasAtomicBitWidth(_128)", .inactive)
assertIfConfig("_objectFileFormat(ELF)", .active)
assertIfConfig("_objectFileFormat(MachO)", .inactive)

assertIfConfig(
"_endian(mid)",
Expand Down
4 changes: 4 additions & 0 deletions Tests/SwiftIfConfigTest/TestingBuildConfiguration.swift
Original file line number Diff line number Diff line change
Expand Up @@ -99,6 +99,10 @@ struct TestingBuildConfiguration: BuildConfiguration {
name == "arm64e"
}

func isActiveTargetObjectFileFormat(name: String) throws -> Bool {
name == "ELF"
}

var targetPointerBitWidth: Int { 64 }

var targetAtomicBitWidths: [Int] { [32, 64] }
Expand Down