WebGPUAttributes.js 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187
  1. class WebGPUAttributes {
  2. constructor( device ) {
  3. this.buffers = new WeakMap();
  4. this.device = device;
  5. }
  6. get( attribute ) {
  7. if ( attribute.isInterleavedBufferAttribute ) attribute = attribute.data;
  8. return this.buffers.get( attribute );
  9. }
  10. remove( attribute ) {
  11. if ( attribute.isInterleavedBufferAttribute ) attribute = attribute.data;
  12. const data = this.buffers.get( attribute );
  13. if ( data ) {
  14. this._destroyBuffers( data );
  15. this.buffers.delete( attribute );
  16. }
  17. }
  18. update( attribute, isIndex = false, usage = null ) {
  19. if ( attribute.isInterleavedBufferAttribute ) attribute = attribute.data;
  20. let data = this.buffers.get( attribute );
  21. if ( data === undefined ) {
  22. if ( usage === null ) {
  23. usage = ( isIndex === true ) ? GPUBufferUsage.INDEX : GPUBufferUsage.VERTEX;
  24. }
  25. data = this._createBuffer( attribute, usage );
  26. this.buffers.set( attribute, data );
  27. } else if ( usage && usage !== data.usage ) {
  28. this._destroyBuffers( data );
  29. data = this._createBuffer( attribute, usage );
  30. this.buffers.set( attribute, data );
  31. } else if ( data.version < attribute.version ) {
  32. this._writeBuffer( data.buffer, attribute );
  33. data.version = attribute.version;
  34. }
  35. }
  36. async getArrayBuffer( attribute ) {
  37. const data = this.get( attribute );
  38. const device = this.device;
  39. const gpuBuffer = data.buffer;
  40. const size = gpuBuffer.size;
  41. let gpuReadBuffer = data.readBuffer;
  42. let needsUnmap = true;
  43. if ( gpuReadBuffer === null ) {
  44. gpuReadBuffer = device.createBuffer( {
  45. size,
  46. usage: GPUBufferUsage.COPY_DST | GPUBufferUsage.MAP_READ
  47. } );
  48. needsUnmap = false;
  49. data.readBuffer = gpuReadBuffer;
  50. }
  51. const cmdEncoder = device.createCommandEncoder( {} );
  52. cmdEncoder.copyBufferToBuffer(
  53. gpuBuffer,
  54. 0,
  55. gpuReadBuffer,
  56. 0,
  57. size
  58. );
  59. if ( needsUnmap ) gpuReadBuffer.unmap();
  60. const gpuCommands = cmdEncoder.finish();
  61. device.queue.submit( [ gpuCommands ] );
  62. await gpuReadBuffer.mapAsync( GPUMapMode.READ );
  63. const arrayBuffer = gpuReadBuffer.getMappedRange();
  64. return arrayBuffer;
  65. }
  66. _createBuffer( attribute, usage ) {
  67. const array = attribute.array;
  68. const size = array.byteLength + ( ( 4 - ( array.byteLength % 4 ) ) % 4 ); // ensure 4 byte alignment, see #20441
  69. const buffer = this.device.createBuffer( {
  70. size,
  71. usage: usage | GPUBufferUsage.COPY_SRC | GPUBufferUsage.COPY_DST,
  72. mappedAtCreation: true
  73. } );
  74. new array.constructor( buffer.getMappedRange() ).set( array );
  75. buffer.unmap();
  76. attribute.onUploadCallback();
  77. return {
  78. version: attribute.version,
  79. buffer,
  80. readBuffer: null,
  81. usage
  82. };
  83. }
  84. _writeBuffer( buffer, attribute ) {
  85. const device = this.device;
  86. const array = attribute.array;
  87. const updateRange = attribute.updateRange;
  88. if ( updateRange.count === - 1 ) {
  89. // Not using update ranges
  90. device.queue.writeBuffer(
  91. buffer,
  92. 0,
  93. array,
  94. 0
  95. );
  96. } else {
  97. device.queue.writeBuffer(
  98. buffer,
  99. 0,
  100. array,
  101. updateRange.offset * array.BYTES_PER_ELEMENT,
  102. updateRange.count * array.BYTES_PER_ELEMENT
  103. );
  104. updateRange.count = - 1; // reset range
  105. }
  106. }
  107. _destroyBuffers( { buffer, readBuffer } ) {
  108. buffer.destroy();
  109. if ( readBuffer !== null ) readBuffer.destroy();
  110. }
  111. }
  112. export default WebGPUAttributes;