2020-05-04 17:05:54 +02:00
|
|
|
mod pico_svg;
|
2020-05-11 20:01:06 -07:00
|
|
|
mod render_ctx;
|
2020-05-04 17:05:54 +02:00
|
|
|
|
|
|
|
pub use render_ctx::PietGpuRenderContext;
|
|
|
|
|
|
|
|
use rand::{Rng, RngCore};
|
|
|
|
|
|
|
|
use piet::kurbo::{BezPath, Circle, Line, Point, Vec2};
|
|
|
|
use piet::{Color, RenderContext};
|
|
|
|
|
2020-05-11 20:01:06 -07:00
|
|
|
use piet_gpu_types::encoder::Encode;
|
|
|
|
|
2020-05-04 17:05:54 +02:00
|
|
|
use piet_gpu_hal::{CmdBuf, Device, Error, ImageLayout, MemFlags};
|
|
|
|
|
|
|
|
use pico_svg::PicoSvg;
|
|
|
|
|
|
|
|
pub const WIDTH: usize = TILE_W * WIDTH_IN_TILES;
|
|
|
|
pub const HEIGHT: usize = TILE_H * HEIGHT_IN_TILES;
|
|
|
|
|
|
|
|
const TILE_W: usize = 16;
|
|
|
|
const TILE_H: usize = 16;
|
|
|
|
|
|
|
|
const WIDTH_IN_TILEGROUPS: usize = 4;
|
|
|
|
const HEIGHT_IN_TILEGROUPS: usize = 96;
|
|
|
|
const TILEGROUP_STRIDE: usize = 2048;
|
|
|
|
|
|
|
|
const WIDTH_IN_TILES: usize = 128;
|
|
|
|
const HEIGHT_IN_TILES: usize = 96;
|
|
|
|
const PTCL_INITIAL_ALLOC: usize = 1024;
|
|
|
|
|
|
|
|
const K2_PER_TILE_SIZE: usize = 8;
|
|
|
|
|
2020-05-15 12:28:29 -07:00
|
|
|
const N_CIRCLES: usize = 0;
|
2020-05-04 17:05:54 +02:00
|
|
|
|
2020-05-12 13:38:26 -07:00
|
|
|
const N_WG: u32 = 16;
|
|
|
|
|
2020-05-04 17:05:54 +02:00
|
|
|
pub fn render_scene(rc: &mut impl RenderContext) {
|
|
|
|
let mut rng = rand::thread_rng();
|
|
|
|
for _ in 0..N_CIRCLES {
|
|
|
|
let color = Color::from_rgba32_u32(rng.next_u32());
|
|
|
|
let center = Point::new(
|
|
|
|
rng.gen_range(0.0, WIDTH as f64),
|
|
|
|
rng.gen_range(0.0, HEIGHT as f64),
|
|
|
|
);
|
|
|
|
let radius = rng.gen_range(0.0, 50.0);
|
|
|
|
let circle = Circle::new(center, radius);
|
|
|
|
rc.fill(circle, &color);
|
|
|
|
}
|
2020-05-20 16:36:09 -07:00
|
|
|
/*
|
2020-05-25 09:08:21 -07:00
|
|
|
let mut path = BezPath::new();
|
2020-05-04 17:05:54 +02:00
|
|
|
path.move_to((100.0, 1150.0));
|
|
|
|
path.line_to((200.0, 1200.0));
|
|
|
|
path.line_to((150.0, 1250.0));
|
|
|
|
path.close_path();
|
|
|
|
rc.fill(path, &Color::rgb8(128, 0, 128));
|
2020-05-15 12:28:29 -07:00
|
|
|
*/
|
2020-05-04 17:05:54 +02:00
|
|
|
rc.stroke(
|
|
|
|
Line::new((100.0, 100.0), (200.0, 150.0)),
|
|
|
|
&Color::WHITE,
|
|
|
|
5.0,
|
|
|
|
);
|
2020-05-25 09:08:21 -07:00
|
|
|
//render_cardioid(rc);
|
|
|
|
render_tiger(rc);
|
2020-05-04 17:05:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[allow(unused)]
|
|
|
|
fn render_cardioid(rc: &mut impl RenderContext) {
|
2020-05-15 20:57:07 -07:00
|
|
|
let n = 601;
|
2020-05-04 17:05:54 +02:00
|
|
|
let dth = std::f64::consts::PI * 2.0 / (n as f64);
|
|
|
|
let center = Point::new(1024.0, 768.0);
|
|
|
|
let r = 750.0;
|
|
|
|
let mut path = BezPath::new();
|
|
|
|
for i in 1..n {
|
|
|
|
let p0 = center + Vec2::from_angle(i as f64 * dth) * r;
|
|
|
|
let p1 = center + Vec2::from_angle(((i * 2) % n) as f64 * dth) * r;
|
2020-05-15 20:57:07 -07:00
|
|
|
//rc.fill(&Circle::new(p0, 8.0), &Color::WHITE);
|
2020-05-04 17:05:54 +02:00
|
|
|
path.move_to(p0);
|
|
|
|
path.line_to(p1);
|
|
|
|
//rc.stroke(Line::new(p0, p1), &Color::BLACK, 2.0);
|
|
|
|
}
|
|
|
|
rc.stroke(&path, &Color::BLACK, 2.0);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn render_tiger(rc: &mut impl RenderContext) {
|
|
|
|
let xml_str = std::str::from_utf8(include_bytes!("../Ghostscript_Tiger.svg")).unwrap();
|
|
|
|
let start = std::time::Instant::now();
|
|
|
|
let svg = PicoSvg::load(xml_str, 8.0).unwrap();
|
|
|
|
println!("parsing time: {:?}", start.elapsed());
|
|
|
|
|
|
|
|
let start = std::time::Instant::now();
|
|
|
|
svg.render(rc);
|
|
|
|
println!("flattening and encoding time: {:?}", start.elapsed());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[allow(unused)]
|
|
|
|
fn dump_scene(buf: &[u8]) {
|
|
|
|
for i in 0..(buf.len() / 4) {
|
|
|
|
let mut buf_u32 = [0u8; 4];
|
|
|
|
buf_u32.copy_from_slice(&buf[i * 4..i * 4 + 4]);
|
|
|
|
println!("{:4x}: {:8x}", i * 4, u32::from_le_bytes(buf_u32));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[allow(unused)]
|
2020-05-12 13:38:26 -07:00
|
|
|
pub fn dump_k1_data(k1_buf: &[u32]) {
|
2020-05-04 17:05:54 +02:00
|
|
|
for i in 0..k1_buf.len() {
|
|
|
|
if k1_buf[i] != 0 {
|
2020-05-12 13:38:26 -07:00
|
|
|
println!("{:4x}: {:8x}", i * 4, k1_buf[i]);
|
2020-05-04 17:05:54 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct Renderer<D: Device> {
|
|
|
|
pub image_dev: D::Image, // resulting image
|
|
|
|
|
|
|
|
scene_buf: D::Buffer,
|
|
|
|
scene_dev: D::Buffer,
|
|
|
|
|
2020-05-11 20:01:06 -07:00
|
|
|
pub state_buf: D::Buffer,
|
2020-05-12 10:53:54 -07:00
|
|
|
pub anno_buf: D::Buffer,
|
2020-05-12 13:38:26 -07:00
|
|
|
pub bin_buf: D::Buffer,
|
2020-05-13 15:35:19 -07:00
|
|
|
pub ptcl_buf: D::Buffer,
|
2020-05-11 20:01:06 -07:00
|
|
|
|
|
|
|
el_pipeline: D::Pipeline,
|
|
|
|
el_ds: D::DescriptorSet,
|
|
|
|
|
2020-05-12 13:38:26 -07:00
|
|
|
bin_pipeline: D::Pipeline,
|
|
|
|
bin_ds: D::DescriptorSet,
|
|
|
|
|
|
|
|
bin_alloc_buf_host: D::Buffer,
|
|
|
|
bin_alloc_buf_dev: D::Buffer,
|
|
|
|
|
2020-05-13 15:35:19 -07:00
|
|
|
coarse_pipeline: D::Pipeline,
|
|
|
|
coarse_ds: D::DescriptorSet,
|
|
|
|
|
|
|
|
coarse_alloc_buf_host: D::Buffer,
|
|
|
|
coarse_alloc_buf_dev: D::Buffer,
|
|
|
|
|
2020-05-04 17:05:54 +02:00
|
|
|
k4_pipeline: D::Pipeline,
|
|
|
|
k4_ds: D::DescriptorSet,
|
2020-05-15 12:28:29 -07:00
|
|
|
|
2020-05-11 20:01:06 -07:00
|
|
|
n_elements: usize,
|
2020-05-04 17:05:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<D: Device> Renderer<D> {
|
|
|
|
pub unsafe fn new(device: &D, scene: &[u8]) -> Result<Self, Error> {
|
|
|
|
let host = MemFlags::host_coherent();
|
|
|
|
let dev = MemFlags::device_local();
|
|
|
|
|
2020-05-12 13:38:26 -07:00
|
|
|
let n_elements = scene.len() / piet_gpu_types::scene::Element::fixed_size();
|
|
|
|
println!("scene: {} elements", n_elements);
|
|
|
|
|
2020-05-04 17:05:54 +02:00
|
|
|
let scene_buf = device
|
|
|
|
.create_buffer(std::mem::size_of_val(&scene[..]) as u64, host)
|
|
|
|
.unwrap();
|
|
|
|
let scene_dev = device
|
|
|
|
.create_buffer(std::mem::size_of_val(&scene[..]) as u64, dev)
|
|
|
|
.unwrap();
|
|
|
|
device.write_buffer(&scene_buf, &scene)?;
|
|
|
|
|
2020-05-15 20:09:39 -07:00
|
|
|
let state_buf = device.create_buffer(1 * 1024 * 1024, dev)?;
|
2020-05-12 10:53:54 -07:00
|
|
|
let anno_buf = device.create_buffer(64 * 1024 * 1024, dev)?;
|
2020-05-29 20:06:16 -07:00
|
|
|
let bin_buf = device.create_buffer(64 * 1024 * 1024, host)?;
|
2020-05-13 15:35:19 -07:00
|
|
|
let ptcl_buf = device.create_buffer(48 * 1024 * 1024, dev)?;
|
2020-05-11 20:01:06 -07:00
|
|
|
let image_dev = device.create_image2d(WIDTH as u32, HEIGHT as u32, dev)?;
|
|
|
|
|
|
|
|
let el_code = include_bytes!("../shader/elements.spv");
|
2020-05-12 10:53:54 -07:00
|
|
|
let el_pipeline = device.create_simple_compute_pipeline(el_code, 3, 0)?;
|
2020-05-11 20:01:06 -07:00
|
|
|
let el_ds = device.create_descriptor_set(
|
|
|
|
&el_pipeline,
|
2020-05-12 10:53:54 -07:00
|
|
|
&[&scene_dev, &state_buf, &anno_buf],
|
2020-05-11 20:01:06 -07:00
|
|
|
&[],
|
|
|
|
)?;
|
|
|
|
|
2020-05-12 13:38:26 -07:00
|
|
|
let bin_alloc_buf_host = device.create_buffer(12, host)?;
|
|
|
|
let bin_alloc_buf_dev = device.create_buffer(12, dev)?;
|
|
|
|
|
|
|
|
// TODO: constants
|
2020-05-29 20:06:16 -07:00
|
|
|
let bin_alloc_start = ((n_elements + 255) & !255) * 8;
|
2020-05-12 13:38:26 -07:00
|
|
|
device
|
|
|
|
.write_buffer(&bin_alloc_buf_host, &[
|
|
|
|
n_elements as u32,
|
|
|
|
0,
|
2020-05-29 20:06:16 -07:00
|
|
|
bin_alloc_start as u32,
|
2020-05-12 13:38:26 -07:00
|
|
|
])
|
|
|
|
?;
|
|
|
|
let bin_code = include_bytes!("../shader/binning.spv");
|
2020-05-19 08:21:09 -07:00
|
|
|
let bin_pipeline = device.create_simple_compute_pipeline(bin_code, 4, 0)?;
|
2020-05-12 13:38:26 -07:00
|
|
|
let bin_ds = device.create_descriptor_set(
|
|
|
|
&bin_pipeline,
|
2020-05-19 08:21:09 -07:00
|
|
|
&[&anno_buf, &state_buf, &bin_alloc_buf_dev, &bin_buf],
|
2020-05-12 13:38:26 -07:00
|
|
|
&[],
|
|
|
|
)?;
|
2020-05-11 20:01:06 -07:00
|
|
|
|
2020-05-13 15:35:19 -07:00
|
|
|
let coarse_alloc_buf_host = device.create_buffer(4, host)?;
|
|
|
|
let coarse_alloc_buf_dev = device.create_buffer(4, dev)?;
|
|
|
|
|
2020-05-15 12:28:29 -07:00
|
|
|
let coarse_alloc_start = WIDTH_IN_TILES * HEIGHT_IN_TILES * PTCL_INITIAL_ALLOC;
|
2020-05-13 15:35:19 -07:00
|
|
|
device
|
|
|
|
.write_buffer(&coarse_alloc_buf_host, &[
|
2020-05-15 12:28:29 -07:00
|
|
|
coarse_alloc_start as u32,
|
2020-05-13 15:35:19 -07:00
|
|
|
])
|
|
|
|
?;
|
|
|
|
let coarse_code = include_bytes!("../shader/coarse.spv");
|
|
|
|
let coarse_pipeline = device.create_simple_compute_pipeline(coarse_code, 4, 0)?;
|
|
|
|
let coarse_ds = device.create_descriptor_set(
|
|
|
|
&coarse_pipeline,
|
|
|
|
&[&anno_buf, &bin_buf, &coarse_alloc_buf_dev, &ptcl_buf],
|
|
|
|
&[],
|
|
|
|
)?;
|
|
|
|
|
2020-05-04 17:05:54 +02:00
|
|
|
let k4_code = include_bytes!("../shader/kernel4.spv");
|
2020-05-15 15:20:25 -07:00
|
|
|
let k4_pipeline = device.create_simple_compute_pipeline(k4_code, 1, 1)?;
|
2020-05-11 20:01:06 -07:00
|
|
|
let k4_ds = device.create_descriptor_set(
|
|
|
|
&k4_pipeline,
|
2020-05-15 15:20:25 -07:00
|
|
|
&[&ptcl_buf],
|
2020-05-11 20:01:06 -07:00
|
|
|
&[&image_dev],
|
|
|
|
)?;
|
2020-05-04 17:05:54 +02:00
|
|
|
|
|
|
|
Ok(Renderer {
|
|
|
|
scene_buf,
|
|
|
|
scene_dev,
|
|
|
|
image_dev,
|
2020-05-11 20:01:06 -07:00
|
|
|
el_pipeline,
|
|
|
|
el_ds,
|
2020-05-12 13:38:26 -07:00
|
|
|
bin_pipeline,
|
|
|
|
bin_ds,
|
2020-05-13 15:35:19 -07:00
|
|
|
coarse_pipeline,
|
|
|
|
coarse_ds,
|
2020-05-15 12:28:29 -07:00
|
|
|
k4_pipeline,
|
|
|
|
k4_ds,
|
2020-05-11 20:01:06 -07:00
|
|
|
state_buf,
|
2020-05-12 10:53:54 -07:00
|
|
|
anno_buf,
|
2020-05-12 13:38:26 -07:00
|
|
|
bin_buf,
|
2020-05-13 15:35:19 -07:00
|
|
|
ptcl_buf,
|
2020-05-12 13:38:26 -07:00
|
|
|
bin_alloc_buf_host,
|
|
|
|
bin_alloc_buf_dev,
|
2020-05-13 15:35:19 -07:00
|
|
|
coarse_alloc_buf_host,
|
|
|
|
coarse_alloc_buf_dev,
|
2020-05-11 20:01:06 -07:00
|
|
|
n_elements,
|
2020-05-04 17:05:54 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
pub unsafe fn record(&self, cmd_buf: &mut impl CmdBuf<D>, query_pool: &D::QueryPool) {
|
|
|
|
cmd_buf.copy_buffer(&self.scene_buf, &self.scene_dev);
|
2020-05-12 13:38:26 -07:00
|
|
|
cmd_buf.copy_buffer(&self.bin_alloc_buf_host, &self.bin_alloc_buf_dev);
|
2020-05-13 15:35:19 -07:00
|
|
|
cmd_buf.copy_buffer(&self.coarse_alloc_buf_host, &self.coarse_alloc_buf_dev);
|
2020-05-15 20:09:39 -07:00
|
|
|
cmd_buf.clear_buffer(&self.state_buf);
|
2020-05-04 17:05:54 +02:00
|
|
|
cmd_buf.memory_barrier();
|
2020-05-11 20:01:06 -07:00
|
|
|
cmd_buf.image_barrier(
|
|
|
|
&self.image_dev,
|
|
|
|
ImageLayout::Undefined,
|
|
|
|
ImageLayout::General,
|
|
|
|
);
|
2020-05-04 17:05:54 +02:00
|
|
|
cmd_buf.reset_query_pool(&query_pool);
|
|
|
|
cmd_buf.write_timestamp(&query_pool, 0);
|
|
|
|
cmd_buf.dispatch(
|
2020-05-11 20:01:06 -07:00
|
|
|
&self.el_pipeline,
|
|
|
|
&self.el_ds,
|
2020-05-15 12:28:29 -07:00
|
|
|
(((self.n_elements + 127) / 128) as u32, 1, 1),
|
2020-05-04 17:05:54 +02:00
|
|
|
);
|
|
|
|
cmd_buf.write_timestamp(&query_pool, 1);
|
|
|
|
cmd_buf.memory_barrier();
|
2020-05-12 13:38:26 -07:00
|
|
|
cmd_buf.dispatch(
|
|
|
|
&self.bin_pipeline,
|
|
|
|
&self.bin_ds,
|
|
|
|
(N_WG, 1, 1),
|
|
|
|
);
|
|
|
|
cmd_buf.write_timestamp(&query_pool, 2);
|
|
|
|
cmd_buf.memory_barrier();
|
2020-05-29 20:06:16 -07:00
|
|
|
/*
|
2020-05-13 15:35:19 -07:00
|
|
|
cmd_buf.dispatch(
|
|
|
|
&self.coarse_pipeline,
|
|
|
|
&self.coarse_ds,
|
|
|
|
(WIDTH as u32 / 256, HEIGHT as u32 / 256, 1),
|
|
|
|
);
|
2020-05-29 20:06:16 -07:00
|
|
|
*/
|
2020-05-13 15:35:19 -07:00
|
|
|
cmd_buf.write_timestamp(&query_pool, 3);
|
|
|
|
cmd_buf.memory_barrier();
|
2020-05-29 20:06:16 -07:00
|
|
|
/*
|
2020-05-15 12:28:29 -07:00
|
|
|
cmd_buf.dispatch(
|
|
|
|
&self.k4_pipeline,
|
|
|
|
&self.k4_ds,
|
|
|
|
((WIDTH / TILE_W) as u32, (HEIGHT / TILE_H) as u32, 1),
|
|
|
|
);
|
2020-05-29 20:06:16 -07:00
|
|
|
*/
|
2020-05-15 12:28:29 -07:00
|
|
|
cmd_buf.write_timestamp(&query_pool, 4);
|
|
|
|
cmd_buf.memory_barrier();
|
2020-05-04 17:05:54 +02:00
|
|
|
cmd_buf.image_barrier(&self.image_dev, ImageLayout::General, ImageLayout::BlitSrc);
|
|
|
|
}
|
|
|
|
}
|