Add bounds for fields in derive macro

This commit is contained in:
hkalbasi 2023-04-06 16:14:00 +03:30
parent e739c999cd
commit 0241b52dad
3 changed files with 66 additions and 27 deletions

View File

@ -16,7 +16,7 @@ fn test_copy_expand_simple() {
#[derive(Copy)] #[derive(Copy)]
struct Foo; struct Foo;
impl < > core::marker::Copy for Foo< > {}"#]], impl < > core::marker::Copy for Foo< > where {}"#]],
); );
} }
@ -41,7 +41,7 @@ fn test_copy_expand_in_core() {
#[derive(Copy)] #[derive(Copy)]
struct Foo; struct Foo;
impl < > crate ::marker::Copy for Foo< > {}"#]], impl < > crate ::marker::Copy for Foo< > where {}"#]],
); );
} }
@ -57,7 +57,7 @@ fn test_copy_expand_with_type_params() {
#[derive(Copy)] #[derive(Copy)]
struct Foo<A, B>; struct Foo<A, B>;
impl <T0: core::marker::Copy, T1: core::marker::Copy, > core::marker::Copy for Foo<T0, T1, > {}"#]], impl <A: core::marker::Copy, B: core::marker::Copy, > core::marker::Copy for Foo<A, B, > where {}"#]],
); );
} }
@ -74,7 +74,7 @@ fn test_copy_expand_with_lifetimes() {
#[derive(Copy)] #[derive(Copy)]
struct Foo<A, B, 'a, 'b>; struct Foo<A, B, 'a, 'b>;
impl <T0: core::marker::Copy, T1: core::marker::Copy, > core::marker::Copy for Foo<T0, T1, > {}"#]], impl <A: core::marker::Copy, B: core::marker::Copy, > core::marker::Copy for Foo<A, B, > where {}"#]],
); );
} }
@ -90,7 +90,7 @@ fn test_clone_expand() {
#[derive(Clone)] #[derive(Clone)]
struct Foo<A, B>; struct Foo<A, B>;
impl <T0: core::clone::Clone, T1: core::clone::Clone, > core::clone::Clone for Foo<T0, T1, > {}"#]], impl <A: core::clone::Clone, B: core::clone::Clone, > core::clone::Clone for Foo<A, B, > where {}"#]],
); );
} }
@ -106,6 +106,6 @@ fn test_clone_expand_with_const_generics() {
#[derive(Clone)] #[derive(Clone)]
struct Foo<const X: usize, T>(u32); struct Foo<const X: usize, T>(u32);
impl <const T0: usize, T1: core::clone::Clone, > core::clone::Clone for Foo<T0, T1, > {}"#]], impl <const X: usize, T: core::clone::Clone, > core::clone::Clone for Foo<X, T, > where u32: core::clone::Clone, {}"#]],
); );
} }

View File

