Skip to content

Commit 92d802e

Browse files
committed
expand: Leave traces when expanding cfg attributes
1 parent 65899c0 commit 92d802e

File tree

24 files changed

+88
-108
lines changed

24 files changed

+88
-108
lines changed

compiler/rustc_ast_passes/src/ast_validation.rs

+1-2
Original file line numberDiff line numberDiff line change
@@ -334,8 +334,7 @@ impl<'a> AstValidator<'a> {
334334
.filter(|attr| {
335335
let arr = [
336336
sym::allow,
337-
sym::cfg,
338-
sym::cfg_attr,
337+
sym::cfg_trace,
339338
sym::cfg_attr_trace,
340339
sym::deny,
341340
sym::expect,

compiler/rustc_ast_pretty/src/pprust/state.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -593,7 +593,7 @@ pub trait PrintState<'a>: std::ops::Deref<Target = pp::Printer> + std::ops::Dere
593593
}
594594

595595
fn print_attribute_inline(&mut self, attr: &ast::Attribute, is_inline: bool) -> bool {
596-
if attr.has_name(sym::cfg_attr_trace) {
596+
if attr.has_name(sym::cfg_trace) || attr.has_name(sym::cfg_attr_trace) {
597597
// It's not a valid identifier, so avoid printing it
598598
// to keep the printed code reasonably parse-able.
599599
return false;

compiler/rustc_expand/src/config.rs

+14-10
Original file line numberDiff line numberDiff line change
@@ -156,6 +156,19 @@ pub fn pre_configure_attrs(sess: &Session, attrs: &[Attribute]) -> ast::AttrVec
156156
.collect()
157157
}
158158

159+
pub(crate) fn attr_into_trace(mut attr: Attribute, trace_name: Symbol) -> Attribute {
160+
match &mut attr.kind {
161+
AttrKind::Normal(normal) => {
162+
let NormalAttr { item, tokens } = &mut **normal;
163+
item.path.segments[0].ident.name = trace_name;
164+
// This makes the trace attributes unobservable to token-based proc macros.
165+
*tokens = Some(LazyAttrTokenStream::new(AttrTokenStream::default()));
166+
}
167+
AttrKind::DocComment(..) => unreachable!(),
168+
}
169+
attr
170+
}
171+
159172
#[macro_export]
160173
macro_rules! configure {
161174
($this:ident, $node:ident) => {
@@ -280,16 +293,7 @@ impl<'a> StripUnconfigured<'a> {
280293

281294
// A trace attribute left in AST in place of the original `cfg_attr` attribute.
282295
// It can later be used by lints or other diagnostics.
283-
let mut trace_attr = cfg_attr.clone();
284-
match &mut trace_attr.kind {
285-
AttrKind::Normal(normal) => {
286-
let NormalAttr { item, tokens } = &mut **normal;
287-
item.path.segments[0].ident.name = sym::cfg_attr_trace;
288-
// This makes the trace attributes unobservable to token-based proc macros.
289-
*tokens = Some(LazyAttrTokenStream::new(AttrTokenStream::default()));
290-
}
291-
AttrKind::DocComment(..) => unreachable!(),
292-
}
296+
let trace_attr = attr_into_trace(cfg_attr.clone(), sym::cfg_attr_trace);
293297

294298
let Some((cfg_predicate, expanded_attrs)) =
295299
rustc_parse::parse_cfg_attr(cfg_attr, &self.sess.psess)

compiler/rustc_expand/src/expand.rs

+6-7
Original file line numberDiff line numberDiff line change
@@ -33,7 +33,7 @@ use rustc_span::{ErrorGuaranteed, FileName, Ident, LocalExpnId, Span, sym};
3333
use smallvec::SmallVec;
3434

3535
use crate::base::*;
36-
use crate::config::StripUnconfigured;
36+
use crate::config::{StripUnconfigured, attr_into_trace};
3737
use crate::errors::{
3838
EmptyDelegationMac, GlobDelegationOutsideImpls, GlobDelegationTraitlessQpath, IncompleteParse,
3939
RecursionLimitReached, RemoveExprNotSupported, RemoveNodeNotSupported, UnsupportedKeyValue,
@@ -2003,7 +2003,7 @@ impl<'a, 'b> InvocationCollector<'a, 'b> {
20032003
let attr_name = attr.ident().unwrap().name;
20042004
// `#[cfg]` and `#[cfg_attr]` are special - they are
20052005
// eagerly evaluated.
2006-
if attr_name != sym::cfg && attr_name != sym::cfg_attr_trace {
2006+
if attr_name != sym::cfg_trace && attr_name != sym::cfg_attr_trace {
20072007
self.cx.sess.psess.buffer_lint(
20082008
UNUSED_ATTRIBUTES,
20092009
attr.span,
@@ -2027,11 +2027,10 @@ impl<'a, 'b> InvocationCollector<'a, 'b> {
20272027
) -> (bool, Option<ast::MetaItem>) {
20282028
let (res, meta_item) = self.cfg().cfg_true(&attr);
20292029
if res {
2030-
// FIXME: `cfg(TRUE)` attributes do not currently remove themselves during expansion,
2031-
// and some tools like rustdoc and clippy rely on that. Find a way to remove them
2032-
// while keeping the tools working.
2033-
self.cx.expanded_inert_attrs.mark(&attr);
2034-
node.visit_attrs(|attrs| attrs.insert(pos, attr));
2030+
// A trace attribute left in AST in place of the original `cfg` attribute.
2031+
// It can later be used by lints or other diagnostics.
2032+
let trace_attr = attr_into_trace(attr, sym::cfg_trace);
2033+
node.visit_attrs(|attrs| attrs.insert(pos, trace_attr));
20352034
}
20362035

20372036
(res, meta_item)

compiler/rustc_feature/src/builtin_attrs.rs

+8-4
Original file line numberDiff line numberDiff line change
@@ -760,10 +760,14 @@ pub static BUILTIN_ATTRIBUTES: &[BuiltinAttribute] = &[
760760
template!(Word, List: r#""...""#), DuplicatesOk,
761761
EncodeCrossCrate::Yes, INTERNAL_UNSTABLE
762762
),
763-
// Trace that is left when a `cfg_attr` attribute is expanded.
764-
// The attribute is not gated, to avoid stability errors, but it cannot be used in stable or
765-
// unstable code directly because `sym::cfg_attr_trace` is not a valid identifier, it can only
766-
// be generated by the compiler.
763+
// Traces that are left when `cfg` and `cfg_attr` attributes are expanded.
764+
// The attributes are not gated, to avoid stability errors, but they cannot be used in stable
765+
// or unstable code directly because `sym::cfg_(attr_)trace` are not valid identifiers, they
766+
// can only be generated by the compiler.
767+
ungated!(
768+
cfg_trace, Normal, template!(Word /* irrelevant */), DuplicatesOk,
769+
EncodeCrossCrate::No
770+
),
767771
ungated!(
768772
cfg_attr_trace, Normal, template!(Word /* irrelevant */), DuplicatesOk,
769773
EncodeCrossCrate::No

compiler/rustc_parse/src/validate_attr.rs

+3-6
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,8 @@ use rustc_span::{Span, Symbol, sym};
1616
use crate::{errors, parse_in};
1717

1818
pub fn check_attr(psess: &ParseSess, attr: &Attribute) {
19-
if attr.is_doc_comment() || attr.has_name(sym::cfg_attr_trace) {
19+
if attr.is_doc_comment() || attr.has_name(sym::cfg_trace) || attr.has_name(sym::cfg_attr_trace)
20+
{
2021
return;
2122
}
2223

@@ -215,11 +216,7 @@ pub fn check_builtin_meta_item(
215216
template: AttributeTemplate,
216217
deny_unsafety: bool,
217218
) {
218-
// Some special attributes like `cfg` must be checked
219-
// before the generic check, so we skip them here.
220-
let should_skip = |name| name == sym::cfg;
221-
222-
if !should_skip(name) && !is_attr_template_compatible(&template, &meta.kind) {
219+
if !is_attr_template_compatible(&template, &meta.kind) {
223220
emit_malformed_attribute(psess, style, meta.span, name, template);
224221
}
225222

compiler/rustc_passes/src/check_attr.rs

+3-3
Original file line numberDiff line numberDiff line change
@@ -272,6 +272,7 @@ impl<'tcx> CheckAttrVisitor<'tcx> {
272272
| sym::forbid
273273
| sym::cfg
274274
| sym::cfg_attr
275+
| sym::cfg_trace
275276
| sym::cfg_attr_trace
276277
// need to be fixed
277278
| sym::cfi_encoding // FIXME(cfi_encoding)
@@ -574,8 +575,7 @@ impl<'tcx> CheckAttrVisitor<'tcx> {
574575
// NOTE: when making changes to this list, check that `error_codes/E0736.md` remains accurate
575576
const ALLOW_LIST: &[rustc_span::Symbol] = &[
576577
// conditional compilation
577-
sym::cfg,
578-
sym::cfg_attr,
578+
sym::cfg_trace,
579579
sym::cfg_attr_trace,
580580
// testing (allowed here so better errors can be generated in `rustc_builtin_macros::test`)
581581
sym::test,
@@ -2656,7 +2656,7 @@ impl<'tcx> Visitor<'tcx> for CheckAttrVisitor<'tcx> {
26562656
// only `#[cfg]` and `#[cfg_attr]` are allowed, but it should be removed
26572657
// if we allow more attributes (e.g., tool attributes and `allow/deny/warn`)
26582658
// in where clauses. After that, only `self.check_attributes` should be enough.
2659-
const ATTRS_ALLOWED: &[Symbol] = &[sym::cfg, sym::cfg_attr, sym::cfg_attr_trace];
2659+
const ATTRS_ALLOWED: &[Symbol] = &[sym::cfg_trace, sym::cfg_attr_trace];
26602660
let spans = self
26612661
.tcx
26622662
.hir_attrs(where_predicate.hir_id)

compiler/rustc_query_system/src/ich/mod.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@ mod hcx;
88
mod impls_syntax;
99

1010
pub const IGNORED_ATTRIBUTES: &[Symbol] = &[
11-
sym::cfg,
11+
sym::cfg_trace, // FIXME should this really be ignored?
1212
sym::rustc_if_this_changed,
1313
sym::rustc_then_this_would_need,
1414
sym::rustc_dirty,

compiler/rustc_span/src/symbol.rs

+1
Original file line numberDiff line numberDiff line change
@@ -623,6 +623,7 @@ symbols! {
623623
cfg_target_has_atomic_equal_alignment,
624624
cfg_target_thread_local,
625625
cfg_target_vendor,
626+
cfg_trace: "<cfg>", // must not be a valid identifier
626627
cfg_ub_checks,
627628
cfg_version,
628629
cfi,

src/librustdoc/clean/mod.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -2773,7 +2773,7 @@ fn add_without_unwanted_attributes<'hir>(
27732773
if ident == sym::doc {
27742774
filter_doc_attr(&mut normal.args, is_inline);
27752775
attrs.push((Cow::Owned(attr), import_parent));
2776-
} else if is_inline || ident != sym::cfg {
2776+
} else if is_inline || ident != sym::cfg_trace {
27772777
// If it's not a `cfg()` attribute, we keep it.
27782778
attrs.push((Cow::Owned(attr), import_parent));
27792779
}

src/librustdoc/clean/types.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1059,7 +1059,7 @@ pub(crate) fn extract_cfg_from_attrs<'a, I: Iterator<Item = &'a hir::Attribute>
10591059
// `doc(cfg())` overrides `cfg()`).
10601060
attrs
10611061
.clone()
1062-
.filter(|attr| attr.has_name(sym::cfg))
1062+
.filter(|attr| attr.has_name(sym::cfg_trace))
10631063
.filter_map(|attr| single(attr.meta_item_list()?))
10641064
.filter_map(|attr| Cfg::parse_without(attr.meta_item()?, hidden_cfg).ok().flatten())
10651065
.fold(Cfg::True, |cfg, new_cfg| cfg & new_cfg)

src/tools/clippy/clippy_lints/src/attrs/duplicated_attributes.rs

+1-2
Original file line numberDiff line numberDiff line change
@@ -37,7 +37,6 @@ fn check_duplicated_attr(
3737
let Some(ident) = attr.ident() else { return };
3838
let name = ident.name;
3939
if name == sym::doc
40-
|| name == sym::cfg_attr
4140
|| name == sym::cfg_attr_trace
4241
|| name == sym::rustc_on_unimplemented
4342
|| name == sym::reason {
@@ -47,7 +46,7 @@ fn check_duplicated_attr(
4746
return;
4847
}
4948
if let Some(direct_parent) = parent.last()
50-
&& ["cfg", "cfg_attr"].contains(&direct_parent.as_str())
49+
&& direct_parent == sym::cfg_trace.as_str()
5150
&& [sym::all, sym::not, sym::any].contains(&name)
5251
{
5352
// FIXME: We don't correctly check `cfg`s for now, so if it's more complex than just a one

src/tools/clippy/clippy_lints/src/cfg_not_test.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -32,7 +32,7 @@ declare_lint_pass!(CfgNotTest => [CFG_NOT_TEST]);
3232

3333
impl EarlyLintPass for CfgNotTest {
3434
fn check_attribute(&mut self, cx: &EarlyContext<'_>, attr: &rustc_ast::Attribute) {
35-
if attr.has_name(rustc_span::sym::cfg) && contains_not_test(attr.meta_item_list().as_deref(), false) {
35+
if attr.has_name(rustc_span::sym::cfg_trace) && contains_not_test(attr.meta_item_list().as_deref(), false) {
3636
span_lint_and_then(
3737
cx,
3838
CFG_NOT_TEST,

src/tools/clippy/clippy_lints/src/methods/is_empty.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -41,7 +41,7 @@ pub(super) fn check(cx: &LateContext<'_>, expr: &'_ Expr<'_>, receiver: &Expr<'_
4141
fn is_under_cfg(cx: &LateContext<'_>, id: HirId) -> bool {
4242
cx.tcx
4343
.hir_parent_id_iter(id)
44-
.any(|id| cx.tcx.hir_attrs(id).iter().any(|attr| attr.has_name(sym::cfg)))
44+
.any(|id| cx.tcx.hir_attrs(id).iter().any(|attr| attr.has_name(sym::cfg_trace)))
4545
}
4646

4747
/// Similar to [`clippy_utils::expr_or_init`], but does not go up the chain if the initialization

src/tools/clippy/clippy_utils/src/lib.rs

+4-4
Original file line numberDiff line numberDiff line change
@@ -2629,7 +2629,7 @@ pub fn peel_ref_operators<'hir>(cx: &LateContext<'_>, mut expr: &'hir Expr<'hir>
26292629
pub fn is_hir_ty_cfg_dependant(cx: &LateContext<'_>, ty: &hir::Ty<'_>) -> bool {
26302630
if let TyKind::Path(QPath::Resolved(_, path)) = ty.kind {
26312631
if let Res::Def(_, def_id) = path.res {
2632-
return cx.tcx.has_attr(def_id, sym::cfg) || cx.tcx.has_attr(def_id, sym::cfg_attr);
2632+
return cx.tcx.has_attr(def_id, sym::cfg_trace) || cx.tcx.has_attr(def_id, sym::cfg_attr);
26332633
}
26342634
}
26352635
false
@@ -2699,7 +2699,7 @@ pub fn is_in_test_function(tcx: TyCtxt<'_>, id: HirId) -> bool {
26992699
/// use [`is_in_cfg_test`]
27002700
pub fn is_cfg_test(tcx: TyCtxt<'_>, id: HirId) -> bool {
27012701
tcx.hir_attrs(id).iter().any(|attr| {
2702-
if attr.has_name(sym::cfg)
2702+
if attr.has_name(sym::cfg_trace)
27032703
&& let Some(items) = attr.meta_item_list()
27042704
&& let [item] = &*items
27052705
&& item.has_name(sym::test)
@@ -2723,11 +2723,11 @@ pub fn is_in_test(tcx: TyCtxt<'_>, hir_id: HirId) -> bool {
27232723

27242724
/// Checks if the item of any of its parents has `#[cfg(...)]` attribute applied.
27252725
pub fn inherits_cfg(tcx: TyCtxt<'_>, def_id: LocalDefId) -> bool {
2726-
tcx.has_attr(def_id, sym::cfg)
2726+
tcx.has_attr(def_id, sym::cfg_trace)
27272727
|| tcx
27282728
.hir_parent_iter(tcx.local_def_id_to_hir_id(def_id))
27292729
.flat_map(|(parent_id, _)| tcx.hir_attrs(parent_id))
2730-
.any(|attr| attr.has_name(sym::cfg))
2730+
.any(|attr| attr.has_name(sym::cfg_trace))
27312731
}
27322732

27332733
/// Walks up the HIR tree from the given expression in an attempt to find where the value is

tests/pretty/tests-are-sorted.pp

-3
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,6 @@
1010
//@ pp-exact:tests-are-sorted.pp
1111

1212
extern crate test;
13-
#[cfg(test)]
1413
#[rustc_test_marker = "m_test"]
1514
#[doc(hidden)]
1615
pub const m_test: test::TestDescAndFn =
@@ -35,7 +34,6 @@
3534
fn m_test() {}
3635

3736
extern crate test;
38-
#[cfg(test)]
3937
#[rustc_test_marker = "z_test"]
4038
#[doc(hidden)]
4139
pub const z_test: test::TestDescAndFn =
@@ -61,7 +59,6 @@
6159
fn z_test() {}
6260

6361
extern crate test;
64-
#[cfg(test)]
6562
#[rustc_test_marker = "a_test"]
6663
#[doc(hidden)]
6764
pub const a_test: test::TestDescAndFn =

tests/ui/conditional-compilation/cfg-attr-syntax-validation.rs

-1
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,6 @@ macro_rules! generate_s10 {
2929
($expr: expr) => {
3030
#[cfg(feature = $expr)]
3131
//~^ ERROR expected unsuffixed literal, found expression `concat!("nonexistent")`
32-
//~| ERROR expected unsuffixed literal, found expression `concat!("nonexistent")`
3332
struct S10;
3433
}
3534
}

tests/ui/conditional-compilation/cfg-attr-syntax-validation.stderr

+1-13
Original file line numberDiff line numberDiff line change
@@ -65,19 +65,7 @@ LL | generate_s10!(concat!("nonexistent"));
6565
|
6666
= note: this error originates in the macro `generate_s10` (in Nightly builds, run with -Z macro-backtrace for more info)
6767

68-
error: expected unsuffixed literal, found expression `concat!("nonexistent")`
69-
--> $DIR/cfg-attr-syntax-validation.rs:30:25
70-
|
71-
LL | #[cfg(feature = $expr)]
72-
| ^^^^^
73-
...
74-
LL | generate_s10!(concat!("nonexistent"));
75-
| ------------------------------------- in this macro invocation
76-
|
77-
= note: duplicate diagnostic emitted due to `-Z deduplicate-diagnostics=no`
78-
= note: this error originates in the macro `generate_s10` (in Nightly builds, run with -Z macro-backtrace for more info)
79-
80-
error: aborting due to 11 previous errors
68+
error: aborting due to 10 previous errors
8169

8270
Some errors have detailed explanations: E0537, E0565.
8371
For more information about an error, try `rustc --explain E0537`.
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,12 @@
11
// This was triggering an assertion failure in `NodeRange::new`.
22

3+
//@ check-pass
4+
35
#![feature(cfg_eval)]
46
#![feature(stmt_expr_attributes)]
57

68
fn f() -> u32 {
79
#[cfg_eval] #[cfg(not(FALSE))] 0
8-
//~^ ERROR removing an expression is not supported in this position
910
}
1011

1112
fn main() {}

tests/ui/conditional-compilation/invalid-node-range-issue-129166.stderr

-8
This file was deleted.

tests/ui/parser/attribute/attr-bad-meta-4.rs

-2
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,6 @@ macro_rules! mac {
22
($attr_item: meta) => {
33
#[cfg($attr_item)]
44
//~^ ERROR expected unsuffixed literal, found `meta` metavariable
5-
//~| ERROR expected unsuffixed literal, found `meta` metavariable
65
struct S;
76
}
87
}
@@ -11,7 +10,6 @@ mac!(an(arbitrary token stream));
1110

1211
#[cfg(feature = -1)]
1312
//~^ ERROR expected unsuffixed literal, found `-`
14-
//~| ERROR expected unsuffixed literal, found `-`
1513
fn handler() {}
1614

1715
fn main() {}
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
error: expected unsuffixed literal, found `-`
2-
--> $DIR/attr-bad-meta-4.rs:12:17
2+
--> $DIR/attr-bad-meta-4.rs:11:17
33
|
44
LL | #[cfg(feature = -1)]
55
| ^
@@ -15,25 +15,5 @@ LL | mac!(an(arbitrary token stream));
1515
|
1616
= note: this error originates in the macro `mac` (in Nightly builds, run with -Z macro-backtrace for more info)
1717

18-
error: expected unsuffixed literal, found `meta` metavariable
19-
--> $DIR/attr-bad-meta-4.rs:3:15
20-
|
21-
LL | #[cfg($attr_item)]
22-
| ^^^^^^^^^^
23-
...
24-
LL | mac!(an(arbitrary token stream));
25-
| -------------------------------- in this macro invocation
26-
|
27-
= note: duplicate diagnostic emitted due to `-Z deduplicate-diagnostics=no`
28-
= note: this error originates in the macro `mac` (in Nightly builds, run with -Z macro-backtrace for more info)
29-
30-
error: expected unsuffixed literal, found `-`
31-
--> $DIR/attr-bad-meta-4.rs:12:17
32-
|
33-
LL | #[cfg(feature = -1)]
34-
| ^
35-
|
36-
= note: duplicate diagnostic emitted due to `-Z deduplicate-diagnostics=no`
37-
38-
error: aborting due to 4 previous errors
18+
error: aborting due to 2 previous errors
3919

0 commit comments

Comments
 (0)