diff --git a/library/proc_macro/src/bridge/client.rs b/library/proc_macro/src/bridge/client.rs index cdb2bac260757..144de5822c50c 100644 --- a/library/proc_macro/src/bridge/client.rs +++ b/library/proc_macro/src/bridge/client.rs @@ -2,6 +2,7 @@ use super::*; +use std::cell::Cell; use std::marker::PhantomData; macro_rules! define_handles { @@ -256,7 +257,7 @@ macro_rules! define_client_side { api_tags::Method::$name(api_tags::$name::$method).encode(&mut b, &mut ()); reverse_encode!(b; $($arg),*); - b = bridge.dispatch.call(b); + b = (bridge.dispatch)(b); let r = Result::<_, PanicMessage>::decode(&mut &b[..], &mut ()); @@ -270,48 +271,64 @@ macro_rules! define_client_side { } with_api!(self, self, define_client_side); -enum BridgeState<'a> { +enum BridgeState { /// No server is currently connected to this client. NotConnected, /// A server is connected and available for requests. - Connected(Bridge<'a>), + Connected(Bridge), /// Access to the bridge is being exclusively acquired /// (e.g., during `BridgeState::with`). InUse, } -enum BridgeStateL {} +impl BridgeState { + /// Sets the thread-local `BridgeState` to `replacement` while + /// running `f`, which gets the old `BridgeState`, mutably. + /// + /// The state will be restored after `f` exits, even + /// by panic, including modifications made to it by `f`. + fn replace_during(replacement: Self, f: impl FnOnce(&mut Self) -> R) -> R { + /// Wrapper that ensures that a cell always gets filled + /// (with the original state, optionally changed by `f`), + /// even if `f` had panicked. + struct PutBackOnDrop<'a, T> { + cell: &'a Cell, + value: Option, + } -impl<'a> scoped_cell::ApplyL<'a> for BridgeStateL { - type Out = BridgeState<'a>; -} + impl<'a, T> Drop for PutBackOnDrop<'a, T> { + fn drop(&mut self) { + self.cell.set(self.value.take().unwrap()); + } + } -thread_local! { - static BRIDGE_STATE: scoped_cell::ScopedCell = - scoped_cell::ScopedCell::new(BridgeState::NotConnected); -} + thread_local! { + static BRIDGE_STATE: Cell = Cell::new(BridgeState::NotConnected); + } + BRIDGE_STATE.with(|state| { + let mut put_back_on_drop = + PutBackOnDrop { cell: state, value: Some(state.replace(replacement)) }; + + f(put_back_on_drop.value.as_mut().unwrap()) + }) + } -impl BridgeState<'_> { /// Take exclusive control of the thread-local /// `BridgeState`, and pass it to `f`, mutably. + /// /// The state will be restored after `f` exits, even /// by panic, including modifications made to it by `f`. /// /// N.B., while `f` is running, the thread-local state /// is `BridgeState::InUse`. - fn with(f: impl FnOnce(&mut BridgeState<'_>) -> R) -> R { - BRIDGE_STATE.with(|state| { - state.replace(BridgeState::InUse, |mut state| { - // FIXME(#52812) pass `f` directly to `replace` when `RefMutL` is gone - f(&mut *state) - }) - }) + fn with(f: impl FnOnce(&mut Self) -> R) -> R { + Self::replace_during(BridgeState::InUse, f) } } -impl Bridge<'_> { +impl Bridge { pub(crate) fn is_available() -> bool { BridgeState::with(|state| match state { BridgeState::Connected(_) | BridgeState::InUse => true, @@ -337,10 +354,10 @@ impl Bridge<'_> { })); }); - BRIDGE_STATE.with(|state| state.set(BridgeState::Connected(self), f)) + BridgeState::replace_during(BridgeState::Connected(self), |_| f()) } - fn with(f: impl FnOnce(&mut Bridge<'_>) -> R) -> R { + fn with(f: impl FnOnce(&mut Self) -> R) -> R { BridgeState::with(|state| match state { BridgeState::NotConnected => { panic!("procedural macro API is used outside of a procedural macro"); @@ -367,7 +384,7 @@ pub struct Client { // FIXME(eddyb) use a reference to the `static COUNTERS`, instead of // a wrapper `fn` pointer, once `const fn` can reference `static`s. pub(super) get_handle_counters: extern "C" fn() -> &'static HandleCounters, - pub(super) run: extern "C" fn(Bridge<'_>, F) -> Buffer, + pub(super) run: extern "C" fn(Bridge, F) -> Buffer, pub(super) f: F, } @@ -375,7 +392,7 @@ pub struct Client { /// deserializing input and serializing output. // FIXME(eddyb) maybe replace `Bridge::enter` with this? fn run_client DecodeMut<'a, 's, ()>, R: Encode<()>>( - mut bridge: Bridge<'_>, + mut bridge: Bridge, f: impl FnOnce(A) -> R, ) -> Buffer { // The initial `cached_buffer` contains the input. @@ -418,7 +435,7 @@ fn run_client DecodeMut<'a, 's, ()>, R: Encode<()>>( impl Client crate::TokenStream> { pub const fn expand1(f: fn(crate::TokenStream) -> crate::TokenStream) -> Self { extern "C" fn run( - bridge: Bridge<'_>, + bridge: Bridge, f: impl FnOnce(crate::TokenStream) -> crate::TokenStream, ) -> Buffer { run_client(bridge, |input| f(crate::TokenStream(input)).0) @@ -432,7 +449,7 @@ impl Client crate::TokenStream> { f: fn(crate::TokenStream, crate::TokenStream) -> crate::TokenStream, ) -> Self { extern "C" fn run( - bridge: Bridge<'_>, + bridge: Bridge, f: impl FnOnce(crate::TokenStream, crate::TokenStream) -> crate::TokenStream, ) -> Buffer { run_client(bridge, |(input, input2)| { diff --git a/library/proc_macro/src/bridge/closure.rs b/library/proc_macro/src/bridge/closure.rs deleted file mode 100644 index 06f76d2fc9140..0000000000000 --- a/library/proc_macro/src/bridge/closure.rs +++ /dev/null @@ -1,28 +0,0 @@ -//! Closure type (equivalent to `&mut dyn FnMut(A) -> R`) that's `repr(C)`. - -use std::marker::PhantomData; - -#[repr(C)] -pub struct Closure<'a, A, R> { - call: unsafe extern "C" fn(*mut Env, A) -> R, - env: *mut Env, - // Ensure Closure is !Send and !Sync - _marker: PhantomData<*mut &'a mut ()>, -} - -struct Env; - -impl<'a, A, R, F: FnMut(A) -> R> From<&'a mut F> for Closure<'a, A, R> { - fn from(f: &'a mut F) -> Self { - unsafe extern "C" fn call R>(env: *mut Env, arg: A) -> R { - (*(env as *mut _ as *mut F))(arg) - } - Closure { call: call::, env: f as *mut _ as *mut Env, _marker: PhantomData } - } -} - -impl<'a, A, R> Closure<'a, A, R> { - pub fn call(&mut self, arg: A) -> R { - unsafe { (self.call)(self.env, arg) } - } -} diff --git a/library/proc_macro/src/bridge/mod.rs b/library/proc_macro/src/bridge/mod.rs index f7c9df6564f87..0368a4f2243c9 100644 --- a/library/proc_macro/src/bridge/mod.rs +++ b/library/proc_macro/src/bridge/mod.rs @@ -199,8 +199,6 @@ macro_rules! reverse_decode { mod buffer; #[forbid(unsafe_code)] pub mod client; -#[allow(unsafe_code)] -mod closure; #[forbid(unsafe_code)] mod handle; #[macro_use] @@ -221,13 +219,13 @@ use rpc::{Decode, DecodeMut, Encode, Reader, Writer}; /// field of `client::Client`. The client holds its copy of the `Bridge` /// in TLS during its execution (`Bridge::{enter, with}` in `client.rs`). #[repr(C)] -pub struct Bridge<'a> { +pub struct Bridge { /// Reusable buffer (only `clear`-ed, never shrunk), primarily /// used for making requests, but also for passing input to client. cached_buffer: Buffer, /// Server-side function that the client uses to make requests. - dispatch: closure::Closure<'a, Buffer, Buffer>, + dispatch: extern "C" fn(Buffer) -> Buffer, /// If 'true', always invoke the default panic hook force_show_panics: bool, diff --git a/library/proc_macro/src/bridge/scoped_cell.rs b/library/proc_macro/src/bridge/scoped_cell.rs index 2cde1f65adf9c..99d3c468b1c7d 100644 --- a/library/proc_macro/src/bridge/scoped_cell.rs +++ b/library/proc_macro/src/bridge/scoped_cell.rs @@ -41,9 +41,10 @@ impl ScopedCell { /// Sets the value in `self` to `replacement` while /// running `f`, which gets the old value, mutably. + /// /// The old value will be restored after `f` exits, even /// by panic, including modifications made to it by `f`. - pub fn replace<'a, R>( + pub fn replace_during<'a, R>( &self, replacement: >::Out, f: impl for<'b, 'c> FnOnce(RefMutL<'b, 'c, T>) -> R, @@ -76,6 +77,6 @@ impl ScopedCell { /// Sets the value in `self` to `value` while running `f`. pub fn set(&self, value: >::Out, f: impl FnOnce() -> R) -> R { - self.replace(value, |_| f()) + self.replace_during(value, |_| f()) } } diff --git a/library/proc_macro/src/bridge/server.rs b/library/proc_macro/src/bridge/server.rs index 28a67663efcc3..185b9c9377efd 100644 --- a/library/proc_macro/src/bridge/server.rs +++ b/library/proc_macro/src/bridge/server.rs @@ -243,7 +243,7 @@ fn run_bridge_and_client( strategy: &impl ExecutionStrategy, server_thread_dispatch: impl FnMut(Buffer) -> Buffer, input: Buffer, - run_client: extern "C" fn(Bridge<'_>, D) -> Buffer, + run_client: extern "C" fn(Bridge, D) -> Buffer, client_data: D, force_show_panics: bool, ) -> Buffer { @@ -268,18 +268,21 @@ fn run_bridge_and_client( strategy.cross_thread_dispatch(server_thread_dispatch, move |client_thread_dispatch| { CLIENT_THREAD_DISPATCH.with(|dispatch_slot| { dispatch_slot.set(Some(client_thread_dispatch), || { - let mut dispatch = |b| { + extern "C" fn dispatch(b: Buffer) -> Buffer { CLIENT_THREAD_DISPATCH.with(|dispatch_slot| { - dispatch_slot.replace(None, |mut client_thread_dispatch| { - client_thread_dispatch.as_mut().unwrap()(b) + dispatch_slot.replace_during(None, |mut client_thread_dispatch| { + client_thread_dispatch.as_mut().expect( + "proc_macro::bridge::server: dispatch used on \ + thread lacking an active server-side dispatcher", + )(b) }) }) - }; + } run_client( Bridge { cached_buffer: input, - dispatch: (&mut dispatch).into(), + dispatch, force_show_panics, _marker: marker::PhantomData, }, @@ -300,7 +303,7 @@ fn run_server< handle_counters: &'static client::HandleCounters, server: S, input: I, - run_client: extern "C" fn(Bridge<'_>, D) -> Buffer, + run_client: extern "C" fn(Bridge, D) -> Buffer, client_data: D, force_show_panics: bool, ) -> Result {