1use std::fmt::Write;
2use std::mem;
3
4use ast::token::IdentIsRaw;
5use rustc_ast::ast::*;
6use rustc_ast::token::{self, Delimiter, InvisibleOrigin, MetaVarKind, TokenKind};
7use rustc_ast::tokenstream::{DelimSpan, TokenStream, TokenTree};
8use rustc_ast::util::case::Case;
9use rustc_ast::{
10 attr, {self as ast},
11};
12use rustc_ast_pretty::pprust;
13use rustc_errors::codes::*;
14use rustc_errors::{Applicability, PResult, StashKey, struct_span_code_err};
15use rustc_session::lint::builtin::VARARGS_WITHOUT_PATTERN;
16use rustc_span::edit_distance::edit_distance;
17use rustc_span::edition::Edition;
18use rustc_span::{DUMMY_SP, ErrorGuaranteed, Ident, Span, Symbol, kw, source_map, sym};
19use thin_vec::{ThinVec, thin_vec};
20use tracing::debug;
21
22use super::diagnostics::{ConsumeClosingDelim, dummy_arg};
23use super::ty::{AllowPlus, RecoverQPath, RecoverReturnSign};
24use super::{
25 AttrWrapper, ExpKeywordPair, ExpTokenPair, FollowedByType, ForceCollect, Parser, PathStyle,
26 Recovered, Trailing, UsePreAttrPos,
27};
28use crate::errors::{self, FnPointerCannotBeAsync, FnPointerCannotBeConst, MacroExpandsToAdtField};
29use crate::{exp, fluent_generated as fluent};
30
31impl<'a> Parser<'a> {
32 pub fn parse_crate_mod(&mut self) -> PResult<'a, ast::Crate> {
34 let (attrs, items, spans) = self.parse_mod(exp!(Eof))?;
35 Ok(ast::Crate { attrs, items, spans, id: DUMMY_NODE_ID, is_placeholder: false })
36 }
37
38 fn parse_item_mod(&mut self, attrs: &mut AttrVec) -> PResult<'a, ItemKind> {
40 let safety = self.parse_safety(Case::Sensitive);
41 self.expect_keyword(exp!(Mod))?;
42 let ident = self.parse_ident()?;
43 let mod_kind = if self.eat(exp!(Semi)) {
44 ModKind::Unloaded
45 } else {
46 self.expect(exp!(OpenBrace))?;
47 let (inner_attrs, items, inner_span) = self.parse_mod(exp!(CloseBrace))?;
48 attrs.extend(inner_attrs);
49 ModKind::Loaded(items, Inline::Yes, inner_span)
50 };
51 Ok(ItemKind::Mod(safety, ident, mod_kind))
52 }
53
54 pub fn parse_mod(
59 &mut self,
60 term: ExpTokenPair,
61 ) -> PResult<'a, (AttrVec, ThinVec<Box<Item>>, ModSpans)> {
62 let lo = self.token.span;
63 let attrs = self.parse_inner_attributes()?;
64
65 let post_attr_lo = self.token.span;
66 let mut items: ThinVec<Box<_>> = ThinVec::new();
67
68 loop {
71 while self.maybe_consume_incorrect_semicolon(items.last().map(|x| &**x)) {} let Some(item) = self.parse_item(ForceCollect::No)? else {
73 break;
74 };
75 items.push(item);
76 }
77
78 if !self.eat(term) {
79 let token_str = super::token_descr(&self.token);
80 if !self.maybe_consume_incorrect_semicolon(items.last().map(|x| &**x)) {
81 let is_let = self.token.is_keyword(kw::Let);
82 let is_let_mut = is_let && self.look_ahead(1, |t| t.is_keyword(kw::Mut));
83 let let_has_ident = is_let && !is_let_mut && self.is_kw_followed_by_ident(kw::Let);
84
85 let msg = format!("expected item, found {token_str}");
86 let mut err = self.dcx().struct_span_err(self.token.span, msg);
87
88 let label = if is_let {
89 "`let` cannot be used for global variables"
90 } else {
91 "expected item"
92 };
93 err.span_label(self.token.span, label);
94
95 if is_let {
96 if is_let_mut {
97 err.help("consider using `static` and a `Mutex` instead of `let mut`");
98 } else if let_has_ident {
99 err.span_suggestion_short(
100 self.token.span,
101 "consider using `static` or `const` instead of `let`",
102 "static",
103 Applicability::MaybeIncorrect,
104 );
105 } else {
106 err.help("consider using `static` or `const` instead of `let`");
107 }
108 }
109 err.note("for a full list of items that can appear in modules, see <https://doc.rust-lang.org/reference/items.html>");
110 return Err(err);
111 }
112 }
113
114 let inject_use_span = post_attr_lo.data().with_hi(post_attr_lo.lo());
115 let mod_spans = ModSpans { inner_span: lo.to(self.prev_token.span), inject_use_span };
116 Ok((attrs, items, mod_spans))
117 }
118}
119
120impl<'a> Parser<'a> {
121 pub fn parse_item(&mut self, force_collect: ForceCollect) -> PResult<'a, Option<Box<Item>>> {
122 let fn_parse_mode =
123 FnParseMode { req_name: |_, _| true, context: FnContext::Free, req_body: true };
124 self.parse_item_(fn_parse_mode, force_collect).map(|i| i.map(Box::new))
125 }
126
127 fn parse_item_(
128 &mut self,
129 fn_parse_mode: FnParseMode,
130 force_collect: ForceCollect,
131 ) -> PResult<'a, Option<Item>> {
132 self.recover_vcs_conflict_marker();
133 let attrs = self.parse_outer_attributes()?;
134 self.recover_vcs_conflict_marker();
135 self.parse_item_common(attrs, true, false, fn_parse_mode, force_collect)
136 }
137
138 pub(super) fn parse_item_common(
139 &mut self,
140 attrs: AttrWrapper,
141 mac_allowed: bool,
142 attrs_allowed: bool,
143 fn_parse_mode: FnParseMode,
144 force_collect: ForceCollect,
145 ) -> PResult<'a, Option<Item>> {
146 if let Some(item) =
147 self.eat_metavar_seq(MetaVarKind::Item, |this| this.parse_item(ForceCollect::Yes))
148 {
149 let mut item = item.expect("an actual item");
150 attrs.prepend_to_nt_inner(&mut item.attrs);
151 return Ok(Some(*item));
152 }
153
154 self.collect_tokens(None, attrs, force_collect, |this, mut attrs| {
155 let lo = this.token.span;
156 let vis = this.parse_visibility(FollowedByType::No)?;
157 let mut def = this.parse_defaultness();
158 let kind = this.parse_item_kind(
159 &mut attrs,
160 mac_allowed,
161 lo,
162 &vis,
163 &mut def,
164 fn_parse_mode,
165 Case::Sensitive,
166 )?;
167 if let Some(kind) = kind {
168 this.error_on_unconsumed_default(def, &kind);
169 let span = lo.to(this.prev_token.span);
170 let id = DUMMY_NODE_ID;
171 let item = Item { attrs, id, kind, vis, span, tokens: None };
172 return Ok((Some(item), Trailing::No, UsePreAttrPos::No));
173 }
174
175 if !matches!(vis.kind, VisibilityKind::Inherited) {
177 this.dcx().emit_err(errors::VisibilityNotFollowedByItem { span: vis.span, vis });
178 }
179
180 if let Defaultness::Default(span) = def {
181 this.dcx().emit_err(errors::DefaultNotFollowedByItem { span });
182 }
183
184 if !attrs_allowed {
185 this.recover_attrs_no_item(&attrs)?;
186 }
187 Ok((None, Trailing::No, UsePreAttrPos::No))
188 })
189 }
190
191 fn error_on_unconsumed_default(&self, def: Defaultness, kind: &ItemKind) {
193 if let Defaultness::Default(span) = def {
194 self.dcx().emit_err(errors::InappropriateDefault {
195 span,
196 article: kind.article(),
197 descr: kind.descr(),
198 });
199 }
200 }
201
202 fn parse_item_kind(
204 &mut self,
205 attrs: &mut AttrVec,
206 macros_allowed: bool,
207 lo: Span,
208 vis: &Visibility,
209 def: &mut Defaultness,
210 fn_parse_mode: FnParseMode,
211 case: Case,
212 ) -> PResult<'a, Option<ItemKind>> {
213 let check_pub = def == &Defaultness::Final;
214 let mut def_ = || mem::replace(def, Defaultness::Final);
215
216 let info = if !self.is_use_closure() && self.eat_keyword_case(exp!(Use), case) {
217 self.parse_use_item()?
218 } else if self.check_fn_front_matter(check_pub, case) {
219 let (ident, sig, generics, contract, body) =
221 self.parse_fn(attrs, fn_parse_mode, lo, vis, case)?;
222 ItemKind::Fn(Box::new(Fn {
223 defaultness: def_(),
224 ident,
225 sig,
226 generics,
227 contract,
228 body,
229 define_opaque: None,
230 }))
231 } else if self.eat_keyword_case(exp!(Extern), case) {
232 if self.eat_keyword_case(exp!(Crate), case) {
233 self.parse_item_extern_crate()?
235 } else {
236 self.parse_item_foreign_mod(attrs, Safety::Default)?
238 }
239 } else if self.is_unsafe_foreign_mod() {
240 let safety = self.parse_safety(Case::Sensitive);
242 self.expect_keyword(exp!(Extern))?;
243 self.parse_item_foreign_mod(attrs, safety)?
244 } else if let Some(safety) = self.parse_global_static_front_matter(case) {
245 let mutability = self.parse_mutability();
247 self.parse_static_item(safety, mutability)?
248 } else if self.check_keyword_case(exp!(Trait), case) || self.check_trait_front_matter() {
249 self.parse_item_trait(attrs, lo)?
251 } else if self.check_impl_frontmatter() {
252 self.parse_item_impl(attrs, def_())?
254 } else if let Const::Yes(const_span) = self.parse_constness(case) {
255 self.recover_const_mut(const_span);
257 self.recover_missing_kw_before_item()?;
258 let (ident, generics, ty, rhs) = self.parse_const_item(attrs)?;
259 ItemKind::Const(Box::new(ConstItem {
260 defaultness: def_(),
261 ident,
262 generics,
263 ty,
264 rhs,
265 define_opaque: None,
266 }))
267 } else if self.is_reuse_path_item() {
268 self.parse_item_delegation()?
269 } else if self.check_keyword_case(exp!(Mod), case)
270 || self.check_keyword_case(exp!(Unsafe), case) && self.is_keyword_ahead(1, &[kw::Mod])
271 {
272 self.parse_item_mod(attrs)?
274 } else if self.eat_keyword_case(exp!(Type), case) {
275 self.parse_type_alias(def_())?
277 } else if self.eat_keyword_case(exp!(Enum), case) {
278 self.parse_item_enum()?
280 } else if self.eat_keyword_case(exp!(Struct), case) {
281 self.parse_item_struct()?
283 } else if self.is_kw_followed_by_ident(kw::Union) {
284 self.bump(); self.parse_item_union()?
287 } else if self.is_builtin() {
288 return self.parse_item_builtin();
290 } else if self.eat_keyword_case(exp!(Macro), case) {
291 self.parse_item_decl_macro(lo)?
293 } else if let IsMacroRulesItem::Yes { has_bang } = self.is_macro_rules_item() {
294 self.parse_item_macro_rules(vis, has_bang)?
296 } else if self.isnt_macro_invocation()
297 && (self.token.is_ident_named(sym::import)
298 || self.token.is_ident_named(sym::using)
299 || self.token.is_ident_named(sym::include)
300 || self.token.is_ident_named(sym::require))
301 {
302 return self.recover_import_as_use();
303 } else if self.isnt_macro_invocation() && vis.kind.is_pub() {
304 self.recover_missing_kw_before_item()?;
305 return Ok(None);
306 } else if self.isnt_macro_invocation() && case == Case::Sensitive {
307 _ = def_;
308
309 return self.parse_item_kind(
311 attrs,
312 macros_allowed,
313 lo,
314 vis,
315 def,
316 fn_parse_mode,
317 Case::Insensitive,
318 );
319 } else if macros_allowed && self.check_path() {
320 if self.isnt_macro_invocation() {
321 self.recover_missing_kw_before_item()?;
322 }
323 ItemKind::MacCall(Box::new(self.parse_item_macro(vis)?))
325 } else {
326 return Ok(None);
327 };
328 Ok(Some(info))
329 }
330
331 fn recover_import_as_use(&mut self) -> PResult<'a, Option<ItemKind>> {
332 let span = self.token.span;
333 let token_name = super::token_descr(&self.token);
334 let snapshot = self.create_snapshot_for_diagnostic();
335 self.bump();
336 match self.parse_use_item() {
337 Ok(u) => {
338 self.dcx().emit_err(errors::RecoverImportAsUse { span, token_name });
339 Ok(Some(u))
340 }
341 Err(e) => {
342 e.cancel();
343 self.restore_snapshot(snapshot);
344 Ok(None)
345 }
346 }
347 }
348
349 fn parse_use_item(&mut self) -> PResult<'a, ItemKind> {
350 let tree = self.parse_use_tree()?;
351 if let Err(mut e) = self.expect_semi() {
352 match tree.kind {
353 UseTreeKind::Glob => {
354 e.note("the wildcard token must be last on the path");
355 }
356 UseTreeKind::Nested { .. } => {
357 e.note("glob-like brace syntax must be last on the path");
358 }
359 _ => (),
360 }
361 return Err(e);
362 }
363 Ok(ItemKind::Use(tree))
364 }
365
366 pub(super) fn is_path_start_item(&mut self) -> bool {
368 self.is_kw_followed_by_ident(kw::Union) || self.is_reuse_path_item()
370 || self.check_trait_front_matter() || self.is_async_fn() || matches!(self.is_macro_rules_item(), IsMacroRulesItem::Yes{..}) }
374
375 fn is_reuse_path_item(&mut self) -> bool {
376 self.token.is_keyword(kw::Reuse)
378 && self.look_ahead(1, |t| t.is_path_start() && *t != token::PathSep)
379 }
380
381 fn isnt_macro_invocation(&mut self) -> bool {
383 self.check_ident() && self.look_ahead(1, |t| *t != token::Bang && *t != token::PathSep)
384 }
385
386 fn recover_missing_kw_before_item(&mut self) -> PResult<'a, ()> {
389 let is_pub = self.prev_token.is_keyword(kw::Pub);
390 let is_const = self.prev_token.is_keyword(kw::Const);
391 let ident_span = self.token.span;
392 let span = if is_pub { self.prev_token.span.to(ident_span) } else { ident_span };
393 let insert_span = ident_span.shrink_to_lo();
394
395 let ident = if self.token.is_ident()
396 && (!is_const || self.look_ahead(1, |t| *t == token::OpenParen))
397 && self.look_ahead(1, |t| {
398 matches!(t.kind, token::Lt | token::OpenBrace | token::OpenParen)
399 }) {
400 self.parse_ident().unwrap()
401 } else {
402 return Ok(());
403 };
404
405 let mut found_generics = false;
406 if self.check(exp!(Lt)) {
407 found_generics = true;
408 self.eat_to_tokens(&[exp!(Gt)]);
409 self.bump(); }
411
412 let err = if self.check(exp!(OpenBrace)) {
413 if self.look_ahead(1, |t| *t == token::CloseBrace) {
415 Some(errors::MissingKeywordForItemDefinition::EnumOrStruct { span })
417 } else if self.look_ahead(2, |t| *t == token::Colon)
418 || self.look_ahead(3, |t| *t == token::Colon)
419 {
420 Some(errors::MissingKeywordForItemDefinition::Struct { span, insert_span, ident })
422 } else {
423 Some(errors::MissingKeywordForItemDefinition::Enum { span, insert_span, ident })
424 }
425 } else if self.check(exp!(OpenParen)) {
426 self.bump(); let is_method = self.recover_self_param();
429
430 self.consume_block(exp!(OpenParen), exp!(CloseParen), ConsumeClosingDelim::Yes);
431
432 let err = if self.check(exp!(RArrow)) || self.check(exp!(OpenBrace)) {
433 self.eat_to_tokens(&[exp!(OpenBrace)]);
434 self.bump(); self.consume_block(exp!(OpenBrace), exp!(CloseBrace), ConsumeClosingDelim::Yes);
436 if is_method {
437 errors::MissingKeywordForItemDefinition::Method { span, insert_span, ident }
438 } else {
439 errors::MissingKeywordForItemDefinition::Function { span, insert_span, ident }
440 }
441 } else if is_pub && self.check(exp!(Semi)) {
442 errors::MissingKeywordForItemDefinition::Struct { span, insert_span, ident }
443 } else {
444 errors::MissingKeywordForItemDefinition::Ambiguous {
445 span,
446 subdiag: if found_generics {
447 None
448 } else if let Ok(snippet) = self.span_to_snippet(ident_span) {
449 Some(errors::AmbiguousMissingKwForItemSub::SuggestMacro {
450 span: ident_span,
451 snippet,
452 })
453 } else {
454 Some(errors::AmbiguousMissingKwForItemSub::HelpMacro)
455 },
456 }
457 };
458 Some(err)
459 } else if found_generics {
460 Some(errors::MissingKeywordForItemDefinition::Ambiguous { span, subdiag: None })
461 } else {
462 None
463 };
464
465 if let Some(err) = err { Err(self.dcx().create_err(err)) } else { Ok(()) }
466 }
467
468 fn parse_item_builtin(&mut self) -> PResult<'a, Option<ItemKind>> {
469 Ok(None)
471 }
472
473 fn parse_item_macro(&mut self, vis: &Visibility) -> PResult<'a, MacCall> {
475 let path = self.parse_path(PathStyle::Mod)?; self.expect(exp!(Bang))?; match self.parse_delim_args() {
478 Ok(args) => {
480 self.eat_semi_for_macro_if_needed(&args);
481 self.complain_if_pub_macro(vis, false);
482 Ok(MacCall { path, args })
483 }
484
485 Err(mut err) => {
486 if self.token.is_ident()
488 && let [segment] = path.segments.as_slice()
489 && edit_distance("macro_rules", &segment.ident.to_string(), 2).is_some()
490 {
491 err.span_suggestion(
492 path.span,
493 "perhaps you meant to define a macro",
494 "macro_rules",
495 Applicability::MachineApplicable,
496 );
497 }
498 Err(err)
499 }
500 }
501 }
502
503 fn recover_attrs_no_item(&mut self, attrs: &[Attribute]) -> PResult<'a, ()> {
505 let ([start @ end] | [start, .., end]) = attrs else {
506 return Ok(());
507 };
508 let msg = if end.is_doc_comment() {
509 "expected item after doc comment"
510 } else {
511 "expected item after attributes"
512 };
513 let mut err = self.dcx().struct_span_err(end.span, msg);
514 if end.is_doc_comment() {
515 err.span_label(end.span, "this doc comment doesn't document anything");
516 } else if self.token == TokenKind::Semi {
517 err.span_suggestion_verbose(
518 self.token.span,
519 "consider removing this semicolon",
520 "",
521 Applicability::MaybeIncorrect,
522 );
523 }
524 if let [.., penultimate, _] = attrs {
525 err.span_label(start.span.to(penultimate.span), "other attributes here");
526 }
527 Err(err)
528 }
529
530 fn is_async_fn(&self) -> bool {
531 self.token.is_keyword(kw::Async) && self.is_keyword_ahead(1, &[kw::Fn])
532 }
533
534 fn parse_polarity(&mut self) -> ast::ImplPolarity {
535 if self.check(exp!(Bang)) && self.look_ahead(1, |t| t.can_begin_type()) {
537 self.bump(); ast::ImplPolarity::Negative(self.prev_token.span)
539 } else {
540 ast::ImplPolarity::Positive
541 }
542 }
543
544 fn parse_item_impl(
559 &mut self,
560 attrs: &mut AttrVec,
561 defaultness: Defaultness,
562 ) -> PResult<'a, ItemKind> {
563 let mut constness = self.parse_constness(Case::Sensitive);
564 let safety = self.parse_safety(Case::Sensitive);
565 self.expect_keyword(exp!(Impl))?;
566
567 let mut generics = if self.choose_generics_over_qpath(0) {
569 self.parse_generics()?
570 } else {
571 let mut generics = Generics::default();
572 generics.span = self.prev_token.span.shrink_to_hi();
575 generics
576 };
577
578 if let Const::No = constness {
579 constness = self.parse_constness(Case::Sensitive);
581 }
582
583 if let Const::Yes(span) = constness {
584 self.psess.gated_spans.gate(sym::const_trait_impl, span);
585 }
586
587 if (self.token_uninterpolated_span().at_least_rust_2018()
589 && self.token.is_keyword(kw::Async))
590 || self.is_kw_followed_by_ident(kw::Async)
591 {
592 self.bump();
593 self.dcx().emit_err(errors::AsyncImpl { span: self.prev_token.span });
594 }
595
596 let polarity = self.parse_polarity();
597
598 let ty_first = if self.token.is_keyword(kw::For) && self.look_ahead(1, |t| t != &token::Lt)
600 {
601 let span = self.prev_token.span.between(self.token.span);
602 return Err(self.dcx().create_err(errors::MissingTraitInTraitImpl {
603 span,
604 for_span: span.to(self.token.span),
605 }));
606 } else {
607 self.parse_ty_with_generics_recovery(&generics)?
608 };
609
610 let has_for = self.eat_keyword(exp!(For));
612 let missing_for_span = self.prev_token.span.between(self.token.span);
613
614 let ty_second = if self.token == token::DotDot {
615 self.bump(); Some(self.mk_ty(self.prev_token.span, TyKind::Dummy))
622 } else if has_for || self.token.can_begin_type() {
623 Some(self.parse_ty()?)
624 } else {
625 None
626 };
627
628 generics.where_clause = self.parse_where_clause()?;
629
630 let impl_items = self.parse_item_list(attrs, |p| p.parse_impl_item(ForceCollect::No))?;
631
632 let (of_trait, self_ty) = match ty_second {
633 Some(ty_second) => {
634 if !has_for {
636 self.dcx().emit_err(errors::MissingForInTraitImpl { span: missing_for_span });
637 }
638
639 let ty_first = *ty_first;
640 let path = match ty_first.kind {
641 TyKind::Path(None, path) => path,
643 other => {
644 if let TyKind::ImplTrait(_, bounds) = other
645 && let [bound] = bounds.as_slice()
646 && let GenericBound::Trait(poly_trait_ref) = bound
647 {
648 let extra_impl_kw = ty_first.span.until(bound.span());
652 self.dcx().emit_err(errors::ExtraImplKeywordInTraitImpl {
653 extra_impl_kw,
654 impl_trait_span: ty_first.span,
655 });
656 poly_trait_ref.trait_ref.path.clone()
657 } else {
658 return Err(self.dcx().create_err(
659 errors::ExpectedTraitInTraitImplFoundType { span: ty_first.span },
660 ));
661 }
662 }
663 };
664 let trait_ref = TraitRef { path, ref_id: ty_first.id };
665
666 let of_trait =
667 Some(Box::new(TraitImplHeader { defaultness, safety, polarity, trait_ref }));
668 (of_trait, ty_second)
669 }
670 None => {
671 let self_ty = ty_first;
672 let error = |modifier, modifier_name, modifier_span| {
673 self.dcx().create_err(errors::TraitImplModifierInInherentImpl {
674 span: self_ty.span,
675 modifier,
676 modifier_name,
677 modifier_span,
678 self_ty: self_ty.span,
679 })
680 };
681
682 if let Safety::Unsafe(span) = safety {
683 error("unsafe", "unsafe", span).with_code(E0197).emit();
684 }
685 if let ImplPolarity::Negative(span) = polarity {
686 error("!", "negative", span).emit();
687 }
688 if let Defaultness::Default(def_span) = defaultness {
689 error("default", "default", def_span).emit();
690 }
691 if let Const::Yes(span) = constness {
692 self.psess.gated_spans.gate(sym::const_trait_impl, span);
693 }
694 (None, self_ty)
695 }
696 };
697
698 Ok(ItemKind::Impl(Impl { generics, of_trait, self_ty, items: impl_items, constness }))
699 }
700
701 fn parse_item_delegation(&mut self) -> PResult<'a, ItemKind> {
702 let span = self.token.span;
703 self.expect_keyword(exp!(Reuse))?;
704
705 let (qself, path) = if self.eat_lt() {
706 let (qself, path) = self.parse_qpath(PathStyle::Expr)?;
707 (Some(qself), path)
708 } else {
709 (None, self.parse_path(PathStyle::Expr)?)
710 };
711
712 let rename = |this: &mut Self| {
713 Ok(if this.eat_keyword(exp!(As)) { Some(this.parse_ident()?) } else { None })
714 };
715 let body = |this: &mut Self| {
716 Ok(if this.check(exp!(OpenBrace)) {
717 Some(this.parse_block()?)
718 } else {
719 this.expect(exp!(Semi))?;
720 None
721 })
722 };
723
724 let item_kind = if self.eat_path_sep() {
725 let suffixes = if self.eat(exp!(Star)) {
726 None
727 } else {
728 let parse_suffix = |p: &mut Self| Ok((p.parse_path_segment_ident()?, rename(p)?));
729 Some(self.parse_delim_comma_seq(exp!(OpenBrace), exp!(CloseBrace), parse_suffix)?.0)
730 };
731 let deleg = DelegationMac { qself, prefix: path, suffixes, body: body(self)? };
732 ItemKind::DelegationMac(Box::new(deleg))
733 } else {
734 let rename = rename(self)?;
735 let ident = rename.unwrap_or_else(|| path.segments.last().unwrap().ident);
736 let deleg = Delegation {
737 id: DUMMY_NODE_ID,
738 qself,
739 path,
740 ident,
741 rename,
742 body: body(self)?,
743 from_glob: false,
744 };
745 ItemKind::Delegation(Box::new(deleg))
746 };
747
748 let span = span.to(self.prev_token.span);
749 self.psess.gated_spans.gate(sym::fn_delegation, span);
750
751 Ok(item_kind)
752 }
753
754 fn parse_item_list<T>(
755 &mut self,
756 attrs: &mut AttrVec,
757 mut parse_item: impl FnMut(&mut Parser<'a>) -> PResult<'a, Option<Option<T>>>,
758 ) -> PResult<'a, ThinVec<T>> {
759 let open_brace_span = self.token.span;
760
761 if self.token == TokenKind::Semi {
763 self.dcx().emit_err(errors::UseEmptyBlockNotSemi { span: self.token.span });
764 self.bump();
765 return Ok(ThinVec::new());
766 }
767
768 self.expect(exp!(OpenBrace))?;
769 attrs.extend(self.parse_inner_attributes()?);
770
771 let mut items = ThinVec::new();
772 while !self.eat(exp!(CloseBrace)) {
773 if self.recover_doc_comment_before_brace() {
774 continue;
775 }
776 self.recover_vcs_conflict_marker();
777 match parse_item(self) {
778 Ok(None) => {
779 let mut is_unnecessary_semicolon = !items.is_empty()
780 && self
798 .span_to_snippet(self.prev_token.span)
799 .is_ok_and(|snippet| snippet == "}")
800 && self.token == token::Semi;
801 let mut semicolon_span = self.token.span;
802 if !is_unnecessary_semicolon {
803 is_unnecessary_semicolon =
805 self.token == token::OpenBrace && self.prev_token == token::Semi;
806 semicolon_span = self.prev_token.span;
807 }
808 let non_item_span = self.token.span;
810 let is_let = self.token.is_keyword(kw::Let);
811
812 let mut err =
813 self.dcx().struct_span_err(non_item_span, "non-item in item list");
814 self.consume_block(exp!(OpenBrace), exp!(CloseBrace), ConsumeClosingDelim::Yes);
815 if is_let {
816 err.span_suggestion_verbose(
817 non_item_span,
818 "consider using `const` instead of `let` for associated const",
819 "const",
820 Applicability::MachineApplicable,
821 );
822 } else {
823 err.span_label(open_brace_span, "item list starts here")
824 .span_label(non_item_span, "non-item starts here")
825 .span_label(self.prev_token.span, "item list ends here");
826 }
827 if is_unnecessary_semicolon {
828 err.span_suggestion(
829 semicolon_span,
830 "consider removing this semicolon",
831 "",
832 Applicability::MaybeIncorrect,
833 );
834 }
835 err.emit();
836 break;
837 }
838 Ok(Some(item)) => items.extend(item),
839 Err(err) => {
840 self.consume_block(exp!(OpenBrace), exp!(CloseBrace), ConsumeClosingDelim::Yes);
841 err.with_span_label(
842 open_brace_span,
843 "while parsing this item list starting here",
844 )
845 .with_span_label(self.prev_token.span, "the item list ends here")
846 .emit();
847 break;
848 }
849 }
850 }
851 Ok(items)
852 }
853
854 fn recover_doc_comment_before_brace(&mut self) -> bool {
856 if let token::DocComment(..) = self.token.kind {
857 if self.look_ahead(1, |tok| tok == &token::CloseBrace) {
858 struct_span_code_err!(
860 self.dcx(),
861 self.token.span,
862 E0584,
863 "found a documentation comment that doesn't document anything",
864 )
865 .with_span_label(self.token.span, "this doc comment doesn't document anything")
866 .with_help(
867 "doc comments must come before what they document, if a comment was \
868 intended use `//`",
869 )
870 .emit();
871 self.bump();
872 return true;
873 }
874 }
875 false
876 }
877
878 fn parse_defaultness(&mut self) -> Defaultness {
880 if self.check_keyword(exp!(Default))
884 && self.look_ahead(1, |t| t.is_non_raw_ident_where(|i| i.name != kw::As))
885 {
886 self.bump(); Defaultness::Default(self.prev_token_uninterpolated_span())
888 } else {
889 Defaultness::Final
890 }
891 }
892
893 fn check_trait_front_matter(&mut self) -> bool {
895 self.check_keyword(exp!(Auto)) && self.is_keyword_ahead(1, &[kw::Trait])
897 || self.check_keyword(exp!(Unsafe)) && self.is_keyword_ahead(1, &[kw::Trait, kw::Auto])
899 || self.check_keyword(exp!(Const)) && ((self.is_keyword_ahead(1, &[kw::Trait]) || self.is_keyword_ahead(1, &[kw::Auto]) && self.is_keyword_ahead(2, &[kw::Trait]))
900 || self.is_keyword_ahead(1, &[kw::Unsafe]) && self.is_keyword_ahead(2, &[kw::Trait, kw::Auto]))
901 }
902
903 fn parse_item_trait(&mut self, attrs: &mut AttrVec, lo: Span) -> PResult<'a, ItemKind> {
905 let constness = self.parse_constness(Case::Sensitive);
906 if let Const::Yes(span) = constness {
907 self.psess.gated_spans.gate(sym::const_trait_impl, span);
908 }
909 let safety = self.parse_safety(Case::Sensitive);
910 let is_auto = if self.eat_keyword(exp!(Auto)) {
912 self.psess.gated_spans.gate(sym::auto_traits, self.prev_token.span);
913 IsAuto::Yes
914 } else {
915 IsAuto::No
916 };
917
918 self.expect_keyword(exp!(Trait))?;
919 let ident = self.parse_ident()?;
920 let mut generics = self.parse_generics()?;
921
922 let had_colon = self.eat(exp!(Colon));
924 let span_at_colon = self.prev_token.span;
925 let bounds = if had_colon { self.parse_generic_bounds()? } else { Vec::new() };
926
927 let span_before_eq = self.prev_token.span;
928 if self.eat(exp!(Eq)) {
929 if had_colon {
931 let span = span_at_colon.to(span_before_eq);
932 self.dcx().emit_err(errors::BoundsNotAllowedOnTraitAliases { span });
933 }
934
935 let bounds = self.parse_generic_bounds()?;
936 generics.where_clause = self.parse_where_clause()?;
937 self.expect_semi()?;
938
939 let whole_span = lo.to(self.prev_token.span);
940 if is_auto == IsAuto::Yes {
941 self.dcx().emit_err(errors::TraitAliasCannotBeAuto { span: whole_span });
942 }
943 if let Safety::Unsafe(_) = safety {
944 self.dcx().emit_err(errors::TraitAliasCannotBeUnsafe { span: whole_span });
945 }
946
947 self.psess.gated_spans.gate(sym::trait_alias, whole_span);
948
949 Ok(ItemKind::TraitAlias(Box::new(TraitAlias { constness, ident, generics, bounds })))
950 } else {
951 generics.where_clause = self.parse_where_clause()?;
953 let items = self.parse_item_list(attrs, |p| p.parse_trait_item(ForceCollect::No))?;
954 Ok(ItemKind::Trait(Box::new(Trait {
955 constness,
956 is_auto,
957 safety,
958 ident,
959 generics,
960 bounds,
961 items,
962 })))
963 }
964 }
965
966 pub fn parse_impl_item(
967 &mut self,
968 force_collect: ForceCollect,
969 ) -> PResult<'a, Option<Option<Box<AssocItem>>>> {
970 let fn_parse_mode =
971 FnParseMode { req_name: |_, _| true, context: FnContext::Impl, req_body: true };
972 self.parse_assoc_item(fn_parse_mode, force_collect)
973 }
974
975 pub fn parse_trait_item(
976 &mut self,
977 force_collect: ForceCollect,
978 ) -> PResult<'a, Option<Option<Box<AssocItem>>>> {
979 let fn_parse_mode = FnParseMode {
980 req_name: |edition, _| edition >= Edition::Edition2018,
981 context: FnContext::Trait,
982 req_body: false,
983 };
984 self.parse_assoc_item(fn_parse_mode, force_collect)
985 }
986
987 fn parse_assoc_item(
989 &mut self,
990 fn_parse_mode: FnParseMode,
991 force_collect: ForceCollect,
992 ) -> PResult<'a, Option<Option<Box<AssocItem>>>> {
993 Ok(self.parse_item_(fn_parse_mode, force_collect)?.map(
994 |Item { attrs, id, span, vis, kind, tokens }| {
995 let kind = match AssocItemKind::try_from(kind) {
996 Ok(kind) => kind,
997 Err(kind) => match kind {
998 ItemKind::Static(box StaticItem {
999 ident,
1000 ty,
1001 safety: _,
1002 mutability: _,
1003 expr,
1004 define_opaque,
1005 }) => {
1006 self.dcx().emit_err(errors::AssociatedStaticItemNotAllowed { span });
1007 let rhs = expr.map(ConstItemRhs::Body);
1008 AssocItemKind::Const(Box::new(ConstItem {
1009 defaultness: Defaultness::Final,
1010 ident,
1011 generics: Generics::default(),
1012 ty,
1013 rhs,
1014 define_opaque,
1015 }))
1016 }
1017 _ => return self.error_bad_item_kind(span, &kind, "`trait`s or `impl`s"),
1018 },
1019 };
1020 Some(Box::new(Item { attrs, id, span, vis, kind, tokens }))
1021 },
1022 ))
1023 }
1024
1025 fn parse_type_alias(&mut self, defaultness: Defaultness) -> PResult<'a, ItemKind> {
1031 let ident = self.parse_ident()?;
1032 let mut generics = self.parse_generics()?;
1033
1034 let bounds = if self.eat(exp!(Colon)) { self.parse_generic_bounds()? } else { Vec::new() };
1036 generics.where_clause = self.parse_where_clause()?;
1037
1038 let ty = if self.eat(exp!(Eq)) { Some(self.parse_ty()?) } else { None };
1039
1040 let after_where_clause = self.parse_where_clause()?;
1041
1042 self.expect_semi()?;
1043
1044 Ok(ItemKind::TyAlias(Box::new(TyAlias {
1045 defaultness,
1046 ident,
1047 generics,
1048 after_where_clause,
1049 bounds,
1050 ty,
1051 })))
1052 }
1053
1054 fn parse_use_tree(&mut self) -> PResult<'a, UseTree> {
1064 let lo = self.token.span;
1065
1066 let mut prefix =
1067 ast::Path { segments: ThinVec::new(), span: lo.shrink_to_lo(), tokens: None };
1068 let kind =
1069 if self.check(exp!(OpenBrace)) || self.check(exp!(Star)) || self.is_import_coupler() {
1070 let mod_sep_ctxt = self.token.span.ctxt();
1072 if self.eat_path_sep() {
1073 prefix
1074 .segments
1075 .push(PathSegment::path_root(lo.shrink_to_lo().with_ctxt(mod_sep_ctxt)));
1076 }
1077
1078 self.parse_use_tree_glob_or_nested()?
1079 } else {
1080 prefix = self.parse_path(PathStyle::Mod)?;
1082
1083 if self.eat_path_sep() {
1084 self.parse_use_tree_glob_or_nested()?
1085 } else {
1086 while self.eat_noexpect(&token::Colon) {
1088 self.dcx()
1089 .emit_err(errors::SingleColonImportPath { span: self.prev_token.span });
1090
1091 self.parse_path_segments(&mut prefix.segments, PathStyle::Mod, None)?;
1093 prefix.span = lo.to(self.prev_token.span);
1094 }
1095
1096 UseTreeKind::Simple(self.parse_rename()?)
1097 }
1098 };
1099
1100 Ok(UseTree { prefix, kind, span: lo.to(self.prev_token.span) })
1101 }
1102
1103 fn parse_use_tree_glob_or_nested(&mut self) -> PResult<'a, UseTreeKind> {
1105 Ok(if self.eat(exp!(Star)) {
1106 UseTreeKind::Glob
1107 } else {
1108 let lo = self.token.span;
1109 UseTreeKind::Nested {
1110 items: self.parse_use_tree_list()?,
1111 span: lo.to(self.prev_token.span),
1112 }
1113 })
1114 }
1115
1116 fn parse_use_tree_list(&mut self) -> PResult<'a, ThinVec<(UseTree, ast::NodeId)>> {
1122 self.parse_delim_comma_seq(exp!(OpenBrace), exp!(CloseBrace), |p| {
1123 p.recover_vcs_conflict_marker();
1124 Ok((p.parse_use_tree()?, DUMMY_NODE_ID))
1125 })
1126 .map(|(r, _)| r)
1127 }
1128
1129 fn parse_rename(&mut self) -> PResult<'a, Option<Ident>> {
1130 if self.eat_keyword(exp!(As)) {
1131 self.parse_ident_or_underscore().map(Some)
1132 } else {
1133 Ok(None)
1134 }
1135 }
1136
1137 fn parse_ident_or_underscore(&mut self) -> PResult<'a, Ident> {
1138 match self.token.ident() {
1139 Some((ident @ Ident { name: kw::Underscore, .. }, IdentIsRaw::No)) => {
1140 self.bump();
1141 Ok(ident)
1142 }
1143 _ => self.parse_ident(),
1144 }
1145 }
1146
1147 fn parse_item_extern_crate(&mut self) -> PResult<'a, ItemKind> {
1156 let orig_ident = self.parse_crate_name_with_dashes()?;
1158 let (orig_name, item_ident) = if let Some(rename) = self.parse_rename()? {
1159 (Some(orig_ident.name), rename)
1160 } else {
1161 (None, orig_ident)
1162 };
1163 self.expect_semi()?;
1164 Ok(ItemKind::ExternCrate(orig_name, item_ident))
1165 }
1166
1167 fn parse_crate_name_with_dashes(&mut self) -> PResult<'a, Ident> {
1168 let ident = if self.token.is_keyword(kw::SelfLower) {
1169 self.parse_path_segment_ident()
1170 } else {
1171 self.parse_ident()
1172 }?;
1173
1174 let dash = exp!(Minus);
1175 if self.token != dash.tok {
1176 return Ok(ident);
1177 }
1178
1179 let mut dashes = vec![];
1181 let mut idents = vec![];
1182 while self.eat(dash) {
1183 dashes.push(self.prev_token.span);
1184 idents.push(self.parse_ident()?);
1185 }
1186
1187 let fixed_name_sp = ident.span.to(idents.last().unwrap().span);
1188 let mut fixed_name = ident.name.to_string();
1189 for part in idents {
1190 write!(fixed_name, "_{}", part.name).unwrap();
1191 }
1192
1193 self.dcx().emit_err(errors::ExternCrateNameWithDashes {
1194 span: fixed_name_sp,
1195 sugg: errors::ExternCrateNameWithDashesSugg { dashes },
1196 });
1197
1198 Ok(Ident::from_str_and_span(&fixed_name, fixed_name_sp))
1199 }
1200
1201 fn parse_item_foreign_mod(
1212 &mut self,
1213 attrs: &mut AttrVec,
1214 mut safety: Safety,
1215 ) -> PResult<'a, ItemKind> {
1216 let extern_span = self.prev_token_uninterpolated_span();
1217 let abi = self.parse_abi(); if safety == Safety::Default
1220 && self.token.is_keyword(kw::Unsafe)
1221 && self.look_ahead(1, |t| *t == token::OpenBrace)
1222 {
1223 self.expect(exp!(OpenBrace)).unwrap_err().emit();
1224 safety = Safety::Unsafe(self.token.span);
1225 let _ = self.eat_keyword(exp!(Unsafe));
1226 }
1227 Ok(ItemKind::ForeignMod(ast::ForeignMod {
1228 extern_span,
1229 safety,
1230 abi,
1231 items: self.parse_item_list(attrs, |p| p.parse_foreign_item(ForceCollect::No))?,
1232 }))
1233 }
1234
1235 pub fn parse_foreign_item(
1237 &mut self,
1238 force_collect: ForceCollect,
1239 ) -> PResult<'a, Option<Option<Box<ForeignItem>>>> {
1240 let fn_parse_mode = FnParseMode {
1241 req_name: |_, is_dot_dot_dot| is_dot_dot_dot == IsDotDotDot::No,
1242 context: FnContext::Free,
1243 req_body: false,
1244 };
1245 Ok(self.parse_item_(fn_parse_mode, force_collect)?.map(
1246 |Item { attrs, id, span, vis, kind, tokens }| {
1247 let kind = match ForeignItemKind::try_from(kind) {
1248 Ok(kind) => kind,
1249 Err(kind) => match kind {
1250 ItemKind::Const(box ConstItem { ident, ty, rhs, .. }) => {
1251 let const_span = Some(span.with_hi(ident.span.lo()))
1252 .filter(|span| span.can_be_used_for_suggestions());
1253 self.dcx().emit_err(errors::ExternItemCannotBeConst {
1254 ident_span: ident.span,
1255 const_span,
1256 });
1257 ForeignItemKind::Static(Box::new(StaticItem {
1258 ident,
1259 ty,
1260 mutability: Mutability::Not,
1261 expr: rhs.map(|b| match b {
1262 ConstItemRhs::TypeConst(anon_const) => anon_const.value,
1263 ConstItemRhs::Body(expr) => expr,
1264 }),
1265 safety: Safety::Default,
1266 define_opaque: None,
1267 }))
1268 }
1269 _ => return self.error_bad_item_kind(span, &kind, "`extern` blocks"),
1270 },
1271 };
1272 Some(Box::new(Item { attrs, id, span, vis, kind, tokens }))
1273 },
1274 ))
1275 }
1276
1277 fn error_bad_item_kind<T>(&self, span: Span, kind: &ItemKind, ctx: &'static str) -> Option<T> {
1278 let span = self.psess.source_map().guess_head_span(span);
1280 let descr = kind.descr();
1281 let help = match kind {
1282 ItemKind::DelegationMac(deleg) if deleg.suffixes.is_none() => false,
1283 _ => true,
1284 };
1285 self.dcx().emit_err(errors::BadItemKind { span, descr, ctx, help });
1286 None
1287 }
1288
1289 fn is_use_closure(&self) -> bool {
1290 if self.token.is_keyword(kw::Use) {
1291 self.look_ahead(1, |token| {
1293 let dist =
1295 if token.is_keyword(kw::Move) || token.is_keyword(kw::Async) { 2 } else { 1 };
1296
1297 self.look_ahead(dist, |token| matches!(token.kind, token::Or | token::OrOr))
1298 })
1299 } else {
1300 false
1301 }
1302 }
1303
1304 fn is_unsafe_foreign_mod(&self) -> bool {
1305 if !self.token.is_keyword(kw::Unsafe) {
1307 return false;
1308 }
1309 if !self.is_keyword_ahead(1, &[kw::Extern]) {
1311 return false;
1312 }
1313
1314 let n = if self.look_ahead(2, |t| t.can_begin_string_literal()) { 3 } else { 2 };
1316
1317 self.tree_look_ahead(n, |t| matches!(t, TokenTree::Delimited(_, _, Delimiter::Brace, _)))
1322 == Some(true)
1323 }
1324
1325 fn parse_global_static_front_matter(&mut self, case: Case) -> Option<Safety> {
1326 let is_global_static = if self.check_keyword_case(exp!(Static), case) {
1327 !self.look_ahead(1, |token| {
1329 if token.is_keyword_case(kw::Move, case) || token.is_keyword_case(kw::Use, case) {
1330 return true;
1331 }
1332 matches!(token.kind, token::Or | token::OrOr)
1333 })
1334 } else {
1335 (self.check_keyword_case(exp!(Unsafe), case)
1337 || self.check_keyword_case(exp!(Safe), case))
1338 && self.look_ahead(1, |t| t.is_keyword_case(kw::Static, case))
1339 };
1340
1341 if is_global_static {
1342 let safety = self.parse_safety(case);
1343 let _ = self.eat_keyword_case(exp!(Static), case);
1344 Some(safety)
1345 } else {
1346 None
1347 }
1348 }
1349
1350 fn recover_const_mut(&mut self, const_span: Span) {
1352 if self.eat_keyword(exp!(Mut)) {
1353 let span = self.prev_token.span;
1354 self.dcx()
1355 .emit_err(errors::ConstGlobalCannotBeMutable { ident_span: span, const_span });
1356 } else if self.eat_keyword(exp!(Let)) {
1357 let span = self.prev_token.span;
1358 self.dcx().emit_err(errors::ConstLetMutuallyExclusive { span: const_span.to(span) });
1359 }
1360 }
1361
1362 fn parse_static_item(
1369 &mut self,
1370 safety: Safety,
1371 mutability: Mutability,
1372 ) -> PResult<'a, ItemKind> {
1373 let ident = self.parse_ident()?;
1374
1375 if self.token == TokenKind::Lt && self.may_recover() {
1376 let generics = self.parse_generics()?;
1377 self.dcx().emit_err(errors::StaticWithGenerics { span: generics.span });
1378 }
1379
1380 let ty = match (self.eat(exp!(Colon)), self.check(exp!(Eq)) | self.check(exp!(Semi))) {
1383 (true, false) => self.parse_ty()?,
1384 (colon, _) => self.recover_missing_global_item_type(colon, Some(mutability)),
1387 };
1388
1389 let expr = if self.eat(exp!(Eq)) { Some(self.parse_expr()?) } else { None };
1390
1391 self.expect_semi()?;
1392
1393 let item = StaticItem { ident, ty, safety, mutability, expr, define_opaque: None };
1394 Ok(ItemKind::Static(Box::new(item)))
1395 }
1396
1397 fn parse_const_item(
1403 &mut self,
1404 attrs: &[Attribute],
1405 ) -> PResult<'a, (Ident, Generics, Box<Ty>, Option<ast::ConstItemRhs>)> {
1406 let ident = self.parse_ident_or_underscore()?;
1407
1408 let mut generics = self.parse_generics()?;
1409
1410 if !generics.span.is_empty() {
1413 self.psess.gated_spans.gate(sym::generic_const_items, generics.span);
1414 }
1415
1416 let ty = match (
1419 self.eat(exp!(Colon)),
1420 self.check(exp!(Eq)) | self.check(exp!(Semi)) | self.check_keyword(exp!(Where)),
1421 ) {
1422 (true, false) => self.parse_ty()?,
1423 (colon, _) => self.recover_missing_global_item_type(colon, None),
1425 };
1426
1427 let before_where_clause =
1430 if self.may_recover() { self.parse_where_clause()? } else { WhereClause::default() };
1431
1432 let rhs = if self.eat(exp!(Eq)) {
1433 if attr::contains_name(attrs, sym::type_const) {
1434 Some(ConstItemRhs::TypeConst(self.parse_const_arg()?))
1435 } else {
1436 Some(ConstItemRhs::Body(self.parse_expr()?))
1437 }
1438 } else {
1439 None
1440 };
1441
1442 let after_where_clause = self.parse_where_clause()?;
1443
1444 if before_where_clause.has_where_token
1448 && let Some(rhs) = &rhs
1449 {
1450 self.dcx().emit_err(errors::WhereClauseBeforeConstBody {
1451 span: before_where_clause.span,
1452 name: ident.span,
1453 body: rhs.span(),
1454 sugg: if !after_where_clause.has_where_token {
1455 self.psess.source_map().span_to_snippet(rhs.span()).ok().map(|body_s| {
1456 errors::WhereClauseBeforeConstBodySugg {
1457 left: before_where_clause.span.shrink_to_lo(),
1458 snippet: body_s,
1459 right: before_where_clause.span.shrink_to_hi().to(rhs.span()),
1460 }
1461 })
1462 } else {
1463 None
1466 },
1467 });
1468 }
1469
1470 let mut predicates = before_where_clause.predicates;
1477 predicates.extend(after_where_clause.predicates);
1478 let where_clause = WhereClause {
1479 has_where_token: before_where_clause.has_where_token
1480 || after_where_clause.has_where_token,
1481 predicates,
1482 span: if after_where_clause.has_where_token {
1483 after_where_clause.span
1484 } else {
1485 before_where_clause.span
1486 },
1487 };
1488
1489 if where_clause.has_where_token {
1490 self.psess.gated_spans.gate(sym::generic_const_items, where_clause.span);
1491 }
1492
1493 generics.where_clause = where_clause;
1494
1495 self.expect_semi()?;
1496
1497 Ok((ident, generics, ty, rhs))
1498 }
1499
1500 fn recover_missing_global_item_type(
1503 &mut self,
1504 colon_present: bool,
1505 m: Option<Mutability>,
1506 ) -> Box<Ty> {
1507 let kind = match m {
1510 Some(Mutability::Mut) => "static mut",
1511 Some(Mutability::Not) => "static",
1512 None => "const",
1513 };
1514
1515 let colon = match colon_present {
1516 true => "",
1517 false => ":",
1518 };
1519
1520 let span = self.prev_token.span.shrink_to_hi();
1521 let err = self.dcx().create_err(errors::MissingConstType { span, colon, kind });
1522 err.stash(span, StashKey::ItemNoType);
1523
1524 Box::new(Ty { kind: TyKind::Infer, span, id: ast::DUMMY_NODE_ID, tokens: None })
1527 }
1528
1529 fn parse_item_enum(&mut self) -> PResult<'a, ItemKind> {
1531 if self.token.is_keyword(kw::Struct) {
1532 let span = self.prev_token.span.to(self.token.span);
1533 let err = errors::EnumStructMutuallyExclusive { span };
1534 if self.look_ahead(1, |t| t.is_ident()) {
1535 self.bump();
1536 self.dcx().emit_err(err);
1537 } else {
1538 return Err(self.dcx().create_err(err));
1539 }
1540 }
1541
1542 let prev_span = self.prev_token.span;
1543 let ident = self.parse_ident()?;
1544 let mut generics = self.parse_generics()?;
1545 generics.where_clause = self.parse_where_clause()?;
1546
1547 let (variants, _) = if self.token == TokenKind::Semi {
1549 self.dcx().emit_err(errors::UseEmptyBlockNotSemi { span: self.token.span });
1550 self.bump();
1551 (thin_vec![], Trailing::No)
1552 } else {
1553 self.parse_delim_comma_seq(exp!(OpenBrace), exp!(CloseBrace), |p| {
1554 p.parse_enum_variant(ident.span)
1555 })
1556 .map_err(|mut err| {
1557 err.span_label(ident.span, "while parsing this enum");
1558 if self.prev_token.is_non_reserved_ident() && self.token == token::Colon {
1560 let snapshot = self.create_snapshot_for_diagnostic();
1561 self.bump();
1562 match self.parse_ty() {
1563 Ok(_) => {
1564 err.span_suggestion_verbose(
1565 prev_span,
1566 "perhaps you meant to use `struct` here",
1567 "struct",
1568 Applicability::MaybeIncorrect,
1569 );
1570 }
1571 Err(e) => {
1572 e.cancel();
1573 }
1574 }
1575 self.restore_snapshot(snapshot);
1576 }
1577 self.eat_to_tokens(&[exp!(CloseBrace)]);
1578 self.bump(); err
1580 })?
1581 };
1582
1583 let enum_definition = EnumDef { variants: variants.into_iter().flatten().collect() };
1584 Ok(ItemKind::Enum(ident, generics, enum_definition))
1585 }
1586
1587 fn parse_enum_variant(&mut self, span: Span) -> PResult<'a, Option<Variant>> {
1588 self.recover_vcs_conflict_marker();
1589 let variant_attrs = self.parse_outer_attributes()?;
1590 self.recover_vcs_conflict_marker();
1591 let help = "enum variants can be `Variant`, `Variant = <integer>`, \
1592 `Variant(Type, ..., TypeN)` or `Variant { fields: Types }`";
1593 self.collect_tokens(None, variant_attrs, ForceCollect::No, |this, variant_attrs| {
1594 let vlo = this.token.span;
1595
1596 let vis = this.parse_visibility(FollowedByType::No)?;
1597 if !this.recover_nested_adt_item(kw::Enum)? {
1598 return Ok((None, Trailing::No, UsePreAttrPos::No));
1599 }
1600 let ident = this.parse_field_ident("enum", vlo)?;
1601
1602 if this.token == token::Bang {
1603 if let Err(err) = this.unexpected() {
1604 err.with_note(fluent::parse_macro_expands_to_enum_variant).emit();
1605 }
1606
1607 this.bump();
1608 this.parse_delim_args()?;
1609
1610 return Ok((None, Trailing::from(this.token == token::Comma), UsePreAttrPos::No));
1611 }
1612
1613 let struct_def = if this.check(exp!(OpenBrace)) {
1614 let (fields, recovered) =
1616 match this.parse_record_struct_body("struct", ident.span, false) {
1617 Ok((fields, recovered)) => (fields, recovered),
1618 Err(mut err) => {
1619 if this.token == token::Colon {
1620 return Err(err);
1622 }
1623 this.eat_to_tokens(&[exp!(CloseBrace)]);
1624 this.bump(); err.span_label(span, "while parsing this enum");
1626 err.help(help);
1627 let guar = err.emit();
1628 (thin_vec![], Recovered::Yes(guar))
1629 }
1630 };
1631 VariantData::Struct { fields, recovered }
1632 } else if this.check(exp!(OpenParen)) {
1633 let body = match this.parse_tuple_struct_body() {
1634 Ok(body) => body,
1635 Err(mut err) => {
1636 if this.token == token::Colon {
1637 return Err(err);
1639 }
1640 this.eat_to_tokens(&[exp!(CloseParen)]);
1641 this.bump(); err.span_label(span, "while parsing this enum");
1643 err.help(help);
1644 err.emit();
1645 thin_vec![]
1646 }
1647 };
1648 VariantData::Tuple(body, DUMMY_NODE_ID)
1649 } else {
1650 VariantData::Unit(DUMMY_NODE_ID)
1651 };
1652
1653 let disr_expr = if this.eat(exp!(Eq)) {
1654 Some(this.parse_expr_anon_const(|_, _| MgcaDisambiguation::AnonConst)?)
1655 } else {
1656 None
1657 };
1658
1659 let vr = ast::Variant {
1660 ident,
1661 vis,
1662 id: DUMMY_NODE_ID,
1663 attrs: variant_attrs,
1664 data: struct_def,
1665 disr_expr,
1666 span: vlo.to(this.prev_token.span),
1667 is_placeholder: false,
1668 };
1669
1670 Ok((Some(vr), Trailing::from(this.token == token::Comma), UsePreAttrPos::No))
1671 })
1672 .map_err(|mut err| {
1673 err.help(help);
1674 err
1675 })
1676 }
1677
1678 fn parse_item_struct(&mut self) -> PResult<'a, ItemKind> {
1680 let ident = self.parse_ident()?;
1681
1682 let mut generics = self.parse_generics()?;
1683
1684 let vdata = if self.token.is_keyword(kw::Where) {
1699 let tuple_struct_body;
1700 (generics.where_clause, tuple_struct_body) =
1701 self.parse_struct_where_clause(ident, generics.span)?;
1702
1703 if let Some(body) = tuple_struct_body {
1704 let body = VariantData::Tuple(body, DUMMY_NODE_ID);
1706 self.expect_semi()?;
1707 body
1708 } else if self.eat(exp!(Semi)) {
1709 VariantData::Unit(DUMMY_NODE_ID)
1711 } else {
1712 let (fields, recovered) = self.parse_record_struct_body(
1714 "struct",
1715 ident.span,
1716 generics.where_clause.has_where_token,
1717 )?;
1718 VariantData::Struct { fields, recovered }
1719 }
1720 } else if self.eat(exp!(Semi)) {
1722 VariantData::Unit(DUMMY_NODE_ID)
1723 } else if self.token == token::OpenBrace {
1725 let (fields, recovered) = self.parse_record_struct_body(
1726 "struct",
1727 ident.span,
1728 generics.where_clause.has_where_token,
1729 )?;
1730 VariantData::Struct { fields, recovered }
1731 } else if self.token == token::OpenParen {
1733 let body = VariantData::Tuple(self.parse_tuple_struct_body()?, DUMMY_NODE_ID);
1734 generics.where_clause = self.parse_where_clause()?;
1735 self.expect_semi()?;
1736 body
1737 } else {
1738 let err = errors::UnexpectedTokenAfterStructName::new(self.token.span, self.token);
1739 return Err(self.dcx().create_err(err));
1740 };
1741
1742 Ok(ItemKind::Struct(ident, generics, vdata))
1743 }
1744
1745 fn parse_item_union(&mut self) -> PResult<'a, ItemKind> {
1747 let ident = self.parse_ident()?;
1748
1749 let mut generics = self.parse_generics()?;
1750
1751 let vdata = if self.token.is_keyword(kw::Where) {
1752 generics.where_clause = self.parse_where_clause()?;
1753 let (fields, recovered) = self.parse_record_struct_body(
1754 "union",
1755 ident.span,
1756 generics.where_clause.has_where_token,
1757 )?;
1758 VariantData::Struct { fields, recovered }
1759 } else if self.token == token::OpenBrace {
1760 let (fields, recovered) = self.parse_record_struct_body(
1761 "union",
1762 ident.span,
1763 generics.where_clause.has_where_token,
1764 )?;
1765 VariantData::Struct { fields, recovered }
1766 } else {
1767 let token_str = super::token_descr(&self.token);
1768 let msg = format!("expected `where` or `{{` after union name, found {token_str}");
1769 let mut err = self.dcx().struct_span_err(self.token.span, msg);
1770 err.span_label(self.token.span, "expected `where` or `{` after union name");
1771 return Err(err);
1772 };
1773
1774 Ok(ItemKind::Union(ident, generics, vdata))
1775 }
1776
1777 pub(crate) fn parse_record_struct_body(
1782 &mut self,
1783 adt_ty: &str,
1784 ident_span: Span,
1785 parsed_where: bool,
1786 ) -> PResult<'a, (ThinVec<FieldDef>, Recovered)> {
1787 let mut fields = ThinVec::new();
1788 let mut recovered = Recovered::No;
1789 if self.eat(exp!(OpenBrace)) {
1790 while self.token != token::CloseBrace {
1791 match self.parse_field_def(adt_ty, ident_span) {
1792 Ok(field) => {
1793 fields.push(field);
1794 }
1795 Err(mut err) => {
1796 self.consume_block(
1797 exp!(OpenBrace),
1798 exp!(CloseBrace),
1799 ConsumeClosingDelim::No,
1800 );
1801 err.span_label(ident_span, format!("while parsing this {adt_ty}"));
1802 let guar = err.emit();
1803 recovered = Recovered::Yes(guar);
1804 break;
1805 }
1806 }
1807 }
1808 self.expect(exp!(CloseBrace))?;
1809 } else {
1810 let token_str = super::token_descr(&self.token);
1811 let where_str = if parsed_where { "" } else { "`where`, or " };
1812 let msg = format!("expected {where_str}`{{` after struct name, found {token_str}");
1813 let mut err = self.dcx().struct_span_err(self.token.span, msg);
1814 err.span_label(self.token.span, format!("expected {where_str}`{{` after struct name",));
1815 return Err(err);
1816 }
1817
1818 Ok((fields, recovered))
1819 }
1820
1821 fn parse_unsafe_field(&mut self) -> Safety {
1822 if self.eat_keyword(exp!(Unsafe)) {
1824 let span = self.prev_token.span;
1825 self.psess.gated_spans.gate(sym::unsafe_fields, span);
1826 Safety::Unsafe(span)
1827 } else {
1828 Safety::Default
1829 }
1830 }
1831
1832 pub(super) fn parse_tuple_struct_body(&mut self) -> PResult<'a, ThinVec<FieldDef>> {
1833 self.parse_paren_comma_seq(|p| {
1836 let attrs = p.parse_outer_attributes()?;
1837 p.collect_tokens(None, attrs, ForceCollect::No, |p, attrs| {
1838 let mut snapshot = None;
1839 if p.is_vcs_conflict_marker(&TokenKind::Shl, &TokenKind::Lt) {
1840 snapshot = Some(p.create_snapshot_for_diagnostic());
1844 }
1845 let lo = p.token.span;
1846 let vis = match p.parse_visibility(FollowedByType::Yes) {
1847 Ok(vis) => vis,
1848 Err(err) => {
1849 if let Some(ref mut snapshot) = snapshot {
1850 snapshot.recover_vcs_conflict_marker();
1851 }
1852 return Err(err);
1853 }
1854 };
1855 let ty = match p.parse_ty() {
1858 Ok(ty) => ty,
1859 Err(err) => {
1860 if let Some(ref mut snapshot) = snapshot {
1861 snapshot.recover_vcs_conflict_marker();
1862 }
1863 return Err(err);
1864 }
1865 };
1866 let mut default = None;
1867 if p.token == token::Eq {
1868 let mut snapshot = p.create_snapshot_for_diagnostic();
1869 snapshot.bump();
1870 match snapshot.parse_expr_anon_const(|_, _| MgcaDisambiguation::AnonConst) {
1871 Ok(const_expr) => {
1872 let sp = ty.span.shrink_to_hi().to(const_expr.value.span);
1873 p.psess.gated_spans.gate(sym::default_field_values, sp);
1874 p.restore_snapshot(snapshot);
1875 default = Some(const_expr);
1876 }
1877 Err(err) => {
1878 err.cancel();
1879 }
1880 }
1881 }
1882
1883 Ok((
1884 FieldDef {
1885 span: lo.to(ty.span),
1886 vis,
1887 safety: Safety::Default,
1888 ident: None,
1889 id: DUMMY_NODE_ID,
1890 ty,
1891 default,
1892 attrs,
1893 is_placeholder: false,
1894 },
1895 Trailing::from(p.token == token::Comma),
1896 UsePreAttrPos::No,
1897 ))
1898 })
1899 })
1900 .map(|(r, _)| r)
1901 }
1902
1903 fn parse_field_def(&mut self, adt_ty: &str, ident_span: Span) -> PResult<'a, FieldDef> {
1905 self.recover_vcs_conflict_marker();
1906 let attrs = self.parse_outer_attributes()?;
1907 self.recover_vcs_conflict_marker();
1908 self.collect_tokens(None, attrs, ForceCollect::No, |this, attrs| {
1909 let lo = this.token.span;
1910 let vis = this.parse_visibility(FollowedByType::No)?;
1911 let safety = this.parse_unsafe_field();
1912 this.parse_single_struct_field(adt_ty, lo, vis, safety, attrs, ident_span)
1913 .map(|field| (field, Trailing::No, UsePreAttrPos::No))
1914 })
1915 }
1916
1917 fn parse_single_struct_field(
1919 &mut self,
1920 adt_ty: &str,
1921 lo: Span,
1922 vis: Visibility,
1923 safety: Safety,
1924 attrs: AttrVec,
1925 ident_span: Span,
1926 ) -> PResult<'a, FieldDef> {
1927 let a_var = self.parse_name_and_ty(adt_ty, lo, vis, safety, attrs)?;
1928 match self.token.kind {
1929 token::Comma => {
1930 self.bump();
1931 }
1932 token::Semi => {
1933 self.bump();
1934 let sp = self.prev_token.span;
1935 let mut err =
1936 self.dcx().struct_span_err(sp, format!("{adt_ty} fields are separated by `,`"));
1937 err.span_suggestion_short(
1938 sp,
1939 "replace `;` with `,`",
1940 ",",
1941 Applicability::MachineApplicable,
1942 );
1943 err.span_label(ident_span, format!("while parsing this {adt_ty}"));
1944 err.emit();
1945 }
1946 token::CloseBrace => {}
1947 token::DocComment(..) => {
1948 let previous_span = self.prev_token.span;
1949 let mut err = errors::DocCommentDoesNotDocumentAnything {
1950 span: self.token.span,
1951 missing_comma: None,
1952 };
1953 self.bump(); if self.eat(exp!(Comma)) || self.token == token::CloseBrace {
1955 self.dcx().emit_err(err);
1956 } else {
1957 let sp = previous_span.shrink_to_hi();
1958 err.missing_comma = Some(sp);
1959 return Err(self.dcx().create_err(err));
1960 }
1961 }
1962 _ => {
1963 let sp = self.prev_token.span.shrink_to_hi();
1964 let msg =
1965 format!("expected `,`, or `}}`, found {}", super::token_descr(&self.token));
1966
1967 if let TyKind::Path(_, Path { segments, .. }) = &a_var.ty.kind
1969 && let Some(last_segment) = segments.last()
1970 {
1971 let guar = self.check_trailing_angle_brackets(
1972 last_segment,
1973 &[exp!(Comma), exp!(CloseBrace)],
1974 );
1975 if let Some(_guar) = guar {
1976 let _ = self.eat(exp!(Comma));
1979
1980 return Ok(a_var);
1983 }
1984 }
1985
1986 let mut err = self.dcx().struct_span_err(sp, msg);
1987
1988 if self.token.is_ident()
1989 || (self.token == TokenKind::Pound
1990 && (self.look_ahead(1, |t| t == &token::OpenBracket)))
1991 {
1992 err.span_suggestion(
1995 sp,
1996 "try adding a comma",
1997 ",",
1998 Applicability::MachineApplicable,
1999 );
2000 err.emit();
2001 } else {
2002 return Err(err);
2003 }
2004 }
2005 }
2006 Ok(a_var)
2007 }
2008
2009 fn expect_field_ty_separator(&mut self) -> PResult<'a, ()> {
2010 if let Err(err) = self.expect(exp!(Colon)) {
2011 let sm = self.psess.source_map();
2012 let eq_typo = self.token == token::Eq && self.look_ahead(1, |t| t.is_path_start());
2013 let semi_typo = self.token == token::Semi
2014 && self.look_ahead(1, |t| {
2015 t.is_path_start()
2016 && match (sm.lookup_line(self.token.span.hi()), sm.lookup_line(t.span.lo())) {
2019 (Ok(l), Ok(r)) => l.line == r.line,
2020 _ => true,
2021 }
2022 });
2023 if eq_typo || semi_typo {
2024 self.bump();
2025 err.with_span_suggestion_short(
2027 self.prev_token.span,
2028 "field names and their types are separated with `:`",
2029 ":",
2030 Applicability::MachineApplicable,
2031 )
2032 .emit();
2033 } else {
2034 return Err(err);
2035 }
2036 }
2037 Ok(())
2038 }
2039
2040 fn parse_name_and_ty(
2042 &mut self,
2043 adt_ty: &str,
2044 lo: Span,
2045 vis: Visibility,
2046 safety: Safety,
2047 attrs: AttrVec,
2048 ) -> PResult<'a, FieldDef> {
2049 let name = self.parse_field_ident(adt_ty, lo)?;
2050 if self.token == token::Bang {
2051 if let Err(mut err) = self.unexpected() {
2052 err.subdiagnostic(MacroExpandsToAdtField { adt_ty });
2054 return Err(err);
2055 }
2056 }
2057 self.expect_field_ty_separator()?;
2058 let ty = self.parse_ty()?;
2059 if self.token == token::Colon && self.look_ahead(1, |&t| t != token::Colon) {
2060 self.dcx()
2061 .struct_span_err(self.token.span, "found single colon in a struct field type path")
2062 .with_span_suggestion_verbose(
2063 self.token.span,
2064 "write a path separator here",
2065 "::",
2066 Applicability::MaybeIncorrect,
2067 )
2068 .emit();
2069 }
2070 let default = if self.token == token::Eq {
2071 self.bump();
2072 let const_expr = self.parse_expr_anon_const(|_, _| MgcaDisambiguation::AnonConst)?;
2073 let sp = ty.span.shrink_to_hi().to(const_expr.value.span);
2074 self.psess.gated_spans.gate(sym::default_field_values, sp);
2075 Some(const_expr)
2076 } else {
2077 None
2078 };
2079 Ok(FieldDef {
2080 span: lo.to(self.prev_token.span),
2081 ident: Some(name),
2082 vis,
2083 safety,
2084 id: DUMMY_NODE_ID,
2085 ty,
2086 default,
2087 attrs,
2088 is_placeholder: false,
2089 })
2090 }
2091
2092 fn parse_field_ident(&mut self, adt_ty: &str, lo: Span) -> PResult<'a, Ident> {
2095 let (ident, is_raw) = self.ident_or_err(true)?;
2096 if matches!(is_raw, IdentIsRaw::No) && ident.is_reserved() {
2097 let snapshot = self.create_snapshot_for_diagnostic();
2098 let err = if self.check_fn_front_matter(false, Case::Sensitive) {
2099 let inherited_vis =
2100 Visibility { span: DUMMY_SP, kind: VisibilityKind::Inherited, tokens: None };
2101 let fn_parse_mode =
2103 FnParseMode { req_name: |_, _| true, context: FnContext::Free, req_body: true };
2104 match self.parse_fn(
2105 &mut AttrVec::new(),
2106 fn_parse_mode,
2107 lo,
2108 &inherited_vis,
2109 Case::Insensitive,
2110 ) {
2111 Ok(_) => {
2112 self.dcx().struct_span_err(
2113 lo.to(self.prev_token.span),
2114 format!("functions are not allowed in {adt_ty} definitions"),
2115 )
2116 .with_help(
2117 "unlike in C++, Java, and C#, functions are declared in `impl` blocks",
2118 )
2119 .with_help("see https://doc.rust-lang.org/book/ch05-03-method-syntax.html for more information")
2120 }
2121 Err(err) => {
2122 err.cancel();
2123 self.restore_snapshot(snapshot);
2124 self.expected_ident_found_err()
2125 }
2126 }
2127 } else if self.eat_keyword(exp!(Struct)) {
2128 match self.parse_item_struct() {
2129 Ok(item) => {
2130 let ItemKind::Struct(ident, ..) = item else { unreachable!() };
2131 self.dcx()
2132 .struct_span_err(
2133 lo.with_hi(ident.span.hi()),
2134 format!("structs are not allowed in {adt_ty} definitions"),
2135 )
2136 .with_help(
2137 "consider creating a new `struct` definition instead of nesting",
2138 )
2139 }
2140 Err(err) => {
2141 err.cancel();
2142 self.restore_snapshot(snapshot);
2143 self.expected_ident_found_err()
2144 }
2145 }
2146 } else {
2147 let mut err = self.expected_ident_found_err();
2148 if self.eat_keyword_noexpect(kw::Let)
2149 && let removal_span = self.prev_token.span.until(self.token.span)
2150 && let Ok(ident) = self
2151 .parse_ident_common(false)
2152 .map_err(|err| err.cancel())
2154 && self.token == TokenKind::Colon
2155 {
2156 err.span_suggestion(
2157 removal_span,
2158 "remove this `let` keyword",
2159 String::new(),
2160 Applicability::MachineApplicable,
2161 );
2162 err.note("the `let` keyword is not allowed in `struct` fields");
2163 err.note("see <https://doc.rust-lang.org/book/ch05-01-defining-structs.html> for more information");
2164 err.emit();
2165 return Ok(ident);
2166 } else {
2167 self.restore_snapshot(snapshot);
2168 }
2169 err
2170 };
2171 return Err(err);
2172 }
2173 self.bump();
2174 Ok(ident)
2175 }
2176
2177 fn parse_item_decl_macro(&mut self, lo: Span) -> PResult<'a, ItemKind> {
2185 let ident = self.parse_ident()?;
2186 let body = if self.check(exp!(OpenBrace)) {
2187 self.parse_delim_args()? } else if self.check(exp!(OpenParen)) {
2189 let params = self.parse_token_tree(); let pspan = params.span();
2191 if !self.check(exp!(OpenBrace)) {
2192 self.unexpected()?;
2193 }
2194 let body = self.parse_token_tree(); let bspan = body.span();
2197 let arrow = TokenTree::token_alone(token::FatArrow, pspan.between(bspan)); let tokens = TokenStream::new(vec![params, arrow, body]);
2199 let dspan = DelimSpan::from_pair(pspan.shrink_to_lo(), bspan.shrink_to_hi());
2200 Box::new(DelimArgs { dspan, delim: Delimiter::Brace, tokens })
2201 } else {
2202 self.unexpected_any()?
2203 };
2204
2205 self.psess.gated_spans.gate(sym::decl_macro, lo.to(self.prev_token.span));
2206 Ok(ItemKind::MacroDef(ident, ast::MacroDef { body, macro_rules: false }))
2207 }
2208
2209 fn is_macro_rules_item(&mut self) -> IsMacroRulesItem {
2211 if self.check_keyword(exp!(MacroRules)) {
2212 let macro_rules_span = self.token.span;
2213
2214 if self.look_ahead(1, |t| *t == token::Bang) && self.look_ahead(2, |t| t.is_ident()) {
2215 return IsMacroRulesItem::Yes { has_bang: true };
2216 } else if self.look_ahead(1, |t| t.is_ident()) {
2217 self.dcx().emit_err(errors::MacroRulesMissingBang {
2219 span: macro_rules_span,
2220 hi: macro_rules_span.shrink_to_hi(),
2221 });
2222
2223 return IsMacroRulesItem::Yes { has_bang: false };
2224 }
2225 }
2226
2227 IsMacroRulesItem::No
2228 }
2229
2230 fn parse_item_macro_rules(
2232 &mut self,
2233 vis: &Visibility,
2234 has_bang: bool,
2235 ) -> PResult<'a, ItemKind> {
2236 self.expect_keyword(exp!(MacroRules))?; if has_bang {
2239 self.expect(exp!(Bang))?; }
2241 let ident = self.parse_ident()?;
2242
2243 if self.eat(exp!(Bang)) {
2244 let span = self.prev_token.span;
2246 self.dcx().emit_err(errors::MacroNameRemoveBang { span });
2247 }
2248
2249 let body = self.parse_delim_args()?;
2250 self.eat_semi_for_macro_if_needed(&body);
2251 self.complain_if_pub_macro(vis, true);
2252
2253 Ok(ItemKind::MacroDef(ident, ast::MacroDef { body, macro_rules: true }))
2254 }
2255
2256 fn complain_if_pub_macro(&self, vis: &Visibility, macro_rules: bool) {
2259 if let VisibilityKind::Inherited = vis.kind {
2260 return;
2261 }
2262
2263 let vstr = pprust::vis_to_string(vis);
2264 let vstr = vstr.trim_end();
2265 if macro_rules {
2266 self.dcx().emit_err(errors::MacroRulesVisibility { span: vis.span, vis: vstr });
2267 } else {
2268 self.dcx().emit_err(errors::MacroInvocationVisibility { span: vis.span, vis: vstr });
2269 }
2270 }
2271
2272 fn eat_semi_for_macro_if_needed(&mut self, args: &DelimArgs) {
2273 if args.need_semicolon() && !self.eat(exp!(Semi)) {
2274 self.report_invalid_macro_expansion_item(args);
2275 }
2276 }
2277
2278 fn report_invalid_macro_expansion_item(&self, args: &DelimArgs) {
2279 let span = args.dspan.entire();
2280 let mut err = self.dcx().struct_span_err(
2281 span,
2282 "macros that expand to items must be delimited with braces or followed by a semicolon",
2283 );
2284 if !span.from_expansion() {
2287 let DelimSpan { open, close } = args.dspan;
2288 err.multipart_suggestion(
2289 "change the delimiters to curly braces",
2290 vec![(open, "{".to_string()), (close, '}'.to_string())],
2291 Applicability::MaybeIncorrect,
2292 );
2293 err.span_suggestion(
2294 span.with_neighbor(self.token.span).shrink_to_hi(),
2295 "add a semicolon",
2296 ';',
2297 Applicability::MaybeIncorrect,
2298 );
2299 }
2300 err.emit();
2301 }
2302
2303 fn recover_nested_adt_item(&mut self, keyword: Symbol) -> PResult<'a, bool> {
2306 if (self.token.is_keyword(kw::Enum)
2307 || self.token.is_keyword(kw::Struct)
2308 || self.token.is_keyword(kw::Union))
2309 && self.look_ahead(1, |t| t.is_ident())
2310 {
2311 let kw_token = self.token;
2312 let kw_str = pprust::token_to_string(&kw_token);
2313 let item = self.parse_item(ForceCollect::No)?;
2314 let mut item = item.unwrap().span;
2315 if self.token == token::Comma {
2316 item = item.to(self.token.span);
2317 }
2318 self.dcx().emit_err(errors::NestedAdt {
2319 span: kw_token.span,
2320 item,
2321 kw_str,
2322 keyword: keyword.as_str(),
2323 });
2324 return Ok(false);
2326 }
2327 Ok(true)
2328 }
2329}
2330
2331type ReqName = fn(Edition, IsDotDotDot) -> bool;
2340
2341#[derive(Copy, Clone, PartialEq)]
2342pub(crate) enum IsDotDotDot {
2343 Yes,
2344 No,
2345}
2346
2347#[derive(Clone, Copy)]
2355pub(crate) struct FnParseMode {
2356 pub(super) req_name: ReqName,
2382 pub(super) context: FnContext,
2385 pub(super) req_body: bool,
2404}
2405
2406#[derive(Clone, Copy, PartialEq, Eq)]
2409pub(crate) enum FnContext {
2410 Free,
2412 Trait,
2414 Impl,
2416}
2417
2418impl<'a> Parser<'a> {
2420 fn parse_fn(
2422 &mut self,
2423 attrs: &mut AttrVec,
2424 fn_parse_mode: FnParseMode,
2425 sig_lo: Span,
2426 vis: &Visibility,
2427 case: Case,
2428 ) -> PResult<'a, (Ident, FnSig, Generics, Option<Box<FnContract>>, Option<Box<Block>>)> {
2429 let fn_span = self.token.span;
2430 let header = self.parse_fn_front_matter(vis, case, FrontMatterParsingMode::Function)?; let ident = self.parse_ident()?; let mut generics = self.parse_generics()?; let decl = match self.parse_fn_decl(&fn_parse_mode, AllowPlus::Yes, RecoverReturnSign::Yes)
2434 {
2435 Ok(decl) => decl,
2436 Err(old_err) => {
2437 if self.token.is_keyword(kw::For) {
2439 old_err.cancel();
2440 return Err(self.dcx().create_err(errors::FnTypoWithImpl { fn_span }));
2441 } else {
2442 return Err(old_err);
2443 }
2444 }
2445 };
2446
2447 let fn_params_end = self.prev_token.span.shrink_to_hi();
2450
2451 let contract = self.parse_contract()?;
2452
2453 generics.where_clause = self.parse_where_clause()?; let fn_params_end =
2457 if generics.where_clause.has_where_token { Some(fn_params_end) } else { None };
2458
2459 let mut sig_hi = self.prev_token.span;
2460 let body =
2462 self.parse_fn_body(attrs, &ident, &mut sig_hi, fn_parse_mode.req_body, fn_params_end)?;
2463 let fn_sig_span = sig_lo.to(sig_hi);
2464 Ok((ident, FnSig { header, decl, span: fn_sig_span }, generics, contract, body))
2465 }
2466
2467 fn error_fn_body_not_found(
2469 &mut self,
2470 ident_span: Span,
2471 req_body: bool,
2472 fn_params_end: Option<Span>,
2473 ) -> PResult<'a, ErrorGuaranteed> {
2474 let expected: &[_] =
2475 if req_body { &[exp!(OpenBrace)] } else { &[exp!(Semi), exp!(OpenBrace)] };
2476 match self.expected_one_of_not_found(&[], expected) {
2477 Ok(error_guaranteed) => Ok(error_guaranteed),
2478 Err(mut err) => {
2479 if self.token == token::CloseBrace {
2480 err.span_label(ident_span, "while parsing this `fn`");
2483 Ok(err.emit())
2484 } else if self.token == token::RArrow
2485 && let Some(fn_params_end) = fn_params_end
2486 {
2487 let fn_trait_span =
2493 [sym::FnOnce, sym::FnMut, sym::Fn].into_iter().find_map(|symbol| {
2494 if self.prev_token.is_ident_named(symbol) {
2495 Some(self.prev_token.span)
2496 } else {
2497 None
2498 }
2499 });
2500
2501 let arrow_span = self.token.span;
2506 let ty_span = match self.parse_ret_ty(
2507 AllowPlus::Yes,
2508 RecoverQPath::Yes,
2509 RecoverReturnSign::Yes,
2510 ) {
2511 Ok(ty_span) => ty_span.span().shrink_to_hi(),
2512 Err(parse_error) => {
2513 parse_error.cancel();
2514 return Err(err);
2515 }
2516 };
2517 let ret_ty_span = arrow_span.to(ty_span);
2518
2519 if let Some(fn_trait_span) = fn_trait_span {
2520 err.subdiagnostic(errors::FnTraitMissingParen { span: fn_trait_span });
2523 } else if let Ok(snippet) = self.psess.source_map().span_to_snippet(ret_ty_span)
2524 {
2525 err.primary_message(
2529 "return type should be specified after the function parameters",
2530 );
2531 err.subdiagnostic(errors::MisplacedReturnType {
2532 fn_params_end,
2533 snippet,
2534 ret_ty_span,
2535 });
2536 }
2537 Err(err)
2538 } else {
2539 Err(err)
2540 }
2541 }
2542 }
2543 }
2544
2545 fn parse_fn_body(
2549 &mut self,
2550 attrs: &mut AttrVec,
2551 ident: &Ident,
2552 sig_hi: &mut Span,
2553 req_body: bool,
2554 fn_params_end: Option<Span>,
2555 ) -> PResult<'a, Option<Box<Block>>> {
2556 let has_semi = if req_body {
2557 self.token == TokenKind::Semi
2558 } else {
2559 self.check(exp!(Semi))
2561 };
2562 let (inner_attrs, body) = if has_semi {
2563 self.expect_semi()?;
2565 *sig_hi = self.prev_token.span;
2566 (AttrVec::new(), None)
2567 } else if self.check(exp!(OpenBrace)) || self.token.is_metavar_block() {
2568 self.parse_block_common(self.token.span, BlockCheckMode::Default, None)
2569 .map(|(attrs, body)| (attrs, Some(body)))?
2570 } else if self.token == token::Eq {
2571 self.bump(); let eq_sp = self.prev_token.span;
2574 let _ = self.parse_expr()?;
2575 self.expect_semi()?; let span = eq_sp.to(self.prev_token.span);
2577 let guar = self.dcx().emit_err(errors::FunctionBodyEqualsExpr {
2578 span,
2579 sugg: errors::FunctionBodyEqualsExprSugg { eq: eq_sp, semi: self.prev_token.span },
2580 });
2581 (AttrVec::new(), Some(self.mk_block_err(span, guar)))
2582 } else {
2583 self.error_fn_body_not_found(ident.span, req_body, fn_params_end)?;
2584 (AttrVec::new(), None)
2585 };
2586 attrs.extend(inner_attrs);
2587 Ok(body)
2588 }
2589
2590 fn check_impl_frontmatter(&mut self) -> bool {
2591 const ALL_QUALS: &[Symbol] = &[kw::Const, kw::Unsafe];
2592 if self.check_keyword(exp!(Impl)) {
2595 return true;
2596 }
2597 let mut i = 0;
2598 while i < ALL_QUALS.len() {
2599 let action = self.look_ahead(i, |token| {
2600 if token.is_keyword(kw::Impl) {
2601 return Some(true);
2602 }
2603 if ALL_QUALS.iter().any(|&qual| token.is_keyword(qual)) {
2604 return None;
2606 }
2607 Some(false)
2608 });
2609 if let Some(ret) = action {
2610 return ret;
2611 }
2612 i += 1;
2613 }
2614 self.is_keyword_ahead(i, &[kw::Impl])
2615 }
2616
2617 pub(super) fn check_fn_front_matter(&mut self, check_pub: bool, case: Case) -> bool {
2622 const ALL_QUALS: &[ExpKeywordPair] = &[
2623 exp!(Pub),
2624 exp!(Gen),
2625 exp!(Const),
2626 exp!(Async),
2627 exp!(Unsafe),
2628 exp!(Safe),
2629 exp!(Extern),
2630 ];
2631
2632 let quals: &[_] = if check_pub {
2637 ALL_QUALS
2638 } else {
2639 &[exp!(Gen), exp!(Const), exp!(Async), exp!(Unsafe), exp!(Safe), exp!(Extern)]
2640 };
2641 self.check_keyword_case(exp!(Fn), case) || quals.iter().any(|&exp| self.check_keyword_case(exp, case))
2644 && self.look_ahead(1, |t| {
2645 t.is_keyword_case(kw::Fn, case)
2647 || (
2649 (
2650 t.is_non_raw_ident_where(|i|
2651 quals.iter().any(|exp| exp.kw == i.name)
2652 && i.is_reserved()
2654 )
2655 || case == Case::Insensitive
2656 && t.is_non_raw_ident_where(|i| quals.iter().any(|exp| {
2657 exp.kw.as_str() == i.name.as_str().to_lowercase()
2658 }))
2659 )
2660 && !self.is_unsafe_foreign_mod()
2662 && !self.is_async_gen_block()
2664 && !self.is_keyword_ahead(2, &[kw::Auto, kw::Trait])
2666 )
2667 })
2668 || self.check_keyword_case(exp!(Extern), case)
2670 && self.look_ahead(1, |t| t.can_begin_string_literal())
2674 && (self.tree_look_ahead(2, |tt| {
2675 match tt {
2676 TokenTree::Token(t, _) => t.is_keyword_case(kw::Fn, case),
2677 TokenTree::Delimited(..) => false,
2678 }
2679 }) == Some(true) ||
2680 (self.may_recover()
2683 && self.tree_look_ahead(2, |tt| {
2684 match tt {
2685 TokenTree::Token(t, _) =>
2686 ALL_QUALS.iter().any(|exp| {
2687 t.is_keyword(exp.kw)
2688 }),
2689 TokenTree::Delimited(..) => false,
2690 }
2691 }) == Some(true)
2692 && self.tree_look_ahead(3, |tt| {
2693 match tt {
2694 TokenTree::Token(t, _) => t.is_keyword_case(kw::Fn, case),
2695 TokenTree::Delimited(..) => false,
2696 }
2697 }) == Some(true)
2698 )
2699 )
2700 }
2701
2702 pub(super) fn parse_fn_front_matter(
2717 &mut self,
2718 orig_vis: &Visibility,
2719 case: Case,
2720 parsing_mode: FrontMatterParsingMode,
2721 ) -> PResult<'a, FnHeader> {
2722 let sp_start = self.token.span;
2723 let constness = self.parse_constness(case);
2724 if parsing_mode == FrontMatterParsingMode::FunctionPtrType
2725 && let Const::Yes(const_span) = constness
2726 {
2727 self.dcx().emit_err(FnPointerCannotBeConst {
2728 span: const_span,
2729 suggestion: const_span.until(self.token.span),
2730 });
2731 }
2732
2733 let async_start_sp = self.token.span;
2734 let coroutine_kind = self.parse_coroutine_kind(case);
2735 if parsing_mode == FrontMatterParsingMode::FunctionPtrType
2736 && let Some(ast::CoroutineKind::Async { span: async_span, .. }) = coroutine_kind
2737 {
2738 self.dcx().emit_err(FnPointerCannotBeAsync {
2739 span: async_span,
2740 suggestion: async_span.until(self.token.span),
2741 });
2742 }
2743 let unsafe_start_sp = self.token.span;
2746 let safety = self.parse_safety(case);
2747
2748 let ext_start_sp = self.token.span;
2749 let ext = self.parse_extern(case);
2750
2751 if let Some(CoroutineKind::Async { span, .. }) = coroutine_kind {
2752 if span.is_rust_2015() {
2753 self.dcx().emit_err(errors::AsyncFnIn2015 {
2754 span,
2755 help: errors::HelpUseLatestEdition::new(),
2756 });
2757 }
2758 }
2759
2760 match coroutine_kind {
2761 Some(CoroutineKind::Gen { span, .. }) | Some(CoroutineKind::AsyncGen { span, .. }) => {
2762 self.psess.gated_spans.gate(sym::gen_blocks, span);
2763 }
2764 Some(CoroutineKind::Async { .. }) | None => {}
2765 }
2766
2767 if !self.eat_keyword_case(exp!(Fn), case) {
2768 match self.expect_one_of(&[], &[]) {
2772 Ok(Recovered::Yes(_)) => {}
2773 Ok(Recovered::No) => unreachable!(),
2774 Err(mut err) => {
2775 enum WrongKw {
2777 Duplicated(Span),
2778 Misplaced(Span),
2779 MisplacedDisallowedQualifier,
2784 }
2785
2786 let mut recover_constness = constness;
2788 let mut recover_coroutine_kind = coroutine_kind;
2789 let mut recover_safety = safety;
2790 let wrong_kw = if self.check_keyword(exp!(Const)) {
2793 match constness {
2794 Const::Yes(sp) => Some(WrongKw::Duplicated(sp)),
2795 Const::No => {
2796 recover_constness = Const::Yes(self.token.span);
2797 match parsing_mode {
2798 FrontMatterParsingMode::Function => {
2799 Some(WrongKw::Misplaced(async_start_sp))
2800 }
2801 FrontMatterParsingMode::FunctionPtrType => {
2802 self.dcx().emit_err(FnPointerCannotBeConst {
2803 span: self.token.span,
2804 suggestion: self
2805 .token
2806 .span
2807 .with_lo(self.prev_token.span.hi()),
2808 });
2809 Some(WrongKw::MisplacedDisallowedQualifier)
2810 }
2811 }
2812 }
2813 }
2814 } else if self.check_keyword(exp!(Async)) {
2815 match coroutine_kind {
2816 Some(CoroutineKind::Async { span, .. }) => {
2817 Some(WrongKw::Duplicated(span))
2818 }
2819 Some(CoroutineKind::AsyncGen { span, .. }) => {
2820 Some(WrongKw::Duplicated(span))
2821 }
2822 Some(CoroutineKind::Gen { .. }) => {
2823 recover_coroutine_kind = Some(CoroutineKind::AsyncGen {
2824 span: self.token.span,
2825 closure_id: DUMMY_NODE_ID,
2826 return_impl_trait_id: DUMMY_NODE_ID,
2827 });
2828 Some(WrongKw::Misplaced(unsafe_start_sp))
2830 }
2831 None => {
2832 recover_coroutine_kind = Some(CoroutineKind::Async {
2833 span: self.token.span,
2834 closure_id: DUMMY_NODE_ID,
2835 return_impl_trait_id: DUMMY_NODE_ID,
2836 });
2837 match parsing_mode {
2838 FrontMatterParsingMode::Function => {
2839 Some(WrongKw::Misplaced(async_start_sp))
2840 }
2841 FrontMatterParsingMode::FunctionPtrType => {
2842 self.dcx().emit_err(FnPointerCannotBeAsync {
2843 span: self.token.span,
2844 suggestion: self
2845 .token
2846 .span
2847 .with_lo(self.prev_token.span.hi()),
2848 });
2849 Some(WrongKw::MisplacedDisallowedQualifier)
2850 }
2851 }
2852 }
2853 }
2854 } else if self.check_keyword(exp!(Unsafe)) {
2855 match safety {
2856 Safety::Unsafe(sp) => Some(WrongKw::Duplicated(sp)),
2857 Safety::Safe(sp) => {
2858 recover_safety = Safety::Unsafe(self.token.span);
2859 Some(WrongKw::Misplaced(sp))
2860 }
2861 Safety::Default => {
2862 recover_safety = Safety::Unsafe(self.token.span);
2863 Some(WrongKw::Misplaced(ext_start_sp))
2864 }
2865 }
2866 } else if self.check_keyword(exp!(Safe)) {
2867 match safety {
2868 Safety::Safe(sp) => Some(WrongKw::Duplicated(sp)),
2869 Safety::Unsafe(sp) => {
2870 recover_safety = Safety::Safe(self.token.span);
2871 Some(WrongKw::Misplaced(sp))
2872 }
2873 Safety::Default => {
2874 recover_safety = Safety::Safe(self.token.span);
2875 Some(WrongKw::Misplaced(ext_start_sp))
2876 }
2877 }
2878 } else {
2879 None
2880 };
2881
2882 if let Some(WrongKw::Duplicated(original_sp)) = wrong_kw {
2884 let original_kw = self
2885 .span_to_snippet(original_sp)
2886 .expect("Span extracted directly from keyword should always work");
2887
2888 err.span_suggestion(
2889 self.token_uninterpolated_span(),
2890 format!("`{original_kw}` already used earlier, remove this one"),
2891 "",
2892 Applicability::MachineApplicable,
2893 )
2894 .span_note(original_sp, format!("`{original_kw}` first seen here"));
2895 }
2896 else if let Some(WrongKw::Misplaced(correct_pos_sp)) = wrong_kw {
2898 let correct_pos_sp = correct_pos_sp.to(self.prev_token.span);
2899 if let Ok(current_qual) = self.span_to_snippet(correct_pos_sp) {
2900 let misplaced_qual_sp = self.token_uninterpolated_span();
2901 let misplaced_qual = self.span_to_snippet(misplaced_qual_sp).unwrap();
2902
2903 err.span_suggestion(
2904 correct_pos_sp.to(misplaced_qual_sp),
2905 format!("`{misplaced_qual}` must come before `{current_qual}`"),
2906 format!("{misplaced_qual} {current_qual}"),
2907 Applicability::MachineApplicable,
2908 ).note("keyword order for functions declaration is `pub`, `default`, `const`, `async`, `unsafe`, `extern`");
2909 }
2910 }
2911 else if self.check_keyword(exp!(Pub)) {
2913 let sp = sp_start.to(self.prev_token.span);
2914 if let Ok(snippet) = self.span_to_snippet(sp) {
2915 let current_vis = match self.parse_visibility(FollowedByType::No) {
2916 Ok(v) => v,
2917 Err(d) => {
2918 d.cancel();
2919 return Err(err);
2920 }
2921 };
2922 let vs = pprust::vis_to_string(¤t_vis);
2923 let vs = vs.trim_end();
2924
2925 if matches!(orig_vis.kind, VisibilityKind::Inherited) {
2927 err.span_suggestion(
2928 sp_start.to(self.prev_token.span),
2929 format!("visibility `{vs}` must come before `{snippet}`"),
2930 format!("{vs} {snippet}"),
2931 Applicability::MachineApplicable,
2932 );
2933 }
2934 else {
2936 err.span_suggestion(
2937 current_vis.span,
2938 "there is already a visibility modifier, remove one",
2939 "",
2940 Applicability::MachineApplicable,
2941 )
2942 .span_note(orig_vis.span, "explicit visibility first seen here");
2943 }
2944 }
2945 }
2946
2947 if let Some(wrong_kw) = wrong_kw
2950 && self.may_recover()
2951 && self.look_ahead(1, |tok| tok.is_keyword_case(kw::Fn, case))
2952 {
2953 self.bump();
2955 self.bump();
2956 if matches!(wrong_kw, WrongKw::MisplacedDisallowedQualifier) {
2959 err.cancel();
2960 } else {
2961 err.emit();
2962 }
2963 return Ok(FnHeader {
2964 constness: recover_constness,
2965 safety: recover_safety,
2966 coroutine_kind: recover_coroutine_kind,
2967 ext,
2968 });
2969 }
2970
2971 return Err(err);
2972 }
2973 }
2974 }
2975
2976 Ok(FnHeader { constness, safety, coroutine_kind, ext })
2977 }
2978
2979 pub(super) fn parse_fn_decl(
2981 &mut self,
2982 fn_parse_mode: &FnParseMode,
2983 ret_allow_plus: AllowPlus,
2984 recover_return_sign: RecoverReturnSign,
2985 ) -> PResult<'a, Box<FnDecl>> {
2986 Ok(Box::new(FnDecl {
2987 inputs: self.parse_fn_params(fn_parse_mode)?,
2988 output: self.parse_ret_ty(ret_allow_plus, RecoverQPath::Yes, recover_return_sign)?,
2989 }))
2990 }
2991
2992 pub(super) fn parse_fn_params(
2994 &mut self,
2995 fn_parse_mode: &FnParseMode,
2996 ) -> PResult<'a, ThinVec<Param>> {
2997 let mut first_param = true;
2998 if self.token != TokenKind::OpenParen
3000 && !self.token.is_keyword(kw::For)
3002 {
3003 self.dcx()
3005 .emit_err(errors::MissingFnParams { span: self.prev_token.span.shrink_to_hi() });
3006 return Ok(ThinVec::new());
3007 }
3008
3009 let (mut params, _) = self.parse_paren_comma_seq(|p| {
3010 p.recover_vcs_conflict_marker();
3011 let snapshot = p.create_snapshot_for_diagnostic();
3012 let param = p.parse_param_general(fn_parse_mode, first_param, true).or_else(|e| {
3013 let guar = e.emit();
3014 let lo = if let TokenKind::OpenParen = p.prev_token.kind {
3018 p.prev_token.span.shrink_to_hi()
3019 } else {
3020 p.prev_token.span
3021 };
3022 p.restore_snapshot(snapshot);
3023 p.eat_to_tokens(&[exp!(Comma), exp!(CloseParen)]);
3025 Ok(dummy_arg(Ident::new(sym::dummy, lo.to(p.prev_token.span)), guar))
3027 });
3028 first_param = false;
3030 param
3031 })?;
3032 self.deduplicate_recovered_params_names(&mut params);
3034 Ok(params)
3035 }
3036
3037 pub(super) fn parse_param_general(
3042 &mut self,
3043 fn_parse_mode: &FnParseMode,
3044 first_param: bool,
3045 recover_arg_parse: bool,
3046 ) -> PResult<'a, Param> {
3047 let lo = self.token.span;
3048 let attrs = self.parse_outer_attributes()?;
3049 self.collect_tokens(None, attrs, ForceCollect::No, |this, attrs| {
3050 if let Some(mut param) = this.parse_self_param()? {
3052 param.attrs = attrs;
3053 let res = if first_param { Ok(param) } else { this.recover_bad_self_param(param) };
3054 return Ok((res?, Trailing::No, UsePreAttrPos::No));
3055 }
3056
3057 let is_dot_dot_dot = if this.token.kind == token::DotDotDot {
3058 IsDotDotDot::Yes
3059 } else {
3060 IsDotDotDot::No
3061 };
3062 let is_name_required = (fn_parse_mode.req_name)(
3063 this.token.span.with_neighbor(this.prev_token.span).edition(),
3064 is_dot_dot_dot,
3065 );
3066 let is_name_required = if is_name_required && is_dot_dot_dot == IsDotDotDot::Yes {
3067 this.psess.buffer_lint(
3068 VARARGS_WITHOUT_PATTERN,
3069 this.token.span,
3070 ast::CRATE_NODE_ID,
3071 errors::VarargsWithoutPattern { span: this.token.span },
3072 );
3073 false
3074 } else {
3075 is_name_required
3076 };
3077 let (pat, ty) = if is_name_required || this.is_named_param() {
3078 debug!("parse_param_general parse_pat (is_name_required:{})", is_name_required);
3079 let (pat, colon) = this.parse_fn_param_pat_colon()?;
3080 if !colon {
3081 let mut err = this.unexpected().unwrap_err();
3082 return if let Some(ident) = this.parameter_without_type(
3083 &mut err,
3084 pat,
3085 is_name_required,
3086 first_param,
3087 fn_parse_mode,
3088 ) {
3089 let guar = err.emit();
3090 Ok((dummy_arg(ident, guar), Trailing::No, UsePreAttrPos::No))
3091 } else {
3092 Err(err)
3093 };
3094 }
3095
3096 this.eat_incorrect_doc_comment_for_param_type();
3097 (pat, this.parse_ty_for_param()?)
3098 } else {
3099 debug!("parse_param_general ident_to_pat");
3100 let parser_snapshot_before_ty = this.create_snapshot_for_diagnostic();
3101 this.eat_incorrect_doc_comment_for_param_type();
3102 let mut ty = this.parse_ty_for_param();
3103
3104 if let Ok(t) = &ty {
3105 if let TyKind::Path(_, Path { segments, .. }) = &t.kind
3107 && let Some(segment) = segments.last()
3108 && let Some(guar) =
3109 this.check_trailing_angle_brackets(segment, &[exp!(CloseParen)])
3110 {
3111 return Ok((
3112 dummy_arg(segment.ident, guar),
3113 Trailing::No,
3114 UsePreAttrPos::No,
3115 ));
3116 }
3117
3118 if this.token != token::Comma && this.token != token::CloseParen {
3119 ty = this.unexpected_any();
3122 }
3123 }
3124 match ty {
3125 Ok(ty) => {
3126 let pat = this.mk_pat(ty.span, PatKind::Missing);
3127 (Box::new(pat), ty)
3128 }
3129 Err(err) if this.token == token::DotDotDot => return Err(err),
3131 Err(err) if this.unmatched_angle_bracket_count > 0 => return Err(err),
3132 Err(err) if recover_arg_parse => {
3133 err.cancel();
3135 this.restore_snapshot(parser_snapshot_before_ty);
3136 this.recover_arg_parse()?
3137 }
3138 Err(err) => return Err(err),
3139 }
3140 };
3141
3142 let span = lo.to(this.prev_token.span);
3143
3144 Ok((
3145 Param { attrs, id: ast::DUMMY_NODE_ID, is_placeholder: false, pat, span, ty },
3146 Trailing::No,
3147 UsePreAttrPos::No,
3148 ))
3149 })
3150 }
3151
3152 fn parse_self_param(&mut self) -> PResult<'a, Option<Param>> {
3154 let expect_self_ident = |this: &mut Self| match this.token.ident() {
3156 Some((ident, IdentIsRaw::No)) => {
3157 this.bump();
3158 ident
3159 }
3160 _ => unreachable!(),
3161 };
3162 let is_lifetime = |this: &Self, n| this.look_ahead(n, |t| t.is_lifetime());
3164 let is_isolated_self = |this: &Self, n| {
3166 this.is_keyword_ahead(n, &[kw::SelfLower])
3167 && this.look_ahead(n + 1, |t| t != &token::PathSep)
3168 };
3169 let is_isolated_pin_const_self = |this: &Self, n| {
3171 this.look_ahead(n, |token| token.is_ident_named(sym::pin))
3172 && this.is_keyword_ahead(n + 1, &[kw::Const])
3173 && is_isolated_self(this, n + 2)
3174 };
3175 let is_isolated_mut_self =
3177 |this: &Self, n| this.is_keyword_ahead(n, &[kw::Mut]) && is_isolated_self(this, n + 1);
3178 let is_isolated_pin_mut_self = |this: &Self, n| {
3180 this.look_ahead(n, |token| token.is_ident_named(sym::pin))
3181 && is_isolated_mut_self(this, n + 1)
3182 };
3183 let parse_self_possibly_typed = |this: &mut Self, m| {
3185 let eself_ident = expect_self_ident(this);
3186 let eself_hi = this.prev_token.span;
3187 let eself = if this.eat(exp!(Colon)) {
3188 SelfKind::Explicit(this.parse_ty()?, m)
3189 } else {
3190 SelfKind::Value(m)
3191 };
3192 Ok((eself, eself_ident, eself_hi))
3193 };
3194 let expect_self_ident_not_typed =
3195 |this: &mut Self, modifier: &SelfKind, modifier_span: Span| {
3196 let eself_ident = expect_self_ident(this);
3197
3198 if this.may_recover() && this.eat_noexpect(&token::Colon) {
3200 let snap = this.create_snapshot_for_diagnostic();
3201 match this.parse_ty() {
3202 Ok(ty) => {
3203 this.dcx().emit_err(errors::IncorrectTypeOnSelf {
3204 span: ty.span,
3205 move_self_modifier: errors::MoveSelfModifier {
3206 removal_span: modifier_span,
3207 insertion_span: ty.span.shrink_to_lo(),
3208 modifier: modifier.to_ref_suggestion(),
3209 },
3210 });
3211 }
3212 Err(diag) => {
3213 diag.cancel();
3214 this.restore_snapshot(snap);
3215 }
3216 }
3217 }
3218 eself_ident
3219 };
3220 let recover_self_ptr = |this: &mut Self| {
3222 this.dcx().emit_err(errors::SelfArgumentPointer { span: this.token.span });
3223
3224 Ok((SelfKind::Value(Mutability::Not), expect_self_ident(this), this.prev_token.span))
3225 };
3226
3227 let eself_lo = self.token.span;
3231 let (eself, eself_ident, eself_hi) = match self.token.uninterpolate().kind {
3232 token::And => {
3233 let has_lifetime = is_lifetime(self, 1);
3234 let skip_lifetime_count = has_lifetime as usize;
3235 let eself = if is_isolated_self(self, skip_lifetime_count + 1) {
3236 self.bump(); let lifetime = has_lifetime.then(|| self.expect_lifetime());
3239 SelfKind::Region(lifetime, Mutability::Not)
3240 } else if is_isolated_mut_self(self, skip_lifetime_count + 1) {
3241 self.bump(); let lifetime = has_lifetime.then(|| self.expect_lifetime());
3244 self.bump(); SelfKind::Region(lifetime, Mutability::Mut)
3246 } else if is_isolated_pin_const_self(self, skip_lifetime_count + 1) {
3247 self.bump(); let lifetime = has_lifetime.then(|| self.expect_lifetime());
3250 self.psess.gated_spans.gate(sym::pin_ergonomics, self.token.span);
3251 self.bump(); self.bump(); SelfKind::Pinned(lifetime, Mutability::Not)
3254 } else if is_isolated_pin_mut_self(self, skip_lifetime_count + 1) {
3255 self.bump(); let lifetime = has_lifetime.then(|| self.expect_lifetime());
3258 self.psess.gated_spans.gate(sym::pin_ergonomics, self.token.span);
3259 self.bump(); self.bump(); SelfKind::Pinned(lifetime, Mutability::Mut)
3262 } else {
3263 return Ok(None);
3265 };
3266 let hi = self.token.span;
3267 let self_ident = expect_self_ident_not_typed(self, &eself, eself_lo.until(hi));
3268 (eself, self_ident, hi)
3269 }
3270 token::Star if is_isolated_self(self, 1) => {
3272 self.bump();
3273 recover_self_ptr(self)?
3274 }
3275 token::Star
3277 if self.look_ahead(1, |t| t.is_mutability()) && is_isolated_self(self, 2) =>
3278 {
3279 self.bump();
3280 self.bump();
3281 recover_self_ptr(self)?
3282 }
3283 token::Ident(..) if is_isolated_self(self, 0) => {
3285 parse_self_possibly_typed(self, Mutability::Not)?
3286 }
3287 token::Ident(..) if is_isolated_mut_self(self, 0) => {
3289 self.bump();
3290 parse_self_possibly_typed(self, Mutability::Mut)?
3291 }
3292 _ => return Ok(None),
3293 };
3294
3295 let eself = source_map::respan(eself_lo.to(eself_hi), eself);
3296 Ok(Some(Param::from_self(AttrVec::default(), eself, eself_ident)))
3297 }
3298
3299 fn is_named_param(&self) -> bool {
3300 let offset = match &self.token.kind {
3301 token::OpenInvisible(origin) => match origin {
3302 InvisibleOrigin::MetaVar(MetaVarKind::Pat(_)) => {
3303 return self.check_noexpect_past_close_delim(&token::Colon);
3304 }
3305 _ => 0,
3306 },
3307 token::And | token::AndAnd => 1,
3308 _ if self.token.is_keyword(kw::Mut) => 1,
3309 _ => 0,
3310 };
3311
3312 self.look_ahead(offset, |t| t.is_ident())
3313 && self.look_ahead(offset + 1, |t| t == &token::Colon)
3314 }
3315
3316 fn recover_self_param(&mut self) -> bool {
3317 matches!(
3318 self.parse_outer_attributes()
3319 .and_then(|_| self.parse_self_param())
3320 .map_err(|e| e.cancel()),
3321 Ok(Some(_))
3322 )
3323 }
3324}
3325
3326enum IsMacroRulesItem {
3327 Yes { has_bang: bool },
3328 No,
3329}
3330
3331#[derive(Copy, Clone, PartialEq, Eq)]
3332pub(super) enum FrontMatterParsingMode {
3333 Function,
3335 FunctionPtrType,
3338}