export async function getProvokingVertexForFlatInterpolationEitherSampling()

in src/webgpu/inter_stage.ts [13:96]


export async function getProvokingVertexForFlatInterpolationEitherSampling(
  t: GPUTest
): Promise<FlatSampling> {
  const { device } = t;
  let sampling = s_deviceToEitherSamplingIndex.get(device);
  if (!sampling) {
    const module = device.createShaderModule({
      code: `
        struct VSOut {
          @builtin(position) position: vec4f,
          @location(0) @interpolate(flat, either) vertexIndex: u32,
        };

        @vertex fn vs(
          @builtin(vertex_index) vertexIndex : u32,
        ) -> VSOut {
          let pos = array(vec2f(-1, 3), vec2f(3, -1), vec2f(-1, -1));
          var vsOutput: VSOut;
          vsOutput.position = vec4f(pos[vertexIndex], 0, 1);
          vsOutput.vertexIndex = vertexIndex;
          return vsOutput;
        }

        @fragment fn fs(@location(0) @interpolate(flat, either) vertexIndex: u32) -> @location(0) vec4u {
          return vec4u(vertexIndex);
        }
      `,
    });

    const pipeline = device.createRenderPipeline({
      layout: 'auto',
      vertex: {
        module,
      },
      fragment: {
        module,
        targets: [{ format: 'rgba8uint' }],
      },
    });

    const texture = t.createTextureTracked({
      format: 'rgba8uint',
      size: [1, 1],
      usage: GPUTextureUsage.RENDER_ATTACHMENT | GPUTextureUsage.COPY_SRC,
    });

    const encoder = device.createCommandEncoder({
      label: 'getProvokingVertexForFlatInterpolationEitherSampling',
    });
    const pass = encoder.beginRenderPass({
      colorAttachments: [
        {
          view: texture.createView(),
          clearValue: [255, 255, 255, 255],
          loadOp: 'clear',
          storeOp: 'store',
        },
      ],
    });
    pass.setPipeline(pipeline);
    pass.draw(3);
    pass.end();

    const buffer = t.createBufferTracked({
      size: 4,
      usage: GPUBufferUsage.COPY_DST | GPUBufferUsage.MAP_READ,
    });
    encoder.copyTextureToBuffer({ texture }, { buffer }, [1, 1]);

    const commandBuffer = encoder.finish();
    device.queue.submit([commandBuffer]);

    await buffer.mapAsync(GPUMapMode.READ);
    const result = new Uint8Array(buffer.getMappedRange())[0];
    buffer.unmap();
    buffer.destroy();
    texture.destroy();

    assert(result === 0 || result === 2, `expected result to be 0 or 2, was ${result}`);
    sampling = result === 2 ? 'last' : 'first';
    s_deviceToEitherSamplingIndex.set(device, sampling);
  }
  return sampling;
}