pub struct DataTextureSource<'a, T: Pod + Send + Sync> {
    ctx: &'a RenderContext,
    buffers: Vec<CpuWriteGpuReadBuffer<T>>,
    active_buffer_index: usize,
}
Expand description

Utility for writing data to a dynamically sized “data textures”.

For WebGL compatibility we sometimes have to use textures instead of buffers. We call these textures “data textures”. This construct allows to write data directly to gpu readable memory which then upon finishing is automatically copied into an appropriately sized texture which receives all data written to DataTextureSource. Each texel in the data texture represents a single element of the type T.

This is implemented by dynamically allocating cpu-write-gpu-read buffers from the central super::CpuWriteGpuReadBelt and copying all of them to the texture in the end.

Fields§

§ctx: &'a RenderContext§buffers: Vec<CpuWriteGpuReadBuffer<T>>

Buffers that need to be transferred to the data texture in the end.

We have two options on how to deal with buffer allocation and filling:

  1. fill the last buffer to its maximum capacity before starting writing to the next, allow arbitrary amount of empty buffers -> Pro: makes the final gpu copies easy, we don’t have to juggle weird offsets and several copies per buffer! -> Con: may need to spread writes over several buffers
  2. create a new buffer whenever a write doesn’t fully fit into the active buffer, even if the last buffer has some remaining capacity, allow arbitrary amount of half-filled buffers -> Pro: All writes can go to a single buffer, which is simpler & faster. -> Con: We waste space and copying to the texture is harder

We’re going with option (1)!

This means that there might be ‘n’ full buffers followed by a single active buffer, followed by ‘m’ empty buffers.

§active_buffer_index: usize

Buffer in which data is currently written.

Between all public calls:

  • all buffers before are full
  • all buffers after are empty (if any)
  • the buffer at this index either does not exist or has remaining capacity

At the end of any operation that adds new elements, call ensure_active_buffer_invariant_after_adding_elements to ensure this invariant.

Implementations§

source§

impl<'a, T: Pod + Send + Sync> DataTextureSource<'a, T>

source

