-
Notifications
You must be signed in to change notification settings - Fork 142
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add frame statistics UI to with_winit example #288
Conversation
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Nothing blocking, some thoughts on potential future work.
examples/with_winit/src/stats.rs
Outdated
impl<'a> Drop for FrameScope<'a> { | ||
fn drop(&mut self) { | ||
self.stats | ||
.add_sample(self.start.elapsed().as_micros() as u64); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
To ensure that this is measuring time between the same point, rather than rendering time, I think we need to inline the call to elapsed
, store the new Instant
, and use that as the start time in frame_scope
. At the moment, we ignore input processing time, for example, which I don't think is right.
Actually working out the best measure of FPS has always eluded me.
We also should measure GPU time (e.g. using https://github.com/Wumpf/wgpu-profiler), but wiring that up is non-trivial.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We also should measure GPU time (e.g. using https://github.com/Wumpf/wgpu-profiler), but wiring that up is non-trivial.
Oh, the GPU time measurement looks neat. I haven't looked into wgpu-profiler
very closely but supporting GPU timers out of the box would be great (so far I've had to rely on external tools, such as the Metal GPU Performance monitor). I'll investigate this.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I made the elapsed time measurement tighter so it should closely match the presentation time. I haven't really had time to investigate wgpu-profiler yet but I plan to spend some time on wiring it up soon.
A couple other thoughts; we should record startup time (ideally both time spent in wgpu |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looks good, just a few thoughts on cross-platform support.
I haven't properly reviewed the code for the frame time graph, but it seems to work OK.
We can get overflow in cases of C being pressed, but that's unavoidable.
I'd also like to get some way to enable this on Android in (without plugging in a USB keyboard, which might just work), but I'm not sure what that would be.
examples/with_winit/src/lib.rs
Outdated
if vsync_on { | ||
wgpu::PresentMode::Fifo | ||
} else { | ||
wgpu::PresentMode::Immediate |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
wgpu::PresentMode::Immediate | |
wgpu::PresentMode::AutoNoVsync |
Should be supported on all platforms. There's also a question of doing PresentMode::AutoVsync
for vsync. I don't have a machine which it would make a difference on; apparently it's only Amd Vulkan.
The alternative might also resolve #266
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Done, I changed the code to always use the Auto*
modes. On web this likely means that "vsync off" will be incorrect so maybe the stats UI shouldn't just blindly display "VSync: off", but it's probably not a huge deal.
examples/with_winit/src/lib.rs
Outdated
stats.add_sample(stats::Sample { | ||
frame_time_us: frame_start_time.elapsed().as_micros() as u64, | ||
}); | ||
frame_start_time = Instant::now(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
stats.add_sample(stats::Sample { | |
frame_time_us: frame_start_time.elapsed().as_micros() as u64, | |
}); | |
frame_start_time = Instant::now(); | |
let new_time = Instant::now(); | |
stats.add_sample(stats::Sample { | |
frame_time_us: (new_time - frame_start_time).as_micros() as u64, | |
}); | |
frame_start_time = new_time; |
Should be slightly more correct. In practise, it probably doesn't matter
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Done.
I made the stats display on by default and I can see it on my Android device. Does it show up on your device? For controls to hide it I thought of adding a "double tap" gesture or recognizing taps over some visible rect which could serve as a button. |
Ah yes, I hadn't realised the consequences of that; I hadn't tried it on my device. I'm not certain it being displayed by default is what we want, but I don't have a strong objection to it either. |
Also there is currently no way to clear min/max on mobile, so we'll want some UI for it. I think I can make some simple "double-tap" or some other gesture could work, maybe in another PR. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Code mostly looks good to me, although I'm not convinced by the non-linearity.
So to confirm, the current status is:
- Add frame statistics UI to with_winit example #288 (comment) (mobile buttons) can be done in a different PR
- Add frame statistics UI to with_winit example #288 (comment) (shader startup time) should be reasonably straightforward, but isn't critical
- Add frame statistics UI to with_winit example #288 (comment)
wgpu_profiler
- doesn't need to be this PR - Add frame statistics UI to with_winit example #288 (comment) (Colour coding) - would be nice to get in this PR - we should also have dashed lines for (at least) 16.6ms and 33.3 ms. It would also be good to use accessible colours here, rather than red and green, although the colour coding is not functional.
examples/with_winit/src/stats.rs
Outdated
// shrink the draw size of the overall average sample, so scale the size non-linearly to | ||
// emphasize smaller samples. | ||
let h = (*sample as f64) * 0.001 / self.frame_time_max_ms; | ||
let s = Affine::scale_non_uniform(1., -h.sqrt()); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This logic is non-obvious to me; I suspect a maximum height based on the maximum seen last render might be better. I'd much rather keep things linear.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I switched this to linear now. I thought using sqrt here added value initially but now that we can reset the max/min frame times with a keypress (which resets the scale) this type of visualization is not necessary.
I'll follow up with the dashed lines for 16.6ms/33.3ms as you suggested, which should provide a good relative visual anchor.
Added a module for frame time statistics and UI layer that displays the average, minimum, and maximum frame time alongside FPS. The UI can be toggled by pressing the `S` key.
Also set the stats layer toggle to be on by default until we add some UI to toggle it on mobile.
Also abandoned the FrameScope idea and revised the `Stats::add_sample` to accept a struct to accept a variety of future measurements.
We now track the frame time from snapshot to snapshot corresponding to the exact presentation time.
…e start time This does not account for the time spent in processing `Stats::add_sample` but it should be very close.
…mode The Auto* modes should have wider compatibility as they implement fallback behavior based on what the platform supports. This also means that on web "vsync off" will likely be incorrect.
The sqrt scale doesn't add much value any more since the max frame time can be reset with a keypress.
Also added color coding for the bar graph based on these thresholds.
Working on these in follow up PRs sounds good to me.
I added color coding and lines and picked accessible colors for the coding. I added markers for 8.33, 16.66, and 30.33. I used a solid stroke instead of a dashed stroke as the dashed stroke style didn't work. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm happy for this to land as is. I've added some more thoughts on Zulip about the scale, so they might be worth considering. We might discuss tomorrow
examples/with_winit/src/stats.rs
Outdated
left_margin_padding, | ||
(1 + labels.len()) as f64 * text_height, | ||
)) * s, | ||
if *sample < 16667 { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think a match expression should also work, but this is fine.
match *sample {
..16_667 =>...,
..33_334 => ...,
.. => ...
}
I think that should probably be in its own variable declaration if we used it
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Done.
left_margin, | ||
(2. - y) * graph_max_height + thres_text_height_2, | ||
)), | ||
&format!("{}", t), |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Can we right-justify this? I presume that would be a job for the SimpleText layer to handle?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think that could be neat, though it's a minor visual improvement given how small the label is rendered. @dfrg Is this straightforward to do this today with the existing code?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We don’t currently do any sort of alignment but we could probably hack something in.
This allows the graph to display at a reasonable scale in the face of fluctuations and a max recorded sample that is much larger than the current average.
Added a module for frame time statistics and a UI layer that displays the average, minimum, and maximum frame time alongside FPS. The UI can be toggled by pressing the
S
key.