1use std::iter;
4
5use rustc_ast::token::{Delimiter, Token, TokenKind};
6use rustc_ast::tokenstream::{
7 AttrTokenStream, AttrTokenTree, LazyAttrTokenStream, Spacing, TokenTree,
8};
9use rustc_ast::{
10 self as ast, AttrKind, AttrStyle, Attribute, HasAttrs, HasTokens, MetaItem, MetaItemInner,
11 NodeId, NormalAttr,
12};
13use rustc_attr_parsing as attr;
14use rustc_attr_parsing::validate_attr::deny_builtin_meta_unsafety;
15use rustc_attr_parsing::{
16 AttributeParser, CFG_TEMPLATE, EvalConfigResult, ShouldEmit, eval_config_entry, parse_cfg,
17 validate_attr,
18};
19use rustc_data_structures::flat_map_in_place::FlatMapInPlace;
20use rustc_feature::{
21 ACCEPTED_LANG_FEATURES, AttributeSafety, EnabledLangFeature, EnabledLibFeature, Features,
22 REMOVED_LANG_FEATURES, UNSTABLE_LANG_FEATURES,
23};
24use rustc_session::Session;
25use rustc_session::parse::feature_err;
26use rustc_span::{STDLIB_STABLE_CRATES, Span, Symbol, sym};
27use thin_vec::ThinVec;
28use tracing::instrument;
29
30use crate::errors::{
31 CrateNameInCfgAttr, CrateTypeInCfgAttr, FeatureNotAllowed, FeatureRemoved,
32 FeatureRemovedReason, InvalidCfg, MalformedFeatureAttribute, MalformedFeatureAttributeHelp,
33 RemoveExprNotSupported,
34};
35
36pub struct StripUnconfigured<'a> {
38 pub sess: &'a Session,
39 pub features: Option<&'a Features>,
40 pub config_tokens: bool,
44 pub lint_node_id: NodeId,
45}
46
47pub fn features(sess: &Session, krate_attrs: &[Attribute], crate_name: Symbol) -> Features {
48 fn feature_list(attr: &Attribute) -> ThinVec<ast::MetaItemInner> {
49 if attr.has_name(sym::feature)
50 && let Some(list) = attr.meta_item_list()
51 {
52 list
53 } else {
54 ThinVec::new()
55 }
56 }
57
58 let mut features = Features::default();
59
60 for attr in krate_attrs {
62 for mi in feature_list(attr) {
63 let name = match mi.ident() {
64 Some(ident) if mi.is_word() => ident.name,
65 Some(ident) => {
66 sess.dcx().emit_err(MalformedFeatureAttribute {
67 span: mi.span(),
68 help: MalformedFeatureAttributeHelp::Suggestion {
69 span: mi.span(),
70 suggestion: ident.name,
71 },
72 });
73 continue;
74 }
75 None => {
76 sess.dcx().emit_err(MalformedFeatureAttribute {
77 span: mi.span(),
78 help: MalformedFeatureAttributeHelp::Label { span: mi.span() },
79 });
80 continue;
81 }
82 };
83
84 if let Some(f) = REMOVED_LANG_FEATURES.iter().find(|f| name == f.feature.name) {
86 let pull_note = if let Some(pull) = f.pull {
87 format!(
88 "; see <https://github.com/rust-lang/rust/pull/{}> for more information",
89 pull
90 )
91 } else {
92 "".to_owned()
93 };
94 sess.dcx().emit_err(FeatureRemoved {
95 span: mi.span(),
96 reason: f.reason.map(|reason| FeatureRemovedReason { reason }),
97 removed_rustc_version: f.feature.since,
98 pull_note,
99 });
100 continue;
101 }
102
103 if let Some(f) = ACCEPTED_LANG_FEATURES.iter().find(|f| name == f.name) {
105 features.set_enabled_lang_feature(EnabledLangFeature {
106 gate_name: name,
107 attr_sp: mi.span(),
108 stable_since: Some(Symbol::intern(f.since)),
109 });
110 continue;
111 }
112
113 if let Some(allowed) = sess.opts.unstable_opts.allow_features.as_ref() {
117 if allowed.iter().all(|f| name.as_str() != f) {
118 sess.dcx().emit_err(FeatureNotAllowed { span: mi.span(), name });
119 continue;
120 }
121 }
122
123 if UNSTABLE_LANG_FEATURES.iter().find(|f| name == f.name).is_some() {
125 if features.internal(name) && !STDLIB_STABLE_CRATES.contains(&crate_name) {
130 sess.using_internal_features.store(true, std::sync::atomic::Ordering::Relaxed);
131 }
132
133 features.set_enabled_lang_feature(EnabledLangFeature {
134 gate_name: name,
135 attr_sp: mi.span(),
136 stable_since: None,
137 });
138 continue;
139 }
140
141 features
144 .set_enabled_lib_feature(EnabledLibFeature { gate_name: name, attr_sp: mi.span() });
145
146 if features.internal(name) && !STDLIB_STABLE_CRATES.contains(&crate_name) {
149 sess.using_internal_features.store(true, std::sync::atomic::Ordering::Relaxed);
150 }
151 }
152 }
153
154 features
155}
156
157pub fn pre_configure_attrs(sess: &Session, attrs: &[Attribute]) -> ast::AttrVec {
158 let strip_unconfigured = StripUnconfigured {
159 sess,
160 features: None,
161 config_tokens: false,
162 lint_node_id: ast::CRATE_NODE_ID,
163 };
164 attrs
165 .iter()
166 .flat_map(|attr| strip_unconfigured.process_cfg_attr(attr))
167 .take_while(|attr| {
168 !is_cfg(attr)
169 || strip_unconfigured
170 .cfg_true(attr, strip_unconfigured.lint_node_id, ShouldEmit::Nothing)
171 .as_bool()
172 })
173 .collect()
174}
175
176pub(crate) fn attr_into_trace(mut attr: Attribute, trace_name: Symbol) -> Attribute {
177 match &mut attr.kind {
178 AttrKind::Normal(normal) => {
179 let NormalAttr { item, tokens } = &mut **normal;
180 item.path.segments[0].ident.name = trace_name;
181 *tokens = Some(LazyAttrTokenStream::new_direct(AttrTokenStream::default()));
183 }
184 AttrKind::DocComment(..) => unreachable!(),
185 }
186 attr
187}
188
189#[macro_export]
190macro_rules! configure {
191 ($this:ident, $node:ident) => {
192 match $this.configure($node) {
193 Some(node) => node,
194 None => return Default::default(),
195 }
196 };
197}
198
199impl<'a> StripUnconfigured<'a> {
200 pub fn configure<T: HasAttrs + HasTokens>(&self, mut node: T) -> Option<T> {
201 self.process_cfg_attrs(&mut node);
202 self.in_cfg(node.attrs()).then(|| {
203 self.try_configure_tokens(&mut node);
204 node
205 })
206 }
207
208 fn try_configure_tokens<T: HasTokens>(&self, node: &mut T) {
209 if self.config_tokens {
210 if let Some(Some(tokens)) = node.tokens_mut() {
211 let attr_stream = tokens.to_attr_token_stream();
212 *tokens = LazyAttrTokenStream::new_direct(self.configure_tokens(&attr_stream));
213 }
214 }
215 }
216
217 fn configure_tokens(&self, stream: &AttrTokenStream) -> AttrTokenStream {
222 fn can_skip(stream: &AttrTokenStream) -> bool {
223 stream.0.iter().all(|tree| match tree {
224 AttrTokenTree::AttrsTarget(_) => false,
225 AttrTokenTree::Token(..) => true,
226 AttrTokenTree::Delimited(.., inner) => can_skip(inner),
227 })
228 }
229
230 if can_skip(stream) {
231 return stream.clone();
232 }
233
234 let trees: Vec<_> = stream
235 .0
236 .iter()
237 .filter_map(|tree| match tree.clone() {
238 AttrTokenTree::AttrsTarget(mut target) => {
239 target.attrs.flat_map_in_place(|attr| self.process_cfg_attr(&attr));
241
242 if self.in_cfg(&target.attrs) {
243 target.tokens = LazyAttrTokenStream::new_direct(
244 self.configure_tokens(&target.tokens.to_attr_token_stream()),
245 );
246 Some(AttrTokenTree::AttrsTarget(target))
247 } else {
248 None
251 }
252 }
253 AttrTokenTree::Delimited(sp, spacing, delim, mut inner) => {
254 inner = self.configure_tokens(&inner);
255 Some(AttrTokenTree::Delimited(sp, spacing, delim, inner))
256 }
257 AttrTokenTree::Token(Token { kind, .. }, _) if kind.is_delim() => {
258 panic!("Should be `AttrTokenTree::Delimited`, not delim tokens: {:?}", tree);
259 }
260 AttrTokenTree::Token(token, spacing) => Some(AttrTokenTree::Token(token, spacing)),
261 })
262 .collect();
263 AttrTokenStream::new(trees)
264 }
265
266 fn process_cfg_attrs<T: HasAttrs>(&self, node: &mut T) {
273 node.visit_attrs(|attrs| {
274 attrs.flat_map_in_place(|attr| self.process_cfg_attr(&attr));
275 });
276 }
277
278 fn process_cfg_attr(&self, attr: &Attribute) -> Vec<Attribute> {
279 if attr.has_name(sym::cfg_attr) {
280 self.expand_cfg_attr(attr, true)
281 } else {
282 vec![attr.clone()]
283 }
284 }
285
286 pub(crate) fn expand_cfg_attr(&self, cfg_attr: &Attribute, recursive: bool) -> Vec<Attribute> {
294 validate_attr::check_attribute_safety(
295 &self.sess.psess,
296 Some(AttributeSafety::Normal),
297 &cfg_attr,
298 ast::CRATE_NODE_ID,
299 );
300
301 let trace_attr = attr_into_trace(cfg_attr.clone(), sym::cfg_attr_trace);
304
305 let Some((cfg_predicate, expanded_attrs)) =
306 rustc_attr_parsing::parse_cfg_attr(cfg_attr, &self.sess, self.features)
307 else {
308 return vec![trace_attr];
309 };
310
311 if expanded_attrs.is_empty() {
313 self.sess.psess.buffer_lint(
314 rustc_lint_defs::builtin::UNUSED_ATTRIBUTES,
315 cfg_attr.span,
316 ast::CRATE_NODE_ID,
317 crate::errors::CfgAttrNoAttributes,
318 );
319 }
320
321 if !attr::eval_config_entry(
322 self.sess,
323 &cfg_predicate,
324 ast::CRATE_NODE_ID,
325 self.features,
326 ShouldEmit::ErrorsAndLints,
327 )
328 .as_bool()
329 {
330 return vec![trace_attr];
331 }
332
333 if recursive {
334 let expanded_attrs = expanded_attrs
338 .into_iter()
339 .flat_map(|item| self.process_cfg_attr(&self.expand_cfg_attr_item(cfg_attr, item)));
340 iter::once(trace_attr).chain(expanded_attrs).collect()
341 } else {
342 let expanded_attrs =
343 expanded_attrs.into_iter().map(|item| self.expand_cfg_attr_item(cfg_attr, item));
344 iter::once(trace_attr).chain(expanded_attrs).collect()
345 }
346 }
347
348 fn expand_cfg_attr_item(
349 &self,
350 cfg_attr: &Attribute,
351 (item, item_span): (ast::AttrItem, Span),
352 ) -> Attribute {
353 let mut orig_trees = cfg_attr.token_trees().into_iter();
357 let Some(TokenTree::Token(pound_token @ Token { kind: TokenKind::Pound, .. }, _)) =
358 orig_trees.next()
359 else {
360 panic!("Bad tokens for attribute {cfg_attr:?}");
361 };
362
363 let mut trees = if cfg_attr.style == AttrStyle::Inner {
365 let Some(TokenTree::Token(bang_token @ Token { kind: TokenKind::Bang, .. }, _)) =
366 orig_trees.next()
367 else {
368 panic!("Bad tokens for attribute {cfg_attr:?}");
369 };
370 vec![
371 AttrTokenTree::Token(pound_token, Spacing::Joint),
372 AttrTokenTree::Token(bang_token, Spacing::JointHidden),
373 ]
374 } else {
375 vec![AttrTokenTree::Token(pound_token, Spacing::JointHidden)]
376 };
377
378 let Some(TokenTree::Delimited(delim_span, delim_spacing, Delimiter::Bracket, _)) =
380 orig_trees.next()
381 else {
382 panic!("Bad tokens for attribute {cfg_attr:?}");
383 };
384 trees.push(AttrTokenTree::Delimited(
385 delim_span,
386 delim_spacing,
387 Delimiter::Bracket,
388 item.tokens
389 .as_ref()
390 .unwrap_or_else(|| panic!("Missing tokens for {item:?}"))
391 .to_attr_token_stream(),
392 ));
393
394 let tokens = Some(LazyAttrTokenStream::new_direct(AttrTokenStream::new(trees)));
395 let attr = ast::attr::mk_attr_from_item(
396 &self.sess.psess.attr_id_generator,
397 item,
398 tokens,
399 cfg_attr.style,
400 item_span,
401 );
402 if attr.has_name(sym::crate_type) {
403 self.sess.dcx().emit_err(CrateTypeInCfgAttr { span: attr.span });
404 }
405 if attr.has_name(sym::crate_name) {
406 self.sess.dcx().emit_err(CrateNameInCfgAttr { span: attr.span });
407 }
408 attr
409 }
410
411 fn in_cfg(&self, attrs: &[Attribute]) -> bool {
413 attrs.iter().all(|attr| {
414 !is_cfg(attr)
415 || self.cfg_true(attr, self.lint_node_id, ShouldEmit::ErrorsAndLints).as_bool()
416 })
417 }
418
419 pub(crate) fn cfg_true(
420 &self,
421 attr: &Attribute,
422 node: NodeId,
423 emit_errors: ShouldEmit,
424 ) -> EvalConfigResult {
425 deny_builtin_meta_unsafety(
427 self.sess.dcx(),
428 attr.get_normal_item().unsafety,
429 &rustc_ast::Path::from_ident(attr.ident().unwrap()),
430 );
431
432 let Some(cfg) = AttributeParser::parse_single(
433 self.sess,
434 attr,
435 attr.span,
436 node,
437 self.features,
438 emit_errors,
439 parse_cfg,
440 &CFG_TEMPLATE,
441 ) else {
442 return EvalConfigResult::True;
444 };
445
446 eval_config_entry(self.sess, &cfg, self.lint_node_id, self.features, emit_errors)
447 }
448
449 #[instrument(level = "trace", skip(self))]
451 pub(crate) fn maybe_emit_expr_attr_err(&self, attr: &Attribute) {
452 if self.features.is_some_and(|features| !features.stmt_expr_attributes())
453 && !attr.span.allows_unstable(sym::stmt_expr_attributes)
454 {
455 let mut err = feature_err(
456 &self.sess,
457 sym::stmt_expr_attributes,
458 attr.span,
459 crate::fluent_generated::expand_attributes_on_expressions_experimental,
460 );
461
462 if attr.is_doc_comment() {
463 err.help(if attr.style == AttrStyle::Outer {
464 crate::fluent_generated::expand_help_outer_doc
465 } else {
466 crate::fluent_generated::expand_help_inner_doc
467 });
468 }
469
470 err.emit();
471 }
472 }
473
474 #[instrument(level = "trace", skip(self))]
475 pub fn configure_expr(&self, expr: &mut ast::Expr, method_receiver: bool) {
476 if !method_receiver {
477 for attr in expr.attrs.iter() {
478 self.maybe_emit_expr_attr_err(attr);
479 }
480 }
481
482 if let Some(attr) = expr.attrs().iter().find(|a| is_cfg(a)) {
490 self.sess.dcx().emit_err(RemoveExprNotSupported { span: attr.span });
491 }
492
493 self.process_cfg_attrs(expr);
494 self.try_configure_tokens(&mut *expr);
495 }
496}
497
498pub fn parse_cfg_old<'a>(meta_item: &'a MetaItem, sess: &Session) -> Option<&'a MetaItemInner> {
500 let span = meta_item.span;
501 match meta_item.meta_item_list() {
502 None => {
503 sess.dcx().emit_err(InvalidCfg::NotFollowedByParens { span });
504 None
505 }
506 Some([]) => {
507 sess.dcx().emit_err(InvalidCfg::NoPredicate { span });
508 None
509 }
510 Some([_, .., l]) => {
511 sess.dcx().emit_err(InvalidCfg::MultiplePredicates { span: l.span() });
512 None
513 }
514 Some([single]) => match single.meta_item_or_bool() {
515 Some(meta_item) => Some(meta_item),
516 None => {
517 sess.dcx().emit_err(InvalidCfg::PredicateLiteral { span: single.span() });
518 None
519 }
520 },
521 }
522}
523
524fn is_cfg(attr: &Attribute) -> bool {
525 attr.has_name(sym::cfg)
526}