blob: c5a499f5708cc953d3bc4008e4d1490b0d414481 [file] [log] [blame] [view] [edit]
# Feature Gate Checking
For the how-to steps to add, remove, rename, or stabilize feature gates,
see [Feature gates][feature-gates].
Feature gates prevent usage of unstable language and library features without a
nightly-only `#![feature(...)]` opt-in. This chapter documents the implementation
of feature gating: where gates are defined, how they are enabled, and how usage
is verified.
<!-- data-check: Feb 2026 -->
## Feature Definitions
All feature gate definitions are located in the `rustc_feature` crate:
- **Unstable features** are declared in [`rustc_feature/src/unstable.rs`] via
the `declare_features!` macro. This associates features with issue numbers and
tracking metadata.
- **Accepted features** (stabilized) are listed in [`rustc_feature/src/accepted.rs`].
- **Removed features** (explicitly disallowed) are listed in [`rustc_feature/src/removed.rs`].
- **Gated built-in attributes and cfgs** are declared in [`rustc_feature/src/builtin_attrs.rs`].
The [`rustc_feature::Features`] type represents the **active feature set** for a
crate. Helpers like `enabled`, `incomplete`, and `internal` are used during
compilation to check status.
## Collecting Features
Before AST validation or expansion, `rustc` collects crate-level
`#![feature(...)]` attributes to build the active `Features` set.
- The collection happens in [`rustc_expand/src/config.rs`] in [`features`].
- Each `#![feature]` entry is classified against the `unstable`, `accepted`, and
`removed` tables:
- **Removed** features cause an immediate error.
- **Accepted** features are recorded but do not require nightly. On
stable/beta, `maybe_stage_features` in
[`rustc_ast_passes/src/feature_gate.rs`] emits the non-nightly
diagnostic and lists stable features, which is where the "already
stabilized" messaging comes from.
- **Unstable** features are recorded as enabled.
- Unknown features are treated as **library features** and validated later.
- With `-Z allow-features=...`, any **unstable** or **unknown** feature
not in the allowlist is rejected.
- [`RUSTC_BOOTSTRAP`] feeds into `UnstableFeatures::from_environment`. This
variable controls whether the compiler is treated as "nightly", allowing
feature gates to be bypassed during bootstrapping or explicitly disabled (`-1`).
## Parser Gating
Some syntax is detected and gated during parsing. The parser records spans for
later checking to keep diagnostics consistent and deferred until after parsing.
- [`rustc_session/src/parse.rs`] defines [`GatedSpans`] and the `gate` method.
- The parser uses it in [`rustc_parse/src/parser/*`] when it encounters
syntax that requires a gate (e.g., `async for`, `yield`, experimental patterns).
## Checking Pass
The central logic lives in [`rustc_ast_passes/src/feature_gate.rs`], primarily
in `check_crate` and its AST visitor.
### `check_crate`
`check_crate` performs high-level validation:
- `maybe_stage_features`: Rejects `#![feature]` on stable/beta.
- `check_incompatible_features`: Ensures incompatible feature combinations
(declared in `rustc_feature::INCOMPATIBLE_FEATURES`) are not used together.
- `check_new_solver_banned_features`: Bans features incompatible with
compiler mode for the next trait solver.
- **Parser-gated spans**: Processes the `GatedSpans` recorded during parsing
(see [Checking `GatedSpans`](#checking-gatedspans)).
### Checking `GatedSpans`
`check_crate` iterates over `sess.psess.gated_spans`:
- The `gate_all!` macro emits diagnostics for each gated span if the feature is
not enabled.
- Some gates have extra logic (e.g., `yield` can be allowed by `coroutines` or
`gen_blocks`).
- Legacy gates (e.g., `box_patterns`, `try_blocks`) may use a separate path that
emits future-incompatibility warnings instead of hard errors.
### AST Visitor
A `PostExpansionVisitor` walks the expanded AST to check constructs that are
easier to validate after expansion.
- The visitor uses helper macros (`gate!`, `gate_alt!`, `gate_multi!`) to check:
1. Is the feature enabled?
2. Does `span.allows_unstable` permit it (for internal compiler macros)?
- Examples include `trait_alias`, `decl_macro`, `extern types`, and various
`impl Trait` forms.
## Attributes and `cfg`
Beyond syntax, rustc also gates attributes and `cfg` options.
### Built-in attributes
- [`rustc_ast_passes::check_attribute`] inspects attributes against
`BUILTIN_ATTRIBUTE_MAP`.
- If the attribute is `AttributeGate::Gated` and the feature isnt enabled,
`feature_err` is emitted.
### `cfg` options
- [`rustc_attr_parsing/src/attributes/cfg.rs`] defines `gate_cfg` and uses
[`rustc_feature::find_gated_cfg`] to reject gated `cfg`s.
- `gate_cfg` respects `Span::allows_unstable`, allowing internal compiler
macros to bypass `cfg` gates when marked with `#[allow_internal_unstable]`.
- The gated cfg list is defined in [`rustc_feature/src/builtin_attrs.rs`].
## Diagnostics
Diagnostic helpers are located in [`rustc_session/src/parse.rs`].
- `feature_err` and `feature_warn` emit standardized diagnostics, attaching the
tracking issue number where possible.
- `Span::allows_unstable` in [`rustc_span/src/lib.rs`] checks if a span originates
from a macro marked with `#[allow_internal_unstable]`. This allows internal
macros to use unstable features on stable channels while enforcing gates for
user code.
[`rustc_feature/src/unstable.rs`]: https://github.com/rust-lang/rust/blob/HEAD/compiler/rustc_feature/src/unstable.rs
[`rustc_feature/src/removed.rs`]: https://github.com/rust-lang/rust/blob/HEAD/compiler/rustc_feature/src/removed.rs
[`rustc_feature/src/accepted.rs`]: https://github.com/rust-lang/rust/blob/HEAD/compiler/rustc_feature/src/accepted.rs
[`rustc_feature/src/builtin_attrs.rs`]: https://github.com/rust-lang/rust/blob/HEAD/compiler/rustc_feature/src/builtin_attrs.rs
[`rustc_feature::Features`]: https://doc.rust-lang.org/nightly/nightly-rustc/rustc_feature/struct.Features.html
[`rustc_expand/src/config.rs`]: https://github.com/rust-lang/rust/blob/HEAD/compiler/rustc_expand/src/config.rs
[`features`]: https://doc.rust-lang.org/nightly/nightly-rustc/rustc_expand/config/fn.features.html
[`RUSTC_BOOTSTRAP`]: https://doc.rust-lang.org/beta/unstable-book/compiler-environment-variables/RUSTC_BOOTSTRAP.html
[`rustc_session/src/parse.rs`]: https://github.com/rust-lang/rust/blob/HEAD/compiler/rustc_session/src/parse.rs
[`GatedSpans`]: https://doc.rust-lang.org/nightly/nightly-rustc/rustc_session/parse/struct.GatedSpans.html
[`rustc_ast_passes/src/feature_gate.rs`]: https://github.com/rust-lang/rust/blob/HEAD/compiler/rustc_ast_passes/src/feature_gate.rs
[`rustc_parse/src/parser/*`]: https://doc.rust-lang.org/nightly/nightly-rustc/rustc_parse/parser/index.html
[`rustc_ast_passes::check_attribute`]: https://doc.rust-lang.org/nightly/nightly-rustc/rustc_ast_passes/feature_gate/fn.check_attribute.html
[`rustc_attr_parsing/src/attributes/cfg.rs`]: https://github.com/rust-lang/rust/blob/HEAD/compiler/rustc_attr_parsing/src/attributes/cfg.rs
[`rustc_feature::find_gated_cfg`]: https://doc.rust-lang.org/nightly/nightly-rustc/rustc_feature/fn.find_gated_cfg.html
[`rustc_span/src/lib.rs`]: https://github.com/rust-lang/rust/blob/HEAD/compiler/rustc_span/src/lib.rs
[feature-gates]: ./feature-gates.md