169 lines
4.4 KiB
JavaScript
169 lines
4.4 KiB
JavaScript
import { mat4 } from './glm.js';
|
|
|
|
// Initialize WebGPU
|
|
const adapter = await navigator.gpu.requestAdapter();
|
|
const device = await adapter.requestDevice();
|
|
const canvas = document.querySelector('canvas');
|
|
const context = canvas.getContext('webgpu');
|
|
const format = navigator.gpu.getPreferredCanvasFormat();
|
|
context.configure({ device, format });
|
|
|
|
// Create depth texture
|
|
const depthTexture = device.createTexture({
|
|
size: [canvas.width, canvas.height],
|
|
usage: GPUTextureUsage.RENDER_ATTACHMENT,
|
|
format: 'depth24plus',
|
|
});
|
|
|
|
// Create vertex buffer
|
|
// prettier-ignore
|
|
const vertices = new Float32Array([
|
|
// position // color
|
|
-1, -1, -1, 1, 1, 0, 0, 1,
|
|
1, -1, -1, 1, 0, 1, 0, 1,
|
|
-1, 1, -1, 1, 0, 0, 1, 1,
|
|
1, 1, -1, 1, 1, 1, 0, 1,
|
|
|
|
-1, 1, 1, 1, 1, 0, 0, 1,
|
|
1, -1, 1, 1, 0, 1, 0, 1,
|
|
-1, 1, 1, 1, 0, 0, 1, 1,
|
|
1, 1, 1, 1, 1, 1, 0, 1,
|
|
]);
|
|
|
|
const vertexBuffer = device.createBuffer({
|
|
size: vertices.byteLength,
|
|
usage: GPUBufferUsage.VERTEX | GPUBufferUsage.COPY_DST,
|
|
});
|
|
|
|
device.queue.writeBuffer(vertexBuffer, 0, vertices);
|
|
|
|
// Create index buffer
|
|
// prettier-ignore
|
|
const indices = new Uint32Array([
|
|
0, 1, 2, 1, 2, 3,
|
|
0, 1, 3, 1, 4, 5,
|
|
0, 4, 2, 4, 2, 5,
|
|
1, 5, 3, 5, 3, 7,
|
|
2, 3, 6, 3, 6, 7,
|
|
4, 5, 6, 5, 6, 7,
|
|
]);
|
|
|
|
const indexBuffer = device.createBuffer({
|
|
size: indices.byteLength,
|
|
usage: GPUBufferUsage.INDEX | GPUBufferUsage.COPY_DST,
|
|
});
|
|
|
|
device.queue.writeBuffer(indexBuffer, 0, indices);
|
|
|
|
// Fetch and compile shaders
|
|
const code = await fetch('shader.wgsl').then((response) => response.text());
|
|
const module = device.createShaderModule({ code });
|
|
|
|
// Create the pipeline
|
|
/** @type {GPUVertexBufferLayout} */
|
|
const vertexBufferLayout = {
|
|
arrayStride: 24,
|
|
attributes: [
|
|
{
|
|
shaderLocation: 0,
|
|
offset: 0,
|
|
format: 'float32x2',
|
|
},
|
|
{
|
|
shaderLocation: 1,
|
|
offset: 8,
|
|
format: 'float32x4',
|
|
},
|
|
],
|
|
};
|
|
|
|
const pipeline = device.createRenderPipeline({
|
|
vertex: {
|
|
module,
|
|
buffers: [vertexBufferLayout],
|
|
},
|
|
fragment: {
|
|
module,
|
|
targets: [{ format }],
|
|
},
|
|
depthStencil: {
|
|
depthWriteEnabled: true,
|
|
depthCompare: 'less',
|
|
format: 'depth24plus',
|
|
},
|
|
layout: 'auto',
|
|
});
|
|
|
|
// Create uniform buffer
|
|
const uniformBuffer = device.createBuffer({
|
|
size: 4 * 4 * 4,
|
|
usage: GPUBufferUsage.UNIFORM | GPUBufferUsage.COPY_DST,
|
|
});
|
|
|
|
// Create the bind group
|
|
const bindGroup = device.createBindGroup({
|
|
layout: pipeline.getBindGroupLayout(0),
|
|
entries: [{ binding: 0, resource: { buffer: uniformBuffer } }],
|
|
});
|
|
|
|
function update() {
|
|
// Animate the square
|
|
const time = performance.now() / 1000;
|
|
|
|
const modelMatrix = mat4
|
|
.create()
|
|
.rotateX(time * 0.7)
|
|
.rotateY(time);
|
|
const viewMatrix = mat4.create().translate([0, 0, 5]).invert();
|
|
const projectionMatrix = mat4.create().perspectiveZO(1, 1, 0.01, 100);
|
|
|
|
const matrix = mat4
|
|
.create()
|
|
.multiply(projectionMatrix)
|
|
.multiply(viewMatrix)
|
|
.multiply(modelMatrix);
|
|
|
|
// const radius = 0.5;
|
|
// const frequency = 0.5;
|
|
// const x = radius * Math.cos(frequency * time * 2 * Math.PI);
|
|
// const y = radius * Math.sin(frequency * time * 2 * Math.PI);
|
|
|
|
device.queue.writeBuffer(uniformBuffer, 0, matrix);
|
|
}
|
|
|
|
function render() {
|
|
// Render the square
|
|
const commandEncoder = device.createCommandEncoder();
|
|
const renderPass = commandEncoder.beginRenderPass({
|
|
colorAttachments: [
|
|
{
|
|
view: context.getCurrentTexture().createView(),
|
|
loadOp: 'clear',
|
|
clearValue: [1, 1, 1, 1],
|
|
storeOp: 'store',
|
|
},
|
|
],
|
|
depthStencilAttachment: {
|
|
view: depthTexture.createView(),
|
|
depthLoadOp: 'clear',
|
|
depthClearValue: 1,
|
|
depthStoreOp: 'discard',
|
|
},
|
|
});
|
|
renderPass.setPipeline(pipeline);
|
|
renderPass.setVertexBuffer(0, vertexBuffer);
|
|
renderPass.setIndexBuffer(indexBuffer, 'uint32');
|
|
renderPass.setBindGroup(0, bindGroup);
|
|
renderPass.drawIndexed(indices.length);
|
|
renderPass.end();
|
|
device.queue.submit([commandEncoder.finish()]);
|
|
}
|
|
|
|
function frame() {
|
|
update();
|
|
render();
|
|
requestAnimationFrame(frame);
|
|
}
|
|
|
|
requestAnimationFrame(frame);
|