UNPKG

@animech-public/playcanvas

Version:
152 lines (140 loc) 4.76 kB
import { Debug, DebugHelper } from '../../../core/debug.js'; import { WebgpuDebug } from './webgpu-debug.js'; /** * A WebGPU implementation of the BindGroup, which is a wrapper over GPUBindGroup. * * @ignore */ class WebgpuBindGroup { constructor() { /** * @type {GPUBindGroup} * @private */ this.bindGroup = void 0; } update(bindGroup) { this.destroy(); const device = bindGroup.device; /** @type {GPUBindGroupDescriptor} */ const descr = this.createDescriptor(device, bindGroup); WebgpuDebug.validate(device); this.bindGroup = device.wgpu.createBindGroup(descr); WebgpuDebug.end(device, { debugFormat: this.debugFormat, descr: descr, format: bindGroup.format, bindGroup: bindGroup }); } destroy() { // this.bindGroup?.destroy(); this.bindGroup = null; } /** * Creates a bind group descriptor in WebGPU format * * @param {import('./webgpu-graphics-device.js').WebgpuGraphicsDevice} device - Graphics device. * @param {import('../bind-group.js').BindGroup} bindGroup - Bind group to create the * descriptor for. * @returns {object} - Returns the generated descriptor of type * GPUBindGroupDescriptor, which can be used to create a GPUBindGroup */ createDescriptor(device, bindGroup) { // Note: This needs to match WebgpuBindGroupFormat.createDescriptor const entries = []; const format = bindGroup.format; Debug.call(() => { this.debugFormat = ''; }); // uniform buffers const uniformBufferFormats = bindGroup.format.uniformBufferFormats; bindGroup.uniformBuffers.forEach((ub, i) => { const slot = uniformBufferFormats[i].slot; const buffer = ub.persistent ? ub.impl.buffer : ub.allocation.gpuBuffer.buffer; Debug.assert(buffer, 'NULL uniform buffer cannot be used by the bind group'); Debug.call(() => { this.debugFormat += `${slot}: UB\n`; }); entries.push({ binding: slot, resource: { buffer: buffer, offset: 0, size: ub.format.byteSize } }); }); // textures const textureFormats = bindGroup.format.textureFormats; bindGroup.textures.forEach((tex, textureIndex) => { /** @type {import('./webgpu-texture.js').WebgpuTexture} */ const wgpuTexture = tex.impl; const textureFormat = format.textureFormats[textureIndex]; const slot = textureFormats[textureIndex].slot; // texture const view = wgpuTexture.getView(device); Debug.assert(view, 'NULL texture view cannot be used by the bind group'); Debug.call(() => { this.debugFormat += `${slot}: ${bindGroup.format.textureFormats[textureIndex].name}\n`; }); entries.push({ binding: slot, resource: view }); // sampler if (textureFormat.hasSampler) { const sampler = wgpuTexture.getSampler(device, textureFormat.sampleType); Debug.assert(sampler, 'NULL sampler cannot be used by the bind group'); Debug.call(() => { this.debugFormat += `${slot + 1}: ${sampler.label}\n`; }); entries.push({ binding: slot + 1, resource: sampler }); } }); // storage textures const storageTextureFormats = bindGroup.format.storageTextureFormats; bindGroup.storageTextures.forEach((tex, textureIndex) => { /** @type {import('./webgpu-texture.js').WebgpuTexture} */ const wgpuTexture = tex.impl; const slot = storageTextureFormats[textureIndex].slot; // texture const view = wgpuTexture.getView(device); Debug.assert(view, 'NULL texture view cannot be used by the bind group'); Debug.call(() => { this.debugFormat += `${slot}: ${bindGroup.format.storageTextureFormats[textureIndex].name}\n`; }); entries.push({ binding: slot, resource: view }); }); // storage buffers const storageBufferFormats = bindGroup.format.storageBufferFormats; bindGroup.storageBuffers.forEach((buffer, bufferIndex) => { /** @type {GPUBuffer} */ const wgpuBuffer = buffer.impl.buffer; const slot = storageBufferFormats[bufferIndex].slot; Debug.assert(wgpuBuffer, 'NULL storage buffer cannot be used by the bind group'); Debug.call(() => { this.debugFormat += `${slot}: SB\n`; }); entries.push({ binding: slot, resource: { buffer: wgpuBuffer } }); }); const descr = { layout: bindGroup.format.impl.bindGroupLayout, entries: entries }; DebugHelper.setLabel(descr, bindGroup.name); return descr; } } export { WebgpuBindGroup };