Skip to content

Commit 9577c14

Browse files
daevmax
authored andcommitted
incremental compilation without a worker
This is considerably less invasive than bazelbuild#667, bazelbuild#517 and bazelbuild#421 - there is no extra code to bootstrap, and no toolchains have to be plumbed into the rules. We had been using a worker to get around sandboxing restrictions, but as @hlopko pointed out on bazelbuild#667, that turns out not to be necessary - even normal rules have access to /tmp. Unfortunately it seems that rustc does not expect the source files to change location, and it will consistently crash when used in a sandboxed context. So the approach this PR takes is to disable sandboxing on targets that are being compiled incrementally. This fixes the compiler crashes, and as a bonus, means we're not limited to saving the cache in /tmp. This PR adds a --@rules_rust//:experimental_incremental_base flag to specify the path where incremental build products are stored - if not provided, the rules will function as they normally do. The default behaviour will incrementally compile crates in the local workspace, like cargo does. The behaviour can be adjusted with another flag, which is covered in docs/index.md.
1 parent d2bf64f commit 9577c14

File tree

6 files changed

+193
-5
lines changed

6 files changed

+193
-5
lines changed

BUILD.bazel

+19-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,10 @@
11
load("@bazel_skylib//:bzl_library.bzl", "bzl_library")
2-
load("//rust:rust.bzl", "error_format")
2+
load(
3+
"//rust:defs.bzl",
4+
"error_format",
5+
"experimental_incremental_base",
6+
"experimental_incremental_prefixes",
7+
)
38

49
exports_files(["LICENSE"])
510

@@ -37,3 +42,16 @@ alias(
3742
actual = "//tools/rustfmt",
3843
visibility = ["//visibility:public"],
3944
)
45+
46+
# Optional incremental compilation - see docs/index.md
47+
experimental_incremental_base(
48+
name = "experimental_incremental_base",
49+
build_setting_default = "",
50+
visibility = ["//visibility:public"],
51+
)
52+
53+
experimental_incremental_prefixes(
54+
name = "experimental_incremental_prefixes",
55+
build_setting_default = "//,-//vendored//",
56+
visibility = ["//visibility:public"],
57+
)

docs/BUILD.bazel

