mirror of
https://github.com/digital-asset/daml.git
synced 2024-11-10 10:46:11 +03:00
142 lines
5.3 KiB
Python
142 lines
5.3 KiB
Python
# Copyright (c) 2019 The DAML Authors. All rights reserved.
|
||
# SPDX-License-Identifier: Apache-2.0
|
||
|
||
load("@os_info//:os_info.bzl", "is_darwin", "is_windows")
|
||
|
||
def _fat_cc_library_impl(ctx):
|
||
input_lib = ctx.attr.input_lib
|
||
cc_info = input_lib[CcInfo]
|
||
static_libs = []
|
||
|
||
# For now we assume that we have static PIC libs for all libs.
|
||
# It should be possible to extend this but we do not have a need
|
||
# for it so far and it would complicate things.
|
||
for lib in cc_info.linking_context.libraries_to_link.to_list():
|
||
static_lib = None
|
||
if lib.pic_static_library:
|
||
static_lib = lib.pic_static_library
|
||
elif is_windows and lib.static_library:
|
||
# On Windows we don't seem to have `pic_static_library`s available.
|
||
static_lib = lib.static_library
|
||
else:
|
||
fail("No (PIC) static library found for '{}'.".format(
|
||
str(lib.dynamic_library.path),
|
||
))
|
||
static_libs += [static_lib]
|
||
|
||
dyn_lib = ctx.outputs.dynamic_library
|
||
static_lib = ctx.outputs.static_library
|
||
|
||
toolchain = ctx.attr._cc_toolchain[cc_common.CcToolchainInfo]
|
||
feature_configuration = cc_common.configure_features(ctx = ctx, cc_toolchain = toolchain)
|
||
|
||
compiler = None
|
||
if is_darwin:
|
||
# toolchain.compiler_executable() fails on MacOS, see https://github.com/bazelbuild/bazel/issues/7105
|
||
compiler = ctx.executable._cc_compiler
|
||
elif is_windows:
|
||
compiler = toolchain.compiler_executable() + ".exe"
|
||
else:
|
||
compiler = toolchain.compiler_executable()
|
||
ctx.actions.run(
|
||
mnemonic = "CppLinkFatDynLib",
|
||
outputs = [dyn_lib],
|
||
executable = compiler,
|
||
arguments =
|
||
["-o", dyn_lib.path, "-shared"] +
|
||
ctx.attr.whole_archive_flag +
|
||
[f.path for f in static_libs] +
|
||
ctx.attr.no_whole_archive_flag +
|
||
# Some libs seems to depend on libstdc++ implicitely
|
||
["-lstdc++"] +
|
||
(["-framework", "CoreFoundation"] if is_darwin else []) +
|
||
# On Windows some libs seems to depend on Windows sockets
|
||
(["-lws2_32"] if is_windows else []),
|
||
inputs = static_libs,
|
||
env = {"PATH": ""},
|
||
)
|
||
|
||
mri_script_content = "\n".join(
|
||
["create {}".format(static_lib.path)] +
|
||
["addlib {}".format(lib.path) for lib in static_libs] +
|
||
["save", "end"],
|
||
) + "\n"
|
||
|
||
mri_script = ctx.actions.declare_file(ctx.label.name + "_mri")
|
||
ctx.actions.write(mri_script, mri_script_content)
|
||
|
||
ar = toolchain.ar_executable()
|
||
|
||
if ar.find("libtool") >= 0:
|
||
# We are on MacOS where ar_executable is actually libtool, see
|
||
# https://github.com/bazelbuild/bazel/issues/5127.
|
||
ctx.actions.run(
|
||
mnemonic = "CppLinkFatStaticLib",
|
||
outputs = [static_lib],
|
||
executable = ar,
|
||
inputs = static_libs,
|
||
arguments =
|
||
["-no_warning_for_no_symbols", "-static", "-o", static_lib.path] +
|
||
[f.path for f in static_libs],
|
||
)
|
||
else:
|
||
ctx.actions.run_shell(
|
||
mnemonic = "CppLinkFatStaticLib",
|
||
outputs = [static_lib],
|
||
inputs = [mri_script] + static_libs,
|
||
command = "{ar} -M < {mri_script}".format(ar = ar, mri_script = mri_script.path),
|
||
)
|
||
|
||
fat_lib = cc_common.create_library_to_link(
|
||
actions = ctx.actions,
|
||
feature_configuration = feature_configuration,
|
||
cc_toolchain = toolchain,
|
||
dynamic_library = dyn_lib,
|
||
static_library = static_lib,
|
||
)
|
||
|
||
new_linking_context = cc_common.create_linking_context(
|
||
libraries_to_link = [fat_lib],
|
||
)
|
||
new_cc_info = CcInfo(
|
||
linking_context = new_linking_context,
|
||
compilation_context = cc_info.compilation_context,
|
||
)
|
||
return [new_cc_info]
|
||
|
||
# Shared libraries built with Bazel do not declare their dependencies on other libraries properly.
|
||
# Instead that dependency is tracked in Bazel internally. This breaks the GHCi linker if
|
||
# RTLD_LAZY doesn’t work which happens quite often. To make matters worse, we also cannot use
|
||
# linkstatic = True, since the GHCi linker cannot handle some relocations.
|
||
# To work around this mess, we create fat libraries that do not have additional dependencies.
|
||
# See https://github.com/tweag/rules_haskell/issues/720
|
||
fat_cc_library = rule(
|
||
_fat_cc_library_impl,
|
||
attrs = dict({
|
||
"input_lib": attr.label(),
|
||
"_cc_toolchain": attr.label(
|
||
default = Label("@bazel_tools//tools/cpp:current_cc_toolchain"),
|
||
),
|
||
"_cc_compiler": attr.label(
|
||
allow_files = True,
|
||
executable = True,
|
||
cfg = "host",
|
||
default =
|
||
# bin/cc is gcc on Darwin which fails to find libc++
|
||
Label("@nixpkgs_cc_toolchain//:bin/clang") if is_darwin else None,
|
||
),
|
||
"whole_archive_flag": attr.string_list(
|
||
# ld on MacOS doesn’t understand --whole-archive
|
||
default = ["-Wl,-all_load"] if is_darwin else ["-Wl,--whole-archive"],
|
||
),
|
||
"no_whole_archive_flag": attr.string_list(
|
||
default = [] if is_darwin else ["-Wl,--no-whole-archive"],
|
||
),
|
||
}),
|
||
fragments = ["cpp"],
|
||
outputs = {
|
||
"dynamic_library": "lib%{name}.dll" if is_windows else "lib%{name}.so",
|
||
"static_library": "lib%{name}.a",
|
||
},
|
||
)
|