fwe | acae1cd | 2022-02-17 09:45:38 -0800 | [diff] [blame] | 1 | Project: /_project.yaml |
| 2 | Book: /_book.yaml |
| 3 | |
| 4 | # Configurable Build Attributes |
| 5 | |
Googler | 3b9ed6e | 2022-11-08 02:19:42 -0800 | [diff] [blame] | 6 | {% include "_buttons.html" %} |
| 7 | |
fwe | acae1cd | 2022-02-17 09:45:38 -0800 | [diff] [blame] | 8 | **_Configurable attributes_**, commonly known as [`select()`]( |
| 9 | /reference/be/functions#select), is a Bazel feature that lets users toggle the values |
| 10 | of build rule attributes at the command line. |
| 11 | |
| 12 | This can be used, for example, for a multiplatform library that automatically |
| 13 | chooses the appropriate implementation for the architecture, or for a |
| 14 | feature-configurable binary that can be customized at build time. |
| 15 | |
| 16 | ## Example {:#configurable-build-example} |
| 17 | |
| 18 | ```python |
| 19 | # myapp/BUILD |
| 20 | |
| 21 | cc_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 | |
| 31 | config_setting( |
| 32 | name = "arm_build", |
| 33 | values = {"cpu": "arm"}, |
| 34 | ) |
| 35 | |
| 36 | config_setting( |
| 37 | name = "x86_debug_build", |
| 38 | values = { |
| 39 | "cpu": "x86", |
| 40 | "compilation_mode": "dbg", |
| 41 | }, |
| 42 | ) |
| 43 | ``` |
| 44 | |
| 45 | This declares a `cc_binary` that "chooses" its deps based on the flags at the |
Googler | 61211f2 | 2022-05-04 13:35:49 -0700 | [diff] [blame] | 46 | command line. Specifically, `deps` becomes: |
fwe | acae1cd | 2022-02-17 09:45:38 -0800 | [diff] [blame] | 47 | |
| 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) |
| 73 | targets. By using `select()` in a configurable attribute, the attribute |
| 74 | effectively adopts different values when different conditions hold. |
| 75 | |
Alessandro Patti | d43737f | 2023-04-11 21:31:54 -0700 | [diff] [blame] | 76 | Matches 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 | |
| 82 | The built-in condition [`//conditions:default`](#default-condition) automatically matches when |
fwe | acae1cd | 2022-02-17 09:45:38 -0800 | [diff] [blame] | 83 | nothing else does. |
| 84 | |
| 85 | While this example uses `deps`, `select()` works just as well on `srcs`, |
| 86 | `resources`, `cmd`, and most other attributes. Only a small number of attributes |
| 87 | are *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 | |
| 93 | Certain attributes change the build parameters for all transitive dependencies |
| 94 | under a target. For example, `genrule`'s `tools` changes `--cpu` to the CPU of |
| 95 | the machine running Bazel (which, thanks to cross-compilation, may be different |
| 96 | than the CPU the target is built for). This is known as a |
| 97 | [configuration transition](/reference/glossary#transition). |
| 98 | |
| 99 | Given |
| 100 | |
| 101 | ```python |
| 102 | #myapp/BUILD |
| 103 | |
| 104 | config_setting( |
| 105 | name = "arm_cpu", |
| 106 | values = {"cpu": "arm"}, |
| 107 | ) |
| 108 | |
| 109 | config_setting( |
| 110 | name = "x86_cpu", |
| 111 | values = {"cpu": "x86"}, |
| 112 | ) |
| 113 | |
| 114 | genrule( |
| 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 | |
| 126 | cc_binary( |
| 127 | name = "tool1", |
| 128 | srcs = select({ |
| 129 | ":arm_cpu": ["armtool.cc"], |
| 130 | ":x86_cpu": ["x86tool.cc"], |
| 131 | }), |
| 132 | ) |
| 133 | ``` |
| 134 | |
| 135 | running |
| 136 | |
| 137 | ```sh |
| 138 | $ bazel build //myapp:my_genrule --cpu=arm |
| 139 | ``` |
| 140 | |
| 141 | on 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 |
| 143 | build 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 | |
| 149 | Each key in a configurable attribute is a label reference to a |
| 150 | [`config_setting`](/reference/be/general#config_setting) or |
Googler | f1c2e44 | 2023-09-06 09:20:49 -0700 | [diff] [blame] | 151 | [`constraint_value`](/reference/be/platforms-and-toolchains#constraint_value). |
fwe | acae1cd | 2022-02-17 09:45:38 -0800 | [diff] [blame] | 152 | |
| 153 | `config_setting` is just a collection of |
| 154 | expected command line flag settings. By encapsulating these in a target, it's |
| 155 | easy 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 | |
| 161 | Flags like `--cpu` are built into Bazel: the build tool natively understands |
| 162 | them 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 |
| 167 | config_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` |
| 178 | is 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 |
| 189 | it only affects how Bazel reports progress to the user. Targets can't use that |
| 190 | flag to construct their results. The exact set of supported flags isn't |
| 191 | documented. In practice, most flags that "make sense" work. |
| 192 | |
| 193 | ### Custom flags {:#custom-flags} |
| 194 | |
| 195 | You can model your own project-specific flags with |
| 196 | [Starlark build settings][BuildSettings]. Unlike built-in flags, these are |
| 197 | defined as build targets, so Bazel references them with target labels. |
| 198 | |
| 199 | These are triggered with [`config_setting`](/reference/be/general#config_setting)'s |
| 200 | [`flag_values`](/reference/be/general#config_setting.flag_values) |
| 201 | attribute: |
| 202 | |
| 203 | ```python |
| 204 | config_setting( |
| 205 | name = "meaningful_condition_name", |
| 206 | flag_values = { |
| 207 | "//myflags:flag1": "value1", |
| 208 | "//myflags:flag2": "value2", |
| 209 | ... |
| 210 | }, |
| 211 | ) |
| 212 | ``` |
| 213 | |
Googler | 504639c | 2022-06-01 14:19:14 -0700 | [diff] [blame] | 214 | Behavior 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} |
fwe | acae1cd | 2022-02-17 09:45:38 -0800 | [diff] [blame] | 215 | for a working example. |
| 216 | |
| 217 | [`--define`](/reference/command-line-reference#flag--define) |
| 218 | is 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 |
| 224 | compatibility. 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 | |
| 231 | The built-in condition `//conditions:default` matches when no other condition |
| 232 | matches. |
| 233 | |
| 234 | Because of the "exactly one match" rule, a configurable attribute with no match |
| 235 | and no default condition emits a `"no matching conditions"` error. This can |
| 236 | protect against silent failures from unexpected settings: |
| 237 | |
| 238 | ```python |
| 239 | # myapp/BUILD |
| 240 | |
| 241 | config_setting( |
| 242 | name = "x86_cpu", |
| 243 | values = {"cpu": "x86"}, |
| 244 | ) |
| 245 | |
| 246 | cc_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 |
| 256 | ERROR: Configurable attribute "srcs" doesn't match this configuration (would |
| 257 | a default condition help?). |
| 258 | Conditions checked: |
| 259 | //myapp:x86_cpu |
| 260 | ``` |
| 261 | |
| 262 | For even clearer errors, you can set custom messages with `select()`'s |
| 263 | [`no_match_error`](#custom-error-messages) attribute. |
| 264 | |
| 265 | ## Platforms {:#platforms} |
| 266 | |
| 267 | While the ability to specify multiple flags on the command line provides |
| 268 | flexibility, it can also be burdensome to individually set each one every time |
| 269 | you want to build a target. |
Googler | 0c4cf50 | 2022-12-19 13:43:43 -0800 | [diff] [blame] | 270 | [Platforms](/extending/platforms) |
fwe | acae1cd | 2022-02-17 09:45:38 -0800 | [diff] [blame] | 271 | let you consolidate these into simple bundles. |
| 272 | |
| 273 | ```python |
| 274 | # myapp/BUILD |
| 275 | |
| 276 | sh_binary( |
| 277 | name = "my_rocks", |
| 278 | srcs = select({ |
| 279 | ":basalt": ["pyroxene.sh"], |
| 280 | ":marble": ["calcite.sh"], |
| 281 | "//conditions:default": ["feldspar.sh"], |
| 282 | }), |
| 283 | ) |
| 284 | |
| 285 | config_setting( |
| 286 | name = "basalt", |
| 287 | constraint_values = [ |
| 288 | ":black", |
| 289 | ":igneous", |
| 290 | ], |
| 291 | ) |
| 292 | |
| 293 | config_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. |
| 302 | constraint_setting(name = "color") |
| 303 | constraint_value(name = "black", constraint_setting = "color") |
| 304 | constraint_value(name = "white", constraint_setting = "color") |
| 305 | constraint_setting(name = "texture") |
| 306 | constraint_value(name = "smooth", constraint_setting = "texture") |
| 307 | constraint_setting(name = "type") |
| 308 | constraint_value(name = "igneous", constraint_setting = "type") |
| 309 | constraint_value(name = "metamorphic", constraint_setting = "type") |
| 310 | |
| 311 | platform( |
| 312 | name = "basalt_platform", |
| 313 | constraint_values = [ |
| 314 | ":black", |
| 315 | ":igneous", |
| 316 | ], |
| 317 | ) |
| 318 | |
| 319 | platform( |
| 320 | name = "marble_platform", |
| 321 | constraint_values = [ |
| 322 | ":white", |
| 323 | ":smooth", |
| 324 | ":metamorphic", |
| 325 | ], |
| 326 | ) |
| 327 | ``` |
| 328 | |
| 329 | The 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`, |
| 331 | allowing those `config_setting`s to match in `select()` expressions. |
| 332 | |
| 333 | For example, in order to set the `srcs` attribute of `my_rocks` to `calcite.sh`, |
| 334 | you can simply run |
| 335 | |
| 336 | ```sh |
| 337 | bazel build //my_app:my_rocks --platforms=//myapp:marble_platform |
| 338 | ``` |
| 339 | |
| 340 | Without platforms, this might look something like |
| 341 | |
| 342 | ```sh |
| 343 | bazel 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 |
| 349 | constraint_setting(name = "type") |
| 350 | constraint_value(name = "igneous", constraint_setting = "type") |
| 351 | constraint_value(name = "metamorphic", constraint_setting = "type") |
| 352 | sh_binary( |
| 353 | name = "my_rocks", |
| 354 | srcs = select({ |
| 355 | ":igneous": ["igneous.sh"], |
| 356 | ":metamorphic" ["metamorphic.sh"], |
| 357 | }), |
| 358 | ) |
| 359 | ``` |
| 360 | |
| 361 | This saves the need for boilerplate `config_setting`s when you only need to |
| 362 | check against single values. |
| 363 | |
| 364 | Platforms are still under development. See the |
Googler | 0c4cf50 | 2022-12-19 13:43:43 -0800 | [diff] [blame] | 365 | [documentation](/concepts/platforms) for details. |
fwe | acae1cd | 2022-02-17 09:45:38 -0800 | [diff] [blame] | 366 | |
| 367 | ## Combining `select()`s {:#combining-selects} |
| 368 | |
| 369 | `select` can appear multiple times in the same attribute: |
| 370 | |
| 371 | ```python |
| 372 | sh_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 Brandstetter | 86c9c3d | 2022-06-22 13:39:59 -0700 | [diff] [blame] | 386 | Note: 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 | |
fwe | acae1cd | 2022-02-17 09:45:38 -0800 | [diff] [blame] | 391 | `select` cannot appear inside another `select`. If you need to nest `selects` |
| 392 | and your attribute takes other targets as values, use an intermediate target: |
| 393 | |
| 394 | ```python |
| 395 | sh_binary( |
| 396 | name = "my_target", |
| 397 | srcs = ["always_include.sh"], |
| 398 | deps = select({ |
| 399 | ":armeabi_mode": [":armeabi_lib"], |
| 400 | ... |
| 401 | }), |
| 402 | ) |
| 403 | |
| 404 | sh_library( |
| 405 | name = "armeabi_lib", |
| 406 | srcs = select({ |
| 407 | ":opt_mode": ["armeabi_with_opt.sh"], |
| 408 | ... |
| 409 | }), |
| 410 | ) |
| 411 | ``` |
| 412 | |
| 413 | If you need a `select` to match when multiple conditions match, consider [AND |
| 414 | chaining](#and-chaining). |
| 415 | |
| 416 | ## OR chaining {:#or-chaining} |
| 417 | |
| 418 | Consider the following: |
| 419 | |
| 420 | ```python |
| 421 | sh_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 | |
| 433 | Most conditions evaluate to the same dep. But this syntax is hard to read and |
| 434 | maintain. It would be nice to not have to repeat `[":standard_lib"]` multiple |
| 435 | times. |
| 436 | |
| 437 | One option is to predefine the value as a BUILD variable: |
| 438 | |
| 439 | ```python |
| 440 | STANDARD_DEP = [":standard_lib"] |
| 441 | |
| 442 | sh_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 | |
| 454 | This makes it easier to manage the dependency. But it still causes unnecessary |
| 455 | duplication. |
| 456 | |
| 457 | For more direct support, use one of the following: |
| 458 | |
| 459 | ### `selects.with_or` {:#selects-with-or} |
| 460 | |
| 461 | The |
| 462 | [with_or](https://github.com/bazelbuild/bazel-skylib/blob/main/docs/selects_doc.md#selectswith_or){: .external} |
| 463 | macro 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} |
| 465 | module supports `OR`ing conditions directly inside a `select`: |
| 466 | |
| 467 | ```python |
| 468 | load("@bazel_skylib//lib:selects.bzl", "selects") |
| 469 | ``` |
| 470 | |
| 471 | ```python |
| 472 | sh_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 | |
| 485 | The |
| 486 | [config_setting_group](https://github.com/bazelbuild/bazel-skylib/blob/main/docs/selects_doc.md#selectsconfig_setting_group){: .external} |
| 487 | macro 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} |
| 489 | module supports `OR`ing multiple `config_setting`s: |
| 490 | |
| 491 | ```python |
| 492 | load("@bazel_skylib//lib:selects.bzl", "selects") |
| 493 | ``` |
| 494 | |
| 495 | |
| 496 | ```python |
| 497 | config_setting( |
| 498 | name = "config1", |
| 499 | values = {"cpu": "arm"}, |
| 500 | ) |
| 501 | config_setting( |
| 502 | name = "config2", |
| 503 | values = {"compilation_mode": "dbg"}, |
| 504 | ) |
| 505 | selects.config_setting_group( |
| 506 | name = "config1_or_2", |
| 507 | match_any = [":config1", ":config2"], |
| 508 | ) |
| 509 | sh_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 | |
| 519 | Unlike `selects.with_or`, different targets can share `:config1_or_2` across |
| 520 | different attributes. |
| 521 | |
| 522 | It's an error for multiple conditions to match unless one is an unambiguous |
Alessandro Patti | d43737f | 2023-04-11 21:31:54 -0700 | [diff] [blame] | 523 | "specialization" of the others or they all resolve to the same value. See [here](#configurable-build-example) for details. |
fwe | acae1cd | 2022-02-17 09:45:38 -0800 | [diff] [blame] | 524 | |
| 525 | ## AND chaining {:#and-chaining} |
| 526 | |
| 527 | If 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 |
| 532 | config_setting( |
| 533 | name = "config1", |
| 534 | values = {"cpu": "arm"}, |
| 535 | ) |
| 536 | config_setting( |
| 537 | name = "config2", |
| 538 | values = {"compilation_mode": "dbg"}, |
| 539 | ) |
| 540 | selects.config_setting_group( |
| 541 | name = "config1_and_2", |
| 542 | match_all = [":config1", ":config2"], |
| 543 | ) |
| 544 | sh_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 | |
| 554 | Unlike OR chaining, existing `config_setting`s can't be directly `AND`ed |
| 555 | inside a `select`. You have to explicitly wrap them in a `config_setting_group`. |
| 556 | |
| 557 | ## Custom error messages {:#custom-error-messages} |
| 558 | |
| 559 | By default, when no condition matches, the target the `select()` is attached to |
| 560 | fails with the error: |
| 561 | |
| 562 | ```sh |
| 563 | ERROR: Configurable attribute "deps" doesn't match this configuration (would |
| 564 | a default condition help?). |
| 565 | Conditions checked: |
| 566 | //tools/cc_target_os:darwin |
| 567 | //tools/cc_target_os:android |
| 568 | ``` |
| 569 | |
| 570 | This can be customized with the [`no_match_error`](/reference/be/functions#select) |
| 571 | attribute: |
| 572 | |
| 573 | ```python |
| 574 | cc_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 |
| 588 | ERROR: Configurable attribute "deps" doesn't match this configuration: Please |
| 589 | build with an Android or Windows toolchain |
| 590 | ``` |
| 591 | |
| 592 | ## Rules compatibility {:#rules-compatibility} |
| 593 | |
| 594 | Rule implementations receive the *resolved values* of configurable |
| 595 | attributes. For example, given: |
| 596 | |
| 597 | ```python |
| 598 | # myapp/BUILD |
| 599 | |
| 600 | some_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 | |
| 613 | Rule implementation code sees `ctx.attr.some_attr` as `[":foo"]`. |
| 614 | |
| 615 | Macros can accept `select()` clauses and pass them through to native |
| 616 | rules. But *they cannot directly manipulate them*. For example, there's no way |
| 617 | for a macro to convert |
| 618 | |
| 619 | ```python |
| 620 | select({"foo": "val"}, ...) |
| 621 | ``` |
| 622 | |
| 623 | to |
| 624 | |
| 625 | ```python |
| 626 | select({"foo": "val_with_suffix"}, ...) |
| 627 | ``` |
| 628 | |
| 629 | This is for two reasons. |
| 630 | |
| 631 | First, macros that need to know which path a `select` will choose *cannot work* |
Googler | 0c4cf50 | 2022-12-19 13:43:43 -0800 | [diff] [blame] | 632 | because macros are evaluated in Bazel's [loading phase](/run/build#loading), |
fwe | acae1cd | 2022-02-17 09:45:38 -0800 | [diff] [blame] | 633 | which occurs before flag values are known. |
| 634 | This is a core Bazel design restriction that's unlikely to change any time soon. |
| 635 | |
| 636 | Second, macros that just need to iterate over *all* `select` paths, while |
| 637 | technically feasible, lack a coherent UI. Further design is necessary to change |
| 638 | this. |
| 639 | |
| 640 | ## Bazel query and cquery {:#query-and-cquery} |
| 641 | |
Googler | df7617e | 2022-12-20 09:44:29 -0800 | [diff] [blame] | 642 | Bazel [`query`](/query/guide) operates over Bazel's |
fwe | acae1cd | 2022-02-17 09:45:38 -0800 | [diff] [blame] | 643 | [loading phase](/reference/glossary#loading-phase). |
| 644 | This means it doesn't know what command line flags a target uses since those |
| 645 | flags aren't evaluated until later in the build (in the |
| 646 | [analysis phase](/reference/glossary#analysis-phase)). |
| 647 | So it can't determine which `select()` branches are chosen. |
| 648 | |
Googler | 0c4cf50 | 2022-12-19 13:43:43 -0800 | [diff] [blame] | 649 | Bazel [`cquery`](/query/cquery) operates after Bazel's analysis phase, so it has |
fwe | acae1cd | 2022-02-17 09:45:38 -0800 | [diff] [blame] | 650 | all this information and can accurately resolve `select()`s. |
| 651 | |
| 652 | Consider: |
| 653 | |
| 654 | ```python |
| 655 | load("@bazel_skylib//rules:common_settings.bzl", "string_flag") |
| 656 | ``` |
| 657 | ```python |
| 658 | # myapp/BUILD |
| 659 | |
| 660 | string_flag( |
| 661 | name = "dog_type", |
| 662 | build_setting_default = "cat" |
| 663 | ) |
| 664 | |
| 665 | cc_library( |
| 666 | name = "my_lib", |
| 667 | deps = select({ |
| 668 | ":long": [":foo_dep"], |
| 669 | ":short": [":bar_dep"], |
| 670 | }), |
| 671 | ) |
| 672 | |
| 673 | config_setting( |
| 674 | name = "long", |
| 675 | flag_values = {":dog_type": "dachshund"}, |
| 676 | ) |
| 677 | |
| 678 | config_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 | |
| 693 | while `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 | |
fwe | 6c2bd4a | 2022-02-18 10:47:46 -0800 | [diff] [blame] | 705 | select() *does* work in rules! See [Rules compatibility](#rules-compatibility) for |
fwe | acae1cd | 2022-02-17 09:45:38 -0800 | [diff] [blame] | 706 | details. |
| 707 | |
| 708 | The key issue this question usually means is that select() doesn't work in |
| 709 | *macros*. These are different than *rules*. See the |
Googler | 0c4cf50 | 2022-12-19 13:43:43 -0800 | [diff] [blame] | 710 | documentation on [rules](/extending/rules) and [macros](/extending/macros) |
fwe | acae1cd | 2022-02-17 09:45:38 -0800 | [diff] [blame] | 711 | to understand the difference. |
| 712 | Here's an end-to-end example: |
| 713 | |
| 714 | Define 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. |
| 721 | def _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: |
| 727 | my_custom_bazel_rule = rule( |
| 728 | implementation = _impl, |
| 729 | attrs = {"my_config_string": attr.string()}, |
| 730 | ) |
| 731 | |
| 732 | # Macro declaration: |
| 733 | def 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 | |
| 738 | Instantiate the rule and macro: |
| 739 | |
| 740 | ```python |
| 741 | # myapp/BUILD |
| 742 | |
| 743 | load("//myapp:defs.bzl", "my_custom_bazel_rule") |
| 744 | load("//myapp:defs.bzl", "my_custom_bazel_macro") |
| 745 | |
| 746 | my_custom_bazel_rule( |
| 747 | name = "happy_rule", |
| 748 | my_config_string = select({ |
Googler | ac2e8e5 | 2024-04-15 04:34:10 -0700 | [diff] [blame] | 749 | "//third_party/bazel_platforms/cpu:x86_32": "first string", |
Googler | 09c927a | 2023-05-11 06:11:11 -0700 | [diff] [blame] | 750 | "//third_party/bazel_platforms/cpu:ppc": "second string", |
fwe | acae1cd | 2022-02-17 09:45:38 -0800 | [diff] [blame] | 751 | }), |
| 752 | ) |
| 753 | |
| 754 | my_custom_bazel_macro( |
| 755 | name = "happy_macro", |
| 756 | my_config_string = "fixed string", |
| 757 | ) |
| 758 | |
| 759 | my_custom_bazel_macro( |
| 760 | name = "sad_macro", |
| 761 | my_config_string = select({ |
Googler | ac2e8e5 | 2024-04-15 04:34:10 -0700 | [diff] [blame] | 762 | "//third_party/bazel_platforms/cpu:x86_32": "first string", |
Googler | 09c927a | 2023-05-11 06:11:11 -0700 | [diff] [blame] | 763 | "//third_party/bazel_platforms/cpu:ppc": "other string", |
fwe | acae1cd | 2022-02-17 09:45:38 -0800 | [diff] [blame] | 764 | }), |
| 765 | ) |
| 766 | ``` |
| 767 | |
| 768 | Building fails because `sad_macro` can't process the `select()`: |
| 769 | |
| 770 | ```sh |
| 771 | $ bazel build //myapp:all |
| 772 | ERROR: /myworkspace/myapp/BUILD:17:1: Traceback |
| 773 | (most recent call last): |
| 774 | File "/myworkspace/myapp/BUILD", line 17 |
| 775 | my_custom_bazel_macro(name = "sad_macro", my_config_stri..."})) |
| 776 | File "/myworkspace/myapp/defs.bzl", line 4, in |
| 777 | my_custom_bazel_macro |
| 778 | my_config_string.upper() |
| 779 | type 'select' has no method upper(). |
| 780 | ERROR: error loading package 'myapp': Package 'myapp' contains errors. |
| 781 | ``` |
| 782 | |
| 783 | Building 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 |
| 788 | DEBUG: /myworkspace/myapp/defs.bzl:5:3: My name is happy_macro with custom message: FIXED STRING. |
| 789 | DEBUG: /myworkspace/myapp/hi.bzl:15:3: My name is happy_rule with custom message: FIRST STRING. |
| 790 | ``` |
| 791 | |
| 792 | This is impossible to change because *by definition* macros are evaluated before |
| 793 | Bazel reads the build's command line flags. That means there isn't enough |
| 794 | information to evaluate select()s. |
| 795 | |
| 796 | Macros can, however, pass `select()`s as opaque blobs to rules: |
| 797 | |
| 798 | ```python |
| 799 | # myapp/defs.bzl |
| 800 | |
| 801 | def 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 |
| 811 | DEBUG: /myworkspace/myapp/defs.bzl:23:3: Invoking macro sad_macro_less_sad. |
| 812 | DEBUG: /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 | |
| 817 | Because *macros* (but not rules) by definition |
| 818 | [can't evaluate `select()`s](#faq-select-macro), any attempt to do so |
| 819 | usually produces an error: |
| 820 | |
| 821 | ```sh |
| 822 | ERROR: /myworkspace/myapp/BUILD:17:1: Traceback |
| 823 | (most recent call last): |
| 824 | File "/myworkspace/myapp/BUILD", line 17 |
| 825 | my_custom_bazel_macro(name = "sad_macro", my_config_stri..."})) |
| 826 | File "/myworkspace/myapp/defs.bzl", line 4, in |
| 827 | my_custom_bazel_macro |
| 828 | my_config_string.upper() |
| 829 | type 'select' has no method upper(). |
| 830 | ``` |
| 831 | |
| 832 | Booleans are a special case that fail silently, so you should be particularly |
| 833 | vigilant with them: |
| 834 | |
| 835 | ```sh |
| 836 | $ cat myapp/defs.bzl |
| 837 | def my_boolean_macro(boolval): |
| 838 | print("TRUE" if boolval else "FALSE") |
| 839 | |
| 840 | $ cat myapp/BUILD |
| 841 | load("//myapp:defs.bzl", "my_boolean_macro") |
| 842 | my_boolean_macro( |
| 843 | boolval = select({ |
Googler | ac2e8e5 | 2024-04-15 04:34:10 -0700 | [diff] [blame] | 844 | "//third_party/bazel_platforms/cpu:x86_32": True, |
Googler | 09c927a | 2023-05-11 06:11:11 -0700 | [diff] [blame] | 845 | "//third_party/bazel_platforms/cpu:ppc": False, |
fwe | acae1cd | 2022-02-17 09:45:38 -0800 | [diff] [blame] | 846 | }), |
| 847 | ) |
| 848 | |
| 849 | $ bazel build //myapp:all --cpu=x86 |
| 850 | DEBUG: /myworkspace/myapp/defs.bzl:4:3: TRUE. |
| 851 | $ bazel build //mypro:all --cpu=ppc |
| 852 | DEBUG: /myworkspace/myapp/defs.bzl:4:3: TRUE. |
| 853 | ``` |
| 854 | |
| 855 | This happens because macros don't understand the contents of `select()`. |
| 856 | So 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 |
| 858 | standards, all objects aside from a very small number of exceptions |
| 859 | automatically return true. |
| 860 | |
| 861 | ### Can I read select() like a dict? {:#faq-inspectable-select} |
| 862 | |
| 863 | Macros [can't](#faq-select-macro) evaluate select(s) because macros evaluate before |
| 864 | Bazel knows what the build's command line parameters are. Can they at least read |
| 865 | the `select()`'s dictionary to, for example, add a suffix to each value? |
| 866 | |
Christopher Sauer | a35d2a6 | 2022-11-28 04:02:47 -0800 | [diff] [blame] | 867 | Conceptually this is possible, but [it isn't yet a Bazel feature](https://github.com/bazelbuild/bazel/issues/8419). |
fwe | acae1cd | 2022-02-17 09:45:38 -0800 | [diff] [blame] | 868 | What you *can* do today is prepare a straight dictionary, then feed it into a |
| 869 | `select()`: |
| 870 | |
| 871 | ```sh |
| 872 | $ cat myapp/defs.bzl |
| 873 | def 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 |
| 885 | selecty_genrule( |
| 886 | name = "selecty", |
| 887 | select_cmd = { |
Googler | ac2e8e5 | 2024-04-15 04:34:10 -0700 | [diff] [blame] | 888 | "//third_party/bazel_platforms/cpu:x86_32": "x86 mode", |
fwe | acae1cd | 2022-02-17 09:45:38 -0800 | [diff] [blame] | 889 | }, |
| 890 | ) |
| 891 | |
| 892 | $ bazel build //testapp:selecty --cpu=x86 && cat bazel-genfiles/testapp/selecty.out |
| 893 | x86 mode WITH SUFFIX |
| 894 | ``` |
| 895 | |
| 896 | If you'd like to support both `select()` and native types, you can do this: |
| 897 | |
| 898 | ```sh |
| 899 | $ cat myapp/defs.bzl |
| 900 | def 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 | |
Googler | 13ecdf5 | 2024-01-02 12:15:30 -0800 | [diff] [blame] | 919 | First of all, do not use `bind()`. It is deprecated in favor of `alias()`. |
fwe | acae1cd | 2022-02-17 09:45:38 -0800 | [diff] [blame] | 920 | |
Googler | 13ecdf5 | 2024-01-02 12:15:30 -0800 | [diff] [blame] | 921 | The technical answer is that [`bind()`](/reference/be/workspace#bind) is a repo |
| 922 | rule, not a BUILD rule. |
| 923 | |
| 924 | Repo rules do not have a specific configuration, and aren't evaluated in |
fwe | acae1cd | 2022-02-17 09:45:38 -0800 | [diff] [blame] | 925 | the same way as BUILD rules. Therefore, a `select()` in a `bind()` can't |
| 926 | actually evaluate to any specific branch. |
| 927 | |
| 928 | Instead, you should use [`alias()`](/reference/be/general#alias), with a `select()` in |
| 929 | the `actual` attribute, to perform this type of run-time determination. This |
| 930 | works correctly, since `alias()` is a BUILD rule, and is evaluated with a |
| 931 | specific configuration. |
| 932 | |
| 933 | You can even have a `bind()` target point to an `alias()`, if needed. |
| 934 | |
| 935 | ```sh |
| 936 | $ cat WORKSPACE |
| 937 | workspace(name = "myapp") |
| 938 | bind(name = "openssl", actual = "//:ssl") |
| 939 | http_archive(name = "alternative", ...) |
| 940 | http_archive(name = "boringssl", ...) |
| 941 | |
| 942 | $ cat BUILD |
| 943 | config_setting( |
| 944 | name = "alt_ssl", |
| 945 | define_values = { |
| 946 | "ssl_library": "alternative", |
| 947 | }, |
| 948 | ) |
| 949 | |
| 950 | alias( |
| 951 | name = "ssl", |
| 952 | actual = select({ |
| 953 | "//:alt_ssl": "@alternative//:ssl", |
| 954 | "//conditions:default": "@boringssl//:ssl", |
| 955 | }), |
| 956 | ) |
| 957 | ``` |
| 958 | |
| 959 | With this setup, you can pass `--define ssl_library=alternative`, and any target |
| 960 | that depends on either `//:ssl` or `//external:ssl` will see the alternative |
| 961 | located at `@alternative//:ssl`. |
| 962 | |
Googler | 13ecdf5 | 2024-01-02 12:15:30 -0800 | [diff] [blame] | 963 | But really, stop using `bind()`. |
| 964 | |
fwe | acae1cd | 2022-02-17 09:45:38 -0800 | [diff] [blame] | 965 | ### Why doesn't my select() choose what I expect? {:#faq-select-choose-condition} |
| 966 | |
| 967 | If `//myapp:foo` has a `select()` that doesn't choose the condition you expect, |
Googler | 0c4cf50 | 2022-12-19 13:43:43 -0800 | [diff] [blame] | 968 | use [cquery](/query/cquery) and `bazel config` to debug: |
fwe | acae1cd | 2022-02-17 09:45:38 -0800 | [diff] [blame] | 969 | |
| 970 | If `//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 | |
| 977 | If 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 | |
| 987 | The `(12e23b9a2b534a)` next to `//myapp:foo` is a *hash* of the |
| 988 | configuration that resolves `//myapp:foo`'s `select()`. You can inspect its |
| 989 | values with `bazel config`: |
| 990 | |
| 991 | ```sh |
| 992 | $ bazel config 12e23b9a2b534a |
| 993 | BuildConfigurationValue 12e23b9a2b534a |
| 994 | Fragment com.google.devtools.build.lib.analysis.config.CoreOptions { |
| 995 | cpu: darwin |
| 996 | compilation_mode: fastbuild |
| 997 | ... |
| 998 | } |
| 999 | Fragment com.google.devtools.build.lib.rules.cpp.CppOptions { |
| 1000 | linkopt: [-Dfoo=bar] |
| 1001 | ... |
| 1002 | } |
| 1003 | ... |
| 1004 | ``` |
| 1005 | |
| 1006 | Then 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 |
Googler | 0c4cf50 | 2022-12-19 13:43:43 -0800 | [diff] [blame] | 1009 | [cquery docs](/query/cquery) for guidance on using `somepath` to get the right |
fwe | acae1cd | 2022-02-17 09:45:38 -0800 | [diff] [blame] | 1010 | one. |
| 1011 | |
| 1012 | Caution: To prevent restarting the Bazel server, invoke `bazel config` with the |
| 1013 | same command line flags as the `bazel cquery`. The `config` command relies on |
| 1014 | the configuration nodes from the still-running server of the previous command. |
| 1015 | |
John Cater | e0e5896 | 2022-05-26 08:52:08 -0700 | [diff] [blame] | 1016 | ### Why doesn't `select()` work with platforms? {:#faq-select-platforms} |
| 1017 | |
| 1018 | Bazel doesn't support configurable attributes checking whether a given platform |
| 1019 | is the target platform because the semantics are unclear. |
| 1020 | |
| 1021 | For example: |
| 1022 | |
| 1023 | ```py |
| 1024 | platform( |
| 1025 | name = "x86_linux_platform", |
| 1026 | constraint_values = [ |
| 1027 | "@platforms//cpu:x86", |
| 1028 | "@platforms//os:linux", |
| 1029 | ], |
| 1030 | ) |
| 1031 | |
| 1032 | cc_library( |
| 1033 | name = "lib", |
| 1034 | srcs = [...], |
| 1035 | linkopts = select({ |
| 1036 | ":x86_linux_platform": ["--enable_x86_optimizations"], |
| 1037 | "//conditions:default": [], |
| 1038 | }), |
| 1039 | ) |
| 1040 | ``` |
| 1041 | |
| 1042 | In 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 |
| 1045 | who defined the separate platform may have different ideas. |
| 1046 | |
| 1047 | #### What should I do instead? |
| 1048 | |
| 1049 | Instead, define a `config_setting` that matches **any** platform with |
| 1050 | these constraints: |
| 1051 | |
| 1052 | ```py |
| 1053 | config_setting( |
| 1054 | name = "is_x86_linux", |
| 1055 | constraint_values = [ |
| 1056 | "@platforms//cpu:x86", |
| 1057 | "@platforms//os:linux", |
| 1058 | ], |
| 1059 | ) |
| 1060 | |
| 1061 | cc_library( |
| 1062 | name = "lib", |
| 1063 | srcs = [...], |
| 1064 | linkopts = select({ |
| 1065 | ":is_x86_linux": ["--enable_x86_optimizations"], |
| 1066 | "//conditions:default": [], |
| 1067 | }), |
| 1068 | ) |
| 1069 | ``` |
| 1070 | |
| 1071 | This process defines specific semantics, making it clearer to users what |
| 1072 | platforms meet the desired conditions. |
| 1073 | |
| 1074 | #### What if I really, really want to `select` on the platform? |
| 1075 | |
| 1076 | If your build requirements specifically require checking the platform, you |
| 1077 | can flip the value of the `--platforms` flag in a `config_setting`: |
| 1078 | |
| 1079 | ```py |
| 1080 | config_setting( |
| 1081 | name = "is_specific_x86_linux_platform", |
| 1082 | values = { |
| 1083 | "platforms": ["//package:x86_linux_platform"], |
| 1084 | }, |
| 1085 | ) |
| 1086 | |
| 1087 | cc_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 | |
| 1097 | The Bazel team doesn't endorse doing this; it overly constrains your build and |
| 1098 | confuses users when the expected condition does not match. |
| 1099 | |
Googler | df7617e | 2022-12-20 09:44:29 -0800 | [diff] [blame] | 1100 | [BuildSettings]: /extending/config#user-defined-build-settings |