blob: f63c82d50d3f2d8379222e9d27e38bf2a3a77fdd [file] [log] [blame]
# 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/compilation_support.bzl", "compilation_support")
load("@_builtins//:common/objc/attrs.bzl", "common_attrs")
load("@_builtins//:common/cc/cc_helper.bzl", "cc_helper")
objc_internal = _builtins.internal.objc_internal
CcInfo = _builtins.toplevel.CcInfo
cc_common = _builtins.toplevel.cc_common
transition = _builtins.toplevel.transition
coverage_common = _builtins.toplevel.coverage_common
apple_common = _builtins.toplevel.apple_common
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 _build_linking_context(ctx, feature_configuration, cc_toolchain, objc_provider, common_variables):
libraries = []
if common_variables.compilation_artifacts.archive != None:
library_to_link = _static_library(ctx, feature_configuration, cc_toolchain, common_variables.compilation_artifacts.archive)
libraries.append(library_to_link)
archives_from_objc_library = {}
for library in objc_provider.library.to_list():
archives_from_objc_library[library.path] = library
objc_libraries_cc_infos = []
for dep in ctx.attr.deps:
if apple_common.Objc in dep and CcInfo in dep:
objc_libraries_cc_infos.append(dep[CcInfo])
merged_objc_library_cc_infos = cc_common.merge_cc_infos(cc_infos = objc_libraries_cc_infos)
for linker_input in merged_objc_library_cc_infos.linking_context.linker_inputs.to_list():
for lib in linker_input.libraries:
path = None
if lib.static_library != None:
path = lib.static_library.path
elif lib.pic_static_library != None:
path = lib.pic_static_library.path
if path in archives_from_objc_library and archives_from_objc_library[path]:
libraries.append(lib)
archives_from_objc_library[path] = None
for archive in archives_from_objc_library.values():
if archive:
library_to_link = _static_library(ctx, feature_configuration, cc_toolchain, archive)
libraries.append(library_to_link)
libraries.extend(objc_provider.cc_library.to_list())
sdk_frameworks = objc_provider.sdk_framework.to_list()
user_link_flags = []
for sdk_framework in sdk_frameworks:
user_link_flags.append(["-framework", sdk_framework])
direct_linker_inputs = []
if len(user_link_flags) != 0 or len(libraries) != 0 or objc_provider.linkstamp:
linker_input = cc_common.create_linker_input(
owner = ctx.label,
libraries = depset(libraries),
user_link_flags = user_link_flags,
linkstamps = objc_provider.linkstamp,
)
direct_linker_inputs.append(linker_input)
return cc_common.create_linking_context(
linker_inputs = depset(direct = direct_linker_inputs, order = "topological"),
)
def _static_library(
ctx,
feature_configuration,
cc_toolchain,
library):
alwayslink = False
if library.extension == "lo":
alwayslink = True
return cc_common.create_library_to_link(
actions = ctx.actions,
feature_configuration = feature_configuration,
cc_toolchain = cc_toolchain,
static_library = library,
alwayslink = alwayslink,
)
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)
cc_toolchain = cc_helper.find_cpp_toolchain(ctx)
common_variables = compilation_support.build_common_variables(
ctx,
cc_toolchain,
True,
False,
False,
False,
ctx.attr.deps,
ctx.attr.runtime_deps,
[],
ctx.attr.linkopts,
)
files = []
if common_variables.compilation_artifacts.archive != None:
files.append(common_variables.compilation_artifacts.archive)
(cc_compilation_context, compilation_outputs, output_group_info) = compilation_support.register_compile_and_archive_actions(
common_variables,
[],
[],
)
compilation_support.validate_attributes(common_variables)
j2objc_providers = objc_internal.j2objc_providers_from_deps(ctx = ctx)
objc_provider = common_variables.objc_provider
feature_configuration = compilation_support.build_feature_configuration(common_variables, False, True)
linking_context = _build_linking_context(ctx, feature_configuration, cc_toolchain, objc_provider, common_variables)
cc_info = CcInfo(
compilation_context = cc_compilation_context,
linking_context = linking_context,
)
return [
DefaultInfo(files = depset(files), data_runfiles = ctx.runfiles(files = files)),
cc_info,
objc_provider,
j2objc_providers[0],
j2objc_providers[1],
objc_internal.instrumented_files_info(ctx = ctx, object_files = compilation_outputs.objects),
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 {} # No changes necessary.
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.LICENSES,
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,
)