Skip to content

Commit

Permalink
Make lto toolchain options public and don't link proc-macros with LTO (
Browse files Browse the repository at this point in the history
…#3147)

I would like to make lto setting public in toolchain so that when we are
providing custom toolchains it would be easier to make configurations
for LTO with `select` statements.

The logic which determines the flags based on cargo outputs.

Proc-macro crates can not be linked with lto and we should not emit
bitcode either. This fixes #3143

---------

Co-authored-by: Daniel Wagner-Hall <[email protected]>
  • Loading branch information
havasd and illicitonion authored Jan 2, 2025
1 parent c0fe933 commit 568bb7b
Show file tree
Hide file tree
Showing 4 changed files with 38 additions and 13 deletions.
6 changes: 4 additions & 2 deletions docs/src/rust_toolchains.md
Original file line number Diff line number Diff line change
Expand Up @@ -46,8 +46,9 @@ rust_toolchain(<a href="#rust_toolchain-name">name</a>, <a href="#rust_toolchain
<a href="#rust_toolchain-default_edition">default_edition</a>, <a href="#rust_toolchain-dylib_ext">dylib_ext</a>, <a href="#rust_toolchain-env">env</a>, <a href="#rust_toolchain-exec_triple">exec_triple</a>, <a href="#rust_toolchain-experimental_link_std_dylib">experimental_link_std_dylib</a>,
<a href="#rust_toolchain-experimental_use_cc_common_link">experimental_use_cc_common_link</a>, <a href="#rust_toolchain-extra_exec_rustc_flags">extra_exec_rustc_flags</a>, <a href="#rust_toolchain-extra_rustc_flags">extra_rustc_flags</a>,
<a href="#rust_toolchain-extra_rustc_flags_for_crate_types">extra_rustc_flags_for_crate_types</a>, <a href="#rust_toolchain-global_allocator_library">global_allocator_library</a>, <a href="#rust_toolchain-llvm_cov">llvm_cov</a>, <a href="#rust_toolchain-llvm_profdata">llvm_profdata</a>,
<a href="#rust_toolchain-llvm_tools">llvm_tools</a>, <a href="#rust_toolchain-opt_level">opt_level</a>, <a href="#rust_toolchain-per_crate_rustc_flags">per_crate_rustc_flags</a>, <a href="#rust_toolchain-rust_doc">rust_doc</a>, <a href="#rust_toolchain-rust_std">rust_std</a>, <a href="#rust_toolchain-rustc">rustc</a>, <a href="#rust_toolchain-rustc_lib">rustc_lib</a>,
<a href="#rust_toolchain-rustfmt">rustfmt</a>, <a href="#rust_toolchain-staticlib_ext">staticlib_ext</a>, <a href="#rust_toolchain-stdlib_linkflags">stdlib_linkflags</a>, <a href="#rust_toolchain-strip_level">strip_level</a>, <a href="#rust_toolchain-target_json">target_json</a>, <a href="#rust_toolchain-target_triple">target_triple</a>)
<a href="#rust_toolchain-llvm_tools">llvm_tools</a>, <a href="#rust_toolchain-lto">lto</a>, <a href="#rust_toolchain-opt_level">opt_level</a>, <a href="#rust_toolchain-per_crate_rustc_flags">per_crate_rustc_flags</a>, <a href="#rust_toolchain-rust_doc">rust_doc</a>, <a href="#rust_toolchain-rust_std">rust_std</a>, <a href="#rust_toolchain-rustc">rustc</a>,
<a href="#rust_toolchain-rustc_lib">rustc_lib</a>, <a href="#rust_toolchain-rustfmt">rustfmt</a>, <a href="#rust_toolchain-staticlib_ext">staticlib_ext</a>, <a href="#rust_toolchain-stdlib_linkflags">stdlib_linkflags</a>, <a href="#rust_toolchain-strip_level">strip_level</a>, <a href="#rust_toolchain-target_json">target_json</a>,
<a href="#rust_toolchain-target_triple">target_triple</a>)
</pre>

