1use std::fmt::Write;
2use std::mem;
3
4use ast::token::IdentIsRaw;
5use rustc_ast::ast::*;
6use rustc_ast::ptr::P;
7use rustc_ast::token::{self, Delimiter, InvisibleOrigin, MetaVarKind, TokenKind};
8use rustc_ast::tokenstream::{DelimSpan, TokenStream, TokenTree};
9use rustc_ast::util::case::Case;
10use rustc_ast::{self as ast};
11use rustc_ast_pretty::pprust;
12use rustc_errors::codes::*;
13use rustc_errors::{Applicability, PResult, StashKey, struct_span_code_err};
14use rustc_span::edit_distance::edit_distance;
15use rustc_span::edition::Edition;
16use rustc_span::{DUMMY_SP, ErrorGuaranteed, Ident, Span, Symbol, kw, source_map, sym};
17use thin_vec::{ThinVec, thin_vec};
18use tracing::debug;
19
20use super::diagnostics::{ConsumeClosingDelim, dummy_arg};
21use super::ty::{AllowPlus, RecoverQPath, RecoverReturnSign};
22use super::{
23 AttrWrapper, ExpKeywordPair, ExpTokenPair, FollowedByType, ForceCollect, Parser, PathStyle,
24 Recovered, Trailing, UsePreAttrPos,
25};
26use crate::errors::{self, MacroExpandsToAdtField};
27use crate::{exp, fluent_generated as fluent};
28
29impl<'a> Parser<'a> {
30 pub fn parse_crate_mod(&mut self) -> PResult<'a, ast::Crate> {
32 let (attrs, items, spans) = self.parse_mod(exp!(Eof))?;
33 Ok(ast::Crate { attrs, items, spans, id: DUMMY_NODE_ID, is_placeholder: false })
34 }
35
36 fn parse_item_mod(&mut self, attrs: &mut AttrVec) -> PResult<'a, ItemInfo> {
38 let safety = self.parse_safety(Case::Sensitive);
39 self.expect_keyword(exp!(Mod))?;
40 let id = self.parse_ident()?;
41 let mod_kind = if self.eat(exp!(Semi)) {
42 ModKind::Unloaded
43 } else {
44 self.expect(exp!(OpenBrace))?;
45 let (inner_attrs, items, inner_span) = self.parse_mod(exp!(CloseBrace))?;
46 attrs.extend(inner_attrs);
47 ModKind::Loaded(items, Inline::Yes, inner_span, Ok(()))
48 };
49 Ok((id, ItemKind::Mod(safety, mod_kind)))
50 }
51
52 pub fn parse_mod(
57 &mut self,
58 term: ExpTokenPair<'_>,
59 ) -> PResult<'a, (AttrVec, ThinVec<P<Item>>, ModSpans)> {
60 let lo = self.token.span;
61 let attrs = self.parse_inner_attributes()?;
62
63 let post_attr_lo = self.token.span;
64 let mut items: ThinVec<P<_>> = ThinVec::new();
65
66 loop {
69 while self.maybe_consume_incorrect_semicolon(items.last().map(|x| &**x)) {} let Some(item) = self.parse_item(ForceCollect::No)? else {
71 break;
72 };
73 items.push(item);
74 }
75
76 if !self.eat(term) {
77 let token_str = super::token_descr(&self.token);
78 if !self.maybe_consume_incorrect_semicolon(items.last().map(|x| &**x)) {
79 let is_let = self.token.is_keyword(kw::Let);
80 let is_let_mut = is_let && self.look_ahead(1, |t| t.is_keyword(kw::Mut));
81 let let_has_ident = is_let && !is_let_mut && self.is_kw_followed_by_ident(kw::Let);
82
83 let msg = format!("expected item, found {token_str}");
84 let mut err = self.dcx().struct_span_err(self.token.span, msg);
85
86 let label = if is_let {
87 "`let` cannot be used for global variables"
88 } else {
89 "expected item"
90 };
91 err.span_label(self.token.span, label);
92
93 if is_let {
94 if is_let_mut {
95 err.help("consider using `static` and a `Mutex` instead of `let mut`");
96 } else if let_has_ident {
97 err.span_suggestion_short(
98 self.token.span,
99 "consider using `static` or `const` instead of `let`",
100 "static",
101 Applicability::MaybeIncorrect,
102 );
103 } else {
104 err.help("consider using `static` or `const` instead of `let`");
105 }
106 }
107 err.note("for a full list of items that can appear in modules, see <https://doc.rust-lang.org/reference/items.html>");
108 return Err(err);
109 }
110 }
111
112 let inject_use_span = post_attr_lo.data().with_hi(post_attr_lo.lo());
113 let mod_spans = ModSpans { inner_span: lo.to(self.prev_token.span), inject_use_span };
114 Ok((attrs, items, mod_spans))
115 }
116}
117
118pub(super) type ItemInfo = (Ident, ItemKind);
119
120impl<'a> Parser<'a> {
121 pub fn parse_item(&mut self, force_collect: ForceCollect) -> PResult<'a, Option<P<Item>>> {
122 let fn_parse_mode = FnParseMode { req_name: |_| true, req_body: true };
123 self.parse_item_(fn_parse_mode, force_collect).map(|i| i.map(P))
124 }
125
126 fn parse_item_(
127 &mut self,
128 fn_parse_mode: FnParseMode,
129 force_collect: ForceCollect,
130 ) -> PResult<'a, Option<Item>> {
131 self.recover_vcs_conflict_marker();
132 let attrs = self.parse_outer_attributes()?;
133 self.recover_vcs_conflict_marker();
134 self.parse_item_common(attrs, true, false, fn_parse_mode, force_collect)
135 }
136
137 pub(super) fn parse_item_common(
138 &mut self,
139 attrs: AttrWrapper,
140 mac_allowed: bool,
141 attrs_allowed: bool,
142 fn_parse_mode: FnParseMode,
143 force_collect: ForceCollect,
144 ) -> PResult<'a, Option<Item>> {
145 if let Some(item) =
146 self.eat_metavar_seq(MetaVarKind::Item, |this| this.parse_item(ForceCollect::Yes))
147 {
148 let mut item = item.expect("an actual item");
149 attrs.prepend_to_nt_inner(&mut item.attrs);
150 return Ok(Some(item.into_inner()));
151 }
152
153 self.collect_tokens(None, attrs, force_collect, |this, mut attrs| {
154 let lo = this.token.span;
155 let vis = this.parse_visibility(FollowedByType::No)?;
156 let mut def = this.parse_defaultness();
157 let kind = this.parse_item_kind(
158 &mut attrs,
159 mac_allowed,
160 lo,
161 &vis,
162 &mut def,
163 fn_parse_mode,
164 Case::Sensitive,
165 )?;
166 if let Some((ident, kind)) = kind {
167 this.error_on_unconsumed_default(def, &kind);
168 let span = lo.to(this.prev_token.span);
169 let id = DUMMY_NODE_ID;
170 let item = Item { ident, attrs, id, kind, vis, span, tokens: None };
171 return Ok((Some(item), Trailing::No, UsePreAttrPos::No));
172 }
173
174 if !matches!(vis.kind, VisibilityKind::Inherited) {
176 this.dcx().emit_err(errors::VisibilityNotFollowedByItem { span: vis.span, vis });
177 }
178
179 if let Defaultness::Default(span) = def {
180 this.dcx().emit_err(errors::DefaultNotFollowedByItem { span });
181 }
182
183 if !attrs_allowed {
184 this.recover_attrs_no_item(&attrs)?;
185 }
186 Ok((None, Trailing::No, UsePreAttrPos::No))
187 })
188 }
189
190 fn error_on_unconsumed_default(&self, def: Defaultness, kind: &ItemKind) {
192 if let Defaultness::Default(span) = def {
193 self.dcx().emit_err(errors::InappropriateDefault {
194 span,
195 article: kind.article(),
196 descr: kind.descr(),
197 });
198 }
199 }
200
201 fn parse_item_kind(
203 &mut self,
204 attrs: &mut AttrVec,
205 macros_allowed: bool,
206 lo: Span,
207 vis: &Visibility,
208 def: &mut Defaultness,
209 fn_parse_mode: FnParseMode,
210 case: Case,
211 ) -> PResult<'a, Option<ItemInfo>> {
212 let check_pub = def == &Defaultness::Final;
213 let mut def_ = || mem::replace(def, Defaultness::Final);
214
215 let info = if !self.is_use_closure() && self.eat_keyword_case(exp!(Use), case) {
216 self.parse_use_item()?
217 } else if self.check_fn_front_matter(check_pub, case) {
218 let (ident, sig, generics, contract, body) =
220 self.parse_fn(attrs, fn_parse_mode, lo, vis, case)?;
221 (
222 ident,
223 ItemKind::Fn(Box::new(Fn {
224 defaultness: def_(),
225 sig,
226 generics,
227 contract,
228 body,
229 define_opaque: None,
230 })),
231 )
232 } else if self.eat_keyword(exp!(Extern)) {
233 if self.eat_keyword(exp!(Crate)) {
234 self.parse_item_extern_crate()?
236 } else {
237 self.parse_item_foreign_mod(attrs, Safety::Default)?
239 }
240 } else if self.is_unsafe_foreign_mod() {
241 let safety = self.parse_safety(Case::Sensitive);
243 self.expect_keyword(exp!(Extern))?;
244 self.parse_item_foreign_mod(attrs, safety)?
245 } else if self.is_static_global() {
246 let safety = self.parse_safety(Case::Sensitive);
247 self.bump(); let mutability = self.parse_mutability();
250 let (ident, item) = self.parse_static_item(safety, mutability)?;
251 (ident, ItemKind::Static(Box::new(item)))
252 } else if let Const::Yes(const_span) = self.parse_constness(Case::Sensitive) {
253 if self.token.is_keyword(kw::Impl) {
255 self.recover_const_impl(const_span, attrs, def_())?
257 } else {
258 self.recover_const_mut(const_span);
259 self.recover_missing_kw_before_item()?;
260 let (ident, generics, ty, expr) = self.parse_const_item()?;
261 (
262 ident,
263 ItemKind::Const(Box::new(ConstItem {
264 defaultness: def_(),
265 generics,
266 ty,
267 expr,
268 define_opaque: None,
269 })),
270 )
271 }
272 } else if self.check_keyword(exp!(Trait)) || self.check_auto_or_unsafe_trait_item() {
273 self.parse_item_trait(attrs, lo)?
275 } else if self.check_keyword(exp!(Impl))
276 || self.check_keyword(exp!(Unsafe)) && self.is_keyword_ahead(1, &[kw::Impl])
277 {
278 self.parse_item_impl(attrs, def_())?
280 } else if self.is_reuse_path_item() {
281 self.parse_item_delegation()?
282 } else if self.check_keyword(exp!(Mod))
283 || self.check_keyword(exp!(Unsafe)) && self.is_keyword_ahead(1, &[kw::Mod])
284 {
285 self.parse_item_mod(attrs)?
287 } else if self.eat_keyword(exp!(Type)) {
288 self.parse_type_alias(def_())?
290 } else if self.eat_keyword(exp!(Enum)) {
291 self.parse_item_enum()?
293 } else if self.eat_keyword(exp!(Struct)) {
294 self.parse_item_struct()?
296 } else if self.is_kw_followed_by_ident(kw::Union) {
297 self.bump(); self.parse_item_union()?
300 } else if self.is_builtin() {
301 return self.parse_item_builtin();
303 } else if self.eat_keyword(exp!(Macro)) {
304 self.parse_item_decl_macro(lo)?
306 } else if let IsMacroRulesItem::Yes { has_bang } = self.is_macro_rules_item() {
307 self.parse_item_macro_rules(vis, has_bang)?
309 } else if self.isnt_macro_invocation()
310 && (self.token.is_ident_named(sym::import)
311 || self.token.is_ident_named(sym::using)
312 || self.token.is_ident_named(sym::include)
313 || self.token.is_ident_named(sym::require))
314 {
315 return self.recover_import_as_use();
316 } else if self.isnt_macro_invocation() && vis.kind.is_pub() {
317 self.recover_missing_kw_before_item()?;
318 return Ok(None);
319 } else if self.isnt_macro_invocation() && case == Case::Sensitive {
320 _ = def_;
321
322 return self.parse_item_kind(
324 attrs,
325 macros_allowed,
326 lo,
327 vis,
328 def,
329 fn_parse_mode,
330 Case::Insensitive,
331 );
332 } else if macros_allowed && self.check_path() {
333 if self.isnt_macro_invocation() {
334 self.recover_missing_kw_before_item()?;
335 }
336 (Ident::empty(), ItemKind::MacCall(P(self.parse_item_macro(vis)?)))
338 } else {
339 return Ok(None);
340 };
341 Ok(Some(info))
342 }
343
344 fn recover_import_as_use(&mut self) -> PResult<'a, Option<ItemInfo>> {
345 let span = self.token.span;
346 let token_name = super::token_descr(&self.token);
347 let snapshot = self.create_snapshot_for_diagnostic();
348 self.bump();
349 match self.parse_use_item() {
350 Ok(u) => {
351 self.dcx().emit_err(errors::RecoverImportAsUse { span, token_name });
352 Ok(Some(u))
353 }
354 Err(e) => {
355 e.cancel();
356 self.restore_snapshot(snapshot);
357 Ok(None)
358 }
359 }
360 }
361
362 fn parse_use_item(&mut self) -> PResult<'a, ItemInfo> {
363 let tree = self.parse_use_tree()?;
364 if let Err(mut e) = self.expect_semi() {
365 match tree.kind {
366 UseTreeKind::Glob => {
367 e.note("the wildcard token must be last on the path");
368 }
369 UseTreeKind::Nested { .. } => {
370 e.note("glob-like brace syntax must be last on the path");
371 }
372 _ => (),
373 }
374 return Err(e);
375 }
376 Ok((Ident::empty(), ItemKind::Use(tree)))
377 }
378
379 pub(super) fn is_path_start_item(&mut self) -> bool {
381 self.is_kw_followed_by_ident(kw::Union) || self.is_reuse_path_item()
383 || self.check_auto_or_unsafe_trait_item() || self.is_async_fn() || matches!(self.is_macro_rules_item(), IsMacroRulesItem::Yes{..}) }
387
388 fn is_reuse_path_item(&mut self) -> bool {
389 self.token.is_keyword(kw::Reuse)
391 && self.look_ahead(1, |t| t.is_path_start() && *t != token::PathSep)
392 }
393
394 fn isnt_macro_invocation(&mut self) -> bool {
396 self.check_ident() && self.look_ahead(1, |t| *t != token::Bang && *t != token::PathSep)
397 }
398
399 fn recover_missing_kw_before_item(&mut self) -> PResult<'a, ()> {
402 let is_pub = self.prev_token.is_keyword(kw::Pub);
403 let is_const = self.prev_token.is_keyword(kw::Const);
404 let ident_span = self.token.span;
405 let span = if is_pub { self.prev_token.span.to(ident_span) } else { ident_span };
406 let insert_span = ident_span.shrink_to_lo();
407
408 let ident = if self.token.is_ident()
409 && (!is_const || self.look_ahead(1, |t| *t == token::OpenDelim(Delimiter::Parenthesis)))
410 && self.look_ahead(1, |t| {
411 [
412 token::Lt,
413 token::OpenDelim(Delimiter::Brace),
414 token::OpenDelim(Delimiter::Parenthesis),
415 ]
416 .contains(&t.kind)
417 }) {
418 self.parse_ident().unwrap()
419 } else {
420 return Ok(());
421 };
422
423 let mut found_generics = false;
424 if self.check(exp!(Lt)) {
425 found_generics = true;
426 self.eat_to_tokens(&[exp!(Gt)]);
427 self.bump(); }
429
430 let err = if self.check(exp!(OpenBrace)) {
431 if self.look_ahead(1, |t| *t == token::CloseDelim(Delimiter::Brace)) {
433 Some(errors::MissingKeywordForItemDefinition::EnumOrStruct { span })
435 } else if self.look_ahead(2, |t| *t == token::Colon)
436 || self.look_ahead(3, |t| *t == token::Colon)
437 {
438 Some(errors::MissingKeywordForItemDefinition::Struct { span, insert_span, ident })
440 } else {
441 Some(errors::MissingKeywordForItemDefinition::Enum { span, insert_span, ident })
442 }
443 } else if self.check(exp!(OpenParen)) {
444 self.bump(); let is_method = self.recover_self_param();
447
448 self.consume_block(exp!(OpenParen), exp!(CloseParen), ConsumeClosingDelim::Yes);
449
450 let err = if self.check(exp!(RArrow)) || self.check(exp!(OpenBrace)) {
451 self.eat_to_tokens(&[exp!(OpenBrace)]);
452 self.bump(); self.consume_block(exp!(OpenBrace), exp!(CloseBrace), ConsumeClosingDelim::Yes);
454 if is_method {
455 errors::MissingKeywordForItemDefinition::Method { span, insert_span, ident }
456 } else {
457 errors::MissingKeywordForItemDefinition::Function { span, insert_span, ident }
458 }
459 } else if is_pub && self.check(exp!(Semi)) {
460 errors::MissingKeywordForItemDefinition::Struct { span, insert_span, ident }
461 } else {
462 errors::MissingKeywordForItemDefinition::Ambiguous {
463 span,
464 subdiag: if found_generics {
465 None
466 } else if let Ok(snippet) = self.span_to_snippet(ident_span) {
467 Some(errors::AmbiguousMissingKwForItemSub::SuggestMacro {
468 span: ident_span,
469 snippet,
470 })
471 } else {
472 Some(errors::AmbiguousMissingKwForItemSub::HelpMacro)
473 },
474 }
475 };
476 Some(err)
477 } else if found_generics {
478 Some(errors::MissingKeywordForItemDefinition::Ambiguous { span, subdiag: None })
479 } else {
480 None
481 };
482
483 if let Some(err) = err { Err(self.dcx().create_err(err)) } else { Ok(()) }
484 }
485
486 fn parse_item_builtin(&mut self) -> PResult<'a, Option<ItemInfo>> {
487 Ok(None)
489 }
490
491 fn parse_item_macro(&mut self, vis: &Visibility) -> PResult<'a, MacCall> {
493 let path = self.parse_path(PathStyle::Mod)?; self.expect(exp!(Bang))?; match self.parse_delim_args() {
496 Ok(args) => {
498 self.eat_semi_for_macro_if_needed(&args);
499 self.complain_if_pub_macro(vis, false);
500 Ok(MacCall { path, args })
501 }
502
503 Err(mut err) => {
504 if self.token.is_ident()
506 && let [segment] = path.segments.as_slice()
507 && edit_distance("macro_rules", &segment.ident.to_string(), 2).is_some()
508 {
509 err.span_suggestion(
510 path.span,
511 "perhaps you meant to define a macro",
512 "macro_rules",
513 Applicability::MachineApplicable,
514 );
515 }
516 Err(err)
517 }
518 }
519 }
520
521 fn recover_attrs_no_item(&mut self, attrs: &[Attribute]) -> PResult<'a, ()> {
523 let ([start @ end] | [start, .., end]) = attrs else {
524 return Ok(());
525 };
526 let msg = if end.is_doc_comment() {
527 "expected item after doc comment"
528 } else {
529 "expected item after attributes"
530 };
531 let mut err = self.dcx().struct_span_err(end.span, msg);
532 if end.is_doc_comment() {
533 err.span_label(end.span, "this doc comment doesn't document anything");
534 } else if self.token == TokenKind::Semi {
535 err.span_suggestion_verbose(
536 self.token.span,
537 "consider removing this semicolon",
538 "",
539 Applicability::MaybeIncorrect,
540 );
541 }
542 if let [.., penultimate, _] = attrs {
543 err.span_label(start.span.to(penultimate.span), "other attributes here");
544 }
545 Err(err)
546 }
547
548 fn is_async_fn(&self) -> bool {
549 self.token.is_keyword(kw::Async) && self.is_keyword_ahead(1, &[kw::Fn])
550 }
551
552 fn parse_polarity(&mut self) -> ast::ImplPolarity {
553 if self.check(exp!(Bang)) && self.look_ahead(1, |t| t.can_begin_type()) {
555 self.bump(); ast::ImplPolarity::Negative(self.prev_token.span)
557 } else {
558 ast::ImplPolarity::Positive
559 }
560 }
561
562 fn parse_item_impl(
577 &mut self,
578 attrs: &mut AttrVec,
579 defaultness: Defaultness,
580 ) -> PResult<'a, ItemInfo> {
581 let safety = self.parse_safety(Case::Sensitive);
582 self.expect_keyword(exp!(Impl))?;
583
584 let mut generics = if self.choose_generics_over_qpath(0) {
586 self.parse_generics()?
587 } else {
588 let mut generics = Generics::default();
589 generics.span = self.prev_token.span.shrink_to_hi();
592 generics
593 };
594
595 let constness = self.parse_constness(Case::Sensitive);
596 if let Const::Yes(span) = constness {
597 self.psess.gated_spans.gate(sym::const_trait_impl, span);
598 }
599
600 if (self.token.uninterpolated_span().at_least_rust_2018()
602 && self.token.is_keyword(kw::Async))
603 || self.is_kw_followed_by_ident(kw::Async)
604 {
605 self.bump();
606 self.dcx().emit_err(errors::AsyncImpl { span: self.prev_token.span });
607 }
608
609 let polarity = self.parse_polarity();
610
611 let err_path = |span| ast::Path::from_ident(Ident::new(kw::Empty, span));
613 let ty_first = if self.token.is_keyword(kw::For) && self.look_ahead(1, |t| t != &token::Lt)
614 {
615 let span = self.prev_token.span.between(self.token.span);
616 self.dcx().emit_err(errors::MissingTraitInTraitImpl {
617 span,
618 for_span: span.to(self.token.span),
619 });
620
621 P(Ty {
622 kind: TyKind::Path(None, err_path(span)),
623 span,
624 id: DUMMY_NODE_ID,
625 tokens: None,
626 })
627 } else {
628 self.parse_ty_with_generics_recovery(&generics)?
629 };
630
631 let has_for = self.eat_keyword(exp!(For));
633 let missing_for_span = self.prev_token.span.between(self.token.span);
634
635 let ty_second = if self.token == token::DotDot {
636 self.bump(); Some(self.mk_ty(self.prev_token.span, TyKind::Dummy))
643 } else if has_for || self.token.can_begin_type() {
644 Some(self.parse_ty()?)
645 } else {
646 None
647 };
648
649 generics.where_clause = self.parse_where_clause()?;
650
651 let impl_items = self.parse_item_list(attrs, |p| p.parse_impl_item(ForceCollect::No))?;
652
653 let (of_trait, self_ty) = match ty_second {
654 Some(ty_second) => {
655 if !has_for {
657 self.dcx().emit_err(errors::MissingForInTraitImpl { span: missing_for_span });
658 }
659
660 let ty_first = ty_first.into_inner();
661 let path = match ty_first.kind {
662 TyKind::Path(None, path) => path,
664 other => {
665 if let TyKind::ImplTrait(_, bounds) = other
666 && let [bound] = bounds.as_slice()
667 {
668 let extra_impl_kw = ty_first.span.until(bound.span());
672 self.dcx().emit_err(errors::ExtraImplKeywordInTraitImpl {
673 extra_impl_kw,
674 impl_trait_span: ty_first.span,
675 });
676 } else {
677 self.dcx().emit_err(errors::ExpectedTraitInTraitImplFoundType {
678 span: ty_first.span,
679 });
680 }
681 err_path(ty_first.span)
682 }
683 };
684 let trait_ref = TraitRef { path, ref_id: ty_first.id };
685
686 (Some(trait_ref), ty_second)
687 }
688 None => (None, ty_first), };
690 let item_kind = ItemKind::Impl(Box::new(Impl {
691 safety,
692 polarity,
693 defaultness,
694 constness,
695 generics,
696 of_trait,
697 self_ty,
698 items: impl_items,
699 }));
700
701 Ok((Ident::empty(), item_kind))
702 }
703
704 fn parse_item_delegation(&mut self) -> PResult<'a, ItemInfo> {
705 let span = self.token.span;
706 self.expect_keyword(exp!(Reuse))?;
707
708 let (qself, path) = if self.eat_lt() {
709 let (qself, path) = self.parse_qpath(PathStyle::Expr)?;
710 (Some(qself), path)
711 } else {
712 (None, self.parse_path(PathStyle::Expr)?)
713 };
714
715 let rename = |this: &mut Self| {
716 Ok(if this.eat_keyword(exp!(As)) { Some(this.parse_ident()?) } else { None })
717 };
718 let body = |this: &mut Self| {
719 Ok(if this.check(exp!(OpenBrace)) {
720 Some(this.parse_block()?)
721 } else {
722 this.expect(exp!(Semi))?;
723 None
724 })
725 };
726
727 let (ident, item_kind) = if self.eat_path_sep() {
728 let suffixes = if self.eat(exp!(Star)) {
729 None
730 } else {
731 let parse_suffix = |p: &mut Self| Ok((p.parse_path_segment_ident()?, rename(p)?));
732 Some(self.parse_delim_comma_seq(exp!(OpenBrace), exp!(CloseBrace), parse_suffix)?.0)
733 };
734 let deleg = DelegationMac { qself, prefix: path, suffixes, body: body(self)? };
735 (Ident::empty(), ItemKind::DelegationMac(Box::new(deleg)))
736 } else {
737 let rename = rename(self)?;
738 let ident = rename.unwrap_or_else(|| path.segments.last().unwrap().ident);
739 let deleg = Delegation {
740 id: DUMMY_NODE_ID,
741 qself,
742 path,
743 rename,
744 body: body(self)?,
745 from_glob: false,
746 };
747 (ident, ItemKind::Delegation(Box::new(deleg)))
748 };
749
750 let span = span.to(self.prev_token.span);
751 self.psess.gated_spans.gate(sym::fn_delegation, span);
752
753 Ok((ident, item_kind))
754 }
755
756 fn parse_item_list<T>(
757 &mut self,
758 attrs: &mut AttrVec,
759 mut parse_item: impl FnMut(&mut Parser<'a>) -> PResult<'a, Option<Option<T>>>,
760 ) -> PResult<'a, ThinVec<T>> {
761 let open_brace_span = self.token.span;
762
763 if self.token == TokenKind::Semi {
765 self.dcx().emit_err(errors::UseEmptyBlockNotSemi { span: self.token.span });
766 self.bump();
767 return Ok(ThinVec::new());
768 }
769
770 self.expect(exp!(OpenBrace))?;
771 attrs.extend(self.parse_inner_attributes()?);
772
773 let mut items = ThinVec::new();
774 while !self.eat(exp!(CloseBrace)) {
775 if self.recover_doc_comment_before_brace() {
776 continue;
777 }
778 self.recover_vcs_conflict_marker();
779 match parse_item(self) {
780 Ok(None) => {
781 let mut is_unnecessary_semicolon = !items.is_empty()
782 && self
799 .span_to_snippet(self.prev_token.span)
800 .is_ok_and(|snippet| snippet == "}")
801 && self.token == token::Semi;
802 let mut semicolon_span = self.token.span;
803 if !is_unnecessary_semicolon {
804 is_unnecessary_semicolon = self.token == token::OpenDelim(Delimiter::Brace)
806 && self.prev_token == token::Semi;
807 semicolon_span = self.prev_token.span;
808 }
809 let non_item_span = self.token.span;
811 let is_let = self.token.is_keyword(kw::Let);
812
813 let mut err =
814 self.dcx().struct_span_err(non_item_span, "non-item in item list");
815 self.consume_block(exp!(OpenBrace), exp!(CloseBrace), ConsumeClosingDelim::Yes);
816 if is_let {
817 err.span_suggestion_verbose(
818 non_item_span,
819 "consider using `const` instead of `let` for associated const",
820 "const",
821 Applicability::MachineApplicable,
822 );
823 } else {
824 err.span_label(open_brace_span, "item list starts here")
825 .span_label(non_item_span, "non-item starts here")
826 .span_label(self.prev_token.span, "item list ends here");
827 }
828 if is_unnecessary_semicolon {
829 err.span_suggestion(
830 semicolon_span,
831 "consider removing this semicolon",
832 "",
833 Applicability::MaybeIncorrect,
834 );
835 }
836 err.emit();
837 break;
838 }
839 Ok(Some(item)) => items.extend(item),
840 Err(err) => {
841 self.consume_block(exp!(OpenBrace), exp!(CloseBrace), ConsumeClosingDelim::Yes);
842 err.with_span_label(
843 open_brace_span,
844 "while parsing this item list starting here",
845 )
846 .with_span_label(self.prev_token.span, "the item list ends here")
847 .emit();
848 break;
849 }
850 }
851 }
852 Ok(items)
853 }
854
855 fn recover_doc_comment_before_brace(&mut self) -> bool {
857 if let token::DocComment(..) = self.token.kind {
858 if self.look_ahead(1, |tok| tok == &token::CloseDelim(Delimiter::Brace)) {
859 struct_span_code_err!(
861 self.dcx(),
862 self.token.span,
863 E0584,
864 "found a documentation comment that doesn't document anything",
865 )
866 .with_span_label(self.token.span, "this doc comment doesn't document anything")
867 .with_help(
868 "doc comments must come before what they document, if a comment was \
869 intended use `//`",
870 )
871 .emit();
872 self.bump();
873 return true;
874 }
875 }
876 false
877 }
878
879 fn parse_defaultness(&mut self) -> Defaultness {
881 if self.check_keyword(exp!(Default))
885 && self.look_ahead(1, |t| t.is_non_raw_ident_where(|i| i.name != kw::As))
886 {
887 self.bump(); Defaultness::Default(self.prev_token.uninterpolated_span())
889 } else {
890 Defaultness::Final
891 }
892 }
893
894 fn check_auto_or_unsafe_trait_item(&mut self) -> bool {
896 self.check_keyword(exp!(Auto)) && self.is_keyword_ahead(1, &[kw::Trait])
898 || self.check_keyword(exp!(Unsafe)) && self.is_keyword_ahead(1, &[kw::Trait, kw::Auto])
900 }
901
902 fn parse_item_trait(&mut self, attrs: &mut AttrVec, lo: Span) -> PResult<'a, ItemInfo> {
904 let safety = self.parse_safety(Case::Sensitive);
905 let is_auto = if self.eat_keyword(exp!(Auto)) {
907 self.psess.gated_spans.gate(sym::auto_traits, self.prev_token.span);
908 IsAuto::Yes
909 } else {
910 IsAuto::No
911 };
912
913 self.expect_keyword(exp!(Trait))?;
914 let ident = self.parse_ident()?;
915 let mut generics = self.parse_generics()?;
916
917 let had_colon = self.eat(exp!(Colon));
919 let span_at_colon = self.prev_token.span;
920 let bounds = if had_colon { self.parse_generic_bounds()? } else { Vec::new() };
921
922 let span_before_eq = self.prev_token.span;
923 if self.eat(exp!(Eq)) {
924 if had_colon {
926 let span = span_at_colon.to(span_before_eq);
927 self.dcx().emit_err(errors::BoundsNotAllowedOnTraitAliases { span });
928 }
929
930 let bounds = self.parse_generic_bounds()?;
931 generics.where_clause = self.parse_where_clause()?;
932 self.expect_semi()?;
933
934 let whole_span = lo.to(self.prev_token.span);
935 if is_auto == IsAuto::Yes {
936 self.dcx().emit_err(errors::TraitAliasCannotBeAuto { span: whole_span });
937 }
938 if let Safety::Unsafe(_) = safety {
939 self.dcx().emit_err(errors::TraitAliasCannotBeUnsafe { span: whole_span });
940 }
941
942 self.psess.gated_spans.gate(sym::trait_alias, whole_span);
943
944 Ok((ident, ItemKind::TraitAlias(generics, bounds)))
945 } else {
946 generics.where_clause = self.parse_where_clause()?;
948 let items = self.parse_item_list(attrs, |p| p.parse_trait_item(ForceCollect::No))?;
949 Ok((
950 ident,
951 ItemKind::Trait(Box::new(Trait { is_auto, safety, generics, bounds, items })),
952 ))
953 }
954 }
955
956 pub fn parse_impl_item(
957 &mut self,
958 force_collect: ForceCollect,
959 ) -> PResult<'a, Option<Option<P<AssocItem>>>> {
960 let fn_parse_mode = FnParseMode { req_name: |_| true, req_body: true };
961 self.parse_assoc_item(fn_parse_mode, force_collect)
962 }
963
964 pub fn parse_trait_item(
965 &mut self,
966 force_collect: ForceCollect,
967 ) -> PResult<'a, Option<Option<P<AssocItem>>>> {
968 let fn_parse_mode =
969 FnParseMode { req_name: |edition| edition >= Edition::Edition2018, req_body: false };
970 self.parse_assoc_item(fn_parse_mode, force_collect)
971 }
972
973 fn parse_assoc_item(
975 &mut self,
976 fn_parse_mode: FnParseMode,
977 force_collect: ForceCollect,
978 ) -> PResult<'a, Option<Option<P<AssocItem>>>> {
979 Ok(self.parse_item_(fn_parse_mode, force_collect)?.map(
980 |Item { attrs, id, span, vis, ident, kind, tokens }| {
981 let kind = match AssocItemKind::try_from(kind) {
982 Ok(kind) => kind,
983 Err(kind) => match kind {
984 ItemKind::Static(box StaticItem {
985 ty,
986 safety: _,
987 mutability: _,
988 expr,
989 define_opaque,
990 }) => {
991 self.dcx().emit_err(errors::AssociatedStaticItemNotAllowed { span });
992 AssocItemKind::Const(Box::new(ConstItem {
993 defaultness: Defaultness::Final,
994 generics: Generics::default(),
995 ty,
996 expr,
997 define_opaque,
998 }))
999 }
1000 _ => return self.error_bad_item_kind(span, &kind, "`trait`s or `impl`s"),
1001 },
1002 };
1003 Some(P(Item { attrs, id, span, vis, ident, kind, tokens }))
1004 },
1005 ))
1006 }
1007
1008 fn parse_type_alias(&mut self, defaultness: Defaultness) -> PResult<'a, ItemInfo> {
1014 let ident = self.parse_ident()?;
1015 let mut generics = self.parse_generics()?;
1016
1017 let bounds = if self.eat(exp!(Colon)) { self.parse_generic_bounds()? } else { Vec::new() };
1019 let before_where_clause = self.parse_where_clause()?;
1020
1021 let ty = if self.eat(exp!(Eq)) { Some(self.parse_ty()?) } else { None };
1022
1023 let after_where_clause = self.parse_where_clause()?;
1024
1025 let where_clauses = TyAliasWhereClauses {
1026 before: TyAliasWhereClause {
1027 has_where_token: before_where_clause.has_where_token,
1028 span: before_where_clause.span,
1029 },
1030 after: TyAliasWhereClause {
1031 has_where_token: after_where_clause.has_where_token,
1032 span: after_where_clause.span,
1033 },
1034 split: before_where_clause.predicates.len(),
1035 };
1036 let mut predicates = before_where_clause.predicates;
1037 predicates.extend(after_where_clause.predicates);
1038 let where_clause = WhereClause {
1039 has_where_token: before_where_clause.has_where_token
1040 || after_where_clause.has_where_token,
1041 predicates,
1042 span: DUMMY_SP,
1043 };
1044 generics.where_clause = where_clause;
1045
1046 self.expect_semi()?;
1047
1048 Ok((
1049 ident,
1050 ItemKind::TyAlias(Box::new(TyAlias {
1051 defaultness,
1052 generics,
1053 where_clauses,
1054 bounds,
1055 ty,
1056 })),
1057 ))
1058 }
1059
1060 fn parse_use_tree(&mut self) -> PResult<'a, UseTree> {
1070 let lo = self.token.span;
1071
1072 let mut prefix =
1073 ast::Path { segments: ThinVec::new(), span: lo.shrink_to_lo(), tokens: None };
1074 let kind =
1075 if self.check(exp!(OpenBrace)) || self.check(exp!(Star)) || self.is_import_coupler() {
1076 let mod_sep_ctxt = self.token.span.ctxt();
1078 if self.eat_path_sep() {
1079 prefix
1080 .segments
1081 .push(PathSegment::path_root(lo.shrink_to_lo().with_ctxt(mod_sep_ctxt)));
1082 }
1083
1084 self.parse_use_tree_glob_or_nested()?
1085 } else {
1086 prefix = self.parse_path(PathStyle::Mod)?;
1088
1089 if self.eat_path_sep() {
1090 self.parse_use_tree_glob_or_nested()?
1091 } else {
1092 while self.eat_noexpect(&token::Colon) {
1094 self.dcx()
1095 .emit_err(errors::SingleColonImportPath { span: self.prev_token.span });
1096
1097 self.parse_path_segments(&mut prefix.segments, PathStyle::Mod, None)?;
1099 prefix.span = lo.to(self.prev_token.span);
1100 }
1101
1102 UseTreeKind::Simple(self.parse_rename()?)
1103 }
1104 };
1105
1106 Ok(UseTree { prefix, kind, span: lo.to(self.prev_token.span) })
1107 }
1108
1109 fn parse_use_tree_glob_or_nested(&mut self) -> PResult<'a, UseTreeKind> {
1111 Ok(if self.eat(exp!(Star)) {
1112 UseTreeKind::Glob
1113 } else {
1114 let lo = self.token.span;
1115 UseTreeKind::Nested {
1116 items: self.parse_use_tree_list()?,
1117 span: lo.to(self.prev_token.span),
1118 }
1119 })
1120 }
1121
1122 fn parse_use_tree_list(&mut self) -> PResult<'a, ThinVec<(UseTree, ast::NodeId)>> {
1128 self.parse_delim_comma_seq(exp!(OpenBrace), exp!(CloseBrace), |p| {
1129 p.recover_vcs_conflict_marker();
1130 Ok((p.parse_use_tree()?, DUMMY_NODE_ID))
1131 })
1132 .map(|(r, _)| r)
1133 }
1134
1135 fn parse_rename(&mut self) -> PResult<'a, Option<Ident>> {
1136 if self.eat_keyword(exp!(As)) {
1137 self.parse_ident_or_underscore().map(Some)
1138 } else {
1139 Ok(None)
1140 }
1141 }
1142
1143 fn parse_ident_or_underscore(&mut self) -> PResult<'a, Ident> {
1144 match self.token.ident() {
1145 Some((ident @ Ident { name: kw::Underscore, .. }, IdentIsRaw::No)) => {
1146 self.bump();
1147 Ok(ident)
1148 }
1149 _ => self.parse_ident(),
1150 }
1151 }
1152
1153 fn parse_item_extern_crate(&mut self) -> PResult<'a, ItemInfo> {
1162 let orig_name = self.parse_crate_name_with_dashes()?;
1164 let (item_name, orig_name) = if let Some(rename) = self.parse_rename()? {
1165 (rename, Some(orig_name.name))
1166 } else {
1167 (orig_name, None)
1168 };
1169 self.expect_semi()?;
1170 Ok((item_name, ItemKind::ExternCrate(orig_name)))
1171 }
1172
1173 fn parse_crate_name_with_dashes(&mut self) -> PResult<'a, Ident> {
1174 let ident = if self.token.is_keyword(kw::SelfLower) {
1175 self.parse_path_segment_ident()
1176 } else {
1177 self.parse_ident()
1178 }?;
1179
1180 let dash = exp!(Minus);
1181 if self.token != *dash.tok {
1182 return Ok(ident);
1183 }
1184
1185 let mut dashes = vec![];
1187 let mut idents = vec![];
1188 while self.eat(dash) {
1189 dashes.push(self.prev_token.span);
1190 idents.push(self.parse_ident()?);
1191 }
1192
1193 let fixed_name_sp = ident.span.to(idents.last().unwrap().span);
1194 let mut fixed_name = ident.name.to_string();
1195 for part in idents {
1196 write!(fixed_name, "_{}", part.name).unwrap();
1197 }
1198
1199 self.dcx().emit_err(errors::ExternCrateNameWithDashes {
1200 span: fixed_name_sp,
1201 sugg: errors::ExternCrateNameWithDashesSugg { dashes },
1202 });
1203
1204 Ok(Ident::from_str_and_span(&fixed_name, fixed_name_sp))
1205 }
1206
1207 fn parse_item_foreign_mod(
1218 &mut self,
1219 attrs: &mut AttrVec,
1220 mut safety: Safety,
1221 ) -> PResult<'a, ItemInfo> {
1222 let extern_span = self.prev_token.uninterpolated_span();
1223 let abi = self.parse_abi(); if safety == Safety::Default
1226 && self.token.is_keyword(kw::Unsafe)
1227 && self.look_ahead(1, |t| *t == token::OpenDelim(Delimiter::Brace))
1228 {
1229 self.expect(exp!(OpenBrace)).unwrap_err().emit();
1230 safety = Safety::Unsafe(self.token.span);
1231 let _ = self.eat_keyword(exp!(Unsafe));
1232 }
1233 let module = ast::ForeignMod {
1234 extern_span,
1235 safety,
1236 abi,
1237 items: self.parse_item_list(attrs, |p| p.parse_foreign_item(ForceCollect::No))?,
1238 };
1239 Ok((Ident::empty(), ItemKind::ForeignMod(module)))
1240 }
1241
1242 pub fn parse_foreign_item(
1244 &mut self,
1245 force_collect: ForceCollect,
1246 ) -> PResult<'a, Option<Option<P<ForeignItem>>>> {
1247 let fn_parse_mode = FnParseMode { req_name: |_| true, req_body: false };
1248 Ok(self.parse_item_(fn_parse_mode, force_collect)?.map(
1249 |Item { attrs, id, span, vis, ident, kind, tokens }| {
1250 let kind = match ForeignItemKind::try_from(kind) {
1251 Ok(kind) => kind,
1252 Err(kind) => match kind {
1253 ItemKind::Const(box ConstItem { ty, expr, .. }) => {
1254 let const_span = Some(span.with_hi(ident.span.lo()))
1255 .filter(|span| span.can_be_used_for_suggestions());
1256 self.dcx().emit_err(errors::ExternItemCannotBeConst {
1257 ident_span: ident.span,
1258 const_span,
1259 });
1260 ForeignItemKind::Static(Box::new(StaticItem {
1261 ty,
1262 mutability: Mutability::Not,
1263 expr,
1264 safety: Safety::Default,
1265 define_opaque: None,
1266 }))
1267 }
1268 _ => return self.error_bad_item_kind(span, &kind, "`extern` blocks"),
1269 },
1270 };
1271 Some(P(Item { attrs, id, span, vis, ident, kind, tokens }))
1272 },
1273 ))
1274 }
1275
1276 fn error_bad_item_kind<T>(&self, span: Span, kind: &ItemKind, ctx: &'static str) -> Option<T> {
1277 let span = self.psess.source_map().guess_head_span(span);
1279 let descr = kind.descr();
1280 let help = match kind {
1281 ItemKind::DelegationMac(deleg) if deleg.suffixes.is_none() => false,
1282 _ => true,
1283 };
1284 self.dcx().emit_err(errors::BadItemKind { span, descr, ctx, help });
1285 None
1286 }
1287
1288 fn is_use_closure(&self) -> bool {
1289 if self.token.is_keyword(kw::Use) {
1290 self.look_ahead(1, |token| {
1292 let dist =
1294 if token.is_keyword(kw::Move) || token.is_keyword(kw::Async) { 2 } else { 1 };
1295
1296 self.look_ahead(dist, |token| matches!(token.kind, token::Or | token::OrOr))
1297 })
1298 } else {
1299 false
1300 }
1301 }
1302
1303 fn is_unsafe_foreign_mod(&self) -> bool {
1304 self.token.is_keyword(kw::Unsafe)
1305 && self.is_keyword_ahead(1, &[kw::Extern])
1306 && self.look_ahead(
1307 2 + self.look_ahead(2, |t| t.can_begin_string_literal() as usize),
1308 |t| *t == token::OpenDelim(Delimiter::Brace),
1309 )
1310 }
1311
1312 fn is_static_global(&mut self) -> bool {
1313 if self.check_keyword(exp!(Static)) {
1314 !self.look_ahead(1, |token| {
1316 if token.is_keyword(kw::Move) || token.is_keyword(kw::Use) {
1317 return true;
1318 }
1319 matches!(token.kind, token::Or | token::OrOr)
1320 })
1321 } else {
1322 (self.check_keyword(exp!(Unsafe)) || self.check_keyword(exp!(Safe)))
1324 && self.look_ahead(1, |t| t.is_keyword(kw::Static))
1325 }
1326 }
1327
1328 fn recover_const_mut(&mut self, const_span: Span) {
1330 if self.eat_keyword(exp!(Mut)) {
1331 let span = self.prev_token.span;
1332 self.dcx()
1333 .emit_err(errors::ConstGlobalCannotBeMutable { ident_span: span, const_span });
1334 } else if self.eat_keyword(exp!(Let)) {
1335 let span = self.prev_token.span;
1336 self.dcx().emit_err(errors::ConstLetMutuallyExclusive { span: const_span.to(span) });
1337 }
1338 }
1339
1340 fn recover_const_impl(
1342 &mut self,
1343 const_span: Span,
1344 attrs: &mut AttrVec,
1345 defaultness: Defaultness,
1346 ) -> PResult<'a, ItemInfo> {
1347 let impl_span = self.token.span;
1348 let err = self.expected_ident_found_err();
1349
1350 let mut impl_info = match self.parse_item_impl(attrs, defaultness) {
1352 Ok(impl_info) => impl_info,
1353 Err(recovery_error) => {
1354 recovery_error.cancel();
1356 return Err(err);
1357 }
1358 };
1359
1360 match &mut impl_info.1 {
1361 ItemKind::Impl(box Impl { of_trait: Some(trai), constness, .. }) => {
1362 *constness = Const::Yes(const_span);
1363
1364 let before_trait = trai.path.span.shrink_to_lo();
1365 let const_up_to_impl = const_span.with_hi(impl_span.lo());
1366 err.with_multipart_suggestion(
1367 "you might have meant to write a const trait impl",
1368 vec![(const_up_to_impl, "".to_owned()), (before_trait, "const ".to_owned())],
1369 Applicability::MaybeIncorrect,
1370 )
1371 .emit();
1372 }
1373 ItemKind::Impl { .. } => return Err(err),
1374 _ => unreachable!(),
1375 }
1376
1377 Ok(impl_info)
1378 }
1379
1380 fn parse_static_item(
1386 &mut self,
1387 safety: Safety,
1388 mutability: Mutability,
1389 ) -> PResult<'a, (Ident, StaticItem)> {
1390 let ident = self.parse_ident()?;
1391
1392 if self.token == TokenKind::Lt && self.may_recover() {
1393 let generics = self.parse_generics()?;
1394 self.dcx().emit_err(errors::StaticWithGenerics { span: generics.span });
1395 }
1396
1397 let ty = match (self.eat(exp!(Colon)), self.check(exp!(Eq)) | self.check(exp!(Semi))) {
1400 (true, false) => self.parse_ty()?,
1401 (colon, _) => self.recover_missing_global_item_type(colon, Some(mutability)),
1403 };
1404
1405 let expr = if self.eat(exp!(Eq)) { Some(self.parse_expr()?) } else { None };
1406
1407 self.expect_semi()?;
1408
1409 Ok((ident, StaticItem { ty, safety, mutability, expr, define_opaque: None }))
1410 }
1411
1412 fn parse_const_item(&mut self) -> PResult<'a, (Ident, Generics, P<Ty>, Option<P<ast::Expr>>)> {
1418 let ident = self.parse_ident_or_underscore()?;
1419
1420 let mut generics = self.parse_generics()?;
1421
1422 if !generics.span.is_empty() {
1425 self.psess.gated_spans.gate(sym::generic_const_items, generics.span);
1426 }
1427
1428 let ty = match (
1431 self.eat(exp!(Colon)),
1432 self.check(exp!(Eq)) | self.check(exp!(Semi)) | self.check_keyword(exp!(Where)),
1433 ) {
1434 (true, false) => self.parse_ty()?,
1435 (colon, _) => self.recover_missing_global_item_type(colon, None),
1437 };
1438
1439 let before_where_clause =
1442 if self.may_recover() { self.parse_where_clause()? } else { WhereClause::default() };
1443
1444 let expr = if self.eat(exp!(Eq)) { Some(self.parse_expr()?) } else { None };
1445
1446 let after_where_clause = self.parse_where_clause()?;
1447
1448 if before_where_clause.has_where_token
1452 && let Some(expr) = &expr
1453 {
1454 self.dcx().emit_err(errors::WhereClauseBeforeConstBody {
1455 span: before_where_clause.span,
1456 name: ident.span,
1457 body: expr.span,
1458 sugg: if !after_where_clause.has_where_token {
1459 self.psess.source_map().span_to_snippet(expr.span).ok().map(|body| {
1460 errors::WhereClauseBeforeConstBodySugg {
1461 left: before_where_clause.span.shrink_to_lo(),
1462 snippet: body,
1463 right: before_where_clause.span.shrink_to_hi().to(expr.span),
1464 }
1465 })
1466 } else {
1467 None
1470 },
1471 });
1472 }
1473
1474 let mut predicates = before_where_clause.predicates;
1481 predicates.extend(after_where_clause.predicates);
1482 let where_clause = WhereClause {
1483 has_where_token: before_where_clause.has_where_token
1484 || after_where_clause.has_where_token,
1485 predicates,
1486 span: if after_where_clause.has_where_token {
1487 after_where_clause.span
1488 } else {
1489 before_where_clause.span
1490 },
1491 };
1492
1493 if where_clause.has_where_token {
1494 self.psess.gated_spans.gate(sym::generic_const_items, where_clause.span);
1495 }
1496
1497 generics.where_clause = where_clause;
1498
1499 self.expect_semi()?;
1500
1501 Ok((ident, generics, ty, expr))
1502 }
1503
1504 fn recover_missing_global_item_type(
1507 &mut self,
1508 colon_present: bool,
1509 m: Option<Mutability>,
1510 ) -> P<Ty> {
1511 let kind = match m {
1514 Some(Mutability::Mut) => "static mut",
1515 Some(Mutability::Not) => "static",
1516 None => "const",
1517 };
1518
1519 let colon = match colon_present {
1520 true => "",
1521 false => ":",
1522 };
1523
1524 let span = self.prev_token.span.shrink_to_hi();
1525 let err = self.dcx().create_err(errors::MissingConstType { span, colon, kind });
1526 err.stash(span, StashKey::ItemNoType);
1527
1528 P(Ty { kind: TyKind::Infer, span, id: ast::DUMMY_NODE_ID, tokens: None })
1531 }
1532
1533 fn parse_item_enum(&mut self) -> PResult<'a, ItemInfo> {
1535 if self.token.is_keyword(kw::Struct) {
1536 let span = self.prev_token.span.to(self.token.span);
1537 let err = errors::EnumStructMutuallyExclusive { span };
1538 if self.look_ahead(1, |t| t.is_ident()) {
1539 self.bump();
1540 self.dcx().emit_err(err);
1541 } else {
1542 return Err(self.dcx().create_err(err));
1543 }
1544 }
1545
1546 let prev_span = self.prev_token.span;
1547 let id = self.parse_ident()?;
1548 let mut generics = self.parse_generics()?;
1549 generics.where_clause = self.parse_where_clause()?;
1550
1551 let (variants, _) = if self.token == TokenKind::Semi {
1553 self.dcx().emit_err(errors::UseEmptyBlockNotSemi { span: self.token.span });
1554 self.bump();
1555 (thin_vec![], Trailing::No)
1556 } else {
1557 self.parse_delim_comma_seq(exp!(OpenBrace), exp!(CloseBrace), |p| {
1558 p.parse_enum_variant(id.span)
1559 })
1560 .map_err(|mut err| {
1561 err.span_label(id.span, "while parsing this enum");
1562 if self.token == token::Colon {
1563 let snapshot = self.create_snapshot_for_diagnostic();
1564 self.bump();
1565 match self.parse_ty() {
1566 Ok(_) => {
1567 err.span_suggestion_verbose(
1568 prev_span,
1569 "perhaps you meant to use `struct` here",
1570 "struct",
1571 Applicability::MaybeIncorrect,
1572 );
1573 }
1574 Err(e) => {
1575 e.cancel();
1576 }
1577 }
1578 self.restore_snapshot(snapshot);
1579 }
1580 self.eat_to_tokens(&[exp!(CloseBrace)]);
1581 self.bump(); err
1583 })?
1584 };
1585
1586 let enum_definition = EnumDef { variants: variants.into_iter().flatten().collect() };
1587 Ok((id, ItemKind::Enum(enum_definition, generics)))
1588 }
1589
1590 fn parse_enum_variant(&mut self, span: Span) -> PResult<'a, Option<Variant>> {
1591 self.recover_vcs_conflict_marker();
1592 let variant_attrs = self.parse_outer_attributes()?;
1593 self.recover_vcs_conflict_marker();
1594 let help = "enum variants can be `Variant`, `Variant = <integer>`, \
1595 `Variant(Type, ..., TypeN)` or `Variant { fields: Types }`";
1596 self.collect_tokens(None, variant_attrs, ForceCollect::No, |this, variant_attrs| {
1597 let vlo = this.token.span;
1598
1599 let vis = this.parse_visibility(FollowedByType::No)?;
1600 if !this.recover_nested_adt_item(kw::Enum)? {
1601 return Ok((None, Trailing::No, UsePreAttrPos::No));
1602 }
1603 let ident = this.parse_field_ident("enum", vlo)?;
1604
1605 if this.token == token::Bang {
1606 if let Err(err) = this.unexpected() {
1607 err.with_note(fluent::parse_macro_expands_to_enum_variant).emit();
1608 }
1609
1610 this.bump();
1611 this.parse_delim_args()?;
1612
1613 return Ok((None, Trailing::from(this.token == token::Comma), UsePreAttrPos::No));
1614 }
1615
1616 let struct_def = if this.check(exp!(OpenBrace)) {
1617 let (fields, recovered) =
1619 match this.parse_record_struct_body("struct", ident.span, false) {
1620 Ok((fields, recovered)) => (fields, recovered),
1621 Err(mut err) => {
1622 if this.token == token::Colon {
1623 return Err(err);
1625 }
1626 this.eat_to_tokens(&[exp!(CloseBrace)]);
1627 this.bump(); err.span_label(span, "while parsing this enum");
1629 err.help(help);
1630 let guar = err.emit();
1631 (thin_vec![], Recovered::Yes(guar))
1632 }
1633 };
1634 VariantData::Struct { fields, recovered }
1635 } else if this.check(exp!(OpenParen)) {
1636 let body = match this.parse_tuple_struct_body() {
1637 Ok(body) => body,
1638 Err(mut err) => {
1639 if this.token == token::Colon {
1640 return Err(err);
1642 }
1643 this.eat_to_tokens(&[exp!(CloseParen)]);
1644 this.bump(); err.span_label(span, "while parsing this enum");
1646 err.help(help);
1647 err.emit();
1648 thin_vec![]
1649 }
1650 };
1651 VariantData::Tuple(body, DUMMY_NODE_ID)
1652 } else {
1653 VariantData::Unit(DUMMY_NODE_ID)
1654 };
1655
1656 let disr_expr =
1657 if this.eat(exp!(Eq)) { Some(this.parse_expr_anon_const()?) } else { None };
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, ItemInfo> {
1680 let class_name = 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(class_name, 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 class_name.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::OpenDelim(Delimiter::Brace) {
1725 let (fields, recovered) = self.parse_record_struct_body(
1726 "struct",
1727 class_name.span,
1728 generics.where_clause.has_where_token,
1729 )?;
1730 VariantData::Struct { fields, recovered }
1731 } else if self.token == token::OpenDelim(Delimiter::Parenthesis) {
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 =
1739 errors::UnexpectedTokenAfterStructName::new(self.token.span, self.token.clone());
1740 return Err(self.dcx().create_err(err));
1741 };
1742
1743 Ok((class_name, ItemKind::Struct(vdata, generics)))
1744 }
1745
1746 fn parse_item_union(&mut self) -> PResult<'a, ItemInfo> {
1748 let class_name = self.parse_ident()?;
1749
1750 let mut generics = self.parse_generics()?;
1751
1752 let vdata = if self.token.is_keyword(kw::Where) {
1753 generics.where_clause = self.parse_where_clause()?;
1754 let (fields, recovered) = self.parse_record_struct_body(
1755 "union",
1756 class_name.span,
1757 generics.where_clause.has_where_token,
1758 )?;
1759 VariantData::Struct { fields, recovered }
1760 } else if self.token == token::OpenDelim(Delimiter::Brace) {
1761 let (fields, recovered) = self.parse_record_struct_body(
1762 "union",
1763 class_name.span,
1764 generics.where_clause.has_where_token,
1765 )?;
1766 VariantData::Struct { fields, recovered }
1767 } else {
1768 let token_str = super::token_descr(&self.token);
1769 let msg = format!("expected `where` or `{{` after union name, found {token_str}");
1770 let mut err = self.dcx().struct_span_err(self.token.span, msg);
1771 err.span_label(self.token.span, "expected `where` or `{` after union name");
1772 return Err(err);
1773 };
1774
1775 Ok((class_name, ItemKind::Union(vdata, generics)))
1776 }
1777
1778 pub(crate) fn parse_record_struct_body(
1783 &mut self,
1784 adt_ty: &str,
1785 ident_span: Span,
1786 parsed_where: bool,
1787 ) -> PResult<'a, (ThinVec<FieldDef>, Recovered)> {
1788 let mut fields = ThinVec::new();
1789 let mut recovered = Recovered::No;
1790 if self.eat(exp!(OpenBrace)) {
1791 while self.token != token::CloseDelim(Delimiter::Brace) {
1792 match self.parse_field_def(adt_ty) {
1793 Ok(field) => {
1794 fields.push(field);
1795 }
1796 Err(mut err) => {
1797 self.consume_block(
1798 exp!(OpenBrace),
1799 exp!(CloseBrace),
1800 ConsumeClosingDelim::No,
1801 );
1802 err.span_label(ident_span, format!("while parsing this {adt_ty}"));
1803 let guar = err.emit();
1804 recovered = Recovered::Yes(guar);
1805 break;
1806 }
1807 }
1808 }
1809 self.expect(exp!(CloseBrace))?;
1810 } else {
1811 let token_str = super::token_descr(&self.token);
1812 let where_str = if parsed_where { "" } else { "`where`, or " };
1813 let msg = format!("expected {where_str}`{{` after struct name, found {token_str}");
1814 let mut err = self.dcx().struct_span_err(self.token.span, msg);
1815 err.span_label(self.token.span, format!("expected {where_str}`{{` after struct name",));
1816 return Err(err);
1817 }
1818
1819 Ok((fields, recovered))
1820 }
1821
1822 fn parse_unsafe_field(&mut self) -> Safety {
1823 if self.eat_keyword(exp!(Unsafe)) {
1825 let span = self.prev_token.span;
1826 self.psess.gated_spans.gate(sym::unsafe_fields, span);
1827 Safety::Unsafe(span)
1828 } else {
1829 Safety::Default
1830 }
1831 }
1832
1833 pub(super) fn parse_tuple_struct_body(&mut self) -> PResult<'a, ThinVec<FieldDef>> {
1834 self.parse_paren_comma_seq(|p| {
1837 let attrs = p.parse_outer_attributes()?;
1838 p.collect_tokens(None, attrs, ForceCollect::No, |p, attrs| {
1839 let mut snapshot = None;
1840 if p.is_vcs_conflict_marker(&TokenKind::Shl, &TokenKind::Lt) {
1841 snapshot = Some(p.create_snapshot_for_diagnostic());
1845 }
1846 let lo = p.token.span;
1847 let vis = match p.parse_visibility(FollowedByType::Yes) {
1848 Ok(vis) => vis,
1849 Err(err) => {
1850 if let Some(ref mut snapshot) = snapshot {
1851 snapshot.recover_vcs_conflict_marker();
1852 }
1853 return Err(err);
1854 }
1855 };
1856 let ty = match p.parse_ty() {
1859 Ok(ty) => ty,
1860 Err(err) => {
1861 if let Some(ref mut snapshot) = snapshot {
1862 snapshot.recover_vcs_conflict_marker();
1863 }
1864 return Err(err);
1865 }
1866 };
1867 let mut default = None;
1868 if p.token == token::Eq {
1869 let mut snapshot = p.create_snapshot_for_diagnostic();
1870 snapshot.bump();
1871 match snapshot.parse_expr_anon_const() {
1872 Ok(const_expr) => {
1873 let sp = ty.span.shrink_to_hi().to(const_expr.value.span);
1874 p.psess.gated_spans.gate(sym::default_field_values, sp);
1875 p.restore_snapshot(snapshot);
1876 default = Some(const_expr);
1877 }
1878 Err(err) => {
1879 err.cancel();
1880 }
1881 }
1882 }
1883
1884 Ok((
1885 FieldDef {
1886 span: lo.to(ty.span),
1887 vis,
1888 safety: Safety::Default,
1889 ident: None,
1890 id: DUMMY_NODE_ID,
1891 ty,
1892 default,
1893 attrs,
1894 is_placeholder: false,
1895 },
1896 Trailing::from(p.token == token::Comma),
1897 UsePreAttrPos::No,
1898 ))
1899 })
1900 })
1901 .map(|(r, _)| r)
1902 }
1903
1904 fn parse_field_def(&mut self, adt_ty: &str) -> PResult<'a, FieldDef> {
1906 self.recover_vcs_conflict_marker();
1907 let attrs = self.parse_outer_attributes()?;
1908 self.recover_vcs_conflict_marker();
1909 self.collect_tokens(None, attrs, ForceCollect::No, |this, attrs| {
1910 let lo = this.token.span;
1911 let vis = this.parse_visibility(FollowedByType::No)?;
1912 let safety = this.parse_unsafe_field();
1913 this.parse_single_struct_field(adt_ty, lo, vis, safety, attrs)
1914 .map(|field| (field, Trailing::No, UsePreAttrPos::No))
1915 })
1916 }
1917
1918 fn parse_single_struct_field(
1920 &mut self,
1921 adt_ty: &str,
1922 lo: Span,
1923 vis: Visibility,
1924 safety: Safety,
1925 attrs: AttrVec,
1926 ) -> PResult<'a, FieldDef> {
1927 let mut seen_comma: bool = false;
1928 let a_var = self.parse_name_and_ty(adt_ty, lo, vis, safety, attrs)?;
1929 if self.token == token::Comma {
1930 seen_comma = true;
1931 }
1932 if self.eat(exp!(Semi)) {
1933 let sp = self.prev_token.span;
1934 let mut err =
1935 self.dcx().struct_span_err(sp, format!("{adt_ty} fields are separated by `,`"));
1936 err.span_suggestion_short(
1937 sp,
1938 "replace `;` with `,`",
1939 ",",
1940 Applicability::MachineApplicable,
1941 );
1942 return Err(err);
1943 }
1944 match self.token.kind {
1945 token::Comma => {
1946 self.bump();
1947 }
1948 token::CloseDelim(Delimiter::Brace) => {}
1949 token::DocComment(..) => {
1950 let previous_span = self.prev_token.span;
1951 let mut err = errors::DocCommentDoesNotDocumentAnything {
1952 span: self.token.span,
1953 missing_comma: None,
1954 };
1955 self.bump(); let comma_after_doc_seen = self.eat(exp!(Comma));
1957 if !seen_comma && comma_after_doc_seen {
1960 seen_comma = true;
1961 }
1962 if comma_after_doc_seen || self.token == token::CloseDelim(Delimiter::Brace) {
1963 self.dcx().emit_err(err);
1964 } else {
1965 if !seen_comma {
1966 let sp = previous_span.shrink_to_hi();
1967 err.missing_comma = Some(sp);
1968 }
1969 return Err(self.dcx().create_err(err));
1970 }
1971 }
1972 _ => {
1973 let sp = self.prev_token.span.shrink_to_hi();
1974 let msg =
1975 format!("expected `,`, or `}}`, found {}", super::token_descr(&self.token));
1976
1977 if let TyKind::Path(_, Path { segments, .. }) = &a_var.ty.kind {
1979 if let Some(last_segment) = segments.last() {
1980 let guar = self.check_trailing_angle_brackets(
1981 last_segment,
1982 &[exp!(Comma), exp!(CloseBrace)],
1983 );
1984 if let Some(_guar) = guar {
1985 let _ = self.eat(exp!(Comma));
1988
1989 return Ok(a_var);
1992 }
1993 }
1994 }
1995
1996 let mut err = self.dcx().struct_span_err(sp, msg);
1997
1998 if self.token.is_ident()
1999 || (self.token == TokenKind::Pound
2000 && (self.look_ahead(1, |t| t == &token::OpenDelim(Delimiter::Bracket))))
2001 {
2002 err.span_suggestion(
2005 sp,
2006 "try adding a comma",
2007 ",",
2008 Applicability::MachineApplicable,
2009 );
2010 err.emit();
2011 } else {
2012 return Err(err);
2013 }
2014 }
2015 }
2016 Ok(a_var)
2017 }
2018
2019 fn expect_field_ty_separator(&mut self) -> PResult<'a, ()> {
2020 if let Err(err) = self.expect(exp!(Colon)) {
2021 let sm = self.psess.source_map();
2022 let eq_typo = self.token == token::Eq && self.look_ahead(1, |t| t.is_path_start());
2023 let semi_typo = self.token == token::Semi
2024 && self.look_ahead(1, |t| {
2025 t.is_path_start()
2026 && match (sm.lookup_line(self.token.span.hi()), sm.lookup_line(t.span.lo())) {
2029 (Ok(l), Ok(r)) => l.line == r.line,
2030 _ => true,
2031 }
2032 });
2033 if eq_typo || semi_typo {
2034 self.bump();
2035 err.with_span_suggestion_short(
2037 self.prev_token.span,
2038 "field names and their types are separated with `:`",
2039 ":",
2040 Applicability::MachineApplicable,
2041 )
2042 .emit();
2043 } else {
2044 return Err(err);
2045 }
2046 }
2047 Ok(())
2048 }
2049
2050 fn parse_name_and_ty(
2052 &mut self,
2053 adt_ty: &str,
2054 lo: Span,
2055 vis: Visibility,
2056 safety: Safety,
2057 attrs: AttrVec,
2058 ) -> PResult<'a, FieldDef> {
2059 let name = self.parse_field_ident(adt_ty, lo)?;
2060 if self.token == token::Bang {
2061 if let Err(mut err) = self.unexpected() {
2062 err.subdiagnostic(MacroExpandsToAdtField { adt_ty });
2064 return Err(err);
2065 }
2066 }
2067 self.expect_field_ty_separator()?;
2068 let ty = self.parse_ty()?;
2069 if self.token == token::Colon && self.look_ahead(1, |t| *t != token::Colon) {
2070 self.dcx()
2071 .struct_span_err(self.token.span, "found single colon in a struct field type path")
2072 .with_span_suggestion_verbose(
2073 self.token.span,
2074 "write a path separator here",
2075 "::",
2076 Applicability::MaybeIncorrect,
2077 )
2078 .emit();
2079 }
2080 let default = if self.token == token::Eq {
2081 self.bump();
2082 let const_expr = self.parse_expr_anon_const()?;
2083 let sp = ty.span.shrink_to_hi().to(const_expr.value.span);
2084 self.psess.gated_spans.gate(sym::default_field_values, sp);
2085 Some(const_expr)
2086 } else {
2087 None
2088 };
2089 Ok(FieldDef {
2090 span: lo.to(self.prev_token.span),
2091 ident: Some(name),
2092 vis,
2093 safety,
2094 id: DUMMY_NODE_ID,
2095 ty,
2096 default,
2097 attrs,
2098 is_placeholder: false,
2099 })
2100 }
2101
2102 fn parse_field_ident(&mut self, adt_ty: &str, lo: Span) -> PResult<'a, Ident> {
2105 let (ident, is_raw) = self.ident_or_err(true)?;
2106 if matches!(is_raw, IdentIsRaw::No) && ident.is_reserved() {
2107 let snapshot = self.create_snapshot_for_diagnostic();
2108 let err = if self.check_fn_front_matter(false, Case::Sensitive) {
2109 let inherited_vis =
2110 Visibility { span: DUMMY_SP, kind: VisibilityKind::Inherited, tokens: None };
2111 let fn_parse_mode = FnParseMode { req_name: |_| true, req_body: true };
2113 match self.parse_fn(
2114 &mut AttrVec::new(),
2115 fn_parse_mode,
2116 lo,
2117 &inherited_vis,
2118 Case::Insensitive,
2119 ) {
2120 Ok(_) => {
2121 self.dcx().struct_span_err(
2122 lo.to(self.prev_token.span),
2123 format!("functions are not allowed in {adt_ty} definitions"),
2124 )
2125 .with_help(
2126 "unlike in C++, Java, and C#, functions are declared in `impl` blocks",
2127 )
2128 .with_help("see https://doc.rust-lang.org/book/ch05-03-method-syntax.html for more information")
2129 }
2130 Err(err) => {
2131 err.cancel();
2132 self.restore_snapshot(snapshot);
2133 self.expected_ident_found_err()
2134 }
2135 }
2136 } else if self.eat_keyword(exp!(Struct)) {
2137 match self.parse_item_struct() {
2138 Ok((ident, _)) => self
2139 .dcx()
2140 .struct_span_err(
2141 lo.with_hi(ident.span.hi()),
2142 format!("structs are not allowed in {adt_ty} definitions"),
2143 )
2144 .with_help(
2145 "consider creating a new `struct` definition instead of nesting",
2146 ),
2147 Err(err) => {
2148 err.cancel();
2149 self.restore_snapshot(snapshot);
2150 self.expected_ident_found_err()
2151 }
2152 }
2153 } else {
2154 let mut err = self.expected_ident_found_err();
2155 if self.eat_keyword_noexpect(kw::Let)
2156 && let removal_span = self.prev_token.span.until(self.token.span)
2157 && let Ok(ident) = self
2158 .parse_ident_common(false)
2159 .map_err(|err| err.cancel())
2161 && self.token == TokenKind::Colon
2162 {
2163 err.span_suggestion(
2164 removal_span,
2165 "remove this `let` keyword",
2166 String::new(),
2167 Applicability::MachineApplicable,
2168 );
2169 err.note("the `let` keyword is not allowed in `struct` fields");
2170 err.note("see <https://doc.rust-lang.org/book/ch05-01-defining-structs.html> for more information");
2171 err.emit();
2172 return Ok(ident);
2173 } else {
2174 self.restore_snapshot(snapshot);
2175 }
2176 err
2177 };
2178 return Err(err);
2179 }
2180 self.bump();
2181 Ok(ident)
2182 }
2183
2184 fn parse_item_decl_macro(&mut self, lo: Span) -> PResult<'a, ItemInfo> {
2192 let ident = self.parse_ident()?;
2193 let body = if self.check(exp!(OpenBrace)) {
2194 self.parse_delim_args()? } else if self.check(exp!(OpenParen)) {
2196 let params = self.parse_token_tree(); let pspan = params.span();
2198 if !self.check(exp!(OpenBrace)) {
2199 self.unexpected()?;
2200 }
2201 let body = self.parse_token_tree(); let bspan = body.span();
2204 let arrow = TokenTree::token_alone(token::FatArrow, pspan.between(bspan)); let tokens = TokenStream::new(vec![params, arrow, body]);
2206 let dspan = DelimSpan::from_pair(pspan.shrink_to_lo(), bspan.shrink_to_hi());
2207 P(DelimArgs { dspan, delim: Delimiter::Brace, tokens })
2208 } else {
2209 self.unexpected_any()?
2210 };
2211
2212 self.psess.gated_spans.gate(sym::decl_macro, lo.to(self.prev_token.span));
2213 Ok((ident, ItemKind::MacroDef(ast::MacroDef { body, macro_rules: false })))
2214 }
2215
2216 fn is_macro_rules_item(&mut self) -> IsMacroRulesItem {
2218 if self.check_keyword(exp!(MacroRules)) {
2219 let macro_rules_span = self.token.span;
2220
2221 if self.look_ahead(1, |t| *t == token::Bang) && self.look_ahead(2, |t| t.is_ident()) {
2222 return IsMacroRulesItem::Yes { has_bang: true };
2223 } else if self.look_ahead(1, |t| (t.is_ident())) {
2224 self.dcx().emit_err(errors::MacroRulesMissingBang {
2226 span: macro_rules_span,
2227 hi: macro_rules_span.shrink_to_hi(),
2228 });
2229
2230 return IsMacroRulesItem::Yes { has_bang: false };
2231 }
2232 }
2233
2234 IsMacroRulesItem::No
2235 }
2236
2237 fn parse_item_macro_rules(
2239 &mut self,
2240 vis: &Visibility,
2241 has_bang: bool,
2242 ) -> PResult<'a, ItemInfo> {
2243 self.expect_keyword(exp!(MacroRules))?; if has_bang {
2246 self.expect(exp!(Bang))?; }
2248 let ident = self.parse_ident()?;
2249
2250 if self.eat(exp!(Bang)) {
2251 let span = self.prev_token.span;
2253 self.dcx().emit_err(errors::MacroNameRemoveBang { span });
2254 }
2255
2256 let body = self.parse_delim_args()?;
2257 self.eat_semi_for_macro_if_needed(&body);
2258 self.complain_if_pub_macro(vis, true);
2259
2260 Ok((ident, ItemKind::MacroDef(ast::MacroDef { body, macro_rules: true })))
2261 }
2262
2263 fn complain_if_pub_macro(&self, vis: &Visibility, macro_rules: bool) {
2266 if let VisibilityKind::Inherited = vis.kind {
2267 return;
2268 }
2269
2270 let vstr = pprust::vis_to_string(vis);
2271 let vstr = vstr.trim_end();
2272 if macro_rules {
2273 self.dcx().emit_err(errors::MacroRulesVisibility { span: vis.span, vis: vstr });
2274 } else {
2275 self.dcx().emit_err(errors::MacroInvocationVisibility { span: vis.span, vis: vstr });
2276 }
2277 }
2278
2279 fn eat_semi_for_macro_if_needed(&mut self, args: &DelimArgs) {
2280 if args.need_semicolon() && !self.eat(exp!(Semi)) {
2281 self.report_invalid_macro_expansion_item(args);
2282 }
2283 }
2284
2285 fn report_invalid_macro_expansion_item(&self, args: &DelimArgs) {
2286 let span = args.dspan.entire();
2287 let mut err = self.dcx().struct_span_err(
2288 span,
2289 "macros that expand to items must be delimited with braces or followed by a semicolon",
2290 );
2291 if !span.from_expansion() {
2294 let DelimSpan { open, close } = args.dspan;
2295 err.multipart_suggestion(
2296 "change the delimiters to curly braces",
2297 vec![(open, "{".to_string()), (close, '}'.to_string())],
2298 Applicability::MaybeIncorrect,
2299 );
2300 err.span_suggestion(
2301 span.with_neighbor(self.token.span).shrink_to_hi(),
2302 "add a semicolon",
2303 ';',
2304 Applicability::MaybeIncorrect,
2305 );
2306 }
2307 err.emit();
2308 }
2309
2310 fn recover_nested_adt_item(&mut self, keyword: Symbol) -> PResult<'a, bool> {
2313 if (self.token.is_keyword(kw::Enum)
2314 || self.token.is_keyword(kw::Struct)
2315 || self.token.is_keyword(kw::Union))
2316 && self.look_ahead(1, |t| t.is_ident())
2317 {
2318 let kw_token = self.token.clone();
2319 let kw_str = pprust::token_to_string(&kw_token);
2320 let item = self.parse_item(ForceCollect::No)?;
2321 let mut item = item.unwrap().span;
2322 if self.token == token::Comma {
2323 item = item.to(self.token.span);
2324 }
2325 self.dcx().emit_err(errors::NestedAdt {
2326 span: kw_token.span,
2327 item,
2328 kw_str,
2329 keyword: keyword.as_str(),
2330 });
2331 return Ok(false);
2333 }
2334 Ok(true)
2335 }
2336}
2337
2338type ReqName = fn(Edition) -> bool;
2345
2346#[derive(Clone, Copy)]
2354pub(crate) struct FnParseMode {
2355 pub(super) req_name: ReqName,
2378 pub(super) req_body: bool,
2397}
2398
2399impl<'a> Parser<'a> {
2401 fn parse_fn(
2403 &mut self,
2404 attrs: &mut AttrVec,
2405 fn_parse_mode: FnParseMode,
2406 sig_lo: Span,
2407 vis: &Visibility,
2408 case: Case,
2409 ) -> PResult<'a, (Ident, FnSig, Generics, Option<P<FnContract>>, Option<P<Block>>)> {
2410 let fn_span = self.token.span;
2411 let header = self.parse_fn_front_matter(vis, case)?; let ident = self.parse_ident()?; let mut generics = self.parse_generics()?; let decl = match self.parse_fn_decl(
2415 fn_parse_mode.req_name,
2416 AllowPlus::Yes,
2417 RecoverReturnSign::Yes,
2418 ) {
2419 Ok(decl) => decl,
2420 Err(old_err) => {
2421 if self.token.is_keyword(kw::For) {
2423 old_err.cancel();
2424 return Err(self.dcx().create_err(errors::FnTypoWithImpl { fn_span }));
2425 } else {
2426 return Err(old_err);
2427 }
2428 }
2429 };
2430
2431 let fn_params_end = self.prev_token.span.shrink_to_hi();
2434
2435 let contract = self.parse_contract()?;
2436
2437 generics.where_clause = self.parse_where_clause()?; let fn_params_end =
2441 if generics.where_clause.has_where_token { Some(fn_params_end) } else { None };
2442
2443 let mut sig_hi = self.prev_token.span;
2444 let body =
2446 self.parse_fn_body(attrs, &ident, &mut sig_hi, fn_parse_mode.req_body, fn_params_end)?;
2447 let fn_sig_span = sig_lo.to(sig_hi);
2448 Ok((ident, FnSig { header, decl, span: fn_sig_span }, generics, contract, body))
2449 }
2450
2451 fn error_fn_body_not_found(
2453 &mut self,
2454 ident_span: Span,
2455 req_body: bool,
2456 fn_params_end: Option<Span>,
2457 ) -> PResult<'a, ErrorGuaranteed> {
2458 let expected: &[_] =
2459 if req_body { &[exp!(OpenBrace)] } else { &[exp!(Semi), exp!(OpenBrace)] };
2460 match self.expected_one_of_not_found(&[], expected) {
2461 Ok(error_guaranteed) => Ok(error_guaranteed),
2462 Err(mut err) => {
2463 if self.token == token::CloseDelim(Delimiter::Brace) {
2464 err.span_label(ident_span, "while parsing this `fn`");
2467 Ok(err.emit())
2468 } else if self.token == token::RArrow
2469 && let Some(fn_params_end) = fn_params_end
2470 {
2471 let fn_trait_span =
2477 [sym::FnOnce, sym::FnMut, sym::Fn].into_iter().find_map(|symbol| {
2478 if self.prev_token.is_ident_named(symbol) {
2479 Some(self.prev_token.span)
2480 } else {
2481 None
2482 }
2483 });
2484
2485 let arrow_span = self.token.span;
2490 let ty_span = match self.parse_ret_ty(
2491 AllowPlus::Yes,
2492 RecoverQPath::Yes,
2493 RecoverReturnSign::Yes,
2494 ) {
2495 Ok(ty_span) => ty_span.span().shrink_to_hi(),
2496 Err(parse_error) => {
2497 parse_error.cancel();
2498 return Err(err);
2499 }
2500 };
2501 let ret_ty_span = arrow_span.to(ty_span);
2502
2503 if let Some(fn_trait_span) = fn_trait_span {
2504 err.subdiagnostic(errors::FnTraitMissingParen { span: fn_trait_span });
2507 } else if let Ok(snippet) = self.psess.source_map().span_to_snippet(ret_ty_span)
2508 {
2509 err.primary_message(
2513 "return type should be specified after the function parameters",
2514 );
2515 err.subdiagnostic(errors::MisplacedReturnType {
2516 fn_params_end,
2517 snippet,
2518 ret_ty_span,
2519 });
2520 }
2521 Err(err)
2522 } else {
2523 Err(err)
2524 }
2525 }
2526 }
2527 }
2528
2529 fn parse_fn_body(
2533 &mut self,
2534 attrs: &mut AttrVec,
2535 ident: &Ident,
2536 sig_hi: &mut Span,
2537 req_body: bool,
2538 fn_params_end: Option<Span>,
2539 ) -> PResult<'a, Option<P<Block>>> {
2540 let has_semi = if req_body {
2541 self.token == TokenKind::Semi
2542 } else {
2543 self.check(exp!(Semi))
2545 };
2546 let (inner_attrs, body) = if has_semi {
2547 self.expect_semi()?;
2549 *sig_hi = self.prev_token.span;
2550 (AttrVec::new(), None)
2551 } else if self.check(exp!(OpenBrace)) || self.token.is_whole_block() {
2552 self.parse_block_common(self.token.span, BlockCheckMode::Default, None)
2553 .map(|(attrs, body)| (attrs, Some(body)))?
2554 } else if self.token == token::Eq {
2555 self.bump(); let eq_sp = self.prev_token.span;
2558 let _ = self.parse_expr()?;
2559 self.expect_semi()?; let span = eq_sp.to(self.prev_token.span);
2561 let guar = self.dcx().emit_err(errors::FunctionBodyEqualsExpr {
2562 span,
2563 sugg: errors::FunctionBodyEqualsExprSugg { eq: eq_sp, semi: self.prev_token.span },
2564 });
2565 (AttrVec::new(), Some(self.mk_block_err(span, guar)))
2566 } else {
2567 self.error_fn_body_not_found(ident.span, req_body, fn_params_end)?;
2568 (AttrVec::new(), None)
2569 };
2570 attrs.extend(inner_attrs);
2571 Ok(body)
2572 }
2573
2574 pub(super) fn check_fn_front_matter(&mut self, check_pub: bool, case: Case) -> bool {
2579 const ALL_QUALS: &[ExpKeywordPair] = &[
2580 exp!(Pub),
2581 exp!(Gen),
2582 exp!(Const),
2583 exp!(Async),
2584 exp!(Unsafe),
2585 exp!(Safe),
2586 exp!(Extern),
2587 ];
2588
2589 let quals: &[_] = if check_pub {
2594 ALL_QUALS
2595 } else {
2596 &[exp!(Gen), exp!(Const), exp!(Async), exp!(Unsafe), exp!(Safe), exp!(Extern)]
2597 };
2598 self.check_keyword_case(exp!(Fn), case) || quals.iter().any(|&exp| self.check_keyword_case(exp, case))
2601 && self.look_ahead(1, |t| {
2602 t.is_keyword_case(kw::Fn, case)
2604 || (
2606 (
2607 t.is_non_raw_ident_where(|i|
2608 quals.iter().any(|exp| exp.kw == i.name)
2609 && i.is_reserved()
2611 )
2612 || case == Case::Insensitive
2613 && t.is_non_raw_ident_where(|i| quals.iter().any(|exp| {
2614 exp.kw.as_str() == i.name.as_str().to_lowercase()
2615 }))
2616 )
2617 && !self.is_unsafe_foreign_mod()
2619 && !self.is_async_gen_block())
2621 })
2622 || self.check_keyword_case(exp!(Extern), case)
2624 && self.look_ahead(1, |t| t.can_begin_string_literal())
2625 && (self.look_ahead(2, |t| t.is_keyword_case(kw::Fn, case)) ||
2626 (self.may_recover()
2629 && self.look_ahead(2, |t| ALL_QUALS.iter().any(|exp| t.is_keyword(exp.kw)))
2630 && self.look_ahead(3, |t| t.is_keyword_case(kw::Fn, case))))
2631 }
2632
2633 pub(super) fn parse_fn_front_matter(
2645 &mut self,
2646 orig_vis: &Visibility,
2647 case: Case,
2648 ) -> PResult<'a, FnHeader> {
2649 let sp_start = self.token.span;
2650 let constness = self.parse_constness(case);
2651
2652 let async_start_sp = self.token.span;
2653 let coroutine_kind = self.parse_coroutine_kind(case);
2654
2655 let unsafe_start_sp = self.token.span;
2656 let safety = self.parse_safety(case);
2657
2658 let ext_start_sp = self.token.span;
2659 let ext = self.parse_extern(case);
2660
2661 if let Some(CoroutineKind::Async { span, .. }) = coroutine_kind {
2662 if span.is_rust_2015() {
2663 self.dcx().emit_err(errors::AsyncFnIn2015 {
2664 span,
2665 help: errors::HelpUseLatestEdition::new(),
2666 });
2667 }
2668 }
2669
2670 match coroutine_kind {
2671 Some(CoroutineKind::Gen { span, .. }) | Some(CoroutineKind::AsyncGen { span, .. }) => {
2672 self.psess.gated_spans.gate(sym::gen_blocks, span);
2673 }
2674 Some(CoroutineKind::Async { .. }) | None => {}
2675 }
2676
2677 if !self.eat_keyword_case(exp!(Fn), case) {
2678 match self.expect_one_of(&[], &[]) {
2682 Ok(Recovered::Yes(_)) => {}
2683 Ok(Recovered::No) => unreachable!(),
2684 Err(mut err) => {
2685 enum WrongKw {
2687 Duplicated(Span),
2688 Misplaced(Span),
2689 }
2690
2691 let mut recover_constness = constness;
2693 let mut recover_coroutine_kind = coroutine_kind;
2694 let mut recover_safety = safety;
2695 let wrong_kw = if self.check_keyword(exp!(Const)) {
2698 match constness {
2699 Const::Yes(sp) => Some(WrongKw::Duplicated(sp)),
2700 Const::No => {
2701 recover_constness = Const::Yes(self.token.span);
2702 Some(WrongKw::Misplaced(async_start_sp))
2703 }
2704 }
2705 } else if self.check_keyword(exp!(Async)) {
2706 match coroutine_kind {
2707 Some(CoroutineKind::Async { span, .. }) => {
2708 Some(WrongKw::Duplicated(span))
2709 }
2710 Some(CoroutineKind::AsyncGen { span, .. }) => {
2711 Some(WrongKw::Duplicated(span))
2712 }
2713 Some(CoroutineKind::Gen { .. }) => {
2714 recover_coroutine_kind = Some(CoroutineKind::AsyncGen {
2715 span: self.token.span,
2716 closure_id: DUMMY_NODE_ID,
2717 return_impl_trait_id: DUMMY_NODE_ID,
2718 });
2719 Some(WrongKw::Misplaced(unsafe_start_sp))
2721 }
2722 None => {
2723 recover_coroutine_kind = Some(CoroutineKind::Async {
2724 span: self.token.span,
2725 closure_id: DUMMY_NODE_ID,
2726 return_impl_trait_id: DUMMY_NODE_ID,
2727 });
2728 Some(WrongKw::Misplaced(unsafe_start_sp))
2729 }
2730 }
2731 } else if self.check_keyword(exp!(Unsafe)) {
2732 match safety {
2733 Safety::Unsafe(sp) => Some(WrongKw::Duplicated(sp)),
2734 Safety::Safe(sp) => {
2735 recover_safety = Safety::Unsafe(self.token.span);
2736 Some(WrongKw::Misplaced(sp))
2737 }
2738 Safety::Default => {
2739 recover_safety = Safety::Unsafe(self.token.span);
2740 Some(WrongKw::Misplaced(ext_start_sp))
2741 }
2742 }
2743 } else if self.check_keyword(exp!(Safe)) {
2744 match safety {
2745 Safety::Safe(sp) => Some(WrongKw::Duplicated(sp)),
2746 Safety::Unsafe(sp) => {
2747 recover_safety = Safety::Safe(self.token.span);
2748 Some(WrongKw::Misplaced(sp))
2749 }
2750 Safety::Default => {
2751 recover_safety = Safety::Safe(self.token.span);
2752 Some(WrongKw::Misplaced(ext_start_sp))
2753 }
2754 }
2755 } else {
2756 None
2757 };
2758
2759 if let Some(WrongKw::Duplicated(original_sp)) = wrong_kw {
2761 let original_kw = self
2762 .span_to_snippet(original_sp)
2763 .expect("Span extracted directly from keyword should always work");
2764
2765 err.span_suggestion(
2766 self.token.uninterpolated_span(),
2767 format!("`{original_kw}` already used earlier, remove this one"),
2768 "",
2769 Applicability::MachineApplicable,
2770 )
2771 .span_note(original_sp, format!("`{original_kw}` first seen here"));
2772 }
2773 else if let Some(WrongKw::Misplaced(correct_pos_sp)) = wrong_kw {
2775 let correct_pos_sp = correct_pos_sp.to(self.prev_token.span);
2776 if let Ok(current_qual) = self.span_to_snippet(correct_pos_sp) {
2777 let misplaced_qual_sp = self.token.uninterpolated_span();
2778 let misplaced_qual = self.span_to_snippet(misplaced_qual_sp).unwrap();
2779
2780 err.span_suggestion(
2781 correct_pos_sp.to(misplaced_qual_sp),
2782 format!("`{misplaced_qual}` must come before `{current_qual}`"),
2783 format!("{misplaced_qual} {current_qual}"),
2784 Applicability::MachineApplicable,
2785 ).note("keyword order for functions declaration is `pub`, `default`, `const`, `async`, `unsafe`, `extern`");
2786 }
2787 }
2788 else if self.check_keyword(exp!(Pub)) {
2790 let sp = sp_start.to(self.prev_token.span);
2791 if let Ok(snippet) = self.span_to_snippet(sp) {
2792 let current_vis = match self.parse_visibility(FollowedByType::No) {
2793 Ok(v) => v,
2794 Err(d) => {
2795 d.cancel();
2796 return Err(err);
2797 }
2798 };
2799 let vs = pprust::vis_to_string(¤t_vis);
2800 let vs = vs.trim_end();
2801
2802 if matches!(orig_vis.kind, VisibilityKind::Inherited) {
2804 err.span_suggestion(
2805 sp_start.to(self.prev_token.span),
2806 format!("visibility `{vs}` must come before `{snippet}`"),
2807 format!("{vs} {snippet}"),
2808 Applicability::MachineApplicable,
2809 );
2810 }
2811 else {
2813 err.span_suggestion(
2814 current_vis.span,
2815 "there is already a visibility modifier, remove one",
2816 "",
2817 Applicability::MachineApplicable,
2818 )
2819 .span_note(orig_vis.span, "explicit visibility first seen here");
2820 }
2821 }
2822 }
2823
2824 if wrong_kw.is_some()
2827 && self.may_recover()
2828 && self.look_ahead(1, |tok| tok.is_keyword_case(kw::Fn, case))
2829 {
2830 self.bump();
2832 self.bump();
2833 err.emit();
2834 return Ok(FnHeader {
2835 constness: recover_constness,
2836 safety: recover_safety,
2837 coroutine_kind: recover_coroutine_kind,
2838 ext,
2839 });
2840 }
2841
2842 return Err(err);
2843 }
2844 }
2845 }
2846
2847 Ok(FnHeader { constness, safety, coroutine_kind, ext })
2848 }
2849
2850 pub(super) fn parse_fn_decl(
2852 &mut self,
2853 req_name: ReqName,
2854 ret_allow_plus: AllowPlus,
2855 recover_return_sign: RecoverReturnSign,
2856 ) -> PResult<'a, P<FnDecl>> {
2857 Ok(P(FnDecl {
2858 inputs: self.parse_fn_params(req_name)?,
2859 output: self.parse_ret_ty(ret_allow_plus, RecoverQPath::Yes, recover_return_sign)?,
2860 }))
2861 }
2862
2863 pub(super) fn parse_fn_params(&mut self, req_name: ReqName) -> PResult<'a, ThinVec<Param>> {
2865 let mut first_param = true;
2866 if self.token != TokenKind::OpenDelim(Delimiter::Parenthesis)
2868 && !self.token.is_keyword(kw::For)
2870 {
2871 self.dcx()
2873 .emit_err(errors::MissingFnParams { span: self.prev_token.span.shrink_to_hi() });
2874 return Ok(ThinVec::new());
2875 }
2876
2877 let (mut params, _) = self.parse_paren_comma_seq(|p| {
2878 p.recover_vcs_conflict_marker();
2879 let snapshot = p.create_snapshot_for_diagnostic();
2880 let param = p.parse_param_general(req_name, first_param).or_else(|e| {
2881 let guar = e.emit();
2882 let lo = if let TokenKind::OpenDelim(Delimiter::Parenthesis) = p.prev_token.kind {
2886 p.prev_token.span.shrink_to_hi()
2887 } else {
2888 p.prev_token.span
2889 };
2890 p.restore_snapshot(snapshot);
2891 p.eat_to_tokens(&[exp!(Comma), exp!(CloseParen)]);
2893 Ok(dummy_arg(Ident::new(sym::dummy, lo.to(p.prev_token.span)), guar))
2895 });
2896 first_param = false;
2898 param
2899 })?;
2900 self.deduplicate_recovered_params_names(&mut params);
2902 Ok(params)
2903 }
2904
2905 fn parse_param_general(&mut self, req_name: ReqName, first_param: bool) -> PResult<'a, Param> {
2909 let lo = self.token.span;
2910 let attrs = self.parse_outer_attributes()?;
2911 self.collect_tokens(None, attrs, ForceCollect::No, |this, attrs| {
2912 if let Some(mut param) = this.parse_self_param()? {
2914 param.attrs = attrs;
2915 let res = if first_param { Ok(param) } else { this.recover_bad_self_param(param) };
2916 return Ok((res?, Trailing::No, UsePreAttrPos::No));
2917 }
2918
2919 let is_name_required = match this.token.kind {
2920 token::DotDotDot => false,
2921 _ => req_name(this.token.span.with_neighbor(this.prev_token.span).edition()),
2922 };
2923 let (pat, ty) = if is_name_required || this.is_named_param() {
2924 debug!("parse_param_general parse_pat (is_name_required:{})", is_name_required);
2925 let (pat, colon) = this.parse_fn_param_pat_colon()?;
2926 if !colon {
2927 let mut err = this.unexpected().unwrap_err();
2928 return if let Some(ident) =
2929 this.parameter_without_type(&mut err, pat, is_name_required, first_param)
2930 {
2931 let guar = err.emit();
2932 Ok((dummy_arg(ident, guar), Trailing::No, UsePreAttrPos::No))
2933 } else {
2934 Err(err)
2935 };
2936 }
2937
2938 this.eat_incorrect_doc_comment_for_param_type();
2939 (pat, this.parse_ty_for_param()?)
2940 } else {
2941 debug!("parse_param_general ident_to_pat");
2942 let parser_snapshot_before_ty = this.create_snapshot_for_diagnostic();
2943 this.eat_incorrect_doc_comment_for_param_type();
2944 let mut ty = this.parse_ty_for_param();
2945 if ty.is_ok()
2946 && this.token != token::Comma
2947 && this.token != token::CloseDelim(Delimiter::Parenthesis)
2948 {
2949 ty = this.unexpected_any();
2952 }
2953 match ty {
2954 Ok(ty) => {
2955 let ident = Ident::new(kw::Empty, this.prev_token.span);
2956 let bm = BindingMode::NONE;
2957 let pat = this.mk_pat_ident(ty.span, bm, ident);
2958 (pat, ty)
2959 }
2960 Err(err) if this.token == token::DotDotDot => return Err(err),
2962 Err(err) => {
2964 err.cancel();
2965 this.restore_snapshot(parser_snapshot_before_ty);
2966 this.recover_arg_parse()?
2967 }
2968 }
2969 };
2970
2971 let span = lo.to(this.prev_token.span);
2972
2973 Ok((
2974 Param { attrs, id: ast::DUMMY_NODE_ID, is_placeholder: false, pat, span, ty },
2975 Trailing::No,
2976 UsePreAttrPos::No,
2977 ))
2978 })
2979 }
2980
2981 fn parse_self_param(&mut self) -> PResult<'a, Option<Param>> {
2983 let expect_self_ident = |this: &mut Self| match this.token.ident() {
2985 Some((ident, IdentIsRaw::No)) => {
2986 this.bump();
2987 ident
2988 }
2989 _ => unreachable!(),
2990 };
2991 let is_lifetime = |this: &Self, n| this.look_ahead(n, |t| t.is_lifetime());
2993 let is_isolated_self = |this: &Self, n| {
2995 this.is_keyword_ahead(n, &[kw::SelfLower])
2996 && this.look_ahead(n + 1, |t| t != &token::PathSep)
2997 };
2998 let is_isolated_pin_const_self = |this: &Self, n| {
3000 this.look_ahead(n, |token| token.is_ident_named(sym::pin))
3001 && this.is_keyword_ahead(n + 1, &[kw::Const])
3002 && is_isolated_self(this, n + 2)
3003 };
3004 let is_isolated_mut_self =
3006 |this: &Self, n| this.is_keyword_ahead(n, &[kw::Mut]) && is_isolated_self(this, n + 1);
3007 let is_isolated_pin_mut_self = |this: &Self, n| {
3009 this.look_ahead(n, |token| token.is_ident_named(sym::pin))
3010 && is_isolated_mut_self(this, n + 1)
3011 };
3012 let parse_self_possibly_typed = |this: &mut Self, m| {
3014 let eself_ident = expect_self_ident(this);
3015 let eself_hi = this.prev_token.span;
3016 let eself = if this.eat(exp!(Colon)) {
3017 SelfKind::Explicit(this.parse_ty()?, m)
3018 } else {
3019 SelfKind::Value(m)
3020 };
3021 Ok((eself, eself_ident, eself_hi))
3022 };
3023 let expect_self_ident_not_typed =
3024 |this: &mut Self, modifier: &SelfKind, modifier_span: Span| {
3025 let eself_ident = expect_self_ident(this);
3026
3027 if this.may_recover() && this.eat_noexpect(&token::Colon) {
3029 let snap = this.create_snapshot_for_diagnostic();
3030 match this.parse_ty() {
3031 Ok(ty) => {
3032 this.dcx().emit_err(errors::IncorrectTypeOnSelf {
3033 span: ty.span,
3034 move_self_modifier: errors::MoveSelfModifier {
3035 removal_span: modifier_span,
3036 insertion_span: ty.span.shrink_to_lo(),
3037 modifier: modifier.to_ref_suggestion(),
3038 },
3039 });
3040 }
3041 Err(diag) => {
3042 diag.cancel();
3043 this.restore_snapshot(snap);
3044 }
3045 }
3046 }
3047 eself_ident
3048 };
3049 let recover_self_ptr = |this: &mut Self| {
3051 this.dcx().emit_err(errors::SelfArgumentPointer { span: this.token.span });
3052
3053 Ok((SelfKind::Value(Mutability::Not), expect_self_ident(this), this.prev_token.span))
3054 };
3055
3056 let eself_lo = self.token.span;
3060 let (eself, eself_ident, eself_hi) = match self.token.uninterpolate().kind {
3061 token::And => {
3062 let has_lifetime = is_lifetime(self, 1);
3063 let skip_lifetime_count = has_lifetime as usize;
3064 let eself = if is_isolated_self(self, skip_lifetime_count + 1) {
3065 self.bump(); let lifetime = has_lifetime.then(|| self.expect_lifetime());
3068 SelfKind::Region(lifetime, Mutability::Not)
3069 } else if is_isolated_mut_self(self, skip_lifetime_count + 1) {
3070 self.bump(); let lifetime = has_lifetime.then(|| self.expect_lifetime());
3073 self.bump(); SelfKind::Region(lifetime, Mutability::Mut)
3075 } else if is_isolated_pin_const_self(self, skip_lifetime_count + 1) {
3076 self.bump(); let lifetime = has_lifetime.then(|| self.expect_lifetime());
3079 self.psess.gated_spans.gate(sym::pin_ergonomics, self.token.span);
3080 self.bump(); self.bump(); SelfKind::Pinned(lifetime, Mutability::Not)
3083 } else if is_isolated_pin_mut_self(self, skip_lifetime_count + 1) {
3084 self.bump(); let lifetime = has_lifetime.then(|| self.expect_lifetime());
3087 self.psess.gated_spans.gate(sym::pin_ergonomics, self.token.span);
3088 self.bump(); self.bump(); SelfKind::Pinned(lifetime, Mutability::Mut)
3091 } else {
3092 return Ok(None);
3094 };
3095 let hi = self.token.span;
3096 let self_ident = expect_self_ident_not_typed(self, &eself, eself_lo.until(hi));
3097 (eself, self_ident, hi)
3098 }
3099 token::Star if is_isolated_self(self, 1) => {
3101 self.bump();
3102 recover_self_ptr(self)?
3103 }
3104 token::Star
3106 if self.look_ahead(1, |t| t.is_mutability()) && is_isolated_self(self, 2) =>
3107 {
3108 self.bump();
3109 self.bump();
3110 recover_self_ptr(self)?
3111 }
3112 token::Ident(..) if is_isolated_self(self, 0) => {
3114 parse_self_possibly_typed(self, Mutability::Not)?
3115 }
3116 token::Ident(..) if is_isolated_mut_self(self, 0) => {
3118 self.bump();
3119 parse_self_possibly_typed(self, Mutability::Mut)?
3120 }
3121 _ => return Ok(None),
3122 };
3123
3124 let eself = source_map::respan(eself_lo.to(eself_hi), eself);
3125 Ok(Some(Param::from_self(AttrVec::default(), eself, eself_ident)))
3126 }
3127
3128 fn is_named_param(&self) -> bool {
3129 let offset = match &self.token.kind {
3130 token::OpenDelim(Delimiter::Invisible(origin)) => match origin {
3131 InvisibleOrigin::MetaVar(MetaVarKind::Pat(_)) => {
3132 return self.check_noexpect_past_close_delim(&token::Colon);
3133 }
3134 _ => 0,
3135 },
3136 token::And | token::AndAnd => 1,
3137 _ if self.token.is_keyword(kw::Mut) => 1,
3138 _ => 0,
3139 };
3140
3141 self.look_ahead(offset, |t| t.is_ident())
3142 && self.look_ahead(offset + 1, |t| t == &token::Colon)
3143 }
3144
3145 fn recover_self_param(&mut self) -> bool {
3146 matches!(
3147 self.parse_outer_attributes()
3148 .and_then(|_| self.parse_self_param())
3149 .map_err(|e| e.cancel()),
3150 Ok(Some(_))
3151 )
3152 }
3153}
3154
3155enum IsMacroRulesItem {
3156 Yes { has_bang: bool },
3157 No,
3158}