bevy/pipelined/bevy_render2/src/render_resource/buffer_vec.rs

89 lines
2.2 KiB
Rust
Raw Normal View History

use crate::{
render_resource::Buffer,
renderer::{RenderDevice, RenderQueue},
};
use bevy_core::{cast_slice, Pod};
use wgpu::BufferUsages;
2021-04-11 20:13:07 +00:00
pub struct BufferVec<T: Pod> {
values: Vec<T>,
2021-06-21 23:28:52 +00:00
buffer: Option<Buffer>,
2021-04-11 20:13:07 +00:00
capacity: usize,
item_size: usize,
buffer_usage: BufferUsages,
2021-04-11 20:13:07 +00:00
}
impl<T: Pod> Default for BufferVec<T> {
fn default() -> Self {
Self {
values: Vec::new(),
buffer: None,
capacity: 0,
buffer_usage: BufferUsages::all(),
2021-04-11 20:13:07 +00:00
item_size: std::mem::size_of::<T>(),
}
}
}
impl<T: Pod> BufferVec<T> {
pub fn new(buffer_usage: BufferUsages) -> Self {
2021-04-11 20:13:07 +00:00
Self {
buffer_usage,
..Default::default()
}
}
#[inline]
2021-06-21 23:28:52 +00:00
pub fn buffer(&self) -> Option<&Buffer> {
self.buffer.as_ref()
2021-04-11 20:13:07 +00:00
}
#[inline]
pub fn capacity(&self) -> usize {
self.capacity
}
pub fn push(&mut self, value: T) -> usize {
let len = self.values.len();
if len < self.capacity {
2021-04-11 20:13:07 +00:00
self.values.push(value);
len
2021-04-11 20:13:07 +00:00
} else {
panic!(
"Cannot push value because capacity of {} has been reached",
self.capacity
);
}
}
2021-06-21 23:28:52 +00:00
pub fn reserve(&mut self, capacity: usize, device: &RenderDevice) {
2021-04-11 20:13:07 +00:00
if capacity > self.capacity {
self.capacity = capacity;
let size = self.item_size * capacity;
2021-06-21 23:28:52 +00:00
self.buffer = Some(device.create_buffer(&wgpu::BufferDescriptor {
label: None,
size: size as wgpu::BufferAddress,
usage: BufferUsages::COPY_DST | self.buffer_usage,
2021-04-11 20:13:07 +00:00
mapped_at_creation: false,
}));
}
}
2021-06-21 23:28:52 +00:00
pub fn reserve_and_clear(&mut self, capacity: usize, device: &RenderDevice) {
2021-04-11 20:13:07 +00:00
self.clear();
2021-06-21 23:28:52 +00:00
self.reserve(capacity, device);
2021-04-11 20:13:07 +00:00
}
pub fn write_buffer(&mut self, queue: &RenderQueue) {
if let Some(buffer) = &self.buffer {
let range = 0..self.item_size * self.values.len();
let bytes: &[u8] = cast_slice(&self.values);
queue.write_buffer(buffer, 0, &bytes[range]);
2021-04-11 20:13:07 +00:00
}
}
pub fn clear(&mut self) {
self.values.clear();
}
}