+1
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@ package(default_visibility = ["//visibility:private"])
77
bzl_library(
88
name = "docs_deps",
99
srcs = [
10+
"@bazel_skylib//rules:common_settings",
1011
"@bazel_tools//tools:bzl_srcs",
1112
"@build_bazel_rules_nodejs//internal/providers:bzl",
1213
"@com_google_protobuf//:bzl_srcs",

docs/index.md

+26
Original file line numberDiff line numberDiff line change
@@ -81,3 +81,29 @@ rust_repositories(rustfmt_version = "1.53.0")
8181

8282
Currently, the most common approach to managing external dependencies is using
8383
[cargo-raze](https://github.com/google/cargo-raze) to generate `BUILD` files for Cargo crates.
84+
85+
86+
## Incremental Compilation
87+
88+
There is an experimental flag that enables incremental compilation, which can
89+
considerably speed up rebuilds during development.
90+
91+
Targets built with incremental compilation enabled will have sandboxing
92+
disabled, so enabling this option is trading off some of Bazel's hermeticity in
93+
the name of performance. This option is not recommended for CI or release
94+
builds.
95+
96+
To enable incremental compilation, add the following argument to your bazel
97+
build command, adjusting the directory path to one that suits:
98+
99+
--@rules_rust//:experimental_incremental_base=/home/user/cache/bazel_incremental
100+
101+
A separate flag allows you to control which crates are incrementally compiled.
102+
The default is:
103+
104+
--@rules_rust//:experimental_incremental_prefixes=//,-//vendored
105+
106+
This will incrementally compile any crates in the local workspace, but exclude
107+
other repositories like `@raze__*`, and anything in the //vendored package. This
108+
behaviour is similar to cargo, which only incrementally compiles the local
109+
workspace.

rust/defs.bzl

+8
Original file line numberDiff line numberDiff line change
@@ -43,6 +43,8 @@ load(
4343
load(
4444
"//rust/private:rustc.bzl",
4545
_error_format = "error_format",
46+
_incremental_base = "incremental_base",
47+
_incremental_prefixes = "incremental_prefixes",
4648
)
4749
load(
4850
"//rust/private:rustdoc.bzl",
@@ -97,6 +99,12 @@ rust_clippy = _rust_clippy
9799
error_format = _error_format
98100
# See @rules_rust//rust/private:rustc.bzl for a complete description.
99101

102+
experimental_incremental_base = _incremental_base
103+
# See @rules_rust//rust/private:rustc.bzl for a complete description.
104+
105+
experimental_incremental_prefixes = _incremental_prefixes
106+
# See @rules_rust//rust/private:rustc.bzl for a complete description.
107+
100108
rust_common = _rust_common
101109
# See @rules_rust//rust/private:common.bzl for a complete description.
102110

rust/private/rust.bzl

+25-1
Original file line numberDiff line numberDiff line change
@@ -13,8 +13,8 @@
1313
# limitations under the License.
1414

1515
# buildifier: disable=module-docstring
16+
load("@bazel_skylib//rules:common_settings.bzl", "BuildSettingInfo")
1617
load("//rust/private:common.bzl", "rust_common")
17-
load("//rust/private:rustc.bzl", "rustc_compile_action")
1818
load(
1919
"//rust/private:utils.bzl",
2020
"crate_name_from_attr",
@@ -23,6 +23,7 @@ load(
2323
"expand_dict_value_locations",
2424
"find_toolchain",
2525
)
26+
load("//rust/private:rustc.bzl", "IncrementalInfo", "IncrementalPrefixInfo", "rustc_compile_action")
2627

2728
# TODO(marco): Separate each rule into its own file.
2829

@@ -103,6 +104,24 @@ def _determine_lib_name(name, crate_type, toolchain, lib_hash = ""):
103104
extension = extension,
104105
)
105106

107+
def incremental_info(attr):
108+
"""Extract incremental compilation info from common rule attributes.
109+
110+
Expects to find the following attributes:
111+
112+
_incremental_base
113+
_incremental_prefixes
114+
115+
Returns:
116+
IncrementalInfo: incremental compilation configuration
117+
"""
118+
prefixes = attr._incremental_prefixes[IncrementalPrefixInfo]
119+
base = attr._incremental_base[BuildSettingInfo].value
120+
return IncrementalInfo(
121+
prefixes = prefixes,
122+
base = base,
123+
)
124+
106125
def get_edition(attr, toolchain):
107126
"""Returns the Rust edition from either the current rule's attirbutes or the current `rust_toolchain`
108127
@@ -269,6 +288,7 @@ def _rust_library_common(ctx, crate_type):
269288
compile_data = depset(ctx.files.compile_data),
270289
),
271290
output_hash = output_hash,
291+
incremental_info = incremental_info(ctx.attr),
272292
)
273293

274294
def _rust_binary_impl(ctx):
@@ -304,6 +324,7 @@ def _rust_binary_impl(ctx):
304324
is_test = False,
305325
compile_data = depset(ctx.files.compile_data),
306326
),
327+
incremental_info = incremental_info(ctx.attr),
307328
)
308329

309330
def _create_test_launcher(ctx, toolchain, output, providers):
@@ -453,6 +474,7 @@ def _rust_test_common(ctx, toolchain, output):
453474
toolchain = toolchain,
454475
crate_info = crate_info,
455476
rust_flags = ["--test"] if ctx.attr.use_libtest_harness else ["--cfg", "test"],
477+
incremental_info = incremental_info(ctx.attr),
456478
)
457479

458480
return _create_test_launcher(ctx, toolchain, output, providers)
@@ -663,6 +685,8 @@ _common_attrs = {
663685
default = "@bazel_tools//tools/cpp:current_cc_toolchain",
664686
),
665687
"_error_format": attr.label(default = "//:error_format"),
688+
"_incremental_base": attr.label(default = "//:experimental_incremental_base"),
689+
"_incremental_prefixes": attr.label(default = "//:experimental_incremental_prefixes"),
666690
"_process_wrapper": attr.label(
667691
default = Label("//util/process_wrapper"),
668692
executable = True,

rust/private/rustc.bzl

+114-3
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,7 @@
1414

1515
# buildifier: disable=module-docstring
1616
load("@bazel_skylib//lib:paths.bzl", "paths")
17+
load("@bazel_skylib//rules:common_settings.bzl", "BuildSettingInfo")
1718
load(
1819
"@bazel_tools//tools/build_defs/cc:action_names.bzl",
1920
"CPP_LINK_EXECUTABLE_ACTION_NAME",
@@ -50,6 +51,22 @@ AliasableDepInfo = provider(
5051
},
5152
)
5253

54+
IncrementalInfo = provider(
55+
doc = "Data relating to incremental compilation.",
56+
fields = {
57+
"base": "string: base folder to store incremental build products",
58+
"prefixes": "IncrementalPrefixInfo: prefixes to include and exclude",
59+
},
60+
)
61+
62+
IncrementalPrefixInfo = provider(
63+
doc = "Prefixes to include and exclude in incremental compilation.",
64+
fields = {
65+
"exclude": "List[string]: prefixes that will exclude a label if matched",
66+
"include": "List[string]: prefixes that will include a label if matched",
67+
},
68+
)
69+
5370
_error_format_values = ["human", "json", "short"]
5471

5572
ErrorFormatInfo = provider(
@@ -537,7 +554,8 @@ def rustc_compile_action(
537554
crate_info,
538555
output_hash = None,
539556
rust_flags = [],
540-
environ = {}):
557+
environ = {},
558+
incremental_info = None):
541559
"""Create and run a rustc compile action based on the current rule's attributes
542560
543561
Args:
@@ -548,6 +566,7 @@ def rustc_compile_action(
548566
output_hash (str, optional): The hashed path of the crate root. Defaults to None.
549567
rust_flags (list, optional): Additional flags to pass to rustc. Defaults to [].
550568
environ (dict, optional): A set of makefile expandable environment variables for the action
569+
incremental_info (str, optional): path to store incremental build products in.
551570
552571
Returns:
553572
list: A list of the following providers:
@@ -597,14 +616,34 @@ def rustc_compile_action(
597616
else:
598617
formatted_version = ""
599618

619+
if (incremental_info and
620+
incremental_info.base and
621+
_want_incremental_compile(ctx.label, incremental_info.prefixes)):
622+
incremental_dir = "{}/{}_{}".format(
623+
incremental_info.base,
624+
ctx.var["COMPILATION_MODE"],
625+
toolchain.target_triple,
626+
)
627+
args.all.extend(["--codegen", "incremental=" + incremental_dir])
628+
629+
# with sandboxing enabled, subsequent rustc invocations will crash,
630+
# as it doesn't expect the source files to have moved
631+
execution_requirements = {"no-sandbox": "1"}
632+
mnemonic = "RustIncr"
633+
else:
634+
execution_requirements = {}
635+
mnemonic = "Rustc"
636+
600637
ctx.actions.run(
601638
executable = ctx.executable._process_wrapper,
602639
inputs = compile_inputs,
603640
outputs = [crate_info.output],
604641
env = env,
605642
arguments = args.all,
606-
mnemonic = "Rustc",
607-
progress_message = "Compiling Rust {} {}{} ({} files)".format(
643+
mnemonic = mnemonic,
644+
execution_requirements = execution_requirements,
645+
progress_message = "Compiling {} {} {}{} ({} files)".format(
646+
mnemonic,
608647
crate_info.type,
609648
ctx.label.name,
610649
formatted_version,
@@ -1022,3 +1061,75 @@ error_format = rule(
10221061
implementation = _error_format_impl,
10231062
build_setting = config.string(flag = True),
10241063
)
1064+
1065+
def _incremental_base_impl(ctx):
1066+
"""Implementation for the incremental_base() rule
1067+
1068+
Args:
1069+
ctx (ctx): The rule's context object
1070+
1071+
Returns:
1072+
BuildSettingInfo: an object with a `value` attribute containing the string.
1073+
"""
1074+
value = ctx.build_setting_value
1075+
return BuildSettingInfo(value = value)
1076+
1077+
incremental_base = rule(
1078+
build_setting = config.string(flag = True),
1079+
implementation = _incremental_base_impl,
1080+
doc = "Declares a command line argument that accepts an arbitrary string.",
1081+
)
1082+
1083+
def _incremental_prefixes_impl(ctx):
1084+
"""Implementation for the incremental_prefixes_flag() rule
1085+
1086+
Splits the provided string on commas, and then partitions prefixes starting
1087+
with a hypen into the exclude list, returning a provider with the include
1088+
and exclude list. The hypens are stripped from the entries in the exclude list.
1089+
1090+
Args:
1091+
ctx (ctx): The rule's context object
1092+
1093+
Returns:
1094+
(IncrementalPrefixInfo): a list of prefixes to include and exclude
1095+
"""
1096+
values = ctx.build_setting_value.split(",")
1097+
include = []
1098+
exclude = []
1099+
for value in ctx.build_setting_value.split(","):
1100+
if not value:
1101+
continue
1102+
elif value.startswith("-"):
1103+
exclude.append(value[1:])
1104+
else:
1105+
include.append(value)
1106+
return IncrementalPrefixInfo(include = include, exclude = exclude)
1107+
1108+
incremental_prefixes = rule(
1109+
build_setting = config.string(flag = True),
1110+
implementation = _incremental_prefixes_impl,
1111+
doc = """Declares a command line argument for incremental prefixes.
1112+
1113+
See _incremental_prefixes_impl() for the details.
1114+
""",
1115+
)
1116+
1117+
def _want_incremental_compile(label, prefixes):
1118+
"""True if the provided prefixes indicate the target should be incrementally compiled.
1119+
1120+
Args:
1121+
label (Label): the label for a given target
1122+
prefixes (IncrementalPrefixInfo): prefixes to include and exclude
1123+
1124+
Returns:
1125+
bool
1126+
"""
1127+
label = str(label)
1128+
for prefix in prefixes.exclude:
1129+
if label.startswith(prefix):
1130+
return False
1131+
for prefix in prefixes.include:
1132+
if label.startswith(prefix):
1133+
return True
1134+
1135+
return False

0 commit comments

Comments
 (0)