Skip to content

Commit 8d55d63

Browse files
committed
foreigh free
1 parent b7c762a commit 8d55d63

File tree

3 files changed

+46
-10
lines changed

3 files changed

+46
-10
lines changed

bench/Cargo.toml

+1-1
Original file line numberDiff line numberDiff line change
@@ -122,4 +122,4 @@ mi = { env = { MALLOC = "mi", IS_MALLOCKIT = "0" } }
122122
# je = { env = { MALLOC = "je", IS_MALLOCKIT = "0" } }
123123

124124
[package.metadata.harness.profiles.default.probes]
125-
harness-probe-perf = { events = "PERF_COUNT_SW_TASK_CLOCK,PERF_COUNT_HW_CPU_CYCLES,PERF_COUNT_HW_INSTRUCTIONS", inherit = true }
125+
harness-probe-perf = { events = "PERF_COUNT_SW_TASK_CLOCK,PERF_COUNT_HW_CPU_CYCLES,PERF_COUNT_HW_INSTRUCTIONS,PERF_COUNT_SW_PAGE_FAULTS,PERF_COUNT_SW_CONTEXT_SWITCHES", inherit = true }

ix/src/block.rs

+35-8
Original file line numberDiff line numberDiff line change
@@ -78,15 +78,12 @@ impl Block {
7878
pub const LINES: usize = Self::BYTES / Line::BYTES;
7979
#[allow(unused)]
8080
pub const DATA_LINES: usize = Self::DATA_BYTES / Line::BYTES;
81+
pub const REUSABLE_THRESHOLD: usize = Self::LINES / 2;
8182

8283
pub fn init(&mut self, owner: usize) {
83-
// debug_assert_eq!(Self::META_BYTES, Address::BYTES * 8);
8484
self.owner = owner;
8585
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);
9087
self.state = BlockState::Allocating;
9188
self.next = None;
9289
self.prev = None;
@@ -95,7 +92,7 @@ impl Block {
9592
pub fn deinit(&mut self) {
9693
self.state = BlockState::Free;
9794
self.live_lines = 0;
98-
self.foreign_free.store(Address::ZERO, Ordering::SeqCst);
95+
self.foreign_free.store(Address::ZERO, Ordering::Relaxed);
9996
self.next = None;
10097
self.prev = None;
10198
}
@@ -132,7 +129,8 @@ impl Block {
132129
}
133130
cursor += 1;
134131
}
135-
if cursor != start_cursor {
132+
let first_cursor = self.lines().start.get_index_within_block();
133+
if cursor != start_cursor && cursor != first_cursor {
136134
cursor += 1;
137135
}
138136
if cursor >= self.line_liveness.len() {
@@ -191,6 +189,7 @@ impl Block {
191189
self.line_liveness[i] += 1;
192190
}
193191
self.live_lines += lines;
192+
// self.drain_foreign_free();
194193
// println!(" - BA {:x?} live-lines {}", self, self.live_lines);
195194
}
196195

@@ -237,14 +236,42 @@ impl Block {
237236
// println!(" - S-{:?} dead-{}", self.state, dead_lines);
238237
if dead_lines >= 1 && self.state == BlockState::Full {
239238
let live_lines = self.live_lines;
240-
if live_lines < Block::DATA_LINES / 2 {
239+
if live_lines <= Block::REUSABLE_THRESHOLD {
241240
let owner = &mut crate::ImmixMutator::current().ix;
242241
owner.add_reusable_block(*self);
243242
}
244243
}
245244
}
246245
}
247246

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+
248275
#[inline]
249276
pub fn on_dealloc(&mut self, ptr: Address, layout: Layout) {
250277
// println!("FLocal {:?}", ptr..(ptr + layout.size()));

ix/src/immix_space.rs

+10-1
Original file line numberDiff line numberDiff line change
@@ -110,6 +110,7 @@ impl ImmixAllocator {
110110
if let Some(mut next) = next {
111111
next.prev = None;
112112
}
113+
b.drain_foreign_free();
113114
self.reusable_blocks = next;
114115
// println!(
115116
// " - acquire_reusable_block {:x?} next = {:x?}",
@@ -134,7 +135,7 @@ impl ImmixAllocator {
134135
debug_assert!(b.prev.is_none());
135136
debug_assert_ne!(Some(b), self.reusable_blocks);
136137
self.space.release_block(b);
137-
} else if live_lines < Block::DATA_LINES / 2 {
138+
} else if live_lines <= Block::REUSABLE_THRESHOLD {
138139
self.add_reusable_block(b);
139140
} else {
140141
b.state = BlockState::Full;
@@ -257,6 +258,8 @@ impl Allocator for ImmixAllocator {
257258
let result = self.cursor;
258259
let new_cursor = self.cursor + layout.size();
259260
let result = if new_cursor > self.limit {
261+
self.large_block.map(|mut b| b.drain_foreign_free());
262+
self.block.map(|mut b| b.drain_foreign_free());
260263
if layout.size() > Line::BYTES {
261264
// Size larger than a line: do large allocation
262265
self.overflow_alloc(layout)
@@ -289,5 +292,11 @@ impl Drop for ImmixAllocator {
289292
fn drop(&mut self) {
290293
// self.tlab
291294
// .clear(|cell| self.local.free_cell(cell, self.space));
295+
// println!(" - ImmixAllocator.drop {:x?}", self as *const Self);
296+
let mut reusable_blocks = self.reusable_blocks;
297+
while let Some(mut block) = reusable_blocks {
298+
reusable_blocks = block.next;
299+
block.drain_foreign_free();
300+
}
292301
}
293302
}

0 commit comments

Comments
 (0)