musli/context/
capture.rs
1use core::cell::UnsafeCell;
2use core::error::Error;
3use core::fmt;
4use core::marker::PhantomData;
5
6use crate::alloc::{self, Allocator, String};
7#[cfg(feature = "alloc")]
8use crate::alloc::{System, SYSTEM};
9use crate::Context;
10
11use super::{ContextError, ErrorMarker};
12
13pub struct Capture<M, E, A>
15where
16 E: ContextError,
17{
18 alloc: A,
19 error: UnsafeCell<Option<E>>,
20 _marker: PhantomData<M>,
21}
22
23#[cfg(feature = "alloc")]
24#[cfg_attr(doc_cfg, doc(cfg(feature = "alloc")))]
25impl<M, E> Capture<M, E, &'static System>
26where
27 E: ContextError,
28{
29 pub fn new() -> Self {
31 Self::with_alloc(&SYSTEM)
32 }
33}
34
35impl<M, E, A> Capture<M, E, A>
36where
37 E: ContextError,
38{
39 pub fn with_alloc(alloc: A) -> Self {
41 Self {
42 alloc,
43 error: UnsafeCell::new(None),
44 _marker: PhantomData,
45 }
46 }
47
48 pub fn unwrap(self) -> E {
50 let Some(error) = self.error.into_inner() else {
51 return E::message("no error captured");
52 };
53
54 error
55 }
56}
57
58impl<M, E, A> Context for Capture<M, E, A>
59where
60 M: 'static,
61 E: ContextError,
62 A: Allocator,
63{
64 type Mode = M;
65 type Error = ErrorMarker;
66 type Mark = ();
67 type Allocator = A;
68 type String<'this> = String<'this, A> where Self: 'this;
69
70 #[inline]
71 fn clear(&self) {
72 unsafe {
75 (*self.error.get()) = None;
76 }
77 }
78
79 #[inline]
80 fn alloc(&self) -> &Self::Allocator {
81 &self.alloc
82 }
83
84 #[inline]
85 fn collect_string<T>(&self, value: &T) -> Result<Self::String<'_>, Self::Error>
86 where
87 T: ?Sized + fmt::Display,
88 {
89 alloc::collect_string(self, value)
90 }
91
92 #[inline]
93 fn custom<T>(&self, error: T) -> ErrorMarker
94 where
95 T: 'static + Send + Sync + Error,
96 {
97 unsafe {
100 self.error.get().replace(Some(E::custom(error)));
101 }
102
103 ErrorMarker
104 }
105
106 #[inline]
107 fn message<T>(&self, message: T) -> ErrorMarker
108 where
109 T: fmt::Display,
110 {
111 unsafe {
114 self.error.get().replace(Some(E::message(message)));
115 }
116
117 ErrorMarker
118 }
119}
120
121#[cfg(feature = "alloc")]
122#[cfg_attr(doc_cfg, doc(cfg(feature = "alloc")))]
123impl<M, E> Default for Capture<M, E, &'static System>
124where
125 E: ContextError,
126{
127 #[inline]
128 fn default() -> Self {
129 Self::new()
130 }
131}