c4872ce644
The big change here is having the C++ toolchain use
Bazel platforms instead of the C++ specific flags/setup.
In Bazel, platforms are a general purpose way to define
things like os, cpu architecture, etc. We were not using
platforms previously, because the best documentation at
the time focused on the old ways.
However, the old ways were clumsy/difficult when trying
to manage cross-compilation, specifically when trying
to have a Mac host trigger a build on our Linux RBE
system targeting a Linux x64 system. Thus, rather than
keep investing in the legacy system, this CL migrates
us to using platforms where possible.
Suggested background reading to better understand this CL:
- https://bazel.build/concepts/platforms-intro
- https://bazel.build/docs/platforms
- https://bazel.build/docs/toolchains#registering-building-toolchains
The hermetic toolchain itself is not changing in this CL
(and likely does not need to), only how we tell Bazel
about it (i.e. registering it) and how Bazel decides
to use it (i.e. resolving toolchains).
Here is my understanding of how platforms and toolchains
interact (supported by some evidence from [1][2])
- Bazel needs to resolve platforms for the Host, Execution,
and Target.
- If not specified via flags, these are the machine from
which Bazel is invoked, aka "@local_config_platform//:host".
- With this CL, the Host could be a Mac laptop, the Execution
platform is our Linux RBE pool, and the Target is "a Linux
system with a x64 CPU"
- To specify the Host, that is, describe to Bazel the
capabilities of the system it is running on, one can
set --host_platform [3] with a label pointing to a platform()
containing the appropriate settings. Tip: have this
platform inherit from @local_config_platform//:host
so it can add to any of the constraint_settings and
constraint_values that Bazel deduces automatically.
- To specify the Target platform(s), that is, the system
on which a final output resides and can execute, one
can set the --platforms flag with a label referencing
a platform().
- Bazel will then choose an execution platform to fulfill
that request. Bazel will look through a list of available
platforms, which can be augmented* with the
--extra_execution_platforms. Platforms specified by this
flag will be considered higher than the default platforms!
- Having selected the appropriate platforms, Bazel now
needs to select a toolchain to actually run the actions
of the appropriate type.
- Bazel looks through the list of available toolchains
and finds one that "matches" the Execution and the Target
platform. This means, the toolchain's exec_compatible_with
is a strict subset of the Execution platform and
the toolchain's target_compatible_with is a strict subset
of the Target platform. To register toolchains* (i.e. add
them to the resolution list), we use --extra_toolchains.
Once Bazel finds a match, it stops looking.
Using --toolchain_resolution_debug=".*" makes Bazel log
how it is resolving these toolchains and what execution
platform it picked.
* We can also register execution platforms and toolchains in
WORKSPACE.bazel [4], but the flags come with higher priority
and that made resolution a bit tricky. Also, when we want
to conditionally add them (e.g. --config=linux_rbe), we
cannot remove them conditionally in the WORKSPACE.bazel file.
The above resolution flow directly necessitated the changes
in this CL.
Example usage of the new configs and platforms:
# Can be run on a x64 Linux host and uses the hermetic toolchain.
bazel build //:skia_public
# Can be run on Mac or Linux and uses the Linux RBE system along
# with the hermetic toolchain to compile a binary for Linux x64.
bazel build //:skia_public --config=linux_rbe --config=for_linux_x64
# Shorthand for above
bazel build //:skia_public --config=for_linux_x64_with_rbe
Notice we don't have to type out --config=clang_linux anymore!
That was due to me reading the Bazel docs more carefully and
realizing we can set options for *all* Bazel build commands.
Current Limitations:
- Targets which require a py_binary (e.g. Dawn's genrules)
will not work on RBE when cross compiling because the
python runtime we download is for the host machine, not
the executor. This means //example:hello_world_dawn does
not work on Mac when cross-compiling via linux_rbe.
- Mac M1 linking not quite working with SkOpts settings.
Probably need to set -target [5]
Suggested Review order:
- toolchain/BUILD.bazel Notice how we do away with
cc_toolchain_suite for toolchain. These have the same
role: giving Bazel the information about where a toolchain
can run. The platforms one is more expressive (IMO), allowing
us to say both where to run the toolchain and what it can
make. In order to more easily force the use of our hermetic
toolchain, but also allow the hermetic toolchain to be used
on RBE, we specify "use_hermetic_toolchain" only on the target,
because the RBE image does not have the hermetic toolchain
on it by default (but can certainly run it).
- bazel/platform/BUILD.bazel to see the custom constraint_setting
and corresponding constraint_value. The names for both of these
are completely arbitrary - they do not need to have any deeper
meaning or relation to any file or Docker image or system or
any other constraints. Think of the constraint_setting as
an Enum and the constraint_value being the one and only member.
We need to pass around a constant value, not a type, so we
need to provide the constraint_value (e.g. in toolchain/BUILD.bazel)
but not a constraint_setting. However we need a
constraint_setting declared so we can make a constraint_value
of that "type".
Notice the platform declared here - it allows us to force
Bazel to use the hermetic toolchain because of the extra
constraint_value.
- .bazelrc I set a few flags that will be on for all
bazel build commands. Importantly, this causes the C++
build logic to use platforms and not the old, bespoke way.
I also found a way to avoid using the local toolchain on
the host, which will hopefully lead to clearer errors
if platforms are mis-specified instead of odd compile
errors because the host toolchain is too old or something.
There are also a few RBE settings tweaked to be a bit
more modern, as well the new shorthands for specifying
target platforms (e.g. for_linux_x64).
- bazel/buildrc where we have to turn off the platforms
logic for emscripten https://github.com/emscripten-core/emsdk/issues/984
- bazel/rbe/BUILD.bazel for a fix in the platform description
that makes it work on Mac.
- Notice that _m1 has been removed from the mac-related toolchain
files because the same toolchain should work on both
architectures.
- All other changes in any order.
[1] https://bazel.build/docs/toolchains#debugging-toolchains
[2] https://bazel.build/docs/toolchains#toolchain-resolution
[3] https://bazel.build/reference/command-line-reference
[4] https://bazel.build/docs/toolchains#registering-building-toolchains
[5] 17dc3f16fc/gn/skia/BUILD.gn (L258-L271)
Change-Id: I515c114099d659639a808f74e47d489a68b7af62
Bug: skia:12541
Reviewed-on: https://skia-review.googlesource.com/c/skia/+/549737
Reviewed-by: Erik Rose <erikrose@google.com>
Reviewed-by: Jorge Betancourt <jmbetancourt@google.com>
143 lines
4.5 KiB
Python
143 lines
4.5 KiB
Python
load("//bazel:macros.bzl", "exports_files_legacy")
|
|
load("@rules_cc//cc:defs.bzl", "cc_toolchain", "cc_toolchain_suite")
|
|
load(":linux_amd64_toolchain_config.bzl", "provide_linux_amd64_toolchain_config")
|
|
load(":mac_toolchain_config.bzl", "provide_mac_toolchain_config")
|
|
|
|
licenses(["notice"])
|
|
|
|
exports_files_legacy()
|
|
|
|
# https://bazel.build/docs/toolchains
|
|
# https://bazel.build/reference/be/platform#toolchain
|
|
toolchain(
|
|
name = "clang_linux_x64_toolchain",
|
|
# Where should we run this toolchain?
|
|
exec_compatible_with = [
|
|
"@platforms//os:linux",
|
|
"@platforms//cpu:x86_64",
|
|
# We do not want an extra constraint here related to the hermetic toolchain because
|
|
# we want the toolchain to run on any Linux x64 machine (and it certainly can).
|
|
],
|
|
# What can this toolchain build?
|
|
target_compatible_with = [
|
|
"@platforms//os:linux",
|
|
"@platforms//cpu:x86_64",
|
|
# We want to be able to explicitly tell Bazel to use this toolchain, and not the
|
|
# default one on a user's machine or on the RBE worker. Thus we need an extra constraint
|
|
# that we can use to differentiate the "stock" C++ toolchain from our hermetic one and
|
|
# force that use by specifying the target platform.
|
|
"//bazel/platform:use_hermetic_toolchain",
|
|
],
|
|
toolchain = ":linux_amd64_host",
|
|
# https://github.com/bazelbuild/rules_cc/blob/8bb0eb5c5ccd96b91753bb112096bb6993d16d13/cc/BUILD#L32-L36
|
|
toolchain_type = "@rules_cc//cc:toolchain_type",
|
|
)
|
|
|
|
# Our one mac toolchain can run on either Intel Macs or M1 Macs, however Bazel does not allow you to specify
|
|
# more than one cpu type in exec_compatible_with. Thus, we list the same toolchain twice.
|
|
toolchain(
|
|
name = "clang_mac_x64_toolchain",
|
|
exec_compatible_with = [
|
|
"@platforms//os:macos",
|
|
"@platforms//cpu:x86_64",
|
|
],
|
|
target_compatible_with = [
|
|
"@platforms//os:macos",
|
|
"@platforms//cpu:x86_64",
|
|
"//bazel/platform:use_hermetic_toolchain",
|
|
],
|
|
toolchain = ":mac_host",
|
|
# https://github.com/bazelbuild/rules_cc/blob/8bb0eb5c5ccd96b91753bb112096bb6993d16d13/cc/BUILD#L32-L36
|
|
toolchain_type = "@rules_cc//cc:toolchain_type",
|
|
)
|
|
|
|
toolchain(
|
|
name = "clang_mac_arm64_toolchain",
|
|
exec_compatible_with = [
|
|
"@platforms//os:macos",
|
|
"@platforms//cpu:arm64",
|
|
],
|
|
target_compatible_with = [
|
|
"@platforms//os:macos",
|
|
"@platforms//cpu:arm64",
|
|
"//bazel/platform:use_hermetic_toolchain",
|
|
],
|
|
toolchain = ":mac_host",
|
|
# https://github.com/bazelbuild/rules_cc/blob/8bb0eb5c5ccd96b91753bb112096bb6993d16d13/cc/BUILD#L32-L36
|
|
toolchain_type = "@rules_cc//cc:toolchain_type",
|
|
)
|
|
|
|
|
|
filegroup(name = "not_implemented")
|
|
|
|
filegroup(
|
|
name = "archive_linux_amd64_files",
|
|
srcs = [
|
|
"linux_trampolines/ar_trampoline_linux.sh",
|
|
"@clang_linux_amd64//:archive_files",
|
|
],
|
|
)
|
|
|
|
filegroup(
|
|
name = "compile_linux_amd64_files",
|
|
srcs = [
|
|
"linux_trampolines/IWYU_mapping.imp",
|
|
"linux_trampolines/clang_trampoline_linux.sh",
|
|
"@clang_linux_amd64//:compile_files",
|
|
],
|
|
)
|
|
|
|
filegroup(
|
|
name = "link_linux_amd64_files",
|
|
srcs = [
|
|
# Bazel assumes it is talking to Clang when linking.
|
|
"linux_trampolines/clang_trampoline_linux.sh",
|
|
"@clang_linux_amd64//:link_files",
|
|
],
|
|
)
|
|
|
|
filegroup(
|
|
name = "all_mac_files",
|
|
srcs = [
|
|
"mac_trampolines/ar_trampoline_mac.sh",
|
|
"mac_trampolines/clang_trampoline_mac.sh",
|
|
"mac_trampolines/lld_trampoline_mac.sh",
|
|
"@clang_mac//:all_files",
|
|
],
|
|
)
|
|
|
|
provide_linux_amd64_toolchain_config(
|
|
name = "linux_amd64_toolchain_config",
|
|
)
|
|
|
|
provide_mac_toolchain_config(
|
|
name = "mac_toolchain_config",
|
|
)
|
|
|
|
# https://bazel.build/reference/be/c-cpp#cc_toolchain
|
|
cc_toolchain(
|
|
name = "linux_amd64_host",
|
|
all_files = ":not_implemented",
|
|
ar_files = ":archive_linux_amd64_files",
|
|
compiler_files = ":compile_linux_amd64_files",
|
|
dwp_files = ":not_implemented",
|
|
linker_files = ":link_linux_amd64_files",
|
|
objcopy_files = ":not_implemented",
|
|
strip_files = ":not_implemented",
|
|
supports_param_files = False,
|
|
toolchain_config = ":linux_amd64_toolchain_config",
|
|
)
|
|
|
|
cc_toolchain(
|
|
name = "mac_host",
|
|
all_files = ":all_mac_files",
|
|
ar_files = ":all_mac_files",
|
|
compiler_files = ":all_mac_files",
|
|
dwp_files = ":not_implemented",
|
|
linker_files = ":all_mac_files",
|
|
objcopy_files = ":not_implemented",
|
|
strip_files = ":not_implemented",
|
|
supports_param_files = False,
|
|
toolchain_config = ":mac_toolchain_config",
|
|
)
|