| # Copyright 2020 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. |
| |
| """objc_library Starlark implementation replacing native""" |
| |
| load("@_builtins//:common/objc/semantics.bzl", "semantics") |
| load("@_builtins//:common/objc/attrs.bzl", "common_attrs") |
| |
| objc_internal = _builtins.internal.objc_internal |
| CcInfo = _builtins.toplevel.CcInfo |
| cc_common = _builtins.toplevel.cc_common |
| transition = _builtins.toplevel.transition |
| |
| def _rule_error(msg): |
| fail(msg) |
| |
| def _attribute_error(attr_name, msg): |
| fail("in attribute '" + attr_name + "': " + msg) |
| |
| def _validate_attributes(ctx): |
| if ctx.label.name.find("/") != -1: |
| _attribute_error("name", "this attribute has unsupported character '/'") |
| |
| def _create_common(ctx): |
| compilation_attributes = objc_internal.create_compilation_attributes(ctx = ctx) |
| intermediate_artifacts = objc_internal.create_intermediate_artifacts(ctx = ctx) |
| compilation_artifacts = objc_internal.create_compilation_artifacts(ctx = ctx) |
| common = objc_internal.create_common( |
| purpose = "COMPILE_AND_LINK", |
| ctx = ctx, |
| compilation_attributes = compilation_attributes, |
| compilation_artifacts = compilation_artifacts, |
| deps = ctx.attr.deps, |
| runtime_deps = ctx.attr.runtime_deps, |
| intermediate_artifacts = intermediate_artifacts, |
| alwayslink = ctx.attr.alwayslink, |
| has_module_map = True, |
| ) |
| return common |
| |
| def _build_linking_context(ctx, feature_configuration, cc_toolchain, objc_provider): |
| libraries = objc_provider.library.to_list() |
| cc_libraries = objc_provider.cc_library.to_list() |
| |
| libraries_to_link = {} |
| |
| for library in libraries: |
| library_to_link = _static_library(ctx, feature_configuration, cc_toolchain, library) |
| libraries_to_link[library_to_link] = library_to_link |
| |
| for library in cc_libraries: |
| library_to_link = _to_static_library(ctx, feature_configuration, cc_toolchain, library) |
| libraries_to_link[library_to_link] = library_to_link |
| |
| sdk_frameworks = objc_provider.sdk_framework.to_list() |
| user_link_flags = [] |
| for sdk_framework in sdk_frameworks: |
| user_link_flags.append("-framework") |
| user_link_flags.append(sdk_framework) |
| |
| linker_input = cc_common.create_linker_input( |
| owner = ctx.label, |
| libraries = depset(libraries_to_link.values()), |
| user_link_flags = depset(user_link_flags), |
| linkstamps = depset(objc_provider.linkstamp.to_list()), |
| ) |
| |
| return cc_common.create_linking_context( |
| linker_inputs = depset([linker_input], order = "topological"), |
| ) |
| |
| def _static_library( |
| ctx, |
| feature_configuration, |
| cc_toolchain, |
| library): |
| return cc_common.create_library_to_link( |
| actions = ctx.actions, |
| feature_configuration = feature_configuration, |
| cc_toolchain = cc_toolchain, |
| static_library = library, |
| ) |
| |
| def _to_static_library( |
| ctx, |
| feature_configuration, |
| cc_toolchain, |
| library): |
| if ((library.pic_static_library == None and |
| library.static_library == None) or |
| (library.dynamic_library == None and |
| library.interface_library == None)): |
| return library |
| |
| return cc_common.create_library_to_link( |
| actions = ctx.actions, |
| feature_configuration = feature_configuration, |
| cc_toolchain = cc_toolchain, |
| alwayslink = library.alwayslink, |
| pic_objects = library.pic_objects, |
| objects = library.objects, |
| static_library = library.static_library, |
| pic_static_library = library.pic_static_library, |
| ) |
| |
| def _objc_library_impl(ctx): |
| _validate_attributes(ctx) |
| common = _create_common(ctx) |
| files = [] |
| if common.compiled_archive != None: |
| files.append(common.compiled_archive) |
| compilation_support = objc_internal.create_compilation_support( |
| ctx = ctx, |
| semantics = semantics.get_semantics(), |
| ) |
| |
| compilation_support.register_compile_and_archive_actions(common = common) |
| compilation_support.validate_attributes() |
| |
| j2objc_providers = objc_internal.j2objc_providers_from_deps(ctx = ctx) |
| |
| objc_provider = common.objc_provider |
| feature_configuration = compilation_support.feature_configuration |
| cc_toolchain = compilation_support.cc_toolchain |
| linking_context = _build_linking_context(ctx, feature_configuration, cc_toolchain, objc_provider) |
| cc_info = CcInfo( |
| compilation_context = compilation_support.compilation_context, |
| linking_context = linking_context, |
| ) |
| |
| return [ |
| DefaultInfo(files = depset(files), data_runfiles = ctx.runfiles(files = files)), |
| cc_info, |
| common.objc_provider, |
| j2objc_providers[0], |
| j2objc_providers[1], |
| compilation_support.instrumented_files_info, |
| compilation_support.output_group_info, |
| ] |
| |
| def _cpu_string(platform_type, settings): |
| arch = _determine_single_architecture(platform_type, settings) |
| if platform_type == MACOS: |
| return "darwin_{}".format(arch) |
| |
| return "{}_{}".format(platform_type, arch) |
| |
| def _determine_single_architecture(platform_type, settings): |
| apple_split_cpu = settings["//command_line_option:apple_split_cpu"] |
| if apple_split_cpu != None and len(apple_split_cpu) > 0: |
| return apple_split_cpu |
| if platform_type == IOS: |
| ios_cpus = settings["//command_line_option:ios_multi_cpus"] |
| if len(ios_cpus) > 0: |
| return ios_cpus[0] |
| return _ios_cpu_from_cpu(settings["//command_line_option:cpu"]) |
| if platform_type == WATCHOS: |
| watchos_cpus = settings["//command_line_option:watchos_cpus"] |
| if len(watchos_cpus) == 0: |
| return DEFAULT_WATCHOS_CPU |
| return watchos_cpus[0] |
| if platform_type == TVOS: |
| tvos_cpus = settings["//command_line_option:tvos_cpus"] |
| if len(tvos_cpus) == 0: |
| return DEFAULT_TVOS_CPU |
| return tvos_cpus[0] |
| if platform_type == MACOS: |
| macos_cpus = settings["//command_line_option:macos_cpus"] |
| if len(macos_cpus) == 0: |
| return DEFAULT_MACOS_CPU |
| return macos_cpus[0] |
| if platform_type == CATALYST: |
| catalyst_cpus = settings["//command_line_option:catalyst_cpus"] |
| if len(catalyst_cpus) == 0: |
| return DEFAULT_CATALYST_CPU |
| return catalyst_cpus[0] |
| |
| _rule_error("ERROR: Unhandled platform type {}".format(platform_type)) |
| return None |
| |
| IOS = "ios" |
| WATCHOS = "watchos" |
| TVOS = "tvos" |
| MACOS = "macos" |
| CATALYST = "catalyst" |
| IOS_CPU_PREFIX = "ios_" |
| DEFAULT_IOS_CPU = "x86_64" |
| DEFAULT_WATCHOS_CPU = "i386" |
| DEFAULT_TVOS_CPU = "x86_64" |
| DEFAULT_MACOS_CPU = "x86_64" |
| DEFAULT_CATALYST_CPU = "x86_64" |
| |
| def _ios_cpu_from_cpu(cpu): |
| if cpu.startswith(IOS_CPU_PREFIX): |
| return cpu[len(IOS_CPU_PREFIX):] |
| return DEFAULT_IOS_CPU |
| |
| def _apple_crosstool_transition_impl(settings, attr): |
| platform_type = str(settings["//command_line_option:apple_platform_type"]) |
| cpu = _cpu_string(platform_type, settings) |
| if cpu == settings["//command_line_option:cpu"] and settings["//command_line_option:crosstool_top"] == settings["//command_line_option:apple_crosstool_top"]: |
| return { |
| "//command_line_option:apple configuration distinguisher": settings["//command_line_option:apple configuration distinguisher"], |
| "//command_line_option:apple_platform_type": settings["//command_line_option:apple_platform_type"], |
| "//command_line_option:apple_split_cpu": settings["//command_line_option:apple_split_cpu"], |
| "//command_line_option:compiler": settings["//command_line_option:compiler"], |
| "//command_line_option:cpu": settings["//command_line_option:cpu"], |
| "//command_line_option:crosstool_top": settings["//command_line_option:crosstool_top"], |
| "//command_line_option:platforms": settings["//command_line_option:platforms"], |
| "//command_line_option:fission": settings["//command_line_option:fission"], |
| "//command_line_option:grte_top": settings["//command_line_option:grte_top"], |
| "//command_line_option:ios_minimum_os": settings["//command_line_option:ios_minimum_os"], |
| "//command_line_option:macos_minimum_os": settings["//command_line_option:macos_minimum_os"], |
| "//command_line_option:tvos_minimum_os": settings["//command_line_option:tvos_minimum_os"], |
| "//command_line_option:watchos_minimum_os": settings["//command_line_option:watchos_minimum_os"], |
| } |
| |
| return { |
| "//command_line_option:apple configuration distinguisher": "applebin_" + platform_type, |
| "//command_line_option:apple_platform_type": settings["//command_line_option:apple_platform_type"], |
| "//command_line_option:apple_split_cpu": settings["//command_line_option:apple_split_cpu"], |
| "//command_line_option:compiler": settings["//command_line_option:apple_compiler"], |
| "//command_line_option:cpu": cpu, |
| "//command_line_option:crosstool_top": ( |
| settings["//command_line_option:apple_crosstool_top"] |
| ), |
| "//command_line_option:platforms": [], |
| "//command_line_option:fission": [], |
| "//command_line_option:grte_top": settings["//command_line_option:apple_grte_top"], |
| "//command_line_option:ios_minimum_os": settings["//command_line_option:ios_minimum_os"], |
| "//command_line_option:macos_minimum_os": settings["//command_line_option:macos_minimum_os"], |
| "//command_line_option:tvos_minimum_os": settings["//command_line_option:tvos_minimum_os"], |
| "//command_line_option:watchos_minimum_os": settings["//command_line_option:watchos_minimum_os"], |
| } |
| |
| _apple_rule_base_transition_inputs = [ |
| "//command_line_option:apple configuration distinguisher", |
| "//command_line_option:apple_compiler", |
| "//command_line_option:compiler", |
| "//command_line_option:apple_platform_type", |
| "//command_line_option:apple_crosstool_top", |
| "//command_line_option:crosstool_top", |
| "//command_line_option:apple_split_cpu", |
| "//command_line_option:apple_grte_top", |
| "//command_line_option:cpu", |
| "//command_line_option:ios_multi_cpus", |
| "//command_line_option:macos_cpus", |
| "//command_line_option:tvos_cpus", |
| "//command_line_option:watchos_cpus", |
| "//command_line_option:catalyst_cpus", |
| "//command_line_option:ios_minimum_os", |
| "//command_line_option:macos_minimum_os", |
| "//command_line_option:tvos_minimum_os", |
| "//command_line_option:watchos_minimum_os", |
| "//command_line_option:platforms", |
| "//command_line_option:fission", |
| "//command_line_option:grte_top", |
| ] |
| _apple_rule_base_transition_outputs = [ |
| "//command_line_option:apple configuration distinguisher", |
| "//command_line_option:apple_platform_type", |
| "//command_line_option:apple_split_cpu", |
| "//command_line_option:compiler", |
| "//command_line_option:cpu", |
| "//command_line_option:crosstool_top", |
| "//command_line_option:platforms", |
| "//command_line_option:fission", |
| "//command_line_option:grte_top", |
| "//command_line_option:ios_minimum_os", |
| "//command_line_option:macos_minimum_os", |
| "//command_line_option:tvos_minimum_os", |
| "//command_line_option:watchos_minimum_os", |
| ] |
| |
| apple_crosstool_transition = transition( |
| implementation = _apple_crosstool_transition_impl, |
| inputs = _apple_rule_base_transition_inputs, |
| outputs = _apple_rule_base_transition_outputs, |
| ) |
| |
| objc_library = rule( |
| implementation = _objc_library_impl, |
| attrs = common_attrs.union( |
| { |
| "data": attr.label_list(allow_files = True), |
| "_cc_toolchain": attr.label( |
| default = "@" + semantics.get_repo() + "//tools/cpp:current_cc_toolchain", |
| ), |
| }, |
| common_attrs.COMPILING_RULE, |
| common_attrs.COMPILE_DEPENDENCY_RULE, |
| common_attrs.INCLUDE_SCANNING_RULE, |
| common_attrs.SDK_FRAMEWORK_DEPENDER_RULE, |
| common_attrs.COPTS_RULE, |
| common_attrs.X_C_RUNE_RULE, |
| ), |
| fragments = ["objc", "apple", "cpp"], |
| cfg = apple_crosstool_transition, |
| toolchains = ["@" + semantics.get_repo() + "//tools/cpp:toolchain_type"], |
| incompatible_use_toolchain_transition = True, |
| ) |