diff --git a/Cargo.toml b/Cargo.toml index 550674b..1d8c700 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -46,6 +46,7 @@ bytemuck = { version = "1.12.1", features = ["derive"] } smallvec = "1.8.0" moscato = { git = "https://github.com/dfrg/pinot", rev = "59db153" } peniko = { git = "https://github.com/linebender/peniko", rev = "cafdac9a211a0fb2fec5656bd663d1ac770bcc81" } +guillotiere = "0.6.2" [workspace.dependencies] wgpu = "0.15" diff --git a/examples/assets/piet-logo.png b/examples/assets/piet-logo.png new file mode 100644 index 0000000..1f5a48f Binary files /dev/null and b/examples/assets/piet-logo.png differ diff --git a/examples/headless/src/main.rs b/examples/headless/src/main.rs index de8d961..0111909 100644 --- a/examples/headless/src/main.rs +++ b/examples/headless/src/main.rs @@ -6,7 +6,7 @@ use std::{ use anyhow::{anyhow, bail, Context, Result}; use clap::{CommandFactory, Parser}; -use scenes::{SceneParams, SceneSet, SimpleText}; +use scenes::{ImageCache, SceneParams, SceneSet, SimpleText}; use vello::{ block_on_wgpu, kurbo::{Affine, Vec2}, @@ -97,9 +97,11 @@ async fn render(mut scenes: SceneSet, index: usize, args: &Args) -> Result<()> { let mut builder = SceneBuilder::for_fragment(&mut fragment); let example_scene = &mut scenes.scenes[index]; let mut text = SimpleText::new(); + let mut images = ImageCache::new(); let mut scene_params = SceneParams { time: args.time.unwrap_or(0.), text: &mut text, + images: &mut images, resolution: None, base_color: None, }; diff --git a/examples/scenes/Cargo.toml b/examples/scenes/Cargo.toml index 9a21a5b..7037d28 100644 --- a/examples/scenes/Cargo.toml +++ b/examples/scenes/Cargo.toml @@ -15,6 +15,7 @@ vello = { path = "../../" } vello_svg = { path = "../../integrations/vello_svg" } anyhow = { workspace = true } clap = { workspace = true, features = ["derive"] } +image = "0.24.5" # Used for the `download` command byte-unit = "4.0" diff --git a/examples/scenes/src/images.rs b/examples/scenes/src/images.rs new file mode 100644 index 0000000..2086351 --- /dev/null +++ b/examples/scenes/src/images.rs @@ -0,0 +1,53 @@ +use std::collections::HashMap; +use std::path::{Path, PathBuf}; +use std::sync::Arc; + +use vello::peniko::{Blob, Format, Image}; + +/// Simple hack to support loading images for examples. +#[derive(Default)] +pub struct ImageCache { + files: HashMap, + bytes: HashMap, +} + +impl ImageCache { + pub fn new() -> Self { + Self::default() + } + + pub fn from_file(&mut self, path: impl AsRef) -> Option { + let path = path.as_ref(); + if let Some(image) = self.files.get(path) { + Some(image.clone()) + } else { + let data = std::fs::read(path).ok()?; + let image = decode_image(&data)?; + self.files.insert(path.to_owned(), image.clone()); + Some(image) + } + } + + pub fn from_bytes(&mut self, key: usize, bytes: &[u8]) -> Option { + if let Some(image) = self.bytes.get(&key) { + Some(image.clone()) + } else { + let image = decode_image(bytes)?; + self.bytes.insert(key, image.clone()); + Some(image) + } + } +} + +fn decode_image(data: &[u8]) -> Option { + let image = image::io::Reader::new(std::io::Cursor::new(data)) + .with_guessed_format() + .ok()? + .decode() + .ok()?; + let width = image.width(); + let height = image.height(); + let data = Arc::new(image.into_rgba8().into_vec()); + let blob = Blob::new(data); + Some(Image::new(blob, Format::Rgba8, width, height)) +} diff --git a/examples/scenes/src/lib.rs b/examples/scenes/src/lib.rs index 242cbe1..a1bf160 100644 --- a/examples/scenes/src/lib.rs +++ b/examples/scenes/src/lib.rs @@ -1,4 +1,5 @@ pub mod download; +mod images; mod simple_text; mod svg; mod test_scenes; @@ -7,6 +8,7 @@ use std::path::PathBuf; use anyhow::{anyhow, Result}; use clap::{Args, Subcommand}; use download::Download; +pub use images::ImageCache; pub use simple_text::SimpleText; pub use svg::{default_scene, scene_from_files}; pub use test_scenes::test_scenes; @@ -16,6 +18,7 @@ use vello::{kurbo::Vec2, peniko::Color, SceneBuilder}; pub struct SceneParams<'a> { pub time: f64, pub text: &'a mut SimpleText, + pub images: &'a mut ImageCache, pub resolution: Option, pub base_color: Option, } diff --git a/examples/scenes/src/test_scenes.rs b/examples/scenes/src/test_scenes.rs index 499e2a4..721b921 100644 --- a/examples/scenes/src/test_scenes.rs +++ b/examples/scenes/src/test_scenes.rs @@ -1,8 +1,12 @@ +use std::sync::Arc; + use crate::{ExampleScene, SceneConfig, SceneParams, SceneSet}; -use vello::kurbo::{Affine, BezPath, Ellipse, PathEl, Point, Rect}; +use vello::kurbo::{Affine, BezPath, Ellipse, PathEl, Point, Rect, Vec2}; use vello::peniko::*; use vello::*; +const PIET_LOGO_IMAGE: &[u8] = include_bytes!("../../assets/piet-logo.png"); + macro_rules! scene { ($name: ident) => { scene!($name: false) @@ -97,6 +101,11 @@ fn cardioid_and_friends(sb: &mut SceneBuilder, _: &mut SceneParams) { } fn animated_text(sb: &mut SceneBuilder, params: &mut SceneParams) { + let piet_logo = params + .images + .from_bytes(PIET_LOGO_IMAGE.as_ptr() as usize, PIET_LOGO_IMAGE) + .unwrap(); + use PathEl::*; let rect = Rect::from_origin_size(Point::new(0.0, 0.0), (1000.0, 1000.0)); let star = [ @@ -184,6 +193,10 @@ fn animated_text(sb: &mut SceneBuilder, params: &mut SceneParams) { None, &star, ); + sb.draw_image( + &piet_logo, + Affine::translate((550.0, 250.0)) * Affine::skew(-20f64.to_radians().tan(), 0.0), + ); } fn brush_transform(sb: &mut SceneBuilder, params: &mut SceneParams) { diff --git a/examples/with_winit/src/lib.rs b/examples/with_winit/src/lib.rs index 9ed9598..c7d8d4f 100644 --- a/examples/with_winit/src/lib.rs +++ b/examples/with_winit/src/lib.rs @@ -19,7 +19,7 @@ use std::time::Instant; use anyhow::Result; use clap::{CommandFactory, Parser}; -use scenes::{SceneParams, SceneSet, SimpleText}; +use scenes::{ImageCache, SceneParams, SceneSet, SimpleText}; use vello::peniko::Color; use vello::util::RenderSurface; use vello::{ @@ -96,6 +96,7 @@ fn run( let mut scene = Scene::new(); let mut fragment = SceneFragment::new(); let mut simple_text = SimpleText::new(); + let mut images = ImageCache::new(); let start = Instant::now(); let mut touch_state = multi_touch::TouchState::new(); @@ -264,6 +265,7 @@ fn run( let mut scene_params = SceneParams { time: start.elapsed().as_secs_f64(), text: &mut simple_text, + images: &mut images, resolution: None, base_color: None, }; diff --git a/shader/coarse.wgsl b/shader/coarse.wgsl index 0963c29..6eced0d 100644 --- a/shader/coarse.wgsl +++ b/shader/coarse.wgsl @@ -126,6 +126,15 @@ fn write_grad(ty: u32, index: u32, info_offset: u32) { cmd_offset += 3u; } +fn write_image(xy: u32, width_height: u32, info_offset: u32) { + alloc_cmd(4u); + ptcl[cmd_offset] = CMD_IMAGE; + ptcl[cmd_offset + 1u] = info_offset; + ptcl[cmd_offset + 2u] = xy; + ptcl[cmd_offset + 3u] = width_height; + cmd_offset += 4u; +} + fn write_begin_clip() { alloc_cmd(1u); ptcl[cmd_offset] = CMD_BEGIN_CLIP; @@ -377,6 +386,15 @@ fn main( write_grad(CMD_RAD_GRAD, index, info_offset); } } + // DRAWTAG_FILL_IMAGE + case 0x1c8u: { + let linewidth = bitcast(info_bin_data[di]); + if write_path(tile, linewidth) { + let xy = scene[dd]; + let width_height = scene[dd + 1u]; + write_image(xy, width_height, di + 1u); + } + } // DRAWTAG_BEGIN_CLIP case 0x9u: { if tile.segments == 0u && tile.backdrop == 0 { diff --git a/shader/draw_leaf.wgsl b/shader/draw_leaf.wgsl index 9183c95..d0873de 100644 --- a/shader/draw_leaf.wgsl +++ b/shader/draw_leaf.wgsl @@ -113,7 +113,9 @@ fn main( var matrx: vec4; var translate: vec2; var linewidth = bbox.linewidth; - if linewidth >= 0.0 || tag_word == DRAWTAG_FILL_LIN_GRADIENT || tag_word == DRAWTAG_FILL_RAD_GRADIENT { + if linewidth >= 0.0 || tag_word == DRAWTAG_FILL_LIN_GRADIENT || tag_word == DRAWTAG_FILL_RAD_GRADIENT || + tag_word == DRAWTAG_FILL_IMAGE + { let transform = read_transform(config.transform_base, bbox.trans_ix); matrx = transform.matrx; translate = transform.translate; @@ -124,7 +126,7 @@ fn main( } switch tag_word { // DRAWTAG_FILL_COLOR, DRAWTAG_FILL_IMAGE - case 0x44u, 0x48u: { + case 0x44u: { info[di] = bitcast(linewidth); } // DRAWTAG_FILL_LIN_GRADIENT @@ -169,6 +171,19 @@ fn main( info[di + 9u] = bitcast(ra); info[di + 10u] = bitcast(roff); } + // DRAWTAG_FILL_IMAGE + case 0x1c8u: { + info[di] = bitcast(linewidth); + let inv_det = 1.0 / (matrx.x * matrx.w - matrx.y * matrx.z); + let inv_mat = inv_det * vec4(matrx.w, -matrx.y, -matrx.z, matrx.x); + let inv_tr = mat2x2(inv_mat.xy, inv_mat.zw) * -translate; + info[di + 1u] = bitcast(inv_mat.x); + info[di + 2u] = bitcast(inv_mat.y); + info[di + 3u] = bitcast(inv_mat.z); + info[di + 4u] = bitcast(inv_mat.w); + info[di + 5u] = bitcast(inv_tr.x); + info[di + 6u] = bitcast(inv_tr.y); + } default: {} } } diff --git a/shader/fine.wgsl b/shader/fine.wgsl index 6851bae..38e6b96 100644 --- a/shader/fine.wgsl +++ b/shader/fine.wgsl @@ -40,6 +40,9 @@ var gradients: texture_2d; @group(0) @binding(6) var info: array; +@group(0) @binding(7) +var image_atlas: texture_2d; + fn read_fill(cmd_ix: u32) -> CmdFill { let tile = ptcl[cmd_ix + 1u]; let backdrop = i32(ptcl[cmd_ix + 2u]); @@ -81,6 +84,24 @@ fn read_rad_grad(cmd_ix: u32) -> CmdRadGrad { return CmdRadGrad(index, matrx, xlat, c1, ra, roff); } +fn read_image(cmd_ix: u32) -> CmdImage { + let info_offset = ptcl[cmd_ix + 1u]; + let xy = ptcl[cmd_ix + 2u]; + let width_height = ptcl[cmd_ix + 3u]; + let m0 = bitcast(info[info_offset]); + let m1 = bitcast(info[info_offset + 1u]); + let m2 = bitcast(info[info_offset + 2u]); + let m3 = bitcast(info[info_offset + 3u]); + let matrx = vec4(m0, m1, m2, m3); + let xlat = vec2(bitcast(info[info_offset + 4u]), bitcast(info[info_offset + 5u])); + // The following are not intended to be bitcasts + let x = f32(xy >> 16u); + let y = f32(xy & 0xffffu); + let width = f32(width_height >> 16u); + let height = f32(width_height & 0xffffu); + return CmdImage(matrx, xlat, vec2(x, y), vec2(width, height)); +} + fn read_end_clip(cmd_ix: u32) -> CmdEndClip { let blend = ptcl[cmd_ix + 1u]; let alpha = bitcast(ptcl[cmd_ix + 2u]); @@ -265,6 +286,29 @@ fn main( } cmd_ix += 3u; } + // CMD_IMAGE + case 8u: { + let image = read_image(cmd_ix); + let atlas_extents = image.atlas_offset + image.extents; + for (var i = 0u; i < PIXELS_PER_THREAD; i += 1u) { + let my_xy = vec2(xy.x + f32(i), xy.y); + let atlas_uv = image.matrx.xy * my_xy.x + image.matrx.zw * my_xy.y + image.xlat + image.atlas_offset; + // This currently clips to the image bounds. TODO: extend modes + if all(atlas_uv < atlas_extents) { + let uv_quad = vec4(max(floor(atlas_uv), image.atlas_offset), min(ceil(atlas_uv), atlas_extents)); + let uv_frac = fract(atlas_uv); + let a = textureLoad(image_atlas, vec2(uv_quad.xy), 0); + let b = textureLoad(image_atlas, vec2(uv_quad.xw), 0); + let c = textureLoad(image_atlas, vec2(uv_quad.zy), 0); + let d = textureLoad(image_atlas, vec2(uv_quad.zw), 0); + let color = mix(mix(a, b, uv_frac.y), mix(c, d, uv_frac.y), uv_frac.x); + let fg_rgba = vec4(color.rgb * color.a, color.a); + let fg_i = fg_rgba * area[i]; + rgba[i] = rgba[i] * (1.0 - fg_i.a) + fg_i; + } + } + cmd_ix += 4u; + } // CMD_BEGIN_CLIP case 9u: { if clip_depth < BLEND_STACK_SPLIT { diff --git a/shader/shared/drawtag.wgsl b/shader/shared/drawtag.wgsl index 432cf1b..d2646b8 100644 --- a/shader/shared/drawtag.wgsl +++ b/shader/shared/drawtag.wgsl @@ -19,7 +19,7 @@ let DRAWTAG_NOP = 0u; let DRAWTAG_FILL_COLOR = 0x44u; let DRAWTAG_FILL_LIN_GRADIENT = 0x114u; let DRAWTAG_FILL_RAD_GRADIENT = 0x2dcu; -let DRAWTAG_FILL_IMAGE = 0x48u; +let DRAWTAG_FILL_IMAGE = 0x1c8u; let DRAWTAG_BEGIN_CLIP = 0x9u; let DRAWTAG_END_CLIP = 0x21u; diff --git a/shader/shared/ptcl.wgsl b/shader/shared/ptcl.wgsl index 3527bd0..5d5e528 100644 --- a/shader/shared/ptcl.wgsl +++ b/shader/shared/ptcl.wgsl @@ -16,6 +16,7 @@ let CMD_SOLID = 3u; let CMD_COLOR = 5u; let CMD_LIN_GRAD = 6u; let CMD_RAD_GRAD = 7u; +let CMD_IMAGE = 8u; let CMD_BEGIN_CLIP = 9u; let CMD_END_CLIP = 10u; let CMD_JUMP = 11u; @@ -57,6 +58,13 @@ struct CmdRadGrad { roff: f32, } +struct CmdImage { + matrx: vec4, + xlat: vec2, + atlas_offset: vec2, + extents: vec2, +} + struct CmdEndClip { blend: u32, alpha: f32, diff --git a/src/encoding.rs b/src/encoding.rs index 239bd54..2925256 100644 --- a/src/encoding.rs +++ b/src/encoding.rs @@ -20,6 +20,7 @@ mod draw; mod encoding; mod glyph; mod glyph_cache; +mod image_cache; mod math; mod monoid; mod path; diff --git a/src/encoding/draw.rs b/src/encoding/draw.rs index 1ddaead..fb3aa1f 100644 --- a/src/encoding/draw.rs +++ b/src/encoding/draw.rs @@ -38,7 +38,7 @@ impl DrawTag { pub const RADIAL_GRADIENT: Self = Self(0x2dc); /// Image fill. - pub const IMAGE: Self = Self(0x48); + pub const IMAGE: Self = Self(0x1c8); /// Begin layer/clip. pub const BEGIN_CLIP: Self = Self(0x9); @@ -104,10 +104,10 @@ pub struct DrawRadialGradient { #[derive(Clone, Copy, Debug, Default, Zeroable, Pod)] #[repr(C)] pub struct DrawImage { - /// Image index. - pub index: u32, - /// Packed image offset. - pub offset: u32, + /// Packed atlas coordinates. + pub xy: u32, + /// Packed image dimensions. + pub width_height: u32, } /// Draw data for a clip or layer. diff --git a/src/encoding/encoding.rs b/src/encoding/encoding.rs index 0bee364..d212a89 100644 --- a/src/encoding/encoding.rs +++ b/src/encoding/encoding.rs @@ -14,12 +14,14 @@ // // Also licensed under MIT license, at your choice. +use crate::encoding::DrawImage; + use super::{ resolve::Patch, DrawColor, DrawLinearGradient, DrawRadialGradient, DrawTag, Glyph, GlyphRun, PathEncoder, PathTag, Transform, }; -use peniko::{kurbo::Shape, BlendMode, BrushRef, ColorStop, Extend, GradientKind}; +use peniko::{kurbo::Shape, BlendMode, BrushRef, ColorStop, Extend, GradientKind, Image}; /// Encoded data streams for a scene. #[derive(Clone, Default)] @@ -122,16 +124,26 @@ impl Encoding { self.n_open_clips += other.n_open_clips; self.patches .extend(other.patches.iter().map(|patch| match patch { - Patch::Ramp { offset, stops } => { + Patch::Ramp { + draw_data_offset: offset, + stops, + } => { let stops = stops.start + stops_base..stops.end + stops_base; Patch::Ramp { - offset: offset + offsets.draw_data, + draw_data_offset: offset + offsets.draw_data, stops, } } Patch::GlyphRun { index } => Patch::GlyphRun { index: index + glyph_runs_base, }, + Patch::Image { + image, + draw_data_offset, + } => Patch::Image { + image: image.clone(), + draw_data_offset: *draw_data_offset + offsets.draw_data, + }, })); self.color_stops.extend_from_slice(&other.color_stops); if let Some(transform) = *transform { @@ -250,8 +262,8 @@ impl Encoding { todo!("sweep gradients aren't supported yet!") } }, - BrushRef::Image(_) => { - todo!("images aren't supported yet!") + BrushRef::Image(image) => { + self.encode_image(image, alpha); } } } @@ -290,6 +302,22 @@ impl Encoding { .extend_from_slice(bytemuck::bytes_of(&gradient)); } + /// Encodes an image brush. + pub fn encode_image(&mut self, image: &Image, _alpha: f32) { + // TODO: feed the alpha multiplier through the full pipeline for consistency + // with other brushes? + self.patches.push(Patch::Image { + image: image.clone(), + draw_data_offset: self.draw_data.len(), + }); + self.draw_tags.push(DrawTag::IMAGE); + self.draw_data + .extend_from_slice(bytemuck::bytes_of(&DrawImage { + xy: 0, + width_height: (image.width << 16) | (image.height & 0xFFFF), + })); + } + /// Encodes a begin clip command. pub fn encode_begin_clip(&mut self, blend_mode: BlendMode, alpha: f32) { use super::DrawBeginClip; @@ -329,7 +357,7 @@ impl Encoding { self.color_stops.extend(color_stops); } self.patches.push(Patch::Ramp { - offset, + draw_data_offset: offset, stops: stops_start..self.color_stops.len(), }); } diff --git a/src/encoding/image_cache.rs b/src/encoding/image_cache.rs new file mode 100644 index 0000000..cd2734e --- /dev/null +++ b/src/encoding/image_cache.rs @@ -0,0 +1,92 @@ +// Copyright 2022 Google LLC +// +// 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. + +use guillotiere::{size2, AtlasAllocator}; +use peniko::Image; +use std::collections::{hash_map::Entry, HashMap}; + +const DEFAULT_ATLAS_SIZE: i32 = 1024; +const MAX_ATLAS_SIZE: i32 = 8192; + +pub struct Images<'a> { + pub width: u32, + pub height: u32, + pub images: &'a [(Image, u32, u32)], +} + +pub struct ImageCache { + atlas: AtlasAllocator, + /// Map from image blob id to atlas location. + map: HashMap, + /// List of all allocated images with associated atlas location. + images: Vec<(Image, u32, u32)>, +} + +impl Default for ImageCache { + fn default() -> Self { + Self::new() + } +} + +impl ImageCache { + pub fn new() -> Self { + Self { + atlas: AtlasAllocator::new(size2(DEFAULT_ATLAS_SIZE, DEFAULT_ATLAS_SIZE)), + map: Default::default(), + images: Default::default(), + } + } + + pub fn images(&self) -> Images { + Images { + width: self.atlas.size().width as u32, + height: self.atlas.size().height as u32, + images: &self.images, + } + } + + pub fn bump_size(&mut self) -> bool { + let new_size = self.atlas.size().width * 2; + if new_size > MAX_ATLAS_SIZE { + return false; + } + self.atlas = AtlasAllocator::new(size2(new_size, new_size)); + self.map.clear(); + self.images.clear(); + true + } + + pub fn clear(&mut self) { + self.atlas.clear(); + self.map.clear(); + self.images.clear(); + } + + pub fn get_or_insert(&mut self, image: &Image) -> Option<(u32, u32)> { + match self.map.entry(image.data.id()) { + Entry::Occupied(occupied) => Some(*occupied.get()), + Entry::Vacant(vacant) => { + let alloc = self + .atlas + .allocate(size2(image.width as _, image.height as _))?; + let x = alloc.rectangle.min.x as u32; + let y = alloc.rectangle.min.y as u32; + self.images.push((image.clone(), x, y)); + Some(*vacant.insert((x, y))) + } + } + } +} diff --git a/src/encoding/resolve.rs b/src/encoding/resolve.rs index d36a05b..58b942f 100644 --- a/src/encoding/resolve.rs +++ b/src/encoding/resolve.rs @@ -18,9 +18,11 @@ use std::ops::Range; use bytemuck::{Pod, Zeroable}; use moscato::pinot::FontRef; +use peniko::Image; use super::{ glyph_cache::{CachedRange, GlyphCache, GlyphKey}, + image_cache::{ImageCache, Images}, ramp_cache::{RampCache, Ramps}, DrawTag, Encoding, PathTag, StreamOffsets, Transform, }; @@ -144,6 +146,8 @@ pub struct Resolver { glyph_ranges: Vec, glyph_cx: GlyphContext, ramp_cache: RampCache, + image_cache: ImageCache, + pending_images: Vec, patches: Vec, } @@ -159,8 +163,9 @@ impl Resolver { &'a mut self, encoding: &Encoding, packed: &mut Vec, - ) -> (Layout, Ramps<'a>) { + ) -> (Layout, Ramps<'a>, Images<'a>) { let sizes = self.resolve_patches(encoding); + self.resolve_pending_images(); let data = packed; data.clear(); let mut layout = Layout::default(); @@ -261,6 +266,26 @@ impl Resolver { pos = *draw_data_offset + 4; } ResolvedPatch::GlyphRun { .. } => {} + ResolvedPatch::Image { + index, + draw_data_offset, + } => { + if pos < *draw_data_offset { + data.extend_from_slice(&encoding.draw_data[pos..*draw_data_offset]); + } + if let Some((x, y)) = self.pending_images[*index].xy { + let xy = (x << 16) | y; + data.extend_from_slice(bytemuck::bytes_of(&xy)); + pos = *draw_data_offset + 4; + } else { + // If we get here, we failed to allocate a slot for this image in the atlas. + // In this case, let's zero out the dimensions so we don't attempt to render + // anything. + // TODO: a better strategy: texture array? downsample large images? + data.extend_from_slice(&[0u8; 8]); + pos = *draw_data_offset + 8; + } + } } } if pos < stream.len() { @@ -336,21 +361,26 @@ impl Resolver { } layout.n_draw_objects = layout.n_paths; assert_eq!(capacity, data.len()); - (layout, self.ramp_cache.ramps()) + (layout, self.ramp_cache.ramps(), self.image_cache.images()) } fn resolve_patches(&mut self, encoding: &Encoding) -> StreamOffsets { self.ramp_cache.advance(); self.glyph_cache.clear(); self.glyph_ranges.clear(); + self.image_cache.clear(); + self.pending_images.clear(); self.patches.clear(); let mut sizes = StreamOffsets::default(); for patch in &encoding.patches { match patch { - Patch::Ramp { offset, stops } => { + Patch::Ramp { + draw_data_offset, + stops, + } => { let ramp_id = self.ramp_cache.add(&encoding.color_stops[stops.clone()]); self.patches.push(ResolvedPatch::Ramp { - draw_data_offset: *offset + sizes.draw_data, + draw_data_offset: *draw_data_offset + sizes.draw_data, ramp_id, }); } @@ -414,10 +444,50 @@ impl Resolver { transform, }); } + Patch::Image { + draw_data_offset, + image, + } => { + let index = self.pending_images.len(); + self.pending_images.push(PendingImage { + image: image.clone(), + xy: None, + }); + self.patches.push(ResolvedPatch::Image { + index, + draw_data_offset: *draw_data_offset + sizes.draw_data, + }); + } } } sizes } + + fn resolve_pending_images(&mut self) { + self.image_cache.clear(); + 'outer: loop { + // Loop over the images, attempting to allocate them all into the atlas. + for pending_image in &mut self.pending_images { + if let Some(xy) = self.image_cache.get_or_insert(&pending_image.image) { + pending_image.xy = Some(xy); + } else { + // We failed to allocate. Try to bump the atlas size. + if self.image_cache.bump_size() { + // We were able to increase the atlas size. Restart the outer loop. + continue 'outer; + } else { + // If the atlas is already maximum size, there's nothing we can do. Set + // the xy field to None so this image isn't rendered and then carry on-- + // other images might still fit. + pending_image.xy = None; + } + } + } + // If we made it here, we've either successfully allocated all images or we reached + // the maximum atlas size. + break; + } + } } #[derive(Clone)] @@ -426,7 +496,7 @@ pub enum Patch { /// Gradient ramp resource. Ramp { /// Byte offset to the ramp id in the draw data stream. - offset: usize, + draw_data_offset: usize, /// Range of the gradient stops in the resource set. stops: Range, }, @@ -435,6 +505,20 @@ pub enum Patch { /// Index in the glyph run buffer. index: usize, }, + /// Image resource. + Image { + /// Offset to the atlas coordinates in the draw data stream. + draw_data_offset: usize, + /// Underlying image data. + image: Image, + }, +} + +/// Image to be allocated in the atlas. +#[derive(Clone, Debug)] +struct PendingImage { + image: Image, + xy: Option<(u32, u32)>, } #[derive(Clone, Debug)] @@ -453,6 +537,12 @@ enum ResolvedPatch { /// Global transform. transform: Transform, }, + Image { + /// Index of pending image element. + index: usize, + /// Offset to the atlas location in the draw data stream. + draw_data_offset: usize, + }, } fn slice_size_in_bytes(slice: &[T], extra: usize) -> usize { diff --git a/src/engine.rs b/src/engine.rs index 9250265..805b915 100644 --- a/src/engine.rs +++ b/src/engine.rs @@ -89,6 +89,7 @@ pub enum Command { Upload(BufProxy, Vec), UploadUniform(BufProxy, Vec), UploadImage(ImageProxy, Vec), + WriteImage(ImageProxy, [u32; 4], Vec), // Discussion question: third argument is vec of resources? // Maybe use tricks to make more ergonomic? // Alternative: provide bufs & images as separate sequences @@ -275,11 +276,6 @@ impl Engine { self.bind_map.insert_buf(buf_proxy, buf); } Command::UploadImage(image_proxy, bytes) => { - let buf = device.create_buffer_init(&wgpu::util::BufferInitDescriptor { - label: None, - contents: bytes, - usage: wgpu::BufferUsages::COPY_SRC, - }); let format = image_proxy.format.to_wgpu(); let texture = device.create_texture(&wgpu::TextureDescriptor { label: None, @@ -305,21 +301,19 @@ impl Engine { array_layer_count: None, format: Some(TextureFormat::Rgba8Unorm), }); - encoder.copy_buffer_to_texture( - wgpu::ImageCopyBuffer { - buffer: &buf, - layout: wgpu::ImageDataLayout { - offset: 0, - bytes_per_row: NonZeroU32::new(image_proxy.width * 4), - rows_per_image: None, - }, - }, + queue.write_texture( wgpu::ImageCopyTexture { texture: &texture, mip_level: 0, origin: wgpu::Origin3d { x: 0, y: 0, z: 0 }, aspect: TextureAspect::All, }, + bytes, + wgpu::ImageDataLayout { + offset: 0, + bytes_per_row: NonZeroU32::new(image_proxy.width * 4), + rows_per_image: None, + }, wgpu::Extent3d { width: image_proxy.width, height: image_proxy.height, @@ -329,6 +323,29 @@ impl Engine { self.bind_map .insert_image(image_proxy.id, texture, texture_view) } + Command::WriteImage(proxy, [x, y, width, height], data) => { + if let Ok((texture, _)) = self.bind_map.get_or_create_image(*proxy, device) { + queue.write_texture( + wgpu::ImageCopyTexture { + texture: &texture, + mip_level: 0, + origin: wgpu::Origin3d { x: *x, y: *y, z: 0 }, + aspect: TextureAspect::All, + }, + &data[..], + wgpu::ImageDataLayout { + offset: 0, + bytes_per_row: NonZeroU32::new(*width * 4), + rows_per_image: None, + }, + wgpu::Extent3d { + width: *width, + height: *height, + depth_or_array_layers: 1, + }, + ); + } + } Command::Dispatch(shader_id, wg_size, bindings) => { // println!("dispatching {:?} with {} bindings", wg_size, bindings.len()); let shader = &self.shaders[shader_id.0]; @@ -444,6 +461,19 @@ impl Recording { image_proxy } + pub fn write_image( + &mut self, + image: ImageProxy, + x: u32, + y: u32, + width: u32, + height: u32, + data: impl Into>, + ) { + let data = data.into(); + self.push(Command::WriteImage(image, [x, y, width, height], data)); + } + pub fn dispatch(&mut self, shader: ShaderId, wg_size: (u32, u32, u32), resources: R) where R: IntoIterator, @@ -716,6 +746,44 @@ impl BindMap { } } } + + fn get_or_create_image( + &mut self, + proxy: ImageProxy, + device: &Device, + ) -> Result<&(Texture, TextureView), Error> { + match self.image_map.entry(proxy.id) { + Entry::Occupied(occupied) => Ok(occupied.into_mut()), + Entry::Vacant(vacant) => { + let format = proxy.format.to_wgpu(); + let texture = device.create_texture(&wgpu::TextureDescriptor { + label: None, + size: wgpu::Extent3d { + width: proxy.width, + height: proxy.height, + depth_or_array_layers: 1, + }, + mip_level_count: 1, + sample_count: 1, + dimension: wgpu::TextureDimension::D2, + usage: TextureUsages::TEXTURE_BINDING | TextureUsages::COPY_DST, + format, + view_formats: &[], + }); + let texture_view = texture.create_view(&wgpu::TextureViewDescriptor { + label: None, + dimension: Some(TextureViewDimension::D2), + aspect: TextureAspect::All, + mip_level_count: None, + base_mip_level: 0, + base_array_layer: 0, + array_layer_count: None, + format: Some(TextureFormat::Rgba8Unorm), + }); + Ok(vacant.insert((texture, texture_view))) + } + } + } } const SIZE_CLASS_BITS: u32 = 1; diff --git a/src/render.rs b/src/render.rs index 4ad5083..d96815e 100644 --- a/src/render.rs +++ b/src/render.rs @@ -33,6 +33,7 @@ struct FineResources { ptcl_buf: ResourceProxy, gradient_image: ResourceProxy, info_bin_data_buf: ResourceProxy, + image_atlas: ResourceProxy, out_image: ImageProxy, } @@ -216,7 +217,7 @@ impl Render { let mut recording = Recording::default(); let mut resolver = Resolver::new(); let mut packed = vec![]; - let (layout, ramps) = resolver.resolve(encoding, &mut packed); + let (layout, ramps, images) = resolver.resolve(encoding, &mut packed); let gradient_image = if ramps.height == 0 { ResourceProxy::new_image(1, 1, ImageFormat::Rgba8) } else { @@ -228,6 +229,11 @@ impl Render { data, )) }; + let image_atlas = if images.images.is_empty() { + ImageProxy::new(1, 1, ImageFormat::Rgba8) + } else { + ImageProxy::new(images.width, images.height, ImageFormat::Rgba8) + }; // TODO: calculate for real when we do rectangles let n_pathtag = layout.path_tags(&packed).len(); let pathtag_padded = align_up(n_pathtag, 4 * shaders::PATHTAG_REDUCE_WG); @@ -251,6 +257,16 @@ impl Render { ptcl_size: self.ptcl_size, layout: layout, }; + for image in images.images { + recording.write_image( + image_atlas, + image.1, + image.2, + image.0.width, + image.0.height, + image.0.data.data(), + ); + } // println!("{:?}", config); let scene_buf = ResourceProxy::Buf(recording.upload("scene", packed)); let config_buf = @@ -504,6 +520,7 @@ impl Render { ptcl_buf, gradient_image, info_bin_data_buf, + image_atlas: ResourceProxy::Image(image_atlas), out_image, }); if robust { @@ -527,6 +544,7 @@ impl Render { fine.ptcl_buf, fine.gradient_image, fine.info_bin_data_buf, + fine.image_atlas, ], ); recording.free_resource(fine.config_buf); @@ -534,6 +552,7 @@ impl Render { recording.free_resource(fine.segments_buf); recording.free_resource(fine.ptcl_buf); recording.free_resource(fine.gradient_image); + recording.free_resource(fine.image_atlas); recording.free_resource(fine.info_bin_data_buf); } diff --git a/src/scene.rs b/src/scene.rs index 29fba23..1d2b345 100644 --- a/src/scene.rs +++ b/src/scene.rs @@ -15,7 +15,7 @@ // Also licensed under MIT license, at your choice. use peniko::kurbo::{Affine, Rect, Shape}; -use peniko::{BlendMode, BrushRef, Color, Fill, Font, Stroke, StyleRef}; +use peniko::{BlendMode, BrushRef, Color, Fill, Font, Image, Stroke, StyleRef}; use crate::encoding::{Encoding, Glyph, GlyphRun, Patch, Transform}; @@ -168,6 +168,17 @@ impl<'a> SceneBuilder<'a> { } } + /// Draws an image at its natural size with the given transform. + pub fn draw_image(&mut self, image: &Image, transform: Affine) { + self.fill( + Fill::NonZero, + transform, + image, + None, + &Rect::new(0.0, 0.0, image.width as f64, image.height as f64), + ); + } + /// Returns a builder for encoding a glyph run. pub fn draw_glyphs(&mut self, font: &Font) -> DrawGlyphs { DrawGlyphs::new(self.scene, font) diff --git a/src/shaders.rs b/src/shaders.rs index 4c7b3da..01c3b38 100644 --- a/src/shaders.rs +++ b/src/shaders.rs @@ -351,6 +351,7 @@ pub fn full_shaders(device: &Device, engine: &mut Engine) -> Result