@@ -78,15 +78,12 @@ impl Block {
78
78
pub const LINES : usize = Self :: BYTES / Line :: BYTES ;
79
79
#[ allow( unused) ]
80
80
pub const DATA_LINES : usize = Self :: DATA_BYTES / Line :: BYTES ;
81
+ pub const REUSABLE_THRESHOLD : usize = Self :: LINES / 2 ;
81
82
82
83
pub fn init ( & mut self , owner : usize ) {
83
- // debug_assert_eq!(Self::META_BYTES, Address::BYTES * 8);
84
84
self . owner = owner;
85
85
self . live_lines = 0 ;
86
- for i in 0 ..LINES_IN_BLOCK {
87
- self . line_liveness [ i] = 0 ;
88
- }
89
- self . foreign_free . store ( Address :: ZERO , Ordering :: SeqCst ) ;
86
+ self . foreign_free . store ( Address :: ZERO , Ordering :: Relaxed ) ;
90
87
self . state = BlockState :: Allocating ;
91
88
self . next = None ;
92
89
self . prev = None ;
@@ -95,7 +92,7 @@ impl Block {
95
92
pub fn deinit ( & mut self ) {
96
93
self . state = BlockState :: Free ;
97
94
self . live_lines = 0 ;
98
- self . foreign_free . store ( Address :: ZERO , Ordering :: SeqCst ) ;
95
+ self . foreign_free . store ( Address :: ZERO , Ordering :: Relaxed ) ;
99
96
self . next = None ;
100
97
self . prev = None ;
101
98
}
@@ -132,7 +129,8 @@ impl Block {
132
129
}
133
130
cursor += 1 ;
134
131
}
135
- if cursor != start_cursor {
132
+ let first_cursor = self . lines ( ) . start . get_index_within_block ( ) ;
133
+ if cursor != start_cursor && cursor != first_cursor {
136
134
cursor += 1 ;
137
135
}
138
136
if cursor >= self . line_liveness . len ( ) {
@@ -191,6 +189,7 @@ impl Block {
191
189
self . line_liveness [ i] += 1 ;
192
190
}
193
191
self . live_lines += lines;
192
+ // self.drain_foreign_free();
194
193
// println!(" - BA {:x?} live-lines {}", self, self.live_lines);
195
194
}
196
195
@@ -237,14 +236,42 @@ impl Block {
237
236
// println!(" - S-{:?} dead-{}", self.state, dead_lines);
238
237
if dead_lines >= 1 && self . state == BlockState :: Full {
239
238
let live_lines = self . live_lines ;
240
- if live_lines < Block :: DATA_LINES / 2 {
239
+ if live_lines <= Block :: REUSABLE_THRESHOLD {
241
240
let owner = & mut crate :: ImmixMutator :: current ( ) . ix ;
242
241
owner. add_reusable_block ( * self ) ;
243
242
}
244
243
}
245
244
}
246
245
}
247
246
247
+ pub fn drain_foreign_free ( & mut self ) {
248
+ let mut ptr;
249
+ loop {
250
+ ptr = self . foreign_free . load ( Ordering :: Relaxed ) ;
251
+ if ptr. is_zero ( ) {
252
+ break ;
253
+ }
254
+ if self
255
+ . foreign_free
256
+ . compare_exchange ( ptr, Address :: ZERO , Ordering :: SeqCst , Ordering :: SeqCst )
257
+ . is_ok ( )
258
+ {
259
+ break ;
260
+ }
261
+ }
262
+ // let mut count = 0;
263
+ while !ptr. is_zero ( ) {
264
+ let next = unsafe { ptr. load ( ) } ;
265
+ self . dealloc_impl ( ptr, self . get_layout ( ptr) ) ;
266
+ ptr = next;
267
+ // count += 1;
268
+ }
269
+ // if count > 0 {
270
+ // //
271
+ // println!(" - DF {:x?} freed {}", self, count);
272
+ // }
273
+ }
274
+
248
275
#[ inline]
249
276
pub fn on_dealloc ( & mut self , ptr : Address , layout : Layout ) {
250
277
// println!("FLocal {:?}", ptr..(ptr + layout.size()));
0 commit comments