reddb_server/storage/engine/
freelist.rs1use super::page::{Page, PageType, HEADER_SIZE, PAGE_SIZE};
30
31pub const FREE_IDS_PER_TRUNK: usize = (PAGE_SIZE - HEADER_SIZE - 8) / 4;
33
34const NEXT_TRUNK_OFFSET: usize = HEADER_SIZE;
36
37const COUNT_OFFSET: usize = HEADER_SIZE + 4;
39
40const PAGE_IDS_OFFSET: usize = HEADER_SIZE + 8;
42
43#[derive(Debug, Clone)]
45pub enum FreeListError {
46 Empty,
48 Corrupted(String),
50 InvalidTrunk(u32),
52}
53
54impl std::fmt::Display for FreeListError {
55 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
56 match self {
57 Self::Empty => write!(f, "Freelist is empty"),
58 Self::Corrupted(msg) => write!(f, "Freelist corrupted: {}", msg),
59 Self::InvalidTrunk(id) => write!(f, "Invalid trunk page: {}", id),
60 }
61 }
62}
63
64impl std::error::Error for FreeListError {}
65
66#[derive(Debug)]
71pub struct FreeList {
72 trunk_head: u32,
74 free_pages: Vec<u32>,
76 total_free: u32,
78 dirty: bool,
80}
81
82impl FreeList {
83 pub fn new() -> Self {
85 Self {
86 trunk_head: 0,
87 free_pages: Vec::new(),
88 total_free: 0,
89 dirty: false,
90 }
91 }
92
93 pub fn from_header(trunk_head: u32, total_free: u32) -> Self {
95 Self {
96 trunk_head,
97 free_pages: Vec::new(),
98 total_free,
99 dirty: false,
100 }
101 }
102
103 pub fn trunk_head(&self) -> u32 {
105 self.trunk_head
106 }
107
108 pub fn total_free(&self) -> u32 {
110 self.total_free
111 }
112
113 pub fn is_empty(&self) -> bool {
115 self.free_pages.is_empty() && self.trunk_head == 0
116 }
117
118 pub fn is_dirty(&self) -> bool {
120 self.dirty
121 }
122
123 pub fn mark_clean(&mut self) {
125 self.dirty = false;
126 }
127
128 pub fn allocate(&mut self) -> Option<u32> {
132 if let Some(page_id) = self.free_pages.pop() {
134 self.total_free = self.total_free.saturating_sub(1);
135 self.dirty = true;
136 return Some(page_id);
137 }
138
139 None
142 }
143
144 pub fn free(&mut self, page_id: u32) {
146 self.free_pages.push(page_id);
147 self.total_free += 1;
148 self.dirty = true;
149 }
150
151 pub fn free_batch(&mut self, page_ids: &[u32]) {
153 self.free_pages.extend_from_slice(page_ids);
154 self.total_free += page_ids.len() as u32;
155 self.dirty = true;
156 }
157
158 pub fn in_memory_count(&self) -> usize {
160 self.free_pages.len()
161 }
162
163 pub fn load_from_trunk(&mut self, trunk: &Page) -> Result<Option<u32>, FreeListError> {
165 if trunk
167 .page_type()
168 .map_err(|_| FreeListError::InvalidTrunk(trunk.page_id()))?
169 != PageType::FreelistTrunk
170 {
171 return Err(FreeListError::InvalidTrunk(trunk.page_id()));
172 }
173
174 let data = trunk.as_bytes();
175
176 let next_trunk = u32::from_le_bytes([
178 data[NEXT_TRUNK_OFFSET],
179 data[NEXT_TRUNK_OFFSET + 1],
180 data[NEXT_TRUNK_OFFSET + 2],
181 data[NEXT_TRUNK_OFFSET + 3],
182 ]);
183
184 let count = u32::from_le_bytes([
186 data[COUNT_OFFSET],
187 data[COUNT_OFFSET + 1],
188 data[COUNT_OFFSET + 2],
189 data[COUNT_OFFSET + 3],
190 ]) as usize;
191
192 if count > FREE_IDS_PER_TRUNK {
193 return Err(FreeListError::Corrupted(format!(
194 "Trunk has {} entries, max is {}",
195 count, FREE_IDS_PER_TRUNK
196 )));
197 }
198
199 self.free_pages.push(trunk.page_id());
200
201 for i in 0..count {
203 let offset = PAGE_IDS_OFFSET + i * 4;
204 let page_id = u32::from_le_bytes([
205 data[offset],
206 data[offset + 1],
207 data[offset + 2],
208 data[offset + 3],
209 ]);
210 self.free_pages.push(page_id);
211 }
212 self.total_free = self.total_free.saturating_add(count as u32 + 1);
213 self.dirty = true;
214
215 self.trunk_head = next_trunk;
217
218 Ok(if next_trunk != 0 {
219 Some(next_trunk)
220 } else {
221 None
222 })
223 }
224
225 pub fn create_trunk(&mut self, trunk_page_id: u32, next_trunk: u32) -> Page {
230 let mut trunk = Page::new(PageType::FreelistTrunk, trunk_page_id);
231 let data = trunk.as_bytes_mut();
232
233 data[NEXT_TRUNK_OFFSET..NEXT_TRUNK_OFFSET + 4].copy_from_slice(&next_trunk.to_le_bytes());
235
236 let count = self.free_pages.len().min(FREE_IDS_PER_TRUNK);
238
239 data[COUNT_OFFSET..COUNT_OFFSET + 4].copy_from_slice(&(count as u32).to_le_bytes());
241
242 for i in 0..count {
244 let page_id = self.free_pages.pop().unwrap();
245 let offset = PAGE_IDS_OFFSET + i * 4;
246 data[offset..offset + 4].copy_from_slice(&page_id.to_le_bytes());
247 }
248
249 trunk.update_checksum();
250 self.dirty = true;
251
252 trunk
253 }
254
255 pub fn flush_to_trunks(
260 &mut self,
261 threshold: usize,
262 mut allocate_page: impl FnMut() -> u32,
263 ) -> Vec<Page> {
264 let mut trunks = Vec::new();
265
266 while self.free_pages.len() > threshold {
267 let trunk_page_id = allocate_page();
269
270 let trunk = self.create_trunk(trunk_page_id, self.trunk_head);
272 self.trunk_head = trunk_page_id;
273
274 trunks.push(trunk);
275 }
276
277 trunks
278 }
279
280 pub fn merge_all_trunks(
284 &mut self,
285 load_page: impl Fn(u32) -> Option<Page>,
286 ) -> Result<Vec<u32>, FreeListError> {
287 let mut reclaimed_trunks = Vec::new();
288
289 while self.trunk_head != 0 {
290 let trunk_id = self.trunk_head;
291 let trunk = load_page(trunk_id).ok_or(FreeListError::InvalidTrunk(trunk_id))?;
292
293 self.load_from_trunk(&trunk)?;
294 reclaimed_trunks.push(trunk_id);
295 }
296
297 Ok(reclaimed_trunks)
298 }
299}
300
301impl Default for FreeList {
302 fn default() -> Self {
303 Self::new()
304 }
305}
306
307#[cfg(test)]
308mod tests {
309 use super::*;
310
311 #[test]
312 fn test_freelist_basic() {
313 let mut fl = FreeList::new();
314
315 assert!(fl.is_empty());
316 assert_eq!(fl.total_free(), 0);
317
318 fl.free(10);
320 fl.free(20);
321 fl.free(30);
322
323 assert!(!fl.is_empty());
324 assert_eq!(fl.total_free(), 3);
325
326 assert_eq!(fl.allocate(), Some(30));
328 assert_eq!(fl.allocate(), Some(20));
329 assert_eq!(fl.allocate(), Some(10));
330 assert_eq!(fl.allocate(), None);
331
332 assert!(fl.is_empty());
333 }
334
335 #[test]
336 fn test_freelist_batch() {
337 let mut fl = FreeList::new();
338
339 fl.free_batch(&[1, 2, 3, 4, 5]);
340 assert_eq!(fl.total_free(), 5);
341 assert_eq!(fl.in_memory_count(), 5);
342 }
343
344 #[test]
345 fn test_freelist_dirty() {
346 let mut fl = FreeList::new();
347
348 assert!(!fl.is_dirty());
349
350 fl.free(1);
351 assert!(fl.is_dirty());
352
353 fl.mark_clean();
354 assert!(!fl.is_dirty());
355
356 fl.allocate();
357 assert!(fl.is_dirty());
358 }
359
360 #[test]
361 fn test_trunk_page_creation() {
362 let mut fl = FreeList::new();
363
364 for i in 0..100 {
366 fl.free(i);
367 }
368
369 let trunk = fl.create_trunk(999, 0);
371
372 assert_eq!(trunk.page_type().unwrap(), PageType::FreelistTrunk);
373 assert_eq!(trunk.page_id(), 999);
374
375 assert!(fl.in_memory_count() < 100);
377 }
378
379 #[test]
380 fn test_trunk_page_load() {
381 let mut fl = FreeList::new();
382
383 for i in 0..50 {
385 fl.free(i);
386 }
387
388 let trunk = fl.create_trunk(999, 0);
389 let pages_in_trunk = 50 - fl.in_memory_count();
390
391 fl.free_pages.clear();
393
394 fl.load_from_trunk(&trunk).unwrap();
396
397 assert_eq!(fl.in_memory_count(), pages_in_trunk + 1);
398 }
399
400 #[test]
401 fn test_trunk_page_reuse() {
402 let mut original = FreeList::new();
403
404 for i in 0..8 {
405 original.free(i);
406 }
407
408 let trunk = original.create_trunk(999, 0);
409
410 let mut fl = FreeList::new();
411 fl.load_from_trunk(&trunk).unwrap();
412
413 let mut ids = Vec::new();
414 while let Some(id) = fl.allocate() {
415 ids.push(id);
416 }
417
418 assert!(ids.contains(&999));
419 }
420
421 #[test]
422 fn test_trunk_chain() {
423 let mut fl = FreeList::new();
424
425 for i in 0..2000 {
427 fl.free(i);
428 }
429
430 let mut next_trunk_id = 10000u32;
432 let trunks = fl.flush_to_trunks(100, || {
433 let id = next_trunk_id;
434 next_trunk_id += 1;
435 id
436 });
437
438 assert!(!trunks.is_empty());
439 assert!(fl.in_memory_count() <= 100);
440 }
441
442 #[test]
443 fn test_free_ids_per_trunk() {
444 assert_eq!(FREE_IDS_PER_TRUNK, 1014);
447 assert_eq!(FREE_IDS_PER_TRUNK, 1014);
448 }
449}