| # pylint: disable=g-bad-file-header |
| # 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. |
| """Configuring the C++ toolchain on macOS.""" |
| |
| load("@bazel_tools//tools/osx:xcode_configure.bzl", "run_xcode_locator") |
| load( |
| "@bazel_tools//tools/cpp:lib_cc_configure.bzl", |
| "escape_string", |
| "resolve_labels", |
| ) |
| load( |
| "@bazel_tools//tools/cpp:unix_cc_configure.bzl", |
| "configure_unix_toolchain", |
| "get_env", |
| ) |
| |
| def _get_escaped_xcode_cxx_inc_directories(repository_ctx, cc, xcode_toolchains): |
| """Compute the list of default C++ include paths on Xcode-enabled darwin. |
| |
| Args: |
| repository_ctx: The repository context. |
| cc: The default C++ compiler on the local system. |
| xcode_toolchains: A list containing the xcode toolchains available |
| Returns: |
| include_paths: A list of builtin include paths. |
| """ |
| |
| # Assume that everything is managed by Xcode / toolchain installations |
| include_dirs = [ |
| "/Applications/", |
| "/Library/", |
| ] |
| |
| user = repository_ctx.os.environ.get("USER") |
| if user: |
| include_dirs.append("/Users/{}/Library/".format(user)) |
| |
| # Include extra Xcode paths in case they're installed on other volumes |
| for toolchain in xcode_toolchains: |
| include_dirs.append(escape_string(toolchain.developer_dir)) |
| |
| return include_dirs |
| |
| # TODO: Remove once Xcode 12 is the minimum supported version |
| def _compile_cc_file_single_arch(repository_ctx, src_name, out_name): |
| env = repository_ctx.os.environ |
| xcrun_result = repository_ctx.execute([ |
| "env", |
| "-i", |
| "DEVELOPER_DIR={}".format(env.get("DEVELOPER_DIR", default = "")), |
| "xcrun", |
| "--sdk", |
| "macosx", |
| "clang", |
| "-mmacosx-version-min=10.9", |
| "-std=c++11", |
| "-lc++", |
| "-O3", |
| "-o", |
| out_name, |
| src_name, |
| ], 60) |
| if (xcrun_result.return_code != 0): |
| error_msg = ( |
| "return code {code}, stderr: {err}, stdout: {out}" |
| ).format( |
| code = xcrun_result.return_code, |
| err = xcrun_result.stderr, |
| out = xcrun_result.stdout, |
| ) |
| fail(out_name + " failed to generate. Please file an issue at " + |
| "https://github.com/bazelbuild/bazel/issues with the following:\n" + |
| error_msg) |
| |
| def _compile_cc_file(repository_ctx, src_name, out_name): |
| env = repository_ctx.os.environ |
| xcrun_result = repository_ctx.execute([ |
| "env", |
| "-i", |
| "DEVELOPER_DIR={}".format(env.get("DEVELOPER_DIR", default = "")), |
| "xcrun", |
| "--sdk", |
| "macosx", |
| "clang", |
| "-mmacosx-version-min=10.9", |
| "-std=c++11", |
| "-lc++", |
| "-arch", |
| "arm64", |
| "-arch", |
| "x86_64", |
| "-Wl,-no_adhoc_codesign", |
| "-Wl,-no_uuid", |
| "-O3", |
| "-o", |
| out_name, |
| src_name, |
| ], 60) |
| |
| if xcrun_result.return_code == 0: |
| xcrun_result = repository_ctx.execute([ |
| "env", |
| "-i", |
| "codesign", |
| "--identifier", # Required to be reproducible across archs |
| out_name, |
| "--force", |
| "--sign", |
| "-", |
| out_name, |
| ], 60) |
| if xcrun_result.return_code != 0: |
| error_msg = ( |
| "codesign return code {code}, stderr: {err}, stdout: {out}" |
| ).format( |
| code = xcrun_result.return_code, |
| err = xcrun_result.stderr, |
| out = xcrun_result.stdout, |
| ) |
| fail(out_name + " failed to generate. Please file an issue at " + |
| "https://github.com/bazelbuild/bazel/issues with the following:\n" + |
| error_msg) |
| else: |
| _compile_cc_file_single_arch(repository_ctx, src_name, out_name) |
| |
| def configure_osx_toolchain(repository_ctx, cpu_value, overriden_tools): |
| """Configure C++ toolchain on macOS. |
| |
| Args: |
| repository_ctx: The repository context. |
| overriden_tools: dictionary of overridden tools. |
| """ |
| paths = resolve_labels(repository_ctx, [ |
| "@bazel_tools//tools/cpp:armeabi_cc_toolchain_config.bzl", |
| "@bazel_tools//tools/cpp:osx_cc_wrapper.sh.tpl", |
| "@bazel_tools//tools/objc:libtool.sh", |
| "@bazel_tools//tools/objc:libtool_check_unique.cc", |
| "@bazel_tools//tools/objc:make_hashed_objlist.py", |
| "@bazel_tools//tools/objc:xcrunwrapper.sh", |
| "@bazel_tools//tools/osx/crosstool:BUILD.tpl", |
| "@bazel_tools//tools/osx/crosstool:cc_toolchain_config.bzl", |
| "@bazel_tools//tools/osx/crosstool:wrapped_clang.cc", |
| "@bazel_tools//tools/osx:xcode_locator.m", |
| ]) |
| |
| env = repository_ctx.os.environ |
| should_use_xcode = "BAZEL_USE_XCODE_TOOLCHAIN" in env and env["BAZEL_USE_XCODE_TOOLCHAIN"] == "1" |
| xcode_toolchains = [] |
| |
| # Make the following logic in sync with //tools/cpp:cc_configure.bzl#cc_autoconf_toolchains_impl |
| (xcode_toolchains, xcodeloc_err) = run_xcode_locator( |
| repository_ctx, |
| paths["@bazel_tools//tools/osx:xcode_locator.m"], |
| ) |
| if should_use_xcode and not xcode_toolchains: |
| fail("BAZEL_USE_XCODE_TOOLCHAIN is set to 1 but Bazel couldn't find Xcode installed on the " + |
| "system. Verify that 'xcode-select -p' is correct.") |
| if xcode_toolchains: |
| # For Xcode toolchains, there's no reason to use anything other than |
| # wrapped_clang, so that we still get the Bazel Xcode placeholder |
| # substitution and other behavior for actions that invoke this |
| # cc_wrapper.sh script. The wrapped_clang binary is already hardcoded |
| # into the Objective-C crosstool actions, anyway, so this ensures that |
| # the C++ actions behave consistently. |
| cc = repository_ctx.path("wrapped_clang") |
| |
| cc_path = '"$(/usr/bin/dirname "$0")"/wrapped_clang' |
| repository_ctx.template( |
| "cc_wrapper.sh", |
| paths["@bazel_tools//tools/cpp:osx_cc_wrapper.sh.tpl"], |
| { |
| "%{cc}": escape_string(cc_path), |
| "%{env}": escape_string(get_env(repository_ctx)), |
| }, |
| ) |
| repository_ctx.symlink( |
| paths["@bazel_tools//tools/cpp:armeabi_cc_toolchain_config.bzl"], |
| "armeabi_cc_toolchain_config.bzl", |
| ) |
| repository_ctx.symlink( |
| paths["@bazel_tools//tools/objc:xcrunwrapper.sh"], |
| "xcrunwrapper.sh", |
| ) |
| repository_ctx.symlink( |
| paths["@bazel_tools//tools/objc:libtool.sh"], |
| "libtool", |
| ) |
| repository_ctx.symlink( |
| paths["@bazel_tools//tools/objc:make_hashed_objlist.py"], |
| "make_hashed_objlist.py", |
| ) |
| repository_ctx.symlink( |
| paths["@bazel_tools//tools/osx/crosstool:cc_toolchain_config.bzl"], |
| "cc_toolchain_config.bzl", |
| ) |
| libtool_check_unique_src_path = str(repository_ctx.path( |
| paths["@bazel_tools//tools/objc:libtool_check_unique.cc"], |
| )) |
| _compile_cc_file(repository_ctx, libtool_check_unique_src_path, "libtool_check_unique") |
| wrapped_clang_src_path = str(repository_ctx.path( |
| paths["@bazel_tools//tools/osx/crosstool:wrapped_clang.cc"], |
| )) |
| _compile_cc_file(repository_ctx, wrapped_clang_src_path, "wrapped_clang") |
| repository_ctx.symlink("wrapped_clang", "wrapped_clang_pp") |
| |
| tool_paths = {} |
| gcov_path = repository_ctx.os.environ.get("GCOV") |
| if gcov_path != None: |
| if not gcov_path.startswith("/"): |
| gcov_path = repository_ctx.which(gcov_path) |
| tool_paths["gcov"] = gcov_path |
| |
| escaped_include_paths = _get_escaped_xcode_cxx_inc_directories(repository_ctx, cc, xcode_toolchains) |
| escaped_cxx_include_directories = [] |
| for path in escaped_include_paths: |
| escaped_cxx_include_directories.append((" \"%s\"," % path)) |
| if xcodeloc_err: |
| escaped_cxx_include_directories.append(" # Error: " + xcodeloc_err) |
| repository_ctx.template( |
| "BUILD", |
| paths["@bazel_tools//tools/osx/crosstool:BUILD.tpl"], |
| { |
| "%{cxx_builtin_include_directories}": "\n".join(escaped_cxx_include_directories), |
| "%{tool_paths_overrides}": ",\n ".join( |
| ['"%s": "%s"' % (k, v) for k, v in tool_paths.items()], |
| ), |
| }, |
| ) |
| else: |
| configure_unix_toolchain(repository_ctx, cpu_value, overriden_tools = overriden_tools) |