blob: 657226811aa299ebfd23c37730fabd63f0a33ed7 [file] [log] [blame] [view]
fweacae1cd2022-02-17 09:45:38 -08001Project: /_project.yaml
2Book: /_book.yaml
3
4# Configurable Build Attributes
5
Googler3b9ed6e2022-11-08 02:19:42 -08006{% include "_buttons.html" %}
7
fweacae1cd2022-02-17 09:45:38 -08008**_Configurable attributes_**, commonly known as [`select()`](
9/reference/be/functions#select), is a Bazel feature that lets users toggle the values
10of build rule attributes at the command line.
11
12This can be used, for example, for a multiplatform library that automatically
13chooses the appropriate implementation for the architecture, or for a
14feature-configurable binary that can be customized at build time.
15
16## Example {:#configurable-build-example}
17
18```python
19# myapp/BUILD
20
21cc_binary(
22 name = "mybinary",
23 srcs = ["main.cc"],
24 deps = select({
25 ":arm_build": [":arm_lib"],
26 ":x86_debug_build": [":x86_dev_lib"],
27 "//conditions:default": [":generic_lib"],
28 }),
29)
30
31config_setting(
32 name = "arm_build",
33 values = {"cpu": "arm"},
34)
35
36config_setting(
37 name = "x86_debug_build",
38 values = {
39 "cpu": "x86",
40 "compilation_mode": "dbg",
41 },
42)
43```
44
45This declares a `cc_binary` that "chooses" its deps based on the flags at the
Googler61211f22022-05-04 13:35:49 -070046command line. Specifically, `deps` becomes:
fweacae1cd2022-02-17 09:45:38 -080047
48<table>
49 <tr style="background: #E9E9E9; font-weight: bold">
50 <td>Command</td>
51 <td>deps =</td>
52 </tr>
53 <tr>
54 <td><code>bazel build //myapp:mybinary --cpu=arm</code></td>
55 <td><code>[":arm_lib"]</code></td>
56 </tr>
57 <tr>
58 <td><code>bazel build //myapp:mybinary -c dbg --cpu=x86</code></td>
59 <td><code>[":x86_dev_lib"]</code></td>
60 </tr>
61 <tr>
62 <td><code>bazel build //myapp:mybinary --cpu=ppc</code></td>
63 <td><code>[":generic_lib"]</code></td>
64 </tr>
65 <tr>
66 <td><code>bazel build //myapp:mybinary -c dbg --cpu=ppc</code></td>
67 <td><code>[":generic_lib"]</code></td>
68 </tr>
69</table>
70
71`select()` serves as a placeholder for a value that will be chosen based on
72*configuration conditions*, which are labels referencing [`config_setting`](/reference/be/general#config_setting)
73targets. By using `select()` in a configurable attribute, the attribute
74effectively adopts different values when different conditions hold.
75
Alessandro Pattid43737f2023-04-11 21:31:54 -070076Matches must be unambiguous: if multiple conditions match then either
77* They all resolve to the same value. For example, when running on linux x86, this is unambiguous
78 `{"@platforms//os:linux": "Hello", "@platforms//cpu:x86_64": "Hello"}` because both branches resolve to "hello".
79* One's `values` is a strict superset of all others'. For example, `values = {"cpu": "x86", "compilation_mode": "dbg"}`
80 is an unambiguous specialization of `values = {"cpu": "x86"}`.
81
82The built-in condition [`//conditions:default`](#default-condition) automatically matches when
fweacae1cd2022-02-17 09:45:38 -080083nothing else does.
84
85While this example uses `deps`, `select()` works just as well on `srcs`,
86`resources`, `cmd`, and most other attributes. Only a small number of attributes
87are *non-configurable*, and these are clearly annotated. For example,
88`config_setting`'s own
89[`values`](/reference/be/general#config_setting.values) attribute is non-configurable.
90
91## `select()` and dependencies {:#select-and-dependencies}
92
93Certain attributes change the build parameters for all transitive dependencies
94under a target. For example, `genrule`'s `tools` changes `--cpu` to the CPU of
95the machine running Bazel (which, thanks to cross-compilation, may be different
96than the CPU the target is built for). This is known as a
97[configuration transition](/reference/glossary#transition).
98
99Given
100
101```python
102#myapp/BUILD
103
104config_setting(
105 name = "arm_cpu",
106 values = {"cpu": "arm"},
107)
108
109config_setting(
110 name = "x86_cpu",
111 values = {"cpu": "x86"},
112)
113
114genrule(
115 name = "my_genrule",
116 srcs = select({
117 ":arm_cpu": ["g_arm.src"],
118 ":x86_cpu": ["g_x86.src"],
119 }),
120 tools = select({
121 ":arm_cpu": [":tool1"],
122 ":x86_cpu": [":tool2"],
123 }),
124)
125
126cc_binary(
127 name = "tool1",
128 srcs = select({
129 ":arm_cpu": ["armtool.cc"],
130 ":x86_cpu": ["x86tool.cc"],
131 }),
132)
133```
134
135running
136
137```sh
138$ bazel build //myapp:my_genrule --cpu=arm
139```
140
141on an `x86` developer machine binds the build to `g_arm.src`, `tool1`, and
142`x86tool.cc`. Both of the `select`s attached to `my_genrule` use `my_genrule`'s
143build parameters, which include `--cpu=arm`. The `tools` attribute changes
144`--cpu` to `x86` for `tool1` and its transitive dependencies. The `select` on
145`tool1` uses `tool1`'s build parameters, which include `--cpu=x86`.
146
147## Configuration conditions {:#configuration-conditions}
148
149Each key in a configurable attribute is a label reference to a
150[`config_setting`](/reference/be/general#config_setting) or
Googlerf1c2e442023-09-06 09:20:49 -0700151[`constraint_value`](/reference/be/platforms-and-toolchains#constraint_value).
fweacae1cd2022-02-17 09:45:38 -0800152
153`config_setting` is just a collection of
154expected command line flag settings. By encapsulating these in a target, it's
155easy to maintain "standard" conditions users can reference from multiple places.
156
157`constraint_value` provides support for [multi-platform behavior](#platforms).
158
159### Built-in flags {:#built-in-flags}
160
161Flags like `--cpu` are built into Bazel: the build tool natively understands
162them for all builds in all projects. These are specified with
163[`config_setting`](/reference/be/general#config_setting)'s
164[`values`](/reference/be/general#config_setting.values) attribute:
165
166```python
167config_setting(
168 name = "meaningful_condition_name",
169 values = {
170 "flag1": "value1",
171 "flag2": "value2",
172 ...
173 },
174)
175```
176
177`flagN` is a flag name (without `--`, so `"cpu"` instead of `"--cpu"`). `valueN`
178is the expected value for that flag. `:meaningful_condition_name` matches if
179*every* entry in `values` matches. Order is irrelevant.
180
181`valueN` is parsed as if it was set on the command line. This means:
182
183* `values = { "compilation_mode": "opt" }` matches `bazel build -c opt`
184* `values = { "force_pic": "true" }` matches `bazel build --force_pic=1`
185* `values = { "force_pic": "0" }` matches `bazel build --noforce_pic`
186
187`config_setting` only supports flags that affect target behavior. For example,
188[`--show_progress`](/docs/user-manual#show-progress) isn't allowed because
189it only affects how Bazel reports progress to the user. Targets can't use that
190flag to construct their results. The exact set of supported flags isn't
191documented. In practice, most flags that "make sense" work.
192
193### Custom flags {:#custom-flags}
194
195You can model your own project-specific flags with
196[Starlark build settings][BuildSettings]. Unlike built-in flags, these are
197defined as build targets, so Bazel references them with target labels.
198
199These are triggered with [`config_setting`](/reference/be/general#config_setting)'s
200[`flag_values`](/reference/be/general#config_setting.flag_values)
201attribute:
202
203```python
204config_setting(
205 name = "meaningful_condition_name",
206 flag_values = {
207 "//myflags:flag1": "value1",
208 "//myflags:flag2": "value2",
209 ...
210 },
211)
212```
213
Googler504639c2022-06-01 14:19:14 -0700214Behavior is the same as for [built-in flags](#built-in-flags). See [here](https://github.com/bazelbuild/examples/tree/HEAD/configurations/select_on_build_setting){: .external}
fweacae1cd2022-02-17 09:45:38 -0800215for a working example.
216
217[`--define`](/reference/command-line-reference#flag--define)
218is an alternative legacy syntax for custom flags (for example
219`--define foo=bar`). This can be expressed either in the
220[values](/reference/be/general#config_setting.values) attribute
221(`values = {"define": "foo=bar"}`) or the
222[define_values](/reference/be/general#config_setting.define_values) attribute
223(`define_values = {"foo": "bar"}`). `--define` is only supported for backwards
224compatibility. Prefer Starlark build settings whenever possible.
225
226`values`, `flag_values`, and `define_values` evaluate independently. The
227`config_setting` matches if all values across all of them match.
228
229## The default condition {:#default-condition}
230
231The built-in condition `//conditions:default` matches when no other condition
232matches.
233
234Because of the "exactly one match" rule, a configurable attribute with no match
235and no default condition emits a `"no matching conditions"` error. This can
236protect against silent failures from unexpected settings:
237
238```python
239# myapp/BUILD
240
241config_setting(
242 name = "x86_cpu",
243 values = {"cpu": "x86"},
244)
245
246cc_library(
247 name = "x86_only_lib",
248 srcs = select({
249 ":x86_cpu": ["lib.cc"],
250 }),
251)
252```
253
254```sh
255$ bazel build //myapp:x86_only_lib --cpu=arm
256ERROR: Configurable attribute "srcs" doesn't match this configuration (would
257a default condition help?).
258Conditions checked:
259 //myapp:x86_cpu
260```
261
262For even clearer errors, you can set custom messages with `select()`'s
263[`no_match_error`](#custom-error-messages) attribute.
264
265## Platforms {:#platforms}
266
267While the ability to specify multiple flags on the command line provides
268flexibility, it can also be burdensome to individually set each one every time
269you want to build a target.
Googler0c4cf502022-12-19 13:43:43 -0800270 [Platforms](/extending/platforms)
fweacae1cd2022-02-17 09:45:38 -0800271let you consolidate these into simple bundles.
272
273```python
274# myapp/BUILD
275
276sh_binary(
277 name = "my_rocks",
278 srcs = select({
279 ":basalt": ["pyroxene.sh"],
280 ":marble": ["calcite.sh"],
281 "//conditions:default": ["feldspar.sh"],
282 }),
283)
284
285config_setting(
286 name = "basalt",
287 constraint_values = [
288 ":black",
289 ":igneous",
290 ],
291)
292
293config_setting(
294 name = "marble",
295 constraint_values = [
296 ":white",
297 ":metamorphic",
298 ],
299)
300
301# constraint_setting acts as an enum type, and constraint_value as an enum value.
302constraint_setting(name = "color")
303constraint_value(name = "black", constraint_setting = "color")
304constraint_value(name = "white", constraint_setting = "color")
305constraint_setting(name = "texture")
306constraint_value(name = "smooth", constraint_setting = "texture")
307constraint_setting(name = "type")
308constraint_value(name = "igneous", constraint_setting = "type")
309constraint_value(name = "metamorphic", constraint_setting = "type")
310
311platform(
312 name = "basalt_platform",
313 constraint_values = [
314 ":black",
315 ":igneous",
316 ],
317)
318
319platform(
320 name = "marble_platform",
321 constraint_values = [
322 ":white",
323 ":smooth",
324 ":metamorphic",
325 ],
326)
327```
328
329The platform can be specified on the command line. It activates the
330`config_setting`s that contain a subset of the platform's `constraint_values`,
331allowing those `config_setting`s to match in `select()` expressions.
332
333For example, in order to set the `srcs` attribute of `my_rocks` to `calcite.sh`,
334you can simply run
335
336```sh
337bazel build //my_app:my_rocks --platforms=//myapp:marble_platform
338```
339
340Without platforms, this might look something like
341
342```sh
343bazel build //my_app:my_rocks --define color=white --define texture=smooth --define type=metamorphic
344```
345
346`select()` can also directly read `constraint_value`s:
347
348```python
349constraint_setting(name = "type")
350constraint_value(name = "igneous", constraint_setting = "type")
351constraint_value(name = "metamorphic", constraint_setting = "type")
352sh_binary(
353 name = "my_rocks",
354 srcs = select({
355 ":igneous": ["igneous.sh"],
356 ":metamorphic" ["metamorphic.sh"],
357 }),
358)
359```
360
361This saves the need for boilerplate `config_setting`s when you only need to
362check against single values.
363
364Platforms are still under development. See the
Googler0c4cf502022-12-19 13:43:43 -0800365[documentation](/concepts/platforms) for details.
fweacae1cd2022-02-17 09:45:38 -0800366
367## Combining `select()`s {:#combining-selects}
368
369`select` can appear multiple times in the same attribute:
370
371```python
372sh_binary(
373 name = "my_target",
374 srcs = ["always_include.sh"] +
375 select({
376 ":armeabi_mode": ["armeabi_src.sh"],
377 ":x86_mode": ["x86_src.sh"],
378 }) +
379 select({
380 ":opt_mode": ["opt_extras.sh"],
381 ":dbg_mode": ["dbg_extras.sh"],
382 }),
383)
384```
385
Fabian Brandstetter86c9c3d2022-06-22 13:39:59 -0700386Note: Some restrictions apply on what can be combined in the `select`s values:
387 - Duplicate labels can appear in different paths of the same `select`.
388 - Duplicate labels can *not* appear within the same path of a `select`.
389 - Duplicate labels can *not* appear across multiple combined `select`s (no matter what path)
390
fweacae1cd2022-02-17 09:45:38 -0800391`select` cannot appear inside another `select`. If you need to nest `selects`
392and your attribute takes other targets as values, use an intermediate target:
393
394```python
395sh_binary(
396 name = "my_target",
397 srcs = ["always_include.sh"],
398 deps = select({
399 ":armeabi_mode": [":armeabi_lib"],
400 ...
401 }),
402)
403
404sh_library(
405 name = "armeabi_lib",
406 srcs = select({
407 ":opt_mode": ["armeabi_with_opt.sh"],
408 ...
409 }),
410)
411```
412
413If you need a `select` to match when multiple conditions match, consider [AND
414chaining](#and-chaining).
415
416## OR chaining {:#or-chaining}
417
418Consider the following:
419
420```python
421sh_binary(
422 name = "my_target",
423 srcs = ["always_include.sh"],
424 deps = select({
425 ":config1": [":standard_lib"],
426 ":config2": [":standard_lib"],
427 ":config3": [":standard_lib"],
428 ":config4": [":special_lib"],
429 }),
430)
431```
432
433Most conditions evaluate to the same dep. But this syntax is hard to read and
434maintain. It would be nice to not have to repeat `[":standard_lib"]` multiple
435times.
436
437One option is to predefine the value as a BUILD variable:
438
439```python
440STANDARD_DEP = [":standard_lib"]
441
442sh_binary(
443 name = "my_target",
444 srcs = ["always_include.sh"],
445 deps = select({
446 ":config1": STANDARD_DEP,
447 ":config2": STANDARD_DEP,
448 ":config3": STANDARD_DEP,
449 ":config4": [":special_lib"],
450 }),
451)
452```
453
454This makes it easier to manage the dependency. But it still causes unnecessary
455duplication.
456
457For more direct support, use one of the following:
458
459### `selects.with_or` {:#selects-with-or}
460
461The
462[with_or](https://github.com/bazelbuild/bazel-skylib/blob/main/docs/selects_doc.md#selectswith_or){: .external}
463macro in [Skylib](https://github.com/bazelbuild/bazel-skylib){: .external}'s
464[`selects`](https://github.com/bazelbuild/bazel-skylib/blob/main/docs/selects_doc.md){: .external}
465module supports `OR`ing conditions directly inside a `select`:
466
467```python
468load("@bazel_skylib//lib:selects.bzl", "selects")
469```
470
471```python
472sh_binary(
473 name = "my_target",
474 srcs = ["always_include.sh"],
475 deps = selects.with_or({
476 (":config1", ":config2", ":config3"): [":standard_lib"],
477 ":config4": [":special_lib"],
478 }),
479)
480```
481
482### `selects.config_setting_group` {:#selects-config-setting-or-group}
483
484
485The
486[config_setting_group](https://github.com/bazelbuild/bazel-skylib/blob/main/docs/selects_doc.md#selectsconfig_setting_group){: .external}
487macro in [Skylib](https://github.com/bazelbuild/bazel-skylib){: .external}'s
488[`selects`](https://github.com/bazelbuild/bazel-skylib/blob/main/docs/selects_doc.md){: .external}
489module supports `OR`ing multiple `config_setting`s:
490
491```python
492load("@bazel_skylib//lib:selects.bzl", "selects")
493```
494
495
496```python
497config_setting(
498 name = "config1",
499 values = {"cpu": "arm"},
500)
501config_setting(
502 name = "config2",
503 values = {"compilation_mode": "dbg"},
504)
505selects.config_setting_group(
506 name = "config1_or_2",
507 match_any = [":config1", ":config2"],
508)
509sh_binary(
510 name = "my_target",
511 srcs = ["always_include.sh"],
512 deps = select({
513 ":config1_or_2": [":standard_lib"],
514 "//conditions:default": [":other_lib"],
515 }),
516)
517```
518
519Unlike `selects.with_or`, different targets can share `:config1_or_2` across
520different attributes.
521
522It's an error for multiple conditions to match unless one is an unambiguous
Alessandro Pattid43737f2023-04-11 21:31:54 -0700523"specialization" of the others or they all resolve to the same value. See [here](#configurable-build-example) for details.
fweacae1cd2022-02-17 09:45:38 -0800524
525## AND chaining {:#and-chaining}
526
527If you need a `select` branch to match when multiple conditions match, use the
528[Skylib](https://github.com/bazelbuild/bazel-skylib){: .external} macro
529[config_setting_group](https://github.com/bazelbuild/bazel-skylib/blob/main/docs/selects_doc.md#selectsconfig_setting_group){: .external}:
530
531```python
532config_setting(
533 name = "config1",
534 values = {"cpu": "arm"},
535)
536config_setting(
537 name = "config2",
538 values = {"compilation_mode": "dbg"},
539)
540selects.config_setting_group(
541 name = "config1_and_2",
542 match_all = [":config1", ":config2"],
543)
544sh_binary(
545 name = "my_target",
546 srcs = ["always_include.sh"],
547 deps = select({
548 ":config1_and_2": [":standard_lib"],
549 "//conditions:default": [":other_lib"],
550 }),
551)
552```
553
554Unlike OR chaining, existing `config_setting`s can't be directly `AND`ed
555inside a `select`. You have to explicitly wrap them in a `config_setting_group`.
556
557## Custom error messages {:#custom-error-messages}
558
559By default, when no condition matches, the target the `select()` is attached to
560fails with the error:
561
562```sh
563ERROR: Configurable attribute "deps" doesn't match this configuration (would
564a default condition help?).
565Conditions checked:
566 //tools/cc_target_os:darwin
567 //tools/cc_target_os:android
568```
569
570This can be customized with the [`no_match_error`](/reference/be/functions#select)
571attribute:
572
573```python
574cc_library(
575 name = "my_lib",
576 deps = select(
577 {
578 "//tools/cc_target_os:android": [":android_deps"],
579 "//tools/cc_target_os:windows": [":windows_deps"],
580 },
581 no_match_error = "Please build with an Android or Windows toolchain",
582 ),
583)
584```
585
586```sh
587$ bazel build //myapp:my_lib
588ERROR: Configurable attribute "deps" doesn't match this configuration: Please
589build with an Android or Windows toolchain
590```
591
592## Rules compatibility {:#rules-compatibility}
593
594Rule implementations receive the *resolved values* of configurable
595attributes. For example, given:
596
597```python
598# myapp/BUILD
599
600some_rule(
601 name = "my_target",
602 some_attr = select({
603 ":foo_mode": [":foo"],
604 ":bar_mode": [":bar"],
605 }),
606)
607```
608
609```sh
610$ bazel build //myapp/my_target --define mode=foo
611```
612
613Rule implementation code sees `ctx.attr.some_attr` as `[":foo"]`.
614
615Macros can accept `select()` clauses and pass them through to native
616rules. But *they cannot directly manipulate them*. For example, there's no way
617for a macro to convert
618
619```python
620select({"foo": "val"}, ...)
621```
622
623to
624
625```python
626select({"foo": "val_with_suffix"}, ...)
627```
628
629This is for two reasons.
630
631First, macros that need to know which path a `select` will choose *cannot work*
Googler0c4cf502022-12-19 13:43:43 -0800632because macros are evaluated in Bazel's [loading phase](/run/build#loading),
fweacae1cd2022-02-17 09:45:38 -0800633which occurs before flag values are known.
634This is a core Bazel design restriction that's unlikely to change any time soon.
635
636Second, macros that just need to iterate over *all* `select` paths, while
637technically feasible, lack a coherent UI. Further design is necessary to change
638this.
639
640## Bazel query and cquery {:#query-and-cquery}
641
Googlerdf7617e2022-12-20 09:44:29 -0800642Bazel [`query`](/query/guide) operates over Bazel's
fweacae1cd2022-02-17 09:45:38 -0800643[loading phase](/reference/glossary#loading-phase).
644This means it doesn't know what command line flags a target uses since those
645flags aren't evaluated until later in the build (in the
646[analysis phase](/reference/glossary#analysis-phase)).
647So it can't determine which `select()` branches are chosen.
648
Googler0c4cf502022-12-19 13:43:43 -0800649Bazel [`cquery`](/query/cquery) operates after Bazel's analysis phase, so it has
fweacae1cd2022-02-17 09:45:38 -0800650all this information and can accurately resolve `select()`s.
651
652Consider:
653
654```python
655load("@bazel_skylib//rules:common_settings.bzl", "string_flag")
656```
657```python
658# myapp/BUILD
659
660string_flag(
661 name = "dog_type",
662 build_setting_default = "cat"
663)
664
665cc_library(
666 name = "my_lib",
667 deps = select({
668 ":long": [":foo_dep"],
669 ":short": [":bar_dep"],
670 }),
671)
672
673config_setting(
674 name = "long",
675 flag_values = {":dog_type": "dachshund"},
676)
677
678config_setting(
679 name = "short",
680 flag_values = {":dog_type": "pug"},
681)
682```
683
684`query` overapproximates `:my_lib`'s dependencies:
685
686```sh
687$ bazel query 'deps(//myapp:my_lib)'
688//myapp:my_lib
689//myapp:foo_dep
690//myapp:bar_dep
691```
692
693while `cquery` shows its exact dependencies:
694
695```sh
696$ bazel cquery 'deps(//myapp:my_lib)' --//myapp:dog_type=pug
697//myapp:my_lib
698//myapp:bar_dep
699```
700
701## FAQ {:#faq}
702
703### Why doesn't select() work in macros? {:#faq-select-macro}
704
fwe6c2bd4a2022-02-18 10:47:46 -0800705select() *does* work in rules! See [Rules compatibility](#rules-compatibility) for
fweacae1cd2022-02-17 09:45:38 -0800706details.
707
708The key issue this question usually means is that select() doesn't work in
709*macros*. These are different than *rules*. See the
Googler0c4cf502022-12-19 13:43:43 -0800710documentation on [rules](/extending/rules) and [macros](/extending/macros)
fweacae1cd2022-02-17 09:45:38 -0800711to understand the difference.
712Here's an end-to-end example:
713
714Define a rule and macro:
715
716```python
717# myapp/defs.bzl
718
719# Rule implementation: when an attribute is read, all select()s have already
720# been resolved. So it looks like a plain old attribute just like any other.
721def _impl(ctx):
722 name = ctx.attr.name
723 allcaps = ctx.attr.my_config_string.upper() # This works fine on all values.
724 print("My name is " + name + " with custom message: " + allcaps)
725
726# Rule declaration:
727my_custom_bazel_rule = rule(
728 implementation = _impl,
729 attrs = {"my_config_string": attr.string()},
730)
731
732# Macro declaration:
733def my_custom_bazel_macro(name, my_config_string):
734 allcaps = my_config_string.upper() # This line won't work with select(s).
735 print("My name is " + name + " with custom message: " + allcaps)
736```
737
738Instantiate the rule and macro:
739
740```python
741# myapp/BUILD
742
743load("//myapp:defs.bzl", "my_custom_bazel_rule")
744load("//myapp:defs.bzl", "my_custom_bazel_macro")
745
746my_custom_bazel_rule(
747 name = "happy_rule",
748 my_config_string = select({
Googlerac2e8e52024-04-15 04:34:10 -0700749 "//third_party/bazel_platforms/cpu:x86_32": "first string",
Googler09c927a2023-05-11 06:11:11 -0700750 "//third_party/bazel_platforms/cpu:ppc": "second string",
fweacae1cd2022-02-17 09:45:38 -0800751 }),
752)
753
754my_custom_bazel_macro(
755 name = "happy_macro",
756 my_config_string = "fixed string",
757)
758
759my_custom_bazel_macro(
760 name = "sad_macro",
761 my_config_string = select({
Googlerac2e8e52024-04-15 04:34:10 -0700762 "//third_party/bazel_platforms/cpu:x86_32": "first string",
Googler09c927a2023-05-11 06:11:11 -0700763 "//third_party/bazel_platforms/cpu:ppc": "other string",
fweacae1cd2022-02-17 09:45:38 -0800764 }),
765)
766```
767
768Building fails because `sad_macro` can't process the `select()`:
769
770```sh
771$ bazel build //myapp:all
772ERROR: /myworkspace/myapp/BUILD:17:1: Traceback
773 (most recent call last):
774File "/myworkspace/myapp/BUILD", line 17
775my_custom_bazel_macro(name = "sad_macro", my_config_stri..."}))
776File "/myworkspace/myapp/defs.bzl", line 4, in
777 my_custom_bazel_macro
778my_config_string.upper()
779type 'select' has no method upper().
780ERROR: error loading package 'myapp': Package 'myapp' contains errors.
781```
782
783Building succeeds when you comment out `sad_macro`:
784
785```sh
786# Comment out sad_macro so it doesn't mess up the build.
787$ bazel build //myapp:all
788DEBUG: /myworkspace/myapp/defs.bzl:5:3: My name is happy_macro with custom message: FIXED STRING.
789DEBUG: /myworkspace/myapp/hi.bzl:15:3: My name is happy_rule with custom message: FIRST STRING.
790```
791
792This is impossible to change because *by definition* macros are evaluated before
793Bazel reads the build's command line flags. That means there isn't enough
794information to evaluate select()s.
795
796Macros can, however, pass `select()`s as opaque blobs to rules:
797
798```python
799# myapp/defs.bzl
800
801def my_custom_bazel_macro(name, my_config_string):
802 print("Invoking macro " + name)
803 my_custom_bazel_rule(
804 name = name + "_as_target",
805 my_config_string = my_config_string,
806 )
807```
808
809```sh
810$ bazel build //myapp:sad_macro_less_sad
811DEBUG: /myworkspace/myapp/defs.bzl:23:3: Invoking macro sad_macro_less_sad.
812DEBUG: /myworkspace/myapp/defs.bzl:15:3: My name is sad_macro_less_sad with custom message: FIRST STRING.
813```
814
815### Why does select() always return true? {:#faq-boolean-select}
816
817Because *macros* (but not rules) by definition
818[can't evaluate `select()`s](#faq-select-macro), any attempt to do so
819usually produces an error:
820
821```sh
822ERROR: /myworkspace/myapp/BUILD:17:1: Traceback
823 (most recent call last):
824File "/myworkspace/myapp/BUILD", line 17
825my_custom_bazel_macro(name = "sad_macro", my_config_stri..."}))
826File "/myworkspace/myapp/defs.bzl", line 4, in
827 my_custom_bazel_macro
828my_config_string.upper()
829type 'select' has no method upper().
830```
831
832Booleans are a special case that fail silently, so you should be particularly
833vigilant with them:
834
835```sh
836$ cat myapp/defs.bzl
837def my_boolean_macro(boolval):
838 print("TRUE" if boolval else "FALSE")
839
840$ cat myapp/BUILD
841load("//myapp:defs.bzl", "my_boolean_macro")
842my_boolean_macro(
843 boolval = select({
Googlerac2e8e52024-04-15 04:34:10 -0700844 "//third_party/bazel_platforms/cpu:x86_32": True,
Googler09c927a2023-05-11 06:11:11 -0700845 "//third_party/bazel_platforms/cpu:ppc": False,
fweacae1cd2022-02-17 09:45:38 -0800846 }),
847)
848
849$ bazel build //myapp:all --cpu=x86
850DEBUG: /myworkspace/myapp/defs.bzl:4:3: TRUE.
851$ bazel build //mypro:all --cpu=ppc
852DEBUG: /myworkspace/myapp/defs.bzl:4:3: TRUE.
853```
854
855This happens because macros don't understand the contents of `select()`.
856So what they're really evaluting is the `select()` object itself. According to
857[Pythonic](https://docs.python.org/release/2.5.2/lib/truth.html) design
858standards, all objects aside from a very small number of exceptions
859automatically return true.
860
861### Can I read select() like a dict? {:#faq-inspectable-select}
862
863Macros [can't](#faq-select-macro) evaluate select(s) because macros evaluate before
864Bazel knows what the build's command line parameters are. Can they at least read
865the `select()`'s dictionary to, for example, add a suffix to each value?
866
Christopher Sauera35d2a62022-11-28 04:02:47 -0800867Conceptually this is possible, but [it isn't yet a Bazel feature](https://github.com/bazelbuild/bazel/issues/8419).
fweacae1cd2022-02-17 09:45:38 -0800868What you *can* do today is prepare a straight dictionary, then feed it into a
869`select()`:
870
871```sh
872$ cat myapp/defs.bzl
873def selecty_genrule(name, select_cmd):
874 for key in select_cmd.keys():
875 select_cmd[key] += " WITH SUFFIX"
876 native.genrule(
877 name = name,
878 outs = [name + ".out"],
879 srcs = [],
880 cmd = "echo " + select(select_cmd + {"//conditions:default": "default"})
881 + " > $@"
882 )
883
884$ cat myapp/BUILD
885selecty_genrule(
886 name = "selecty",
887 select_cmd = {
Googlerac2e8e52024-04-15 04:34:10 -0700888 "//third_party/bazel_platforms/cpu:x86_32": "x86 mode",
fweacae1cd2022-02-17 09:45:38 -0800889 },
890)
891
892$ bazel build //testapp:selecty --cpu=x86 && cat bazel-genfiles/testapp/selecty.out
893x86 mode WITH SUFFIX
894```
895
896If you'd like to support both `select()` and native types, you can do this:
897
898```sh
899$ cat myapp/defs.bzl
900def selecty_genrule(name, select_cmd):
901 cmd_suffix = ""
902 if type(select_cmd) == "string":
903 cmd_suffix = select_cmd + " WITH SUFFIX"
904 elif type(select_cmd) == "dict":
905 for key in select_cmd.keys():
906 select_cmd[key] += " WITH SUFFIX"
907 cmd_suffix = select(select_cmd + {"//conditions:default": "default"})
908
909 native.genrule(
910 name = name,
911 outs = [name + ".out"],
912 srcs = [],
913 cmd = "echo " + cmd_suffix + "> $@",
914 )
915```
916
917### Why doesn't select() work with bind()? {:#faq-select-bind}
918
Googler13ecdf52024-01-02 12:15:30 -0800919First of all, do not use `bind()`. It is deprecated in favor of `alias()`.
fweacae1cd2022-02-17 09:45:38 -0800920
Googler13ecdf52024-01-02 12:15:30 -0800921The technical answer is that [`bind()`](/reference/be/workspace#bind) is a repo
922rule, not a BUILD rule.
923
924Repo rules do not have a specific configuration, and aren't evaluated in
fweacae1cd2022-02-17 09:45:38 -0800925the same way as BUILD rules. Therefore, a `select()` in a `bind()` can't
926actually evaluate to any specific branch.
927
928Instead, you should use [`alias()`](/reference/be/general#alias), with a `select()` in
929the `actual` attribute, to perform this type of run-time determination. This
930works correctly, since `alias()` is a BUILD rule, and is evaluated with a
931specific configuration.
932
933You can even have a `bind()` target point to an `alias()`, if needed.
934
935```sh
936$ cat WORKSPACE
937workspace(name = "myapp")
938bind(name = "openssl", actual = "//:ssl")
939http_archive(name = "alternative", ...)
940http_archive(name = "boringssl", ...)
941
942$ cat BUILD
943config_setting(
944 name = "alt_ssl",
945 define_values = {
946 "ssl_library": "alternative",
947 },
948)
949
950alias(
951 name = "ssl",
952 actual = select({
953 "//:alt_ssl": "@alternative//:ssl",
954 "//conditions:default": "@boringssl//:ssl",
955 }),
956)
957```
958
959With this setup, you can pass `--define ssl_library=alternative`, and any target
960that depends on either `//:ssl` or `//external:ssl` will see the alternative
961located at `@alternative//:ssl`.
962
Googler13ecdf52024-01-02 12:15:30 -0800963But really, stop using `bind()`.
964
fweacae1cd2022-02-17 09:45:38 -0800965### Why doesn't my select() choose what I expect? {:#faq-select-choose-condition}
966
967If `//myapp:foo` has a `select()` that doesn't choose the condition you expect,
Googler0c4cf502022-12-19 13:43:43 -0800968use [cquery](/query/cquery) and `bazel config` to debug:
fweacae1cd2022-02-17 09:45:38 -0800969
970If `//myapp:foo` is the top-level target you're building, run:
971
972```sh
973$ bazel cquery //myapp:foo <desired build flags>
974//myapp:foo (12e23b9a2b534a)
975```
976
977If you're building some other target `//bar` that depends on
978//myapp:foo somewhere in its subgraph, run:
979
980```sh
981$ bazel cquery 'somepath(//bar, //myapp:foo)' <desired build flags>
982//bar:bar (3ag3193fee94a2)
983//bar:intermediate_dep (12e23b9a2b534a)
984//myapp:foo (12e23b9a2b534a)
985```
986
987The `(12e23b9a2b534a)` next to `//myapp:foo` is a *hash* of the
988configuration that resolves `//myapp:foo`'s `select()`. You can inspect its
989values with `bazel config`:
990
991```sh
992$ bazel config 12e23b9a2b534a
993BuildConfigurationValue 12e23b9a2b534a
994Fragment com.google.devtools.build.lib.analysis.config.CoreOptions {
995 cpu: darwin
996 compilation_mode: fastbuild
997 ...
998}
999Fragment com.google.devtools.build.lib.rules.cpp.CppOptions {
1000 linkopt: [-Dfoo=bar]
1001 ...
1002}
1003...
1004```
1005
1006Then compare this output against the settings expected by each `config_setting`.
1007
1008`//myapp:foo` may exist in different configurations in the same build. See the
Googler0c4cf502022-12-19 13:43:43 -08001009[cquery docs](/query/cquery) for guidance on using `somepath` to get the right
fweacae1cd2022-02-17 09:45:38 -08001010one.
1011
1012Caution: To prevent restarting the Bazel server, invoke `bazel config` with the
1013same command line flags as the `bazel cquery`. The `config` command relies on
1014the configuration nodes from the still-running server of the previous command.
1015
John Catere0e58962022-05-26 08:52:08 -07001016### Why doesn't `select()` work with platforms? {:#faq-select-platforms}
1017
1018Bazel doesn't support configurable attributes checking whether a given platform
1019is the target platform because the semantics are unclear.
1020
1021For example:
1022
1023```py
1024platform(
1025 name = "x86_linux_platform",
1026 constraint_values = [
1027 "@platforms//cpu:x86",
1028 "@platforms//os:linux",
1029 ],
1030)
1031
1032cc_library(
1033 name = "lib",
1034 srcs = [...],
1035 linkopts = select({
1036 ":x86_linux_platform": ["--enable_x86_optimizations"],
1037 "//conditions:default": [],
1038 }),
1039)
1040```
1041
1042In this `BUILD` file, which `select()` should be used if the target platform has both the
1043`@platforms//cpu:x86` and `@platforms//os:linux` constraints, but is **not** the
1044`:x86_linux_platform` defined here? The author of the `BUILD` file and the user
1045who defined the separate platform may have different ideas.
1046
1047#### What should I do instead?
1048
1049Instead, define a `config_setting` that matches **any** platform with
1050these constraints:
1051
1052```py
1053config_setting(
1054 name = "is_x86_linux",
1055 constraint_values = [
1056 "@platforms//cpu:x86",
1057 "@platforms//os:linux",
1058 ],
1059)
1060
1061cc_library(
1062 name = "lib",
1063 srcs = [...],
1064 linkopts = select({
1065 ":is_x86_linux": ["--enable_x86_optimizations"],
1066 "//conditions:default": [],
1067 }),
1068)
1069```
1070
1071This process defines specific semantics, making it clearer to users what
1072platforms meet the desired conditions.
1073
1074#### What if I really, really want to `select` on the platform?
1075
1076If your build requirements specifically require checking the platform, you
1077can flip the value of the `--platforms` flag in a `config_setting`:
1078
1079```py
1080config_setting(
1081 name = "is_specific_x86_linux_platform",
1082 values = {
1083 "platforms": ["//package:x86_linux_platform"],
1084 },
1085)
1086
1087cc_library(
1088 name = "lib",
1089 srcs = [...],
1090 linkopts = select({
1091 ":is_specific_x86_linux_platform": ["--enable_x86_optimizations"],
1092 "//conditions:default": [],
1093 }),
1094)
1095```
1096
1097The Bazel team doesn't endorse doing this; it overly constrains your build and
1098confuses users when the expected condition does not match.
1099
Googlerdf7617e2022-12-20 09:44:29 -08001100[BuildSettings]: /extending/config#user-defined-build-settings