honeycomb_core/cmap/components/
betas.rs1use std::ops::{Index, IndexMut};
2
3use crate::cmap::{LinkError, NULL_DART_ID};
4use crate::stm::{TVar, Transaction, TransactionClosureResult, abort};
5
6use super::identifiers::DartIdType;
7
8pub struct BetaFunctions<const N: usize>(Vec<[TVar<DartIdType>; N]>);
13
14fn new_beta_entry<const N: usize>() -> [TVar<DartIdType>; N] {
16 (0..N)
17 .map(|_| TVar::new(NULL_DART_ID))
18 .collect::<Vec<_>>()
19 .try_into()
20 .unwrap()
21}
22
23#[allow(unused)]
24impl<const N: usize> BetaFunctions<N> {
25 pub fn new(n_darts: usize) -> Self {
27 Self((0..n_darts).map(|_| new_beta_entry()).collect())
28 }
29
30 pub fn extend(&mut self, len: usize) {
32 self.0.extend((0..len).map(|_| new_beta_entry()));
33 }
34
35 pub fn capacity(&self) -> usize {
37 self.0.capacity()
38 }
39}
40
41impl<const N: usize> Index<(u8, DartIdType)> for BetaFunctions<N> {
42 type Output = TVar<DartIdType>;
43
44 fn index(&self, (beta_id, dart_id): (u8, DartIdType)) -> &Self::Output {
45 &self.0[dart_id as usize][beta_id as usize]
46 }
47}
48
49impl<const N: usize> IndexMut<(u8, DartIdType)> for BetaFunctions<N> {
50 fn index_mut(&mut self, (beta_id, dart_id): (u8, DartIdType)) -> &mut Self::Output {
51 &mut self.0[dart_id as usize][beta_id as usize]
52 }
53}
54
55impl<const N: usize> BetaFunctions<N> {
60 pub fn one_link_core(
77 &self,
78 trans: &mut Transaction,
79 lhs_dart_id: DartIdType,
80 rhs_dart_id: DartIdType,
81 ) -> TransactionClosureResult<(), LinkError> {
82 if self[(1, lhs_dart_id)].read(trans)? != NULL_DART_ID {
83 return abort(LinkError::NonFreeBase(1, lhs_dart_id, rhs_dart_id));
84 }
85 if self[(0, rhs_dart_id)].read(trans)? != NULL_DART_ID {
86 return abort(LinkError::NonFreeImage(0, lhs_dart_id, rhs_dart_id));
87 }
88 self[(1, lhs_dart_id)].write(trans, rhs_dart_id)?;
90 self[(0, rhs_dart_id)].write(trans, lhs_dart_id)?;
92 Ok(())
93 }
94
95 pub fn two_link_core(
110 &self,
111 trans: &mut Transaction,
112 lhs_dart_id: DartIdType,
113 rhs_dart_id: DartIdType,
114 ) -> TransactionClosureResult<(), LinkError> {
115 if self[(2, lhs_dart_id)].read(trans)? != NULL_DART_ID {
116 return abort(LinkError::NonFreeBase(2, lhs_dart_id, rhs_dart_id));
117 }
118 if self[(2, rhs_dart_id)].read(trans)? != NULL_DART_ID {
119 return abort(LinkError::NonFreeImage(2, lhs_dart_id, rhs_dart_id));
120 }
121 self[(2, lhs_dart_id)].write(trans, rhs_dart_id)?;
123 self[(2, rhs_dart_id)].write(trans, lhs_dart_id)?;
125 Ok(())
126 }
127
128 pub fn three_link_core(
129 &self,
130 trans: &mut Transaction,
131 lhs_dart_id: DartIdType,
132 rhs_dart_id: DartIdType,
133 ) -> TransactionClosureResult<(), LinkError> {
134 if self[(3, lhs_dart_id)].read(trans)? != NULL_DART_ID {
135 return abort(LinkError::NonFreeBase(3, lhs_dart_id, rhs_dart_id));
136 }
137 if self[(3, rhs_dart_id)].read(trans)? != NULL_DART_ID {
138 return abort(LinkError::NonFreeImage(3, lhs_dart_id, rhs_dart_id));
139 }
140 self[(3, lhs_dart_id)].write(trans, rhs_dart_id)?;
141 self[(3, rhs_dart_id)].write(trans, lhs_dart_id)?;
142 Ok(())
143 }
144
145 pub fn one_unlink_core(
160 &self,
161 trans: &mut Transaction,
162 lhs_dart_id: DartIdType,
163 ) -> TransactionClosureResult<(), LinkError> {
164 let rhs_dart_id = self[(1, lhs_dart_id)].replace(trans, NULL_DART_ID)?;
166 if rhs_dart_id == NULL_DART_ID {
167 return abort(LinkError::AlreadyFree(1, lhs_dart_id));
168 }
169 self[(0, rhs_dart_id)].write(trans, NULL_DART_ID)?;
171 Ok(())
172 }
173
174 pub fn two_unlink_core(
188 &self,
189 trans: &mut Transaction,
190 lhs_dart_id: DartIdType,
191 ) -> TransactionClosureResult<(), LinkError> {
192 let rhs_dart_id = self[(2, lhs_dart_id)].replace(trans, NULL_DART_ID)?;
194 if rhs_dart_id == NULL_DART_ID {
195 return abort(LinkError::AlreadyFree(2, lhs_dart_id));
196 }
197 self[(2, rhs_dart_id)].write(trans, NULL_DART_ID)?;
199 Ok(())
200 }
201
202 pub fn three_unlink_core(
203 &self,
204 trans: &mut Transaction,
205 lhs_dart_id: DartIdType,
206 ) -> TransactionClosureResult<(), LinkError> {
207 let rhs_dart_id = self[(3, lhs_dart_id)].replace(trans, NULL_DART_ID)?;
209 if rhs_dart_id == NULL_DART_ID {
210 return abort(LinkError::AlreadyFree(3, lhs_dart_id));
211 }
212 self[(3, rhs_dart_id)].write(trans, NULL_DART_ID)?;
214 Ok(())
215 }
216}