14
14
15
15
# buildifier: disable=module-docstring
16
16
load ("@bazel_skylib//lib:paths.bzl" , "paths" )
17
+ load ("@bazel_skylib//rules:common_settings.bzl" , "BuildSettingInfo" )
17
18
load (
18
19
"@bazel_tools//tools/build_defs/cc:action_names.bzl" ,
19
20
"CPP_LINK_EXECUTABLE_ACTION_NAME" ,
@@ -50,6 +51,22 @@ AliasableDepInfo = provider(
50
51
},
51
52
)
52
53
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
+
53
70
_error_format_values = ["human" , "json" , "short" ]
54
71
55
72
ErrorFormatInfo = provider (
@@ -537,7 +554,8 @@ def rustc_compile_action(
537
554
crate_info ,
538
555
output_hash = None ,
539
556
rust_flags = [],
540
- environ = {}):
557
+ environ = {},
558
+ incremental_info = None ):
541
559
"""Create and run a rustc compile action based on the current rule's attributes
542
560
543
561
Args:
@@ -548,6 +566,7 @@ def rustc_compile_action(
548
566
output_hash (str, optional): The hashed path of the crate root. Defaults to None.
549
567
rust_flags (list, optional): Additional flags to pass to rustc. Defaults to [].
550
568
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.
551
570
552
571
Returns:
553
572
list: A list of the following providers:
@@ -597,14 +616,34 @@ def rustc_compile_action(
597
616
else :
598
617
formatted_version = ""
599
618
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
+
600
637
ctx .actions .run (
601
638
executable = ctx .executable ._process_wrapper ,
602
639
inputs = compile_inputs ,
603
640
outputs = [crate_info .output ],
604
641
env = env ,
605
642
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 ,
608
647
crate_info .type ,
609
648
ctx .label .name ,
610
649
formatted_version ,
@@ -1022,3 +1061,75 @@ error_format = rule(
1022
1061
implementation = _error_format_impl ,
1023
1062
build_setting = config .string (flag = True ),
1024
1063
)
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