| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202 |
- # Part of the Carbon Language project, under the Apache License v2.0 with LLVM
- # Exceptions. See /LICENSE for license information.
- # SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- """Provides rules for building Carbon files using the toolchain."""
- load("@rules_cc//cc/common:cc_info.bzl", "CcInfo")
- def _runtimes_path(runtimes_target):
- path = None
- for f in runtimes_target:
- if f.short_path.endswith("clang_resource_dir/lib"):
- path = f.path
- break
- if not path:
- fail("Could not find the `clang_resource_dir` in target {}".format(runtimes_target.label))
- return path[:-len("/clang_resource_dir/lib")]
- def _carbon_binary_impl(ctx):
- toolchain_driver = ctx.executable.internal_exec_toolchain_driver
- toolchain_data = ctx.files.internal_exec_toolchain_data
- prebuilt_runtimes = ctx.files.internal_exec_prebuilt_runtimes
- # If the exec driver isn't provided, that means we're trying to use a target
- # config toolchain, likely to avoid build overhead of two configs.
- if toolchain_driver == None:
- toolchain_driver = ctx.executable.internal_target_toolchain_driver
- toolchain_data = ctx.files.internal_target_toolchain_data
- prebuilt_runtimes = ctx.files.internal_target_prebuilt_runtimes
- # Pass any C++ flags from our dependencies onto Carbon.
- dep_flags = []
- dep_hdrs = []
- dep_link_flags = []
- dep_link_inputs = []
- for dep in ctx.attr.deps:
- if CcInfo in dep:
- cc_info = dep[CcInfo]
- # TODO: We should reuse the feature-based flag generation in
- # bazel/cc_toolchains here.
- dep_flags += ["--clang-arg=-D{0}".format(define) for define in cc_info.compilation_context.defines.to_list()]
- dep_flags += ["--clang-arg=-I{0}".format(path) for path in cc_info.compilation_context.includes.to_list()]
- dep_flags += ["--clang-arg=-iquote{0}".format(path) for path in cc_info.compilation_context.quote_includes.to_list()]
- dep_flags += ["--clang-arg=-isystem{0}".format(path) for path in cc_info.compilation_context.system_includes.to_list()]
- dep_hdrs.append(cc_info.compilation_context.headers)
- for link_input in cc_info.linking_context.linker_inputs.to_list():
- # TODO: `carbon link` doesn't support linker flags yet.
- # dep_link_flags += link_input.user_link_flags
- dep_link_inputs += link_input.additional_inputs
- for lib in link_input.libraries:
- dep_link_inputs += [dep for dep in [lib.dynamic_library, lib.static_library] if dep]
- dep_link_inputs += lib.objects
- if DefaultInfo in dep:
- dep_link_inputs += dep[DefaultInfo].files.to_list()
- dep_link_flags += [dep.path for dep in dep_link_inputs]
- # Build object files for the prelude and for the binary itself.
- # TODO: Eventually the prelude should be build as a separate `carbon_library`.
- srcs_and_flags = [
- (ctx.files.prelude_srcs, ["--no-prelude-import"]),
- (ctx.files.srcs, dep_flags),
- ]
- objs = []
- for (srcs, extra_flags) in srcs_and_flags:
- for src in srcs:
- # Build each source file. For now, we pass all sources to each compile
- # because we don't have visibility into dependencies and have no way to
- # specify multiple output files. Object code for each input is written
- # into the output file in turn, so the final carbon source file
- # specified ends up determining the contents of the object file.
- #
- # TODO: This is a hack; replace with something better once the toolchain
- # supports doing so.
- #
- # TODO: Switch to the `prefix` based rule similar to linking when
- # the prelude moves there.
- out = ctx.actions.declare_file("_objs/{0}/{1}o".format(
- ctx.label.name,
- src.short_path.removeprefix(ctx.label.package).removesuffix(src.extension),
- ))
- objs.append(out)
- srcs_reordered = [s for s in srcs if s != src] + [src]
- ctx.actions.run(
- outputs = [out],
- inputs = depset(direct = srcs_reordered, transitive = dep_hdrs),
- executable = toolchain_driver,
- tools = depset(toolchain_data),
- arguments = ["compile", "--output=" + out.path] +
- [s.path for s in srcs_reordered] + extra_flags + ctx.attr.flags,
- mnemonic = "CarbonCompile",
- progress_message = "Compiling " + src.short_path,
- )
- bin = ctx.actions.declare_file(ctx.label.name)
- ctx.actions.run(
- outputs = [bin],
- inputs = objs + dep_link_inputs,
- executable = toolchain_driver,
- tools = depset(toolchain_data + prebuilt_runtimes),
- arguments = ["--prebuilt-runtimes=" + _runtimes_path(prebuilt_runtimes), "link", "--output=" + bin.path] + ["--"] + dep_link_flags + [o.path for o in objs],
- mnemonic = "CarbonLink",
- progress_message = "Linking " + bin.short_path,
- )
- return [DefaultInfo(files = depset([bin]), executable = bin)]
- _carbon_binary_internal = rule(
- implementation = _carbon_binary_impl,
- attrs = {
- "deps": attr.label_list(allow_files = True, providers = [[CcInfo]]),
- "flags": attr.string_list(),
- # The exec config toolchain attributes. These will be `None` when using
- # the target config and populated when using the exec config. We have to
- # use duplicate attributes here and below to have different `cfg`
- # settings, as that isn't `select`-able, and we'll use `select`s when
- # populating these.
- "internal_exec_prebuilt_runtimes": attr.label(
- cfg = "exec",
- ),
- "internal_exec_toolchain_data": attr.label(
- cfg = "exec",
- ),
- "internal_exec_toolchain_driver": attr.label(
- allow_single_file = True,
- executable = True,
- cfg = "exec",
- ),
- # The target config toolchain attributes. These will be 'None' when
- # using the exec config and populated when using the target config. We
- # have to use duplicate attributes here and below to have different
- # `cfg` settings, as that isn't `select`-able, and we'll use `select`s
- # when populating these.
- "internal_target_prebuilt_runtimes": attr.label(
- cfg = "target",
- ),
- "internal_target_toolchain_data": attr.label(
- cfg = "target",
- ),
- "internal_target_toolchain_driver": attr.label(
- allow_single_file = True,
- executable = True,
- cfg = "target",
- ),
- "prelude_srcs": attr.label_list(allow_files = [".carbon"]),
- "srcs": attr.label_list(allow_files = [".carbon"]),
- "_cc_toolchain": attr.label(default = "@bazel_tools//tools/cpp:current_cc_toolchain"),
- },
- executable = True,
- )
- def carbon_binary(name, srcs, deps = [], flags = [], tags = []):
- """Compiles a Carbon binary.
- Args:
- name: The name of the build target.
- srcs: List of Carbon source files to compile.
- deps: List of dependencies.
- flags: Extra flags to pass to the Carbon compile command.
- tags: Tags to apply to the rule.
- """
- _carbon_binary_internal(
- name = name,
- srcs = srcs,
- prelude_srcs = ["//core:prelude_files"],
- deps = deps,
- flags = flags,
- tags = tags,
- # We synthesize two sets of attributes from mirrored `select`s here
- # because we want to select on an internal property of these attributes
- # but that isn't `select`-able. Instead, we have both attributes and
- # `select` which one we use.
- internal_exec_toolchain_driver = select({
- "//bazel/carbon_rules:use_target_config_carbon_rules_config": None,
- "//conditions:default": "//toolchain/install:prefix/bin/carbon",
- }),
- internal_exec_toolchain_data = select({
- "//bazel/carbon_rules:use_target_config_carbon_rules_config": None,
- "//conditions:default": "//toolchain/install:install_data",
- }),
- internal_exec_prebuilt_runtimes = select({
- "//bazel/carbon_rules:use_target_config_carbon_rules_config": None,
- "//conditions:default": "//toolchain/driver:prebuilt_runtimes",
- }),
- internal_target_toolchain_driver = select({
- "//bazel/carbon_rules:use_target_config_carbon_rules_config": "//toolchain/install:prefix/bin/carbon",
- "//conditions:default": None,
- }),
- internal_target_toolchain_data = select({
- "//bazel/carbon_rules:use_target_config_carbon_rules_config": "//toolchain/install:install_data",
- "//conditions:default": None,
- }),
- internal_target_prebuilt_runtimes = select({
- "//bazel/carbon_rules:use_target_config_carbon_rules_config": "//toolchain/driver:prebuilt_runtimes",
- "//conditions:default": None,
- }),
- )
|