@ -1,11 +1,12 @@
//! Builtin derives. //! Builtin derives.
use base_db::{CrateOrigin, LangCrateOrigin}; use base_db::{CrateOrigin, LangCrateOrigin};
use either::Either;
use tracing::debug; use tracing::debug;
use crate::tt::{self, TokenId}; use crate::tt::{self, TokenId};
use syntax::{ use syntax::{
ast::{self, AstNode, HasGenericParams, HasModuleItem, HasName}, ast::{self, AstNode, HasGenericParams, HasModuleItem, HasName, HasTypeBounds},
match_ast, match_ast,
}; };
@ -60,8 +61,11 @@ pub fn find_builtin_derive(ident: &name::Name) -> Option<BuiltinDeriveExpander>
struct BasicAdtInfo { struct BasicAdtInfo {
name: tt::Ident, name: tt::Ident,
/// `Some(ty)` if it's a const param of type `ty`, `None` if it's a type param. /// first field is the name, and
param_types: Vec<Option<tt::Subtree>>, /// second field is `Some(ty)` if it's a const param of type `ty`, `None` if it's a type param.
/// third fields is where bounds, if any
param_types: Vec<(tt::Subtree, Option<tt::Subtree>, Option<tt::Subtree>)>,
field_types: Vec<tt::Subtree>,
} }
fn parse_adt(tt: &tt::Subtree) -> Result<BasicAdtInfo, ExpandError> { fn parse_adt(tt: &tt::Subtree) -> Result<BasicAdtInfo, ExpandError> {
@ -75,17 +79,34 @@ fn parse_adt(tt: &tt::Subtree) -> Result<BasicAdtInfo, ExpandError> {
ExpandError::Other("no item found".into()) ExpandError::Other("no item found".into())
})?; })?;
let node = item.syntax(); let node = item.syntax();
let (name, params) = match_ast! { let (name, params, fields) = match_ast! {
match node { match node {
ast::Struct(it) => (it.name(), it.generic_param_list()), ast::Struct(it) => {
ast::Enum(it) => (it.name(), it.generic_param_list()), (it.name(), it.generic_param_list(), it.field_list().into_iter().collect::<Vec<_>>())
ast::Union(it) => (it.name(), it.generic_param_list()), },
ast::Enum(it) => (it.name(), it.generic_param_list(), it.variant_list().into_iter().flat_map(|x| x.variants()).filter_map(|x| x.field_list()).collect()),
ast::Union(it) => (it.name(), it.generic_param_list(), it.record_field_list().into_iter().map(|x| ast::FieldList::RecordFieldList(x)).collect()),
_ => { _ => {
debug!("unexpected node is {:?}", node); debug!("unexpected node is {:?}", node);
return Err(ExpandError::Other("expected struct, enum or union".into())) return Err(ExpandError::Other("expected struct, enum or union".into()))
}, },
} }
}; };
let field_types = fields
.into_iter()
.flat_map(|f| match f {
ast::FieldList::RecordFieldList(x) => Either::Left(
x.fields()
.filter_map(|x| x.ty())
.map(|x| mbe::syntax_node_to_token_tree(x.syntax()).0),
),
ast::FieldList::TupleFieldList(x) => Either::Right(
x.fields()
.filter_map(|x| x.ty())
.map(|x| mbe::syntax_node_to_token_tree(x.syntax()).0),
),
})
.collect::<Vec<_>>();
let name = name.ok_or_else(|| { let name = name.ok_or_else(|| {
debug!("parsed item has no name"); debug!("parsed item has no name");
ExpandError::Other("missing name".into()) ExpandError::Other("missing name".into())
@ -97,7 +118,17 @@ fn parse_adt(tt: &tt::Subtree) -> Result<BasicAdtInfo, ExpandError> {
.into_iter() .into_iter()
.flat_map(|param_list| param_list.type_or_const_params()) .flat_map(|param_list| param_list.type_or_const_params())
.map(|param| { .map(|param| {
if let ast::TypeOrConstParam::Const(param) = param { let name = param
.name()
.map(|x| mbe::syntax_node_to_token_tree(x.syntax()).0)
.unwrap_or_else(tt::Subtree::empty);
let bounds = match &param {
ast::TypeOrConstParam::Type(x) => {
x.type_bound_list().map(|x| mbe::syntax_node_to_token_tree(x.syntax()).0)
}
ast::TypeOrConstParam::Const(_) => None,
};
let ty = if let ast::TypeOrConstParam::Const(param) = param {
let ty = param let ty = param
.ty() .ty()
.map(|ty| mbe::syntax_node_to_token_tree(ty.syntax()).0) .map(|ty| mbe::syntax_node_to_token_tree(ty.syntax()).0)
@ -105,10 +136,11 @@ fn parse_adt(tt: &tt::Subtree) -> Result<BasicAdtInfo, ExpandError> {
Some(ty) Some(ty)
} else { } else {
None None
} };
(name, ty, bounds)
}) })
.collect(); .collect();
Ok(BasicAdtInfo { name: name_token, param_types }) Ok(BasicAdtInfo { name: name_token, param_types, field_types })
} }
fn expand_simple_derive(tt: &tt::Subtree, trait_path: tt::Subtree) -> ExpandResult<tt::Subtree> { fn expand_simple_derive(tt: &tt::Subtree, trait_path: tt::Subtree) -> ExpandResult<tt::Subtree> {
@ -116,16 +148,16 @@ fn expand_simple_derive(tt: &tt::Subtree, trait_path: tt::Subtree) -> ExpandResu
Ok(info) => info, Ok(info) => info,
Err(e) => return ExpandResult::with_err(tt::Subtree::empty(), e), Err(e) => return ExpandResult::with_err(tt::Subtree::empty(), e),
}; };
let mut where_block = vec![];
let (params, args): (Vec<_>, Vec<_>) = info let (params, args): (Vec<_>, Vec<_>) = info
.param_types .param_types
.into_iter() .into_iter()
.enumerate() .map(|(ident, param_ty, bound)| {
.map(|(idx, param_ty)| {
let ident = tt::Leaf::Ident(tt::Ident {
span: tt::TokenId::unspecified(),
text: format!("T{idx}").into(),
});
let ident_ = ident.clone(); let ident_ = ident.clone();
if let Some(b) = bound {
let ident = ident.clone();
where_block.push(quote! { #ident : #b , });
}
if let Some(ty) = param_ty { if let Some(ty) = param_ty {
(quote! { const #ident : #ty , }, quote! { #ident_ , }) (quote! { const #ident : #ty , }, quote! { #ident_ , })
} else { } else {
@ -134,9 +166,16 @@ fn expand_simple_derive(tt: &tt::Subtree, trait_path: tt::Subtree) -> ExpandResu
} }
}) })
.unzip(); .unzip();
where_block.extend(info.field_types.iter().map(|x| {
let x = x.clone();
let bound = trait_path.clone();
quote! { #x : #bound , }
}));
let name = info.name; let name = info.name;
let expanded = quote! { let expanded = quote! {
impl < ##params > #trait_path for #name < ##args > {} impl < ##params > #trait_path for #name < ##args > where ##where_block {}
}; };
ExpandResult::ok(expanded) ExpandResult::ok(expanded)
} }

View File

@ -471,7 +471,7 @@ struct Foo {}
"#, "#,
expect![[r#" expect![[r#"
Clone Clone
impl < >core::clone::Clone for Foo< >{} impl < >core::clone::Clone for Foo< >where{}
"#]], "#]],
); );
} }
@ -488,7 +488,7 @@ struct Foo {}
"#, "#,
expect![[r#" expect![[r#"
Copy Copy
impl < >core::marker::Copy for Foo< >{} impl < >core::marker::Copy for Foo< >where{}
"#]], "#]],
); );
} }
@ -504,7 +504,7 @@ struct Foo {}
"#, "#,
expect![[r#" expect![[r#"
Copy Copy
impl < >core::marker::Copy for Foo< >{} impl < >core::marker::Copy for Foo< >where{}
"#]], "#]],
); );
check( check(
@ -516,7 +516,7 @@ struct Foo {}
"#, "#,
expect![[r#" expect![[r#"
Clone Clone
impl < >core::clone::Clone for Foo< >{} impl < >core::clone::Clone for Foo< >where{}
"#]], "#]],
); );
} }