mirror of
https://github.com/rust-lang/rust-analyzer.git
synced 2025-10-30 19:49:36 +00:00
1540 lines
54 KiB
Rust
1540 lines
54 KiB
Rust
//! Builtin derives.
|
|
|
|
use intern::sym;
|
|
use itertools::{izip, Itertools};
|
|
use parser::SyntaxKind;
|
|
use rustc_hash::FxHashSet;
|
|
use span::{Edition, MacroCallId, Span, SyntaxContextId};
|
|
use stdx::never;
|
|
use syntax_bridge::DocCommentDesugarMode;
|
|
use tracing::debug;
|
|
|
|
use crate::{
|
|
builtin::quote::{dollar_crate, quote},
|
|
db::ExpandDatabase,
|
|
hygiene::span_with_def_site_ctxt,
|
|
name::{self, AsName, Name},
|
|
span_map::ExpansionSpanMap,
|
|
tt, ExpandError, ExpandResult,
|
|
};
|
|
use syntax::{
|
|
ast::{
|
|
self, edit_in_place::GenericParamsOwnerEdit, make, AstNode, FieldList, HasAttrs,
|
|
HasGenericArgs, HasGenericParams, HasModuleItem, HasName, HasTypeBounds,
|
|
},
|
|
ted,
|
|
};
|
|
|
|
macro_rules! register_builtin {
|
|
( $($trait:ident => $expand:ident),* ) => {
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
pub enum BuiltinDeriveExpander {
|
|
$($trait),*
|
|
}
|
|
|
|
impl BuiltinDeriveExpander {
|
|
pub fn expander(&self) -> fn(&dyn ExpandDatabase, Span, &tt::TopSubtree) -> ExpandResult<tt::TopSubtree> {
|
|
match *self {
|
|
$( BuiltinDeriveExpander::$trait => $expand, )*
|
|
}
|
|
}
|
|
|
|
fn find_by_name(name: &name::Name) -> Option<Self> {
|
|
match name {
|
|
$( id if id == &sym::$trait => Some(BuiltinDeriveExpander::$trait), )*
|
|
_ => None,
|
|
}
|
|
}
|
|
}
|
|
|
|
};
|
|
}
|
|
|
|
impl BuiltinDeriveExpander {
|
|
pub fn expand(
|
|
&self,
|
|
db: &dyn ExpandDatabase,
|
|
id: MacroCallId,
|
|
tt: &tt::TopSubtree,
|
|
span: Span,
|
|
) -> ExpandResult<tt::TopSubtree> {
|
|
let span = span_with_def_site_ctxt(db, span, id, Edition::CURRENT);
|
|
self.expander()(db, span, tt)
|
|
}
|
|
}
|
|
|
|
register_builtin! {
|
|
Copy => copy_expand,
|
|
Clone => clone_expand,
|
|
Default => default_expand,
|
|
Debug => debug_expand,
|
|
Hash => hash_expand,
|
|
Ord => ord_expand,
|
|
PartialOrd => partial_ord_expand,
|
|
Eq => eq_expand,
|
|
PartialEq => partial_eq_expand,
|
|
CoercePointee => coerce_pointee_expand
|
|
}
|
|
|
|
pub fn find_builtin_derive(ident: &name::Name) -> Option<BuiltinDeriveExpander> {
|
|
BuiltinDeriveExpander::find_by_name(ident)
|
|
}
|
|
|
|
#[derive(Clone, Copy)]
|
|
enum HasDefault {
|
|
Yes,
|
|
No,
|
|
}
|
|
|
|
#[derive(Clone)]
|
|
enum VariantShape {
|
|
Struct(Vec<(tt::Ident, HasDefault)>),
|
|
Tuple(usize),
|
|
Unit,
|
|
}
|
|
|
|
fn tuple_field_iterator(span: Span, n: usize) -> impl Iterator<Item = tt::Ident> {
|
|
(0..n).map(move |it| tt::Ident::new(&format!("f{it}"), span))
|
|
}
|
|
|
|
impl VariantShape {
|
|
fn as_pattern(&self, path: tt::TopSubtree, span: Span) -> tt::TopSubtree {
|
|
self.as_pattern_map(path, span, |it| quote!(span => #it))
|
|
}
|
|
|
|
fn field_names(&self, span: Span) -> Vec<tt::Ident> {
|
|
match self {
|
|
VariantShape::Struct(s) => s.iter().map(|(ident, _)| ident.clone()).collect(),
|
|
VariantShape::Tuple(n) => tuple_field_iterator(span, *n).collect(),
|
|
VariantShape::Unit => vec![],
|
|
}
|
|
}
|
|
|
|
fn as_pattern_map(
|
|
&self,
|
|
path: tt::TopSubtree,
|
|
span: Span,
|
|
field_map: impl Fn(&tt::Ident) -> tt::TopSubtree,
|
|
) -> tt::TopSubtree {
|
|
match self {
|
|
VariantShape::Struct(fields) => {
|
|
let fields = fields.iter().map(|(it, _)| {
|
|
let mapped = field_map(it);
|
|
quote! {span => #it : #mapped , }
|
|
});
|
|
quote! {span =>
|
|
#path { ##fields }
|
|
}
|
|
}
|
|
&VariantShape::Tuple(n) => {
|
|
let fields = tuple_field_iterator(span, n).map(|it| {
|
|
let mapped = field_map(&it);
|
|
quote! {span =>
|
|
#mapped ,
|
|
}
|
|
});
|
|
quote! {span =>
|
|
#path ( ##fields )
|
|
}
|
|
}
|
|
VariantShape::Unit => path,
|
|
}
|
|
}
|
|
|
|
fn default_expand(
|
|
&self,
|
|
path: tt::TopSubtree,
|
|
span: Span,
|
|
field_map: impl Fn(&tt::Ident) -> tt::TopSubtree,
|
|
) -> tt::TopSubtree {
|
|
match self {
|
|
VariantShape::Struct(fields) => {
|
|
let contains_default = fields.iter().any(|it| matches!(it.1, HasDefault::Yes));
|
|
let fields = fields
|
|
.iter()
|
|
.filter_map(|(it, has_default)| match has_default {
|
|
HasDefault::Yes => None,
|
|
HasDefault::No => Some(it),
|
|
})
|
|
.map(|it| {
|
|
let mapped = field_map(it);
|
|
quote! {span => #it : #mapped , }
|
|
});
|
|
if contains_default {
|
|
let mut double_dots =
|
|
tt::TopSubtreeBuilder::new(tt::Delimiter::invisible_spanned(span));
|
|
double_dots.push(tt::Leaf::Punct(tt::Punct {
|
|
char: '.',
|
|
spacing: tt::Spacing::Joint,
|
|
span,
|
|
}));
|
|
double_dots.push(tt::Leaf::Punct(tt::Punct {
|
|
char: '.',
|
|
spacing: tt::Spacing::Alone,
|
|
span,
|
|
}));
|
|
let double_dots = double_dots.build();
|
|
quote! {span =>
|
|
#path { ##fields #double_dots }
|
|
}
|
|
} else {
|
|
quote! {span =>
|
|
#path { ##fields }
|
|
}
|
|
}
|
|
}
|
|
&VariantShape::Tuple(n) => {
|
|
let fields = tuple_field_iterator(span, n).map(|it| {
|
|
let mapped = field_map(&it);
|
|
quote! {span =>
|
|
#mapped ,
|
|
}
|
|
});
|
|
quote! {span =>
|
|
#path ( ##fields )
|
|
}
|
|
}
|
|
VariantShape::Unit => path,
|
|
}
|
|
}
|
|
|
|
fn from(
|
|
call_site: Span,
|
|
tm: &ExpansionSpanMap,
|
|
value: Option<FieldList>,
|
|
) -> Result<Self, ExpandError> {
|
|
let r = match value {
|
|
None => VariantShape::Unit,
|
|
Some(FieldList::RecordFieldList(it)) => VariantShape::Struct(
|
|
it.fields()
|
|
.map(|it| {
|
|
(
|
|
it.name(),
|
|
if it.expr().is_some() { HasDefault::Yes } else { HasDefault::No },
|
|
)
|
|
})
|
|
.map(|(it, has_default)| {
|
|
name_to_token(call_site, tm, it).map(|ident| (ident, has_default))
|
|
})
|
|
.collect::<Result<_, _>>()?,
|
|
),
|
|
Some(FieldList::TupleFieldList(it)) => VariantShape::Tuple(it.fields().count()),
|
|
};
|
|
Ok(r)
|
|
}
|
|
}
|
|
|
|
#[derive(Clone)]
|
|
enum AdtShape {
|
|
Struct(VariantShape),
|
|
Enum { variants: Vec<(tt::Ident, VariantShape)>, default_variant: Option<usize> },
|
|
Union,
|
|
}
|
|
|
|
impl AdtShape {
|
|
fn as_pattern(&self, span: Span, name: &tt::Ident) -> Vec<tt::TopSubtree> {
|
|
self.as_pattern_map(name, |it| quote!(span =>#it), span)
|
|
}
|
|
|
|
fn field_names(&self, span: Span) -> Vec<Vec<tt::Ident>> {
|
|
match self {
|
|
AdtShape::Struct(s) => {
|
|
vec![s.field_names(span)]
|
|
}
|
|
AdtShape::Enum { variants, .. } => {
|
|
variants.iter().map(|(_, fields)| fields.field_names(span)).collect()
|
|
}
|
|
AdtShape::Union => {
|
|
never!("using fields of union in derive is always wrong");
|
|
vec![]
|
|
}
|
|
}
|
|
}
|
|
|
|
fn as_pattern_map(
|
|
&self,
|
|
name: &tt::Ident,
|
|
field_map: impl Fn(&tt::Ident) -> tt::TopSubtree,
|
|
span: Span,
|
|
) -> Vec<tt::TopSubtree> {
|
|
match self {
|
|
AdtShape::Struct(s) => {
|
|
vec![s.as_pattern_map(quote! {span => #name }, span, field_map)]
|
|
}
|
|
AdtShape::Enum { variants, .. } => variants
|
|
.iter()
|
|
.map(|(v, fields)| {
|
|
fields.as_pattern_map(quote! {span => #name :: #v }, span, &field_map)
|
|
})
|
|
.collect(),
|
|
AdtShape::Union => {
|
|
never!("pattern matching on union is always wrong");
|
|
vec![quote! {span => un }]
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Clone)]
|
|
struct BasicAdtInfo {
|
|
name: tt::Ident,
|
|
shape: AdtShape,
|
|
/// first field is the name, and
|
|
/// 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<AdtParam>,
|
|
where_clause: Vec<tt::TopSubtree>,
|
|
associated_types: Vec<tt::TopSubtree>,
|
|
}
|
|
|
|
#[derive(Clone)]
|
|
struct AdtParam {
|
|
name: tt::TopSubtree,
|
|
/// `None` if this is a type parameter.
|
|
const_ty: Option<tt::TopSubtree>,
|
|
bounds: Option<tt::TopSubtree>,
|
|
}
|
|
|
|
// FIXME: This whole thing needs a refactor. Each derive requires its special values, and the result is a mess.
|
|
fn parse_adt(
|
|
db: &dyn ExpandDatabase,
|
|
tt: &tt::TopSubtree,
|
|
call_site: Span,
|
|
) -> Result<BasicAdtInfo, ExpandError> {
|
|
let (adt, tm) = to_adt_syntax(db, tt, call_site)?;
|
|
parse_adt_from_syntax(&adt, &tm, call_site)
|
|
}
|
|
|
|
fn parse_adt_from_syntax(
|
|
adt: &ast::Adt,
|
|
tm: &span::SpanMap<SyntaxContextId>,
|
|
call_site: Span,
|
|
) -> Result<BasicAdtInfo, ExpandError> {
|
|
let (name, generic_param_list, where_clause, shape) = match &adt {
|
|
ast::Adt::Struct(it) => (
|
|
it.name(),
|
|
it.generic_param_list(),
|
|
it.where_clause(),
|
|
AdtShape::Struct(VariantShape::from(call_site, tm, it.field_list())?),
|
|
),
|
|
ast::Adt::Enum(it) => {
|
|
let default_variant = it
|
|
.variant_list()
|
|
.into_iter()
|
|
.flat_map(|it| it.variants())
|
|
.position(|it| it.attrs().any(|it| it.simple_name() == Some("default".into())));
|
|
(
|
|
it.name(),
|
|
it.generic_param_list(),
|
|
it.where_clause(),
|
|
AdtShape::Enum {
|
|
default_variant,
|
|
variants: it
|
|
.variant_list()
|
|
.into_iter()
|
|
.flat_map(|it| it.variants())
|
|
.map(|it| {
|
|
Ok((
|
|
name_to_token(call_site, tm, it.name())?,
|
|
VariantShape::from(call_site, tm, it.field_list())?,
|
|
))
|
|
})
|
|
.collect::<Result<_, ExpandError>>()?,
|
|
},
|
|
)
|
|
}
|
|
ast::Adt::Union(it) => {
|
|
(it.name(), it.generic_param_list(), it.where_clause(), AdtShape::Union)
|
|
}
|
|
};
|
|
|
|
let mut param_type_set: FxHashSet<Name> = FxHashSet::default();
|
|
let param_types = generic_param_list
|
|
.into_iter()
|
|
.flat_map(|param_list| param_list.type_or_const_params())
|
|
.map(|param| {
|
|
let name = {
|
|
let this = param.name();
|
|
match this {
|
|
Some(it) => {
|
|
param_type_set.insert(it.as_name());
|
|
syntax_bridge::syntax_node_to_token_tree(
|
|
it.syntax(),
|
|
tm,
|
|
call_site,
|
|
DocCommentDesugarMode::ProcMacro,
|
|
)
|
|
}
|
|
None => {
|
|
tt::TopSubtree::empty(::tt::DelimSpan { open: call_site, close: call_site })
|
|
}
|
|
}
|
|
};
|
|
let bounds = match ¶m {
|
|
ast::TypeOrConstParam::Type(it) => it.type_bound_list().map(|it| {
|
|
syntax_bridge::syntax_node_to_token_tree(
|
|
it.syntax(),
|
|
tm,
|
|
call_site,
|
|
DocCommentDesugarMode::ProcMacro,
|
|
)
|
|
}),
|
|
ast::TypeOrConstParam::Const(_) => None,
|
|
};
|
|
let const_ty = if let ast::TypeOrConstParam::Const(param) = param {
|
|
let ty = param
|
|
.ty()
|
|
.map(|ty| {
|
|
syntax_bridge::syntax_node_to_token_tree(
|
|
ty.syntax(),
|
|
tm,
|
|
call_site,
|
|
DocCommentDesugarMode::ProcMacro,
|
|
)
|
|
})
|
|
.unwrap_or_else(|| {
|
|
tt::TopSubtree::empty(::tt::DelimSpan { open: call_site, close: call_site })
|
|
});
|
|
Some(ty)
|
|
} else {
|
|
None
|
|
};
|
|
AdtParam { name, const_ty, bounds }
|
|
})
|
|
.collect();
|
|
|
|
let where_clause = if let Some(w) = where_clause {
|
|
w.predicates()
|
|
.map(|it| {
|
|
syntax_bridge::syntax_node_to_token_tree(
|
|
it.syntax(),
|
|
tm,
|
|
call_site,
|
|
DocCommentDesugarMode::ProcMacro,
|
|
)
|
|
})
|
|
.collect()
|
|
} else {
|
|
vec![]
|
|
};
|
|
|
|
// For a generic parameter `T`, when shorthand associated type `T::Assoc` appears in field
|
|
// types (of any variant for enums), we generate trait bound for it. It sounds reasonable to
|
|
// also generate trait bound for qualified associated type `<T as Trait>::Assoc`, but rustc
|
|
// does not do that for some unknown reason.
|
|
//
|
|
// See the analogous function in rustc [find_type_parameters()] and rust-lang/rust#50730.
|
|
// [find_type_parameters()]: https://github.com/rust-lang/rust/blob/1.70.0/compiler/rustc_builtin_macros/src/deriving/generic/mod.rs#L378
|
|
|
|
// It's cumbersome to deal with the distinct structures of ADTs, so let's just get untyped
|
|
// `SyntaxNode` that contains fields and look for descendant `ast::PathType`s. Of note is that
|
|
// we should not inspect `ast::PathType`s in parameter bounds and where clauses.
|
|
let field_list = match adt {
|
|
ast::Adt::Enum(it) => it.variant_list().map(|list| list.syntax().clone()),
|
|
ast::Adt::Struct(it) => it.field_list().map(|list| list.syntax().clone()),
|
|
ast::Adt::Union(it) => it.record_field_list().map(|list| list.syntax().clone()),
|
|
};
|
|
let associated_types = field_list
|
|
.into_iter()
|
|
.flat_map(|it| it.descendants())
|
|
.filter_map(ast::PathType::cast)
|
|
.filter_map(|p| {
|
|
let name = p.path()?.qualifier()?.as_single_name_ref()?.as_name();
|
|
param_type_set.contains(&name).then_some(p)
|
|
})
|
|
.map(|it| {
|
|
syntax_bridge::syntax_node_to_token_tree(
|
|
it.syntax(),
|
|
tm,
|
|
call_site,
|
|
DocCommentDesugarMode::ProcMacro,
|
|
)
|
|
})
|
|
.collect();
|
|
let name_token = name_to_token(call_site, tm, name)?;
|
|
Ok(BasicAdtInfo { name: name_token, shape, param_types, where_clause, associated_types })
|
|
}
|
|
|
|
fn to_adt_syntax(
|
|
db: &dyn ExpandDatabase,
|
|
tt: &tt::TopSubtree,
|
|
call_site: Span,
|
|
) -> Result<(ast::Adt, span::SpanMap<SyntaxContextId>), ExpandError> {
|
|
let (parsed, tm) = crate::db::token_tree_to_syntax_node(
|
|
db,
|
|
tt,
|
|
crate::ExpandTo::Items,
|
|
parser::Edition::CURRENT_FIXME,
|
|
);
|
|
let macro_items = ast::MacroItems::cast(parsed.syntax_node())
|
|
.ok_or_else(|| ExpandError::other(call_site, "invalid item definition"))?;
|
|
let item =
|
|
macro_items.items().next().ok_or_else(|| ExpandError::other(call_site, "no item found"))?;
|
|
let adt = ast::Adt::cast(item.syntax().clone())
|
|
.ok_or_else(|| ExpandError::other(call_site, "expected struct, enum or union"))?;
|
|
Ok((adt, tm))
|
|
}
|
|
|
|
fn name_to_token(
|
|
call_site: Span,
|
|
token_map: &ExpansionSpanMap,
|
|
name: Option<ast::Name>,
|
|
) -> Result<tt::Ident, ExpandError> {
|
|
let name = name.ok_or_else(|| {
|
|
debug!("parsed item has no name");
|
|
ExpandError::other(call_site, "missing name")
|
|
})?;
|
|
let span = token_map.span_at(name.syntax().text_range().start());
|
|
|
|
let name_token = tt::Ident::new(name.text().as_ref(), span);
|
|
Ok(name_token)
|
|
}
|
|
|
|
/// Given that we are deriving a trait `DerivedTrait` for a type like:
|
|
///
|
|
/// ```ignore (only-for-syntax-highlight)
|
|
/// struct Struct<'a, ..., 'z, A, B: DeclaredTrait, C, ..., Z> where C: WhereTrait {
|
|
/// a: A,
|
|
/// b: B::Item,
|
|
/// b1: <B as DeclaredTrait>::Item,
|
|
/// c1: <C as WhereTrait>::Item,
|
|
/// c2: Option<<C as WhereTrait>::Item>,
|
|
/// ...
|
|
/// }
|
|
/// ```
|
|
///
|
|
/// create an impl like:
|
|
///
|
|
/// ```ignore (only-for-syntax-highlight)
|
|
/// impl<'a, ..., 'z, A, B: DeclaredTrait, C, ... Z> where
|
|
/// C: WhereTrait,
|
|
/// A: DerivedTrait + B1 + ... + BN,
|
|
/// B: DerivedTrait + B1 + ... + BN,
|
|
/// C: DerivedTrait + B1 + ... + BN,
|
|
/// B::Item: DerivedTrait + B1 + ... + BN,
|
|
/// <C as WhereTrait>::Item: DerivedTrait + B1 + ... + BN,
|
|
/// ...
|
|
/// {
|
|
/// ...
|
|
/// }
|
|
/// ```
|
|
///
|
|
/// where B1, ..., BN are the bounds given by `bounds_paths`. Z is a phantom type, and
|
|
/// therefore does not get bound by the derived trait.
|
|
fn expand_simple_derive(
|
|
db: &dyn ExpandDatabase,
|
|
invoc_span: Span,
|
|
tt: &tt::TopSubtree,
|
|
trait_path: tt::TopSubtree,
|
|
make_trait_body: impl FnOnce(&BasicAdtInfo) -> tt::TopSubtree,
|
|
) -> ExpandResult<tt::TopSubtree> {
|
|
let info = match parse_adt(db, tt, invoc_span) {
|
|
Ok(info) => info,
|
|
Err(e) => {
|
|
return ExpandResult::new(
|
|
tt::TopSubtree::empty(tt::DelimSpan { open: invoc_span, close: invoc_span }),
|
|
e,
|
|
)
|
|
}
|
|
};
|
|
ExpandResult::ok(expand_simple_derive_with_parsed(
|
|
invoc_span,
|
|
info,
|
|
trait_path,
|
|
make_trait_body,
|
|
true,
|
|
tt::TopSubtree::empty(tt::DelimSpan::from_single(invoc_span)),
|
|
))
|
|
}
|
|
|
|
fn expand_simple_derive_with_parsed(
|
|
invoc_span: Span,
|
|
info: BasicAdtInfo,
|
|
trait_path: tt::TopSubtree,
|
|
make_trait_body: impl FnOnce(&BasicAdtInfo) -> tt::TopSubtree,
|
|
constrain_to_trait: bool,
|
|
extra_impl_params: tt::TopSubtree,
|
|
) -> tt::TopSubtree {
|
|
let trait_body = make_trait_body(&info);
|
|
let mut where_block: Vec<_> =
|
|
info.where_clause.into_iter().map(|w| quote! {invoc_span => #w , }).collect();
|
|
let (params, args): (Vec<_>, Vec<_>) = info
|
|
.param_types
|
|
.into_iter()
|
|
.map(|param| {
|
|
let ident = param.name;
|
|
if let Some(b) = param.bounds {
|
|
let ident2 = ident.clone();
|
|
where_block.push(quote! {invoc_span => #ident2 : #b , });
|
|
}
|
|
if let Some(ty) = param.const_ty {
|
|
let ident2 = ident.clone();
|
|
(quote! {invoc_span => const #ident : #ty , }, quote! {invoc_span => #ident2 , })
|
|
} else {
|
|
let bound = trait_path.clone();
|
|
let ident2 = ident.clone();
|
|
let param = if constrain_to_trait {
|
|
quote! {invoc_span => #ident : #bound , }
|
|
} else {
|
|
quote! {invoc_span => #ident , }
|
|
};
|
|
(param, quote! {invoc_span => #ident2 , })
|
|
}
|
|
})
|
|
.unzip();
|
|
|
|
if constrain_to_trait {
|
|
where_block.extend(info.associated_types.iter().map(|it| {
|
|
let it = it.clone();
|
|
let bound = trait_path.clone();
|
|
quote! {invoc_span => #it : #bound , }
|
|
}));
|
|
}
|
|
|
|
let name = info.name;
|
|
quote! {invoc_span =>
|
|
impl < ##params #extra_impl_params > #trait_path for #name < ##args > where ##where_block { #trait_body }
|
|
}
|
|
}
|
|
|
|
fn copy_expand(
|
|
db: &dyn ExpandDatabase,
|
|
span: Span,
|
|
tt: &tt::TopSubtree,
|
|
) -> ExpandResult<tt::TopSubtree> {
|
|
let krate = dollar_crate(span);
|
|
expand_simple_derive(db, span, tt, quote! {span => #krate::marker::Copy }, |_| quote! {span =>})
|
|
}
|
|
|
|
fn clone_expand(
|
|
db: &dyn ExpandDatabase,
|
|
span: Span,
|
|
tt: &tt::TopSubtree,
|
|
) -> ExpandResult<tt::TopSubtree> {
|
|
let krate = dollar_crate(span);
|
|
expand_simple_derive(db, span, tt, quote! {span => #krate::clone::Clone }, |adt| {
|
|
if matches!(adt.shape, AdtShape::Union) {
|
|
let star = tt::Punct { char: '*', spacing: ::tt::Spacing::Alone, span };
|
|
return quote! {span =>
|
|
fn clone(&self) -> Self {
|
|
#star self
|
|
}
|
|
};
|
|
}
|
|
if matches!(&adt.shape, AdtShape::Enum { variants, .. } if variants.is_empty()) {
|
|
let star = tt::Punct { char: '*', spacing: ::tt::Spacing::Alone, span };
|
|
return quote! {span =>
|
|
fn clone(&self) -> Self {
|
|
match #star self {}
|
|
}
|
|
};
|
|
}
|
|
let name = &adt.name;
|
|
let patterns = adt.shape.as_pattern(span, name);
|
|
let exprs = adt.shape.as_pattern_map(name, |it| quote! {span => #it .clone() }, span);
|
|
let arms = patterns.into_iter().zip(exprs).map(|(pat, expr)| {
|
|
let fat_arrow = fat_arrow(span);
|
|
quote! {span =>
|
|
#pat #fat_arrow #expr,
|
|
}
|
|
});
|
|
|
|
quote! {span =>
|
|
fn clone(&self) -> Self {
|
|
match self {
|
|
##arms
|
|
}
|
|
}
|
|
}
|
|
})
|
|
}
|
|
|
|
/// This function exists since `quote! {span => => }` doesn't work.
|
|
fn fat_arrow(span: Span) -> tt::TopSubtree {
|
|
let eq = tt::Punct { char: '=', spacing: ::tt::Spacing::Joint, span };
|
|
quote! {span => #eq> }
|
|
}
|
|
|
|
/// This function exists since `quote! {span => && }` doesn't work.
|
|
fn and_and(span: Span) -> tt::TopSubtree {
|
|
let and = tt::Punct { char: '&', spacing: ::tt::Spacing::Joint, span };
|
|
quote! {span => #and& }
|
|
}
|
|
|
|
fn default_expand(
|
|
db: &dyn ExpandDatabase,
|
|
span: Span,
|
|
tt: &tt::TopSubtree,
|
|
) -> ExpandResult<tt::TopSubtree> {
|
|
let krate = &dollar_crate(span);
|
|
expand_simple_derive(db, span, tt, quote! {span => #krate::default::Default }, |adt| {
|
|
let body = match &adt.shape {
|
|
AdtShape::Struct(fields) => {
|
|
let name = &adt.name;
|
|
fields.default_expand(
|
|
quote!(span =>#name),
|
|
span,
|
|
|_| quote!(span =>#krate::default::Default::default()),
|
|
)
|
|
}
|
|
AdtShape::Enum { default_variant, variants } => {
|
|
if let Some(d) = default_variant {
|
|
let (name, fields) = &variants[*d];
|
|
let adt_name = &adt.name;
|
|
fields.default_expand(
|
|
quote!(span =>#adt_name :: #name),
|
|
span,
|
|
|_| quote!(span =>#krate::default::Default::default()),
|
|
)
|
|
} else {
|
|
// FIXME: Return expand error here
|
|
quote!(span =>)
|
|
}
|
|
}
|
|
AdtShape::Union => {
|
|
// FIXME: Return expand error here
|
|
quote!(span =>)
|
|
}
|
|
};
|
|
quote! {span =>
|
|
fn default() -> Self {
|
|
#body
|
|
}
|
|
}
|
|
})
|
|
}
|
|
|
|
fn debug_expand(
|
|
db: &dyn ExpandDatabase,
|
|
span: Span,
|
|
tt: &tt::TopSubtree,
|
|
) -> ExpandResult<tt::TopSubtree> {
|
|
let krate = &dollar_crate(span);
|
|
expand_simple_derive(db, span, tt, quote! {span => #krate::fmt::Debug }, |adt| {
|
|
let for_variant = |name: String, v: &VariantShape| match v {
|
|
VariantShape::Struct(fields) => {
|
|
let for_fields = fields.iter().map(|(it, _)| {
|
|
let x_string = it.to_string();
|
|
quote! {span =>
|
|
.field(#x_string, & #it)
|
|
}
|
|
});
|
|
quote! {span =>
|
|
f.debug_struct(#name) ##for_fields .finish()
|
|
}
|
|
}
|
|
VariantShape::Tuple(n) => {
|
|
let for_fields = tuple_field_iterator(span, *n).map(|it| {
|
|
quote! {span =>
|
|
.field( & #it)
|
|
}
|
|
});
|
|
quote! {span =>
|
|
f.debug_tuple(#name) ##for_fields .finish()
|
|
}
|
|
}
|
|
VariantShape::Unit => quote! {span =>
|
|
f.write_str(#name)
|
|
},
|
|
};
|
|
if matches!(&adt.shape, AdtShape::Enum { variants, .. } if variants.is_empty()) {
|
|
let star = tt::Punct { char: '*', spacing: ::tt::Spacing::Alone, span };
|
|
return quote! {span =>
|
|
fn fmt(&self, f: &mut #krate::fmt::Formatter) -> #krate::fmt::Result {
|
|
match #star self {}
|
|
}
|
|
};
|
|
}
|
|
let arms = match &adt.shape {
|
|
AdtShape::Struct(fields) => {
|
|
let fat_arrow = fat_arrow(span);
|
|
let name = &adt.name;
|
|
let pat = fields.as_pattern(quote!(span =>#name), span);
|
|
let expr = for_variant(name.to_string(), fields);
|
|
vec![quote! {span => #pat #fat_arrow #expr }]
|
|
}
|
|
AdtShape::Enum { variants, .. } => variants
|
|
.iter()
|
|
.map(|(name, v)| {
|
|
let fat_arrow = fat_arrow(span);
|
|
let adt_name = &adt.name;
|
|
let pat = v.as_pattern(quote!(span =>#adt_name :: #name), span);
|
|
let expr = for_variant(name.to_string(), v);
|
|
quote! {span =>
|
|
#pat #fat_arrow #expr ,
|
|
}
|
|
})
|
|
.collect(),
|
|
AdtShape::Union => {
|
|
// FIXME: Return expand error here
|
|
vec![]
|
|
}
|
|
};
|
|
quote! {span =>
|
|
fn fmt(&self, f: &mut #krate::fmt::Formatter) -> #krate::fmt::Result {
|
|
match self {
|
|
##arms
|
|
}
|
|
}
|
|
}
|
|
})
|
|
}
|
|
|
|
fn hash_expand(
|
|
db: &dyn ExpandDatabase,
|
|
span: Span,
|
|
tt: &tt::TopSubtree,
|
|
) -> ExpandResult<tt::TopSubtree> {
|
|
let krate = &dollar_crate(span);
|
|
expand_simple_derive(db, span, tt, quote! {span => #krate::hash::Hash }, |adt| {
|
|
if matches!(adt.shape, AdtShape::Union) {
|
|
// FIXME: Return expand error here
|
|
return quote! {span =>};
|
|
}
|
|
if matches!(&adt.shape, AdtShape::Enum { variants, .. } if variants.is_empty()) {
|
|
let star = tt::Punct { char: '*', spacing: ::tt::Spacing::Alone, span };
|
|
return quote! {span =>
|
|
fn hash<H: #krate::hash::Hasher>(&self, ra_expand_state: &mut H) {
|
|
match #star self {}
|
|
}
|
|
};
|
|
}
|
|
let arms =
|
|
adt.shape.as_pattern(span, &adt.name).into_iter().zip(adt.shape.field_names(span)).map(
|
|
|(pat, names)| {
|
|
let expr = {
|
|
let it =
|
|
names.iter().map(|it| quote! {span => #it . hash(ra_expand_state); });
|
|
quote! {span => {
|
|
##it
|
|
} }
|
|
};
|
|
let fat_arrow = fat_arrow(span);
|
|
quote! {span =>
|
|
#pat #fat_arrow #expr ,
|
|
}
|
|
},
|
|
);
|
|
let check_discriminant = if matches!(&adt.shape, AdtShape::Enum { .. }) {
|
|
quote! {span => #krate::mem::discriminant(self).hash(ra_expand_state); }
|
|
} else {
|
|
quote! {span =>}
|
|
};
|
|
quote! {span =>
|
|
fn hash<H: #krate::hash::Hasher>(&self, ra_expand_state: &mut H) {
|
|
#check_discriminant
|
|
match self {
|
|
##arms
|
|
}
|
|
}
|
|
}
|
|
})
|
|
}
|
|
|
|
fn eq_expand(
|
|
db: &dyn ExpandDatabase,
|
|
span: Span,
|
|
tt: &tt::TopSubtree,
|
|
) -> ExpandResult<tt::TopSubtree> {
|
|
let krate = dollar_crate(span);
|
|
expand_simple_derive(db, span, tt, quote! {span => #krate::cmp::Eq }, |_| quote! {span =>})
|
|
}
|
|
|
|
fn partial_eq_expand(
|
|
db: &dyn ExpandDatabase,
|
|
span: Span,
|
|
tt: &tt::TopSubtree,
|
|
) -> ExpandResult<tt::TopSubtree> {
|
|
let krate = dollar_crate(span);
|
|
expand_simple_derive(db, span, tt, quote! {span => #krate::cmp::PartialEq }, |adt| {
|
|
if matches!(adt.shape, AdtShape::Union) {
|
|
// FIXME: Return expand error here
|
|
return quote! {span =>};
|
|
}
|
|
let name = &adt.name;
|
|
|
|
let (self_patterns, other_patterns) = self_and_other_patterns(adt, name, span);
|
|
let arms = izip!(self_patterns, other_patterns, adt.shape.field_names(span)).map(
|
|
|(pat1, pat2, names)| {
|
|
let fat_arrow = fat_arrow(span);
|
|
let body = match &*names {
|
|
[] => {
|
|
quote!(span =>true)
|
|
}
|
|
[first, rest @ ..] => {
|
|
let rest = rest.iter().map(|it| {
|
|
let t1 = tt::Ident::new(&format!("{}_self", it.sym), it.span);
|
|
let t2 = tt::Ident::new(&format!("{}_other", it.sym), it.span);
|
|
let and_and = and_and(span);
|
|
quote!(span =>#and_and #t1 .eq( #t2 ))
|
|
});
|
|
let first = {
|
|
let t1 = tt::Ident::new(&format!("{}_self", first.sym), first.span);
|
|
let t2 = tt::Ident::new(&format!("{}_other", first.sym), first.span);
|
|
quote!(span =>#t1 .eq( #t2 ))
|
|
};
|
|
quote!(span =>#first ##rest)
|
|
}
|
|
};
|
|
quote! {span => ( #pat1 , #pat2 ) #fat_arrow #body , }
|
|
},
|
|
);
|
|
|
|
let fat_arrow = fat_arrow(span);
|
|
quote! {span =>
|
|
fn eq(&self, other: &Self) -> bool {
|
|
match (self, other) {
|
|
##arms
|
|
_unused #fat_arrow false
|
|
}
|
|
}
|
|
}
|
|
})
|
|
}
|
|
|
|
fn self_and_other_patterns(
|
|
adt: &BasicAdtInfo,
|
|
name: &tt::Ident,
|
|
span: Span,
|
|
) -> (Vec<tt::TopSubtree>, Vec<tt::TopSubtree>) {
|
|
let self_patterns = adt.shape.as_pattern_map(
|
|
name,
|
|
|it| {
|
|
let t = tt::Ident::new(&format!("{}_self", it.sym), it.span);
|
|
quote!(span =>#t)
|
|
},
|
|
span,
|
|
);
|
|
let other_patterns = adt.shape.as_pattern_map(
|
|
name,
|
|
|it| {
|
|
let t = tt::Ident::new(&format!("{}_other", it.sym), it.span);
|
|
quote!(span =>#t)
|
|
},
|
|
span,
|
|
);
|
|
(self_patterns, other_patterns)
|
|
}
|
|
|
|
fn ord_expand(
|
|
db: &dyn ExpandDatabase,
|
|
span: Span,
|
|
tt: &tt::TopSubtree,
|
|
) -> ExpandResult<tt::TopSubtree> {
|
|
let krate = &dollar_crate(span);
|
|
expand_simple_derive(db, span, tt, quote! {span => #krate::cmp::Ord }, |adt| {
|
|
fn compare(
|
|
krate: &tt::Ident,
|
|
left: tt::TopSubtree,
|
|
right: tt::TopSubtree,
|
|
rest: tt::TopSubtree,
|
|
span: Span,
|
|
) -> tt::TopSubtree {
|
|
let fat_arrow1 = fat_arrow(span);
|
|
let fat_arrow2 = fat_arrow(span);
|
|
quote! {span =>
|
|
match #left.cmp(&#right) {
|
|
#krate::cmp::Ordering::Equal #fat_arrow1 {
|
|
#rest
|
|
}
|
|
c #fat_arrow2 return c,
|
|
}
|
|
}
|
|
}
|
|
if matches!(adt.shape, AdtShape::Union) {
|
|
// FIXME: Return expand error here
|
|
return quote!(span =>);
|
|
}
|
|
let (self_patterns, other_patterns) = self_and_other_patterns(adt, &adt.name, span);
|
|
let arms = izip!(self_patterns, other_patterns, adt.shape.field_names(span)).map(
|
|
|(pat1, pat2, fields)| {
|
|
let mut body = quote!(span =>#krate::cmp::Ordering::Equal);
|
|
for f in fields.into_iter().rev() {
|
|
let t1 = tt::Ident::new(&format!("{}_self", f.sym), f.span);
|
|
let t2 = tt::Ident::new(&format!("{}_other", f.sym), f.span);
|
|
body = compare(krate, quote!(span =>#t1), quote!(span =>#t2), body, span);
|
|
}
|
|
let fat_arrow = fat_arrow(span);
|
|
quote! {span => ( #pat1 , #pat2 ) #fat_arrow #body , }
|
|
},
|
|
);
|
|
let fat_arrow = fat_arrow(span);
|
|
let mut body = quote! {span =>
|
|
match (self, other) {
|
|
##arms
|
|
_unused #fat_arrow #krate::cmp::Ordering::Equal
|
|
}
|
|
};
|
|
if matches!(&adt.shape, AdtShape::Enum { .. }) {
|
|
let left = quote!(span =>#krate::intrinsics::discriminant_value(self));
|
|
let right = quote!(span =>#krate::intrinsics::discriminant_value(other));
|
|
body = compare(krate, left, right, body, span);
|
|
}
|
|
quote! {span =>
|
|
fn cmp(&self, other: &Self) -> #krate::cmp::Ordering {
|
|
#body
|
|
}
|
|
}
|
|
})
|
|
}
|
|
|
|
fn partial_ord_expand(
|
|
db: &dyn ExpandDatabase,
|
|
span: Span,
|
|
tt: &tt::TopSubtree,
|
|
) -> ExpandResult<tt::TopSubtree> {
|
|
let krate = &dollar_crate(span);
|
|
expand_simple_derive(db, span, tt, quote! {span => #krate::cmp::PartialOrd }, |adt| {
|
|
fn compare(
|
|
krate: &tt::Ident,
|
|
left: tt::TopSubtree,
|
|
right: tt::TopSubtree,
|
|
rest: tt::TopSubtree,
|
|
span: Span,
|
|
) -> tt::TopSubtree {
|
|
let fat_arrow1 = fat_arrow(span);
|
|
let fat_arrow2 = fat_arrow(span);
|
|
quote! {span =>
|
|
match #left.partial_cmp(&#right) {
|
|
#krate::option::Option::Some(#krate::cmp::Ordering::Equal) #fat_arrow1 {
|
|
#rest
|
|
}
|
|
c #fat_arrow2 return c,
|
|
}
|
|
}
|
|
}
|
|
if matches!(adt.shape, AdtShape::Union) {
|
|
// FIXME: Return expand error here
|
|
return quote!(span =>);
|
|
}
|
|
let left = quote!(span =>#krate::intrinsics::discriminant_value(self));
|
|
let right = quote!(span =>#krate::intrinsics::discriminant_value(other));
|
|
|
|
let (self_patterns, other_patterns) = self_and_other_patterns(adt, &adt.name, span);
|
|
let arms = izip!(self_patterns, other_patterns, adt.shape.field_names(span)).map(
|
|
|(pat1, pat2, fields)| {
|
|
let mut body =
|
|
quote!(span =>#krate::option::Option::Some(#krate::cmp::Ordering::Equal));
|
|
for f in fields.into_iter().rev() {
|
|
let t1 = tt::Ident::new(&format!("{}_self", f.sym), f.span);
|
|
let t2 = tt::Ident::new(&format!("{}_other", f.sym), f.span);
|
|
body = compare(krate, quote!(span =>#t1), quote!(span =>#t2), body, span);
|
|
}
|
|
let fat_arrow = fat_arrow(span);
|
|
quote! {span => ( #pat1 , #pat2 ) #fat_arrow #body , }
|
|
},
|
|
);
|
|
let fat_arrow = fat_arrow(span);
|
|
let body = compare(
|
|
krate,
|
|
left,
|
|
right,
|
|
quote! {span =>
|
|
match (self, other) {
|
|
##arms
|
|
_unused #fat_arrow #krate::option::Option::Some(#krate::cmp::Ordering::Equal)
|
|
}
|
|
},
|
|
span,
|
|
);
|
|
quote! {span =>
|
|
fn partial_cmp(&self, other: &Self) -> #krate::option::Option::Option<#krate::cmp::Ordering> {
|
|
#body
|
|
}
|
|
}
|
|
})
|
|
}
|
|
|
|
fn coerce_pointee_expand(
|
|
db: &dyn ExpandDatabase,
|
|
span: Span,
|
|
tt: &tt::TopSubtree,
|
|
) -> ExpandResult<tt::TopSubtree> {
|
|
let (adt, _span_map) = match to_adt_syntax(db, tt, span) {
|
|
Ok(it) => it,
|
|
Err(err) => {
|
|
return ExpandResult::new(tt::TopSubtree::empty(tt::DelimSpan::from_single(span)), err);
|
|
}
|
|
};
|
|
let adt = adt.clone_for_update();
|
|
let ast::Adt::Struct(strukt) = &adt else {
|
|
return ExpandResult::new(
|
|
tt::TopSubtree::empty(tt::DelimSpan::from_single(span)),
|
|
ExpandError::other(span, "`CoercePointee` can only be derived on `struct`s"),
|
|
);
|
|
};
|
|
let has_at_least_one_field = strukt
|
|
.field_list()
|
|
.map(|it| match it {
|
|
ast::FieldList::RecordFieldList(it) => it.fields().next().is_some(),
|
|
ast::FieldList::TupleFieldList(it) => it.fields().next().is_some(),
|
|
})
|
|
.unwrap_or(false);
|
|
if !has_at_least_one_field {
|
|
return ExpandResult::new(
|
|
tt::TopSubtree::empty(tt::DelimSpan::from_single(span)),
|
|
ExpandError::other(
|
|
span,
|
|
"`CoercePointee` can only be derived on `struct`s with at least one field",
|
|
),
|
|
);
|
|
}
|
|
let is_repr_transparent = strukt.attrs().any(|attr| {
|
|
attr.as_simple_call().is_some_and(|(name, tt)| {
|
|
name == "repr"
|
|
&& tt.syntax().children_with_tokens().any(|it| {
|
|
it.into_token().is_some_and(|it| {
|
|
it.kind() == SyntaxKind::IDENT && it.text() == "transparent"
|
|
})
|
|
})
|
|
})
|
|
});
|
|
if !is_repr_transparent {
|
|
return ExpandResult::new(
|
|
tt::TopSubtree::empty(tt::DelimSpan::from_single(span)),
|
|
ExpandError::other(
|
|
span,
|
|
"`CoercePointee` can only be derived on `struct`s with `#[repr(transparent)]`",
|
|
),
|
|
);
|
|
}
|
|
let type_params = strukt
|
|
.generic_param_list()
|
|
.into_iter()
|
|
.flat_map(|generics| {
|
|
generics.generic_params().filter_map(|param| match param {
|
|
ast::GenericParam::TypeParam(param) => Some(param),
|
|
_ => None,
|
|
})
|
|
})
|
|
.collect_vec();
|
|
if type_params.is_empty() {
|
|
return ExpandResult::new(
|
|
tt::TopSubtree::empty(tt::DelimSpan::from_single(span)),
|
|
ExpandError::other(
|
|
span,
|
|
"`CoercePointee` can only be derived on `struct`s that are generic over at least one type",
|
|
),
|
|
);
|
|
}
|
|
let (pointee_param, pointee_param_idx) = if type_params.len() == 1 {
|
|
// Regardless of the only type param being designed as `#[pointee]` or not, we can just use it as such.
|
|
(type_params[0].clone(), 0)
|
|
} else {
|
|
let mut pointees = type_params.iter().cloned().enumerate().filter(|(_, param)| {
|
|
param.attrs().any(|attr| {
|
|
let is_pointee = attr.as_simple_atom().is_some_and(|name| name == "pointee");
|
|
if is_pointee {
|
|
// Remove the `#[pointee]` attribute so it won't be present in the generated
|
|
// impls (where we cannot resolve it).
|
|
ted::remove(attr.syntax());
|
|
}
|
|
is_pointee
|
|
})
|
|
});
|
|
match (pointees.next(), pointees.next()) {
|
|
(Some((pointee_idx, pointee)), None) => (pointee, pointee_idx),
|
|
(None, _) => {
|
|
return ExpandResult::new(
|
|
tt::TopSubtree::empty(tt::DelimSpan::from_single(span)),
|
|
ExpandError::other(
|
|
span,
|
|
"exactly one generic type parameter must be marked \
|
|
as `#[pointee]` to derive `CoercePointee` traits",
|
|
),
|
|
)
|
|
}
|
|
(Some(_), Some(_)) => {
|
|
return ExpandResult::new(
|
|
tt::TopSubtree::empty(tt::DelimSpan::from_single(span)),
|
|
ExpandError::other(
|
|
span,
|
|
"only one type parameter can be marked as `#[pointee]` \
|
|
when deriving `CoercePointee` traits",
|
|
),
|
|
)
|
|
}
|
|
}
|
|
};
|
|
let (Some(struct_name), Some(pointee_param_name)) = (strukt.name(), pointee_param.name())
|
|
else {
|
|
return ExpandResult::new(
|
|
tt::TopSubtree::empty(tt::DelimSpan::from_single(span)),
|
|
ExpandError::other(span, "invalid item"),
|
|
);
|
|
};
|
|
|
|
{
|
|
let mut pointee_has_maybe_sized_bound = false;
|
|
if let Some(bounds) = pointee_param.type_bound_list() {
|
|
pointee_has_maybe_sized_bound |= bounds.bounds().any(is_maybe_sized_bound);
|
|
}
|
|
if let Some(where_clause) = strukt.where_clause() {
|
|
pointee_has_maybe_sized_bound |= where_clause.predicates().any(|pred| {
|
|
let Some(ast::Type::PathType(ty)) = pred.ty() else { return false };
|
|
let is_not_pointee = ty.path().is_none_or(|path| {
|
|
let is_pointee = path
|
|
.as_single_name_ref()
|
|
.is_some_and(|name| name.text() == pointee_param_name.text());
|
|
!is_pointee
|
|
});
|
|
if is_not_pointee {
|
|
return false;
|
|
}
|
|
pred.type_bound_list()
|
|
.is_some_and(|bounds| bounds.bounds().any(is_maybe_sized_bound))
|
|
})
|
|
}
|
|
if !pointee_has_maybe_sized_bound {
|
|
return ExpandResult::new(
|
|
tt::TopSubtree::empty(tt::DelimSpan::from_single(span)),
|
|
ExpandError::other(
|
|
span,
|
|
format!("`derive(CoercePointee)` requires `{pointee_param_name}` to be marked `?Sized`"),
|
|
),
|
|
);
|
|
}
|
|
}
|
|
|
|
const ADDED_PARAM: &str = "__S";
|
|
|
|
let where_clause = strukt.get_or_create_where_clause();
|
|
|
|
{
|
|
let mut new_predicates = Vec::new();
|
|
|
|
// # Rewrite generic parameter bounds
|
|
// For each bound `U: ..` in `struct<U: ..>`, make a new bound with `__S` in place of `#[pointee]`
|
|
// Example:
|
|
// ```
|
|
// struct<
|
|
// U: Trait<T>,
|
|
// #[pointee] T: Trait<T> + ?Sized,
|
|
// V: Trait<T>> ...
|
|
// ```
|
|
// ... generates this `impl` generic parameters
|
|
// ```
|
|
// impl<
|
|
// U: Trait<T>,
|
|
// T: Trait<T> + ?Sized,
|
|
// V: Trait<T>
|
|
// >
|
|
// where
|
|
// U: Trait<__S>,
|
|
// __S: Trait<__S> + ?Sized,
|
|
// V: Trait<__S> ...
|
|
// ```
|
|
for param in &type_params {
|
|
let Some(param_name) = param.name() else { continue };
|
|
if let Some(bounds) = param.type_bound_list() {
|
|
// If the target type is the pointee, duplicate the bound as whole.
|
|
// Otherwise, duplicate only bounds that mention the pointee.
|
|
let is_pointee = param_name.text() == pointee_param_name.text();
|
|
let new_bounds = bounds
|
|
.bounds()
|
|
.map(|bound| bound.clone_subtree().clone_for_update())
|
|
.filter(|bound| {
|
|
bound.ty().is_some_and(|ty| {
|
|
substitute_type_in_bound(ty, &pointee_param_name.text(), ADDED_PARAM)
|
|
|| is_pointee
|
|
})
|
|
});
|
|
let new_bounds_target = if is_pointee {
|
|
make::name_ref(ADDED_PARAM)
|
|
} else {
|
|
make::name_ref(¶m_name.text())
|
|
};
|
|
new_predicates.push(
|
|
make::where_pred(
|
|
make::ty_path(make::path_from_segments(
|
|
[make::path_segment(new_bounds_target)],
|
|
false,
|
|
)),
|
|
new_bounds,
|
|
)
|
|
.clone_for_update(),
|
|
);
|
|
}
|
|
}
|
|
|
|
// # Rewrite `where` clauses
|
|
//
|
|
// Move on to `where` clauses.
|
|
// Example:
|
|
// ```
|
|
// struct MyPointer<#[pointee] T, ..>
|
|
// where
|
|
// U: Trait<V> + Trait<T>,
|
|
// Companion<T>: Trait<T>,
|
|
// T: Trait<T> + ?Sized,
|
|
// { .. }
|
|
// ```
|
|
// ... will have a impl prelude like so
|
|
// ```
|
|
// impl<..> ..
|
|
// where
|
|
// U: Trait<V> + Trait<T>,
|
|
// U: Trait<__S>,
|
|
// Companion<T>: Trait<T>,
|
|
// Companion<__S>: Trait<__S>,
|
|
// T: Trait<T> + ?Sized,
|
|
// __S: Trait<__S> + ?Sized,
|
|
// ```
|
|
//
|
|
// We should also write a few new `where` bounds from `#[pointee] T` to `__S`
|
|
// as well as any bound that indirectly involves the `#[pointee] T` type.
|
|
for predicate in where_clause.predicates() {
|
|
let predicate = predicate.clone_subtree().clone_for_update();
|
|
let Some(pred_target) = predicate.ty() else { continue };
|
|
|
|
// If the target type references the pointee, duplicate the bound as whole.
|
|
// Otherwise, duplicate only bounds that mention the pointee.
|
|
if substitute_type_in_bound(
|
|
pred_target.clone(),
|
|
&pointee_param_name.text(),
|
|
ADDED_PARAM,
|
|
) {
|
|
if let Some(bounds) = predicate.type_bound_list() {
|
|
for bound in bounds.bounds() {
|
|
if let Some(ty) = bound.ty() {
|
|
substitute_type_in_bound(ty, &pointee_param_name.text(), ADDED_PARAM);
|
|
}
|
|
}
|
|
}
|
|
|
|
new_predicates.push(predicate);
|
|
} else if let Some(bounds) = predicate.type_bound_list() {
|
|
let new_bounds = bounds
|
|
.bounds()
|
|
.map(|bound| bound.clone_subtree().clone_for_update())
|
|
.filter(|bound| {
|
|
bound.ty().is_some_and(|ty| {
|
|
substitute_type_in_bound(ty, &pointee_param_name.text(), ADDED_PARAM)
|
|
})
|
|
});
|
|
new_predicates.push(make::where_pred(pred_target, new_bounds).clone_for_update());
|
|
}
|
|
}
|
|
|
|
for new_predicate in new_predicates {
|
|
where_clause.add_predicate(new_predicate);
|
|
}
|
|
}
|
|
|
|
{
|
|
// # Add `Unsize<__S>` bound to `#[pointee]` at the generic parameter location
|
|
//
|
|
// Find the `#[pointee]` parameter and add an `Unsize<__S>` bound to it.
|
|
where_clause.add_predicate(
|
|
make::where_pred(
|
|
make::ty_path(make::path_from_segments(
|
|
[make::path_segment(make::name_ref(&pointee_param_name.text()))],
|
|
false,
|
|
)),
|
|
[make::type_bound(make::ty_path(make::path_from_segments(
|
|
[
|
|
make::path_segment(make::name_ref("core")),
|
|
make::path_segment(make::name_ref("marker")),
|
|
make::generic_ty_path_segment(
|
|
make::name_ref("Unsize"),
|
|
[make::type_arg(make::ty_path(make::path_from_segments(
|
|
[make::path_segment(make::name_ref(ADDED_PARAM))],
|
|
false,
|
|
)))
|
|
.into()],
|
|
),
|
|
],
|
|
true,
|
|
)))],
|
|
)
|
|
.clone_for_update(),
|
|
);
|
|
}
|
|
|
|
let self_for_traits = {
|
|
// Replace the `#[pointee]` with `__S`.
|
|
let mut type_param_idx = 0;
|
|
let self_params_for_traits = strukt
|
|
.generic_param_list()
|
|
.into_iter()
|
|
.flat_map(|params| params.generic_params())
|
|
.filter_map(|param| {
|
|
Some(match param {
|
|
ast::GenericParam::ConstParam(param) => {
|
|
ast::GenericArg::ConstArg(make::expr_const_value(¶m.name()?.text()))
|
|
}
|
|
ast::GenericParam::LifetimeParam(param) => {
|
|
make::lifetime_arg(param.lifetime()?).into()
|
|
}
|
|
ast::GenericParam::TypeParam(param) => {
|
|
let name = if pointee_param_idx == type_param_idx {
|
|
make::name_ref(ADDED_PARAM)
|
|
} else {
|
|
make::name_ref(¶m.name()?.text())
|
|
};
|
|
type_param_idx += 1;
|
|
make::type_arg(make::ty_path(make::path_from_segments(
|
|
[make::path_segment(name)],
|
|
false,
|
|
)))
|
|
.into()
|
|
}
|
|
})
|
|
});
|
|
let self_for_traits = make::path_from_segments(
|
|
[make::generic_ty_path_segment(
|
|
make::name_ref(&struct_name.text()),
|
|
self_params_for_traits,
|
|
)],
|
|
false,
|
|
)
|
|
.clone_for_update();
|
|
self_for_traits
|
|
};
|
|
|
|
let mut span_map = span::SpanMap::empty();
|
|
// One span for them all.
|
|
span_map.push(adt.syntax().text_range().end(), span);
|
|
|
|
let self_for_traits = syntax_bridge::syntax_node_to_token_tree(
|
|
self_for_traits.syntax(),
|
|
&span_map,
|
|
span,
|
|
DocCommentDesugarMode::ProcMacro,
|
|
);
|
|
let info = match parse_adt_from_syntax(&adt, &span_map, span) {
|
|
Ok(it) => it,
|
|
Err(err) => {
|
|
return ExpandResult::new(tt::TopSubtree::empty(tt::DelimSpan::from_single(span)), err)
|
|
}
|
|
};
|
|
|
|
let self_for_traits2 = self_for_traits.clone();
|
|
let krate = dollar_crate(span);
|
|
let krate2 = krate.clone();
|
|
let dispatch_from_dyn = expand_simple_derive_with_parsed(
|
|
span,
|
|
info.clone(),
|
|
quote! {span => #krate2::ops::DispatchFromDyn<#self_for_traits2> },
|
|
|_adt| quote! {span => },
|
|
false,
|
|
quote! {span => __S },
|
|
);
|
|
let coerce_unsized = expand_simple_derive_with_parsed(
|
|
span,
|
|
info,
|
|
quote! {span => #krate::ops::CoerceUnsized<#self_for_traits> },
|
|
|_adt| quote! {span => },
|
|
false,
|
|
quote! {span => __S },
|
|
);
|
|
return ExpandResult::ok(quote! {span => #dispatch_from_dyn #coerce_unsized });
|
|
|
|
fn is_maybe_sized_bound(bound: ast::TypeBound) -> bool {
|
|
if bound.question_mark_token().is_none() {
|
|
return false;
|
|
}
|
|
let Some(ast::Type::PathType(ty)) = bound.ty() else {
|
|
return false;
|
|
};
|
|
let Some(path) = ty.path() else {
|
|
return false;
|
|
};
|
|
return segments_eq(&path, &["Sized"])
|
|
|| segments_eq(&path, &["core", "marker", "Sized"])
|
|
|| segments_eq(&path, &["std", "marker", "Sized"]);
|
|
|
|
fn segments_eq(path: &ast::Path, expected: &[&str]) -> bool {
|
|
path.segments().zip_longest(expected.iter().copied()).all(|value| {
|
|
value.both().is_some_and(|(segment, expected)| {
|
|
segment.name_ref().is_some_and(|name| name.text() == expected)
|
|
})
|
|
})
|
|
}
|
|
}
|
|
|
|
/// Returns true if any substitution was performed.
|
|
fn substitute_type_in_bound(ty: ast::Type, param_name: &str, replacement: &str) -> bool {
|
|
return match ty {
|
|
ast::Type::ArrayType(ty) => {
|
|
ty.ty().is_some_and(|ty| substitute_type_in_bound(ty, param_name, replacement))
|
|
}
|
|
ast::Type::DynTraitType(ty) => go_bounds(ty.type_bound_list(), param_name, replacement),
|
|
ast::Type::FnPtrType(ty) => any_long(
|
|
ty.param_list()
|
|
.into_iter()
|
|
.flat_map(|params| params.params().filter_map(|param| param.ty()))
|
|
.chain(ty.ret_type().and_then(|it| it.ty())),
|
|
|ty| substitute_type_in_bound(ty, param_name, replacement),
|
|
),
|
|
ast::Type::ForType(ty) => {
|
|
ty.ty().is_some_and(|ty| substitute_type_in_bound(ty, param_name, replacement))
|
|
}
|
|
ast::Type::ImplTraitType(ty) => {
|
|
go_bounds(ty.type_bound_list(), param_name, replacement)
|
|
}
|
|
ast::Type::ParenType(ty) => {
|
|
ty.ty().is_some_and(|ty| substitute_type_in_bound(ty, param_name, replacement))
|
|
}
|
|
ast::Type::PathType(ty) => ty.path().is_some_and(|path| {
|
|
if path.as_single_name_ref().is_some_and(|name| name.text() == param_name) {
|
|
ted::replace(
|
|
path.syntax(),
|
|
make::path_from_segments(
|
|
[make::path_segment(make::name_ref(replacement))],
|
|
false,
|
|
)
|
|
.clone_for_update()
|
|
.syntax(),
|
|
);
|
|
return true;
|
|
}
|
|
|
|
any_long(
|
|
path.segments()
|
|
.filter_map(|segment| segment.generic_arg_list())
|
|
.flat_map(|it| it.generic_args())
|
|
.filter_map(|generic_arg| match generic_arg {
|
|
ast::GenericArg::TypeArg(ty) => ty.ty(),
|
|
_ => None,
|
|
}),
|
|
|ty| substitute_type_in_bound(ty, param_name, replacement),
|
|
)
|
|
}),
|
|
ast::Type::PtrType(ty) => {
|
|
ty.ty().is_some_and(|ty| substitute_type_in_bound(ty, param_name, replacement))
|
|
}
|
|
ast::Type::RefType(ty) => {
|
|
ty.ty().is_some_and(|ty| substitute_type_in_bound(ty, param_name, replacement))
|
|
}
|
|
ast::Type::SliceType(ty) => {
|
|
ty.ty().is_some_and(|ty| substitute_type_in_bound(ty, param_name, replacement))
|
|
}
|
|
ast::Type::TupleType(ty) => {
|
|
any_long(ty.fields(), |ty| substitute_type_in_bound(ty, param_name, replacement))
|
|
}
|
|
ast::Type::InferType(_) | ast::Type::MacroType(_) | ast::Type::NeverType(_) => false,
|
|
};
|
|
|
|
fn go_bounds(
|
|
bounds: Option<ast::TypeBoundList>,
|
|
param_name: &str,
|
|
replacement: &str,
|
|
) -> bool {
|
|
bounds.is_some_and(|bounds| {
|
|
any_long(bounds.bounds(), |bound| {
|
|
bound
|
|
.ty()
|
|
.is_some_and(|ty| substitute_type_in_bound(ty, param_name, replacement))
|
|
})
|
|
})
|
|
}
|
|
|
|
/// Like [`Iterator::any()`], but not short-circuiting.
|
|
fn any_long<I: Iterator, F: FnMut(I::Item) -> bool>(iter: I, mut f: F) -> bool {
|
|
let mut result = false;
|
|
iter.for_each(|item| result |= f(item));
|
|
result
|
|
}
|
|
}
|
|
}
|