-
Notifications
You must be signed in to change notification settings - Fork 13k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Tracking issue for cfg_match
#115585
Comments
I think it needs to be considered how this will interact with Also, if this is intended to be |
Modulo (auto) doc(cfg), this implementation is more complex than it needs to be, since it's only supporting items. simpler impl#[macro_export]
macro_rules! cfg_match {
{
cfg($cfg:meta) => { $($cfg_items:item)* }
$(cfg($rest_cfg:meta) => { $($rest_cfg_items:item)* })*
$(_ => { $($fallthrough_items:item)* })?
} => {
#[cfg($cfg)]
$crate::cfg_match! {
_ => {
$($cfg_items)*
// validate the inactive cfgs as well
#[cfg(any($($rest_cfg),*))]
const _: () = ();
}
}
#[cfg(not($cfg))]
$crate::cfg_match! {
$(cfg($rest_cfg) => { $($rest_cfg_items)* })*
$(_ => { $($fallthrough_items)* })?
}
};
{
$(_ => { $($items:item)* })?
} => {
$($($items)*)?
};
} No internal matching arms necessary; just a normal inductive definition over the input syntax. The one tricky difference is that without the extra This can be rather simply adapted to expression position by matching a block and using It's unfortunate though that implemented without compiler support |
Something else to consider: This should be able to detect invalid cfgs like the following (ideally regardless of whether the match_cfg! {
cfg(unix) => {}
cfg(unix) => {} // Ooops, specified same `cfg` twice
_ => {}
} |
I would prefer if it could support both items and expressions. It being part of the stdlib means that implementing it in the compiler would be completely reasonable, and would also make a lint that warns on redundant cfg arms pretty doable. |
For detecting duplicates, it's important to note that this is more difficult than it seems at first. rustdoc has been dealing with this for |
I tested the current (1.74.0-nightly (e6aabe8b3 2023-09-26)) behavior of doc_auto_cfg: items defined in Fixing this is actually fairly simple once you understand why this is: instead of expanding to I want to additionally point out that doc_auto_cfg fundamentally can only see what items are defined on the current cfg and mark them as only available on whatever cfg this specific implementation is gated to. If you want to expose a portable API, either define a single portable public version of the API (and use private cfg gates to implement it) or use So what I personally think the macro definition wants to be:
in item position#[macro_export]
macro_rules! cfg_match {
// base case without wildcard arm
() => {};
// base case with wildcard arm (items)
(_ => { $($wild_item:item)* }) => {
$($wild_item)*
};
// process first arm (items; propagate cfg)
(
cfg($this_cfg:meta) => { $($this_item:item)* }
$(cfg($rest_cfg:meta) => { $($rest_item:item)* })*
$(_ => { $($wild_item:item)* })?
) => {
$(#[cfg($this_cfg)] $this_item)*
$crate::cfg_match! {
$(cfg($rest_cfg) => { $(#[cfg(not($this_cfg))] $rest_item)* })*
$(_ => { $(#[cfg(not($this_cfg))] $wild_item)* })?
}
};
} in expression position#[macro_export]
macro_rules! cfg_match {
// base case without wildcard arm
() => {};
// base case with wildcard arm (expression block)
(_ => $wild:block) => {
$wild
};
// process first arm (expression block)
(
cfg($this_cfg:meta) => $this:block
$(cfg($rest_cfg:meta) => $rest:block)*
$(_ => $wild:block)?
) => {
{
#[cfg($this_cfg)]
$crate::cfg_match! {
_ => $this
}
#[cfg(not($this_cfg))]
$crate::cfg_match! {
$(cfg($rest_cfg) => $rest)*
$(_ => $wild)?
}
}
};
} in statement position#[macro_export]
macro_rules! cfg_match {
// base case without wildcard arm
() => {};
// base case with wildcard arm
(_ => { $($wild:tt)* }) => {
$($wild)*
};
// process first arm
(
cfg($this_cfg:meta) => { $($this:tt)* }
$(cfg($rest_cfg:meta) => { $($rest:tt)* })*
$(_ => { $($wild:tt)* })?
) => {
#[cfg($this_cfg)]
$crate::cfg_match! {
_ => { $($this)* }
}
#[cfg(not($this_cfg))]
$crate::cfg_match! {
$(cfg($rest_cfg) => { $($rest)* })*
$(_ => { $($wild)* })?
}
};
} This technically is the most flexible spelling just passing through tt sequences; it works perfectly for items and statements (modulo preserving doc_auto_cfg) and can even be used for expressions if wrapped inside an outer block (e.g. There's also the further question of how much syntax checking is expected for inactive arms. The current implementation that matches everything as A far too overly clever implementation:#[macro_export]
macro_rules! cfg_match {
// base case (wildcard arm)
() => {};
// base case (no wildcard arm)
(_ => { $($wild:tt)* }) => {
$($wild)*
};
// Note: macro patterns matching $:stmt need to come before those matching
// $:item, as otherwise statements will match as the $:item pattern instead
// even when not valid as items (and obviously thus cause a syntax error).
// apply cfg to wildcard arm (expression position)
(_ => #[cfg($cfg:meta)] { $wild:expr }) => {
{ #[cfg($cfg)] { $wild } }
};
// apply cfg to wildcard arm (statement position)
(_ => #[cfg($cfg:meta)] { $wild_stmt:stmt; $($wild:tt)* }) => {
#[cfg($cfg)]
$crate::cfg_match! {
_ => { $wild_stmt; $($wild)* }
}
// We only parse the first statement in the macro pattern, so we need to
// emit the captured syntax even when the cfg is inactive such that any
// syntax errors still get reported. We do the macro capture this way as
// if we match as statements, minor typos try to parse as items instead.
#[cfg(any())]
const _: () = { wild_stmt; $($wild)* };
};
// apply cfg to wildcard arm (item position)
(_ => #[cfg($cfg:meta)] { $($wild:item)* }) => {
$(#[cfg($cfg)] $wild)*
};
// merge multiple cfg into a single cfg(all()) predicate
(_ => $(#[cfg($cfg:meta)])+ { $($wild:tt)* }) => {
$crate::cfg_match! {
_ => #[cfg(all($($cfg),+))] { $($wild)* }
}
};
// split off first arm (empty)
(
cfg($this_cfg_arm:meta) => $(#[cfg($this_cfg_pre:meta)])* { /* empty */ }
$(cfg($rest_cfg_arm:meta) => $(#[cfg($rest_cfg_pre:meta)])* { $($rest:tt)* })*
$(_ => $(#[cfg($wild_cfg_pre:meta)])* { $($wild:tt)* })?
) => {
$crate::cfg_match! {
$(cfg($rest_cfg_arm) => $(#[cfg($rest_cfg_pre)])* #[cfg(not($this_cfg_arm))] { $($rest)* })*
$(_ => $(#[cfg($wild_cfg_pre)])* #[cfg(not($this_cfg_arm))] { $($wild)* })?
}
};
// split off first arm (expression position)
(
cfg($this_cfg_arm:meta) => $(#[cfg($this_cfg_pre:meta)])* { $this:expr }
$(cfg($rest_cfg_arm:meta) => $(#[cfg($rest_cfg_pre:meta)])* { $rest:expr })*
$(_ => $(#[cfg($wild_cfg_pre:meta)])* { $wild:expr })?
) => {{
$crate::cfg_match! {
_ => $(#[cfg($this_cfg_pre)])* #[cfg($this_cfg_arm)] { $this }
}
$crate::cfg_match! {
$(cfg($rest_cfg_arm) => $(#[cfg($rest_cfg_pre)])* #[cfg(not($this_cfg_arm))] { $rest })*
$(_ => $(#[cfg($wild_cfg_pre)])* #[cfg(not($this_cfg_arm))] { $wild })?
}
}};
// split off first arm (statement position)
(
cfg($this_cfg_arm:meta) => $(#[cfg($this_cfg_pre:meta)])* { $this_stmt:stmt; $($this:tt)* }
$(cfg($rest_cfg_arm:meta) => $(#[cfg($rest_cfg_pre:meta)])* { $($rest:tt)* })*
$(_ => $(#[cfg($wild_cfg_pre:meta)])* { $($wild:tt)* })?
) => {
$crate::cfg_match! {
_ => $(#[cfg($this_cfg_pre)])* #[cfg($this_cfg_arm)] { $this_stmt; $($this)* }
}
// Ensure all arms infer statement position by prefixing a statement. We
// only match the first arm in the macro pattern because otherwise minor
// typos unhelpfully cause all arms to parse as item definitions instead
// and thus reporting an error nowhere near the actual problem, or even
// more amusingly, accidentally providing automatic semicolon insertion.
$crate::cfg_match! {
$(cfg($rest_cfg_arm) => $(#[cfg($rest_cfg_pre)])* #[cfg(not($this_cfg_arm))] { {}; $($rest)* })*
$(_ => $(#[cfg($wild_cfg_pre)])* #[cfg(not($this_cfg_arm))] { {}; $($wild)* })?
}
};
// split off first arm (item position)
(
cfg($this_cfg_arm:meta) => $(#[cfg($this_cfg_pre:meta)])* { $($this:item)* }
$(cfg($rest_cfg_arm:meta) => $(#[cfg($rest_cfg_pre:meta)])* { $($rest:item)* })*
$(_ => $(#[cfg($wild_cfg_pre:meta)])* { $($wild:item)* })?
) => {
$crate::cfg_match! {
_ => $(#[cfg($this_cfg_pre)])* #[cfg($this_cfg_arm)] { $($this)* }
}
// Items just match as item in the macro pattern. The error messages
// would be improved if we could pass through tokens without eagerly
// matching them as items, but we need to parse the items in the macro
// so that we can apply the cfg attributes to every item in the arm.
$crate::cfg_match! {
$(cfg($rest_cfg_arm) => $(#[cfg($rest_cfg_pre)])* #[cfg(not($this_cfg_arm))] { $($rest)* })*
$(_ => $(#[cfg($wild_cfg_pre)])* #[cfg(not($this_cfg_arm))] { $($wild)* })?
}
};
} This does almost always behave as desired by using the arm bodies to try to guess how to expand, but is unreasonably complicated by attempting to do so and as a result causes some very poor errors for reasonable syntax errors. |
Well, since #115416 (comment) I have been working in transposing It is worth noting that the current implementation mimics cfg_match! {
cfg(unix) => { fn foo() -> i32 { 1 } },
_ => { fn foo() -> i32 { 2 } },
}
# Expands to ->
#[cfg(all(unix, not(any())))]
cfg_match! { @__identity fn foo() -> i32 { 1 } }
#[cfg(all(not(any(unix))))]
cfg_match! { @__identity fn foo() -> i32 { 2 } } That said, changing the strategy to apply |
FWIW, What we need to do is to keep expansion backtraces for |
Initiate the inner usage of `cfg_match` (Compiler) cc rust-lang#115585 Dogfood to test the implementation and remove dependencies.
Initiate the inner usage of `cfg_match` (Compiler) cc rust-lang#115585 Dogfood to test the implementation and remove dependencies.
Initiate the inner usage of `cfg_match` (Compiler) cc rust-lang#115585 Dogfood to test the implementation and remove dependencies.
Rollup merge of rust-lang#116312 - c410-f3r:try, r=Mark-Simulacrum Initiate the inner usage of `cfg_match` (Compiler) cc rust-lang#115585 Dogfood to test the implementation and remove dependencies.
Initiate the inner usage of `cfg_match` (Library) cc rust-lang#115585 Continuation of rust-lang#116312
Remove `cfg_match` from the prelude Fixes rust-lang#117057 cc rust-lang#115585
Remove `cfg_match` from the prelude Fixes rust-lang#117057 cc rust-lang#115585
Rollup merge of rust-lang#117162 - c410-f3r:try, r=workingjubilee Remove `cfg_match` from the prelude Fixes rust-lang#117057 cc rust-lang#115585
Initiate the inner usage of `cfg_match` (Library) cc rust-lang#115585 Continuation of rust-lang#116312
Initiate the inner usage of `cfg_match` (Library) cc rust-lang#115585 Continuation of rust-lang#116312
Initiate the inner usage of `cfg_match` (Library) cc rust-lang#115585 Continuation of rust-lang#116312
I just found out about this today, and my first impression is that semantically this isn't really a " match () {
() if unix => { ... }
() if target_pointer_width.contains("32") => { ... }
_ => { ... }
} A more similar construct in my mind are the All that is to say that |
[`cfg_match`] Generalize inputs cc rust-lang#115585 Changes the input type from `item` to `tt`, which makes the macro have the same functionality of `cfg_if`. Also adds a test to ensure that `stmt_expr_attributes` is not triggered.
[`cfg_match`] Generalize inputs cc rust-lang#115585 Changes the input type from `item` to `tt`, which makes the macro have the same functionality of `cfg_if`. Also adds a test to ensure that `stmt_expr_attributes` is not triggered.
Rollup merge of rust-lang#130313 - c410-f3r:unlock-rfc-2011, r=thomcc [`cfg_match`] Generalize inputs cc rust-lang#115585 Changes the input type from `item` to `tt`, which makes the macro have the same functionality of `cfg_if`. Also adds a test to ensure that `stmt_expr_attributes` is not triggered.
The syntax of At least for me, it became clear in #116342 that match_cfg! {
unix => {
fn foo() {}
}
target_pointer_width = "32" => {
fn foo() {}
}
_ => {
fn foo() {}
}
} The functionality provided by this macro is useful and can positively impact several projects. If possible, it would be awesome to seek consensus as soon as possible to create a path for stabilization. |
At least in std we rely on both, cfg-if'd sections in function bodies and defining items, e.g. platform-specific modules. See #126827 (comment) for an example where platform-dependent definitions + other macros + ordering constraints on interacted poorly enough that outsourcing things to a module would have been tricky. |
…joshtriplett [cfg_match] Adjust syntax A year has passed since the creation of rust-lang#115585 and the feature, as expected, is not moving forward. Let's change that. This PR proposes changing the arm's syntax from `cfg(SOME_CONDITION) => { ... }` to `SOME_CODITION => {}`. ```rust match_cfg! { unix => { fn foo() { /* unix specific functionality */ } } target_pointer_width = "32" => { fn foo() { /* non-unix, 32-bit functionality */ } } _ => { fn foo() { /* fallback implementation */ } } } ``` Why? Because after several manual migrations in rust-lang#116342 it became clear, at least for me, that `cfg` prefixes are unnecessary, verbose and redundant. Again, everything is just a proposal to move things forward. If the shown syntax isn't ideal, feel free to close this PR or suggest other alternatives.
…joshtriplett [cfg_match] Adjust syntax A year has passed since the creation of rust-lang#115585 and the feature, as expected, is not moving forward. Let's change that. This PR proposes changing the arm's syntax from `cfg(SOME_CONDITION) => { ... }` to `SOME_CODITION => {}`. ```rust match_cfg! { unix => { fn foo() { /* unix specific functionality */ } } target_pointer_width = "32" => { fn foo() { /* non-unix, 32-bit functionality */ } } _ => { fn foo() { /* fallback implementation */ } } } ``` Why? Because after several manual migrations in rust-lang#116342 it became clear, at least for me, that `cfg` prefixes are unnecessary, verbose and redundant. Again, everything is just a proposal to move things forward. If the shown syntax isn't ideal, feel free to close this PR or suggest other alternatives.
…joshtriplett [cfg_match] Adjust syntax A year has passed since the creation of rust-lang#115585 and the feature, as expected, is not moving forward. Let's change that. This PR proposes changing the arm's syntax from `cfg(SOME_CONDITION) => { ... }` to `SOME_CODITION => {}`. ```rust match_cfg! { unix => { fn foo() { /* unix specific functionality */ } } target_pointer_width = "32" => { fn foo() { /* non-unix, 32-bit functionality */ } } _ => { fn foo() { /* fallback implementation */ } } } ``` Why? Because after several manual migrations in rust-lang#116342 it became clear, at least for me, that `cfg` prefixes are unnecessary, verbose and redundant. Again, everything is just a proposal to move things forward. If the shown syntax isn't ideal, feel free to close this PR or suggest other alternatives.
Provides a native way to easily manage multiple conditional flags without having to rewrite each clause multiple times.
Public API
Steps / History
cfg_match!
macro #115416)Unresolved Questions
What should the final syntax be? A match-like syntax feels more natural in the sense that each macro fragment resembles an arm.What should the feature name be?cfg_match
conflicts with the already existingcfg_match
crate.How can we support usage in both expression-position and item position?References
cfg_if
#59442The text was updated successfully, but these errors were encountered: