| // Copyright 2018 Syn Developers |
| // |
| // Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or |
| // http://www.apache.org/licenses/LICENSE-2.0> or the MIT license |
| // <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your |
| // option. This file may not be copied, modified, or distributed |
| // except according to those terms. |
| |
| #[cfg(any(feature = "full", feature = "derive"))] |
| macro_rules! ast_struct { |
| ( |
| $(#[$attr:meta])* |
| pub struct $name:ident #full $($rest:tt)* |
| ) => { |
| #[cfg(feature = "full")] |
| $(#[$attr])* |
| #[cfg_attr(feature = "extra-traits", derive(Debug, Eq, PartialEq, Hash))] |
| #[cfg_attr(feature = "clone-impls", derive(Clone))] |
| pub struct $name $($rest)* |
| |
| #[cfg(not(feature = "full"))] |
| $(#[$attr])* |
| #[cfg_attr(feature = "extra-traits", derive(Debug, Eq, PartialEq, Hash))] |
| #[cfg_attr(feature = "clone-impls", derive(Clone))] |
| pub struct $name { |
| _noconstruct: (), |
| } |
| }; |
| |
| ( |
| $(#[$attr:meta])* |
| pub struct $name:ident #manual_extra_traits $($rest:tt)* |
| ) => { |
| $(#[$attr])* |
| #[cfg_attr(feature = "extra-traits", derive(Debug))] |
| #[cfg_attr(feature = "clone-impls", derive(Clone))] |
| pub struct $name $($rest)* |
| }; |
| |
| ( |
| $(#[$attr:meta])* |
| pub struct $name:ident $($rest:tt)* |
| ) => { |
| $(#[$attr])* |
| #[cfg_attr(feature = "extra-traits", derive(Debug, Eq, PartialEq, Hash))] |
| #[cfg_attr(feature = "clone-impls", derive(Clone))] |
| pub struct $name $($rest)* |
| }; |
| } |
| |
| #[cfg(any(feature = "full", feature = "derive"))] |
| macro_rules! ast_enum { |
| ( |
| $(#[$enum_attr:meta])* |
| pub enum $name:ident $(# $tags:ident)* { $($variants:tt)* } |
| ) => ( |
| $(#[$enum_attr])* |
| #[cfg_attr(feature = "extra-traits", derive(Debug, Eq, PartialEq, Hash))] |
| #[cfg_attr(feature = "clone-impls", derive(Clone))] |
| pub enum $name { |
| $($variants)* |
| } |
| ) |
| } |
| |
| #[cfg(any(feature = "full", feature = "derive"))] |
| macro_rules! ast_enum_of_structs { |
| ( |
| $(#[$enum_attr:meta])* |
| pub enum $name:ident { |
| $( |
| $(#[$variant_attr:meta])* |
| pub $variant:ident $( ($member:ident $($rest:tt)*) )*, |
| )* |
| } |
| |
| $($remaining:tt)* |
| ) => ( |
| ast_enum! { |
| $(#[$enum_attr])* |
| pub enum $name { |
| $( |
| $(#[$variant_attr])* |
| $variant $( ($member) )*, |
| )* |
| } |
| } |
| |
| $( |
| maybe_ast_struct! { |
| $(#[$variant_attr])* |
| $( |
| pub struct $member $($rest)* |
| )* |
| } |
| |
| $( |
| impl From<$member> for $name { |
| fn from(e: $member) -> $name { |
| $name::$variant(e) |
| } |
| } |
| )* |
| )* |
| |
| #[cfg(feature = "printing")] |
| generate_to_tokens! { |
| $($remaining)* |
| () |
| tokens |
| $name { $($variant $( [$($rest)*] )*,)* } |
| } |
| ) |
| } |
| |
| #[cfg(all( |
| feature = "printing", |
| any(feature = "full", feature = "derive") |
| ))] |
| macro_rules! generate_to_tokens { |
| (do_not_generate_to_tokens $($foo:tt)*) => (); |
| |
| (($($arms:tt)*) $tokens:ident $name:ident { $variant:ident, $($next:tt)*}) => { |
| generate_to_tokens!( |
| ($($arms)* $name::$variant => {}) |
| $tokens $name { $($next)* } |
| ); |
| }; |
| |
| (($($arms:tt)*) $tokens:ident $name:ident { $variant:ident [$($rest:tt)*], $($next:tt)*}) => { |
| generate_to_tokens!( |
| ($($arms)* $name::$variant(ref _e) => to_tokens_call!(_e, $tokens, $($rest)*),) |
| $tokens $name { $($next)* } |
| ); |
| }; |
| |
| (($($arms:tt)*) $tokens:ident $name:ident {}) => { |
| impl ::quote::ToTokens for $name { |
| fn to_tokens(&self, $tokens: &mut ::proc_macro2::TokenStream) { |
| match *self { |
| $($arms)* |
| } |
| } |
| } |
| }; |
| } |
| |
| #[cfg(all(feature = "printing", feature = "full"))] |
| macro_rules! to_tokens_call { |
| ($e:ident, $tokens:ident, $($rest:tt)*) => { |
| $e.to_tokens($tokens) |
| }; |
| } |
| |
| #[cfg(all( |
| feature = "printing", |
| feature = "derive", |
| not(feature = "full") |
| ))] |
| macro_rules! to_tokens_call { |
| // If the variant is marked as #full, don't auto-generate to-tokens for it. |
| ($e:ident, $tokens:ident, #full $($rest:tt)*) => { |
| unreachable!() |
| }; |
| ($e:ident, $tokens:ident, $($rest:tt)*) => { |
| $e.to_tokens($tokens) |
| }; |
| } |
| |
| #[cfg(any(feature = "full", feature = "derive"))] |
| macro_rules! maybe_ast_struct { |
| ( |
| $(#[$attr:meta])* |
| $( |
| pub struct $name:ident |
| )* |
| ) => (); |
| |
| ($($rest:tt)*) => (ast_struct! { $($rest)* }); |
| } |