/build/source/nativelink-util/src/origin_context.rs
Line | Count | Source |
1 | | // Copyright 2024 The NativeLink Authors. All rights reserved. |
2 | | // |
3 | | // Licensed under the Apache License, Version 2.0 (the "License"); |
4 | | // you may not use this file except in compliance with the License. |
5 | | // You may obtain a copy of the License at |
6 | | // |
7 | | // http://www.apache.org/licenses/LICENSE-2.0 |
8 | | // |
9 | | // Unless required by applicable law or agreed to in writing, software |
10 | | // distributed under the License is distributed on an "AS IS" BASIS, |
11 | | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
12 | | // See the License for the specific language governing permissions and |
13 | | // limitations under the License. |
14 | | |
15 | | use core::panic; |
16 | | use std::any::Any; |
17 | | use std::cell::RefCell; |
18 | | use std::clone::Clone; |
19 | | use std::collections::HashMap; |
20 | | use std::mem::ManuallyDrop; |
21 | | use std::pin::Pin; |
22 | | use std::ptr::from_ref; |
23 | | use std::sync::Arc; |
24 | | use std::task::{Context, Poll}; |
25 | | |
26 | | use futures::Future; |
27 | | use nativelink_error::{make_err, Code, Error}; |
28 | | use pin_project_lite::pin_project; |
29 | | use tracing::instrument::Instrumented; |
30 | | use tracing::{Instrument, Span}; |
31 | | |
32 | | use crate::background_spawn; |
33 | | |
34 | | /// Make a symbol that represents a unique memory pointer that is |
35 | | /// constant and chosen at compile time. This enables two modules |
36 | | /// to use that memory location to reference data that lives in a |
37 | | /// shared module without the two modules knowing about each other. |
38 | | /// For example, let's say we have a context that holds anonymous |
39 | | /// data; we can use these symbols to let one module set the data |
40 | | /// and tie the data to a symbol and another module read the data |
41 | | /// with the symbol, without the two modules knowing about each other. |
42 | | #[macro_export] |
43 | | macro_rules! make_symbol { |
44 | | ($name:ident, $type:ident) => { |
45 | | #[no_mangle] |
46 | | #[used] |
47 | | pub static $name: $crate::origin_context::NLSymbol<$type> = |
48 | | $crate::origin_context::NLSymbol { |
49 | | name: concat!(module_path!(), "::", stringify!($name)), |
50 | | _phantom: std::marker::PhantomData {}, |
51 | | }; |
52 | | }; |
53 | | } |
54 | | |
55 | | // Symbol that represents the identity of the origin of a request. |
56 | | // See: IdentityHeaderSpec for details. |
57 | | make_symbol!(ORIGIN_IDENTITY, String); |
58 | | |
59 | | pub struct NLSymbol<T: Send + Sync + 'static> { |
60 | | pub name: &'static str, |
61 | | pub _phantom: std::marker::PhantomData<T>, |
62 | | } |
63 | | |
64 | | impl<T: Send + Sync + 'static> Symbol for NLSymbol<T> { |
65 | | type Type = T; |
66 | | |
67 | 0 | fn name(&self) -> &'static str { |
68 | 0 | self.name |
69 | 0 | } |
70 | | } |
71 | | |
72 | | pub type RawSymbol = std::os::raw::c_void; |
73 | | |
74 | | pub trait Symbol { |
75 | | type Type: 'static; |
76 | | |
77 | 0 | fn name(&self) -> &'static str { |
78 | 0 | std::any::type_name::<Self>() |
79 | 0 | } |
80 | | |
81 | 85 | fn as_ptr(&'static self) -> *const RawSymbol { |
82 | 85 | from_ref::<Self>(self).cast::<RawSymbol>() |
83 | 85 | } |
84 | | } |
85 | | |
86 | | /// Simple wrapper around a raw symbol pointer. |
87 | | /// This allows us to bypass the unsafe undefined behavior check |
88 | | /// when using raw pointers by manually implementing Send and Sync. |
89 | | #[derive(Eq, PartialEq, Hash, Clone)] |
90 | | #[repr(transparent)] |
91 | | pub struct RawSymbolWrapper(pub *const RawSymbol); |
92 | | |
93 | | unsafe impl Send for RawSymbolWrapper {} |
94 | | unsafe impl Sync for RawSymbolWrapper {} |
95 | | |
96 | | /// Context used to store data about the origin of a request. |
97 | | #[derive(Default, Clone)] |
98 | | pub struct OriginContext { |
99 | | data: HashMap<RawSymbolWrapper, Arc<dyn Any + Send + Sync + 'static>>, |
100 | | } |
101 | | |
102 | | impl OriginContext { |
103 | | /// Creates a new (empty) context. |
104 | 301 | pub fn new() -> Self { |
105 | 301 | Self::default() |
106 | 301 | } |
107 | | |
108 | | /// Sets the value for a given symbol on the context. |
109 | 40 | pub fn set_value<T: Any + Send + Sync + 'static>( |
110 | 40 | &mut self, |
111 | 40 | symbol: &'static impl Symbol<Type = T>, |
112 | 40 | value: Arc<T>, |
113 | 40 | ) -> Option<Arc<dyn Any + Send + Sync + 'static>> { |
114 | 40 | self.data.insert(RawSymbolWrapper(symbol.as_ptr()), value) |
115 | 40 | } |
116 | | |
117 | | /// Gets the value current set for a given symbol on the context. |
118 | | #[inline] |
119 | 45 | pub fn get_value<T: Send + Sync + 'static>( |
120 | 45 | &self, |
121 | 45 | symbol: &'static impl Symbol<Type = T>, |
122 | 45 | ) -> Result<Option<Arc<T>>, Error> { |
123 | 45 | self.data |
124 | 45 | .get(&RawSymbolWrapper(symbol.as_ptr())) |
125 | 45 | .map_or(Ok(None), |value| { |
126 | 2 | Arc::downcast::<T>(value.clone()).map_or_else( |
127 | 2 | |_| { |
128 | 0 | Err(make_err!( |
129 | 0 | Code::Internal, |
130 | 0 | "Failed to downcast symbol: {}", |
131 | 0 | symbol.name(), |
132 | 0 | )) |
133 | 2 | }, |
134 | 2 | |v| Ok(Some(v)), |
135 | 2 | ) |
136 | 45 | }) |
137 | 45 | } |
138 | | |
139 | | /// Consumes the context and runs the given function with the context set |
140 | | /// as the active context. When the function exits, the context is restored |
141 | | /// to the previous global context. |
142 | | #[inline] |
143 | 4 | pub fn run<T>(self, span: Span, func: impl FnOnce() -> T) -> T { |
144 | 4 | Arc::new(self).wrap(span, func)() |
145 | 4 | } |
146 | | |
147 | | /// Wraps a function so when it is called the passed in context is set as |
148 | | /// the active context and when the function exits, the context is restored |
149 | | /// to the previous global context. |
150 | | #[inline] |
151 | 4 | fn wrap<T>(self: Arc<Self>, span: Span, func: impl FnOnce() -> T) -> impl FnOnce() -> T { |
152 | 4 | move || { |
153 | 4 | let enter = Self::enter(self); |
154 | 4 | let result = span.in_scope(func); |
155 | 4 | enter.release(); |
156 | 4 | result |
157 | 4 | } |
158 | 4 | } |
159 | | |
160 | | /// Wraps a future so when it is called the passed in context is set as |
161 | | /// the active context and when the future exits, the context is restored |
162 | | /// to the previous global context. |
163 | | #[inline] |
164 | 337 | pub fn wrap_async<T>( |
165 | 337 | self: Arc<Self>, |
166 | 337 | span: Span, |
167 | 337 | fut: impl Future<Output = T>, |
168 | 337 | ) -> impl Future<Output = T> { |
169 | 337 | ContextAwareFuture::new(Some(self), fut.instrument(span)) |
170 | 337 | } |
171 | | |
172 | | /// Spawns a future in the background with the given context. |
173 | 0 | pub fn background_spawn( |
174 | 0 | self: Arc<Self>, |
175 | 0 | span: Span, |
176 | 0 | fut: impl Future<Output = ()> + Send + 'static, |
177 | 0 | ) { |
178 | 0 | background_spawn!(span: span, ctx: Some(self), fut: fut); |
179 | 0 | } |
180 | | |
181 | | /// Enters the context, storing the previous context. If the returned |
182 | | /// `ContextDropGuard` is dropped, the previous context is restored. |
183 | | #[inline] |
184 | 16.8k | fn enter(self: Arc<Self>) -> ContextDropGuard { |
185 | 16.8k | ContextDropGuard::new(self) |
186 | 16.8k | } |
187 | | } |
188 | | |
189 | | /// Static struct to interact with the active global context. |
190 | | pub struct ActiveOriginContext; |
191 | | |
192 | | impl ActiveOriginContext { |
193 | | /// Sets the active context for the current thread. |
194 | 1.47k | pub fn get() -> Option<Arc<OriginContext>> { |
195 | 1.47k | GLOBAL_ORIGIN_CONTEXT.with_borrow(Clone::clone) |
196 | 1.47k | } |
197 | | |
198 | | /// Gets the value current set for a given symbol on the |
199 | | /// active context. |
200 | | #[inline] |
201 | 43 | pub fn get_value<T: Send + Sync + 'static>( |
202 | 43 | symbol: &'static impl Symbol<Type = T>, |
203 | 43 | ) -> Result<Option<Arc<T>>, Error> { |
204 | 43 | GLOBAL_ORIGIN_CONTEXT.with_borrow(|maybe_ctx| { |
205 | 43 | maybe_ctx.as_ref().map_or_else( |
206 | 43 | || { |
207 | 0 | Err(make_err!( |
208 | 0 | Code::Internal, |
209 | 0 | "Expected active context to be set" |
210 | 0 | )) |
211 | 43 | }, |
212 | 43 | |ctx| ctx.get_value(symbol), |
213 | 43 | ) |
214 | 43 | }) |
215 | 43 | } |
216 | | |
217 | | /// Forks the active context, returning a new context with the same data. |
218 | | #[inline] |
219 | 40 | pub fn fork() -> Result<OriginContext, Error> { |
220 | 40 | GLOBAL_ORIGIN_CONTEXT.with_borrow(|maybe_ctx| { |
221 | 40 | maybe_ctx.as_ref().map_or_else( |
222 | 40 | || { |
223 | 0 | Err(make_err!( |
224 | 0 | Code::Internal, |
225 | 0 | "Expected active context to be set" |
226 | 0 | )) |
227 | 40 | }, |
228 | 40 | |ctx| Ok(ctx.as_ref().clone()), |
229 | 40 | ) |
230 | 40 | }) |
231 | 40 | } |
232 | | } |
233 | | |
234 | | thread_local! { |
235 | | /// Global context that is used to store the active context. |
236 | | static GLOBAL_ORIGIN_CONTEXT: RefCell<Option<Arc<OriginContext>>> = const { RefCell::new(None) }; |
237 | | } |
238 | | |
239 | | /// Special guard struct that is used to hold the previous context and restore it |
240 | | /// when the guard is released or dropped. |
241 | | struct ContextDropGuard { |
242 | | prev_ctx: Option<Arc<OriginContext>>, |
243 | | new_ctx_ptr: *const OriginContext, |
244 | | } |
245 | | |
246 | | impl ContextDropGuard { |
247 | | /// Places the new context in the global context, stores the previous context |
248 | | /// and returns a new `ContextDropGuard`, so if it is ever released or dropped |
249 | | /// the previous context will be restored. |
250 | | #[inline] |
251 | 16.8k | fn new(new_ctx: Arc<OriginContext>) -> Self { |
252 | 16.8k | let new_ctx_ptr = Arc::as_ptr(&new_ctx); |
253 | 16.8k | let prev_ctx = GLOBAL_ORIGIN_CONTEXT.replace(Some(new_ctx)); |
254 | 16.8k | Self { |
255 | 16.8k | prev_ctx, |
256 | 16.8k | new_ctx_ptr, |
257 | 16.8k | } |
258 | 16.8k | } |
259 | | |
260 | | /// Swap the global context with the previous context. |
261 | | /// Returns the context used in the `.enter()`/`.new()` call. |
262 | 16.8k | fn restore_global_context(&mut self) -> Arc<OriginContext> { |
263 | 16.8k | let new_ctx = GLOBAL_ORIGIN_CONTEXT |
264 | 16.8k | .replace(self.prev_ctx.take()) |
265 | 16.8k | .expect("Expected global context to be set"); |
266 | 16.8k | assert_eq!(self.new_ctx_ptr, Arc::as_ptr(&new_ctx)); |
267 | 16.8k | new_ctx |
268 | 16.8k | } |
269 | | |
270 | | /// Release the context guard and restore the previous context. |
271 | | /// This is an optimization, so we don't need to clone our Arc |
272 | | /// if the caller can use the context after releasing it. |
273 | | #[inline] |
274 | 15.0k | fn release(mut self) -> Arc<OriginContext> { |
275 | 15.0k | let new_ctx = self.restore_global_context(); |
276 | 15.0k | std::mem::forget(self); // Prevent the destructor from being called. |
277 | 15.0k | new_ctx |
278 | 15.0k | } |
279 | | } |
280 | | |
281 | | impl Drop for ContextDropGuard { |
282 | | #[inline] |
283 | 1.80k | fn drop(&mut self) { |
284 | 1.80k | // If the user calls `.release()` the drop() will not be triggered. |
285 | 1.80k | // If a panic happens, the drop() will be called and the `.release()` |
286 | 1.80k | // will not be called, this is our safety net. |
287 | 1.80k | self.restore_global_context(); |
288 | 1.80k | } |
289 | | } |
290 | | |
291 | | pin_project! { |
292 | | #[must_use = "futures do nothing unless you `.await` or poll them"] |
293 | | pub struct ContextAwareFuture<F> { |
294 | | // `ManuallyDrop` is used so we can call `self.span.enter()` in the `drop()` |
295 | | // of our inner future, then drop the span. |
296 | | #[pin] |
297 | | inner: ManuallyDrop<Instrumented<F>>, |
298 | | context: Option<Arc<OriginContext>>, |
299 | | } |
300 | | |
301 | | impl <F> PinnedDrop for ContextAwareFuture<F> { |
302 | | #[inline] |
303 | | fn drop(this: Pin<&mut Self>) { |
304 | | let this = this.project(); |
305 | | // Note: If the future panics, the context will not be restored, so |
306 | | // this is a best effort to provide access to our global context |
307 | | // in the destructors the event of a panic. |
308 | | let _enter = this.context.take().map(OriginContext::enter); |
309 | | // SAFETY: 1. `Pin::get_unchecked_mut()` is safe, because this isn't |
310 | | // different from wrapping `T` in `Option` and calling |
311 | | // `Pin::set(&mut this.inner, None)`, except avoiding |
312 | | // additional memory overhead. |
313 | | // 2. `ManuallyDrop::drop()` is safe, because |
314 | | // `PinnedDrop::drop()` is guaranteed to be called only |
315 | | // once. |
316 | | unsafe { |
317 | | ManuallyDrop::drop(this.inner.get_unchecked_mut()); |
318 | | } |
319 | | } |
320 | | } |
321 | | } |
322 | | |
323 | | impl<T> ContextAwareFuture<T> { |
324 | | /// Utility function to create a new `ContextAwareFuture` from the |
325 | | /// active context. |
326 | | #[must_use = "futures do nothing unless you `.await` or poll them"] |
327 | | #[inline] |
328 | 9 | pub fn new_from_active(inner: Instrumented<T>) -> ContextAwareFuture<T> { |
329 | 9 | if let Some(ctx) = ActiveOriginContext::get() { Branch (329:16): [Folded - Ignored]
Branch (329:16): [True: 2, False: 0]
Branch (329:16): [True: 2, False: 0]
Branch (329:16): [True: 2, False: 0]
Branch (329:16): [True: 2, False: 0]
Branch (329:16): [True: 1, False: 0]
Branch (329:16): [Folded - Ignored]
|
330 | 9 | ContextAwareFuture::new(Some(ctx), inner) |
331 | | } else { |
332 | | // Useful to get tracing stack trace. |
333 | 0 | tracing::error!("OriginContext must be set"); |
334 | 0 | panic!("OriginContext must be set"); |
335 | | } |
336 | 9 | } |
337 | | |
338 | | #[must_use = "futures do nothing unless you `.await` or poll them"] |
339 | | #[inline] |
340 | 1.80k | pub(crate) fn new(context: Option<Arc<OriginContext>>, inner: Instrumented<T>) -> Self { |
341 | 1.80k | Self { |
342 | 1.80k | inner: ManuallyDrop::new(inner), |
343 | 1.80k | context, |
344 | 1.80k | } |
345 | 1.80k | } |
346 | | } |
347 | | |
348 | | impl<T: Future> Future for ContextAwareFuture<T> { |
349 | | type Output = T::Output; |
350 | | |
351 | | #[inline] |
352 | 15.0k | fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> { |
353 | 15.0k | let this = self.project(); |
354 | 15.0k | let Some(ctx) = this.context.take() else { Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 22, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 115, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 6, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 32, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 4, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 54, False: 0]
Branch (354:13): [True: 1.22k, False: 0]
Branch (354:13): [True: 5, False: 0]
Branch (354:13): [True: 27, False: 0]
Branch (354:13): [Folded - Ignored]
Branch (354:13): [True: 74, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 706, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 290, False: 0]
Branch (354:13): [True: 67, False: 0]
Branch (354:13): [True: 270, False: 0]
Branch (354:13): [True: 13, False: 0]
Branch (354:13): [True: 4, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 4, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 5, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 4, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 15, False: 0]
Branch (354:13): [True: 13, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 7, False: 0]
Branch (354:13): [True: 13, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 7, False: 0]
Branch (354:13): [True: 4, False: 0]
Branch (354:13): [True: 5, False: 0]
Branch (354:13): [True: 2.63k, False: 0]
Branch (354:13): [True: 83, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 5, False: 0]
Branch (354:13): [True: 6, False: 0]
Branch (354:13): [True: 32, False: 0]
Branch (354:13): [True: 4, False: 0]
Branch (354:13): [True: 31, False: 0]
Branch (354:13): [True: 3.26k, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 6, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 6, False: 0]
Branch (354:13): [True: 4, False: 0]
Branch (354:13): [True: 5, False: 0]
Branch (354:13): [True: 5, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1.03k, False: 0]
Branch (354:13): [True: 13, False: 0]
Branch (354:13): [True: 26, False: 0]
Branch (354:13): [True: 22, False: 0]
Branch (354:13): [True: 26, False: 0]
Branch (354:13): [True: 24, False: 0]
Branch (354:13): [True: 17, False: 0]
Branch (354:13): [True: 37, False: 0]
Branch (354:13): [True: 22, False: 0]
Branch (354:13): [True: 20, False: 0]
Branch (354:13): [True: 36, False: 0]
Branch (354:13): [True: 28, False: 0]
Branch (354:13): [True: 26, False: 0]
Branch (354:13): [True: 17, False: 0]
Branch (354:13): [True: 36, False: 0]
Branch (354:13): [True: 30, False: 0]
Branch (354:13): [True: 47, False: 0]
Branch (354:13): [True: 29, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 26, False: 0]
Branch (354:13): [True: 1.02k, False: 0]
Branch (354:13): [True: 4, False: 0]
Branch (354:13): [True: 16, False: 0]
Branch (354:13): [True: 16, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 35, False: 0]
Branch (354:13): [True: 37, False: 0]
Branch (354:13): [True: 43, False: 0]
Branch (354:13): [True: 38, False: 0]
Branch (354:13): [True: 38, False: 0]
Branch (354:13): [True: 16, False: 0]
Branch (354:13): [True: 13, False: 0]
Branch (354:13): [True: 16, False: 0]
Branch (354:13): [True: 16, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 9, False: 0]
Branch (354:13): [True: 4, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 5, False: 0]
Branch (354:13): [True: 9, False: 0]
Branch (354:13): [True: 14, False: 0]
Branch (354:13): [True: 19, False: 0]
Branch (354:13): [True: 18, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 23, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 10, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [Folded - Ignored]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 4, False: 0]
Branch (354:13): [True: 12, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 8, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 4, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 8, False: 0]
Branch (354:13): [True: 9, False: 0]
Branch (354:13): [True: 11, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 10, False: 0]
Branch (354:13): [True: 10, False: 0]
Branch (354:13): [True: 57, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 49, False: 0]
Branch (354:13): [True: 77, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 51, False: 0]
Branch (354:13): [True: 83, False: 0]
Branch (354:13): [True: 13, False: 0]
Branch (354:13): [True: 14, False: 0]
Branch (354:13): [True: 53, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 37, False: 0]
Branch (354:13): [True: 23, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 53, False: 0]
Branch (354:13): [True: 33, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 16, False: 0]
Branch (354:13): [True: 12, False: 0]
Branch (354:13): [True: 24, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 57, False: 0]
Branch (354:13): [True: 55, False: 0]
Branch (354:13): [True: 51, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 26, False: 0]
Branch (354:13): [True: 7, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 5, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 4, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 24, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 24, False: 0]
Branch (354:13): [True: 24, False: 0]
Branch (354:13): [True: 24, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 24, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 0, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 4, False: 0]
Branch (354:13): [True: 5, False: 0]
Branch (354:13): [True: 4, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 6, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 4, False: 0]
Branch (354:13): [True: 5, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 4, False: 0]
Branch (354:13): [True: 4, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 4, False: 0]
Branch (354:13): [True: 73, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 502, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 993, False: 0]
Branch (354:13): [True: 4, False: 0]
Branch (354:13): [True: 5, False: 0]
Branch (354:13): [True: 5, False: 0]
Branch (354:13): [True: 4, False: 0]
Branch (354:13): [True: 6, False: 0]
Branch (354:13): [True: 4, False: 0]
Branch (354:13): [True: 4, False: 0]
Branch (354:13): [True: 6, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 4, False: 0]
Branch (354:13): [True: 5, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 5, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 3, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 4, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 2, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
Branch (354:13): [True: 1, False: 0]
|
355 | | // Useful to get tracing stack trace. |
356 | 0 | tracing::error!("Expected context to be set"); |
357 | 0 | panic!("Expected context to be set"); |
358 | | }; |
359 | 15.0k | let enter = ctx.enter(); |
360 | 15.0k | // Since `inner` is only moved when the future itself is dropped, `inner` will |
361 | 15.0k | // never move, so this should be safe. |
362 | 15.0k | // see: https://docs.rs/tracing/0.1.40/src/tracing/instrument.rs.html#297 |
363 | 15.0k | let inner = unsafe { this.inner.map_unchecked_mut(|v| &mut **v) }; |
364 | 15.0k | let result = inner.poll(cx); |
365 | 15.0k | assert!( |
366 | 15.0k | this.context.replace(enter.release()).is_none(), |
367 | 0 | "Expected context to be unset" |
368 | | ); |
369 | 15.0k | result |
370 | 15.0k | } |
371 | | } |