15375
|
1 |
use super::common::GearId;
|
|
2 |
use std::{
|
|
3 |
any::TypeId,
|
|
4 |
mem::{size_of, MaybeUninit},
|
|
5 |
num::NonZeroU16,
|
15379
|
6 |
ptr::{copy_nonoverlapping, NonNull},
|
15375
|
7 |
slice,
|
|
8 |
};
|
15326
|
9 |
|
15378
|
10 |
pub unsafe trait TypeTuple: Sized {
|
15326
|
11 |
fn len() -> usize;
|
|
12 |
fn get_types(dest: &mut Vec<TypeId>);
|
15380
|
13 |
unsafe fn iter<F>(slices: &[NonNull<u8>], count: usize, mut f: F)
|
15375
|
14 |
where
|
15380
|
15 |
F: FnMut(Self);
|
15326
|
16 |
}
|
|
17 |
|
15378
|
18 |
unsafe impl<T: 'static> TypeTuple for (&T,) {
|
15326
|
19 |
fn len() -> usize {
|
|
20 |
1
|
|
21 |
}
|
|
22 |
|
|
23 |
fn get_types(dest: &mut Vec<TypeId>) {
|
|
24 |
dest.push(TypeId::of::<T>());
|
|
25 |
}
|
15375
|
26 |
|
15380
|
27 |
unsafe fn iter<F>(slices: &[NonNull<u8>], count: usize, mut f: F)
|
15375
|
28 |
where
|
15380
|
29 |
F: FnMut(Self),
|
15375
|
30 |
{
|
|
31 |
let slice1 = slice::from_raw_parts(slices[0].as_ptr() as *const T, count);
|
|
32 |
for i in 0..count {
|
|
33 |
f((slice1.get_unchecked(i),));
|
|
34 |
}
|
|
35 |
}
|
15326
|
36 |
}
|
|
37 |
|
15375
|
38 |
const BLOCK_SIZE: usize = 32768;
|
|
39 |
|
|
40 |
struct DataBlock {
|
|
41 |
max_elements: u16,
|
|
42 |
elements_count: u16,
|
|
43 |
data: Box<[u8; BLOCK_SIZE]>,
|
15380
|
44 |
component_blocks: [Option<NonNull<u8>>; 64],
|
15326
|
45 |
}
|
|
46 |
|
15375
|
47 |
impl Unpin for DataBlock {}
|
|
48 |
|
|
49 |
impl DataBlock {
|
|
50 |
fn new(mask: u64, element_sizes: &[u16; 64]) -> Self {
|
|
51 |
let total_size: u16 = element_sizes
|
|
52 |
.iter()
|
|
53 |
.enumerate()
|
15380
|
54 |
.filter(|(i, _)| mask & (1 << *i as u64) != 0)
|
15375
|
55 |
.map(|(_, size)| *size)
|
|
56 |
.sum();
|
|
57 |
let max_elements = (BLOCK_SIZE / total_size as usize) as u16;
|
|
58 |
|
|
59 |
let mut data: Box<[u8; BLOCK_SIZE]> =
|
15379
|
60 |
Box::new(unsafe { MaybeUninit::uninit().assume_init() });
|
15375
|
61 |
let mut blocks = [None; 64];
|
|
62 |
let mut offset = 0;
|
|
63 |
|
|
64 |
for i in 0..64 {
|
15380
|
65 |
if mask & (1 << i) != 0 {
|
15375
|
66 |
blocks[i] = Some(NonNull::new(data[offset..].as_mut_ptr()).unwrap());
|
|
67 |
offset += element_sizes[i] as usize * max_elements as usize;
|
|
68 |
}
|
|
69 |
}
|
15326
|
70 |
Self {
|
15375
|
71 |
elements_count: 0,
|
|
72 |
max_elements,
|
|
73 |
data,
|
15380
|
74 |
component_blocks: blocks,
|
15326
|
75 |
}
|
|
76 |
}
|
|
77 |
|
15375
|
78 |
fn is_full(&self) -> bool {
|
|
79 |
self.elements_count == self.max_elements
|
|
80 |
}
|
15326
|
81 |
}
|
|
82 |
|
15375
|
83 |
#[derive(Clone, Copy, Debug, Default)]
|
|
84 |
pub struct LookupEntry {
|
|
85 |
index: Option<NonZeroU16>,
|
|
86 |
block_index: u16,
|
15326
|
87 |
}
|
|
88 |
|
|
89 |
pub struct GearDataManager {
|
|
90 |
types: Vec<TypeId>,
|
15375
|
91 |
blocks: Vec<DataBlock>,
|
|
92 |
block_masks: Vec<u64>,
|
|
93 |
element_sizes: Box<[u16; 64]>,
|
|
94 |
lookup: Box<[LookupEntry]>,
|
15326
|
95 |
}
|
|
96 |
|
|
97 |
impl GearDataManager {
|
|
98 |
pub fn new() -> Self {
|
|
99 |
Self {
|
|
100 |
types: vec![],
|
15375
|
101 |
blocks: vec![],
|
|
102 |
block_masks: vec![],
|
|
103 |
element_sizes: Box::new([0; 64]),
|
|
104 |
lookup: vec![LookupEntry::default(); u16::max_value() as usize].into_boxed_slice(),
|
|
105 |
}
|
|
106 |
}
|
|
107 |
|
|
108 |
#[inline]
|
|
109 |
fn get_type_index<T: 'static>(&self) -> Option<usize> {
|
|
110 |
let type_id = TypeId::of::<T>();
|
|
111 |
self.types.iter().position(|id| *id == type_id)
|
|
112 |
}
|
|
113 |
|
|
114 |
fn move_between_blocks(&mut self, from_block_index: u16, from_index: u16, to_block_index: u16) {
|
15379
|
115 |
debug_assert!(from_block_index != to_block_index);
|
15375
|
116 |
let source_mask = self.block_masks[from_block_index as usize];
|
|
117 |
let destination_mask = self.block_masks[to_block_index as usize];
|
|
118 |
debug_assert!(source_mask & destination_mask == source_mask);
|
|
119 |
|
|
120 |
let source = &self.blocks[from_block_index as usize];
|
|
121 |
let destination = &self.blocks[to_block_index as usize];
|
15379
|
122 |
debug_assert!(from_index < source.elements_count);
|
|
123 |
debug_assert!(!destination.is_full());
|
|
124 |
|
15375
|
125 |
for i in 0..64 {
|
15380
|
126 |
if source_mask & 1 << i != 0 {
|
15379
|
127 |
unsafe {
|
|
128 |
copy_nonoverlapping(
|
15380
|
129 |
source.component_blocks[i].unwrap().as_ptr(),
|
|
130 |
destination.component_blocks[i].unwrap().as_ptr(),
|
15379
|
131 |
self.element_sizes[i] as usize,
|
|
132 |
);
|
|
133 |
}
|
|
134 |
}
|
15375
|
135 |
}
|
15379
|
136 |
self.blocks[from_block_index as usize].elements_count -= 1;
|
|
137 |
self.blocks[to_block_index as usize].elements_count += 1;
|
15375
|
138 |
}
|
|
139 |
|
15378
|
140 |
fn add_to_block<T: Clone>(&mut self, block_index: u16, value: &T) {
|
|
141 |
debug_assert!(self.block_masks[block_index as usize].count_ones() == 1);
|
|
142 |
|
|
143 |
let block = &mut self.blocks[block_index as usize];
|
|
144 |
debug_assert!(block.elements_count < block.max_elements);
|
|
145 |
|
|
146 |
unsafe {
|
|
147 |
let slice = slice::from_raw_parts_mut(
|
|
148 |
block.data.as_mut_ptr() as *mut T,
|
|
149 |
block.max_elements as usize,
|
|
150 |
);
|
|
151 |
*slice.get_unchecked_mut(block.elements_count as usize) = value.clone();
|
|
152 |
};
|
|
153 |
block.elements_count += 1;
|
15375
|
154 |
}
|
|
155 |
|
|
156 |
fn remove_from_block(&mut self, block_index: u16, index: u16) {
|
15378
|
157 |
let block = &mut self.blocks[block_index as usize];
|
|
158 |
debug_assert!(index < block.elements_count);
|
|
159 |
|
|
160 |
for (i, size) in self.element_sizes.iter().cloned().enumerate() {
|
|
161 |
if index < block.elements_count - 1 {
|
15380
|
162 |
if let Some(ptr) = block.component_blocks[i] {
|
15378
|
163 |
unsafe {
|
15379
|
164 |
copy_nonoverlapping(
|
15378
|
165 |
ptr.as_ptr()
|
|
166 |
.add((size * (block.elements_count - 1)) as usize),
|
|
167 |
ptr.as_ptr().add((size * index) as usize),
|
|
168 |
size as usize,
|
|
169 |
);
|
|
170 |
}
|
|
171 |
}
|
|
172 |
}
|
|
173 |
}
|
|
174 |
block.elements_count -= 1;
|
15375
|
175 |
}
|
|
176 |
|
|
177 |
#[inline]
|
15380
|
178 |
fn ensure_block(&mut self, mask: u64) -> u16 {
|
15375
|
179 |
if let Some(index) = self
|
|
180 |
.block_masks
|
|
181 |
.iter()
|
|
182 |
.enumerate()
|
|
183 |
.position(|(i, m)| *m == mask && !self.blocks[i].is_full())
|
|
184 |
{
|
|
185 |
index as u16
|
|
186 |
} else {
|
|
187 |
self.blocks.push(DataBlock::new(mask, &self.element_sizes));
|
15380
|
188 |
self.block_masks.push(mask);
|
15375
|
189 |
(self.blocks.len() - 1) as u16
|
|
190 |
}
|
|
191 |
}
|
|
192 |
|
|
193 |
pub fn add<T: Clone + 'static>(&mut self, gear_id: GearId, value: &T) {
|
|
194 |
if let Some(type_index) = self.get_type_index::<T>() {
|
15380
|
195 |
let type_bit = 1 << type_index as u64;
|
15375
|
196 |
let entry = self.lookup[gear_id.get() as usize - 1];
|
|
197 |
|
|
198 |
if let Some(index) = entry.index {
|
|
199 |
let mask = self.block_masks[entry.block_index as usize];
|
|
200 |
let new_mask = mask | type_bit;
|
|
201 |
|
|
202 |
if new_mask != mask {
|
15380
|
203 |
let dest_block_index = self.ensure_block(new_mask);
|
15375
|
204 |
self.move_between_blocks(entry.block_index, index.get() - 1, dest_block_index);
|
|
205 |
}
|
|
206 |
} else {
|
15380
|
207 |
let dest_block_index = self.ensure_block(type_bit);
|
15375
|
208 |
self.add_to_block(dest_block_index, value);
|
|
209 |
}
|
|
210 |
} else {
|
|
211 |
panic!("Unregistered type")
|
|
212 |
}
|
|
213 |
}
|
|
214 |
|
|
215 |
pub fn remove<T: 'static>(&mut self, gear_id: GearId) {
|
|
216 |
if let Some(type_index) = self.get_type_index::<T>() {
|
|
217 |
let entry = self.lookup[gear_id.get() as usize - 1];
|
|
218 |
if let Some(index) = entry.index {
|
15379
|
219 |
let destination_mask =
|
15380
|
220 |
self.block_masks[entry.block_index as usize] & !(1 << type_index as u64);
|
15379
|
221 |
|
|
222 |
if destination_mask == 0 {
|
|
223 |
self.remove_all(gear_id)
|
|
224 |
} else {
|
15380
|
225 |
let destination_block_index = self.ensure_block(destination_mask);
|
15379
|
226 |
self.move_between_blocks(
|
|
227 |
entry.block_index,
|
|
228 |
index.get() - 1,
|
|
229 |
destination_block_index,
|
|
230 |
);
|
|
231 |
}
|
15375
|
232 |
}
|
15379
|
233 |
} else {
|
|
234 |
panic!("Unregistered type")
|
|
235 |
}
|
|
236 |
}
|
|
237 |
|
|
238 |
pub fn remove_all(&mut self, gear_id: GearId) {
|
|
239 |
let entry = self.lookup[gear_id.get() as usize - 1];
|
|
240 |
if let Some(index) = entry.index {
|
|
241 |
self.remove_from_block(entry.block_index, index.get() - 1);
|
15326
|
242 |
}
|
|
243 |
}
|
|
244 |
|
|
245 |
pub fn register<T: 'static>(&mut self) {
|
15378
|
246 |
debug_assert!(!std::mem::needs_drop::<T>());
|
|
247 |
debug_assert!(self.types.len() <= 64);
|
|
248 |
debug_assert!(size_of::<T>() <= u16::max_value() as usize);
|
15375
|
249 |
|
15326
|
250 |
let id = TypeId::of::<T>();
|
|
251 |
if !self.types.contains(&id) {
|
15375
|
252 |
self.element_sizes[self.types.len()] = size_of::<T>() as u16;
|
15326
|
253 |
self.types.push(id);
|
|
254 |
}
|
|
255 |
}
|
|
256 |
|
|
257 |
fn create_selector(&self, types: &[TypeId]) -> u64 {
|
15380
|
258 |
let mut selector = 0;
|
15326
|
259 |
for (i, typ) in self.types.iter().enumerate() {
|
|
260 |
if types.contains(&typ) {
|
15380
|
261 |
selector |= 1 << (i as u64)
|
15326
|
262 |
}
|
|
263 |
}
|
|
264 |
selector
|
|
265 |
}
|
|
266 |
|
15380
|
267 |
pub fn iter<T: TypeTuple + 'static, F: FnMut(T)>(&self, mut f: F) {
|
15326
|
268 |
let mut types = vec![];
|
|
269 |
T::get_types(&mut types);
|
15378
|
270 |
debug_assert!(types.iter().all(|t| self.types.contains(t)));
|
15375
|
271 |
|
15378
|
272 |
let types_count = types.len();
|
15326
|
273 |
let selector = self.create_selector(&types);
|
15378
|
274 |
|
15380
|
275 |
let mut slices = Vec::with_capacity(64);
|
|
276 |
|
15375
|
277 |
for (block_index, mask) in self.block_masks.iter().enumerate() {
|
|
278 |
if mask & selector == selector {
|
15380
|
279 |
slices.clear();
|
15375
|
280 |
let block = &self.blocks[block_index];
|
15380
|
281 |
|
|
282 |
for (i, ptr_opt) in block.component_blocks.iter().cloned().enumerate() {
|
|
283 |
if let Some(ptr) = ptr_opt {
|
|
284 |
slices.push(ptr);
|
15378
|
285 |
}
|
15375
|
286 |
}
|
15380
|
287 |
|
|
288 |
unsafe {
|
|
289 |
T::iter(&slices[..], block.elements_count as usize, |x| f(x));
|
|
290 |
}
|
15326
|
291 |
}
|
|
292 |
}
|
|
293 |
}
|
|
294 |
}
|
15375
|
295 |
|
|
296 |
#[cfg(test)]
|
|
297 |
mod test {
|
15380
|
298 |
use super::{super::common::GearId, GearDataManager};
|
15375
|
299 |
|
15380
|
300 |
#[derive(Clone)]
|
15375
|
301 |
struct Datum {
|
|
302 |
value: u32,
|
|
303 |
}
|
|
304 |
|
|
305 |
#[test]
|
15380
|
306 |
fn single_component_iteration() {
|
|
307 |
assert!(std::mem::size_of::<Datum>() > 0);
|
|
308 |
|
15375
|
309 |
let mut manager = GearDataManager::new();
|
|
310 |
manager.register::<Datum>();
|
15380
|
311 |
for i in 1..=5 {
|
|
312 |
manager.add(GearId::new(i as u16).unwrap(), &Datum { value: i });
|
|
313 |
}
|
|
314 |
|
|
315 |
let mut sum = 0;
|
|
316 |
manager.iter(|(d,): (&Datum,)| sum += d.value);
|
|
317 |
|
|
318 |
assert_eq!(sum, 15);
|
15375
|
319 |
}
|
|
320 |
}
|