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_transac(
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_transac::<0>(t, d)?,
172 self.beta_transac::<2>(t, d)?,
173 self.beta_transac::<3>(t, d)?,
174 );
175 [
176 self.beta_transac::<3>(t, b2)?, // b3(b2(d))
177 self.beta_transac::<1>(t, b3)?, // b1(b3(d))
178 self.beta_transac::<1>(t, b2)?, // b1(b2(d))
179 self.beta_transac::<3>(t, b0)?, // b3(b0(d))
180 self.beta_transac::<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_transac::<2>(t, d)?, self.beta_transac::<3>(t, d)?);
189 [
190 self.beta_transac::<3>(t, b2)?, // b3(b2(d))
191 self.beta_transac::<1>(t, b3)?, // b1(b3(d))
192 self.beta_transac::<1>(t, b2)?, // b1(b2(d))
193 ]
194 .into_iter()
195 .for_each(check);
196 }
197 // B2, B3
198 OrbitPolicy::Edge => {
199 [self.beta_transac::<2>(t, d)?, self.beta_transac::<3>(t, d)?]
200 .into_iter()
201 .for_each(check);
202 }
203 // B1, B0, B3
204 OrbitPolicy::Face => {
205 [
206 self.beta_transac::<1>(t, d)?,
207 self.beta_transac::<0>(t, d)?,
208 self.beta_transac::<3>(t, d)?,
209 ]
210 .into_iter()
211 .for_each(check);
212 }
213 // B1, B3
214 OrbitPolicy::FaceLinear => {
215 [
216 self.beta_transac::<1>(t, d)?,
217 self.beta_transac::<3>(t, d)?,
218 ]
219 .into_iter()
220 .for_each(check);
221 }
222 // B1, B0, B2
223 OrbitPolicy::Volume => {
224 [
225 self.beta_transac::<1>(t, d)?,
226 self.beta_transac::<0>(t, d)?,
227 self.beta_transac::<2>(t, d)?,
228 ]
229 .into_iter()
230 .for_each(check);
231 }
232 // B1, B2
233 OrbitPolicy::VolumeLinear => {
234 [self.beta_transac::<1>(t, d)?, self.beta_transac::<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_transac(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}