|  | # Copyright 2016 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. | 
|  | """Rules for configuring the C++ toolchain (experimental).""" | 
|  |  | 
|  | load("@bazel_tools//tools/cpp:windows_cc_configure.bzl", "configure_windows_toolchain") | 
|  | load("@bazel_tools//tools/cpp:osx_cc_configure.bzl", "configure_osx_toolchain") | 
|  | load("@bazel_tools//tools/cpp:unix_cc_configure.bzl", "configure_unix_toolchain") | 
|  | load( | 
|  | "@bazel_tools//tools/cpp:lib_cc_configure.bzl", | 
|  | "get_cpu_value", | 
|  | "resolve_labels", | 
|  | ) | 
|  | load("@bazel_tools//tools/osx:xcode_configure.bzl", "run_xcode_locator") | 
|  |  | 
|  | def _generate_cpp_only_build_file(repository_ctx, cpu_value, paths): | 
|  | repository_ctx.template( | 
|  | "BUILD", | 
|  | paths["@bazel_tools//tools/cpp:BUILD.toolchains.tpl"], | 
|  | {"%{name}": cpu_value}, | 
|  | ) | 
|  |  | 
|  | def cc_autoconf_toolchains_impl(repository_ctx): | 
|  | """Generate BUILD file with 'toolchain' targets for the local host C++ toolchain. | 
|  |  | 
|  | Args: | 
|  | repository_ctx: repository context | 
|  | """ | 
|  | paths = resolve_labels(repository_ctx, [ | 
|  | "@bazel_tools//tools/cpp:BUILD.toolchains.tpl", | 
|  | "@bazel_tools//tools/osx/crosstool:BUILD.toolchains", | 
|  | "@bazel_tools//tools/osx/crosstool:osx_archs.bzl", | 
|  | "@bazel_tools//tools/osx:xcode_locator.m", | 
|  | ]) | 
|  | env = repository_ctx.os.environ | 
|  | cpu_value = get_cpu_value(repository_ctx) | 
|  |  | 
|  | # Should we try to find C++ toolchain at all? If not, we don't have to generate toolchains for C++ at all. | 
|  | should_detect_cpp_toolchain = "BAZEL_DO_NOT_DETECT_CPP_TOOLCHAIN" not in env or env["BAZEL_DO_NOT_DETECT_CPP_TOOLCHAIN"] != "1" | 
|  |  | 
|  | # Should we unconditionally *not* use xcode? If so, we don't have to run Xcode locator ever. | 
|  | should_use_cpp_only_toolchain = "BAZEL_USE_CPP_ONLY_TOOLCHAIN" in env and env["BAZEL_USE_CPP_ONLY_TOOLCHAIN"] == "1" | 
|  |  | 
|  | # Should we unconditionally use xcode? If so, we don't have to run Xcode locator now. | 
|  | should_use_xcode = "BAZEL_USE_XCODE_TOOLCHAIN" in env and env["BAZEL_USE_XCODE_TOOLCHAIN"] == "1" | 
|  |  | 
|  | if not should_detect_cpp_toolchain: | 
|  | repository_ctx.file("BUILD", "# C++ toolchain autoconfiguration was disabled by BAZEL_DO_NOT_DETECT_CPP_TOOLCHAIN env variable.") | 
|  | elif cpu_value.startswith("darwin") and not should_use_cpp_only_toolchain: | 
|  | xcode_toolchains = [] | 
|  |  | 
|  | # Only detect xcode if the user didn't tell us it will be there. | 
|  | if not should_use_xcode: | 
|  | # TODO(#6926): Unify C++ and ObjC toolchains so we don't have to run xcode locator to generate toolchain targets. | 
|  | # And also so we don't have to keep this code in sync with //tools/cpp:osx_cc_configure.bzl. | 
|  | (xcode_toolchains, _xcodeloc_err) = run_xcode_locator( | 
|  | repository_ctx, | 
|  | paths["@bazel_tools//tools/osx:xcode_locator.m"], | 
|  | ) | 
|  |  | 
|  | if should_use_xcode or xcode_toolchains: | 
|  | repository_ctx.symlink(paths["@bazel_tools//tools/osx/crosstool:BUILD.toolchains"], "BUILD") | 
|  | repository_ctx.symlink(paths["@bazel_tools//tools/osx/crosstool:osx_archs.bzl"], "osx_archs.bzl") | 
|  | else: | 
|  | _generate_cpp_only_build_file(repository_ctx, cpu_value, paths) | 
|  | else: | 
|  | _generate_cpp_only_build_file(repository_ctx, cpu_value, paths) | 
|  |  | 
|  | cc_autoconf_toolchains = repository_rule( | 
|  | environ = [ | 
|  | "BAZEL_USE_CPP_ONLY_TOOLCHAIN", | 
|  | "BAZEL_DO_NOT_DETECT_CPP_TOOLCHAIN", | 
|  | ], | 
|  | implementation = cc_autoconf_toolchains_impl, | 
|  | configure = True, | 
|  | ) | 
|  |  | 
|  | def cc_autoconf_impl(repository_ctx, overriden_tools = dict()): | 
|  | """Generate BUILD file with 'cc_toolchain' targets for the local host C++ toolchain. | 
|  |  | 
|  | Args: | 
|  | repository_ctx: repository context | 
|  | overriden_tools: dict of tool paths to use instead of autoconfigured tools | 
|  | """ | 
|  |  | 
|  | env = repository_ctx.os.environ | 
|  | cpu_value = get_cpu_value(repository_ctx) | 
|  | if "BAZEL_DO_NOT_DETECT_CPP_TOOLCHAIN" in env and env["BAZEL_DO_NOT_DETECT_CPP_TOOLCHAIN"] == "1": | 
|  | paths = resolve_labels(repository_ctx, [ | 
|  | "@bazel_tools//tools/cpp:BUILD.empty.tpl", | 
|  | "@bazel_tools//tools/cpp:empty_cc_toolchain_config.bzl", | 
|  | ]) | 
|  | repository_ctx.symlink(paths["@bazel_tools//tools/cpp:empty_cc_toolchain_config.bzl"], "cc_toolchain_config.bzl") | 
|  | repository_ctx.template("BUILD", paths["@bazel_tools//tools/cpp:BUILD.empty.tpl"], { | 
|  | "%{cpu}": get_cpu_value(repository_ctx), | 
|  | }) | 
|  | elif cpu_value == "freebsd" or cpu_value == "openbsd": | 
|  | paths = resolve_labels(repository_ctx, [ | 
|  | "@bazel_tools//tools/cpp:BUILD.static.bsd", | 
|  | "@bazel_tools//tools/cpp:bsd_cc_toolchain_config.bzl", | 
|  | ]) | 
|  |  | 
|  | # This is defaulting to a static crosstool. We should eventually | 
|  | # autoconfigure this platform too. Theorically, FreeBSD and OpenBSD | 
|  | # should be straightforward to add but we cannot run them in a Docker | 
|  | # container so skipping until we have proper tests for these platforms. | 
|  | repository_ctx.symlink(paths["@bazel_tools//tools/cpp:bsd_cc_toolchain_config.bzl"], "cc_toolchain_config.bzl") | 
|  | repository_ctx.symlink(paths["@bazel_tools//tools/cpp:BUILD.static.bsd"], "BUILD") | 
|  | elif cpu_value in ["x64_windows", "arm64_windows"]: | 
|  | # TODO(ibiryukov): overriden_tools are only supported in configure_unix_toolchain. | 
|  | # We might want to add that to Windows too(at least for msys toolchain). | 
|  | configure_windows_toolchain(repository_ctx) | 
|  | elif (cpu_value.startswith("darwin") and | 
|  | ("BAZEL_USE_CPP_ONLY_TOOLCHAIN" not in env or env["BAZEL_USE_CPP_ONLY_TOOLCHAIN"] != "1")): | 
|  | configure_osx_toolchain(repository_ctx, cpu_value, overriden_tools) | 
|  | else: | 
|  | configure_unix_toolchain(repository_ctx, cpu_value, overriden_tools) | 
|  |  | 
|  | MSVC_ENVVARS = [ | 
|  | "BAZEL_VC", | 
|  | "BAZEL_VC_FULL_VERSION", | 
|  | "BAZEL_VS", | 
|  | "BAZEL_WINSDK_FULL_VERSION", | 
|  | "VS90COMNTOOLS", | 
|  | "VS100COMNTOOLS", | 
|  | "VS110COMNTOOLS", | 
|  | "VS120COMNTOOLS", | 
|  | "VS140COMNTOOLS", | 
|  | "VS150COMNTOOLS", | 
|  | "VS160COMNTOOLS", | 
|  | "TMP", | 
|  | "TEMP", | 
|  | ] | 
|  |  | 
|  | cc_autoconf = repository_rule( | 
|  | environ = [ | 
|  | "ABI_LIBC_VERSION", | 
|  | "ABI_VERSION", | 
|  | "BAZEL_COMPILER", | 
|  | "BAZEL_HOST_SYSTEM", | 
|  | "BAZEL_CXXOPTS", | 
|  | "BAZEL_LINKOPTS", | 
|  | "BAZEL_LINKLIBS", | 
|  | "BAZEL_LLVM_COV", | 
|  | "BAZEL_PYTHON", | 
|  | "BAZEL_SH", | 
|  | "BAZEL_TARGET_CPU", | 
|  | "BAZEL_TARGET_LIBC", | 
|  | "BAZEL_TARGET_SYSTEM", | 
|  | "BAZEL_USE_CPP_ONLY_TOOLCHAIN", | 
|  | "BAZEL_USE_XCODE_TOOLCHAIN", | 
|  | "BAZEL_DO_NOT_DETECT_CPP_TOOLCHAIN", | 
|  | "BAZEL_USE_LLVM_NATIVE_COVERAGE", | 
|  | "BAZEL_LLVM", | 
|  | "BAZEL_IGNORE_SYSTEM_HEADERS_VERSIONS", | 
|  | "USE_CLANG_CL", | 
|  | "CC", | 
|  | "CC_CONFIGURE_DEBUG", | 
|  | "CC_TOOLCHAIN_NAME", | 
|  | "CPLUS_INCLUDE_PATH", | 
|  | "DEVELOPER_DIR", | 
|  | "GCOV", | 
|  | "HOMEBREW_RUBY_PATH", | 
|  | "SYSTEMROOT", | 
|  | "USER", | 
|  | ] + MSVC_ENVVARS, | 
|  | implementation = cc_autoconf_impl, | 
|  | configure = True, | 
|  | ) | 
|  |  | 
|  | def cc_configure(): | 
|  | """A C++ configuration rules that generate the crosstool file.""" | 
|  | cc_autoconf_toolchains(name = "local_config_cc_toolchains") | 
|  | cc_autoconf(name = "local_config_cc") | 
|  | native.bind(name = "cc_toolchain", actual = "@local_config_cc//:toolchain") | 
|  | native.register_toolchains( | 
|  | # Use register_toolchain's target pattern expansion to register all toolchains in the package. | 
|  | "@local_config_cc_toolchains//:all", | 
|  | ) |