1use std::cell::RefCell;
4use std::ops::{Bound, Range};
5use std::sync::Once;
6use std::{fmt, mem, panic};
7
8use crate::bridge::{
9 ApiTags, BridgeConfig, Buffer, Decode, Diagnostic, Encode, ExpnGlobals, Literal, PanicMessage,
10 TokenTree, closure, handle,
11};
12
13pub(crate) struct TokenStream {
14 handle: handle::Handle,
15}
16
17impl !Send for TokenStream {}
18impl !Sync for TokenStream {}
19
20impl Drop for TokenStream {
22 fn drop(&mut self) {
23 Methods::ts_drop(TokenStream { handle: self.handle });
24 }
25}
26
27impl<S> Encode<S> for TokenStream {
28 #[inline]
29 fn encode(self, w: &mut Buffer, s: &mut S) {
30 mem::ManuallyDrop::new(self).handle.encode(w, s);
31 }
32}
33
34impl<S> Encode<S> for &TokenStream {
35 #[inline]
36 fn encode(self, w: &mut Buffer, s: &mut S) {
37 self.handle.encode(w, s);
38 }
39}
40
41impl<S> Decode<'_, '_, S> for TokenStream {
42 #[inline]
43 fn decode(r: &mut &[u8], s: &mut S) -> Self {
44 TokenStream { handle: handle::Handle::decode(r, s) }
45 }
46}
47
48impl Encode<()> for crate::TokenStream {
49 #[inline]
50 fn encode(self, w: &mut Buffer, s: &mut ()) {
51 self.0.encode(w, s)
52 }
53}
54
55impl Decode<'_, '_, ()> for crate::TokenStream {
56 #[inline]
57 fn decode(r: &mut &[u8], s: &mut ()) -> Self {
58 crate::TokenStream(Some(Decode::decode(r, s)))
59 }
60}
61
62#[derive(#[automatically_derived]
impl ::core::marker::Copy for Span { }Copy, #[automatically_derived]
impl ::core::clone::Clone for Span {
#[inline]
fn clone(&self) -> Span {
let _: ::core::clone::AssertParamIsClone<handle::Handle>;
*self
}
}Clone, #[automatically_derived]
impl ::core::cmp::PartialEq for Span {
#[inline]
fn eq(&self, other: &Span) -> bool { self.handle == other.handle }
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for Span {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_fields_are_eq(&self) {
let _: ::core::cmp::AssertParamIsEq<handle::Handle>;
}
}Eq, #[automatically_derived]
impl ::core::hash::Hash for Span {
#[inline]
fn hash<__H: ::core::hash::Hasher>(&self, state: &mut __H) {
::core::hash::Hash::hash(&self.handle, state)
}
}Hash)]
63pub(crate) struct Span {
64 handle: handle::Handle,
65}
66
67impl !Send for Span {}
68impl !Sync for Span {}
69
70impl<S> Encode<S> for Span {
71 #[inline]
72 fn encode(self, w: &mut Buffer, s: &mut S) {
73 self.handle.encode(w, s);
74 }
75}
76
77impl<S> Decode<'_, '_, S> for Span {
78 #[inline]
79 fn decode(r: &mut &[u8], s: &mut S) -> Self {
80 Span { handle: handle::Handle::decode(r, s) }
81 }
82}
83
84impl Clone for TokenStream {
85 fn clone(&self) -> Self {
86 Methods::ts_clone(self)
87 }
88}
89
90impl Span {
91 pub(crate) fn def_site() -> Span {
92 Bridge::with(|bridge| bridge.globals.def_site)
93 }
94
95 pub(crate) fn call_site() -> Span {
96 Bridge::with(|bridge| bridge.globals.call_site)
97 }
98
99 pub(crate) fn mixed_site() -> Span {
100 Bridge::with(|bridge| bridge.globals.mixed_site)
101 }
102}
103
104impl fmt::Debug for Span {
105 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
106 f.write_str(&Methods::span_debug(*self))
107 }
108}
109
110pub(crate) use super::Methods;
111pub(crate) use super::symbol::Symbol;
112
113macro_rules! define_client_side {
114 (
115 $(fn $method:ident($($arg:ident: $arg_ty:ty),* $(,)?) $(-> $ret_ty:ty)?;)*
116 ) => {
117 impl Methods {
118 $(pub(crate) fn $method($($arg: $arg_ty),*) $(-> $ret_ty)? {
119 Bridge::with(|bridge| {
120 let mut buf = bridge.cached_buffer.take();
121
122 buf.clear();
123 ApiTags::$method.encode(&mut buf, &mut ());
124 $($arg.encode(&mut buf, &mut ());)*
125
126 buf = bridge.dispatch.call(buf);
127
128 let r = Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
129
130 bridge.cached_buffer = buf;
131
132 r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
133 })
134 })*
135 }
136 }
137}
138impl Methods {
pub(crate) fn injected_env_var(var: &str) -> Option<String> {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::injected_env_var.encode(&mut buf, &mut ());
var.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn track_env_var(var: &str, value: Option<&str>) {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::track_env_var.encode(&mut buf, &mut ());
var.encode(&mut buf, &mut ());
value.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn track_path(path: &str) {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::track_path.encode(&mut buf, &mut ());
path.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn literal_from_str(s: &str)
-> Result<Literal<Span, Symbol>, String> {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::literal_from_str.encode(&mut buf, &mut ());
s.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn emit_diagnostic(diagnostic: Diagnostic<Span>) {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::emit_diagnostic.encode(&mut buf, &mut ());
diagnostic.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn ts_drop(stream: TokenStream) {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::ts_drop.encode(&mut buf, &mut ());
stream.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn ts_clone(stream: &TokenStream) -> TokenStream {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::ts_clone.encode(&mut buf, &mut ());
stream.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn ts_is_empty(stream: &TokenStream) -> bool {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::ts_is_empty.encode(&mut buf, &mut ());
stream.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn ts_expand_expr(stream: &TokenStream)
-> Result<TokenStream, ()> {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::ts_expand_expr.encode(&mut buf, &mut ());
stream.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn ts_from_str(src: &str) -> Result<TokenStream, String> {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::ts_from_str.encode(&mut buf, &mut ());
src.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn ts_to_string(stream: &TokenStream) -> String {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::ts_to_string.encode(&mut buf, &mut ());
stream.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn ts_from_token_tree(tree:
TokenTree<TokenStream, Span, Symbol>) -> TokenStream {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::ts_from_token_tree.encode(&mut buf, &mut ());
tree.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn ts_concat_trees(base: Option<TokenStream>,
trees: Vec<TokenTree<TokenStream, Span, Symbol>>) -> TokenStream {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::ts_concat_trees.encode(&mut buf, &mut ());
base.encode(&mut buf, &mut ());
trees.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn ts_concat_streams(base: Option<TokenStream>,
streams: Vec<TokenStream>) -> TokenStream {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::ts_concat_streams.encode(&mut buf, &mut ());
base.encode(&mut buf, &mut ());
streams.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn ts_into_trees(stream: TokenStream)
-> Vec<TokenTree<TokenStream, Span, Symbol>> {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::ts_into_trees.encode(&mut buf, &mut ());
stream.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn span_debug(span: Span) -> String {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::span_debug.encode(&mut buf, &mut ());
span.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn span_parent(span: Span) -> Option<Span> {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::span_parent.encode(&mut buf, &mut ());
span.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn span_source(span: Span) -> Span {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::span_source.encode(&mut buf, &mut ());
span.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn span_byte_range(span: Span) -> Range<usize> {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::span_byte_range.encode(&mut buf, &mut ());
span.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn span_start(span: Span) -> Span {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::span_start.encode(&mut buf, &mut ());
span.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn span_end(span: Span) -> Span {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::span_end.encode(&mut buf, &mut ());
span.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn span_line(span: Span) -> usize {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::span_line.encode(&mut buf, &mut ());
span.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn span_column(span: Span) -> usize {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::span_column.encode(&mut buf, &mut ());
span.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn span_file(span: Span) -> String {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::span_file.encode(&mut buf, &mut ());
span.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn span_local_file(span: Span) -> Option<String> {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::span_local_file.encode(&mut buf, &mut ());
span.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn span_join(span: Span, other: Span) -> Option<Span> {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::span_join.encode(&mut buf, &mut ());
span.encode(&mut buf, &mut ());
other.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn span_subspan(span: Span, start: Bound<usize>,
end: Bound<usize>) -> Option<Span> {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::span_subspan.encode(&mut buf, &mut ());
span.encode(&mut buf, &mut ());
start.encode(&mut buf, &mut ());
end.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn span_resolved_at(span: Span, at: Span) -> Span {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::span_resolved_at.encode(&mut buf, &mut ());
span.encode(&mut buf, &mut ());
at.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn span_source_text(span: Span) -> Option<String> {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::span_source_text.encode(&mut buf, &mut ());
span.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn span_save_span(span: Span) -> usize {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::span_save_span.encode(&mut buf, &mut ());
span.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn span_recover_proc_macro_span(id: usize) -> Span {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::span_recover_proc_macro_span.encode(&mut buf,
&mut ());
id.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
pub(crate) fn symbol_normalize_and_validate_ident(string: &str)
-> Result<Symbol, ()> {
Bridge::with(|bridge|
{
let mut buf = bridge.cached_buffer.take();
buf.clear();
ApiTags::symbol_normalize_and_validate_ident.encode(&mut buf,
&mut ());
string.encode(&mut buf, &mut ());
buf = bridge.dispatch.call(buf);
let r =
Result::<_, PanicMessage>::decode(&mut &buf[..], &mut ());
bridge.cached_buffer = buf;
r.unwrap_or_else(|e| panic::resume_unwind(e.into()))
})
}
}with_api!(define_client_side, TokenStream, Span, Symbol);
139
140struct Bridge<'a> {
141 cached_buffer: Buffer,
144
145 dispatch: closure::Closure<'a>,
147
148 globals: ExpnGlobals<Span>,
150}
151
152impl<'a> !Send for Bridge<'a> {}
153impl<'a> !Sync for Bridge<'a> {}
154
155#[allow(unsafe_code)]
156mod state {
157 use std::cell::{Cell, RefCell};
158 use std::ptr;
159
160 use super::Bridge;
161
162 const BRIDGE_STATE: ::std::thread::LocalKey<Cell<*const ()>> =
{
const __RUST_STD_INTERNAL_INIT: Cell<*const ()> =
{ Cell::new(ptr::null()) };
unsafe {
::std::thread::LocalKey::new(const {
if ::std::mem::needs_drop::<Cell<*const ()>>() {
|_|
{
#[thread_local]
static __RUST_STD_INTERNAL_VAL:
::std::thread::local_impl::EagerStorage<Cell<*const ()>> =
::std::thread::local_impl::EagerStorage::new(__RUST_STD_INTERNAL_INIT);
__RUST_STD_INTERNAL_VAL.get()
}
} else {
|_|
{
#[thread_local]
static __RUST_STD_INTERNAL_VAL: Cell<*const ()> =
__RUST_STD_INTERNAL_INIT;
&__RUST_STD_INTERNAL_VAL
}
}
})
}
};thread_local! {
163 static BRIDGE_STATE: Cell<*const ()> = const { Cell::new(ptr::null()) };
164 }
165
166 pub(super) fn set<'bridge, R>(state: &RefCell<Bridge<'bridge>>, f: impl FnOnce() -> R) -> R {
167 struct RestoreOnDrop(*const ());
168 impl Drop for RestoreOnDrop {
169 fn drop(&mut self) {
170 BRIDGE_STATE.set(self.0);
171 }
172 }
173
174 let inner = ptr::from_ref(state).cast();
175 let outer = BRIDGE_STATE.replace(inner);
176 let _restore = RestoreOnDrop(outer);
177
178 f()
179 }
180
181 pub(super) fn with<R>(
182 f: impl for<'bridge> FnOnce(Option<&RefCell<Bridge<'bridge>>>) -> R,
183 ) -> R {
184 let state = BRIDGE_STATE.get();
185 let bridge = unsafe { state.cast::<RefCell<Bridge<'static>>>().as_ref() };
193 f(bridge)
194 }
195}
196
197impl Bridge<'_> {
198 fn with<R>(f: impl FnOnce(&mut Bridge<'_>) -> R) -> R {
199 state::with(|state| {
200 let bridge = state.expect("procedural macro API is used outside of a procedural macro");
201 let mut bridge = bridge
202 .try_borrow_mut()
203 .expect("procedural macro API is used while it's already in use");
204 f(&mut bridge)
205 })
206 }
207}
208
209pub(crate) fn is_available() -> bool {
210 state::with(|s| s.is_some())
211}
212
213#[repr(C)]
220#[derive(#[automatically_derived]
impl ::core::marker::Copy for Client { }Copy, #[automatically_derived]
impl ::core::clone::Clone for Client {
#[inline]
fn clone(&self) -> Client {
let _:
::core::clone::AssertParamIsClone<extern "C" fn(BridgeConfig<'_>)
-> Buffer>;
*self
}
}Clone)]
221pub struct Client {
222 pub(super) run: extern "C" fn(BridgeConfig<'_>) -> Buffer,
223}
224
225fn maybe_install_panic_hook(force_show_panics: bool) {
226 static HIDE_PANICS_DURING_EXPANSION: Once = Once::new();
229 HIDE_PANICS_DURING_EXPANSION.call_once(|| {
230 let prev = panic::take_hook();
231 panic::set_hook(Box::new(move |info| {
232 if force_show_panics || !is_available() || !info.can_unwind() {
237 prev(info)
238 }
239 }));
240 });
241}
242
243fn run_client<A: for<'a, 's> Decode<'a, 's, ()>>(
246 config: BridgeConfig<'_>,
247 f: impl FnOnce(A) -> crate::TokenStream,
248) -> Buffer {
249 let BridgeConfig { input: mut buf, dispatch, force_show_panics, .. } = config;
250
251 let res = panic::catch_unwind(panic::AssertUnwindSafe(|| {
252 maybe_install_panic_hook(force_show_panics);
253
254 Symbol::invalidate_all();
256
257 let reader = &mut &buf[..];
258 let (globals, input) = <(ExpnGlobals<Span>, A)>::decode(reader, &mut ());
259
260 let state = RefCell::new(Bridge { cached_buffer: buf.take(), dispatch, globals });
262
263 let output = state::set(&state, || f(input));
264
265 buf = RefCell::into_inner(state).cached_buffer;
267
268 output
269 }));
270
271 buf.clear();
273 res.map_err(PanicMessage::from).encode(&mut buf, &mut ());
274
275 Symbol::invalidate_all();
278 buf
279}
280
281impl Client {
282 pub const fn expand1(f: impl Fn(crate::TokenStream) -> crate::TokenStream + Copy) -> Self {
283 Client {
284 run: super::selfless_reify::reify_to_extern_c_fn_hrt_bridge(move |bridge| {
285 run_client(bridge, |input| f(input))
286 }),
287 }
288 }
289
290 pub const fn expand2(
291 f: impl Fn(crate::TokenStream, crate::TokenStream) -> crate::TokenStream + Copy,
292 ) -> Self {
293 Client {
294 run: super::selfless_reify::reify_to_extern_c_fn_hrt_bridge(move |bridge| {
295 run_client(bridge, |(input, input2)| f(input, input2))
296 }),
297 }
298 }
299}