use crate::bitmap::{BitmapData, BitmapDataMut}; use crate::formats::{self, PixelFormatSpec, NUM_PIXEL_FORMATS}; use crate::op::{self, BinaryOpSpec, UnaryOpSpec}; pub type UnaryFunc = for<'l> fn(&'l mut dyn BitmapDataMut, &'l Params); pub type BinaryFunc = for<'l> fn(&'l mut dyn BitmapDataMut, &'l dyn BitmapData, &'l Params); pub trait GenericUnary { fn perform<'l, Format: PixelFormatSpec>( write_data: &'l mut dyn BitmapDataMut, params: &'l Params, ); } pub trait GenericBinary { fn perform<'l, Format: PixelFormatSpec, ReadFormat: PixelFormatSpec>( write_data: &'l mut dyn BitmapDataMut, read_data: &'l dyn BitmapData, params: &'l Params, ); } pub struct UnaryDispatch { op_array: [UnaryFunc; NUM_PIXEL_FORMATS], changed: [bool; NUM_PIXEL_FORMATS], } impl UnaryDispatch { pub fn with(mut self, specific_imp: UnaryFunc) -> Self { self.op_array[Format::FORMAT_ENUM as usize] = specific_imp; self.changed[Format::FORMAT_ENUM as usize] = true; self } pub fn with_avx(mut self, avx_imp: UnaryFunc) -> Self { if is_x86_feature_detected!("avx") { self.op_array[Format::FORMAT_ENUM as usize] = avx_imp; self.changed[Format::FORMAT_ENUM as usize] = true; self } else { self } } pub fn select(&self) -> UnaryFunc { self.op_array[Format::FORMAT_ENUM as usize] } pub fn is_generic(&self) -> bool { !self.changed[Format::FORMAT_ENUM as usize] } } pub struct BinaryDispatch { op_array: [[BinaryFunc; NUM_PIXEL_FORMATS]; NUM_PIXEL_FORMATS], changed: [[bool; NUM_PIXEL_FORMATS]; NUM_PIXEL_FORMATS], } impl BinaryDispatch { pub fn with( mut self, imp: BinaryFunc, ) -> Self { self.op_array[WriteFormat::FORMAT_ENUM as usize][ReadFormat::FORMAT_ENUM as usize] = imp; self.changed[WriteFormat::FORMAT_ENUM as usize][ReadFormat::FORMAT_ENUM as usize] = true; self } pub fn with_avx( mut self, avx_imp: BinaryFunc, ) -> Self { if is_x86_feature_detected!("avx") { self.op_array[WriteFormat::FORMAT_ENUM as usize][ReadFormat::FORMAT_ENUM as usize] = avx_imp; self.changed[WriteFormat::FORMAT_ENUM as usize][ReadFormat::FORMAT_ENUM as usize] = true; self } else { self } } pub fn select( &self, ) -> BinaryFunc { self.op_array[WriteFormat::FORMAT_ENUM as usize][ReadFormat::FORMAT_ENUM as usize] } pub fn is_generic(&self) -> bool { !self.changed[WriteFormat::FORMAT_ENUM as usize][ReadFormat::FORMAT_ENUM as usize] } } macro_rules! dispatch_entry { (unary, $sn:ident, $opn:ident) => { UnaryDispatch < ::Params<'l>> }; (binary, $sn:ident, $opn:ident) => { BinaryDispatch < ::Params<'l>> }; } pub struct KahloDispatch<'l> { pub composite: dispatch_entry!(binary, composite, CompositeOp), pub copy_from: dispatch_entry!(binary, copy_from, CopyFromOp), pub fill: dispatch_entry!(unary, fill, FillOp), pub fill_region: dispatch_entry!(unary, fill_region, FillRegionOp), pub fill_region_masked: dispatch_entry!(unary, fill_region_masked, FillRegionMaskedOp), pub rectangle: dispatch_entry!(unary, rectangle, RectangleOp), } impl<'l> Default for KahloDispatch<'l> { fn default() -> Self { Self { composite: op::composite::CompositeOp::build(), copy_from: op::copy_from::CopyFromOp::build(), fill: op::fill::FillOp::build(), fill_region: op::fill_region::FillRegionOp::build(), fill_region_masked: op::fill_region_masked::FillRegionMaskedOp::build(), rectangle: op::rectangle::RectangleOp::build(), } } } impl<'l> KahloDispatch<'l> { pub fn use_lifetime(&self) -> &KahloDispatch<'_> { unsafe { std::mem::transmute(self) } } } lazy_static::lazy_static! { pub static ref DISPATCH: KahloDispatch<'static> = KahloDispatch::default(); } // long but simple implementations impl UnaryDispatch { pub(crate) fn from_generic>() -> Self { Self { op_array: [ G::perform::, G::perform::, G::perform::, G::perform::, G::perform::, G::perform::, G::perform::, ], changed: Default::default(), } } } impl BinaryDispatch { pub(crate) fn from_generic>() -> Self { Self { op_array: [ [ G::perform::, G::perform::, G::perform::, G::perform::, G::perform::, G::perform::, G::perform::, ], [ G::perform::, G::perform::, G::perform::, G::perform::, G::perform::, G::perform::, G::perform::, ], [ G::perform::, G::perform::, G::perform::, G::perform::, G::perform::, G::perform::, G::perform::, ], [ G::perform::, G::perform::, G::perform::, G::perform::, G::perform::, G::perform::, G::perform::, ], [ G::perform::, G::perform::, G::perform::, G::perform::, G::perform::, G::perform::, G::perform::, ], [ G::perform::, G::perform::, G::perform::, G::perform::, G::perform::, G::perform::, G::perform::, ], [ G::perform::, G::perform::, G::perform::, G::perform::, G::perform::, G::perform::, G::perform::, ], ], changed: Default::default(), } } }