1
1
extern crate core;
2
- use allocated_memory;
3
- use allocated_memory :: AllocatedStackMemory ;
2
+ use super :: allocated_memory;
3
+ use super :: allocated_stack_memory :: AllocatedStackMemory ;
4
4
5
5
pub trait Allocator < T > {
6
- type AllocatedMemory : allocated_memory:: traits :: AllocatedSlice < T > ;
6
+ type AllocatedMemory : allocated_memory:: AllocatedSlice < T > ;
7
7
fn alloc_cell ( & mut self , len : usize ) -> Self :: AllocatedMemory ;
8
8
fn free_cell ( & mut self , data : Self :: AllocatedMemory ) ;
9
9
}
10
10
11
11
12
- pub trait SystemMemoryPool < ' a , T > {
13
- fn add_slices ( & mut self , input : & mut Option < & ' a mut AllocatedStackMemory < ' a , T > > ) ;
14
- fn add_memory ( & mut self , input : & mut Option < & ' a mut AllocatedStackMemory < ' a , T > > , min_length : usize ) ;
15
- }
16
-
17
-
18
12
19
-
20
- pub struct StackAllocator < ' a , T : ' a , U : SystemMemoryPool < ' a , T > > {
13
+ pub struct StackAllocator < ' a ,
14
+ T : ' a ,
15
+ U : allocated_memory:: AllocatedSlice < & ' a mut [ T ] > > {
16
+ pub nop : & ' a mut [ T ] ,
21
17
pub system_resources : U ,
22
- pub glob : Option < & ' a mut AllocatedStackMemory < ' a , T > > ,
18
+ pub free_list_start : usize ,
19
+ pub free_list_overflow_count : usize ,
23
20
}
24
21
25
22
26
- fn remove_cur < ' a , T > ( mut iter : & mut Option < & mut AllocatedStackMemory < ' a , T > > ) -> & ' a mut AllocatedStackMemory < ' a , T > {
27
- match * iter {
28
- Some ( ref mut glob_next ) => {
29
- let rest : Option < & ' a mut AllocatedStackMemory < ' a , T > > ;
30
- match glob_next . next {
31
- Some ( ref mut root_cell ) => rest = core :: mem :: replace ( & mut root_cell . next , None ) ,
32
- None => rest = None ,
33
- }
34
- match core :: mem :: replace ( & mut glob_next . next , rest ) {
35
- Some ( mut root_cell ) => return root_cell ,
36
- None => panic ! ( "Empty list" ) ,
23
+ impl < ' a , T , U : allocated_memory :: AllocatedSlice < & ' a mut [ T ] > >
24
+ Allocator < T > for StackAllocator < ' a , T , U > {
25
+ type AllocatedMemory = AllocatedStackMemory < ' a , T > ;
26
+ fn alloc_cell ( self : & mut StackAllocator < ' a , T , U > ,
27
+ len : usize ) -> AllocatedStackMemory < ' a , T > {
28
+ let mut index : usize = self . free_list_start ;
29
+ let mut found : bool = false ;
30
+ for free_resource in self . system_resources . slice ( ) [ self . free_list_start .. ] . iter ( ) {
31
+ if free_resource . len ( ) >= len {
32
+ found = true ;
33
+ break ;
37
34
}
38
- } ,
39
- None => panic ! ( "List not initialized" ) ,
40
- }
41
- }
42
-
43
-
44
- fn remove_search_recursive < ' a , T : ' a > ( mut node : & mut Option < & mut AllocatedStackMemory < ' a , T > > ,
45
- searchmin : usize ,
46
- searchmax : usize ) -> & ' a mut AllocatedStackMemory < ' a , T > {
47
- let mut found : bool = false ;
48
- {
49
- match * node {
50
- Some ( ref mut cur_cell) => {
51
- match cur_cell. next {
52
- Some ( ref item) => {
53
- let len = item. mem . len ( ) ;
54
- if ( len >= searchmin && ( searchmax ==0 || len < searchmax) )
55
- || len == 0 || !item. next . is_some ( ) {
56
- found = true ;
57
- }
58
- } ,
59
- None => panic ! ( "Too many outstanding allocated items" ) ,
60
- }
61
- if !found {
62
- return remove_search_recursive ( & mut cur_cell. next , searchmin, searchmax) ;
63
- }
64
- } ,
65
- None => panic ! ( "Too many outstanding allocated items" ) ,
35
+ index += 1 ;
66
36
}
67
- }
68
- assert ! ( found) ;
69
- return remove_cur ( node) ;
70
- }
71
-
72
- fn return_slice_to < ' a , T > ( mut node : & mut Option < & mut AllocatedStackMemory < ' a , T > > ,
73
- mut val : & ' a mut AllocatedStackMemory < ' a , T > ) {
74
- match * node {
75
- Some ( ref mut glob_next) => {
76
- match core:: mem:: replace ( & mut glob_next. next , None ) {
77
- Some ( mut x) => {
78
- let _discard = core:: mem:: replace ( & mut val. next , Some ( x) ) ;
79
- } ,
80
- None => { } ,
81
- }
82
- glob_next. next = Some ( val) ;
83
- } ,
84
- None => panic ! ( "Allocator not initialized" ) ,
85
- }
86
- }
87
-
88
- impl < ' a , T , U : SystemMemoryPool < ' a , T > > StackAllocator < ' a , T , U > {
89
- fn alloc_cell_directive ( self : & mut StackAllocator < ' a , T , U > , len : usize ,
90
- best_match : bool ) -> & ' a mut AllocatedStackMemory < ' a , T > {
91
- let lower_bound : usize ;
92
- let upper_bound: usize ;
93
- if best_match {
94
- lower_bound = len;
95
- upper_bound = len * 3 / 2 ;
96
- } else {
97
- lower_bound = 0 ;
98
- upper_bound = 0 ;
37
+ if !found {
38
+ panic ! ( "OOM" ) ;
99
39
}
100
-
101
- let mut retval = remove_search_recursive ( & mut self . glob , lower_bound, upper_bound) ;
102
- if retval. mem . len ( ) < len {
103
- let oom : bool ;
104
- match self . glob {
105
- Some ( ref mut sentinel) => {
106
- oom = sentinel. mem . len ( ) < len;
107
- } ,
108
- None => panic ! ( "Uninitialized allocator" ) ,
109
- }
110
- if oom {
111
- self . system_resources . add_memory ( & mut self . glob , len) ;
112
- }
113
- match self . glob {
114
- Some ( ref mut sentinel) => {
115
- let mut current_mem = core:: mem:: replace ( & mut sentinel. mem , & mut [ ] ) ;
116
- if current_mem. len ( ) < len {
117
- panic ! ( "OOM" ) ;
118
- }
119
- let ( mut new_chunk, mut remaining_mem) = current_mem. split_at_mut ( len) ;
120
- sentinel. mem = remaining_mem;
121
- retval. mem = new_chunk;
122
- } ,
123
- None => panic ! ( "Uninitalized allocator" ) ,
40
+ let mut available_slice = core:: mem:: replace ( & mut self . system_resources . slice_mut ( ) [ index] ,
41
+ & mut [ ] ) ;
42
+ if available_slice. len ( ) < len + 32 { // we don't want really small wasted slices
43
+ // we must assign free_list_start
44
+ if index != self . free_list_start {
45
+ assert ! ( index > self . free_list_start) ;
46
+ let mut farthest_free_list = core:: mem:: replace (
47
+ & mut self . system_resources . slice_mut ( ) [ self . free_list_start ] ,
48
+ & mut [ ] ) ;
49
+ core:: mem:: replace ( & mut self . system_resources . slice_mut ( ) [ index] ,
50
+ farthest_free_list) ;
124
51
}
125
- }
126
- return retval;
127
- }
128
- fn no_slices_left ( self : & StackAllocator < ' a , T , U > ) -> bool {
129
- match self . glob {
130
- Some ( ref glob_next) => return !glob_next. next . is_some ( ) ,
131
- None => return true ,
52
+ self . free_list_start += 1 ;
53
+ return AllocatedStackMemory :: < ' a , T > { mem : available_slice} ;
54
+ } else { // the memory allocated was not the entire range of items. Split and move on
55
+ let ( mut retval, return_to_sender) = available_slice. split_at_mut ( len) ;
56
+ core:: mem:: replace ( & mut self . system_resources . slice_mut ( ) [ index] , return_to_sender) ;
57
+ return AllocatedStackMemory :: < ' a , T > { mem : retval} ;
132
58
}
133
59
}
134
- }
135
- impl < ' a , T , U : SystemMemoryPool < ' a , T > >
136
- Allocator < T > for StackAllocator < ' a , T , U > {
137
- type AllocatedMemory = & ' a mut AllocatedStackMemory < ' a , T > ;
138
- fn alloc_cell ( self : & mut StackAllocator < ' a , T , U > ,
139
- len : usize ) -> & ' a mut AllocatedStackMemory < ' a , T > {
140
- if self . no_slices_left ( ) {
141
- self . system_resources . add_slices ( & mut self . glob ) ;
142
- }
143
- return self . alloc_cell_directive ( len, true ) ;
144
- }
145
60
fn free_cell ( self : & mut StackAllocator < ' a , T , U > ,
146
- mut val : & ' a mut AllocatedStackMemory < ' a , T > ) {
147
- return_slice_to ( & mut self . glob , val)
61
+ mut val : AllocatedStackMemory < ' a , T > ) {
62
+ if self . free_list_start > 0 {
63
+ self . free_list_start -=1 ;
64
+ core:: mem:: replace ( & mut self . system_resources . slice_mut ( ) [ self . free_list_start ] ,
65
+ val. mem ) ;
66
+ } else {
67
+ for _i in 0 ..3 {
68
+ self . free_list_overflow_count += 1 ;
69
+ self . free_list_overflow_count %= self . system_resources . slice ( ) . len ( ) ;
70
+ if self . system_resources . slice ( ) [ self . free_list_overflow_count ] . len ( ) < val. mem . len ( ) {
71
+ core:: mem:: replace ( & mut self . system_resources . slice_mut ( ) [ self . free_list_overflow_count ] ,
72
+ val. mem ) ;
73
+ return ;
74
+ }
75
+ }
76
+ }
148
77
}
149
78
}
150
79
151
- pub struct AllocatorStackState < ' a , T : ' a > {
152
- pub _cells : & ' a mut [ AllocatedStackMemory < ' a , T > ] ,
153
- }
154
-
155
- impl < ' a , T > SystemMemoryPool < ' a , T > for AllocatorStackState < ' a , T > {
156
- fn add_slices ( self : & mut AllocatorStackState < ' a , T > , _x : & mut Option < & ' a mut AllocatedStackMemory < ' a , T > > ) {
157
- panic ! ( "Out of Slices" ) ;
158
- }
159
- fn add_memory ( self : & mut AllocatorStackState < ' a , T > , _x : & mut Option < & ' a mut AllocatedStackMemory < ' a , T > > , _min_len : usize ) {
160
- panic ! ( "Out of Memory" ) ;
161
- }
162
- }
0 commit comments