Declares a Rust toolchain for use.
Expand Down Expand Up @@ -118,6 +119,7 @@ See `@rules_rust//rust:repositories.bzl` for examples of defining the `@rust_cpu
| <a id="rust_toolchain-llvm_cov"></a>llvm_cov | The location of the `llvm-cov` binary. Can be a direct source or a filegroup containing one item. If None, rust code is not instrumented for coverage. | <a href="https://bazel.build/concepts/labels">Label</a> | optional | `None` |
| <a id="rust_toolchain-llvm_profdata"></a>llvm_profdata | The location of the `llvm-profdata` binary. Can be a direct source or a filegroup containing one item. If `llvm_cov` is None, this can be None as well and rust code is not instrumented for coverage. | <a href="https://bazel.build/concepts/labels">Label</a> | optional | `None` |
| <a id="rust_toolchain-llvm_tools"></a>llvm_tools | LLVM tools that are shipped with the Rust toolchain. | <a href="https://bazel.build/concepts/labels">Label</a> | optional | `None` |
| <a id="rust_toolchain-lto"></a>lto | Label to an LTO setting whether which can enable custom LTO settings | <a href="https://bazel.build/concepts/labels">Label</a> | optional | `"@rules_rust//rust/settings:lto"` |
| <a id="rust_toolchain-opt_level"></a>opt_level | Rustc optimization levels. | <a href="https://bazel.build/rules/lib/dict">Dictionary: String -> String</a> | optional | `{"dbg": "0", "fastbuild": "0", "opt": "3"}` |
| <a id="rust_toolchain-per_crate_rustc_flags"></a>per_crate_rustc_flags | Extra flags to pass to rustc in non-exec configuration | List of strings | optional | `[]` |
| <a id="rust_toolchain-rust_doc"></a>rust_doc | The location of the `rustdoc` binary. Can be a direct source or a filegroup containing one item. | <a href="https://bazel.build/concepts/labels">Label</a> | required | |
Expand Down
10 changes: 6 additions & 4 deletions rust/private/lto.bzl
Original file line number Diff line number Diff line change
Expand Up @@ -60,7 +60,7 @@ def _determine_lto_object_format(ctx, toolchain, crate_info):
if is_exec_configuration(ctx):
return "only_object"

mode = toolchain._lto.mode
mode = toolchain.lto.mode

if mode in ["off", "unspecified"]:
return "only_object"
Expand All @@ -75,9 +75,10 @@ def _determine_lto_object_format(ctx, toolchain, crate_info):
# If we're building an 'rlib' and LTO is enabled, then we can skip
# generating object files entirely.
return "only_bitcode"
elif crate_info.type == "dylib":
elif crate_info.type in ["dylib", "proc-macro"]:
# If we're a dylib and we're running LTO, then only emit object code
# because 'rustc' doesn't currently support LTO with dylibs.
# proc-macros do not benefit from LTO, and cannot be dynamically linked with LTO.
return "only_object"
else:
return "object_and_bitcode"
Expand All @@ -93,7 +94,7 @@ def construct_lto_arguments(ctx, toolchain, crate_info):
Returns:
list: A list of strings that are valid flags for 'rustc'.
"""
mode = toolchain._lto.mode
mode = toolchain.lto.mode

# The user is handling LTO on their own, don't add any arguments.
if mode == "manual":
Expand All @@ -102,7 +103,8 @@ def construct_lto_arguments(ctx, toolchain, crate_info):
format = _determine_lto_object_format(ctx, toolchain, crate_info)
args = []

if mode in ["thin", "fat", "off"] and not is_exec_configuration(ctx):
# proc-macros do not benefit from LTO, and cannot be dynamically linked with LTO.
if mode in ["thin", "fat", "off"] and not is_exec_configuration(ctx) and crate_info.type != "proc-macro":
args.append("lto={}".format(mode))

if format == "object_and_bitcode":
Expand Down
13 changes: 7 additions & 6 deletions rust/toolchain.bzl
Original file line number Diff line number Diff line change
Expand Up @@ -521,7 +521,7 @@ def _rust_toolchain_impl(ctx):
third_party_dir = ctx.attr._third_party_dir[BuildSettingInfo].value
pipelined_compilation = ctx.attr._pipelined_compilation[BuildSettingInfo].value
no_std = ctx.attr._no_std[BuildSettingInfo].value
lto = ctx.attr._lto[RustLtoInfo]
lto = ctx.attr.lto[RustLtoInfo]

experimental_use_global_allocator = ctx.attr._experimental_use_global_allocator[BuildSettingInfo].value
if _experimental_use_cc_common_link(ctx):
Expand Down Expand Up @@ -672,6 +672,7 @@ def _rust_toolchain_impl(ctx):
nostd_and_global_allocator_cc_info = _make_libstd_and_allocator_ccinfo(ctx, rust_std, ctx.attr.global_allocator_library, "no_std_with_alloc"),
llvm_cov = ctx.file.llvm_cov,
llvm_profdata = ctx.file.llvm_profdata,
lto = lto,
make_variables = make_variable_info,
rust_doc = sysroot.rustdoc,
rust_std = sysroot.rust_std,
Expand Down Expand Up @@ -705,7 +706,6 @@ def _rust_toolchain_impl(ctx):
_toolchain_generated_sysroot = ctx.attr._toolchain_generated_sysroot[BuildSettingInfo].value,
_incompatible_do_not_include_data_in_compile_data = ctx.attr._incompatible_do_not_include_data_in_compile_data[IncompatibleFlagInfo].enabled,
_no_std = no_std,
_lto = lto,
_codegen_units = ctx.attr._codegen_units[BuildSettingInfo].value,
)
return [
Expand Down Expand Up @@ -808,6 +808,11 @@ rust_toolchain = rule(
doc = "LLVM tools that are shipped with the Rust toolchain.",
allow_files = True,
),
"lto": attr.label(
providers = [RustLtoInfo],
default = Label("//rust/settings:lto"),
doc = "Label to an LTO setting whether which can enable custom LTO settings",
),
"opt_level": attr.string_dict(
doc = "Rustc optimization levels.",
default = {
Expand Down Expand Up @@ -897,10 +902,6 @@ rust_toolchain = rule(
default = Label("//rust/settings:incompatible_do_not_include_data_in_compile_data"),
doc = "Label to a boolean build setting that controls whether to include data files in compile_data.",
),
"_lto": attr.label(
providers = [RustLtoInfo],
default = Label("//rust/settings:lto"),
),
"_no_std": attr.label(
default = Label("//rust/settings:no_std"),
),
Expand Down
22 changes: 21 additions & 1 deletion test/unit/lto/lto_test_suite.bzl
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@

load("@bazel_skylib//lib:unittest.bzl", "analysistest")
load("@bazel_skylib//rules:write_file.bzl", "write_file")
load("//rust:defs.bzl", "rust_library")
load("//rust:defs.bzl", "rust_library", "rust_proc_macro")
load(
"//test/unit:common.bzl",
"assert_action_mnemonic",
Expand Down Expand Up @@ -78,6 +78,14 @@ _lto_level_fat_test = analysistest.make(
config_settings = {str(Label("//rust/settings:lto")): "fat"},
)

def _lto_proc_macro(ctx):
return _lto_test_impl(ctx, None, "no", False)

_lto_proc_macro_test = analysistest.make(
_lto_proc_macro,
config_settings = {str(Label("//rust/settings:lto")): "thin"},
)

def lto_test_suite(name):
"""Entry-point macro called from the BUILD file.
Expand All @@ -100,6 +108,12 @@ def lto_test_suite(name):
edition = "2021",
)

rust_proc_macro(
name = "proc_macro",
srcs = [":lib.rs"],
edition = "2021",
)

_lto_level_default_test(
name = "lto_level_default_test",
target_under_test = ":lib",
Expand All @@ -125,6 +139,11 @@ def lto_test_suite(name):
target_under_test = ":lib",
)

_lto_proc_macro_test(
name = "lto_proc_macro_test",
target_under_test = ":proc_macro",
)

native.test_suite(
name = name,
tests = [
Expand All @@ -133,5 +152,6 @@ def lto_test_suite(name):
":lto_level_off_test",
":lto_level_thin_test",
":lto_level_fat_test",
":lto_proc_macro_test",
],
)

0 comments on commit 568bb7b

Please sign in to comment.