blob: 7aded22017f4631dcd1404d8431bbd4ded3578f4 [file] [log] [blame]
Devin Jeanpierref643afa2022-02-23 20:50:48 +00001// Part of the Crubit project, under the Apache License v2.0 with LLVM
2// Exceptions. See /LICENSE for license information.
3// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
Googler594745b2023-03-03 13:56:43 -08004#![cfg_attr(not(test), no_std)]
Devin Jeanpierref643afa2022-02-23 20:50:48 +00005
Googler594745b2023-03-03 13:56:43 -08006extern crate alloc;
7
8use alloc::borrow::Cow;
9use alloc::collections::BTreeSet;
10use alloc::format;
11use alloc::vec;
Devin Jeanpierref643afa2022-02-23 20:50:48 +000012use proc_macro::TokenStream;
13use proc_macro2::{Ident, Span};
14use quote::{quote, quote_spanned, ToTokens as _};
Devin Jeanpierrebe1efd22022-05-10 16:52:07 -070015use syn::parse::Parse;
Devin Jeanpierref643afa2022-02-23 20:50:48 +000016use syn::spanned::Spanned as _;
Devin Jeanpierrebe1efd22022-05-10 16:52:07 -070017use syn::Token;
Devin Jeanpierref643afa2022-02-23 20:50:48 +000018
19// TODO(jeanpierreda): derive constructors and assignment for copy and move.
20
Devin Jeanpierree9866732022-07-25 05:52:09 -070021const FIELD_FOR_MUST_USE_CTOR: &'static str = "__must_use_ctor_to_initialize";
22
Devin Jeanpierref643afa2022-02-23 20:50:48 +000023#[proc_macro_derive(CtorFrom_Default)]
24pub fn derive_default(item: TokenStream) -> TokenStream {
25 let input = syn::parse_macro_input!(item as syn::DeriveInput);
26
27 let struct_name = input.ident;
Devin Jeanpierre1b8f4a12022-03-22 21:29:42 +000028 let struct_ctor_name =
29 Ident::new(&format!("_ctor_derive_{}_CtorType_Default", struct_name), Span::call_site());
Devin Jeanpierre31a2f972022-05-09 06:11:05 -070030 let fields: proc_macro2::TokenStream = match &input.data {
Devin Jeanpierref643afa2022-02-23 20:50:48 +000031 syn::Data::Struct(data) => {
32 if let syn::Fields::Unit = data.fields {
Devin Jeanpierre31a2f972022-05-09 06:11:05 -070033 quote! {}
34 } else {
Devin Jeanpierree9866732022-07-25 05:52:09 -070035 let filled_fields = data.fields.iter().enumerate().filter_map(|(i, field)| {
Devin Jeanpierref643afa2022-02-23 20:50:48 +000036 let field_i = syn::Index::from(i);
Devin Jeanpierree9866732022-07-25 05:52:09 -070037 let field_name;
38 // This logic is here in case you derive default on the output of
39 // `#[recursively_pinned]`, but it's obviously not very flexible. For example,
40 // maybe we want to compute a non-colliding field name, and maybe there are
41 // other ordering problems.
42 match &field.ident {
43 Some(name) if name == FIELD_FOR_MUST_USE_CTOR => return None,
44 Some(name) => field_name = quote! {#name},
45 None => field_name = quote! {#field_i},
Devin Jeanpierref643afa2022-02-23 20:50:48 +000046 };
47
48 let field_type = &field.ty;
Devin Jeanpierree9866732022-07-25 05:52:09 -070049 Some(quote_spanned! {field.span() =>
Devin Jeanpierref643afa2022-02-23 20:50:48 +000050 #field_name: <#field_type as ::ctor::CtorNew<()>>::ctor_new(())
Devin Jeanpierree9866732022-07-25 05:52:09 -070051 })
Devin Jeanpierre31a2f972022-05-09 06:11:05 -070052 });
53 quote! {{ #(#filled_fields),* }}
54 }
Devin Jeanpierref643afa2022-02-23 20:50:48 +000055 }
Devin Jeanpierre83f26942022-05-09 09:21:33 -070056 syn::Data::Enum(e) => {
57 return syn::Error::new(e.enum_token.span, "Enums are not supported")
58 .into_compile_error()
59 .into();
60 }
61 syn::Data::Union(u) => {
62 return syn::Error::new(u.union_token.span, "Unions are not supported")
63 .into_compile_error()
64 .into();
65 }
Devin Jeanpierref643afa2022-02-23 20:50:48 +000066 };
67
68 let expanded = quote! {
69 struct #struct_ctor_name();
70
71 impl ::ctor::Ctor for #struct_ctor_name {
72 type Output = #struct_name;
Googler594745b2023-03-03 13:56:43 -080073 unsafe fn ctor(self, dest: ::core::pin::Pin<&mut ::core::mem::MaybeUninit<Self::Output>>) {
Devin Jeanpierref643afa2022-02-23 20:50:48 +000074 ::ctor::ctor!(
Devin Jeanpierre31a2f972022-05-09 06:11:05 -070075 #struct_name #fields
Devin Jeanpierref643afa2022-02-23 20:50:48 +000076 ).ctor(dest)
77 }
78 }
79
Googler594745b2023-03-03 13:56:43 -080080 impl !::core::marker::Unpin for #struct_ctor_name {}
Devin Jeanpierref643afa2022-02-23 20:50:48 +000081
82 impl ::ctor::CtorNew<()> for #struct_name {
83 type CtorType = #struct_ctor_name;
84
85 fn ctor_new(_args: ()) -> #struct_ctor_name { #struct_ctor_name() }
86 }
87 };
88 TokenStream::from(expanded)
89}
90
Devin Jeanpierre57aac932022-05-17 17:18:44 -070091/// `project_pin_type!(foo::T)` is the name of the type returned by
92/// `foo::T::project_pin()`.
Devin Jeanpierref643afa2022-02-23 20:50:48 +000093///
Devin Jeanpierre1c28ff22022-05-09 06:04:57 -070094/// If `foo::T` is not `#[recursively_pinned]`, then this returns the name it
95/// would have used, but is essentially useless.
96#[proc_macro]
Devin Jeanpierre57aac932022-05-17 17:18:44 -070097pub fn project_pin_type(name: TokenStream) -> TokenStream {
Devin Jeanpierre1c28ff22022-05-09 06:04:57 -070098 let mut name = syn::parse_macro_input!(name as syn::Path);
Devin Jeanpierre83f26942022-05-09 09:21:33 -070099 match name.segments.last_mut() {
100 None => {
101 return syn::Error::new(name.span(), "Path must have at least one element")
102 .into_compile_error()
103 .into();
Devin Jeanpierre1c28ff22022-05-09 06:04:57 -0700104 }
Devin Jeanpierre83f26942022-05-09 09:21:33 -0700105 Some(last) => {
106 if let syn::PathArguments::Parenthesized(p) = &last.arguments {
107 return syn::Error::new(
108 p.span(),
109 "Parenthesized paths (e.g. fn, Fn) do not have projected equivalents.",
110 )
111 .into_compile_error()
112 .into();
113 }
Devin Jeanpierre57aac932022-05-17 17:18:44 -0700114 last.ident = project_pin_ident(&last.ident);
Devin Jeanpierre83f26942022-05-09 09:21:33 -0700115 }
Devin Jeanpierre1c28ff22022-05-09 06:04:57 -0700116 }
117 TokenStream::from(quote! { #name })
118}
119
Devin Jeanpierre57aac932022-05-17 17:18:44 -0700120fn project_pin_ident(ident: &Ident) -> Ident {
121 Ident::new(&format!("__CrubitProjectPin{}", ident), Span::call_site())
Devin Jeanpierre1c28ff22022-05-09 06:04:57 -0700122}
123
Devin Jeanpierre190b90a2022-05-24 06:00:34 -0700124/// Defines the `project_pin` function, and its return value.
125///
126/// If the input is a union, this returns nothing, and pin-projection is not
127/// implemented.
Devin Jeanpierreab912362022-06-09 16:49:52 -0700128fn project_pin_impl(input: &syn::DeriveInput) -> syn::Result<proc_macro2::TokenStream> {
129 let is_fieldless = match &input.data {
Devin Jeanpierre239f7612022-05-09 06:09:08 -0700130 syn::Data::Struct(data) => data.fields.is_empty(),
131 syn::Data::Enum(e) => e.variants.iter().all(|variant| variant.fields.is_empty()),
Devin Jeanpierre190b90a2022-05-24 06:00:34 -0700132 syn::Data::Union(_) => {
133 return Ok(quote! {});
Devin Jeanpierre83f26942022-05-09 09:21:33 -0700134 }
Devin Jeanpierre239f7612022-05-09 06:09:08 -0700135 };
136
Devin Jeanpierreab912362022-06-09 16:49:52 -0700137 let mut projected = input.clone();
Devin Jeanpierre190b90a2022-05-24 06:00:34 -0700138 // TODO(jeanpierreda): check attributes for repr(packed)
139 projected.attrs.clear();
140 projected.ident = project_pin_ident(&projected.ident);
141
Devin Jeanpierreab912362022-06-09 16:49:52 -0700142 let lifetime = if is_fieldless {
143 quote! {}
Devin Jeanpierre239f7612022-05-09 06:09:08 -0700144 } else {
Devin Jeanpierreab912362022-06-09 16:49:52 -0700145 add_lifetime(&mut projected.generics, "'proj")
146 };
Devin Jeanpierre1c28ff22022-05-09 06:04:57 -0700147
148 let project_field = |field: &mut syn::Field| {
149 field.attrs.clear();
150 let field_ty = &field.ty;
Googler594745b2023-03-03 13:56:43 -0800151 let pin_ty = syn::parse_quote!(::core::pin::Pin<& #lifetime mut #field_ty>);
Devin Jeanpierre1c28ff22022-05-09 06:04:57 -0700152 field.ty = syn::Type::Path(pin_ty);
153 };
154 // returns the braced parts of a projection pattern and return value.
Devin Jeanpierree9866732022-07-25 05:52:09 -0700155 // e.g. {foo, bar, ..}, {foo: Pin::new_unchecked(foo), bar:
156 // Pin::new_unchecked(bar)}
Devin Jeanpierre1c28ff22022-05-09 06:04:57 -0700157 let pat_project = |fields: &mut syn::Fields| {
158 let mut pat = quote! {};
159 let mut project = quote! {};
160 for (i, field) in fields.iter_mut().enumerate() {
161 // TODO(jeanpierreda): check attributes for e.g. #[unpin]
162 field.attrs.clear();
163 let lhs;
164 let rhs;
165 if let Some(ident) = &field.ident {
166 lhs = quote! {#ident};
167 rhs = ident.clone();
168 pat.extend(quote! {#lhs,});
169 } else {
170 lhs = proc_macro2::Literal::usize_unsuffixed(i).into_token_stream();
171 rhs = Ident::new(&format!("item_{i}"), Span::call_site());
172 pat.extend(quote! {#lhs: #rhs,});
173 }
Googler594745b2023-03-03 13:56:43 -0800174 project.extend(quote! {#lhs: ::core::pin::Pin::new_unchecked(#rhs),});
Devin Jeanpierre1c28ff22022-05-09 06:04:57 -0700175 }
Devin Jeanpierred6f3e2a2022-07-20 18:45:42 -0700176 // Also ignore the __must_use_ctor_to_initialize field, if present.
177 pat.extend(quote! {..});
Devin Jeanpierre1c28ff22022-05-09 06:04:57 -0700178 (quote! {{#pat}}, quote! {{#project}})
179 };
180 let project_body;
Devin Jeanpierreab912362022-06-09 16:49:52 -0700181 let input_ident = &input.ident;
Devin Jeanpierre190b90a2022-05-24 06:00:34 -0700182 let projected_ident = &projected.ident;
Devin Jeanpierre1c28ff22022-05-09 06:04:57 -0700183 match &mut projected.data {
184 syn::Data::Struct(data) => {
185 for field in &mut data.fields {
186 project_field(field);
187 }
188 let (pat, project) = pat_project(&mut data.fields);
189 project_body = quote! {
Devin Jeanpierreab912362022-06-09 16:49:52 -0700190 let #input_ident #pat = from;
Devin Jeanpierre190b90a2022-05-24 06:00:34 -0700191 #projected_ident #project
Devin Jeanpierre1c28ff22022-05-09 06:04:57 -0700192 };
193 }
194 syn::Data::Enum(e) => {
195 let mut match_body = quote! {};
196 for variant in &mut e.variants {
197 for field in &mut variant.fields {
198 project_field(field);
199 }
200 let (pat, project) = pat_project(&mut variant.fields);
201 let variant_ident = &variant.ident;
202 match_body.extend(quote! {
Devin Jeanpierreab912362022-06-09 16:49:52 -0700203 #input_ident::#variant_ident #pat => #projected_ident::#variant_ident #project,
Devin Jeanpierre1c28ff22022-05-09 06:04:57 -0700204 });
205 }
206 project_body = quote! {
207 match from {
208 #match_body
209 }
210 };
211 }
Devin Jeanpierreab912362022-06-09 16:49:52 -0700212 syn::Data::Union(_) => {
213 unreachable!("project_pin_impl should early return when it finds a union")
214 }
Devin Jeanpierre1c28ff22022-05-09 06:04:57 -0700215 }
Devin Jeanpierre190b90a2022-05-24 06:00:34 -0700216
Devin Jeanpierreab912362022-06-09 16:49:52 -0700217 let (input_impl_generics, input_ty_generics, input_where_clause) =
218 input.generics.split_for_impl();
219 let (_, projected_generics, _) = projected.generics.split_for_impl();
220
Devin Jeanpierre190b90a2022-05-24 06:00:34 -0700221 Ok(quote! {
222 #projected
223
Devin Jeanpierreab912362022-06-09 16:49:52 -0700224 impl #input_impl_generics #input_ident #input_ty_generics #input_where_clause {
Devin Jeanpierre190b90a2022-05-24 06:00:34 -0700225 #[must_use]
Googler594745b2023-03-03 13:56:43 -0800226 pub fn project_pin<#lifetime>(self: ::core::pin::Pin<& #lifetime mut Self>) -> #projected_ident #projected_generics {
Devin Jeanpierre1c28ff22022-05-09 06:04:57 -0700227 unsafe {
Googler594745b2023-03-03 13:56:43 -0800228 let from = ::core::pin::Pin::into_inner_unchecked(self);
Devin Jeanpierre1c28ff22022-05-09 06:04:57 -0700229 #project_body
230 }
231 }
232 }
Devin Jeanpierre190b90a2022-05-24 06:00:34 -0700233 })
Devin Jeanpierre1c28ff22022-05-09 06:04:57 -0700234}
235
Devin Jeanpierreab912362022-06-09 16:49:52 -0700236/// Adds a new lifetime to `generics`, returning the quoted lifetime name.
237fn add_lifetime(generics: &mut syn::Generics, prefix: &str) -> proc_macro2::TokenStream {
Googler594745b2023-03-03 13:56:43 -0800238 let taken_lifetimes: BTreeSet<&syn::Lifetime> =
Devin Jeanpierreab912362022-06-09 16:49:52 -0700239 generics.lifetimes().map(|def| &def.lifetime).collect();
240 let mut name = Cow::Borrowed(prefix);
241 let mut i = 1;
242 let lifetime = loop {
243 let lifetime = syn::Lifetime::new(&name, Span::call_site());
244 if !taken_lifetimes.contains(&lifetime) {
245 break lifetime;
246 }
247
248 i += 1;
249 name = Cow::Owned(format!("{prefix}_{i}"));
250 };
251 let quoted_lifetime = quote! {#lifetime};
252 generics.params.push(syn::GenericParam::Lifetime(syn::LifetimeDef::new(lifetime)));
253 quoted_lifetime
254}
255
Devin Jeanpierrebe1efd22022-05-10 16:52:07 -0700256#[derive(Default)]
257struct RecursivelyPinnedArgs {
258 is_pinned_drop: bool,
259}
260
261impl Parse for RecursivelyPinnedArgs {
262 fn parse(input: syn::parse::ParseStream) -> syn::Result<Self> {
263 let args = <syn::punctuated::Punctuated<Ident, Token![,]>>::parse_terminated(input)?;
264 if args.len() > 1 {
265 return Err(syn::Error::new(
266 input.span(), // not args.span(), as that is only for the first argument.
267 &format!("expected at most 1 argument, got: {}", args.len()),
268 ));
269 }
270 let is_pinned_drop = if let Some(arg) = args.first() {
271 if arg != "PinnedDrop" {
272 return Err(syn::Error::new(
273 arg.span(),
274 "unexpected argument (wasn't `PinnedDrop`)",
275 ));
276 }
277 true
278 } else {
279 false
280 };
281 Ok(RecursivelyPinnedArgs { is_pinned_drop })
282 }
283}
284
Devin Jeanpierred6f3e2a2022-07-20 18:45:42 -0700285/// Prevents this type from being directly created outside of this crate in safe
286/// code.
287///
288/// For enums and unit structs, this uses the `#[non_exhaustive]` attribute.
289/// This leads to unfortunate error messages, but there is no other way to
290/// prevent creation of an enum or a unit struct at this time.
291///
292/// For tuple structs, we also use `#[non_exhaustive]`, as it's no worse than
293/// the alternative. Both adding a private field and adding `#[non_exhaustive]`
294/// lead to indirect error messages, but `#[non_exhaustive]` is the more likely
295/// of the two to ever get custom error message support.
296///
297/// Finally, for structs with named fields, we actually *cannot* use
298/// `#[non_exhaustive]`, because it would make the struct not FFI-safe, and
299/// structs with named fields are specifically supported for C++ interop.
300/// Instead, we use a private field with a name that indicates the error.
301/// (`__must_use_ctor_to_initialize`).
302///
303/// Unions are not yet implemented properly.
304///
305/// ---
306///
307/// Note that the use of `#[non_exhaustive]` also has other effects. At the
308/// least: tuple variants and tuple structs marked with `#[non_exhaustive]`
309/// cannot be pattern matched using the "normal" syntax. Instead, one must use
310/// curly braces. (Broken: `T(x, ..)`; woken: `T{0: x, ..}`).
311///
312/// (This does not seem very intentional, and with all luck will be fixed before
313/// too long.)
314fn forbid_initialization(s: &mut syn::DeriveInput) {
315 let non_exhaustive_attr = syn::parse_quote!(#[non_exhaustive]);
316 match &mut s.data {
317 // TODO(b/232969667): prevent creation of unions from safe code.
318 // (E.g. hide inside a struct.)
319 syn::Data::Union(_) => return,
320 syn::Data::Struct(data) => {
321 match &mut data.fields {
322 syn::Fields::Unit | syn::Fields::Unnamed(_) => {
323 s.attrs.insert(0, non_exhaustive_attr);
324 }
325 syn::Fields::Named(fields) => {
326 fields.named.push(syn::Field {
327 attrs: vec![],
328 vis: syn::Visibility::Inherited,
329 // TODO(jeanpierreda): better hygiene: work even if a field has the same name.
Devin Jeanpierree9866732022-07-25 05:52:09 -0700330 ident: Some(Ident::new(FIELD_FOR_MUST_USE_CTOR, Span::call_site())),
Devin Jeanpierred6f3e2a2022-07-20 18:45:42 -0700331 colon_token: Some(<syn::Token![:]>::default()),
332 ty: syn::parse_quote!([u8; 0]),
333 });
334 }
335 }
336 }
337 syn::Data::Enum(e) => {
338 // Enums can't have private fields. Instead, we need to add #[non_exhaustive] to
339 // every variant -- this makes it impossible to construct the
340 // variants.
341 for variant in &mut e.variants {
342 variant.attrs.insert(0, non_exhaustive_attr.clone());
343 }
344 }
345 }
346}
347
Devin Jeanpierrebe1efd22022-05-10 16:52:07 -0700348/// `#[recursively_pinned]` pins every field, similar to `#[pin_project]`, and
349/// marks the struct `!Unpin`.
Devin Jeanpierref643afa2022-02-23 20:50:48 +0000350///
351/// Example:
352///
353/// ```
354/// #[recursively_pinned]
355/// struct S {
356/// field: i32,
357/// }
358/// ```
359///
Devin Jeanpierre1c28ff22022-05-09 06:04:57 -0700360/// This is analogous to using pin_project, pinning every field, as so:
Devin Jeanpierref643afa2022-02-23 20:50:48 +0000361///
362/// ```
Devin Jeanpierre1c28ff22022-05-09 06:04:57 -0700363/// #[pin_project(!Unpin)]
Devin Jeanpierref643afa2022-02-23 20:50:48 +0000364/// struct S {
365/// #[pin]
366/// field: i32,
367/// }
368/// ```
Devin Jeanpierrebe1efd22022-05-10 16:52:07 -0700369///
370/// ## Arguments
371///
372/// ### `PinnedDrop`
373///
374/// To define a destructor for a recursively-pinned struct, pass `PinnedDrop`
375/// and implement the `PinnedDrop` trait.
376///
377/// `#[recursively_pinned]` prohibits implementing `Drop`, as that would make it
378/// easy to violate the `Pin` guarantee. Instead, to define a destructor, one
379/// must define a `PinnedDrop` impl, as so:
380///
381/// ```
382/// #[recursively_pinned(PinnedDrop)]
383/// struct S {
384/// field: i32,
385/// }
386///
387/// impl PinnedDrop for S {
388/// unsafe fn pinned_drop(self: Pin<&mut Self>) {
389/// println!("I am being destroyed!");
390/// }
391/// }
392/// ```
393///
394/// (This is analogous to `#[pin_project(PinnedDrop)]`.)
Devin Jeanpierre190b90a2022-05-24 06:00:34 -0700395///
Devin Jeanpierred6f3e2a2022-07-20 18:45:42 -0700396/// ## Direct initialization
397///
398/// Use the `ctor!` macro to instantiate recursively pinned types. For example:
399///
400/// ```
401/// // equivalent to `let x = Point {x: 3, y: 4}`, but uses pinned construction.
402/// emplace! {
403/// let x = ctor!(Point {x: 3, y: 4});
404/// }
405/// ```
406///
407/// Recursively pinned types cannot be created directly in safe code, as they
408/// are pinned from the very moment of their creation.
409///
410/// This is prevented either using `#[non_exhaustive]` or using a private field,
411/// depending on the type in question. For example, enums use
412/// `#[non_exhaustive]`, and structs with named fields use a private field named
413/// `__must_use_ctor_to_initialize`. This can lead to confusing error messages,
414/// so watch out!
415///
Devin Jeanpierre190b90a2022-05-24 06:00:34 -0700416/// ## Supported types
417///
418/// Structs, enums, and unions are all supported. However, unions do not receive
419/// a `pin_project` method, as there is no way to implement pin projection for
420/// unions. (One cannot know which field is active.)
Devin Jeanpierref643afa2022-02-23 20:50:48 +0000421#[proc_macro_attribute]
422pub fn recursively_pinned(args: TokenStream, item: TokenStream) -> TokenStream {
Devin Jeanpierred6f3e2a2022-07-20 18:45:42 -0700423 match recursively_pinned_impl(args.into(), item.into()) {
424 Ok(t) => t.into(),
425 Err(e) => e.into_compile_error().into(),
426 }
427}
Devin Jeanpierref643afa2022-02-23 20:50:48 +0000428
Devin Jeanpierred6f3e2a2022-07-20 18:45:42 -0700429/// A separate function for calling from tests.
430///
431/// See e.g. https://users.rust-lang.org/t/procedural-macro-api-is-used-outside-of-a-procedural-macro/30841
432fn recursively_pinned_impl(
433 args: proc_macro2::TokenStream,
434 item: proc_macro2::TokenStream,
435) -> syn::Result<proc_macro2::TokenStream> {
436 let args = syn::parse2::<RecursivelyPinnedArgs>(args)?;
437 let mut input = syn::parse2::<syn::DeriveInput>(item)?;
Devin Jeanpierref643afa2022-02-23 20:50:48 +0000438
Devin Jeanpierred6f3e2a2022-07-20 18:45:42 -0700439 let project_pin_impl = project_pin_impl(&input)?;
Devin Jeanpierref643afa2022-02-23 20:50:48 +0000440 let name = input.ident.clone();
Devin Jeanpierref643afa2022-02-23 20:50:48 +0000441
Devin Jeanpierred6f3e2a2022-07-20 18:45:42 -0700442 // Create two copies of input: one (public) has a private field that can't be
443 // instantiated. The other (only visible via
444 // RecursivelyPinned::CtorInitializedFields) doesn't have this field.
445 // This causes `ctor!(Foo {})` to work, but `Foo{}` to complain of a missing
446 // field.
447 let mut ctor_initialized_input = input.clone();
Devin Jeanpierree9866732022-07-25 05:52:09 -0700448 // Removing repr(C) triggers dead-code detection.
449 ctor_initialized_input.attrs = vec![syn::parse_quote!(#[allow(dead_code)])];
Devin Jeanpierred6f3e2a2022-07-20 18:45:42 -0700450 // TODO(jeanpierreda): This should really check for name collisions with any types
451 // used in the fields. Collisions with other names don't matter, because the
452 // type is locally defined within a narrow scope.
453 ctor_initialized_input.ident = syn::Ident::new(&format!("__CrubitCtor{name}"), name.span());
454 let ctor_initialized_name = &ctor_initialized_input.ident;
455 forbid_initialization(&mut input);
456
Devin Jeanpierreab912362022-06-09 16:49:52 -0700457 let (input_impl_generics, input_ty_generics, input_where_clause) =
458 input.generics.split_for_impl();
459
Devin Jeanpierrebe1efd22022-05-10 16:52:07 -0700460 let drop_impl = if args.is_pinned_drop {
461 quote! {
Devin Jeanpierreab912362022-06-09 16:49:52 -0700462 impl #input_impl_generics Drop for #name #input_ty_generics #input_where_clause {
Devin Jeanpierrebe1efd22022-05-10 16:52:07 -0700463 fn drop(&mut self) {
Googler594745b2023-03-03 13:56:43 -0800464 unsafe {::ctor::PinnedDrop::pinned_drop(::core::pin::Pin::new_unchecked(self))}
Devin Jeanpierrebe1efd22022-05-10 16:52:07 -0700465 }
466 }
467 }
468 } else {
469 quote! {
Devin Jeanpierreab912362022-06-09 16:49:52 -0700470 impl #input_impl_generics ::ctor::macro_internal::DoNotImplDrop for #name #input_ty_generics #input_where_clause {}
Devin Jeanpierrebe1efd22022-05-10 16:52:07 -0700471 /// A no-op PinnedDrop that will cause an error if the user also defines PinnedDrop,
472 /// due to forgetting to pass `PinnedDrop` to #[recursively_pinned(PinnedDrop)]`.
Devin Jeanpierreab912362022-06-09 16:49:52 -0700473 impl #input_impl_generics ::ctor::PinnedDrop for #name #input_ty_generics #input_where_clause {
Googler594745b2023-03-03 13:56:43 -0800474 unsafe fn pinned_drop(self: ::core::pin::Pin<&mut Self>) {}
Devin Jeanpierrebe1efd22022-05-10 16:52:07 -0700475 }
476 }
477 };
478
Devin Jeanpierred6f3e2a2022-07-20 18:45:42 -0700479 Ok(quote! {
Devin Jeanpierref643afa2022-02-23 20:50:48 +0000480 #input
Devin Jeanpierre190b90a2022-05-24 06:00:34 -0700481 #project_pin_impl
Devin Jeanpierref643afa2022-02-23 20:50:48 +0000482
Devin Jeanpierrebe1efd22022-05-10 16:52:07 -0700483 #drop_impl
Devin Jeanpierreab912362022-06-09 16:49:52 -0700484 impl #input_impl_generics !Unpin for #name #input_ty_generics #input_where_clause {}
Devin Jeanpierref643afa2022-02-23 20:50:48 +0000485
Devin Jeanpierred6f3e2a2022-07-20 18:45:42 -0700486 // Introduce a new scope to limit the blast radius of the CtorInitializedFields type.
487 // This lets us use relatively readable names: while the impl is visible outside the scope,
488 // type is otherwise not visible.
489 const _ : () = {
490 #ctor_initialized_input
491
492 unsafe impl #input_impl_generics ::ctor::RecursivelyPinned for #name #input_ty_generics #input_where_clause {
493 type CtorInitializedFields = #ctor_initialized_name #input_ty_generics;
494 }
495 };
496 })
497}
498
499#[cfg(test)]
500mod test {
501 use super::*;
502 use token_stream_matchers::assert_rs_matches;
503
504 /// Essentially a change detector, but handy for debugging.
505 ///
506 /// At time of writing, we can't write negative compilation tests, so
507 /// asserting on the output is as close as we can get. Once negative
508 /// compilation tests are added, it would be better to test various
509 /// safety features that way.
510 #[test]
511 fn test_recursively_pinned_struct() {
512 let definition =
513 recursively_pinned_impl(quote! {}, quote! {#[repr(C)] struct S {x: i32}}).unwrap();
514
515 // The struct can't be directly created, but can be created via
516 // CtorInitializedFields:
517 assert_rs_matches!(
518 definition,
519 quote! {
520 #[repr(C)]
521 struct S {
522 x: i32,
523 __must_use_ctor_to_initialize: [u8; 0]
524 }
525 }
526 );
527 assert_rs_matches!(
528 definition,
529 quote! {
530 const _: () = {
Devin Jeanpierree9866732022-07-25 05:52:09 -0700531 #[allow(dead_code)]
532 struct __CrubitCtorS {x: i32}
533 unsafe impl ::ctor::RecursivelyPinned for S {
534 type CtorInitializedFields = __CrubitCtorS;
535 }
Devin Jeanpierred6f3e2a2022-07-20 18:45:42 -0700536 };
537 }
538 );
539
540 // The type is non-Unpin:
541 assert_rs_matches!(
542 definition,
543 quote! {
544 impl !Unpin for S {}
545 }
546 );
547
548 // The remaining features of the generated output are better tested via
549 // real tests that exercise the code.
550 }
551
552 /// The enum version of `test_recursively_pinned_struct`.
553 #[test]
554 fn test_recursively_pinned_enum() {
555 let definition = recursively_pinned_impl(
556 quote! {},
557 quote! {
558 #[repr(C)]
559 enum E {
560 A,
561 B(i32),
562 }
563 },
564 )
565 .unwrap();
566
567 // The enum variants can't be directly created, but can be created via
568 // CtorInitializedFields:
569 assert_rs_matches!(
570 definition,
571 quote! {
572 #[repr(C)]
573 enum E {
574 #[non_exhaustive]
575 A,
576 #[non_exhaustive]
577 B(i32),
578 }
579 }
580 );
581 assert_rs_matches!(
582 definition,
583 quote! {
584 const _: () = {
Devin Jeanpierree9866732022-07-25 05:52:09 -0700585 #[allow(dead_code)]
Devin Jeanpierred6f3e2a2022-07-20 18:45:42 -0700586 enum __CrubitCtorE {
587 A,
588 B(i32),
589 }
590 unsafe impl ::ctor::RecursivelyPinned for E {
591 type CtorInitializedFields = __CrubitCtorE;
592 }
593 };
594 }
595 );
596
597 // The type is non-Unpin:
598 assert_rs_matches!(
599 definition,
600 quote! {
601 impl !Unpin for E {}
602 }
603 );
604
605 // The remaining features of the generated output are better tested via
606 // real tests that exercise the code.
607 }
Devin Jeanpierref643afa2022-02-23 20:50:48 +0000608}