pub fn new(ctx: &'a RenderContext) -> Self

Creates a new DataTextureSource with the given RenderContext.

This operation itself will not allocate any memory, empty DataTextureSource are not a concern.

source

pub fn is_empty(&self) -> bool

Whether no elements have been written at all.

source

pub fn len(&self) -> usize

The number of elements written so far.

source

pub fn capacity(&self) -> usize

The number of elements that can be written without allocating more memory.

source

pub fn remaining_capacity(&self) -> usize

The number of elements that can be written without allocating more memory.

source

fn ensure_active_buffer_invariant_after_adding_elements(&mut self)

Ensure invariant that the active buffer has some remaining capacity or is the next buffer that needs to be allocated.

Since elements were just added to the active buffer, this function assumes that the active_buffer_index points to a valid buffer.

source

pub fn reserve( &mut self, num_elements: usize, ) -> Result<usize, CpuWriteGpuReadError>

Ensures that there’s space internally for at least num_elements more elements.

Returns the number of elements that are currently reserved. This value is smaller than the requested number of elements if the maximum number of elements that can be stored is reached, see max_num_elements_per_data_texture.

Creating new buffers is a relatively expensive operation, so it’s best to reserve gratuitously and often. Ideally, you know exactly how many elements you’re going to write and reserve accordingly at the start.

If there’s no more space, a new buffer is allocated such that:

  • have a total capacity for at least as many elements as requested, clamping total size to max_num_elements_per_data_texture
  • be at least double the size of the last buffer
  • keep it easy to copy to textures by always being a multiple of the maximum row size we use for data textures -> this massively simplifies the buffer->texture copy logic!
source

fn error_on_clamped_write( &self, num_elements_attempted_to_add: usize, num_elements_actually_added: usize, ) -> Result<(), DataTextureSourceWriteError>

source

pub fn extend_from_slice( &mut self, elements: &[T], ) -> Result<(), DataTextureSourceWriteError>

Pushes a slice of elements into the data texture.

source

pub fn add_n( &mut self, element: T, num_elements: usize, ) -> Result<(), DataTextureSourceWriteError>

Fills the data texture with n instances of an element.

source

pub fn push(&mut self, element: T) -> Result<(), DataTextureSourceWriteError>

source

fn data_texture_size(&self, max_texture_dimension_2d: u32) -> Extent3d

source

pub fn finish( self, texture_format: TextureFormat, texture_label: impl Into<DebugLabel>, ) -> Result<Arc<DynamicResource<GpuTextureHandle, TextureDesc, GpuTextureInternal>>, CpuWriteGpuReadError>

Schedules copies of all previous writes to this DataTextureSource to a GpuTexture.

The format has to be uncompressed, not a depth/stencil format and have the exact same block size of the size of type T. The resulting GpuTexture is ready to be bound as a data texture in a shader.

Auto Trait Implementations§

§

impl<'a, T> Freeze for DataTextureSource<'a, T>

§

impl<'a, T> !RefUnwindSafe for DataTextureSource<'a, T>

§

impl<'a, T> Send for DataTextureSource<'a, T>

§

impl<'a, T> Sync for DataTextureSource<'a, T>

§

impl<'a, T> Unpin for DataTextureSource<'a, T>
where T: Unpin,

§

impl<'a, T> !UnwindSafe for DataTextureSource<'a, T>

Blanket Implementations§

source§

impl<T> Any for T
where T: 'static + ?Sized,

source§

fn type_id(&self) -> TypeId

Gets the TypeId of self. Read more
source§

impl<T> Borrow<T> for T
where T: ?Sized,

source§

fn borrow(&self) -> &T

Immutably borrows from an owned value. Read more
source§

impl<T> BorrowMut<T> for T
where T: ?Sized,

source§

fn borrow_mut(&mut self) -> &mut T

Mutably borrows from an owned value. Read more
§

impl<T> Downcast<T> for T

§

fn downcast(&self) -> &T

§

impl<T> Downcast for T
where T: Any,

§

fn into_any(self: Box<T>) -> Box<dyn Any>

Convert Box<dyn Trait> (where Trait: Downcast) to Box<dyn Any>. Box<dyn Any> can then be further downcast into Box<ConcreteType> where ConcreteType implements Trait.
§

fn into_any_rc(self: Rc<T>) -> Rc<dyn Any>

Convert Rc<Trait> (where Trait: Downcast) to Rc<Any>. Rc<Any> can then be further downcast into Rc<ConcreteType> where ConcreteType implements Trait.
§

fn as_any(&self) -> &(dyn Any + 'static)

Convert &Trait (where Trait: Downcast) to &Any. This is needed since Rust cannot generate &Any’s vtable from &Trait’s.
§

fn as_any_mut(&mut self) -> &mut (dyn Any + 'static)

Convert &mut Trait (where Trait: Downcast) to &Any. This is needed since Rust cannot generate &mut Any’s vtable from &mut Trait’s.
§

impl<T> DowncastSync for T
where T: Any + Send + Sync,

§

fn into_any_arc(self: Arc<T>) -> Arc<dyn Any + Sync + Send>

Convert Arc<Trait> (where Trait: Downcast) to Arc<Any>. Arc<Any> can then be further downcast into Arc<ConcreteType> where ConcreteType implements Trait.
source§

impl<T> From<T> for T

source§

fn from(t: T) -> T

Returns the argument unchanged.

§

impl<T> Instrument for T

§

fn instrument(self, span: Span) -> Instrumented<Self>

Instruments this type with the provided [Span], returning an Instrumented wrapper. Read more
§

fn in_current_span(self) -> Instrumented<Self>

Instruments this type with the current Span, returning an Instrumented wrapper. Read more
source§

impl<T, U> Into<U> for T
where U: From<T>,

source§

fn into(self) -> U

Calls U::from(self).

That is, this conversion is whatever the implementation of From<T> for U chooses to do.

source§

impl<T> IntoEither for T

source§

fn into_either(self, into_left: bool) -> Either<Self, Self>

Converts self into a Left variant of Either<Self, Self> if into_left is true. Converts self into a Right variant of Either<Self, Self> otherwise. Read more
source§

fn into_either_with<F>(self, into_left: F) -> Either<Self, Self>
where F: FnOnce(&Self) -> bool,

Converts self into a Left variant of Either<Self, Self> if into_left(&self) returns true. Converts self into a Right variant of Either<Self, Self> otherwise. Read more
§

impl<T> Pointable for T

§

const ALIGN: usize = _

The alignment of pointer.
§

type Init = T

The type for initializers.
§

unsafe fn init(init: <T as Pointable>::Init) -> usize

Initializes a with the given initializer. Read more
§

unsafe fn deref<'a>(ptr: usize) -> &'a T

Dereferences the given pointer. Read more
§

unsafe fn deref_mut<'a>(ptr: usize) -> &'a mut T

Mutably dereferences the given pointer. Read more
§

unsafe fn drop(ptr: usize)

Drops the object pointed to by the given pointer. Read more
source§

impl<T> Same for T

§

type Output = T

Should always be Self
§

impl<T> To for T
where T: ?Sized,

§

fn to<T>(self) -> T
where Self: Into<T>,

Converts to T by calling Into<T>::into.
§

fn try_to<T>(self) -> Result<T, Self::Error>
where Self: TryInto<T>,

Tries to convert to T by calling TryInto<T>::try_into.
source§

impl<T, U> TryFrom<U> for T
where U: Into<T>,

§

type Error = Infallible

The type returned in the event of a conversion error.
source§

fn try_from(value: U) -> Result<T, <T as TryFrom<U>>::Error>

Performs the conversion.
source§

impl<T, U> TryInto<U> for T
where U: TryFrom<T>,

§

type Error = <U as TryFrom<T>>::Error

The type returned in the event of a conversion error.
source§

fn try_into(self) -> Result<U, <U as TryFrom<T>>::Error>

Performs the conversion.
§

impl<T> Upcast<T> for T

§

fn upcast(&self) -> Option<&T>

§

impl<V, T> VZip<V> for T
where V: MultiLane<T>,

§

fn vzip(self) -> V

§

impl<T> WithSubscriber for T

§

fn with_subscriber<S>(self, subscriber: S) -> WithDispatch<Self>
where S: Into<Dispatch>,

Attaches the provided Subscriber to this type, returning a [WithDispatch] wrapper. Read more
§

fn with_current_subscriber(self) -> WithDispatch<Self>

Attaches the current default Subscriber to this type, returning a [WithDispatch] wrapper. Read more
§

impl<T> ErasedDestructor for T
where T: 'static,

§

impl<T> MaybeSendSync for T

§

impl<T> WasmNotSend for T
where T: Send,

§

impl<T> WasmNotSendSync for T
where T: WasmNotSend + WasmNotSync,

§

impl<T> WasmNotSync for T
where T: Sync,