mirror of
https://github.com/vulkano-rs/vulkano.git
synced 2024-11-25 08:14:20 +00:00
Refactor some braindead code in FreeListAllocator
(#2492)
This commit is contained in:
parent
ee0cabfa65
commit
7bba8ccca3
@ -403,17 +403,34 @@ unsafe impl Suballocator for FreeListAllocator {
|
|||||||
unsafe {
|
unsafe {
|
||||||
match state.free_list.last() {
|
match state.free_list.last() {
|
||||||
Some(&last) if state.nodes.get(last).size >= size => {
|
Some(&last) if state.nodes.get(last).size >= size => {
|
||||||
let index = match state
|
// We create a dummy node to compare against in the below binary search. The
|
||||||
.free_list
|
// only fields of importance are `offset` and `size`. It is paramount that we
|
||||||
.binary_search_by_key(&size, |&id| state.nodes.get(id).size)
|
// set `offset` to zero, so that in the case where there are multiple free
|
||||||
{
|
// suballocations with the same size, we get the first one of them, that is,
|
||||||
// Exact fit.
|
// the one with the lowest offset.
|
||||||
Ok(index) => index,
|
let dummy_node = SuballocationListNode {
|
||||||
// Next-best fit. Note that `index == free_list.len()` can't be because we
|
prev: None,
|
||||||
// checked that the free-list contains a suballocation that is big enough.
|
next: None,
|
||||||
Err(index) => index,
|
offset: 0,
|
||||||
|
size,
|
||||||
|
ty: SuballocationType::Unknown,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// This is almost exclusively going to return `Err`, but that's expected: we
|
||||||
|
// are first comparing the size, looking for the smallest one not less than
|
||||||
|
// `size`, however the next-best will do as well (that is, a size somewhat
|
||||||
|
// larger). In that case we get `Err`. If we do find a suballocation with the
|
||||||
|
// exact size however, we are then comparing the offsets to make sure we get
|
||||||
|
// the suballocation with the lowest offset, in case there are multiple with
|
||||||
|
// the same size. In that case we also exclusively get `Err` except when the
|
||||||
|
// offset is zero.
|
||||||
|
//
|
||||||
|
// Note that `index == free_list.len()` can't be because we checked that the
|
||||||
|
// free-list contains a suballocation that is big enough.
|
||||||
|
let (Ok(index) | Err(index)) = state
|
||||||
|
.free_list
|
||||||
|
.binary_search_by_key(&dummy_node, |&id| state.nodes.get(id));
|
||||||
|
|
||||||
for (index, &id) in state.free_list.iter().enumerate().skip(index) {
|
for (index, &id) in state.free_list.iter().enumerate().skip(index) {
|
||||||
let suballoc = state.nodes.get(id);
|
let suballoc = state.nodes.get(id);
|
||||||
|
|
||||||
@ -530,6 +547,31 @@ struct SuballocationListNode {
|
|||||||
ty: SuballocationType,
|
ty: SuballocationType,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl PartialEq for SuballocationListNode {
|
||||||
|
fn eq(&self, other: &Self) -> bool {
|
||||||
|
self.size == other.size && self.offset == other.offset
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Eq for SuballocationListNode {}
|
||||||
|
|
||||||
|
impl PartialOrd for SuballocationListNode {
|
||||||
|
fn partial_cmp(&self, other: &Self) -> Option<cmp::Ordering> {
|
||||||
|
Some(self.cmp(other))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Ord for SuballocationListNode {
|
||||||
|
fn cmp(&self, other: &Self) -> cmp::Ordering {
|
||||||
|
// We want to sort the free-list by size.
|
||||||
|
self.size
|
||||||
|
.cmp(&other.size)
|
||||||
|
// However there might be multiple free suballocations with the same size, so we need
|
||||||
|
// to compare the offset as well to differentiate.
|
||||||
|
.then(self.offset.cmp(&other.offset))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Tells us if a suballocation is free, and if not, whether it is linear or not. This is needed in
|
/// Tells us if a suballocation is free, and if not, whether it is linear or not. This is needed in
|
||||||
/// order to be able to respect the buffer-image granularity.
|
/// order to be able to respect the buffer-image granularity.
|
||||||
#[derive(Clone, Copy, Debug, PartialEq, Eq)]
|
#[derive(Clone, Copy, Debug, PartialEq, Eq)]
|
||||||
@ -564,55 +606,10 @@ impl FreeListAllocatorState {
|
|||||||
|
|
||||||
match self
|
match self
|
||||||
.free_list
|
.free_list
|
||||||
.binary_search_by_key(&node.size, |&id| self.nodes.get(id).size)
|
.binary_search_by_key(&node, |&id| self.nodes.get(id))
|
||||||
{
|
{
|
||||||
Ok(index) => {
|
Ok(index) => {
|
||||||
// If there are multiple free suballocations with the same size, the search might
|
|
||||||
// have returned any one, so we need to find the one corresponding to the target ID.
|
|
||||||
if self.free_list[index] == node_id {
|
|
||||||
self.free_list.remove(index);
|
self.free_list.remove(index);
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check all previous indices that point to suballocations with the same size.
|
|
||||||
{
|
|
||||||
let mut index = index;
|
|
||||||
loop {
|
|
||||||
index = index.wrapping_sub(1);
|
|
||||||
if let Some(&id) = self.free_list.get(index) {
|
|
||||||
if id == node_id {
|
|
||||||
self.free_list.remove(index);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if self.nodes.get(id).size != node.size {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check all next indices that point to suballocations with the same size.
|
|
||||||
{
|
|
||||||
let mut index = index;
|
|
||||||
loop {
|
|
||||||
index += 1;
|
|
||||||
if let Some(&id) = self.free_list.get(index) {
|
|
||||||
if id == node_id {
|
|
||||||
self.free_list.remove(index);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if self.nodes.get(id).size != node.size {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
unreachable!();
|
|
||||||
}
|
}
|
||||||
Err(_) => unreachable!(),
|
Err(_) => unreachable!(),
|
||||||
}
|
}
|
||||||
@ -698,7 +695,7 @@ impl FreeListAllocatorState {
|
|||||||
let node = self.nodes.get(node_id);
|
let node = self.nodes.get(node_id);
|
||||||
let (Ok(index) | Err(index)) = self
|
let (Ok(index) | Err(index)) = self
|
||||||
.free_list
|
.free_list
|
||||||
.binary_search_by_key(&node.size, |&id| self.nodes.get(id).size);
|
.binary_search_by_key(&node, |&id| self.nodes.get(id));
|
||||||
self.free_list.insert(index, node_id);
|
self.free_list.insert(index, node_id);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user