2022-10-25 08:53:12 +11:00
|
|
|
//! Take an encoded scene and create a graph to render it
|
|
|
|
|
|
|
|
use bytemuck::{Pod, Zeroable};
|
|
|
|
|
|
|
|
use crate::{
|
2022-11-26 09:16:56 +11:00
|
|
|
engine::{BufProxy, ImageFormat, ImageProxy, Recording, ResourceProxy},
|
2022-11-04 13:33:11 +11:00
|
|
|
shaders::{self, FullShaders, Shaders},
|
2023-01-07 08:52:41 +11:00
|
|
|
Scene,
|
2022-10-25 08:53:12 +11:00
|
|
|
};
|
|
|
|
|
|
|
|
const TAG_MONOID_SIZE: u64 = 12;
|
2022-11-04 13:33:11 +11:00
|
|
|
const TAG_MONOID_FULL_SIZE: u64 = 20;
|
|
|
|
const PATH_BBOX_SIZE: u64 = 24;
|
2022-11-27 11:51:22 +11:00
|
|
|
const CUBIC_SIZE: u64 = 48;
|
2022-11-04 13:33:11 +11:00
|
|
|
const DRAWMONOID_SIZE: u64 = 16;
|
|
|
|
const MAX_DRAWINFO_SIZE: u64 = 44;
|
2022-11-19 09:26:26 +11:00
|
|
|
const CLIP_BIC_SIZE: u64 = 8;
|
|
|
|
const CLIP_EL_SIZE: u64 = 32;
|
2022-11-30 05:28:25 +11:00
|
|
|
const CLIP_INP_SIZE: u64 = 8;
|
2022-11-19 09:26:26 +11:00
|
|
|
const CLIP_BBOX_SIZE: u64 = 16;
|
2022-11-04 16:00:52 +11:00
|
|
|
const PATH_SIZE: u64 = 32;
|
2022-11-04 13:33:11 +11:00
|
|
|
const DRAW_BBOX_SIZE: u64 = 16;
|
|
|
|
const BUMP_SIZE: u64 = 16;
|
2022-11-04 16:00:52 +11:00
|
|
|
const BIN_HEADER_SIZE: u64 = 8;
|
2022-10-25 08:53:12 +11:00
|
|
|
|
|
|
|
#[repr(C)]
|
2022-11-04 16:00:52 +11:00
|
|
|
#[derive(Clone, Copy, Debug, Default, Zeroable, Pod)]
|
2022-10-25 08:53:12 +11:00
|
|
|
struct Config {
|
|
|
|
width_in_tiles: u32,
|
|
|
|
height_in_tiles: u32,
|
2022-11-26 09:16:56 +11:00
|
|
|
target_width: u32,
|
|
|
|
target_height: u32,
|
2022-11-02 10:20:15 +11:00
|
|
|
n_drawobj: u32,
|
2022-11-04 10:53:34 +11:00
|
|
|
n_path: u32,
|
2022-11-11 14:48:36 +11:00
|
|
|
n_clip: u32,
|
2022-11-30 12:35:19 +11:00
|
|
|
bin_data_start: u32,
|
2022-11-03 12:07:32 +11:00
|
|
|
pathtag_base: u32,
|
|
|
|
pathdata_base: u32,
|
2022-11-02 10:20:15 +11:00
|
|
|
drawtag_base: u32,
|
|
|
|
drawdata_base: u32,
|
2022-11-04 13:33:11 +11:00
|
|
|
transform_base: u32,
|
2022-11-05 06:40:54 +11:00
|
|
|
linewidth_base: u32,
|
2022-10-25 08:53:12 +11:00
|
|
|
}
|
|
|
|
|
2022-11-03 12:07:32 +11:00
|
|
|
fn size_to_words(byte_size: usize) -> u32 {
|
|
|
|
(byte_size / std::mem::size_of::<u32>()) as u32
|
|
|
|
}
|
|
|
|
|
2022-11-24 04:34:04 +11:00
|
|
|
pub const fn next_multiple_of(val: u32, rhs: u32) -> u32 {
|
|
|
|
match val % rhs {
|
|
|
|
0 => val,
|
|
|
|
r => val + (rhs - r),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-27 06:27:19 +11:00
|
|
|
#[allow(unused)]
|
2022-11-24 04:34:04 +11:00
|
|
|
fn render(scene: &Scene, shaders: &Shaders) -> (Recording, BufProxy) {
|
2022-10-25 08:53:12 +11:00
|
|
|
let mut recording = Recording::default();
|
|
|
|
let data = scene.data();
|
2023-01-07 08:52:41 +11:00
|
|
|
let n_pathtag = data.path_tags.len();
|
2022-10-25 08:53:12 +11:00
|
|
|
let pathtag_padded = align_up(n_pathtag, 4 * shaders::PATHTAG_REDUCE_WG);
|
|
|
|
let pathtag_wgs = pathtag_padded / (4 * shaders::PATHTAG_REDUCE_WG as usize);
|
2022-11-03 12:07:32 +11:00
|
|
|
let mut scene: Vec<u8> = Vec::with_capacity(pathtag_padded);
|
|
|
|
let pathtag_base = size_to_words(scene.len());
|
2023-01-07 08:52:41 +11:00
|
|
|
scene.extend(bytemuck::cast_slice(&data.path_tags));
|
2022-11-03 12:07:32 +11:00
|
|
|
scene.resize(pathtag_padded, 0);
|
|
|
|
let pathdata_base = size_to_words(scene.len());
|
2023-01-07 08:52:41 +11:00
|
|
|
scene.extend(&data.path_data);
|
2022-11-03 12:07:32 +11:00
|
|
|
|
|
|
|
let config = Config {
|
|
|
|
width_in_tiles: 64,
|
|
|
|
height_in_tiles: 64,
|
2022-11-26 09:16:56 +11:00
|
|
|
target_width: 64 * 16,
|
|
|
|
target_height: 64 * 16,
|
2022-11-03 12:07:32 +11:00
|
|
|
pathtag_base,
|
|
|
|
pathdata_base,
|
|
|
|
..Default::default()
|
|
|
|
};
|
|
|
|
let scene_buf = recording.upload(scene);
|
2022-11-30 12:23:12 +11:00
|
|
|
let config_buf = recording.upload_uniform(bytemuck::bytes_of(&config));
|
2022-11-03 12:07:32 +11:00
|
|
|
|
2022-10-25 08:53:12 +11:00
|
|
|
let reduced_buf = BufProxy::new(pathtag_wgs as u64 * TAG_MONOID_SIZE);
|
|
|
|
// TODO: really only need pathtag_wgs - 1
|
|
|
|
recording.dispatch(
|
|
|
|
shaders.pathtag_reduce,
|
|
|
|
(pathtag_wgs as u32, 1, 1),
|
2022-11-03 12:07:32 +11:00
|
|
|
[config_buf, scene_buf, reduced_buf],
|
2022-10-25 08:53:12 +11:00
|
|
|
);
|
|
|
|
|
|
|
|
let tagmonoid_buf =
|
|
|
|
BufProxy::new(pathtag_wgs as u64 * shaders::PATHTAG_REDUCE_WG as u64 * TAG_MONOID_SIZE);
|
|
|
|
recording.dispatch(
|
|
|
|
shaders.pathtag_scan,
|
|
|
|
(pathtag_wgs as u32, 1, 1),
|
2022-11-03 12:07:32 +11:00
|
|
|
[config_buf, scene_buf, reduced_buf, tagmonoid_buf],
|
2022-10-25 08:53:12 +11:00
|
|
|
);
|
|
|
|
|
2022-11-17 02:49:38 +11:00
|
|
|
let path_coarse_wgs =
|
|
|
|
(n_pathtag as u32 + shaders::PATH_COARSE_WG - 1) / shaders::PATH_COARSE_WG;
|
2022-10-25 08:53:12 +11:00
|
|
|
// TODO: more principled size calc
|
|
|
|
let tiles_buf = BufProxy::new(4097 * 8);
|
|
|
|
let segments_buf = BufProxy::new(256 * 24);
|
2022-10-26 03:03:13 +11:00
|
|
|
recording.clear_all(tiles_buf);
|
2022-10-25 08:53:12 +11:00
|
|
|
recording.dispatch(
|
|
|
|
shaders.path_coarse,
|
|
|
|
(path_coarse_wgs, 1, 1),
|
|
|
|
[
|
|
|
|
config_buf,
|
2022-11-03 12:07:32 +11:00
|
|
|
scene_buf,
|
|
|
|
tagmonoid_buf,
|
2022-10-25 08:53:12 +11:00
|
|
|
tiles_buf,
|
|
|
|
segments_buf,
|
|
|
|
],
|
|
|
|
);
|
|
|
|
recording.dispatch(
|
|
|
|
shaders.backdrop,
|
|
|
|
(config.height_in_tiles, 1, 1),
|
|
|
|
[config_buf, tiles_buf],
|
|
|
|
);
|
|
|
|
let out_buf_size = config.width_in_tiles * config.height_in_tiles * 256;
|
|
|
|
let out_buf = BufProxy::new(out_buf_size as u64);
|
|
|
|
recording.dispatch(
|
|
|
|
shaders.fine,
|
|
|
|
(config.width_in_tiles, config.height_in_tiles, 1),
|
|
|
|
[config_buf, tiles_buf, segments_buf, out_buf],
|
|
|
|
);
|
|
|
|
|
|
|
|
recording.download(out_buf);
|
|
|
|
(recording, out_buf)
|
|
|
|
}
|
|
|
|
|
2022-11-24 04:34:04 +11:00
|
|
|
pub fn render_full(
|
|
|
|
scene: &Scene,
|
|
|
|
shaders: &FullShaders,
|
2022-11-27 06:27:19 +11:00
|
|
|
width: u32,
|
|
|
|
height: u32,
|
2022-11-26 09:16:56 +11:00
|
|
|
) -> (Recording, ResourceProxy) {
|
2023-01-07 08:52:41 +11:00
|
|
|
use crate::encoding::{resource::ResourceCache, PackedEncoding};
|
2022-11-04 13:33:11 +11:00
|
|
|
let mut recording = Recording::default();
|
2023-01-07 08:52:41 +11:00
|
|
|
let mut resources = ResourceCache::new();
|
|
|
|
let mut packed_scene = PackedEncoding::default();
|
2022-11-04 13:33:11 +11:00
|
|
|
let data = scene.data();
|
2023-01-07 08:52:41 +11:00
|
|
|
packed_scene.pack(&data, &mut resources);
|
|
|
|
let (ramp_data, ramps_width, ramps_height) = resources.ramps(packed_scene.resources).unwrap();
|
|
|
|
let gradient_image = if data.patches.is_empty() {
|
2022-11-26 09:16:56 +11:00
|
|
|
ResourceProxy::new_image(1, 1, ImageFormat::Rgba8)
|
2022-11-17 02:49:38 +11:00
|
|
|
} else {
|
2023-01-07 08:52:41 +11:00
|
|
|
let data: &[u8] = bytemuck::cast_slice(ramp_data);
|
|
|
|
ResourceProxy::Image(recording.upload_image(
|
|
|
|
ramps_width,
|
|
|
|
ramps_height,
|
|
|
|
ImageFormat::Rgba8,
|
|
|
|
data,
|
|
|
|
))
|
2022-11-17 02:49:38 +11:00
|
|
|
};
|
2022-11-04 13:33:11 +11:00
|
|
|
// TODO: calculate for real when we do rectangles
|
2023-01-07 08:52:41 +11:00
|
|
|
let n_pathtag = data.path_tags.len();
|
|
|
|
let pathtag_padded = align_up(data.path_tags.len(), 4 * shaders::PATHTAG_REDUCE_WG);
|
|
|
|
let n_paths = data.n_paths;
|
|
|
|
let n_drawobj = n_paths;
|
|
|
|
let n_clip = data.n_clips;
|
2022-11-24 04:34:04 +11:00
|
|
|
|
2022-11-27 06:27:19 +11:00
|
|
|
let new_width = next_multiple_of(width, 16);
|
|
|
|
let new_height = next_multiple_of(height, 16);
|
2022-11-24 04:34:04 +11:00
|
|
|
|
2023-01-07 08:52:41 +11:00
|
|
|
let config = crate::encoding::Config {
|
2022-11-24 04:34:04 +11:00
|
|
|
width_in_tiles: new_width / 16,
|
|
|
|
height_in_tiles: new_height / 16,
|
2022-11-27 06:27:19 +11:00
|
|
|
target_width: width,
|
|
|
|
target_height: height,
|
2023-01-07 08:52:41 +11:00
|
|
|
layout: packed_scene.layout,
|
2022-11-04 13:33:11 +11:00
|
|
|
};
|
2022-11-26 09:16:56 +11:00
|
|
|
// println!("{:?}", config);
|
2023-01-07 08:52:41 +11:00
|
|
|
let scene_buf = ResourceProxy::Buf(recording.upload(packed_scene.data));
|
2022-11-30 12:23:12 +11:00
|
|
|
let config_buf = ResourceProxy::Buf(recording.upload_uniform(bytemuck::bytes_of(&config)));
|
2022-11-04 13:33:11 +11:00
|
|
|
|
2022-11-05 15:41:37 +11:00
|
|
|
let pathtag_wgs = pathtag_padded / (4 * shaders::PATHTAG_REDUCE_WG as usize);
|
2022-11-17 02:49:38 +11:00
|
|
|
let reduced_buf = ResourceProxy::new_buf(pathtag_wgs as u64 * TAG_MONOID_FULL_SIZE);
|
2022-11-04 13:33:11 +11:00
|
|
|
// TODO: really only need pathtag_wgs - 1
|
|
|
|
recording.dispatch(
|
|
|
|
shaders.pathtag_reduce,
|
|
|
|
(pathtag_wgs as u32, 1, 1),
|
|
|
|
[config_buf, scene_buf, reduced_buf],
|
|
|
|
);
|
|
|
|
|
2022-11-17 02:49:38 +11:00
|
|
|
let tagmonoid_buf = ResourceProxy::new_buf(
|
|
|
|
pathtag_wgs as u64 * shaders::PATHTAG_REDUCE_WG as u64 * TAG_MONOID_FULL_SIZE,
|
|
|
|
);
|
2022-11-04 13:33:11 +11:00
|
|
|
recording.dispatch(
|
|
|
|
shaders.pathtag_scan,
|
|
|
|
(pathtag_wgs as u32, 1, 1),
|
|
|
|
[config_buf, scene_buf, reduced_buf, tagmonoid_buf],
|
|
|
|
);
|
|
|
|
let drawobj_wgs = (n_drawobj + shaders::PATH_BBOX_WG - 1) / shaders::PATH_BBOX_WG;
|
2023-01-07 08:52:41 +11:00
|
|
|
let path_bbox_buf = ResourceProxy::new_buf(n_paths as u64 * PATH_BBOX_SIZE);
|
2022-11-04 13:33:11 +11:00
|
|
|
recording.dispatch(
|
|
|
|
shaders.bbox_clear,
|
|
|
|
(drawobj_wgs, 1, 1),
|
|
|
|
[config_buf, path_bbox_buf],
|
|
|
|
);
|
2022-11-17 02:49:38 +11:00
|
|
|
let cubic_buf = ResourceProxy::new_buf(n_pathtag as u64 * CUBIC_SIZE);
|
|
|
|
let path_coarse_wgs =
|
|
|
|
(n_pathtag as u32 + shaders::PATH_COARSE_WG - 1) / shaders::PATH_COARSE_WG;
|
2022-11-04 13:33:11 +11:00
|
|
|
recording.dispatch(
|
|
|
|
shaders.pathseg,
|
|
|
|
(path_coarse_wgs, 1, 1),
|
|
|
|
[
|
|
|
|
config_buf,
|
|
|
|
scene_buf,
|
|
|
|
tagmonoid_buf,
|
|
|
|
path_bbox_buf,
|
|
|
|
cubic_buf,
|
|
|
|
],
|
|
|
|
);
|
2022-11-17 02:49:38 +11:00
|
|
|
let draw_reduced_buf = ResourceProxy::new_buf(drawobj_wgs as u64 * DRAWMONOID_SIZE);
|
2022-11-04 13:33:11 +11:00
|
|
|
recording.dispatch(
|
|
|
|
shaders.draw_reduce,
|
|
|
|
(drawobj_wgs, 1, 1),
|
|
|
|
[config_buf, scene_buf, draw_reduced_buf],
|
|
|
|
);
|
2022-11-17 02:49:38 +11:00
|
|
|
let draw_monoid_buf = ResourceProxy::new_buf(n_drawobj as u64 * DRAWMONOID_SIZE);
|
2022-11-30 12:35:19 +11:00
|
|
|
let info_bin_data_buf = ResourceProxy::new_buf(1 << 20);
|
2023-01-07 08:52:41 +11:00
|
|
|
let clip_inp_buf = ResourceProxy::new_buf(data.n_clips as u64 * CLIP_INP_SIZE);
|
2022-11-04 13:33:11 +11:00
|
|
|
recording.dispatch(
|
|
|
|
shaders.draw_leaf,
|
|
|
|
(drawobj_wgs, 1, 1),
|
|
|
|
[
|
|
|
|
config_buf,
|
|
|
|
scene_buf,
|
|
|
|
draw_reduced_buf,
|
|
|
|
path_bbox_buf,
|
|
|
|
draw_monoid_buf,
|
2022-11-30 12:35:19 +11:00
|
|
|
info_bin_data_buf,
|
2022-11-19 09:26:26 +11:00
|
|
|
clip_inp_buf,
|
2022-11-04 13:33:11 +11:00
|
|
|
],
|
|
|
|
);
|
2023-01-07 08:52:41 +11:00
|
|
|
let clip_el_buf = ResourceProxy::new_buf(data.n_clips as u64 * CLIP_EL_SIZE);
|
2022-11-19 09:26:26 +11:00
|
|
|
let clip_bic_buf =
|
|
|
|
ResourceProxy::new_buf((n_clip / shaders::CLIP_REDUCE_WG) as u64 * CLIP_BIC_SIZE);
|
|
|
|
let clip_wg_reduce = n_clip.saturating_sub(1) / shaders::CLIP_REDUCE_WG;
|
|
|
|
if clip_wg_reduce > 0 {
|
|
|
|
recording.dispatch(
|
|
|
|
shaders.clip_reduce,
|
|
|
|
(clip_wg_reduce, 1, 1),
|
|
|
|
[
|
|
|
|
config_buf,
|
|
|
|
clip_inp_buf,
|
|
|
|
path_bbox_buf,
|
|
|
|
clip_bic_buf,
|
|
|
|
clip_el_buf,
|
|
|
|
],
|
|
|
|
);
|
|
|
|
}
|
|
|
|
let clip_wg = (n_clip + shaders::CLIP_REDUCE_WG - 1) / shaders::CLIP_REDUCE_WG;
|
|
|
|
let clip_bbox_buf = ResourceProxy::new_buf(n_clip as u64 * CLIP_BBOX_SIZE);
|
|
|
|
if clip_wg > 0 {
|
|
|
|
recording.dispatch(
|
|
|
|
shaders.clip_leaf,
|
|
|
|
(clip_wg, 1, 1),
|
|
|
|
[
|
|
|
|
config_buf,
|
|
|
|
clip_inp_buf,
|
|
|
|
path_bbox_buf,
|
|
|
|
clip_bic_buf,
|
|
|
|
clip_el_buf,
|
|
|
|
draw_monoid_buf,
|
|
|
|
clip_bbox_buf,
|
|
|
|
],
|
|
|
|
);
|
|
|
|
}
|
2023-01-07 08:52:41 +11:00
|
|
|
let draw_bbox_buf = ResourceProxy::new_buf(n_paths as u64 * DRAW_BBOX_SIZE);
|
2022-11-04 13:33:11 +11:00
|
|
|
let bump_buf = BufProxy::new(BUMP_SIZE);
|
2022-11-04 16:00:52 +11:00
|
|
|
let width_in_bins = (config.width_in_tiles + 15) / 16;
|
|
|
|
let height_in_bins = (config.height_in_tiles + 15) / 16;
|
2022-12-06 08:26:16 +11:00
|
|
|
let bin_header_buf = ResourceProxy::new_buf((256 * drawobj_wgs) as u64 * BIN_HEADER_SIZE);
|
2022-11-04 13:33:11 +11:00
|
|
|
recording.clear_all(bump_buf);
|
2022-11-17 02:49:38 +11:00
|
|
|
let bump_buf = ResourceProxy::Buf(bump_buf);
|
2022-11-04 13:33:11 +11:00
|
|
|
recording.dispatch(
|
|
|
|
shaders.binning,
|
|
|
|
(drawobj_wgs, 1, 1),
|
|
|
|
[
|
|
|
|
config_buf,
|
|
|
|
draw_monoid_buf,
|
|
|
|
path_bbox_buf,
|
|
|
|
clip_bbox_buf,
|
|
|
|
draw_bbox_buf,
|
|
|
|
bump_buf,
|
2022-11-30 12:35:19 +11:00
|
|
|
info_bin_data_buf,
|
2022-11-04 16:00:52 +11:00
|
|
|
bin_header_buf,
|
2022-11-04 13:33:11 +11:00
|
|
|
],
|
|
|
|
);
|
2022-12-06 08:26:16 +11:00
|
|
|
// Note: this only needs to be rounded up because of the workaround to store the tile_offset
|
|
|
|
// in storage rather than workgroup memory.
|
2023-01-07 08:52:41 +11:00
|
|
|
let n_path_aligned = align_up(n_paths as usize, 256);
|
2022-12-06 08:26:16 +11:00
|
|
|
let path_buf = ResourceProxy::new_buf(n_path_aligned as u64 * PATH_SIZE);
|
2022-11-17 02:49:38 +11:00
|
|
|
let tile_buf = ResourceProxy::new_buf(1 << 20);
|
2023-01-07 08:52:41 +11:00
|
|
|
let path_wgs = (n_paths + shaders::PATH_BBOX_WG - 1) / shaders::PATH_BBOX_WG;
|
2022-11-04 13:33:11 +11:00
|
|
|
recording.dispatch(
|
|
|
|
shaders.tile_alloc,
|
|
|
|
(path_wgs, 1, 1),
|
|
|
|
[
|
|
|
|
config_buf,
|
|
|
|
scene_buf,
|
|
|
|
draw_bbox_buf,
|
|
|
|
bump_buf,
|
|
|
|
path_buf,
|
|
|
|
tile_buf,
|
|
|
|
],
|
|
|
|
);
|
|
|
|
|
2022-11-17 02:49:38 +11:00
|
|
|
let segments_buf = ResourceProxy::new_buf(1 << 24);
|
2022-11-04 13:33:11 +11:00
|
|
|
recording.dispatch(
|
|
|
|
shaders.path_coarse,
|
|
|
|
(path_coarse_wgs, 1, 1),
|
|
|
|
[
|
|
|
|
config_buf,
|
|
|
|
scene_buf,
|
|
|
|
tagmonoid_buf,
|
|
|
|
cubic_buf,
|
|
|
|
path_buf,
|
|
|
|
bump_buf,
|
2022-11-04 16:00:52 +11:00
|
|
|
tile_buf,
|
2022-11-04 13:33:11 +11:00
|
|
|
segments_buf,
|
|
|
|
],
|
|
|
|
);
|
|
|
|
recording.dispatch(
|
|
|
|
shaders.backdrop,
|
|
|
|
(path_wgs, 1, 1),
|
2022-11-04 16:00:52 +11:00
|
|
|
[config_buf, path_buf, tile_buf],
|
2022-11-04 13:33:11 +11:00
|
|
|
);
|
2022-11-17 02:49:38 +11:00
|
|
|
let ptcl_buf = ResourceProxy::new_buf(1 << 24);
|
2022-11-04 16:00:52 +11:00
|
|
|
recording.dispatch(
|
|
|
|
shaders.coarse,
|
|
|
|
(width_in_bins, height_in_bins, 1),
|
|
|
|
[
|
|
|
|
config_buf,
|
|
|
|
scene_buf,
|
|
|
|
draw_monoid_buf,
|
|
|
|
bin_header_buf,
|
2022-11-30 12:35:19 +11:00
|
|
|
info_bin_data_buf,
|
2022-11-04 16:00:52 +11:00
|
|
|
path_buf,
|
|
|
|
tile_buf,
|
|
|
|
bump_buf,
|
|
|
|
ptcl_buf,
|
|
|
|
],
|
|
|
|
);
|
2022-11-27 06:27:19 +11:00
|
|
|
let out_image = ImageProxy::new(width, height, ImageFormat::Rgba8);
|
2022-11-04 13:33:11 +11:00
|
|
|
recording.dispatch(
|
|
|
|
shaders.fine,
|
|
|
|
(config.width_in_tiles, config.height_in_tiles, 1),
|
2022-11-17 02:49:38 +11:00
|
|
|
[
|
|
|
|
config_buf,
|
|
|
|
tile_buf,
|
|
|
|
segments_buf,
|
2022-11-26 09:16:56 +11:00
|
|
|
ResourceProxy::Image(out_image),
|
2022-11-17 02:49:38 +11:00
|
|
|
ptcl_buf,
|
|
|
|
gradient_image,
|
2022-11-30 12:35:19 +11:00
|
|
|
info_bin_data_buf,
|
2022-11-17 02:49:38 +11:00
|
|
|
],
|
2022-11-04 13:33:11 +11:00
|
|
|
);
|
2022-11-26 09:16:56 +11:00
|
|
|
(recording, ResourceProxy::Image(out_image))
|
2022-11-04 13:33:11 +11:00
|
|
|
}
|
|
|
|
|
2022-10-25 08:53:12 +11:00
|
|
|
pub fn align_up(len: usize, alignment: u32) -> usize {
|
|
|
|
len + (len.wrapping_neg() & alignment as usize - 1)
|
|
|
|
}
|