From 314b5e56e9a82aa616f87e3f3f0f51dcb8550260 Mon Sep 17 00:00:00 2001 From: Yuki Kishimoto Date: Mon, 30 Dec 2024 15:46:34 +0100 Subject: [PATCH] util: simplify `ReusableBoxFuture::try_set` logic --- tokio-util/src/sync/reusable_box.rs | 86 +++-------------------------- 1 file changed, 8 insertions(+), 78 deletions(-) diff --git a/tokio-util/src/sync/reusable_box.rs b/tokio-util/src/sync/reusable_box.rs index 1fae38cc6ff..d9609a9c65a 100644 --- a/tokio-util/src/sync/reusable_box.rs +++ b/tokio-util/src/sync/reusable_box.rs @@ -1,9 +1,7 @@ -use std::alloc::Layout; use std::fmt; -use std::future::{self, Future}; -use std::mem::{self, ManuallyDrop}; +use std::future::Future; +use std::mem::{align_of_val, size_of_val}; use std::pin::Pin; -use std::ptr; use std::task::{Context, Poll}; /// A reusable `Pin + Send + 'a>>`. @@ -47,24 +45,14 @@ impl<'a, T> ReusableBoxFuture<'a, T> { where F: Future + Send + 'a, { - // If we try to inline the contents of this function, the type checker complains because - // the bound `T: 'a` is not satisfied in the call to `pending()`. But by putting it in an - // inner function that doesn't have `T` as a generic parameter, we implicitly get the bound - // `F::Output: 'a` transitively through `F: 'a`, allowing us to call `pending()`. - #[inline(always)] - fn real_try_set<'a, F>( - this: &mut ReusableBoxFuture<'a, F::Output>, - future: F, - ) -> Result<(), F> - where - F: Future + Send + 'a, + if size_of_val(&*self.boxed) == size_of_val(&future) + && align_of_val(&*self.boxed) == align_of_val(&future) { - // future::Pending is a ZST so this never allocates. - let boxed = mem::replace(&mut this.boxed, Box::pin(future::pending())); - reuse_pin_box(boxed, future, |boxed| this.boxed = Pin::from(boxed)) + self.boxed = Box::pin(future); + Ok(()) + } else { + Err(future) } - - real_try_set(self, future) } /// Get a pinned reference to the underlying future. @@ -97,61 +85,3 @@ impl fmt::Debug for ReusableBoxFuture<'_, T> { f.debug_struct("ReusableBoxFuture").finish() } } - -fn reuse_pin_box(boxed: Pin>, new_value: U, callback: F) -> Result -where - F: FnOnce(Box) -> O, -{ - let layout = Layout::for_value::(&*boxed); - if layout != Layout::new::() { - return Err(new_value); - } - - // SAFETY: We don't ever construct a non-pinned reference to the old `T` from now on, and we - // always drop the `T`. - let raw: *mut T = Box::into_raw(unsafe { Pin::into_inner_unchecked(boxed) }); - - // When dropping the old value panics, we still want to call `callback` — so move the rest of - // the code into a guard type. - let guard = CallOnDrop::new(|| { - let raw: *mut U = raw.cast::(); - unsafe { raw.write(new_value) }; - - // SAFETY: - // - `T` and `U` have the same layout. - // - `raw` comes from a `Box` that uses the same allocator as this one. - // - `raw` points to a valid instance of `U` (we just wrote it in). - let boxed = unsafe { Box::from_raw(raw) }; - - callback(boxed) - }); - - // Drop the old value. - unsafe { ptr::drop_in_place(raw) }; - - // Run the rest of the code. - Ok(guard.call()) -} - -struct CallOnDrop O> { - f: ManuallyDrop, -} - -impl O> CallOnDrop { - fn new(f: F) -> Self { - let f = ManuallyDrop::new(f); - Self { f } - } - fn call(self) -> O { - let mut this = ManuallyDrop::new(self); - let f = unsafe { ManuallyDrop::take(&mut this.f) }; - f() - } -} - -impl O> Drop for CallOnDrop { - fn drop(&mut self) { - let f = unsafe { ManuallyDrop::take(&mut self.f) }; - f(); - } -}