Skip to content

Commit

Permalink
Address Comments for Timestamp example
Browse files Browse the repository at this point in the history
It was mentioned that a previous PR did the wrong thing
by possibly applying a time more than once per timer query.
This PR addresses that issue by making the TimestampQueryManager
take a callback at initialization time that is called, only when
a value is available. The value is provided to the callback.
This pattern is similar to ResizeObserver

I agree with another comment, I kind of feel like it would
be more useful to support multiple queries. But, maybe as a
sample it's fine to support just 1.
  • Loading branch information
greggman committed Dec 5, 2024
1 parent 530c33a commit 93c31f3
Show file tree
Hide file tree
Showing 2 changed files with 15 additions and 17 deletions.
10 changes: 5 additions & 5 deletions sample/timestampQuery/TimestampQueryManager.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,15 +14,15 @@ export default class TimestampQueryManager {
// A buffer to map this result back to CPU
#timestampMapBuffer: GPUBuffer;

// Last queried elapsed time of the pass in nanoseconds.
passDurationMeasurementNs: number;
// Callback to call when results are available.
#callback: (deltaTimeMs: number) => void;

// Device must have the "timestamp-query" feature
constructor(device: GPUDevice) {
constructor(device: GPUDevice, callback: (deltaTimeNs: number) => void) {
this.timestampSupported = device.features.has('timestamp-query');
if (!this.timestampSupported) return;

this.passDurationMeasurementNs = 0;
this.#callback = callback;

// Create timestamp queries
this.#timestampQuerySet = device.createQuerySet({
Expand Down Expand Up @@ -101,7 +101,7 @@ export default class TimestampQueryManager {
// It's possible elapsedNs is negative which means it's invalid
// (see spec https://gpuweb.github.io/gpuweb/#timestamp)
if (elapsedNs >= 0) {
this.passDurationMeasurementNs = elapsedNs;
this.#callback(elapsedNs);
}
buffer.unmap();
});
Expand Down
22 changes: 10 additions & 12 deletions sample/timestampQuery/main.ts
Original file line number Diff line number Diff line change
Expand Up @@ -33,7 +33,15 @@ quitIfWebGPUNotAvailable(adapter, device);
// NB: Look for 'timestampQueryManager' in this file to locate parts of this
// snippets that are related to timestamps. Most of the logic is in
// TimestampQueryManager.ts.
const timestampQueryManager = new TimestampQueryManager(device);
const timestampQueryManager = new TimestampQueryManager(device, (elapsedNs) => {
// Convert from nanoseconds to milliseconds:
const elapsedMs = Number(elapsedNs) * 1e-6;
renderPassDurationCounter.addSample(elapsedMs);
perfDisplay.innerHTML = `Render Pass duration: ${renderPassDurationCounter
.getAverage()
.toFixed(3)} ms ± ${renderPassDurationCounter.getStddev().toFixed(3)} ms`;
});

const renderPassDurationCounter = new PerfCounter();

const context = canvas.getContext('webgpu') as GPUCanvasContext;
Expand Down Expand Up @@ -209,17 +217,7 @@ function frame() {

device.queue.submit([commandEncoder.finish()]);

if (timestampQueryManager.timestampSupported) {
// Show the last successfully downloaded elapsed time.
const elapsedNs = timestampQueryManager.passDurationMeasurementNs;
// Convert from nanoseconds to milliseconds:
const elapsedMs = Number(elapsedNs) * 1e-6;
renderPassDurationCounter.addSample(elapsedMs);
perfDisplay.innerHTML = `Render Pass duration: ${renderPassDurationCounter
.getAverage()
.toFixed(3)} ms ± ${renderPassDurationCounter.getStddev().toFixed(3)} ms`;
}

// Try to download the time stamp.
timestampQueryManager.tryInitiateTimestampDownload();

requestAnimationFrame(frame);
Expand Down

0 comments on commit 93c31f3

Please sign in to comment.