diff --git a/doc/BUILD b/doc/BUILD index 5fc30a1a5..1895bfd3c 100644 --- a/doc/BUILD +++ b/doc/BUILD @@ -23,6 +23,7 @@ _DOC_SRCS = { "swift_interop_hint", "swift_library", "swift_library_group", + "mixed_language_library", "swift_module_alias", "swift_package_configuration", "swift_test", @@ -170,6 +171,7 @@ bzl_library( srcs = ["//doc:doc.bzl"], visibility = ["//doc:__pkg__"], deps = [ + "//mixed_language", "//proto", "//swift", ], diff --git a/doc/api.md b/doc/api.md index 6ca045dfa..958fc04bb 100644 --- a/doc/api.md +++ b/doc/api.md @@ -476,9 +476,9 @@ A provider whose type/layout is an implementation detail and should not ## swift_common.create_swift_module
-swift_common.create_swift_module(swiftdoc, swiftmodule, ast_files, defines, indexstore, plugins,
-                                 swiftsourceinfo, swiftinterface, private_swiftinterface,
-                                 const_protocols_to_gather)
+swift_common.create_swift_module(swiftdoc, swiftmodule, ast_files, defines, generated_header,
+                                 indexstore, plugins, swiftsourceinfo, swiftinterface,
+                                 private_swiftinterface, const_protocols_to_gather)
 
Creates a value representing a Swift module use as a Swift dependency. @@ -492,6 +492,7 @@ Creates a value representing a Swift module use as a Swift dependency. | swiftmodule | The `.swiftmodule` file emitted by the compiler for this module. | none | | ast_files | A list of `File`s output from the `DUMP_AST` action. | `[]` | | defines | A list of defines that will be provided as `copts` to targets that depend on this module. If omitted, the empty list will be used. | `[]` | +| generated_header | A `File` representing the Swift generated header. | `None` | | indexstore | A `File` representing the directory that contains the index store data generated by the compiler if the `"swift.index_while_building"` feature is enabled, otherwise this will be `None`. | `None` | | plugins | A list of `SwiftCompilerPluginInfo` providers representing compiler plugins that are required by this module and should be loaded by the compiler when this module is directly depended on. | `[]` | | swiftsourceinfo | The `.swiftsourceinfo` file emitted by the compiler for this module. May be `None` if no source info file was emitted. | `None` | diff --git a/doc/doc.bzl b/doc/doc.bzl index d938e5294..2625ddb4a 100644 --- a/doc/doc.bzl +++ b/doc/doc.bzl @@ -26,6 +26,10 @@ load("@build_bazel_rules_swift//proto:swift_proto_library.bzl", "swift_proto_lib ``` """ +load( + "//mixed_language:mixed_language_library.bzl", + _mixed_language_library = "mixed_language_library", +) load( "//proto:swift_proto_common.bzl", _swift_proto_common = "swift_proto_common", @@ -102,6 +106,7 @@ swift_import = _swift_import swift_interop_hint = _swift_interop_hint swift_library = _swift_library swift_library_group = _swift_library_group +mixed_language_library = _mixed_language_library swift_module_alias = _swift_module_alias swift_package_configuration = _swift_package_configuration swift_test = _swift_test diff --git a/doc/rules.md b/doc/rules.md index 943af7d55..73e355951 100644 --- a/doc/rules.md +++ b/doc/rules.md @@ -22,6 +22,7 @@ On this page: * [swift_interop_hint](#swift_interop_hint) * [swift_library](#swift_library) * [swift_library_group](#swift_library_group) + * [mixed_language_library](#mixed_language_library) * [swift_module_alias](#swift_module_alias) * [swift_package_configuration](#swift_package_configuration) * [swift_test](#swift_test) @@ -802,3 +803,46 @@ swift_library( | plugin | Target to generate a 'fat' binary from. | Label | required | | + + +## mixed_language_library + +
+mixed_language_library(name, clang_copts, clang_defines, clang_srcs, enable_modules, hdrs, includes,
+                       linkopts, module_map, module_name, non_arc_srcs, private_deps, sdk_dylibs,
+                       sdk_frameworks, swift_copts, swift_defines, swift_srcs, swiftc_inputs,
+                       textual_hdrs, umbrella_header, weak_sdk_frameworks, deps, kwargs)
+
+ +Creates a mixed language library from a Clang and Swift library target pair. + +**PARAMETERS** + + +| Name | Description | Default Value | +| :------------- | :------------- | :------------- | +| name | The name of the target. | none | +| clang_copts | The compiler flags for the clang library. These will only be used for the clang library. If you want them to affect the swift library as well, you need to pass them with `-Xcc` in `swift_copts`. | `[]` | +| clang_defines | Extra clang `-D` flags to pass to the compiler. They should be in the form `KEY=VALUE` or simply `KEY` and are passed not only to the compiler for this target (as `clang_copts` are) but also to all dependers of this target. Subject to "Make variable" substitution and Bourne shell tokenization. | `[]` | +| clang_srcs | The list of C, C++, Objective-C, or Objective-C++ sources for the clang library. | none | +| enable_modules | Enables clang module support (via `-fmodules`). Setting this to `True` will allow you to `@import` system headers and other targets: `@import UIKit;` `@import path_to_package_target;`. | `False` | +| hdrs | The list of C, C++, Objective-C, or Objective-C++ header files published by this library to be included by sources in dependent rules. This can't include `umbrella_header`. | `[]` | +| includes | List of `#include`/`#import` search paths to add to this target and all depending targets. | `[]` | +| linkopts | Extra flags to pass to the linker. | `[]` | +| module_map | A `File` representing an existing module map that should be used to represent the module, or `None` (the default) if the module map should be generated based on `hdrs`. If this argument is provided, then `module_name` must also be provided.

Warning: If a module map (whether provided here or not) is able to be found via an include path, it will result in duplicate module definition errors for downstream targets unless sandboxing or remote execution is used. | `None` | +| module_name | The name of the Swift module being built.

If left unspecified, the module name will be computed based on the target's build label, by stripping the leading `//` and replacing `/`, `:`, and other non-identifier characters with underscores. | `None` | +| non_arc_srcs | The list of Objective-C files that are processed to create the library target that DO NOT use ARC. The files in this attribute are treated very similar to those in the `clang_srcs` attribute, but are compiled without ARC enabled. | `[]` | +| private_deps | A list of targets that are implementation-only dependencies of the target being built. Libraries/linker flags from these dependencies will be propagated to dependent for linking, but artifacts/flags required for compilation (such as .swiftmodule files, C headers, and search paths) will not be propagated. | `[]` | +| sdk_dylibs | A list of of SDK `.dylib` libraries to link with. For instance, "libz" or "libarchive". "libc++" is included automatically if the binary has any C++ or Objective-C++ sources in its dependency tree. When linking a binary, all libraries named in that binary's transitive dependency graph are used. | `[]` | +| sdk_frameworks | A list of SDK frameworks to link with (e.g. "AddressBook", "QuartzCore").

When linking a top level Apple binary, all SDK frameworks listed in that binary's transitive dependency graph are linked. | `[]` | +| swift_copts | The compiler flags for the swift library. | `[]` | +| swift_defines | A list of Swift defines to add to the compilation command line.

Note that unlike C-family languages, Swift defines do not have values; they are simply identifiers that are either defined or undefined. So strings in this list should be simple identifiers, not `name=value` pairs.

Each string is prepended with `-D` and added to the command line. Unlike `swift_copts`, these flags are added for the target and every target that depends on it, so use this attribute with caution. It is preferred that you add defines directly to `swift_copts`, only using this feature in the rare case that a library needs to propagate a symbol up to those that depend on it. | `[]` | +| swift_srcs | The sources for the swift library. | none | +| swiftc_inputs | Additional files that are referenced using `$(location ...)` in attributes that support location expansion. | `[]` | +| textual_hdrs | The list of C, C++, Objective-C, or Objective-C++ files that are included as headers by source files in this rule or by users of this library. Unlike `hdrs`, these will not be compiled separately from the sources. | `[]` | +| umbrella_header | A `File` representing an existing umbrella header that should be used in the generated module map or is used in the custom module map, or `None` (the default) if the umbrella header should be generated based on `hdrs`. A symlink to this header is added to an include path such that `#import ` works for this and downstream targets. | `None` | +| weak_sdk_frameworks | A list of SDK frameworks to weakly link with. For instance, "MediaAccessibility". In difference to regularly linked SDK frameworks, symbols from weakly linked frameworks do not cause an error if they are not present at runtime. | `[]` | +| deps | A list of targets that are dependencies of the target being built. | `[]` | +| kwargs | Additional arguments to pass to the underlying clang and swift library targets. | none | + + diff --git a/examples/apple/mixed_language/BUILD b/examples/apple/mixed_language/BUILD new file mode 100644 index 000000000..214b4a378 --- /dev/null +++ b/examples/apple/mixed_language/BUILD @@ -0,0 +1,48 @@ +load("//mixed_language:mixed_language_library.bzl", "mixed_language_library") +load("//swift:swift_library.bzl", "swift_library") +load("//swift:swift_test.bzl", "swift_test") + +mixed_language_library( + name = "MixedAnswer", + hdrs = ["MixedAnswer.h"], + clang_srcs = [ + "MixedAnswer.m", + "MixedAnswerPrivate.m", + "MixedAnswerPrivate.h", + ], + enable_modules = True, + module_name = "MixedAnswer", + swift_srcs = [ + "MixedAnswer.swift", + ], + target_compatible_with = ["@platforms//os:macos"], +) + +swift_library( + name = "SwiftLibDependingOnMixedLib", + srcs = [ + "SwiftLibDependingOnMixedLib.swift", + ], + module_name = "SwiftLibDependingOnMixedLib", + deps = [":MixedAnswer"], +) + +mixed_language_library( + name = "MixedTestsLib", + testonly = True, + clang_srcs = [ + "MixedTests.m", + ], + swift_srcs = [ + "MixedTests.swift", + ], + target_compatible_with = ["@platforms//os:macos"], + deps = [ + ":SwiftLibDependingOnMixedLib", + ], +) + +swift_test( + name = "MixedTests", + deps = [":MixedTestsLib"], +) diff --git a/examples/apple/mixed_language/MixedAnswer.h b/examples/apple/mixed_language/MixedAnswer.h new file mode 100644 index 000000000..24ce6ac6f --- /dev/null +++ b/examples/apple/mixed_language/MixedAnswer.h @@ -0,0 +1,7 @@ +@import Foundation; + +@interface MixedAnswerObjc : NSObject + ++ (NSString *)mixedAnswerObjc; + +@end diff --git a/examples/apple/mixed_language/MixedAnswer.m b/examples/apple/mixed_language/MixedAnswer.m new file mode 100644 index 000000000..c67832fbf --- /dev/null +++ b/examples/apple/mixed_language/MixedAnswer.m @@ -0,0 +1,11 @@ +#import "examples/apple/mixed_language/MixedAnswer.h" +#import "examples/apple/mixed_language/MixedAnswerPrivate.h" +#import "examples/apple/mixed_language/MixedAnswer-Swift.h" + +@implementation MixedAnswerObjc + ++ (NSString *)mixedAnswerObjc { + return [NSString stringWithFormat:@"%@_%@ and %@", @"mixedAnswerObjc", [MixedAnswerSwift swiftToObjcMixedAnswer], [MixedAnswerPrivateObjc mixedAnswerPrivateObjc]]; +} + +@end diff --git a/examples/apple/mixed_language/MixedAnswer.swift b/examples/apple/mixed_language/MixedAnswer.swift new file mode 100644 index 000000000..5e670c5a1 --- /dev/null +++ b/examples/apple/mixed_language/MixedAnswer.swift @@ -0,0 +1,16 @@ +import Foundation + +@objc +public class MixedAnswerSwift: NSObject { + + public + static func swiftMixedAnswer() -> String { + "\(MixedAnswerObjc.mixedAnswerObjc() ?? "invalid")_swiftMixedAnswer" + } + + @objc + public + static func swiftToObjcMixedAnswer() -> String { + "swiftToObjcMixedAnswer" + } +} diff --git a/examples/apple/mixed_language/MixedAnswerPrivate.h b/examples/apple/mixed_language/MixedAnswerPrivate.h new file mode 100644 index 000000000..e95268d59 --- /dev/null +++ b/examples/apple/mixed_language/MixedAnswerPrivate.h @@ -0,0 +1,7 @@ +@import Foundation; + +@interface MixedAnswerPrivateObjc : NSObject + ++ (NSString *)mixedAnswerPrivateObjc; + +@end diff --git a/examples/apple/mixed_language/MixedAnswerPrivate.m b/examples/apple/mixed_language/MixedAnswerPrivate.m new file mode 100644 index 000000000..f31d6fbec --- /dev/null +++ b/examples/apple/mixed_language/MixedAnswerPrivate.m @@ -0,0 +1,10 @@ +#import "examples/apple/mixed_language/MixedAnswerPrivate.h" +#import "examples/apple/mixed_language/MixedAnswer-Swift.h" + +@implementation MixedAnswerPrivateObjc + ++ (NSString *)mixedAnswerPrivateObjc { + return [NSString stringWithFormat:@"%@_%@", @"mixedAnswerPrivateObjc", [MixedAnswerSwift swiftToObjcMixedAnswer]]; +} + +@end diff --git a/examples/apple/mixed_language/MixedTests.m b/examples/apple/mixed_language/MixedTests.m new file mode 100644 index 000000000..cc46e7795 --- /dev/null +++ b/examples/apple/mixed_language/MixedTests.m @@ -0,0 +1,27 @@ +// Copyright 2017 The Bazel Authors. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +#import +#import + +@interface MixedTests : XCTestCase +@end + +@implementation MixedTests + +- (void)testAlwaysPass { + XCTAssertTrue(YES); +} + +@end diff --git a/examples/apple/mixed_language/MixedTests.swift b/examples/apple/mixed_language/MixedTests.swift new file mode 100644 index 000000000..730fb0ff1 --- /dev/null +++ b/examples/apple/mixed_language/MixedTests.swift @@ -0,0 +1,29 @@ +// Copyright 2018 The Bazel Authors. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +import SwiftLibDependingOnMixedLib +import XCTest + +class MixedTests: XCTestCase { + func testValuesSwift() { + XCTAssertEqual( + SwiftLibDependingOnMixedLib.callSwiftMixedAnswer(), + "mixedAnswerObjc_swiftToObjcMixedAnswer and mixedAnswerPrivateObjc_swiftToObjcMixedAnswer_swiftMixedAnswer" + ) + XCTAssertEqual( + SwiftLibDependingOnMixedLib.callObjcMixedAnswer(), + "mixedAnswerObjc_swiftToObjcMixedAnswer and mixedAnswerPrivateObjc_swiftToObjcMixedAnswer" + ) + } +} diff --git a/examples/apple/mixed_language/SwiftLibDependingOnMixedLib.swift b/examples/apple/mixed_language/SwiftLibDependingOnMixedLib.swift new file mode 100644 index 000000000..4c9245b47 --- /dev/null +++ b/examples/apple/mixed_language/SwiftLibDependingOnMixedLib.swift @@ -0,0 +1,11 @@ +import MixedAnswer + +public class SwiftLibDependingOnMixedLib { + public static func callSwiftMixedAnswer() -> String { + MixedAnswerSwift.swiftMixedAnswer() + } + + public static func callObjcMixedAnswer() -> String? { + MixedAnswerObjc.mixedAnswerObjc() + } +} diff --git a/mixed_language/BUILD b/mixed_language/BUILD new file mode 100644 index 000000000..25995fe8a --- /dev/null +++ b/mixed_language/BUILD @@ -0,0 +1,26 @@ +load("@bazel_skylib//:bzl_library.bzl", "bzl_library") + +package(default_visibility = ["//visibility:public"]) + +licenses(["notice"]) + +bzl_library( + name = "mixed_language", + deps = [ + ":mixed_language_library", + ], +) + +bzl_library( + name = "mixed_language_library", + srcs = ["mixed_language_library.bzl"], + deps = [ + "//mixed_language/internal:library", + "//mixed_language/internal:module_map", + "//mixed_language/internal:umbrella_header", + "//swift:swift_interop_hint", + "//swift:swift_library", + "//swift/internal:compiling", + "@bazel_skylib//lib:paths", + ], +) diff --git a/mixed_language/internal/BUILD b/mixed_language/internal/BUILD new file mode 100644 index 000000000..26f370a66 --- /dev/null +++ b/mixed_language/internal/BUILD @@ -0,0 +1,40 @@ +load("@bazel_skylib//:bzl_library.bzl", "bzl_library") + +licenses(["notice"]) + +bzl_library( + name = "library", + srcs = ["library.bzl"], + visibility = ["//mixed_language:__subpackages__"], + deps = [ + "//swift:providers", + "//swift:swift_clang_module_aspect", + "//swift/internal:attrs", + "//swift/internal:compiling", + "//swift/internal:feature_names", + "//swift/internal:features", + "//swift/internal:providers", + "//swift/internal:toolchain_utils", + "//swift/internal:utils", + "@bazel_skylib//lib:dicts", + "@bazel_skylib//lib:paths", + ], +) + +bzl_library( + name = "module_map", + srcs = ["module_map.bzl"], + visibility = ["//mixed_language:__subpackages__"], + deps = [ + "//swift/internal:module_maps", + ], +) + +bzl_library( + name = "umbrella_header", + srcs = ["umbrella_header.bzl"], + visibility = ["//mixed_language:__subpackages__"], + deps = [ + "@bazel_skylib//lib:paths", + ], +) diff --git a/mixed_language/internal/library.bzl b/mixed_language/internal/library.bzl new file mode 100644 index 000000000..f65958588 --- /dev/null +++ b/mixed_language/internal/library.bzl @@ -0,0 +1,262 @@ +# Copyright 2024 The Bazel Authors. All rights reserved. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +"""Implementation of the `mixed_language_library` rule.""" + +load("@bazel_skylib//lib:dicts.bzl", "dicts") +load("@bazel_skylib//lib:paths.bzl", "paths") +load("//swift:providers.bzl", "SwiftInfo") +load("//swift:swift_clang_module_aspect.bzl", "swift_clang_module_aspect") + +# buildifier: disable=bzl-visibility +load("//swift/internal:attrs.bzl", "swift_deps_attr", "swift_toolchain_attrs") + +# buildifier: disable=bzl-visibility +load( + "//swift/internal:feature_names.bzl", + "SWIFT_FEATURE_PROPAGATE_GENERATED_MODULE_MAP", +) + +# buildifier: disable=bzl-visibility +load( + "//swift/internal:features.bzl", + "configure_features", + "is_feature_enabled", +) + +# buildifier: disable=bzl-visibility +load( + "//swift/internal:providers.bzl", + "create_clang_module", + "create_module", + "create_swift_info", +) + +# buildifier: disable=bzl-visibility +load( + "//swift/internal:toolchain_utils.bzl", + "get_swift_toolchain", + "use_swift_toolchain", +) + +# buildifier: disable=bzl-visibility +load("//swift/internal:utils.bzl", "get_providers") + +def _write_extended_module_map( + *, + actions, + module_map_extender, + module_name, + original_module_map, + output_module_map, + swift_generated_header): + args = actions.args() + args.add(output_module_map) + args.add(original_module_map) + args.add(module_name) + args.add(swift_generated_header.basename) + + actions.run( + arguments = [args], + executable = module_map_extender, + inputs = [original_module_map], + mnemonic = "ExtendMixedLanguageModuleMap", + outputs = [output_module_map], + ) + +def _mixed_language_library_impl(ctx): + actions = ctx.actions + clang_target = ctx.attr.clang_target + module_name = ctx.attr.module_name + name = ctx.attr.name + + swift_target = ctx.attr.swift_target + swift_info = swift_target[SwiftInfo] + + feature_configuration = configure_features( + ctx = ctx, + requested_features = ctx.features, + swift_toolchain = get_swift_toolchain(ctx), + unsupported_features = ctx.disabled_features, + ) + + umbrella_header_symlink = actions.declare_file( + "{name}/{module_name}/{module_name}.h".format( + module_name = module_name, + name = name, + ), + ) + ctx.actions.symlink( + output = umbrella_header_symlink, + target_file = ctx.file.umbrella_header, + ) + + swift_module = swift_info.direct_modules[0] + swift_generated_header = swift_module.swift.generated_header + if not swift_generated_header: + fail("{} must have 'generate_header = True'".format(swift_target.label)) + + swift_generated_header_symlink = actions.declare_file( + "{name}/{module_name}/{basename}".format( + module_name = module_name, + name = name, + basename = swift_generated_header.basename, + ), + ) + ctx.actions.symlink( + output = swift_generated_header_symlink, + target_file = swift_generated_header, + ) + + propagate_module_map = is_feature_enabled( + feature_configuration = feature_configuration, + feature_name = SWIFT_FEATURE_PROPAGATE_GENERATED_MODULE_MAP, + ) + + if propagate_module_map: + module_map_basename = "module.modulemap" + else: + module_map_basename = name + ".modulemap" + + extended_module_map = actions.declare_file( + "{name}/{module_name}/{basename}".format( + basename = module_map_basename, + module_name = module_name, + name = name, + ), + ) + _write_extended_module_map( + actions = actions, + module_map_extender = ctx.executable._module_map_extender, + module_name = module_name, + original_module_map = ctx.file.module_map, + output_module_map = extended_module_map, + swift_generated_header = swift_generated_header_symlink, + ) + + outputs = [ + umbrella_header_symlink, + extended_module_map, + swift_generated_header_symlink, + ] + + compilation_context = cc_common.create_compilation_context( + headers = depset(outputs), + direct_public_headers = outputs, + includes = depset( + # This allows `#import ` and + # `#import ` to work + [paths.dirname(umbrella_header_symlink.dirname)], + ), + ) + + cc_info = cc_common.merge_cc_infos( + direct_cc_infos = [ + CcInfo(compilation_context = compilation_context), + ], + cc_infos = [swift_target[CcInfo], clang_target[CcInfo]], + ) + swift_info = create_swift_info( + modules = [ + create_module( + name = module_name, + clang = create_clang_module( + compilation_context = cc_info.compilation_context, + module_map = extended_module_map, + ), + swift = swift_module.swift, + ), + ], + # Collect transitive modules, without including `swift_target` (which is + # covered with the `create_module` above) + swift_infos = get_providers(ctx.attr.deps, SwiftInfo), + ) + + return [ + DefaultInfo(files = depset( + outputs, + transitive = [ + swift_target[DefaultInfo].files, + clang_target[DefaultInfo].files, + ], + )), + cc_info, + coverage_common.instrumented_files_info( + ctx, + dependency_attributes = ["deps"], + ), + swift_info, + # Propagate an `apple_common.Objc` provider with linking info about the + # library so that linking with Apple Starlark APIs/rules works + # correctly. + # TODO(b/171413861): This can be removed when the Obj-C rules are + # migrated to use `CcLinkingContext`. + apple_common.new_objc_provider( + providers = get_providers( + [swift_target, clang_target], + apple_common.Objc, + ), + ), + ] + +mixed_language_library = rule( + attrs = dicts.add( + swift_toolchain_attrs(), + { + "clang_target": attr.label( + doc = """ +The non-Swift portion of the mixed language module. +""", + mandatory = True, + providers = [CcInfo], + ), + "deps": swift_deps_attr( + aspects = [swift_clang_module_aspect], + doc = "Dependencies of the target being built.", + ), + "swift_target": attr.label( + doc = """ +The Swift portion of the mixed language module. +""", + mandatory = True, + providers = [SwiftInfo], + ), + "umbrella_header": attr.label( + allow_single_file = True, + doc = "The umbrella header for the module.", + mandatory = True, + ), + "module_name": attr.string( + doc = "The name of the module.", + mandatory = True, + ), + "module_map": attr.label( + allow_single_file = True, + doc = "The module map for the module.", + mandatory = True, + ), + "_module_map_extender": attr.label( + cfg = "exec", + executable = True, + default = "//tools/mixed_language_module_map_extender", + ), + }, + ), + doc = """\ +Assembles a mixed language library from a clang and swift library target pair. +""", + fragments = ["cpp"], + implementation = _mixed_language_library_impl, + toolchains = use_swift_toolchain(), +) diff --git a/mixed_language/internal/module_map.bzl b/mixed_language/internal/module_map.bzl new file mode 100644 index 000000000..8ed59f3c9 --- /dev/null +++ b/mixed_language/internal/module_map.bzl @@ -0,0 +1,75 @@ +# Copyright 2024 The Bazel Authors. All rights reserved. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +"""Implementation of the `mixed_language_module_map` macro.""" + +# buildifier: disable=bzl-visibility +load("//swift/internal:module_maps.bzl", "write_module_map") + +def _mixed_language_internal_module_map_impl(ctx): + actions = ctx.actions + + module_map = actions.declare_file(ctx.attr.module_map_name + ".modulemap") + + # TODO: Set `dependent_module_names` + write_module_map( + actions = actions, + exported_module_ids = ["*"], + module_map_file = module_map, + module_name = ctx.attr.module_name, + umbrella_header = ctx.file.umbrella_header, + public_headers = ctx.files.hdrs, + public_textual_headers = ctx.files.textual_hdrs, + ) + + return [DefaultInfo(files = depset([module_map]))] + +mixed_language_internal_module_map = rule( + attrs = { + "hdrs": attr.label_list( + allow_files = True, + doc = """ +The list of C, C++, Objective-C, and Objective-C++ header files published by +this library to be included by sources in dependent rules. +""", + ), + "module_map_name": attr.string( + doc = "The name that will be used for the `.modulemap` file.", + mandatory = True, + ), + "module_name": attr.string( + doc = "The name of the module.", + mandatory = True, + ), + "textual_hdrs": attr.label_list( + allow_files = True, + doc = """ +The list of C, C++, Objective-C, and Objective-C++ files that are included as +headers by source files in this rule or by users of this library. Unlike `hdrs`, +these will not be compiled separately from the sources. +""", + ), + "umbrella_header": attr.label( + allow_single_file = True, + doc = """ +An umbrella header. If set, `hdrs` and `textual_hdrs` are ignored. +""", + ), + }, + doc = """\ +Creates the module map file that is used internally by the Clang and Swift +library targets in the `mixed_language_library` macro. +""", + implementation = _mixed_language_internal_module_map_impl, +) diff --git a/mixed_language/internal/umbrella_header.bzl b/mixed_language/internal/umbrella_header.bzl new file mode 100644 index 000000000..5ac070c85 --- /dev/null +++ b/mixed_language/internal/umbrella_header.bzl @@ -0,0 +1,67 @@ +# Copyright 2024 The Bazel Authors. All rights reserved. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +"""Implementation of the `mixed_language_umbrella_header` rule.""" + +load("@bazel_skylib//lib:paths.bzl", "paths") + +def _mixed_language_umbrella_header_impl(ctx): + actions = ctx.actions + + umbrella_header = actions.declare_file( + "{name}/{module_name}/{module_name}.h".format( + module_name = ctx.attr.module_name, + name = ctx.attr.name, + ), + ) + + content = actions.args() + content.set_param_file_format("multiline") + content.add_all(ctx.files.hdrs, format_each = '#import "%s"') + actions.write( + content = content, + output = umbrella_header, + ) + outputs = [umbrella_header] + outputs_depset = depset(outputs) + + compilation_context = cc_common.create_compilation_context( + headers = outputs_depset, + direct_public_headers = outputs, + # This allows `#import ` to work + includes = depset([paths.dirname(umbrella_header.dirname)]), + ) + + return [ + DefaultInfo(files = outputs_depset), + CcInfo(compilation_context = compilation_context), + ] + +mixed_language_umbrella_header = rule( + attrs = { + "hdrs": attr.label_list( + allow_files = True, + doc = """ +The list of C, C++, Objective-C, and Objective-C++ header files published by +this library to be included by sources in dependent rules. +""", + ), + "module_name": attr.string( + doc = "The name of the module.", + mandatory = True, + ), + }, + doc = "Creates an umbrella header for a mixed language target.", + implementation = _mixed_language_umbrella_header_impl, +) diff --git a/mixed_language/mixed_language_library.bzl b/mixed_language/mixed_language_library.bzl new file mode 100644 index 000000000..b92103601 --- /dev/null +++ b/mixed_language/mixed_language_library.bzl @@ -0,0 +1,375 @@ +# Copyright 2024 The Bazel Authors. All rights reserved. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +"""Implementation of the `mixed_language_library` macro.""" + +load( + "//mixed_language/internal:library.bzl", + _mixed_language_library = "mixed_language_library", +) +load( + "//mixed_language/internal:module_map.bzl", + "mixed_language_internal_module_map", +) +load( + "//mixed_language/internal:umbrella_header.bzl", + "mixed_language_umbrella_header", +) +load("//swift:swift_interop_hint.bzl", "swift_interop_hint") +load("//swift:swift_library.bzl", "swift_library") + +# buildifier: disable=bzl-visibility +load("//swift/internal:compiling.bzl", "derive_module_name") + +# `mixed_language_library` + +def mixed_language_library( + *, + name, + clang_copts = [], + clang_defines = [], + clang_srcs, + enable_modules = False, + hdrs = [], + includes = [], + linkopts = [], + module_map = None, + module_name = None, + non_arc_srcs = [], + private_deps = [], + sdk_dylibs = [], + sdk_frameworks = [], + swift_copts = [], + swift_defines = [], + swift_srcs, + swiftc_inputs = [], + textual_hdrs = [], + umbrella_header = None, + weak_sdk_frameworks = [], + deps = [], + **kwargs): + """Creates a mixed language library from a Clang and Swift library target \ + pair. + + Args: + name: The name of the target. + clang_copts: The compiler flags for the clang library. These will only + be used for the clang library. If you want them to affect the swift + library as well, you need to pass them with `-Xcc` in `swift_copts`. + clang_defines: Extra clang `-D` flags to pass to the compiler. They + should be in the form `KEY=VALUE` or simply `KEY` and are passed not + only to the compiler for this target (as `clang_copts` are) but also + to all dependers of this target. Subject to "Make variable" + substitution and Bourne shell tokenization. + clang_srcs: The list of C, C++, Objective-C, or Objective-C++ sources + for the clang library. + enable_modules: Enables clang module support (via `-fmodules`). Setting + this to `True` will allow you to `@import` system headers and other + targets: `@import UIKit;` `@import path_to_package_target;`. + hdrs: The list of C, C++, Objective-C, or Objective-C++ header files + published by this library to be included by sources in dependent + rules. This can't include `umbrella_header`. + includes: List of `#include`/`#import` search paths to add to this + target and all depending targets. + linkopts: Extra flags to pass to the linker. + module_map: A `File` representing an existing module map that should be + used to represent the module, or `None` (the default) if the module + map should be generated based on `hdrs`. If this argument is + provided, then `module_name` must also be provided. + + Warning: If a module map (whether provided here or not) is able to + be found via an include path, it will result in duplicate module + definition errors for downstream targets unless sandboxing or remote + execution is used. + module_name: The name of the Swift module being built. + + If left unspecified, the module name will be computed based on the + target's build label, by stripping the leading `//` and replacing + `/`, `:`, and other non-identifier characters with underscores. + non_arc_srcs: The list of Objective-C files that are processed to create + the library target that DO NOT use ARC. The files in this attribute + are treated very similar to those in the `clang_srcs` attribute, but + are compiled without ARC enabled. + private_deps: A list of targets that are implementation-only + dependencies of the target being built. Libraries/linker flags from + these dependencies will be propagated to dependent for linking, but + artifacts/flags required for compilation (such as .swiftmodule + files, C headers, and search paths) will not be propagated. + sdk_dylibs: A list of of SDK `.dylib` libraries to link with. For + instance, "libz" or "libarchive". "libc++" is included automatically + if the binary has any C++ or Objective-C++ sources in its dependency + tree. When linking a binary, all libraries named in that binary's + transitive dependency graph are used. + sdk_frameworks: A list of SDK frameworks to link with (e.g. + "AddressBook", "QuartzCore"). + + When linking a top level Apple binary, all SDK frameworks listed in + that binary's transitive dependency graph are linked. + swift_copts: The compiler flags for the swift library. + swift_defines: A list of Swift defines to add to the compilation command + line. + + Note that unlike C-family languages, Swift defines do not have + values; they are simply identifiers that are either defined or + undefined. So strings in this list should be simple identifiers, + not `name=value` pairs. + + Each string is prepended with `-D` and added to the command line. + Unlike `swift_copts`, these flags are added for the target and + every target that depends on it, so use this attribute with caution. + It is preferred that you add defines directly to `swift_copts`, only + using this feature in the rare case that a library needs to + propagate a symbol up to those that depend on it. + swift_srcs: The sources for the swift library. + swiftc_inputs: Additional files that are referenced using + `$(location ...)` in attributes that support location expansion. + textual_hdrs: The list of C, C++, Objective-C, or Objective-C++ files + that are included as headers by source files in this rule or by + users of this library. Unlike `hdrs`, these will not be compiled + separately from the sources. + umbrella_header: A `File` representing an existing umbrella header that + should be used in the generated module map or is used in the custom + module map, or `None` (the default) if the umbrella header should be + generated based on `hdrs`. A symlink to this header is added to an + include path such that `#import ` works for + this and downstream targets. + weak_sdk_frameworks: A list of SDK frameworks to weakly link with. For + instance, "MediaAccessibility". In difference to regularly linked + SDK frameworks, symbols from weakly linked frameworks do not cause + an error if they are not present at runtime. + deps: A list of targets that are dependencies of the target being built. + **kwargs: Additional arguments to pass to the underlying clang and swift + library targets. + """ + aspect_hints = kwargs.pop("aspect_hints", []) + features = kwargs.pop("features", []) + tags = kwargs.pop("tags", []) + testonly = kwargs.pop("testonly", None) + visibility = kwargs.pop("visibility", None) + + internal_tags = ( + ((["manual"] + tags) if "manual" not in tags else tags) + + # Allows for easy query filtering of the internal targets + ["mixed_language_library_internal"] + ) + + if "generates_header" in kwargs: + fail( + """\ +'generates_header' is an invalid attribute for 'mixed_language_library'.\ +""", + attr = "generates_header", + ) + if "copts" in kwargs: + fail( + """\ +Use 'clang_copts' and/or 'swift_copts' to set copts with \ +'mixed_language_library'.\ +""", + attr = "copts", + ) + if "defines" in kwargs: + fail( + """\ +Use 'clang_defines' and/or 'swift_defines' to set defines for \ +'mixed_language_library'.\ +""", + attr = "defines", + ) + if "implementation_deps" in kwargs: + fail( + """\ +'implementation_deps' is an invalid attribute for 'mixed_language_library'. \ +Use 'private_deps' to set implementation deps.\ +""", + attr = "implementation_deps", + ) + + if not clang_srcs and not hdrs: + fail( + """\ +'mixed_language_library' requires either 'clang_srcs' and/or 'hdrs' to be \ +non-empty. If this is not a mixed language Swift library, use `swift_library` \ +instead.\ +""", + attr = "clang_srcs", + ) + if not swift_srcs: + fail( + """\ +'mixed_language_library' requires 'swift_srcs' to be non-empty. If this is not \ +a mixed language Swift library, use a clang only library rule like \ +`cc_library` or `objc_library` instead.\ +""", + attr = "swift_srcs", + ) + + if not module_name: + module_name = derive_module_name(native.package_name(), name) + + if not module_map: + internal_modulemap_name = name + "_modulemap" + mixed_language_internal_module_map( + name = internal_modulemap_name, + hdrs = hdrs, + module_map_name = name, + module_name = module_name, + tags = internal_tags, + textual_hdrs = textual_hdrs, + # We can't use a generated umbrella header here, it results in + # duplicated headers being found. There is probvably a way to make + # that work, but I don't know of it right now. + umbrella_header = umbrella_header, + ) + module_map = ":" + internal_modulemap_name + elif not module_name: + fail( + "If 'module_map' is provided, 'module_name' must also be provided.", + attr = "module_name", + ) + + if not umbrella_header: + internal_umbrella_header_name = name + "_umbrella_header" + mixed_language_umbrella_header( + name = internal_umbrella_header_name, + hdrs = hdrs, + module_name = module_name, + tags = internal_tags, + ) + umbrella_header = ":" + internal_umbrella_header_name + + # We need both targets to depend on this to get the includes path that + # lets `#import ` work, since minimally it + # might be in the `-Swift.h` header. + private_deps = private_deps + [umbrella_header] + + # The umbrella header is a public header + adjusted_hdrs = [umbrella_header] + hdrs + + # `_headers` is the public interface of the `_clang` target, to be consumed + # by the `_swift` target. It won't be propagated to dependers of the mixed + # language library. The `_swift_interop` aspect hint allows the `_swift` + # target to import the module map for the `_clang` target. + internal_swift_interop_name = name + "_swift_interop" + swift_interop_hint( + name = internal_swift_interop_name, + module_map = module_map, + module_name = module_name, + ) + headers_library_name = name + "_headers" + native.objc_library( + name = headers_library_name, + hdrs = adjusted_hdrs, + aspect_hints = aspect_hints + [":" + internal_swift_interop_name], + defines = clang_defines, + features = features, + includes = includes, + tags = internal_tags, + textual_hdrs = textual_hdrs, + **kwargs + ) + + swift_library_name = name + "_swift" + swift_library( + name = swift_library_name, + srcs = swift_srcs, + aspect_hints = aspect_hints, + copts = ["-import-underlying-module"] + swift_copts, + defines = swift_defines, + # We generate a module map in `_mixed_language_library` instead + features = features + ["swift.no_generated_module_map"], + generates_header = True, + # TODO: Allow customizing `generated_header_name` + # This (using module name instead of target name) should be the default + # for `swift_library`, but is a breaking change + generated_header_name = module_name + "-Swift.h", + linkopts = linkopts, + module_name = module_name, + private_deps = private_deps, + swiftc_inputs = swiftc_inputs, + tags = internal_tags, + testonly = testonly, + # The `_headers` target will have headers set for inputs, set + # `includes`, and set `SwiftInfo.clang.modulemap` (via the + # `_swift_interop` target). It needs to be a private dep to prevent from + # propagating outside of these targets. The final + # `_mixed_language_library` target will propagate the umbrella header + # and extended module map. + # + # This would use `private_deps`, but that doesn't propagate `includes` + # since that could impact the generated header. We use `deps` instead, + # which is fine because `mixed_language_library` creates a new + # `SwiftInfo` which will ignore the `swift_interop_hint` set on the + # `_headers` target. + deps = [":" + headers_library_name] + deps, + **kwargs + ) + + # TODO: Should this support `cc_library`, and if so, how do we choose? + # We can't look at the `clang_srcs` attribute because it might be a + # `select`. + clang_library_name = name + "_clang" + native.objc_library( + name = clang_library_name, + srcs = clang_srcs, + hdrs = adjusted_hdrs, + non_arc_srcs = non_arc_srcs, + # `internal_swift_interop_name` isn't needed here because + # `_mixed_language_library` will explciitly set the module name. We set + # it here for rules_xcodeproj or other aspects to be able to assign a + # sane module name to the target. Since we use the target for the + # `_headers` target, this has minimal additional overhead. + aspect_hints = aspect_hints + [":" + internal_swift_interop_name], + copts = clang_copts, + defines = clang_defines, + enable_modules = enable_modules, + features = features, + includes = includes, + # The `_swift` target is an implementation dep because + # `_mixed_language_library` will propagate the final `SwiftInfo` + implementation_deps = [":" + swift_library_name] + private_deps, + linkopts = linkopts, + sdk_dylibs = sdk_dylibs, + sdk_frameworks = sdk_frameworks, + tags = internal_tags, + testonly = testonly, + textual_hdrs = textual_hdrs, + weak_sdk_frameworks = weak_sdk_frameworks, + deps = deps, + **kwargs + ) + + # `_mixed_language_library` creates an extended modulemap that includes + # the `.Swift` submodule. It returns a new `SwiftInfo` that has that + # module map set so downstream Swift targets will see it. If the + # `swift.propagate_generated_module_map` feature is set, then an include + # path is set in `CcInfo.compilation+context` so that downstream Clang + # targets will see it via implicit search. + _mixed_language_library( + name = name, + aspect_hints = aspect_hints, + clang_target = ":" + clang_library_name, + features = features, + module_map = module_map, + module_name = module_name, + swift_target = ":" + swift_library_name, + tags = tags, + testonly = testonly, + umbrella_header = umbrella_header, + visibility = visibility, + # We collect the same deps as the `_swift` target in order to propagate + # transitive `SwiftInfo`. It's inefficent to try to do that from the + # `SwiftInfo` of the `_swift` target. + deps = deps, + ) diff --git a/swift/internal/BUILD b/swift/internal/BUILD index 8a1c4bd75..57157cde7 100644 --- a/swift/internal/BUILD +++ b/swift/internal/BUILD @@ -41,6 +41,7 @@ bzl_library( name = "attrs", srcs = ["attrs.bzl"], visibility = [ + "//mixed_language:__subpackages__", "//proto:__subpackages__", "//swift:__subpackages__", ], @@ -72,6 +73,7 @@ bzl_library( name = "compiling", srcs = ["compiling.bzl"], visibility = [ + "//mixed_language:__subpackages__", "//proto:__subpackages__", "//swift:__subpackages__", ], @@ -128,13 +130,19 @@ bzl_library( bzl_library( name = "feature_names", srcs = ["feature_names.bzl"], - visibility = ["//swift:__subpackages__"], + visibility = [ + "//mixed_language:__subpackages__", + "//swift:__subpackages__", + ], ) bzl_library( name = "features", srcs = ["features.bzl"], - visibility = ["//swift:__subpackages__"], + visibility = [ + "//mixed_language:__subpackages__", + "//swift:__subpackages__", + ], deps = [ ":feature_names", ":package_specs", @@ -170,7 +178,10 @@ bzl_library( bzl_library( name = "module_maps", srcs = ["module_maps.bzl"], - visibility = ["//swift:__subpackages__"], + visibility = [ + "//mixed_language:__subpackages__", + "//swift:__subpackages__", + ], deps = ["@bazel_skylib//lib:paths"], ) @@ -192,7 +203,10 @@ bzl_library( bzl_library( name = "providers", srcs = ["providers.bzl"], - visibility = ["//swift:__subpackages__"], + visibility = [ + "//mixed_language:__subpackages__", + "//swift:__subpackages__", + ], deps = [ "//swift:providers.bzl", ], @@ -248,6 +262,7 @@ bzl_library( name = "toolchain_utils", srcs = ["toolchain_utils.bzl"], visibility = [ + "//mixed_language:__subpackages__", "//proto:__subpackages__", "//swift:__subpackages__", ], @@ -257,6 +272,7 @@ bzl_library( name = "utils", srcs = ["utils.bzl"], visibility = [ + "//mixed_language:__subpackages__", "//proto:__subpackages__", "//swift:__subpackages__", ], diff --git a/swift/internal/compiling.bzl b/swift/internal/compiling.bzl index dab4b870f..325993f6f 100644 --- a/swift/internal/compiling.bzl +++ b/swift/internal/compiling.bzl @@ -805,6 +805,7 @@ to use swift_common.compile(include_dev_srch_paths = ...) instead.\ swift = create_swift_module( ast_files = compile_outputs.ast_files, defines = defines, + generated_header = compile_outputs.generated_header_file, indexstore = compile_outputs.indexstore_directory, plugins = depset(plugins), private_swiftinterface = compile_outputs.private_swiftinterface_file, diff --git a/swift/internal/feature_names.bzl b/swift/internal/feature_names.bzl index 11640b1e5..7328ac235 100644 --- a/swift/internal/feature_names.bzl +++ b/swift/internal/feature_names.bzl @@ -132,7 +132,9 @@ SWIFT_FEATURE_NO_GENERATED_MODULE_MAP = "swift.no_generated_module_map" # If enabled, the parent directory of the generated module map is added to # `CcInfo.compilation_context.includes`. This allows `objc_library` to import -# the Swift module. +# the Swift module. If you swap this feature between enabled and disabled, and +# sandboxing is disabled, you may need to clean your output base to prevent +# implicit discovery of the generated module map. SWIFT_FEATURE_PROPAGATE_GENERATED_MODULE_MAP = "swift.propagate_generated_module_map" # If enabled, builds using the "opt" compilation mode will invoke `swiftc` with diff --git a/swift/internal/providers.bzl b/swift/internal/providers.bzl index c3592ab83..9d24e02af 100644 --- a/swift/internal/providers.bzl +++ b/swift/internal/providers.bzl @@ -140,6 +140,7 @@ def create_swift_module( swiftmodule, ast_files = [], defines = [], + generated_header = None, indexstore = None, plugins = [], swiftsourceinfo = None, @@ -155,6 +156,7 @@ def create_swift_module( ast_files: A list of `File`s output from the `DUMP_AST` action. defines: A list of defines that will be provided as `copts` to targets that depend on this module. If omitted, the empty list will be used. + generated_header: A `File` representing the Swift generated header. indexstore: A `File` representing the directory that contains the index store data generated by the compiler if the `"swift.index_while_building"` feature is enabled, otherwise this @@ -181,6 +183,7 @@ def create_swift_module( return struct( ast_files = tuple(ast_files), defines = tuple(defines), + generated_header = generated_header, plugins = plugins, private_swiftinterface = private_swiftinterface, indexstore = indexstore, diff --git a/tools/mixed_language_module_map_extender/BUILD b/tools/mixed_language_module_map_extender/BUILD new file mode 100644 index 000000000..081fb4754 --- /dev/null +++ b/tools/mixed_language_module_map_extender/BUILD @@ -0,0 +1,7 @@ +licenses(["notice"]) + +sh_binary( + name = "mixed_language_module_map_extender", + srcs = ["mixed_language_module_map_extender.sh"], + visibility = ["//visibility:public"], +) diff --git a/tools/mixed_language_module_map_extender/mixed_language_module_map_extender.sh b/tools/mixed_language_module_map_extender/mixed_language_module_map_extender.sh new file mode 100755 index 000000000..a4a614871 --- /dev/null +++ b/tools/mixed_language_module_map_extender/mixed_language_module_map_extender.sh @@ -0,0 +1,45 @@ +#!/bin/bash + +set -eu + +if [ $# -ne 4 ]; then + echo "Usage: $0 " + exit 1 +fi + +readonly output="$1" +readonly modulemap="$2" +readonly module_name="$3" +readonly swift_generated_header="$4" + +relpath() { + perl -MFile::Spec -e 'print File::Spec->abs2rel($ARGV[1], $ARGV[0])' "$1" "$2" +} + +modulemap_dir=$(perl -MFile::Spec -e 'print File::Spec->rel2abs($ARGV[0])' "${modulemap%/*}") +output_dir=$(perl -MFile::Spec -e 'print File::Spec->rel2abs($ARGV[0])' "${output%/*}") + +{ + IFS= + while read -r line || [ -n "$line" ]; do + if [[ "$line" =~ ^([[:space:]]*(private[[:space:]]+|textual[[:space:]]+|umbrella[[:space:]]+)*)header[[:space:]]+\"([^\"]+)\"(.*)$ ]]; then + prefix="${BASH_REMATCH[1]}header " + header_path="${BASH_REMATCH[3]}" + suffix="${BASH_REMATCH[4]}" + + original_header_abs_path=$(perl -MFile::Spec -e 'print File::Spec->rel2abs($ARGV[0], $ARGV[1])' "$header_path" "$modulemap_dir") + new_header_rel_path=$(relpath "$output_dir" "$original_header_abs_path") + + echo "$prefix\"$new_header_rel_path\"$suffix" + else + echo "$line" + fi + done + + echo "" + echo "module \"$module_name\".Swift {" + echo " header \"$swift_generated_header\"" + echo "" + echo " requires objc" + echo "}" +} < "$modulemap" > "$output"