mirror of
https://github.com/italicsjenga/vello.git
synced 2025-01-11 04:51:32 +11:00
875c8badf4
This is one of the stages in the new element pipeline. It's a simple one, just a prefix sum of a couple counts, and some of it will probably get merged with a downstream stage, but we'll do it separately for now for convenience. This patch also contains an update to Vulkan tools 1.2.198, which accounts for the large diff of translated shaders.
148 lines
4.6 KiB
Rust
148 lines
4.6 KiB
Rust
// Copyright 2021 The piet-gpu authors.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// https://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
//
|
|
// Also licensed under MIT license, at your choice.
|
|
|
|
//! Tests for the piet-gpu draw object stage.
|
|
|
|
use piet_gpu_hal::{BufWrite, BufferUsage};
|
|
use rand::Rng;
|
|
|
|
use crate::{Config, Runner, TestResult};
|
|
|
|
use piet_gpu::stages::{self, DrawCode, DrawMonoid, DrawStage};
|
|
|
|
const ELEMENT_SIZE: usize = 36;
|
|
|
|
const ELEMENT_FILLCOLOR: u32 = 4;
|
|
const ELEMENT_FILLLINGRADIENT: u32 = 5;
|
|
const ELEMENT_FILLIMAGE: u32 = 6;
|
|
const ELEMENT_BEGINCLIP: u32 = 9;
|
|
const ELEMENT_ENDCLIP: u32 = 10;
|
|
|
|
struct DrawTestData {
|
|
tags: Vec<u32>,
|
|
}
|
|
|
|
pub unsafe fn draw_test(runner: &mut Runner, config: &Config) -> TestResult {
|
|
let mut result = TestResult::new("draw");
|
|
let n_tag: u64 = config.size.choose(1 << 12, 1 << 20, 1 << 24);
|
|
let data = DrawTestData::new(n_tag);
|
|
let stage_config = data.get_config();
|
|
|
|
let config_buf = runner
|
|
.session
|
|
.create_buffer_init(std::slice::from_ref(&stage_config), BufferUsage::STORAGE)
|
|
.unwrap();
|
|
let scene_size = n_tag * ELEMENT_SIZE as u64;
|
|
let scene_buf = runner
|
|
.session
|
|
.create_buffer_with(scene_size, |b| data.fill_scene(b), BufferUsage::STORAGE)
|
|
.unwrap();
|
|
let memory = runner.buf_down(data.memory_size(), BufferUsage::STORAGE);
|
|
|
|
let code = DrawCode::new(&runner.session);
|
|
let stage = DrawStage::new(&runner.session, &code);
|
|
let binding = stage.bind(
|
|
&runner.session,
|
|
&code,
|
|
&config_buf,
|
|
&scene_buf,
|
|
&memory.dev_buf,
|
|
);
|
|
|
|
let mut total_elapsed = 0.0;
|
|
let n_iter = config.n_iter;
|
|
for i in 0..n_iter {
|
|
let mut commands = runner.commands();
|
|
commands.write_timestamp(0);
|
|
stage.record(&mut commands.cmd_buf, &code, &binding, n_tag);
|
|
commands.write_timestamp(1);
|
|
if i == 0 || config.verify_all {
|
|
commands.cmd_buf.memory_barrier();
|
|
commands.download(&memory);
|
|
}
|
|
total_elapsed += runner.submit(commands);
|
|
if i == 0 || config.verify_all {
|
|
let dst = memory.map_read(..);
|
|
if let Some(failure) = data.verify(&dst) {
|
|
result.fail(failure);
|
|
}
|
|
}
|
|
}
|
|
let n_elements = n_tag;
|
|
result.timing(total_elapsed, n_elements * n_iter);
|
|
|
|
result
|
|
}
|
|
|
|
impl DrawTestData {
|
|
fn new(n: u64) -> DrawTestData {
|
|
let mut rng = rand::thread_rng();
|
|
let tags = (0..n).map(|_| rng.gen_range(0, 12)).collect();
|
|
DrawTestData { tags }
|
|
}
|
|
|
|
fn get_config(&self) -> stages::Config {
|
|
let n_tags = self.tags.len();
|
|
|
|
// Layout of memory
|
|
let drawmonoid_alloc = 0;
|
|
let stage_config = stages::Config {
|
|
n_elements: n_tags as u32,
|
|
drawmonoid_alloc,
|
|
..Default::default()
|
|
};
|
|
stage_config
|
|
}
|
|
|
|
fn memory_size(&self) -> u64 {
|
|
8 + self.tags.len() as u64 * 8
|
|
}
|
|
|
|
fn fill_scene(&self, buf: &mut BufWrite) {
|
|
let mut element = [0u32; ELEMENT_SIZE / 4];
|
|
for tag in &self.tags {
|
|
element[0] = *tag;
|
|
buf.push(element);
|
|
}
|
|
}
|
|
|
|
fn verify(&self, buf: &[u8]) -> Option<String> {
|
|
let size = self.tags.len() * 8;
|
|
let actual = bytemuck::cast_slice::<u8, DrawMonoid>(&buf[8..8 + size]);
|
|
let mut expected = DrawMonoid::default();
|
|
for (i, (tag, actual)) in self.tags.iter().zip(actual).enumerate() {
|
|
// We compute an inclusive prefix sum, but for this application
|
|
// exclusive would be slightly better. We can adapt though.
|
|
let (path_ix, clip_ix) = Self::reduce_tag(*tag);
|
|
expected.path_ix += path_ix;
|
|
expected.clip_ix += clip_ix;
|
|
if *actual != expected {
|
|
return Some(format!("draw mismatch at {}", i));
|
|
}
|
|
}
|
|
None
|
|
}
|
|
|
|
fn reduce_tag(tag: u32) -> (u32, u32) {
|
|
match tag {
|
|
ELEMENT_FILLCOLOR | ELEMENT_FILLLINGRADIENT | ELEMENT_FILLIMAGE => (1, 0),
|
|
ELEMENT_BEGINCLIP => (1, 1),
|
|
ELEMENT_ENDCLIP => (0, 1),
|
|
_ => (0, 0),
|
|
}
|
|
}
|
|
}
|