renderbug/src/buffers.rs

348 lines
9.0 KiB
Rust

use crate::geometry::*;
use crate::lib8::interpolate::Fract8Ops;
use crate::power::AsMilliwatts;
use crate::render::{PixelView, Sample, Shader, Surface, Surfaces, HardwarePixel};
use crate::task::Task;
use std::fmt::Debug;
use std::io;
use std::ops::IndexMut;
use std::sync::atomic::AtomicBool;
use std::sync::RwLock;
use std::sync::{Arc, Mutex};
#[derive(Debug)]
struct ShaderBinding {
shader: Option<Box<dyn Shader>>,
rect: Rectangle<Virtual>,
opacity: u8
}
#[derive(Debug)]
struct SurfaceUpdate {
shader: Option<Option<Box<dyn Shader>>>,
rect: Option<Rectangle<Virtual>>,
opacity: Option<u8>,
slot: usize
}
impl SurfaceUpdate {
fn merge(&mut self, mut other: Self) {
if other.shader.is_some() {
self.shader = other.shader.take()
}
if other.rect.is_some() {
self.rect = other.rect.take()
}
if other.opacity.is_some() {
self.opacity = other.opacity.take()
}
}
}
impl Debug for Box<dyn Shader> {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.debug_struct("Shader").finish()
}
}
impl Default for SurfaceUpdate {
fn default() -> Self {
SurfaceUpdate {
shader: None,
rect: None,
opacity: None,
slot: usize::MAX
}
}
}
#[derive(Debug)]
pub struct BufferedSurface {
updater: Arc<UpdateQueue>,
slot: usize
}
impl Surface for BufferedSurface {
fn clear_shader(&mut self) {
self.updater.push(SurfaceUpdate {
shader: None,
slot: self.slot,
..Default::default()
});
}
fn set_opacity(&mut self, opacity: u8) {
self.updater.push(SurfaceUpdate {
opacity: Some(opacity),
slot: self.slot,
..Default::default()
});
}
fn set_rect(&mut self, rect: &Rectangle<Virtual>) {
self.updater.push(SurfaceUpdate {
rect: Some(rect.clone()),
slot: self.slot,
..Default::default()
});
}
fn set_shader(&mut self, shader: Box<dyn Shader>) {
self.updater.push(SurfaceUpdate {
shader: Some(Some(shader)),
slot: self.slot,
..Default::default()
});
}
}
#[derive(Debug)]
struct UpdateQueue {
pending: Mutex<Vec<SurfaceUpdate>>,
damaged: AtomicBool
}
impl UpdateQueue {
fn new() -> Self {
UpdateQueue {
pending: Mutex::new(Vec::new()),
damaged: AtomicBool::new(false)
}
}
fn push(&self, update: SurfaceUpdate) {
let mut locked = self.pending.lock().unwrap();
let mut existing_slot = None;
for existing in locked.iter_mut() {
if existing.slot == update.slot {
existing_slot = Some(existing);
break
}
}
match existing_slot {
Some(tgt) => {
log::debug!("Updating existing shader update");
tgt.merge(update);
}
_ => {
log::debug!("Pushing new shader update");
locked.push(update);
self.damaged.store(true, std::sync::atomic::Ordering::Relaxed);
}
}
}
}
#[derive(Debug)]
pub struct ShaderChain {
bindings: Vec<ShaderBinding>,
updates: Arc<UpdateQueue>
}
impl ShaderChain {
pub fn new() -> Self {
ShaderChain {
bindings: Vec::new(),
updates: Arc::new(UpdateQueue::new())
}
}
pub fn is_dirty(&self) -> bool {
self.updates.damaged.load(std::sync::atomic::Ordering::Relaxed)
}
pub fn commit(&mut self) {
let mut queue: Vec<SurfaceUpdate> = {
let mut updates = self.updates.pending.lock().unwrap();
std::mem::take(updates.as_mut())
};
for update in queue.iter_mut() {
let target_slot = &mut self.bindings[update.slot];
if let Some(shader) = update.shader.take() {
target_slot.shader = shader;
}
if let Some(opacity) = update.opacity.take() {
target_slot.opacity = opacity;
}
if let Some(rect) = update.rect.take() {
target_slot.rect = rect;
}
}
self.updates.damaged.store(false, std::sync::atomic::Ordering::Relaxed);
}
}
impl Surfaces for ShaderChain {
type Error = ();
type Surface = BufferedSurface;
fn new_surface(&mut self, area: &Rectangle<Virtual>) -> Result<Self::Surface, Self::Error> {
let next_slot = self.bindings.len();
self.bindings.push(ShaderBinding {
opacity: 255,
shader: None,
rect: area.clone()
});
Ok(BufferedSurface {
updater: Arc::clone(&self.updates),
slot: next_slot
})
}
fn render_to<S: Sample>(&self, output: &mut S, frame: usize) {
for surface in self.bindings.iter() {
let opacity = surface.opacity;
if opacity > 0 {
let rect = surface.rect;
let mut sample = output.sample(&rect);
if let Some(ref shader) = surface.shader {
while let Some((virt_coords, pixel)) = sample.next() {
*pixel = pixel.blend8(shader.draw(&virt_coords, frame).into(), opacity);
}
}
}
}
}
}
#[derive(Clone)]
pub struct BufferedSurfacePool {
pool: Arc<RwLock<ShaderChain>>
}
impl BufferedSurfacePool {
pub fn new() -> Self {
BufferedSurfacePool {
pool: Arc::new(RwLock::new(ShaderChain::new()))
}
}
}
impl Surfaces for BufferedSurfacePool {
type Error = ();
type Surface = <ShaderChain as Surfaces>::Surface;
fn new_surface(&mut self, area: &crate::geometry::Rectangle<crate::geometry::Virtual>) -> Result<Self::Surface, Self::Error> {
self.pool.write().unwrap().new_surface(area)
}
fn render_to<S: crate::render::Sample>(&self, output: &mut S, frame: usize) {
self.pool.read().unwrap().render_to(output, frame);
}
}
impl Task for BufferedSurfacePool {
fn tick(&mut self) {
if self.pool.read().unwrap().is_dirty() {
self.pool.write().unwrap().commit();
}
}
}
#[derive(Clone)]
pub struct SharedSurface {
binding: Arc<Mutex<ShaderBinding>>
}
impl Default for SharedSurface {
fn default() -> Self {
Self {
binding: Arc::new(Mutex::new(ShaderBinding {
shader: None,
rect: Rectangle::everything(),
opacity: 255
})),
}
}
}
impl Surface for SharedSurface {
fn set_shader(&mut self, shader: Box<dyn Shader>) {
self.binding.lock().unwrap().shader = Some(shader);
}
fn clear_shader(&mut self) {
self.binding.lock().unwrap().shader = None;
}
fn set_rect(&mut self, rect: &Rectangle<Virtual>) {
self.binding.lock().unwrap().rect = rect.clone();
}
fn set_opacity(&mut self, opacity: u8) {
self.binding.lock().unwrap().opacity = opacity
}
}
#[derive(Clone)]
pub struct SurfacePool {
surfaces: Vec<SharedSurface>
}
impl SurfacePool {
pub const fn new() -> Self {
Self {
surfaces: Vec::new()
}
}
}
impl Surfaces for SurfacePool {
type Surface = SharedSurface;
type Error = io::Error;
fn new_surface(&mut self, area: &Rectangle<Virtual>) -> Result<Self::Surface, Self::Error> {
let mut surface = SharedSurface::default();
surface.set_rect(area);
self.surfaces.push(surface.clone());
return Ok(surface);
}
fn render_to<Sampler: Sample>(&self, output: &mut Sampler, frame: usize) {
for surface in self.surfaces.iter() {
let binding = surface.binding.lock().unwrap();
let opacity = binding.opacity;
if opacity > 0 {
let rect = binding.rect;
let mut sample = output.sample(&rect);
if let Some(ref shader) = binding.shader {
while let Some((virt_coords, pixel)) = sample.next() {
*pixel = pixel.blend8(shader.draw(&virt_coords, frame).into(), opacity);
}
}
}
}
}
}
pub trait Pixbuf: AsMilliwatts + IndexMut<usize, Output=Self::Pixel> + Send {
type Pixel: HardwarePixel;
fn new() -> Self;
fn blank(&mut self);
fn iter_with_brightness(&self, brightness: u8) -> impl Iterator<Item = Self::Pixel> + Send;
fn pixel_count(&self) -> usize;
}
impl<T: HardwarePixel, const PIXEL_NUM: usize> Pixbuf for [T; PIXEL_NUM] {
type Pixel = T;
fn new() -> Self {
[T::default(); PIXEL_NUM]
}
fn pixel_count(&self) -> usize {
self.len()
}
fn blank(&mut self) {
self.fill(T::default())
}
fn iter_with_brightness(&self, brightness: u8) -> impl Iterator<Item=T> + Send {
self.iter().map(move |x| { x.scale8(brightness)})
}
}