зеркало из https://github.com/mozilla/gecko-dev.git
200 строки
7.0 KiB
Rust
200 строки
7.0 KiB
Rust
use proc_macro2::{Span, TokenStream};
|
|
use quote::{quote, quote_spanned, ToTokens};
|
|
use syn::{spanned::Spanned, visit_mut::VisitMut, *};
|
|
|
|
use crate::utils::{
|
|
parse_as_empty, prepend_underscore_to_self, ReplaceReceiver, CURRENT_PRIVATE_MODULE,
|
|
};
|
|
|
|
pub(crate) fn attribute(args: &TokenStream, mut input: ItemImpl) -> TokenStream {
|
|
if let Err(e) = parse_as_empty(args).and_then(|()| parse(&mut input)) {
|
|
let self_ty = &input.self_ty;
|
|
let (impl_generics, _, where_clause) = input.generics.split_for_impl();
|
|
|
|
let mut tokens = e.to_compile_error();
|
|
let private = Ident::new(CURRENT_PRIVATE_MODULE, Span::call_site());
|
|
// Generate a dummy `PinnedDrop` implementation.
|
|
// In many cases, `#[pinned_drop] impl` is declared after `#[pin_project]`.
|
|
// Therefore, if `pinned_drop` compile fails, you will also get an error
|
|
// about `PinnedDrop` not being implemented.
|
|
// This can be prevented to some extent by generating a dummy
|
|
// `PinnedDrop` implementation.
|
|
// We already know that we will get a compile error, so this won't
|
|
// accidentally compile successfully.
|
|
tokens.extend(quote! {
|
|
impl #impl_generics ::pin_project::#private::PinnedDrop for #self_ty #where_clause {
|
|
unsafe fn drop(self: ::core::pin::Pin<&mut Self>) {}
|
|
}
|
|
});
|
|
tokens
|
|
} else {
|
|
input.into_token_stream()
|
|
}
|
|
}
|
|
|
|
fn parse_method(method: &ImplItemMethod) -> Result<()> {
|
|
fn get_ty_path(ty: &Type) -> Option<&Path> {
|
|
if let Type::Path(TypePath { qself: None, path }) = ty { Some(path) } else { None }
|
|
}
|
|
|
|
const INVALID_ARGUMENT: &str = "method `drop` must take an argument `self: Pin<&mut Self>`";
|
|
|
|
if method.sig.ident != "drop" {
|
|
return Err(error!(
|
|
method.sig.ident,
|
|
"method `{}` is not a member of trait `PinnedDrop", method.sig.ident,
|
|
));
|
|
}
|
|
|
|
if let ReturnType::Type(_, ty) = &method.sig.output {
|
|
match &**ty {
|
|
Type::Tuple(TypeTuple { elems, .. }) if elems.is_empty() => {}
|
|
_ => return Err(error!(ty, "method `drop` must return the unit type")),
|
|
}
|
|
}
|
|
|
|
if method.sig.inputs.len() != 1 {
|
|
if method.sig.inputs.is_empty() {
|
|
return Err(syn::Error::new(method.sig.paren_token.span, INVALID_ARGUMENT));
|
|
} else {
|
|
return Err(error!(&method.sig.inputs, INVALID_ARGUMENT));
|
|
}
|
|
}
|
|
|
|
if let FnArg::Typed(PatType { pat, ty, .. }) = &method.sig.inputs[0] {
|
|
// !by_ref (mutability) ident !subpat: path
|
|
if let (Pat::Ident(PatIdent { by_ref: None, ident, subpat: None, .. }), Some(path)) =
|
|
(&**pat, get_ty_path(ty))
|
|
{
|
|
let ty = &path.segments.last().unwrap();
|
|
if let PathArguments::AngleBracketed(args) = &ty.arguments {
|
|
// (mut) self: (path::)Pin<args>
|
|
if ident == "self" && args.args.len() == 1 && ty.ident == "Pin" {
|
|
// &mut <elem>
|
|
if let GenericArgument::Type(Type::Reference(TypeReference {
|
|
mutability: Some(_),
|
|
elem,
|
|
..
|
|
})) = &args.args[0]
|
|
{
|
|
if get_ty_path(elem).map_or(false, |path| path.is_ident("Self")) {
|
|
if method.sig.unsafety.is_some() {
|
|
return Err(error!(
|
|
method.sig.unsafety,
|
|
"implementing the method `drop` is not unsafe"
|
|
));
|
|
}
|
|
return Ok(());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Err(error!(method.sig.inputs[0], INVALID_ARGUMENT))
|
|
}
|
|
|
|
fn parse(item: &mut ItemImpl) -> Result<()> {
|
|
if let Some((_, path, _)) = &mut item.trait_ {
|
|
if path.is_ident("PinnedDrop") {
|
|
let private = Ident::new(CURRENT_PRIVATE_MODULE, Span::call_site());
|
|
*path = syn::parse2(quote_spanned! { path.span() =>
|
|
::pin_project::#private::PinnedDrop
|
|
})
|
|
.unwrap();
|
|
} else {
|
|
return Err(error!(
|
|
path,
|
|
"#[pinned_drop] may only be used on implementation for the `PinnedDrop` trait"
|
|
));
|
|
}
|
|
} else {
|
|
return Err(error!(
|
|
item.self_ty,
|
|
"#[pinned_drop] may only be used on implementation for the `PinnedDrop` trait"
|
|
));
|
|
}
|
|
|
|
if item.unsafety.is_some() {
|
|
return Err(error!(item.unsafety, "implementing the trait `PinnedDrop` is not unsafe"));
|
|
}
|
|
if item.items.is_empty() {
|
|
return Err(error!(item, "not all trait items implemented, missing: `drop`"));
|
|
}
|
|
|
|
for (i, item) in item.items.iter().enumerate() {
|
|
match item {
|
|
ImplItem::Const(item) => {
|
|
return Err(error!(
|
|
item,
|
|
"const `{}` is not a member of trait `PinnedDrop`", item.ident
|
|
));
|
|
}
|
|
ImplItem::Type(item) => {
|
|
return Err(error!(
|
|
item,
|
|
"type `{}` is not a member of trait `PinnedDrop`", item.ident
|
|
));
|
|
}
|
|
ImplItem::Method(method) => {
|
|
parse_method(method)?;
|
|
if i != 0 {
|
|
return Err(error!(method, "duplicate definitions with name `drop`"));
|
|
}
|
|
}
|
|
_ => parse_as_empty(&item.to_token_stream())?,
|
|
}
|
|
}
|
|
|
|
expand_item(item);
|
|
|
|
Ok(())
|
|
}
|
|
|
|
// from:
|
|
//
|
|
// fn drop(self: Pin<&mut Self>) {
|
|
// // something
|
|
// }
|
|
//
|
|
// into:
|
|
//
|
|
// unsafe fn drop(self: Pin<&mut Self>) {
|
|
// fn __drop_inner<T>(__self: Pin<&mut Foo<'_, T>>) {
|
|
// // something
|
|
// }
|
|
// __drop_inner(self);
|
|
// }
|
|
//
|
|
fn expand_item(item: &mut ItemImpl) {
|
|
let method =
|
|
if let ImplItem::Method(method) = &mut item.items[0] { method } else { unreachable!() };
|
|
let mut drop_inner = method.clone();
|
|
|
|
// `fn drop(mut self: Pin<&mut Self>)` -> `fn __drop_inner<T>(mut __self: Pin<&mut Receiver>)`
|
|
drop_inner.sig.ident = Ident::new("__drop_inner", drop_inner.sig.ident.span());
|
|
drop_inner.sig.generics = item.generics.clone();
|
|
if let FnArg::Typed(arg) = &mut drop_inner.sig.inputs[0] {
|
|
if let Pat::Ident(ident) = &mut *arg.pat {
|
|
prepend_underscore_to_self(&mut ident.ident);
|
|
}
|
|
}
|
|
let mut visitor = ReplaceReceiver::new(&item.self_ty);
|
|
visitor.visit_signature_mut(&mut drop_inner.sig);
|
|
visitor.visit_block_mut(&mut drop_inner.block);
|
|
|
|
// `fn drop(mut self: Pin<&mut Self>)` -> `unsafe fn drop(self: Pin<&mut Self>)`
|
|
method.sig.unsafety = Some(token::Unsafe::default());
|
|
if let FnArg::Typed(arg) = &mut method.sig.inputs[0] {
|
|
if let Pat::Ident(ident) = &mut *arg.pat {
|
|
ident.mutability = None;
|
|
}
|
|
}
|
|
|
|
method.block = syn::parse_quote! {{
|
|
#drop_inner
|
|
__drop_inner(self);
|
|
}};
|
|
}
|