#![cfg_attr(not(feature = "default"), allow(dead_code))]
use proc_macro2::{Span, TokenStream};
use quote::{quote, ToTokens};
use syn::{
parse::{Error, Result},
parse_str,
spanned::Spanned,
Attribute, Data, DeriveInput, Field, Fields, FieldsNamed, FieldsUnnamed,
GenericParam, Generics, Ident, ImplGenerics, Index, Meta, NestedMeta, Type,
TypeGenerics, TypeParamBound, Variant, WhereClause,
};
#[derive(Clone, Copy, Eq, PartialEq, Hash)]
pub enum RefType {
No,
Ref,
Mut,
}
impl RefType {
pub fn lifetime(self) -> TokenStream {
match self {
RefType::No => quote!(),
_ => quote!('__deriveMoreLifetime),
}
}
pub fn reference(self) -> TokenStream {
match self {
RefType::No => quote!(),
RefType::Ref => quote!(&),
RefType::Mut => quote!(&mut),
}
}
pub fn mutability(self) -> TokenStream {
match self {
RefType::Mut => quote!(mut),
_ => quote!(),
}
}
pub fn pattern_ref(self) -> TokenStream {
match self {
RefType::Ref => quote!(ref),
RefType::Mut => quote!(ref mut),
RefType::No => quote!(),
}
}
pub fn reference_with_lifetime(self) -> TokenStream {
if !self.is_ref() {
return quote!();
}
let lifetime = self.lifetime();
let mutability = self.mutability();
quote!(&#lifetime #mutability)
}
pub fn is_ref(self) -> bool {
match self {
RefType::No => false,
_ => true,
}
}
}
pub fn numbered_vars(count: usize, prefix: &str) -> Vec<Ident> {
(0..count)
.map(|i| Ident::new(&format!("__{}{}", prefix, i), Span::call_site()))
.collect()
}
pub fn field_idents<'a>(fields: &'a [&'a Field]) -> Vec<&'a Ident> {
fields
.iter()
.map(|f| {
f.ident
.as_ref()
.expect("Tried to get field names of a tuple struct")
})
.collect()
}
pub fn get_field_types_iter<'a>(
fields: &'a [&'a Field],
) -> Box<dyn Iterator<Item = &'a Type> + 'a> {
Box::new(fields.iter().map(|f| &f.ty))
}
pub fn get_field_types<'a>(fields: &'a [&'a Field]) -> Vec<&'a Type> {
get_field_types_iter(fields).collect()
}
pub fn add_extra_type_param_bound_op_output<'a>(
generics: &'a Generics,
trait_ident: &'a Ident,
) -> Generics {
let mut generics = generics.clone();
for type_param in &mut generics.type_params_mut() {
let type_ident = &type_param.ident;
let bound: TypeParamBound = parse_str(
"e!(::core::ops::#trait_ident<Output=#type_ident>).to_string(),
)
.unwrap();
type_param.bounds.push(bound)
}
generics
}
pub fn add_extra_ty_param_bound_op<'a>(
generics: &'a Generics,
trait_ident: &'a Ident,
) -> Generics {
add_extra_ty_param_bound(generics, "e!(::core::ops::#trait_ident))
}
pub fn add_extra_ty_param_bound<'a>(
generics: &'a Generics,
bound: &'a TokenStream,
) -> Generics {
let mut generics = generics.clone();
let bound: TypeParamBound = parse_str(&bound.to_string()).unwrap();
for type_param in &mut generics.type_params_mut() {
type_param.bounds.push(bound.clone())
}
generics
}
pub fn add_extra_ty_param_bound_ref<'a>(
generics: &'a Generics,
bound: &'a TokenStream,
ref_type: RefType,
) -> Generics {
match ref_type {
RefType::No => add_extra_ty_param_bound(generics, bound),
_ => {
let generics = generics.clone();
let idents = generics.type_params().map(|x| &x.ident);
let ref_with_lifetime = ref_type.reference_with_lifetime();
add_extra_where_clauses(
&generics,
quote!(
where #(#ref_with_lifetime #idents: #bound),*
),
)
}
}
}
pub fn add_extra_generic_param(
generics: &Generics,
generic_param: TokenStream,
) -> Generics {
let generic_param: GenericParam = parse_str(&generic_param.to_string()).unwrap();
let mut generics = generics.clone();
generics.params.push(generic_param);
generics
}
pub fn add_extra_where_clauses(
generics: &Generics,
type_where_clauses: TokenStream,
) -> Generics {
let mut type_where_clauses: WhereClause =
parse_str(&type_where_clauses.to_string()).unwrap();
let mut new_generics = generics.clone();
if let Some(old_where) = new_generics.where_clause {
type_where_clauses.predicates.extend(old_where.predicates)
}
new_generics.where_clause = Some(type_where_clauses);
new_generics
}
pub fn add_where_clauses_for_new_ident<'a>(
generics: &'a Generics,
fields: &[&'a Field],
type_ident: &Ident,
type_where_clauses: TokenStream,
sized: bool,
) -> Generics {
let generic_param = if fields.len() > 1 {
quote!(#type_ident: ::core::marker::Copy)
} else if sized {
quote!(#type_ident)
} else {
quote!(#type_ident: ?::core::marker::Sized)
};
let generics = add_extra_where_clauses(generics, type_where_clauses);
add_extra_generic_param(&generics, generic_param)
}
pub fn unnamed_to_vec(fields: &FieldsUnnamed) -> Vec<&Field> {
fields.unnamed.iter().collect()
}
pub fn named_to_vec(fields: &FieldsNamed) -> Vec<&Field> {
fields.named.iter().collect()
}
fn panic_one_field(trait_name: &str, trait_attr: &str) -> ! {
panic!(format!(
"derive({}) only works when forwarding to a single field. Try putting #[{}] or #[{}(ignore)] on the fields in the struct",
trait_name, trait_attr, trait_attr,
))
}
#[derive(PartialEq, Eq)]
pub enum DeriveType {
Unnamed,
Named,
Enum,
}
pub struct State<'input> {
pub input: &'input DeriveInput,
pub trait_name: &'static str,
pub trait_ident: Ident,
pub method_ident: Ident,
pub trait_module: TokenStream,
pub trait_path: TokenStream,
pub trait_path_params: Vec<TokenStream>,
pub trait_attr: String,
pub derive_type: DeriveType,
pub fields: Vec<&'input Field>,
pub variants: Vec<&'input Variant>,
pub variant_states: Vec<State<'input>>,
pub variant: Option<&'input Variant>,
pub generics: Generics,
pub default_info: FullMetaInfo,
full_meta_infos: Vec<FullMetaInfo>,
}
#[derive(Default, Clone)]
pub struct AttrParams {
pub enum_: Vec<&'static str>,
pub variant: Vec<&'static str>,
pub struct_: Vec<&'static str>,
pub field: Vec<&'static str>,
}
impl AttrParams {
pub fn new(params: Vec<&'static str>) -> AttrParams {
AttrParams {
enum_: params.clone(),
struct_: params.clone(),
variant: params.clone(),
field: params,
}
}
pub fn struct_(params: Vec<&'static str>) -> AttrParams {
AttrParams {
enum_: vec![],
struct_: params,
variant: vec![],
field: vec![],
}
}
}
impl<'input> State<'input> {
pub fn new<'arg_input>(
input: &'arg_input DeriveInput,
trait_name: &'static str,
trait_module: TokenStream,
trait_attr: String,
) -> Result<State<'arg_input>> {
State::with_attr_params(
input,
trait_name,
trait_module,
trait_attr,
AttrParams::default(),
)
}
pub fn with_field_ignore<'arg_input>(
input: &'arg_input DeriveInput,
trait_name: &'static str,
trait_module: TokenStream,
trait_attr: String,
) -> Result<State<'arg_input>> {
State::with_attr_params(
input,
trait_name,
trait_module,
trait_attr,
AttrParams::new(vec!["ignore"]),
)
}
pub fn with_field_ignore_and_forward<'arg_input>(
input: &'arg_input DeriveInput,
trait_name: &'static str,
trait_module: TokenStream,
trait_attr: String,
) -> Result<State<'arg_input>> {
State::with_attr_params(
input,
trait_name,
trait_module,
trait_attr,
AttrParams::new(vec!["ignore", "forward"]),
)
}
pub fn with_field_ignore_and_refs<'arg_input>(
input: &'arg_input DeriveInput,
trait_name: &'static str,
trait_module: TokenStream,
trait_attr: String,
) -> Result<State<'arg_input>> {
State::with_attr_params(
input,
trait_name,
trait_module,
trait_attr,
AttrParams::new(vec!["ignore", "owned", "ref", "ref_mut"]),
)
}
pub fn with_attr_params<'arg_input>(
input: &'arg_input DeriveInput,
trait_name: &'static str,
trait_module: TokenStream,
trait_attr: String,
allowed_attr_params: AttrParams,
) -> Result<State<'arg_input>> {
let trait_name = trait_name.trim_end_matches("ToInner");
let trait_ident = Ident::new(trait_name, Span::call_site());
let method_ident = Ident::new(&trait_attr, Span::call_site());
let trait_path = quote!(#trait_module::#trait_ident);
let (derive_type, fields, variants): (_, Vec<_>, Vec<_>) = match input.data {
Data::Struct(ref data_struct) => match data_struct.fields {
Fields::Unnamed(ref fields) => {
(DeriveType::Unnamed, unnamed_to_vec(fields), vec![])
}
Fields::Named(ref fields) => {
(DeriveType::Named, named_to_vec(fields), vec![])
}
Fields::Unit => (DeriveType::Named, vec![], vec![]),
},
Data::Enum(ref data_enum) => (
DeriveType::Enum,
vec![],
data_enum.variants.iter().collect(),
),
Data::Union(_) => {
panic!(format!("can not derive({}) for union", trait_name))
}
};
let attrs: Vec<_> = if derive_type == DeriveType::Enum {
variants.iter().map(|v| &v.attrs).collect()
} else {
fields.iter().map(|f| &f.attrs).collect()
};
let (allowed_attr_params_outer, allowed_attr_params_inner) =
if derive_type == DeriveType::Enum {
(&allowed_attr_params.enum_, &allowed_attr_params.variant)
} else {
(&allowed_attr_params.struct_, &allowed_attr_params.field)
};
let struct_meta_info =
get_meta_info(&trait_attr, &input.attrs, allowed_attr_params_outer)?;
let meta_infos: Result<Vec<_>> = attrs
.iter()
.map(|attrs| get_meta_info(&trait_attr, attrs, allowed_attr_params_inner))
.collect();
let meta_infos = meta_infos?;
let first_match = meta_infos
.iter()
.filter_map(|info| info.enabled.map(|_| info))
.next();
let defaults = struct_meta_info.to_full(FullMetaInfo {
enabled: first_match.map_or(true, |info| !info.enabled.unwrap()),
forward: false,
owned: first_match.map_or(true, |info| {
info.owned.is_none() && info.ref_.is_none() || info.ref_mut.is_none()
}),
ref_: false,
ref_mut: false,
info: MetaInfo::default(),
});
let full_meta_infos: Vec<_> = meta_infos
.iter()
.map(|info| info.to_full(defaults))
.collect();
let variant_states: Result<Vec<_>> = if derive_type == DeriveType::Enum {
variants
.iter()
.zip(full_meta_infos.iter().copied())
.map(|(variant, info)| {
State::from_variant(
input,
trait_name,
trait_module.clone(),
trait_attr.clone(),
allowed_attr_params.clone(),
variant,
info,
)
})
.collect()
} else {
Ok(vec![])
};
let generics = add_extra_ty_param_bound(&input.generics, &trait_path);
Ok(State {
input,
trait_name,
trait_ident,
method_ident,
trait_module,
trait_path,
trait_path_params: vec![],
trait_attr,
fields,
variants,
variant_states: variant_states?,
variant: None,
derive_type,
generics,
full_meta_infos,
default_info: defaults,
})
}
pub fn from_variant<'arg_input>(
input: &'arg_input DeriveInput,
trait_name: &'static str,
trait_module: TokenStream,
trait_attr: String,
allowed_attr_params: AttrParams,
variant: &'arg_input Variant,
default_info: FullMetaInfo,
) -> Result<State<'arg_input>> {
let trait_name = trait_name.trim_end_matches("ToInner");
let trait_ident = Ident::new(trait_name, Span::call_site());
let method_ident = Ident::new(&trait_attr, Span::call_site());
let trait_path = quote!(#trait_module::#trait_ident);
let (derive_type, fields): (_, Vec<_>) = match variant.fields {
Fields::Unnamed(ref fields) => {
(DeriveType::Unnamed, unnamed_to_vec(fields))
}
Fields::Named(ref fields) => (DeriveType::Named, named_to_vec(fields)),
Fields::Unit => (DeriveType::Named, vec![]),
};
let meta_infos: Result<Vec<_>> = fields
.iter()
.map(|f| &f.attrs)
.map(|attrs| get_meta_info(&trait_attr, attrs, &allowed_attr_params.field))
.collect();
let meta_infos = meta_infos?;
let full_meta_infos: Vec<_> = meta_infos
.iter()
.map(|info| info.to_full(default_info))
.collect();
let generics = add_extra_ty_param_bound(&input.generics, &trait_path);
Ok(State {
input,
trait_name,
trait_module,
trait_path,
trait_path_params: vec![],
trait_attr,
trait_ident,
method_ident,
fields,
variants: vec![],
variant_states: vec![],
variant: Some(variant),
derive_type,
generics,
full_meta_infos,
default_info,
})
}
pub fn add_trait_path_type_param(&mut self, param: TokenStream) {
self.trait_path_params.push(param);
}
pub fn assert_single_enabled_field<'state>(
&'state self,
) -> SingleFieldData<'input, 'state> {
if self.derive_type == DeriveType::Enum {
panic_one_field(self.trait_name, &self.trait_attr);
}
let data = self.enabled_fields_data();
if data.fields.len() != 1 {
panic_one_field(self.trait_name, &self.trait_attr);
};
SingleFieldData {
input_type: data.input_type,
field: data.fields[0],
field_type: data.field_types[0],
member: data.members[0].clone(),
info: data.infos[0],
field_ident: data.field_idents[0].clone(),
trait_path: data.trait_path,
trait_path_with_params: data.trait_path_with_params.clone(),
casted_trait: data.casted_traits[0].clone(),
impl_generics: data.impl_generics.clone(),
ty_generics: data.ty_generics.clone(),
where_clause: data.where_clause,
multi_field_data: data,
}
}
pub fn enabled_fields_data<'state>(&'state self) -> MultiFieldData<'input, 'state> {
if self.derive_type == DeriveType::Enum {
panic!(format!("can not derive({}) for enum", self.trait_name))
}
let fields = self.enabled_fields();
let field_idents = self.enabled_fields_idents();
let field_indexes = self.enabled_fields_indexes();
let field_types: Vec<_> = fields.iter().map(|f| &f.ty).collect();
let members: Vec<_> = field_idents
.iter()
.map(|ident| quote!(self.#ident))
.collect();
let trait_path = &self.trait_path;
let trait_path_with_params = if !self.trait_path_params.is_empty() {
let params = self.trait_path_params.iter();
quote!(#trait_path<#(#params),*>)
} else {
self.trait_path.clone()
};
let casted_traits: Vec<_> = field_types
.iter()
.map(|field_type| quote!(<#field_type as #trait_path_with_params>))
.collect();
let (impl_generics, ty_generics, where_clause) = self.generics.split_for_impl();
let input_type = &self.input.ident;
let (variant_name, variant_type) = self.variant.map_or_else(
|| (None, quote!(#input_type)),
|v| {
let variant_name = &v.ident;
(Some(variant_name), quote!(#input_type::#variant_name))
},
);
MultiFieldData {
input_type,
variant_type,
variant_name,
variant_info: self.default_info,
fields,
field_types,
field_indexes,
members,
infos: self.enabled_infos(),
field_idents,
method_ident: &self.method_ident,
trait_path,
trait_path_with_params,
casted_traits,
impl_generics,
ty_generics,
where_clause,
state: self,
}
}
pub fn enabled_variant_data<'state>(
&'state self,
) -> MultiVariantData<'input, 'state> {
if self.derive_type != DeriveType::Enum {
panic!(format!("can only derive({}) for enum", self.trait_name))
}
let variants = self.enabled_variants();
let trait_path = &self.trait_path;
let (impl_generics, ty_generics, where_clause) = self.generics.split_for_impl();
MultiVariantData {
input_type: &self.input.ident,
variants,
variant_states: self.enabled_variant_states(),
infos: self.enabled_infos(),
trait_path,
impl_generics,
ty_generics,
where_clause,
}
}
fn enabled_variants(&self) -> Vec<&'input Variant> {
self.variants
.iter()
.zip(self.full_meta_infos.iter().map(|info| info.enabled))
.filter(|(_, ig)| *ig)
.map(|(v, _)| *v)
.collect()
}
fn enabled_variant_states(&self) -> Vec<&State<'input>> {
self.variant_states
.iter()
.zip(self.full_meta_infos.iter().map(|info| info.enabled))
.filter(|(_, ig)| *ig)
.map(|(v, _)| v)
.collect()
}
pub fn enabled_fields(&self) -> Vec<&'input Field> {
self.fields
.iter()
.zip(self.full_meta_infos.iter().map(|info| info.enabled))
.filter(|(_, ig)| *ig)
.map(|(f, _)| *f)
.collect()
}
fn field_idents(&self) -> Vec<TokenStream> {
if self.derive_type == DeriveType::Named {
self.fields
.iter()
.map(|f| {
f.ident
.as_ref()
.expect("Tried to get field names of a tuple struct")
.to_token_stream()
})
.collect()
} else {
let count = self.fields.len();
(0..count)
.map(|i| Index::from(i).to_token_stream())
.collect()
}
}
fn enabled_fields_idents(&self) -> Vec<TokenStream> {
self.field_idents()
.into_iter()
.zip(self.full_meta_infos.iter().map(|info| info.enabled))
.filter(|(_, ig)| *ig)
.map(|(f, _)| f)
.collect()
}
fn enabled_fields_indexes(&self) -> Vec<usize> {
self.full_meta_infos
.iter()
.map(|info| info.enabled)
.enumerate()
.filter(|(_, ig)| *ig)
.map(|(i, _)| i)
.collect()
}
fn enabled_infos(&self) -> Vec<FullMetaInfo> {
self.full_meta_infos
.iter()
.filter(|info| info.enabled)
.copied()
.collect()
}
}
#[derive(Clone)]
pub struct SingleFieldData<'input, 'state> {
pub input_type: &'input Ident,
pub field: &'input Field,
pub field_type: &'input Type,
pub field_ident: TokenStream,
pub member: TokenStream,
pub info: FullMetaInfo,
pub trait_path: &'state TokenStream,
pub trait_path_with_params: TokenStream,
pub casted_trait: TokenStream,
pub impl_generics: ImplGenerics<'state>,
pub ty_generics: TypeGenerics<'state>,
pub where_clause: Option<&'state WhereClause>,
multi_field_data: MultiFieldData<'input, 'state>,
}
#[derive(Clone)]
pub struct MultiFieldData<'input, 'state> {
pub input_type: &'input Ident,
pub variant_type: TokenStream,
pub variant_name: Option<&'input Ident>,
pub variant_info: FullMetaInfo,
pub fields: Vec<&'input Field>,
pub field_types: Vec<&'input Type>,
pub field_idents: Vec<TokenStream>,
pub field_indexes: Vec<usize>,
pub members: Vec<TokenStream>,
pub infos: Vec<FullMetaInfo>,
pub method_ident: &'state Ident,
pub trait_path: &'state TokenStream,
pub trait_path_with_params: TokenStream,
pub casted_traits: Vec<TokenStream>,
pub impl_generics: ImplGenerics<'state>,
pub ty_generics: TypeGenerics<'state>,
pub where_clause: Option<&'state WhereClause>,
pub state: &'state State<'input>,
}
pub struct MultiVariantData<'input, 'state> {
pub input_type: &'input Ident,
pub variants: Vec<&'input Variant>,
pub variant_states: Vec<&'state State<'input>>,
pub infos: Vec<FullMetaInfo>,
pub trait_path: &'state TokenStream,
pub impl_generics: ImplGenerics<'state>,
pub ty_generics: TypeGenerics<'state>,
pub where_clause: Option<&'state WhereClause>,
}
impl<'input, 'state> MultiFieldData<'input, 'state> {
pub fn initializer<T: ToTokens>(&self, initializers: &[T]) -> TokenStream {
let MultiFieldData {
variant_type,
field_idents,
..
} = self;
if self.state.derive_type == DeriveType::Named {
quote!(#variant_type{#(#field_idents: #initializers),*})
} else {
quote!(#variant_type(#(#initializers),*))
}
}
pub fn matcher<T: ToTokens>(
&self,
indexes: &[usize],
bindings: &[T],
) -> TokenStream {
let MultiFieldData { variant_type, .. } = self;
let full_bindings = (0..self.state.fields.len()).map(|i| {
indexes.iter().position(|index| i == *index).map_or_else(
|| quote!(_),
|found_index| bindings[found_index].to_token_stream(),
)
});
if self.state.derive_type == DeriveType::Named {
let field_idents = self.state.field_idents();
quote!(#variant_type{#(#field_idents: #full_bindings),*})
} else {
quote!(#variant_type(#(#full_bindings),*))
}
}
}
impl<'input, 'state> SingleFieldData<'input, 'state> {
pub fn initializer<T: ToTokens>(&self, initializers: &[T]) -> TokenStream {
self.multi_field_data.initializer(initializers)
}
}
fn get_meta_info(
trait_attr: &str,
attrs: &[Attribute],
allowed_attr_params: &[&'static str],
) -> Result<MetaInfo> {
let mut it = attrs
.iter()
.filter_map(|m| m.parse_meta().ok())
.filter(|m| {
if let Some(ident) = m.path().segments.first().map(|p| &p.ident) {
ident == trait_attr
} else {
false
}
});
let mut info = MetaInfo {
enabled: None,
forward: None,
owned: None,
ref_: None,
ref_mut: None,
};
let meta = if let Some(meta) = it.next() {
meta
} else {
return Ok(info);
};
if allowed_attr_params.is_empty() {
return Err(Error::new(meta.span(), "Attribute is not allowed here"));
}
info.enabled = Some(true);
if let Some(meta2) = it.next() {
return Err(Error::new(
meta2.span(),
"Only a single attribute is allowed",
));
}
let list = match meta.clone() {
Meta::Path(_) => {
if allowed_attr_params.contains(&"ignore") {
return Ok(info);
} else {
return Err(Error::new(meta.span(), format!("Empty attribute is not allowed, add one of the following parameters: {}",
allowed_attr_params.join(", ")
)));
}
}
Meta::List(list) => list,
_ => {
return Err(Error::new(meta.span(), "Attribute format not supported1"));
}
};
for element in list.nested.into_iter() {
let nested_meta = if let NestedMeta::Meta(meta) = element {
meta
} else {
return Err(Error::new(meta.span(), "Attribute format not supported3"));
};
if let Meta::Path(_) = nested_meta {
} else {
return Err(Error::new(meta.span(), "Attribute format not supported4"));
}
let ident = if let Some(ident) =
nested_meta.path().segments.first().map(|p| &p.ident)
{
ident
} else {
return Err(Error::new(meta.span(), "Attribute format not supported5"));
};
let str_ident: &str = &ident.to_string();
if !allowed_attr_params.contains(&str_ident) {
return Err(Error::new(
ident.span(),
format!(
"Attribute parameter not supported. Supported attribute parameters are: {}",
allowed_attr_params.join(", ")
),
));
}
match str_ident {
"ignore" => {
info.enabled = Some(false);
}
"forward" => {
info.forward = Some(true);
}
"owned" => {
info.owned = Some(true);
}
"ref" => {
info.ref_ = Some(true);
}
"ref_mut" => {
info.ref_mut = Some(true);
}
_ => {
return Err(Error::new(meta.span(), "Attribute format not supported7"));
}
}
}
Ok(info)
}
#[derive(Copy, Clone, Debug, Default)]
pub struct FullMetaInfo {
pub enabled: bool,
pub forward: bool,
pub owned: bool,
pub ref_: bool,
pub ref_mut: bool,
pub info: MetaInfo,
}
#[derive(Copy, Clone, Debug, Default)]
pub struct MetaInfo {
pub enabled: Option<bool>,
pub forward: Option<bool>,
pub owned: Option<bool>,
pub ref_: Option<bool>,
pub ref_mut: Option<bool>,
}
impl MetaInfo {
fn to_full(self, defaults: FullMetaInfo) -> FullMetaInfo {
let info = self;
FullMetaInfo {
enabled: self.enabled.unwrap_or(defaults.enabled),
forward: self.forward.unwrap_or(defaults.forward),
owned: self.owned.unwrap_or(defaults.owned),
ref_: self.ref_.unwrap_or(defaults.ref_),
ref_mut: self.ref_mut.unwrap_or(defaults.ref_mut),
info,
}
}
}
impl FullMetaInfo {
pub fn ref_types(self) -> Vec<RefType> {
let mut ref_types = vec![];
if self.owned {
ref_types.push(RefType::No);
}
if self.ref_ {
ref_types.push(RefType::Ref);
}
if self.ref_mut {
ref_types.push(RefType::Mut);
}
ref_types
}
}