gpgpu

A simple GPU compute library based on wgpu. It is meant to be used alongside wgpu if desired.

To start using gpgpu, just create a [Framework] instance and follow the examples in the main repository.

Example

Small program that multiplies 2 vectors A and B; and stores the result in another vector C.

Rust program

```rust use gpgpu::*;

fn main() -> GpuResult<()> { let fw = Framework::default();

// Original CPU data
let cpu_data = (0..10000).into_iter().collect::<Vec<u32>>();

// GPU buffer creation
let buf_a = GpuBuffer::from_slice(&fw, &cpu_data);       // Input
let buf_b = GpuBuffer::from_slice(&fw, &cpu_data);       // Input
let buf_c = GpuBuffer::<u32>::new(&fw, cpu_data.len());  // Output

// Shader load from SPIR-V binary file
let shader_module = utils::shader::from_spirv_file(&fw, "<SPIR-V shader path>")?;
//  or from a WGSL source file
let shader_module = utils::shader::from_wgsl_file(&fw, "<WGSL shader path>")?;    

// Descriptor set creation
let desc_set = DescriptorSet::default()
    .bind_buffer(&buf_a, GpuBufferUsage::ReadOnly)
    .bind_buffer(&buf_b, GpuBufferUsage::ReadOnly)
    .bind_buffer(&buf_c, GpuBufferUsage::ReadWrite);

// Kernel creation and enqueuing
fw.create_kernel_builder(&shader_module, "main")   // Entry point
    .add_descriptor_set(desc_set)                      
    .build()
    .enqueue(cpu_data.len() as u32, 1, 1);         // Enqueuing, not very optimus 😅

let output = buf_c.read()?;                        // Read back C from GPU
for (a, b) in cpu_data.into_iter().zip(output) {
    assert_eq!(a.pow(2), b);
}

Ok(())

} ```

Shader program

The shader is writen in WGSL ```rust // Vector type definition. Used for both input and output [[block]] struct Vector { data: [[stride(4)]] array; };

// A, B and C vectors [[group(0), binding(0)]] var a: Vector; [[group(0), binding(1)]] var b: Vector; [[group(0), binding(2)]] var c: Vector;

[[stage(compute), workgroupsize(1)]] fn main([[builtin(globalinvocationid)]] globalid: vec3) { c.data[globalid.x] = a.data[globalid.x] * b.data[global_id.x]; } ```