facet_core/impls/core/
tuple.rs

1use core::{cmp::Ordering, fmt, mem};
2
3use crate::{
4    Def, Facet, FieldBuilder, HashProxy, OxPtrConst, OxPtrMut, OxRef, PtrConst, PtrMut, Repr,
5    Shape, ShapeBuilder, StructKind, StructType, Type, TypeNameOpts, TypeOpsIndirect, UserType,
6    VTableIndirect,
7};
8
9/// Debug for tuples - formats as tuple literal
10unsafe fn tuple_debug(
11    ox: OxPtrConst,
12    f: &mut core::fmt::Formatter<'_>,
13) -> Option<core::fmt::Result> {
14    let shape = ox.shape();
15    let ty = match shape.ty {
16        Type::User(UserType::Struct(ref st)) => st,
17        _ => return None,
18    };
19
20    let ptr = ox.ptr();
21    let mut tuple = f.debug_tuple("");
22
23    for field in ty.fields {
24        let field_ptr = unsafe { PtrConst::new(ptr.as_byte_ptr().add(field.offset)) };
25        let field_ox = OxRef::new(field_ptr, field.shape.get());
26        tuple.field(&field_ox);
27    }
28
29    Some(tuple.finish())
30}
31
32/// Hash for tuples - hashes each element
33unsafe fn tuple_hash(ox: OxPtrConst, hasher: &mut HashProxy<'_>) -> Option<()> {
34    let shape = ox.shape();
35    let ty = match shape.ty {
36        Type::User(UserType::Struct(ref st)) => st,
37        _ => return None,
38    };
39
40    let ptr = ox.ptr();
41
42    for field in ty.fields {
43        let field_ptr = unsafe { PtrConst::new(ptr.as_byte_ptr().add(field.offset)) };
44        let field_shape = field.shape.get();
45        unsafe { field_shape.call_hash(field_ptr, hasher)? };
46    }
47
48    Some(())
49}
50
51/// PartialEq for tuples
52unsafe fn tuple_partial_eq(a: OxPtrConst, b: OxPtrConst) -> Option<bool> {
53    let shape = a.shape();
54    let ty = match shape.ty {
55        Type::User(UserType::Struct(ref st)) => st,
56        _ => return None,
57    };
58
59    let a_ptr = a.ptr();
60    let b_ptr = b.ptr();
61
62    for field in ty.fields {
63        let a_field = unsafe { PtrConst::new(a_ptr.as_byte_ptr().add(field.offset)) };
64        let b_field = unsafe { PtrConst::new(b_ptr.as_byte_ptr().add(field.offset)) };
65        let field_shape = field.shape.get();
66        if !unsafe { field_shape.call_partial_eq(a_field, b_field)? } {
67            return Some(false);
68        }
69    }
70
71    Some(true)
72}
73
74/// PartialOrd for tuples
75unsafe fn tuple_partial_cmp(a: OxPtrConst, b: OxPtrConst) -> Option<Option<Ordering>> {
76    let shape = a.shape();
77    let ty = match shape.ty {
78        Type::User(UserType::Struct(ref st)) => st,
79        _ => return None,
80    };
81
82    let a_ptr = a.ptr();
83    let b_ptr = b.ptr();
84
85    for field in ty.fields {
86        let a_field = unsafe { PtrConst::new(a_ptr.as_byte_ptr().add(field.offset)) };
87        let b_field = unsafe { PtrConst::new(b_ptr.as_byte_ptr().add(field.offset)) };
88        let field_shape = field.shape.get();
89        match unsafe { field_shape.call_partial_cmp(a_field, b_field)? } {
90            Some(Ordering::Equal) => continue,
91            other => return Some(other),
92        }
93    }
94
95    Some(Some(Ordering::Equal))
96}
97
98/// Ord for tuples
99unsafe fn tuple_cmp(a: OxPtrConst, b: OxPtrConst) -> Option<Ordering> {
100    let shape = a.shape();
101    let ty = match shape.ty {
102        Type::User(UserType::Struct(ref st)) => st,
103        _ => return None,
104    };
105
106    let a_ptr = a.ptr();
107    let b_ptr = b.ptr();
108
109    for field in ty.fields {
110        let a_field = unsafe { PtrConst::new(a_ptr.as_byte_ptr().add(field.offset)) };
111        let b_field = unsafe { PtrConst::new(b_ptr.as_byte_ptr().add(field.offset)) };
112        let field_shape = field.shape.get();
113        match unsafe { field_shape.call_cmp(a_field, b_field)? } {
114            Ordering::Equal => continue,
115            other => return Some(other),
116        }
117    }
118
119    Some(Ordering::Equal)
120}
121
122/// Drop for tuples
123unsafe fn tuple_drop(ox: OxPtrMut) {
124    let shape = ox.shape();
125    let ty = match shape.ty {
126        Type::User(UserType::Struct(ref st)) => st,
127        _ => return,
128    };
129
130    let ptr = ox.ptr();
131
132    for field in ty.fields {
133        let field_ptr = unsafe { PtrMut::new((ptr.as_byte_ptr() as *mut u8).add(field.offset)) };
134        let field_shape = field.shape.get();
135        unsafe { field_shape.call_drop_in_place(field_ptr) };
136    }
137}
138
139// Shared vtable for all tuples
140const TUPLE_VTABLE: VTableIndirect = VTableIndirect {
141    display: None,
142    debug: Some(tuple_debug),
143    hash: Some(tuple_hash),
144    invariants: None,
145    parse: None,
146    try_from: None,
147    try_into_inner: None,
148    try_borrow_inner: None,
149    partial_eq: Some(tuple_partial_eq),
150    partial_cmp: Some(tuple_partial_cmp),
151    cmp: Some(tuple_cmp),
152};
153
154// Type operations for all tuples
155static TUPLE_TYPE_OPS: TypeOpsIndirect = TypeOpsIndirect {
156    drop_in_place: tuple_drop,
157    default_in_place: None,
158    clone_into: None,
159    is_truthy: None,
160};
161
162/// Type-erased type_name for tuples - reads field types from the shape
163fn tuple_type_name(
164    shape: &'static Shape,
165    f: &mut fmt::Formatter<'_>,
166    opts: TypeNameOpts,
167) -> fmt::Result {
168    let st = match &shape.ty {
169        Type::User(UserType::Struct(st)) if st.kind == StructKind::Tuple => st,
170        _ => return write!(f, "(?)"),
171    };
172
173    write!(f, "(")?;
174    // Use for_children() for the element types since they're nested inside the tuple
175    let child_opts = opts.for_children();
176    for (i, field) in st.fields.iter().enumerate() {
177        if i > 0 {
178            write!(f, ", ")?;
179        }
180        if let Some(opts) = child_opts {
181            field.shape.get().write_type_name(f, opts)?;
182        } else {
183            write!(f, "…")?;
184        }
185    }
186    // Trailing comma for single-element tuples
187    if st.fields.len() == 1 {
188        write!(f, ",")?;
189    }
190    write!(f, ")")?;
191    Ok(())
192}
193
194macro_rules! impl_facet_for_tuple {
195    // Used to implement the next bigger tuple type, by taking the next typename & associated index
196    // out of `remaining`, if it exists.
197    {
198        continue from ($($elems:ident.$idx:tt,)+),
199        remaining ()
200    } => {};
201    {
202        continue from ($($elems:ident.$idx:tt,)+),
203        remaining ($next:ident.$nextidx:tt, $($remaining:ident.$remainingidx:tt,)*)
204    } => {
205        impl_facet_for_tuple! {
206            impl ($($elems.$idx,)+ $next.$nextidx,),
207            remaining ($($remaining.$remainingidx,)*)
208        }
209    };
210    // Actually generate the trait implementation, and keep the remaining possible elements around
211    {
212        impl ($($elems:ident.$idx:tt,)+),
213        remaining ($($remaining:ident.$remainingidx:tt,)*)
214    } => {
215        unsafe impl<'a $(, $elems)+> Facet<'a> for ($($elems,)+)
216        where
217            $($elems: Facet<'a>,)+
218        {
219            const SHAPE: &'static Shape = &const {
220                ShapeBuilder::for_sized::<Self>(
221                    if 1 == [$($elems::SHAPE),+].len() {
222                        "(_,)"
223                    } else {
224                        "(…)"
225                    }
226                )
227                .type_name(tuple_type_name)
228                .ty(Type::User(UserType::Struct(StructType {
229                    repr: Repr::default(),
230                    kind: StructKind::Tuple,
231                    fields: &const {[
232                        $(FieldBuilder::new(stringify!($idx), crate::shape_of::<$elems>, mem::offset_of!(Self, $idx)).build(),)+
233                    ]}
234                })))
235                .def(Def::Undefined)
236                .vtable_indirect(&TUPLE_VTABLE)
237                .type_ops_indirect(&TUPLE_TYPE_OPS)
238                .build()
239            };
240        }
241
242        impl_facet_for_tuple! {
243            continue from ($($elems.$idx,)+),
244            remaining ($($remaining.$remainingidx,)*)
245        }
246    };
247    // The entry point into this macro, all smaller tuple types get implemented as well.
248    { ($first:ident.$firstidx:tt $(, $remaining:ident.$remainingidx:tt)* $(,)?) } => {
249        impl_facet_for_tuple! {
250            impl ($first.$firstidx,),
251            remaining ($($remaining.$remainingidx,)*)
252        }
253    };
254}
255
256#[cfg(feature = "tuples-12")]
257impl_facet_for_tuple! {
258    (T0.0, T1.1, T2.2, T3.3, T4.4, T5.5, T6.6, T7.7, T8.8, T9.9, T10.10, T11.11)
259}
260
261#[cfg(not(feature = "tuples-12"))]
262impl_facet_for_tuple! {
263    (T0.0, T1.1, T2.2, T3.3)
264}