honeycomb_core/cmap/dim3/orbits.rs
1//! Orbit implementation
2//!
3//! This module contains all code used to model orbits, a notion defined
4//! along the structure of combinatorial maps.
5
6use std::collections::{HashSet, VecDeque};
7
8use crate::cmap::{CMap3, DartIdType, NULL_DART_ID, OrbitPolicy, try_from_fn};
9use crate::geometry::CoordsFloat;
10use crate::stm::{StmClosureResult, Transaction};
11
12impl<T: CoordsFloat> CMap3<T> {
13    /// Generic orbit implementation.
14    ///
15    /// # Arguments
16    /// - `opolicy: OrbitPolicy` -- Policy used by the orbit for the BFS.
17    /// - `dart_id: DartIdentifier` -- Dart of which the structure will compute the orbit.
18    ///
19    /// # The search algorithm
20    ///
21    /// The search algorithm used to establish the list of dart included in the orbit is a
22    /// [Breadth-First Search algorithm][WIKIBFS]. This means that:
23    ///
24    /// - we look at the images of the current dart through all beta functions,
25    ///   adding those to a queue, before moving on to the next dart.
26    /// - we apply the beta functions in their specified order; This guarantees a consistent and
27    ///   predictable result.
28    ///
29    /// # Performance
30    ///
31    /// Currently, orbits use two dynamically allocated structures for computation: a `VecDeque`,
32    /// and a `HashSet`. There is a possibility to use static thread-local instances to avoid
33    /// ephemeral allocations, but [it would require a guard mechanism][PR].
34    ///
35    /// [WIKIBFS]: https://en.wikipedia.org/wiki/Breadth-first_search
36    /// [PR]: https://github.com/LIHPC-Computational-Geometry/honeycomb/pull/293
37    #[allow(clippy::needless_for_each, clippy::too_many_lines)]
38    #[rustfmt::skip]
39    pub fn orbit(
40        &self,
41        opolicy: OrbitPolicy,
42        dart_id: DartIdType,
43    ) -> impl Iterator<Item = DartIdType> {
44        let mut pending = VecDeque::new();
45        let mut marked: HashSet<DartIdType> = HashSet::new();
46        pending.push_back(dart_id);
47        marked.insert(NULL_DART_ID);
48        marked.insert(dart_id); // we're starting here, so we mark it beforehand
49
50        // FIXME: move the match block out of the iterator
51        std::iter::from_fn(move || {
52            if let Some(d) = pending.pop_front() {
53                let mut check = |d: DartIdType| {
54                    if marked.insert(d) {
55                        pending.push_back(d);
56                    }
57                };
58                // compute the next images
59                match opolicy {
60                    // B3oB2, B1oB3, B1oB2, B3oB0, B2oB0
61                    OrbitPolicy::Vertex => {
62                        [
63                            self.beta::<3>(self.beta::<2>(d)), // b3(b2(d))
64                            self.beta::<1>(self.beta::<3>(d)), // b1(b3(d))
65                            self.beta::<1>(self.beta::<2>(d)), // b1(b2(d))
66                            self.beta::<3>(self.beta::<0>(d)), // b3(b0(d))
67                            self.beta::<2>(self.beta::<0>(d)), // b2(b0(d))
68                        ]
69                        .into_iter()
70                        .for_each(check);
71                    }
72                    // B3oB2, B1oB3, B1oB2
73                    OrbitPolicy::VertexLinear => {
74                        [
75                            self.beta::<3>(self.beta::<2>(d)), // b3(b2(d))
76                            self.beta::<1>(self.beta::<3>(d)), // b1(b3(d))
77                            self.beta::<1>(self.beta::<2>(d)), // b1(b2(d))
78                        ]
79                        .into_iter()
80                        .for_each(check);
81                    }
82                    // B2, B3
83                    OrbitPolicy::Edge => {
84                        [
85                            self.beta::<2>(d),
86                            self.beta::<3>(d),
87                        ]
88                        .into_iter()
89                        .for_each(check);
90                    }
91                    // B1, B0, B3
92                    OrbitPolicy::Face => {
93                        [
94                            self.beta::<1>(d),
95                            self.beta::<0>(d),
96                            self.beta::<3>(d),
97                        ]
98                        .into_iter()
99                        .for_each(check);
100                    }
101                    // B1, B3
102                    OrbitPolicy::FaceLinear => {
103                        [
104                            self.beta::<1>(d),
105                            self.beta::<3>(d),
106                        ]
107                        .into_iter()
108                        .for_each(check);
109                    }
110                    // B1, B0, B2
111                    OrbitPolicy::Volume => {
112                        [
113                            self.beta::<1>(d),
114                            self.beta::<0>(d),
115                            self.beta::<2>(d),
116                        ]
117                        .into_iter()
118                        .for_each(check);
119                    }
120                    // B1, B2
121                    OrbitPolicy::VolumeLinear => {
122                        [
123                            self.beta::<1>(d),
124                            self.beta::<2>(d),
125                        ]
126                        .into_iter()
127                        .for_each(check);
128                    }
129                    OrbitPolicy::Custom(beta_slice) => {
130                        for beta_id in beta_slice {
131                            let im = self.beta_rt(*beta_id, d);
132                            check(im);
133                        }
134                    }
135                }
136
137                return Some(d);
138            }
139            None // queue is empty, we're done
140        })
141    }
142
143    /// Generic orbit transactional implementation.
144    #[allow(clippy::needless_for_each, clippy::too_many_lines)]
145    #[rustfmt::skip]
146    pub fn orbit_tx(
147        &self,
148        t: &mut Transaction,
149        opolicy: OrbitPolicy,
150        dart_id: DartIdType,
151    ) -> impl Iterator<Item = StmClosureResult<DartIdType>> {
152        let mut pending = VecDeque::new();
153        let mut marked: HashSet<DartIdType> = HashSet::new();
154        pending.push_back(dart_id);
155        marked.insert(NULL_DART_ID);
156        marked.insert(dart_id); // we're starting here, so we mark it beforehand
157
158        // FIXME: move the match block out of the iterator
159        try_from_fn(move || {
160            if let Some(d) = pending.pop_front() {
161                let mut check = |d: DartIdType| {
162                    if marked.insert(d) {
163                        pending.push_back(d);
164                    }
165                };
166                // compute the next images
167                match opolicy {
168                    // B3oB2, B1oB3, B1oB2, B3oB0, B2oB0
169                    OrbitPolicy::Vertex => {
170                        let (b0, b2, b3) = (
171                            self.beta_tx::<0>(t, d)?,
172                            self.beta_tx::<2>(t, d)?,
173                            self.beta_tx::<3>(t, d)?,
174                        );
175                        [
176                            self.beta_tx::<3>(t, b2)?, // b3(b2(d))
177                            self.beta_tx::<1>(t, b3)?, // b1(b3(d))
178                            self.beta_tx::<1>(t, b2)?, // b1(b2(d))
179                            self.beta_tx::<3>(t, b0)?, // b3(b0(d))
180                            self.beta_tx::<2>(t, b0)?, // b2(b0(d))
181                        ]
182                        .into_iter()
183                        .for_each(check);
184                    }
185                    // B3oB2, B1oB3, B1oB2
186                    OrbitPolicy::VertexLinear => {
187                        let (b2, b3) =
188                            (self.beta_tx::<2>(t, d)?, self.beta_tx::<3>(t, d)?);
189                        [
190                            self.beta_tx::<3>(t, b2)?, // b3(b2(d))
191                            self.beta_tx::<1>(t, b3)?, // b1(b3(d))
192                            self.beta_tx::<1>(t, b2)?, // b1(b2(d))
193                        ]
194                        .into_iter()
195                        .for_each(check);
196                    }
197                    // B2, B3
198                    OrbitPolicy::Edge => {
199                        [self.beta_tx::<2>(t, d)?, self.beta_tx::<3>(t, d)?]
200                            .into_iter()
201                            .for_each(check);
202                    }
203                    // B1, B0, B3
204                    OrbitPolicy::Face => {
205                        [
206                            self.beta_tx::<1>(t, d)?,
207                            self.beta_tx::<0>(t, d)?,
208                            self.beta_tx::<3>(t, d)?,
209                        ]
210                        .into_iter()
211                        .for_each(check);
212                    }
213                    // B1, B3
214                    OrbitPolicy::FaceLinear => {
215                        [
216                            self.beta_tx::<1>(t, d)?,
217                            self.beta_tx::<3>(t, d)?,
218                        ]
219                        .into_iter()
220                        .for_each(check);
221                    }
222                    // B1, B0, B2
223                    OrbitPolicy::Volume => {
224                        [
225                            self.beta_tx::<1>(t, d)?,
226                            self.beta_tx::<0>(t, d)?,
227                            self.beta_tx::<2>(t, d)?,
228                        ]
229                        .into_iter()
230                        .for_each(check);
231                    }
232                    // B1, B2
233                    OrbitPolicy::VolumeLinear => {
234                        [self.beta_tx::<1>(t, d)?, self.beta_tx::<2>(t, d)?]
235                            .into_iter()
236                            .for_each(check);
237                    }
238                    OrbitPolicy::Custom(beta_slice) => {
239                        for beta_id in beta_slice {
240                            let im = self.beta_rt_tx(t, *beta_id, d)?;
241                            check(im);
242                        }
243                    }
244                }
245
246                return Ok(Some(d));
247            }
248            Ok(None) // queue is empty, we're done
249        })
250    }
251
252    /// Return the orbit defined by a dart and its `I`-cell.
253    ///
254    /// # Usage
255    ///
256    /// The returned item can be iterated upon to retrieve all dart members of the cell. Note that
257    /// **the dart passed as an argument is included as the first element of the returned orbit**.
258    ///
259    /// # Panics
260    ///
261    /// The method will panic if *I* is not 0, 1, 2, or 3.
262    #[must_use = "unused return value"]
263    pub fn i_cell<const I: u8>(&self, dart_id: DartIdType) -> impl Iterator<Item = DartIdType> {
264        assert!(I < 4);
265        match I {
266            0 => self.orbit(OrbitPolicy::Vertex, dart_id),
267            1 => self.orbit(OrbitPolicy::Edge, dart_id),
268            2 => self.orbit(OrbitPolicy::Face, dart_id),
269            3 => self.orbit(OrbitPolicy::Volume, dart_id),
270            _ => unreachable!(),
271        }
272    }
273}