Skip to content

Commit

Permalink
Use Layout::extend and clean up code around alloc
Browse files Browse the repository at this point in the history
  • Loading branch information
taiki-e committed Dec 24, 2023
1 parent 4aeb87e commit 26d6fd9
Show file tree
Hide file tree
Showing 2 changed files with 33 additions and 29 deletions.
36 changes: 20 additions & 16 deletions crossbeam-epoch/src/atomic.rs
Original file line number Diff line number Diff line change
@@ -1,13 +1,14 @@
use alloc::boxed::Box;
use core::alloc::Layout;
use core::borrow::{Borrow, BorrowMut};
use core::cmp;
use core::fmt;
use core::marker::PhantomData;
use core::mem::{self, MaybeUninit};
use core::ops::{Deref, DerefMut};
use core::ptr;
use core::slice;

use crate::alloc::alloc;
use crate::alloc::boxed::Box;
use crate::guard::Guard;
use crate::primitive::sync::atomic::{AtomicUsize, Ordering};
use crossbeam_utils::atomic::AtomicConsume;
Expand Down Expand Up @@ -232,30 +233,35 @@ impl<T> Pointable for T {
///
/// Elements are not present in the type, but they will be in the allocation.
/// ```
///
// TODO(@jeehoonkang): once we bump the minimum required Rust version to 1.44 or newer, use
// [`alloc::alloc::Layout::extend`] instead.
#[repr(C)]
struct Array<T> {
/// The number of elements (not the number of bytes).
len: usize,
elements: [MaybeUninit<T>; 0],
}

impl<T> Array<T> {
fn layout(len: usize) -> Layout {
Layout::new::<Self>()
.extend(Layout::array::<MaybeUninit<T>>(len).unwrap())
.unwrap()
.0
.pad_to_align()
}
}

impl<T> Pointable for [MaybeUninit<T>] {
const ALIGN: usize = mem::align_of::<Array<T>>();

type Init = usize;

unsafe fn init(len: Self::Init) -> usize {
let size = mem::size_of::<Array<T>>() + mem::size_of::<MaybeUninit<T>>() * len;
let align = mem::align_of::<Array<T>>();
let layout = alloc::Layout::from_size_align(size, align).unwrap();
let ptr = alloc::alloc(layout).cast::<Array<T>>();
let layout = Array::<T>::layout(len);
let ptr = alloc::alloc::alloc(layout).cast::<Array<T>>();
if ptr.is_null() {
alloc::handle_alloc_error(layout);
alloc::alloc::handle_alloc_error(layout);
}
(*ptr).len = len;
ptr::addr_of_mut!((*ptr).len).write(len);
ptr as usize
}

Expand All @@ -270,11 +276,9 @@ impl<T> Pointable for [MaybeUninit<T>] {
}

unsafe fn drop(ptr: usize) {
let array = &*(ptr as *mut Array<T>);
let size = mem::size_of::<Array<T>>() + mem::size_of::<MaybeUninit<T>>() * array.len;
let align = mem::align_of::<Array<T>>();
let layout = alloc::Layout::from_size_align(size, align).unwrap();
alloc::dealloc(ptr as *mut u8, layout);
let len = (*(ptr as *mut Array<T>)).len;
let layout = Array::<T>::layout(len);
alloc::alloc::dealloc(ptr as *mut u8, layout);
}
}

Expand Down
26 changes: 13 additions & 13 deletions crossbeam-skiplist/src/base.rs
Original file line number Diff line number Diff line change
Expand Up @@ -104,11 +104,11 @@ impl<K, V> Node<K, V> {
handle_alloc_error(layout);
}

ptr::write(
&mut (*ptr).refs_and_height,
AtomicUsize::new((height - 1) | ref_count << HEIGHT_BITS),
);
ptr::write_bytes((*ptr).tower.pointers.as_mut_ptr(), 0, height);
ptr::addr_of_mut!((*ptr).refs_and_height)
.write(AtomicUsize::new((height - 1) | ref_count << HEIGHT_BITS));
ptr::addr_of_mut!((*ptr).tower.pointers)
.cast::<Atomic<Self>>()
.write_bytes(0, height);
ptr
}

Expand All @@ -122,14 +122,14 @@ impl<K, V> Node<K, V> {
}

/// Returns the layout of a node with the given `height`.
unsafe fn get_layout(height: usize) -> Layout {
fn get_layout(height: usize) -> Layout {
assert!((1..=MAX_HEIGHT).contains(&height));

let size_self = mem::size_of::<Self>();
let align_self = mem::align_of::<Self>();
let size_pointer = mem::size_of::<Atomic<Self>>();

Layout::from_size_align_unchecked(size_self + size_pointer * height, align_self)
Layout::new::<Self>()
.extend(Layout::array::<Atomic<Self>>(height).unwrap())
.unwrap()
.0
.pad_to_align()
}

/// Returns the height of this node's tower.
Expand Down Expand Up @@ -906,8 +906,8 @@ where
let n = Node::<K, V>::alloc(height, 2);

// Write the key and the value into the node.
ptr::write(&mut (*n).key, key);
ptr::write(&mut (*n).value, value);
ptr::addr_of_mut!((*n).key).write(key);
ptr::addr_of_mut!((*n).value).write(value);

(Shared::<Node<K, V>>::from(n as *const _), &*n)
};
Expand Down

0 comments on commit 26d6fd9

Please sign in to comment.