//! A set of traits implemented for various AST nodes, //! typically those used in AST fragments during macro expansion. //! The traits are not implemented exhaustively, only when actually necessary. use crate::ptr::P; use crate::token::Nonterminal; use crate::tokenstream::LazyTokenStream; use crate::{Arm, Crate, ExprField, FieldDef, GenericParam, Param, PatField, Variant}; use crate::{AssocItem, Expr, ForeignItem, Item, NodeId}; use crate::{AttrItem, AttrKind, Block, Pat, Path, Ty, Visibility}; use crate::{AttrVec, Attribute, Stmt, StmtKind}; use rustc_span::Span; use std::fmt; use std::marker::PhantomData; /// A utility trait to reduce boilerplate. /// Standard `Deref(Mut)` cannot be reused due to coherence. pub trait AstDeref { type Target; fn ast_deref(&self) -> &Self::Target; fn ast_deref_mut(&mut self) -> &mut Self::Target; } macro_rules! impl_not_ast_deref { ($($T:ty),+ $(,)?) => { $( impl !AstDeref for $T {} )+ }; } impl_not_ast_deref!(AssocItem, Expr, ForeignItem, Item, Stmt); impl AstDeref for P { type Target = T; fn ast_deref(&self) -> &Self::Target { self } fn ast_deref_mut(&mut self) -> &mut Self::Target { self } } /// A trait for AST nodes having an ID. pub trait HasNodeId { fn node_id(&self) -> NodeId; fn node_id_mut(&mut self) -> &mut NodeId; } macro_rules! impl_has_node_id { ($($T:ty),+ $(,)?) => { $( impl HasNodeId for $T { fn node_id(&self) -> NodeId { self.id } fn node_id_mut(&mut self) -> &mut NodeId { &mut self.id } } )+ }; } impl_has_node_id!( Arm, AssocItem, Crate, Expr, ExprField, FieldDef, ForeignItem, GenericParam, Item, Param, Pat, PatField, Stmt, Ty, Variant, ); impl> HasNodeId for T { fn node_id(&self) -> NodeId { self.ast_deref().node_id() } fn node_id_mut(&mut self) -> &mut NodeId { self.ast_deref_mut().node_id_mut() } } /// A trait for AST nodes having a span. pub trait HasSpan { fn span(&self) -> Span; } macro_rules! impl_has_span { ($($T:ty),+ $(,)?) => { $( impl HasSpan for $T { fn span(&self) -> Span { self.span } } )+ }; } impl_has_span!(AssocItem, Block, Expr, ForeignItem, Item, Pat, Path, Stmt, Ty, Visibility); impl> HasSpan for T { fn span(&self) -> Span { self.ast_deref().span() } } impl HasSpan for AttrItem { fn span(&self) -> Span { self.span() } } /// A trait for AST nodes having (or not having) collected tokens. pub trait HasTokens { fn tokens(&self) -> Option<&LazyTokenStream>; fn tokens_mut(&mut self) -> Option<&mut Option>; } macro_rules! impl_has_tokens { ($($T:ty),+ $(,)?) => { $( impl HasTokens for $T { fn tokens(&self) -> Option<&LazyTokenStream> { self.tokens.as_ref() } fn tokens_mut(&mut self) -> Option<&mut Option> { Some(&mut self.tokens) } } )+ }; } macro_rules! impl_has_tokens_none { ($($T:ty),+ $(,)?) => { $( impl HasTokens for $T { fn tokens(&self) -> Option<&LazyTokenStream> { None } fn tokens_mut(&mut self) -> Option<&mut Option> { None } } )+ }; } impl_has_tokens!(AssocItem, AttrItem, Block, Expr, ForeignItem, Item, Pat, Path, Ty, Visibility); impl_has_tokens_none!(Arm, ExprField, FieldDef, GenericParam, Param, PatField, Variant); impl> HasTokens for T { fn tokens(&self) -> Option<&LazyTokenStream> { self.ast_deref().tokens() } fn tokens_mut(&mut self) -> Option<&mut Option> { self.ast_deref_mut().tokens_mut() } } impl HasTokens for Option { fn tokens(&self) -> Option<&LazyTokenStream> { self.as_ref().and_then(|inner| inner.tokens()) } fn tokens_mut(&mut self) -> Option<&mut Option> { self.as_mut().and_then(|inner| inner.tokens_mut()) } } impl HasTokens for StmtKind { fn tokens(&self) -> Option<&LazyTokenStream> { match self { StmtKind::Local(local) => local.tokens.as_ref(), StmtKind::Item(item) => item.tokens(), StmtKind::Expr(expr) | StmtKind::Semi(expr) => expr.tokens(), StmtKind::Empty => return None, StmtKind::MacCall(mac) => mac.tokens.as_ref(), } } fn tokens_mut(&mut self) -> Option<&mut Option> { match self { StmtKind::Local(local) => Some(&mut local.tokens), StmtKind::Item(item) => item.tokens_mut(), StmtKind::Expr(expr) | StmtKind::Semi(expr) => expr.tokens_mut(), StmtKind::Empty => return None, StmtKind::MacCall(mac) => Some(&mut mac.tokens), } } } impl HasTokens for Stmt { fn tokens(&self) -> Option<&LazyTokenStream> { self.kind.tokens() } fn tokens_mut(&mut self) -> Option<&mut Option> { self.kind.tokens_mut() } } impl HasTokens for Attribute { fn tokens(&self) -> Option<&LazyTokenStream> { match &self.kind { AttrKind::Normal(normal) => normal.tokens.as_ref(), kind @ AttrKind::DocComment(..) => { panic!("Called tokens on doc comment attr {:?}", kind) } } } fn tokens_mut(&mut self) -> Option<&mut Option> { Some(match &mut self.kind { AttrKind::Normal(normal) => &mut normal.tokens, kind @ AttrKind::DocComment(..) => { panic!("Called tokens_mut on doc comment attr {:?}", kind) } }) } } impl HasTokens for Nonterminal { fn tokens(&self) -> Option<&LazyTokenStream> { match self { Nonterminal::NtItem(item) => item.tokens(), Nonterminal::NtStmt(stmt) => stmt.tokens(), Nonterminal::NtExpr(expr) | Nonterminal::NtLiteral(expr) => expr.tokens(), Nonterminal::NtPat(pat) => pat.tokens(), Nonterminal::NtTy(ty) => ty.tokens(), Nonterminal::NtMeta(attr_item) => attr_item.tokens(), Nonterminal::NtPath(path) => path.tokens(), Nonterminal::NtVis(vis) => vis.tokens(), Nonterminal::NtBlock(block) => block.tokens(), Nonterminal::NtIdent(..) | Nonterminal::NtLifetime(..) => None, } } fn tokens_mut(&mut self) -> Option<&mut Option> { match self { Nonterminal::NtItem(item) => item.tokens_mut(), Nonterminal::NtStmt(stmt) => stmt.tokens_mut(), Nonterminal::NtExpr(expr) | Nonterminal::NtLiteral(expr) => expr.tokens_mut(), Nonterminal::NtPat(pat) => pat.tokens_mut(), Nonterminal::NtTy(ty) => ty.tokens_mut(), Nonterminal::NtMeta(attr_item) => attr_item.tokens_mut(), Nonterminal::NtPath(path) => path.tokens_mut(), Nonterminal::NtVis(vis) => vis.tokens_mut(), Nonterminal::NtBlock(block) => block.tokens_mut(), Nonterminal::NtIdent(..) | Nonterminal::NtLifetime(..) => None, } } } /// A trait for AST nodes having (or not having) attributes. pub trait HasAttrs { /// This is `true` if this `HasAttrs` might support 'custom' (proc-macro) inner /// attributes. Attributes like `#![cfg]` and `#![cfg_attr]` are not /// considered 'custom' attributes. /// /// If this is `false`, then this `HasAttrs` definitely does /// not support 'custom' inner attributes, which enables some optimizations /// during token collection. const SUPPORTS_CUSTOM_INNER_ATTRS: bool; fn attrs(&self) -> &[Attribute]; fn visit_attrs(&mut self, f: impl FnOnce(&mut AttrVec)); } macro_rules! impl_has_attrs { (const SUPPORTS_CUSTOM_INNER_ATTRS: bool = $inner:literal, $($T:ty),+ $(,)?) => { $( impl HasAttrs for $T { const SUPPORTS_CUSTOM_INNER_ATTRS: bool = $inner; fn attrs(&self) -> &[Attribute] { &self.attrs } fn visit_attrs(&mut self, f: impl FnOnce(&mut AttrVec)) { f(&mut self.attrs) } } )+ }; } macro_rules! impl_has_attrs_none { ($($T:ty),+ $(,)?) => { $( impl HasAttrs for $T { const SUPPORTS_CUSTOM_INNER_ATTRS: bool = false; fn attrs(&self) -> &[Attribute] { &[] } fn visit_attrs(&mut self, _f: impl FnOnce(&mut AttrVec)) {} } )+ }; } impl_has_attrs!( const SUPPORTS_CUSTOM_INNER_ATTRS: bool = true, AssocItem, ForeignItem, Item, ); impl_has_attrs!( const SUPPORTS_CUSTOM_INNER_ATTRS: bool = false, Arm, Crate, Expr, ExprField, FieldDef, GenericParam, Param, PatField, Variant, ); impl_has_attrs_none!(Attribute, AttrItem, Block, Pat, Path, Ty, Visibility); impl> HasAttrs for T { const SUPPORTS_CUSTOM_INNER_ATTRS: bool = T::Target::SUPPORTS_CUSTOM_INNER_ATTRS; fn attrs(&self) -> &[Attribute] { self.ast_deref().attrs() } fn visit_attrs(&mut self, f: impl FnOnce(&mut AttrVec)) { self.ast_deref_mut().visit_attrs(f) } } impl HasAttrs for Option { const SUPPORTS_CUSTOM_INNER_ATTRS: bool = T::SUPPORTS_CUSTOM_INNER_ATTRS; fn attrs(&self) -> &[Attribute] { self.as_ref().map(|inner| inner.attrs()).unwrap_or(&[]) } fn visit_attrs(&mut self, f: impl FnOnce(&mut AttrVec)) { if let Some(inner) = self.as_mut() { inner.visit_attrs(f); } } } impl HasAttrs for StmtKind { // This might be a `StmtKind::Item`, which contains // an item that supports inner attrs. const SUPPORTS_CUSTOM_INNER_ATTRS: bool = true; fn attrs(&self) -> &[Attribute] { match self { StmtKind::Local(local) => &local.attrs, StmtKind::Expr(expr) | StmtKind::Semi(expr) => expr.attrs(), StmtKind::Item(item) => item.attrs(), StmtKind::Empty => &[], StmtKind::MacCall(mac) => &mac.attrs, } } fn visit_attrs(&mut self, f: impl FnOnce(&mut AttrVec)) { match self { StmtKind::Local(local) => f(&mut local.attrs), StmtKind::Expr(expr) | StmtKind::Semi(expr) => expr.visit_attrs(f), StmtKind::Item(item) => item.visit_attrs(f), StmtKind::Empty => {} StmtKind::MacCall(mac) => f(&mut mac.attrs), } } } impl HasAttrs for Stmt { const SUPPORTS_CUSTOM_INNER_ATTRS: bool = StmtKind::SUPPORTS_CUSTOM_INNER_ATTRS; fn attrs(&self) -> &[Attribute] { self.kind.attrs() } fn visit_attrs(&mut self, f: impl FnOnce(&mut AttrVec)) { self.kind.visit_attrs(f); } } /// A newtype around an AST node that implements the traits above if the node implements them. pub struct AstNodeWrapper { pub wrapped: Wrapped, pub tag: PhantomData, } impl AstNodeWrapper { pub fn new(wrapped: Wrapped, _tag: Tag) -> AstNodeWrapper { AstNodeWrapper { wrapped, tag: Default::default() } } } impl AstDeref for AstNodeWrapper { type Target = Wrapped; fn ast_deref(&self) -> &Self::Target { &self.wrapped } fn ast_deref_mut(&mut self) -> &mut Self::Target { &mut self.wrapped } } impl fmt::Debug for AstNodeWrapper { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { f.debug_struct("AstNodeWrapper") .field("wrapped", &self.wrapped) .field("tag", &self.tag) .finish() } }