2026-06-23 23:27:47 +04:00
|
|
|
#![forbid(unsafe_code)]
|
|
|
|
|
#![cfg_attr(
|
|
|
|
|
test,
|
|
|
|
|
allow(
|
|
|
|
|
clippy::expect_used,
|
|
|
|
|
clippy::needless_raw_string_hashes,
|
|
|
|
|
clippy::panic,
|
|
|
|
|
clippy::unwrap_used
|
|
|
|
|
)
|
|
|
|
|
)]
|
|
|
|
|
#![allow(clippy::print_stderr, clippy::print_stdout)]
|
|
|
|
|
//! Native Vulkan smoke runner entrypoint.
|
|
|
|
|
|
2026-06-25 04:18:32 +04:00
|
|
|
use fparkan_platform_winit::{window_native_handles, WinitWindowPlan};
|
2026-06-23 23:33:42 +04:00
|
|
|
use fparkan_render_vulkan::{
|
2026-06-25 04:18:32 +04:00
|
|
|
VulkanSmokeFrameOutcome, VulkanSmokeRenderer, VulkanSmokeRendererCreateInfo,
|
2026-06-23 23:33:42 +04:00
|
|
|
};
|
2026-06-23 23:27:47 +04:00
|
|
|
use std::path::PathBuf;
|
|
|
|
|
use std::process::Command;
|
2026-06-25 04:18:32 +04:00
|
|
|
use winit::application::ApplicationHandler;
|
|
|
|
|
use winit::dpi::PhysicalSize;
|
|
|
|
|
use winit::event::WindowEvent;
|
|
|
|
|
use winit::event_loop::{ActiveEventLoop, EventLoop};
|
|
|
|
|
use winit::window::{Window, WindowId};
|
2026-06-23 23:27:47 +04:00
|
|
|
|
|
|
|
|
const SCHEMA_VERSION: &str = "fparkan-native-smoke-v1";
|
2026-06-25 04:18:32 +04:00
|
|
|
const DEFAULT_TARGET_FRAMES: u32 = 300;
|
|
|
|
|
const DEFAULT_RESIZE_FRAME: u32 = 120;
|
|
|
|
|
const DEFAULT_RESIZE_WIDTH: u32 = 960;
|
|
|
|
|
const DEFAULT_RESIZE_HEIGHT: u32 = 540;
|
2026-06-23 23:27:47 +04:00
|
|
|
|
|
|
|
|
fn main() {
|
|
|
|
|
let args = std::env::args().skip(1).collect::<Vec<_>>();
|
|
|
|
|
let code = match run(&args) {
|
|
|
|
|
Ok(output) => {
|
|
|
|
|
println!("{output}");
|
|
|
|
|
0
|
|
|
|
|
}
|
|
|
|
|
Err(err) => {
|
|
|
|
|
eprintln!("{err}");
|
|
|
|
|
2
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
std::process::exit(code);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn run(args: &[String]) -> Result<String, String> {
|
|
|
|
|
let options = SmokeOptions::parse(args)?;
|
2026-06-25 04:18:32 +04:00
|
|
|
let event_loop = EventLoop::new().map_err(|err| format!("winit event loop: {err}"))?;
|
|
|
|
|
let mut app = SmokeApp::new(options);
|
|
|
|
|
event_loop
|
|
|
|
|
.run_app(&mut app)
|
|
|
|
|
.map_err(|err| format!("winit event loop: {err}"))?;
|
|
|
|
|
app.finish()
|
2026-06-23 23:27:47 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[derive(Clone, Debug, Eq, PartialEq)]
|
|
|
|
|
struct SmokeOptions {
|
|
|
|
|
out: PathBuf,
|
|
|
|
|
frames: u32,
|
2026-06-25 04:18:32 +04:00
|
|
|
resize_frame: u32,
|
2026-06-23 23:27:47 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl SmokeOptions {
|
|
|
|
|
fn parse(args: &[String]) -> Result<Self, String> {
|
|
|
|
|
let mut out = None;
|
2026-06-25 04:18:32 +04:00
|
|
|
let mut frames = DEFAULT_TARGET_FRAMES;
|
|
|
|
|
let mut resize_frame = DEFAULT_RESIZE_FRAME;
|
2026-06-23 23:27:47 +04:00
|
|
|
let mut iter = args.iter();
|
|
|
|
|
while let Some(arg) = iter.next() {
|
|
|
|
|
match arg.as_str() {
|
|
|
|
|
"--out" => {
|
2026-06-25 04:18:32 +04:00
|
|
|
out = Some(
|
|
|
|
|
iter.next()
|
|
|
|
|
.map(PathBuf::from)
|
|
|
|
|
.ok_or_else(|| "--out requires a path".to_string())?,
|
|
|
|
|
);
|
2026-06-23 23:27:47 +04:00
|
|
|
}
|
|
|
|
|
"--frames" => {
|
2026-06-25 04:18:32 +04:00
|
|
|
frames = iter
|
2026-06-23 23:27:47 +04:00
|
|
|
.next()
|
2026-06-25 04:18:32 +04:00
|
|
|
.ok_or_else(|| "--frames requires a value".to_string())?
|
|
|
|
|
.parse()
|
|
|
|
|
.map_err(|_| "--frames must be an integer".to_string())?;
|
2026-06-23 23:38:09 +04:00
|
|
|
}
|
2026-06-25 04:18:32 +04:00
|
|
|
"--resize-frame" => {
|
|
|
|
|
resize_frame = iter
|
2026-06-23 23:27:47 +04:00
|
|
|
.next()
|
2026-06-25 04:18:32 +04:00
|
|
|
.ok_or_else(|| "--resize-frame requires a value".to_string())?
|
|
|
|
|
.parse()
|
|
|
|
|
.map_err(|_| "--resize-frame must be an integer".to_string())?;
|
2026-06-23 23:27:47 +04:00
|
|
|
}
|
|
|
|
|
_ => return Err(format!("unknown native smoke option: {arg}")),
|
|
|
|
|
}
|
|
|
|
|
}
|
2026-06-25 04:18:32 +04:00
|
|
|
let out = out.ok_or_else(|| "missing --out".to_string())?;
|
|
|
|
|
if frames < DEFAULT_TARGET_FRAMES {
|
|
|
|
|
return Err(format!(
|
|
|
|
|
"native smoke requires --frames >= {DEFAULT_TARGET_FRAMES}"
|
|
|
|
|
));
|
|
|
|
|
}
|
2026-06-23 23:27:47 +04:00
|
|
|
Ok(Self {
|
2026-06-25 04:18:32 +04:00
|
|
|
out,
|
2026-06-23 23:27:47 +04:00
|
|
|
frames,
|
2026-06-25 04:18:32 +04:00
|
|
|
resize_frame,
|
2026-06-23 23:27:47 +04:00
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-06-25 04:18:32 +04:00
|
|
|
struct SmokeApp {
|
|
|
|
|
options: SmokeOptions,
|
|
|
|
|
window_id: Option<WindowId>,
|
|
|
|
|
window: Option<Window>,
|
|
|
|
|
renderer: Option<VulkanSmokeRenderer>,
|
|
|
|
|
error: Option<String>,
|
|
|
|
|
output: Option<String>,
|
|
|
|
|
frames_presented: u32,
|
|
|
|
|
resize_count: u32,
|
|
|
|
|
resize_requested: bool,
|
|
|
|
|
last_size: Option<(u32, u32)>,
|
2026-06-23 23:42:20 +04:00
|
|
|
}
|
|
|
|
|
|
2026-06-25 04:18:32 +04:00
|
|
|
impl SmokeApp {
|
|
|
|
|
const fn new(options: SmokeOptions) -> Self {
|
|
|
|
|
Self {
|
|
|
|
|
options,
|
|
|
|
|
window_id: None,
|
|
|
|
|
window: None,
|
|
|
|
|
renderer: None,
|
|
|
|
|
error: None,
|
|
|
|
|
output: None,
|
|
|
|
|
frames_presented: 0,
|
|
|
|
|
resize_count: 0,
|
|
|
|
|
resize_requested: false,
|
|
|
|
|
last_size: None,
|
|
|
|
|
}
|
2026-06-23 23:42:20 +04:00
|
|
|
}
|
|
|
|
|
|
2026-06-25 04:18:32 +04:00
|
|
|
fn finish(self) -> Result<String, String> {
|
|
|
|
|
if let Some(error) = self.error {
|
|
|
|
|
return Err(error);
|
|
|
|
|
}
|
|
|
|
|
self.output
|
|
|
|
|
.ok_or_else(|| "native smoke exited before producing a report".to_string())
|
2026-06-23 23:42:20 +04:00
|
|
|
}
|
2026-06-24 01:53:39 +04:00
|
|
|
|
2026-06-25 04:18:32 +04:00
|
|
|
fn schedule_next_redraw(&self) {
|
|
|
|
|
if let Some(window) = self.window.as_ref() {
|
|
|
|
|
window.request_redraw();
|
2026-06-23 23:33:42 +04:00
|
|
|
}
|
2026-06-25 04:18:32 +04:00
|
|
|
}
|
2026-06-23 23:33:42 +04:00
|
|
|
|
2026-06-25 04:18:32 +04:00
|
|
|
fn complete(&mut self, event_loop: &ActiveEventLoop) {
|
|
|
|
|
let Some(renderer) = self.renderer.as_ref() else {
|
|
|
|
|
self.error = Some("native smoke renderer was not initialized".to_string());
|
|
|
|
|
event_loop.exit();
|
|
|
|
|
return;
|
2026-06-24 01:53:39 +04:00
|
|
|
};
|
2026-06-25 04:18:32 +04:00
|
|
|
let validation = renderer.validation_report();
|
|
|
|
|
if self.frames_presented < self.options.frames {
|
|
|
|
|
self.error = Some("native smoke did not reach the required frame count".to_string());
|
|
|
|
|
event_loop.exit();
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if self.resize_count == 0 || renderer.swapchain_recreate_count() == 0 {
|
|
|
|
|
self.error = Some(
|
|
|
|
|
"native smoke requires at least one measured resize and swapchain recreation"
|
|
|
|
|
.to_string(),
|
|
|
|
|
);
|
|
|
|
|
event_loop.exit();
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if validation.warning_count != 0 || validation.error_count != 0 {
|
|
|
|
|
self.error = Some(format!(
|
|
|
|
|
"native smoke validation must stay clean (warnings={}, errors={})",
|
|
|
|
|
validation.warning_count, validation.error_count
|
|
|
|
|
));
|
|
|
|
|
event_loop.exit();
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
let report = render_smoke_report_json(
|
|
|
|
|
&self.options,
|
|
|
|
|
renderer,
|
|
|
|
|
self.frames_presented,
|
|
|
|
|
self.resize_count,
|
|
|
|
|
validation.warning_count,
|
|
|
|
|
validation.error_count,
|
|
|
|
|
&validation.vuids,
|
|
|
|
|
);
|
|
|
|
|
if let Some(parent) = self.options.out.parent() {
|
|
|
|
|
if let Err(err) = std::fs::create_dir_all(parent) {
|
|
|
|
|
self.error = Some(format!("{}: {err}", parent.display()));
|
|
|
|
|
event_loop.exit();
|
|
|
|
|
return;
|
2026-06-24 01:53:39 +04:00
|
|
|
}
|
|
|
|
|
}
|
2026-06-25 04:18:32 +04:00
|
|
|
if let Err(err) = std::fs::write(&self.options.out, &report) {
|
|
|
|
|
self.error = Some(format!("{}: {err}", self.options.out.display()));
|
|
|
|
|
event_loop.exit();
|
|
|
|
|
return;
|
2026-06-23 23:56:40 +04:00
|
|
|
}
|
2026-06-25 04:18:32 +04:00
|
|
|
self.output = Some(report);
|
|
|
|
|
event_loop.exit();
|
2026-06-23 23:56:40 +04:00
|
|
|
}
|
|
|
|
|
|
2026-06-25 04:18:32 +04:00
|
|
|
fn request_controlled_resize(&mut self) {
|
|
|
|
|
if self.resize_requested {
|
|
|
|
|
return;
|
2026-06-23 23:56:40 +04:00
|
|
|
}
|
2026-06-25 04:18:32 +04:00
|
|
|
let Some(window) = self.window.as_ref() else {
|
|
|
|
|
return;
|
|
|
|
|
};
|
|
|
|
|
self.resize_requested = true;
|
|
|
|
|
let requested = PhysicalSize::new(DEFAULT_RESIZE_WIDTH, DEFAULT_RESIZE_HEIGHT);
|
|
|
|
|
let _ = window.request_inner_size(requested);
|
2026-06-23 23:56:40 +04:00
|
|
|
}
|
2026-06-25 04:18:32 +04:00
|
|
|
}
|
2026-06-23 23:35:41 +04:00
|
|
|
|
2026-06-25 04:18:32 +04:00
|
|
|
impl ApplicationHandler for SmokeApp {
|
|
|
|
|
fn resumed(&mut self, event_loop: &ActiveEventLoop) {
|
|
|
|
|
if self.window.is_some() {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
let plan = match WinitWindowPlan::smoke().validate() {
|
|
|
|
|
Ok(plan) => plan,
|
|
|
|
|
Err(err) => {
|
|
|
|
|
self.error = Some(err.to_string());
|
|
|
|
|
event_loop.exit();
|
|
|
|
|
return;
|
2026-06-23 23:56:40 +04:00
|
|
|
}
|
2026-06-25 04:18:32 +04:00
|
|
|
};
|
|
|
|
|
let attributes = Window::default_attributes()
|
|
|
|
|
.with_title("FParkan Vulkan smoke")
|
|
|
|
|
.with_inner_size(PhysicalSize::new(plan.width, plan.height));
|
|
|
|
|
let window = match event_loop.create_window(attributes) {
|
|
|
|
|
Ok(window) => window,
|
|
|
|
|
Err(err) => {
|
|
|
|
|
self.error = Some(format!("winit window: {err}"));
|
|
|
|
|
event_loop.exit();
|
|
|
|
|
return;
|
2026-06-23 23:42:20 +04:00
|
|
|
}
|
2026-06-25 04:18:32 +04:00
|
|
|
};
|
|
|
|
|
let Some(native_handles) = window_native_handles(&window) else {
|
|
|
|
|
self.error = Some("winit window does not expose native handles".to_string());
|
|
|
|
|
event_loop.exit();
|
|
|
|
|
return;
|
|
|
|
|
};
|
|
|
|
|
let size = window.inner_size();
|
|
|
|
|
let renderer = match VulkanSmokeRenderer::new(&VulkanSmokeRendererCreateInfo {
|
|
|
|
|
application_name: "fparkan-vulkan-smoke".to_string(),
|
|
|
|
|
native_handles,
|
|
|
|
|
drawable_extent: (size.width.max(1), size.height.max(1)),
|
|
|
|
|
enable_validation: true,
|
|
|
|
|
}) {
|
|
|
|
|
Ok(renderer) => renderer,
|
|
|
|
|
Err(err) => {
|
|
|
|
|
self.error = Some(err.to_string());
|
|
|
|
|
event_loop.exit();
|
|
|
|
|
return;
|
2026-06-23 23:35:41 +04:00
|
|
|
}
|
2026-06-25 04:18:32 +04:00
|
|
|
};
|
|
|
|
|
self.last_size = Some((size.width, size.height));
|
|
|
|
|
self.window_id = Some(window.id());
|
|
|
|
|
self.renderer = Some(renderer);
|
|
|
|
|
self.window = Some(window);
|
|
|
|
|
self.schedule_next_redraw();
|
2026-06-23 23:56:40 +04:00
|
|
|
}
|
|
|
|
|
|
2026-06-25 04:18:32 +04:00
|
|
|
fn window_event(
|
2026-06-23 23:56:40 +04:00
|
|
|
&mut self,
|
2026-06-25 04:18:32 +04:00
|
|
|
event_loop: &ActiveEventLoop,
|
|
|
|
|
window_id: WindowId,
|
|
|
|
|
event: WindowEvent,
|
|
|
|
|
) {
|
|
|
|
|
if Some(window_id) != self.window_id {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
match event {
|
|
|
|
|
WindowEvent::CloseRequested => {
|
|
|
|
|
if self.output.is_none() {
|
|
|
|
|
self.error = Some("native smoke window closed before completion".to_string());
|
|
|
|
|
}
|
|
|
|
|
event_loop.exit();
|
|
|
|
|
}
|
|
|
|
|
WindowEvent::Resized(size) => {
|
|
|
|
|
if self
|
|
|
|
|
.last_size
|
|
|
|
|
.is_some_and(|last| last != (size.width, size.height))
|
|
|
|
|
{
|
|
|
|
|
self.resize_count = self.resize_count.saturating_add(1);
|
2026-06-23 23:38:09 +04:00
|
|
|
}
|
2026-06-25 04:18:32 +04:00
|
|
|
self.last_size = Some((size.width, size.height));
|
|
|
|
|
if let Some(renderer) = self.renderer.as_mut() {
|
|
|
|
|
renderer.request_resize((size.width, size.height));
|
2026-06-23 23:38:09 +04:00
|
|
|
}
|
|
|
|
|
}
|
2026-06-25 04:18:32 +04:00
|
|
|
WindowEvent::RedrawRequested => {
|
|
|
|
|
let Some(renderer) = self.renderer.as_mut() else {
|
|
|
|
|
self.error = Some("native smoke renderer was not initialized".to_string());
|
|
|
|
|
event_loop.exit();
|
|
|
|
|
return;
|
|
|
|
|
};
|
|
|
|
|
match renderer.draw_frame() {
|
|
|
|
|
Ok(VulkanSmokeFrameOutcome::Presented) => {
|
|
|
|
|
self.frames_presented = self.frames_presented.saturating_add(1);
|
|
|
|
|
}
|
|
|
|
|
Ok(
|
|
|
|
|
VulkanSmokeFrameOutcome::Recreated | VulkanSmokeFrameOutcome::ZeroExtent,
|
|
|
|
|
) => {}
|
|
|
|
|
Err(err) => {
|
|
|
|
|
self.error = Some(err.to_string());
|
|
|
|
|
event_loop.exit();
|
|
|
|
|
return;
|
|
|
|
|
}
|
2026-06-24 01:53:39 +04:00
|
|
|
}
|
2026-06-25 04:18:32 +04:00
|
|
|
let recreate_count = renderer.swapchain_recreate_count();
|
|
|
|
|
let should_request_resize =
|
|
|
|
|
!self.resize_requested && self.frames_presented >= self.options.resize_frame;
|
|
|
|
|
let should_complete = self.frames_presented >= self.options.frames
|
|
|
|
|
&& self.resize_count > 0
|
|
|
|
|
&& recreate_count > 0;
|
|
|
|
|
let _ = renderer;
|
|
|
|
|
if should_request_resize {
|
|
|
|
|
self.request_controlled_resize();
|
|
|
|
|
}
|
|
|
|
|
if should_complete {
|
|
|
|
|
self.complete(event_loop);
|
|
|
|
|
} else {
|
|
|
|
|
self.schedule_next_redraw();
|
2026-06-24 01:05:31 +04:00
|
|
|
}
|
2026-06-24 00:05:46 +04:00
|
|
|
}
|
2026-06-25 04:18:32 +04:00
|
|
|
_ => {}
|
2026-06-24 00:05:46 +04:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-06-25 04:18:32 +04:00
|
|
|
fn about_to_wait(&mut self, _event_loop: &ActiveEventLoop) {
|
|
|
|
|
if self.output.is_none() && self.error.is_none() {
|
|
|
|
|
self.schedule_next_redraw();
|
2026-06-23 23:27:47 +04:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-06-23 23:33:42 +04:00
|
|
|
fn render_smoke_report_json(
|
|
|
|
|
options: &SmokeOptions,
|
2026-06-25 04:18:32 +04:00
|
|
|
renderer: &VulkanSmokeRenderer,
|
|
|
|
|
frames_presented: u32,
|
|
|
|
|
resize_count: u32,
|
|
|
|
|
validation_warning_count: u32,
|
|
|
|
|
validation_error_count: u32,
|
|
|
|
|
validation_vuids: &[String],
|
|
|
|
|
) -> String {
|
|
|
|
|
let report = renderer.report();
|
|
|
|
|
let fields = vec![
|
2026-06-24 00:05:46 +04:00
|
|
|
("schema_version", json_string(SCHEMA_VERSION)),
|
|
|
|
|
("commit_sha", json_string(¤t_git_commit_sha())),
|
2026-06-25 04:18:32 +04:00
|
|
|
("rust_toolchain", json_string(¤t_rustc_release())),
|
2026-06-24 00:05:46 +04:00
|
|
|
("target_triple", json_string(¤t_rustc_host_triple())),
|
2026-06-25 04:18:32 +04:00
|
|
|
("platform", json_string(actual_platform())),
|
|
|
|
|
("status", json_string("passed")),
|
|
|
|
|
("frames", frames_presented.to_string()),
|
|
|
|
|
("resize_count", resize_count.to_string()),
|
2026-06-24 00:05:46 +04:00
|
|
|
(
|
|
|
|
|
"swapchain_recreate_count",
|
2026-06-25 04:18:32 +04:00
|
|
|
renderer.swapchain_recreate_count().to_string(),
|
2026-06-24 00:05:46 +04:00
|
|
|
),
|
|
|
|
|
(
|
2026-06-25 04:18:32 +04:00
|
|
|
"validation_warning_count",
|
|
|
|
|
validation_warning_count.to_string(),
|
2026-06-24 00:05:46 +04:00
|
|
|
),
|
2026-06-25 04:18:32 +04:00
|
|
|
("validation_error_count", validation_error_count.to_string()),
|
|
|
|
|
("validation_vuids", render_string_array(validation_vuids)),
|
|
|
|
|
("requested_frames", options.frames.to_string()),
|
2026-06-24 00:05:46 +04:00
|
|
|
(
|
2026-06-25 04:18:32 +04:00
|
|
|
"shader_manifest_hash",
|
|
|
|
|
json_string("849ffae9681f5ff2fc145d7b98f19f69b478d9ea73207efdf5f1748e8d51045c"),
|
2026-06-24 00:14:26 +04:00
|
|
|
),
|
2026-06-25 04:18:32 +04:00
|
|
|
("vulkan_loader_status", json_string("available")),
|
|
|
|
|
("vulkan_instance_status", json_string("created")),
|
|
|
|
|
("window_status", json_string("created")),
|
|
|
|
|
("vulkan_surface_status", json_string("created")),
|
|
|
|
|
("vulkan_device_status", json_string("selected")),
|
|
|
|
|
("vulkan_device_name", json_string(&report.device_name)),
|
|
|
|
|
("vulkan_logical_device_status", json_string("created")),
|
2026-06-24 00:14:26 +04:00
|
|
|
(
|
|
|
|
|
"vulkan_logical_device_graphics_queue_family",
|
2026-06-25 04:18:32 +04:00
|
|
|
report.graphics_queue_family.to_string(),
|
2026-06-24 00:14:26 +04:00
|
|
|
),
|
|
|
|
|
(
|
|
|
|
|
"vulkan_logical_device_present_queue_family",
|
2026-06-25 04:18:32 +04:00
|
|
|
report.present_queue_family.to_string(),
|
2026-06-24 00:14:26 +04:00
|
|
|
),
|
|
|
|
|
(
|
|
|
|
|
"vulkan_logical_device_enabled_extension_count",
|
2026-06-25 04:18:32 +04:00
|
|
|
report.enabled_extension_count.to_string(),
|
2026-06-24 00:05:46 +04:00
|
|
|
),
|
2026-06-25 04:18:32 +04:00
|
|
|
("vulkan_swapchain_status", json_string("created")),
|
2026-06-24 00:05:46 +04:00
|
|
|
(
|
|
|
|
|
"vulkan_swapchain_width",
|
2026-06-25 04:18:32 +04:00
|
|
|
report.swapchain_extent.0.to_string(),
|
2026-06-24 00:05:46 +04:00
|
|
|
),
|
|
|
|
|
(
|
|
|
|
|
"vulkan_swapchain_height",
|
2026-06-25 04:18:32 +04:00
|
|
|
report.swapchain_extent.1.to_string(),
|
2026-06-24 00:05:46 +04:00
|
|
|
),
|
|
|
|
|
(
|
|
|
|
|
"vulkan_swapchain_image_count",
|
2026-06-25 04:18:32 +04:00
|
|
|
report.swapchain_image_count.to_string(),
|
2026-06-24 00:05:46 +04:00
|
|
|
),
|
|
|
|
|
(
|
2026-06-25 04:18:32 +04:00
|
|
|
"vulkan_portability_enumeration",
|
|
|
|
|
bool_json(report.portability_enumeration),
|
2026-06-24 00:05:46 +04:00
|
|
|
),
|
2026-06-25 04:18:32 +04:00
|
|
|
];
|
|
|
|
|
render_json_object(&fields)
|
2026-06-24 00:05:46 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn render_json_object(fields: &[(&str, String)]) -> String {
|
|
|
|
|
let mut out = String::from("{\n");
|
|
|
|
|
for (index, (name, value)) in fields.iter().enumerate() {
|
|
|
|
|
out.push_str(" ");
|
|
|
|
|
out.push_str(&json_string(name));
|
|
|
|
|
out.push_str(": ");
|
|
|
|
|
out.push_str(value);
|
|
|
|
|
if index + 1 < fields.len() {
|
|
|
|
|
out.push(',');
|
|
|
|
|
}
|
|
|
|
|
out.push('\n');
|
|
|
|
|
}
|
|
|
|
|
out.push_str("}\n");
|
|
|
|
|
out
|
|
|
|
|
}
|
|
|
|
|
|
2026-06-25 04:18:32 +04:00
|
|
|
fn render_string_array(values: &[String]) -> String {
|
|
|
|
|
let items = values
|
|
|
|
|
.iter()
|
|
|
|
|
.map(|value| json_string(value))
|
|
|
|
|
.collect::<Vec<_>>()
|
|
|
|
|
.join(", ");
|
|
|
|
|
format!("[{items}]")
|
2026-06-23 23:27:47 +04:00
|
|
|
}
|
|
|
|
|
|
2026-06-25 04:18:32 +04:00
|
|
|
fn actual_platform() -> &'static str {
|
|
|
|
|
match std::env::consts::OS {
|
|
|
|
|
"macos" => "macos",
|
|
|
|
|
"linux" => "linux",
|
|
|
|
|
"windows" => "windows",
|
|
|
|
|
other => other,
|
|
|
|
|
}
|
2026-06-23 23:33:42 +04:00
|
|
|
}
|
|
|
|
|
|
2026-06-23 23:27:47 +04:00
|
|
|
fn current_git_commit_sha() -> String {
|
|
|
|
|
Command::new("git")
|
|
|
|
|
.args(["rev-parse", "HEAD"])
|
|
|
|
|
.output()
|
|
|
|
|
.ok()
|
|
|
|
|
.filter(|output| output.status.success())
|
|
|
|
|
.and_then(|output| String::from_utf8(output.stdout).ok())
|
|
|
|
|
.map(|value| value.trim().to_string())
|
|
|
|
|
.filter(|value| value.len() == 40 && value.chars().all(|ch| ch.is_ascii_hexdigit()))
|
|
|
|
|
.unwrap_or_else(|| "unknown".to_string())
|
|
|
|
|
}
|
|
|
|
|
|
2026-06-25 04:18:32 +04:00
|
|
|
fn current_rustc_release() -> String {
|
|
|
|
|
Command::new("rustc")
|
|
|
|
|
.arg("-Vv")
|
|
|
|
|
.output()
|
|
|
|
|
.ok()
|
|
|
|
|
.filter(|output| output.status.success())
|
|
|
|
|
.and_then(|output| String::from_utf8(output.stdout).ok())
|
|
|
|
|
.and_then(|output| {
|
|
|
|
|
output
|
|
|
|
|
.lines()
|
|
|
|
|
.find_map(|line| line.strip_prefix("release: ").map(ToString::to_string))
|
|
|
|
|
})
|
|
|
|
|
.unwrap_or_else(|| "unknown".to_string())
|
|
|
|
|
}
|
|
|
|
|
|
2026-06-23 23:48:31 +04:00
|
|
|
fn current_rustc_host_triple() -> String {
|
|
|
|
|
Command::new("rustc")
|
2026-06-25 04:18:32 +04:00
|
|
|
.arg("-Vv")
|
2026-06-23 23:48:31 +04:00
|
|
|
.output()
|
|
|
|
|
.ok()
|
|
|
|
|
.filter(|output| output.status.success())
|
|
|
|
|
.and_then(|output| String::from_utf8(output.stdout).ok())
|
|
|
|
|
.and_then(|output| {
|
|
|
|
|
output
|
|
|
|
|
.lines()
|
|
|
|
|
.find_map(|line| line.strip_prefix("host: ").map(ToString::to_string))
|
|
|
|
|
})
|
|
|
|
|
.unwrap_or_else(|| "unknown".to_string())
|
|
|
|
|
}
|
|
|
|
|
|
2026-06-23 23:27:47 +04:00
|
|
|
fn json_string(value: &str) -> String {
|
2026-06-25 04:18:32 +04:00
|
|
|
let mut out = String::with_capacity(value.len() + 2);
|
|
|
|
|
out.push('"');
|
2026-06-23 23:27:47 +04:00
|
|
|
for ch in value.chars() {
|
|
|
|
|
match ch {
|
|
|
|
|
'"' => out.push_str("\\\""),
|
|
|
|
|
'\\' => out.push_str("\\\\"),
|
|
|
|
|
'\n' => out.push_str("\\n"),
|
|
|
|
|
'\r' => out.push_str("\\r"),
|
|
|
|
|
'\t' => out.push_str("\\t"),
|
2026-06-25 04:18:32 +04:00
|
|
|
c if c.is_control() => {
|
2026-06-23 23:27:47 +04:00
|
|
|
use std::fmt::Write as _;
|
2026-06-25 04:18:32 +04:00
|
|
|
let _ = write!(out, "\\u{:04x}", c as u32);
|
2026-06-23 23:27:47 +04:00
|
|
|
}
|
2026-06-25 04:18:32 +04:00
|
|
|
c => out.push(c),
|
2026-06-23 23:27:47 +04:00
|
|
|
}
|
|
|
|
|
}
|
2026-06-25 04:18:32 +04:00
|
|
|
out.push('"');
|
2026-06-23 23:27:47 +04:00
|
|
|
out
|
|
|
|
|
}
|
|
|
|
|
|
2026-06-25 04:18:32 +04:00
|
|
|
fn bool_json(value: bool) -> String {
|
|
|
|
|
if value { "true" } else { "false" }.to_string()
|
|
|
|
|
}
|
|
|
|
|
|
2026-06-23 23:27:47 +04:00
|
|
|
#[cfg(test)]
|
|
|
|
|
mod tests {
|
|
|
|
|
use super::*;
|
|
|
|
|
|
2026-06-23 23:59:07 +04:00
|
|
|
#[test]
|
2026-06-25 04:18:32 +04:00
|
|
|
fn parses_required_args() {
|
|
|
|
|
let parsed = SmokeOptions::parse(&["--out".to_string(), "target/report.json".to_string()]);
|
2026-06-23 23:59:07 +04:00
|
|
|
|
|
|
|
|
assert_eq!(
|
2026-06-25 04:18:32 +04:00
|
|
|
parsed,
|
|
|
|
|
Ok(SmokeOptions {
|
|
|
|
|
out: PathBuf::from("target/report.json"),
|
|
|
|
|
frames: DEFAULT_TARGET_FRAMES,
|
|
|
|
|
resize_frame: DEFAULT_RESIZE_FRAME,
|
|
|
|
|
})
|
2026-06-24 00:05:46 +04:00
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
2026-06-25 04:18:32 +04:00
|
|
|
fn rejects_too_few_frames() {
|
|
|
|
|
let parsed = SmokeOptions::parse(&[
|
|
|
|
|
"--out".to_string(),
|
|
|
|
|
"target/report.json".to_string(),
|
|
|
|
|
"--frames".to_string(),
|
|
|
|
|
"299".to_string(),
|
|
|
|
|
]);
|
2026-06-24 00:05:46 +04:00
|
|
|
|
|
|
|
|
assert_eq!(
|
2026-06-25 04:18:32 +04:00
|
|
|
parsed,
|
|
|
|
|
Err("native smoke requires --frames >= 300".to_string())
|
2026-06-24 00:05:46 +04:00
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2026-06-24 00:14:26 +04:00
|
|
|
#[test]
|
2026-06-25 04:18:32 +04:00
|
|
|
fn renders_string_array_json() {
|
2026-06-24 00:14:26 +04:00
|
|
|
assert_eq!(
|
2026-06-25 04:18:32 +04:00
|
|
|
render_string_array(&["VUID-A".to_string(), "VUID-B".to_string()]),
|
|
|
|
|
"[\"VUID-A\", \"VUID-B\"]"
|
2026-06-24 00:14:26 +04:00
|
|
|
);
|
|
|
|
|
}
|
2026-06-23 23:27:47 +04:00
|
|
|
}
|