-
Notifications
You must be signed in to change notification settings - Fork 215
/
Copy pathframe_allocator.rs
127 lines (118 loc) · 5.1 KB
/
frame_allocator.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
use super::{frame_range, phys_frame_range};
use bootloader::bootinfo::{MemoryMap, MemoryRegion, MemoryRegionType};
use x86_64::structures::paging::{frame::PhysFrameRange, PhysFrame};
pub(crate) struct FrameAllocator<'a> {
pub memory_map: &'a mut MemoryMap,
}
impl<'a> FrameAllocator<'a> {
pub(crate) fn allocate_frame(&mut self, region_type: MemoryRegionType) -> Option<PhysFrame> {
// try to find an existing region of same type that can be enlarged
let mut iter = self.memory_map.iter_mut().peekable();
while let Some(region) = iter.next() {
if region.region_type == region_type {
if let Some(next) = iter.peek() {
if next.range.start_frame_number == region.range.end_frame_number
&& next.region_type == MemoryRegionType::Usable
&& !next.range.is_empty()
{
let frame = phys_frame_range(region.range).end;
region.range.end_frame_number += 1;
iter.next().unwrap().range.start_frame_number += 1;
return Some(frame);
}
}
}
}
fn split_usable_region<'a, I>(iter: &mut I) -> Option<(PhysFrame, PhysFrameRange)>
where
I: Iterator<Item = &'a mut MemoryRegion>,
{
for region in iter {
if region.region_type != MemoryRegionType::Usable {
continue;
}
if region.range.is_empty() {
continue;
}
let frame = phys_frame_range(region.range).start;
region.range.start_frame_number += 1;
return Some((frame, PhysFrame::range(frame, frame + 1)));
}
None
}
let result = if region_type == MemoryRegionType::PageTable {
// prevent fragmentation when page tables are allocated in between
split_usable_region(&mut self.memory_map.iter_mut().rev())
} else {
split_usable_region(&mut self.memory_map.iter_mut())
};
if let Some((frame, range)) = result {
self.memory_map.add_region(MemoryRegion {
range: frame_range(range),
region_type,
});
Some(frame)
} else {
None
}
}
/// Marks the passed region in the memory map.
///
/// Panics if a non-usable region (e.g. a reserved region) overlaps with the passed region.
pub(crate) fn mark_allocated_region(&mut self, region: MemoryRegion) {
for r in self.memory_map.iter_mut() {
if region.range.start_frame_number >= r.range.end_frame_number {
continue;
}
if region.range.end_frame_number <= r.range.start_frame_number {
continue;
}
if r.region_type != MemoryRegionType::Usable {
panic!(
"region {:x?} overlaps with non-usable region {:x?}",
region, r
);
}
if region.range.start_frame_number == r.range.start_frame_number {
if region.range.end_frame_number < r.range.end_frame_number {
// Case: (r = `r`, R = `region`)
// ----rrrrrrrrrrr----
// ----RRRR-----------
r.range.start_frame_number = region.range.end_frame_number;
self.memory_map.add_region(region);
} else {
// Case: (r = `r`, R = `region`)
// ----rrrrrrrrrrr----
// ----RRRRRRRRRRRRRR-
*r = region;
}
} else if region.range.start_frame_number > r.range.start_frame_number {
if region.range.end_frame_number < r.range.end_frame_number {
// Case: (r = `r`, R = `region`)
// ----rrrrrrrrrrr----
// ------RRRR---------
let mut behind_r = r.clone();
behind_r.range.start_frame_number = region.range.end_frame_number;
r.range.end_frame_number = region.range.start_frame_number;
self.memory_map.add_region(behind_r);
self.memory_map.add_region(region);
} else {
// Case: (r = `r`, R = `region`)
// ----rrrrrrrrrrr----
// -----------RRRR---- or
// -------------RRRR--
r.range.end_frame_number = region.range.start_frame_number;
self.memory_map.add_region(region);
}
} else {
// Case: (r = `r`, R = `region`)
// ----rrrrrrrrrrr----
// --RRRR-------------
r.range.start_frame_number = region.range.end_frame_number;
self.memory_map.add_region(region);
}
return;
}
panic!("region {:x?} is not a usable memory region", region);
}
}