dispatch.rs 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. use crate::bitmap::{BitmapData, BitmapDataMut};
  2. use crate::formats::{self, PixelFormatSpec, NUM_PIXEL_FORMATS};
  3. use crate::op::{self, BinaryOpSpec, UnaryOpSpec};
  4. pub type UnaryFunc<Params> = for<'l> fn(&'l mut dyn BitmapDataMut, &'l Params);
  5. pub type BinaryFunc<Params> = for<'l> fn(&'l mut dyn BitmapDataMut, &'l dyn BitmapData, &'l Params);
  6. pub trait GenericUnary<Params> {
  7. fn perform<'l, Format: PixelFormatSpec>(
  8. write_data: &'l mut dyn BitmapDataMut,
  9. params: &'l Params,
  10. );
  11. }
  12. pub trait GenericBinary<Params> {
  13. fn perform<'l, Format: PixelFormatSpec, ReadFormat: PixelFormatSpec>(
  14. write_data: &'l mut dyn BitmapDataMut,
  15. read_data: &'l dyn BitmapData,
  16. params: &'l Params,
  17. );
  18. }
  19. pub struct UnaryDispatch<Params> {
  20. op_array: [UnaryFunc<Params>; NUM_PIXEL_FORMATS],
  21. changed: [bool; NUM_PIXEL_FORMATS],
  22. }
  23. impl<Params> UnaryDispatch<Params> {
  24. pub fn with<Format: PixelFormatSpec>(mut self, specific_imp: UnaryFunc<Params>) -> Self {
  25. self.op_array[Format::FORMAT_ENUM as usize] = specific_imp;
  26. self.changed[Format::FORMAT_ENUM as usize] = true;
  27. self
  28. }
  29. pub fn with_avx<Format: PixelFormatSpec>(mut self, avx_imp: UnaryFunc<Params>) -> Self {
  30. if is_x86_feature_detected!("avx") {
  31. self.op_array[Format::FORMAT_ENUM as usize] = avx_imp;
  32. self.changed[Format::FORMAT_ENUM as usize] = true;
  33. self
  34. } else {
  35. self
  36. }
  37. }
  38. pub fn select<Format: PixelFormatSpec>(&self) -> UnaryFunc<Params> {
  39. self.op_array[Format::FORMAT_ENUM as usize]
  40. }
  41. pub fn is_generic<Format: PixelFormatSpec>(&self) -> bool {
  42. !self.changed[Format::FORMAT_ENUM as usize]
  43. }
  44. }
  45. pub struct BinaryDispatch<Params> {
  46. op_array: [[BinaryFunc<Params>; NUM_PIXEL_FORMATS]; NUM_PIXEL_FORMATS],
  47. changed: [[bool; NUM_PIXEL_FORMATS]; NUM_PIXEL_FORMATS],
  48. }
  49. impl<Params> BinaryDispatch<Params> {
  50. pub fn with<WriteFormat: PixelFormatSpec, ReadFormat: PixelFormatSpec>(
  51. mut self,
  52. imp: BinaryFunc<Params>,
  53. ) -> Self {
  54. self.op_array[WriteFormat::FORMAT_ENUM as usize][ReadFormat::FORMAT_ENUM as usize] = imp;
  55. self.changed[WriteFormat::FORMAT_ENUM as usize][ReadFormat::FORMAT_ENUM as usize] = true;
  56. self
  57. }
  58. pub fn with_avx<WriteFormat: PixelFormatSpec, ReadFormat: PixelFormatSpec>(
  59. mut self,
  60. avx_imp: BinaryFunc<Params>,
  61. ) -> Self {
  62. if is_x86_feature_detected!("avx") {
  63. self.op_array[WriteFormat::FORMAT_ENUM as usize][ReadFormat::FORMAT_ENUM as usize] =
  64. avx_imp;
  65. self.changed[WriteFormat::FORMAT_ENUM as usize][ReadFormat::FORMAT_ENUM as usize] = true;
  66. self
  67. } else {
  68. self
  69. }
  70. }
  71. pub fn select<WriteFormat: PixelFormatSpec, ReadFormat: PixelFormatSpec>(
  72. &self,
  73. ) -> BinaryFunc<Params> {
  74. self.op_array[WriteFormat::FORMAT_ENUM as usize][ReadFormat::FORMAT_ENUM as usize]
  75. }
  76. pub fn is_generic<WriteFormat: PixelFormatSpec, ReadFormat: PixelFormatSpec>(&self) -> bool {
  77. !self.changed[WriteFormat::FORMAT_ENUM as usize][ReadFormat::FORMAT_ENUM as usize]
  78. }
  79. }
  80. macro_rules! dispatch_entry {
  81. (unary, $sn:ident, $opn:ident) => {
  82. UnaryDispatch < <op::$sn::$opn as UnaryOpSpec>::Params<'l>>
  83. };
  84. (binary, $sn:ident, $opn:ident) => {
  85. BinaryDispatch < <op::$sn::$opn as BinaryOpSpec>::Params<'l>>
  86. };
  87. }
  88. pub struct KahloDispatch<'l> {
  89. pub composite: dispatch_entry!(binary, composite, CompositeOp),
  90. pub copy_from: dispatch_entry!(binary, copy_from, CopyFromOp),
  91. pub fill: dispatch_entry!(unary, fill, FillOp),
  92. pub fill_region: dispatch_entry!(unary, fill_region, FillRegionOp),
  93. pub fill_region_masked: dispatch_entry!(unary, fill_region_masked, FillRegionMaskedOp),
  94. pub rectangle: dispatch_entry!(unary, rectangle, RectangleOp),
  95. }
  96. impl<'l> Default for KahloDispatch<'l> {
  97. fn default() -> Self {
  98. Self {
  99. composite: op::composite::CompositeOp::build(),
  100. copy_from: op::copy_from::CopyFromOp::build(),
  101. fill: op::fill::FillOp::build(),
  102. fill_region: op::fill_region::FillRegionOp::build(),
  103. fill_region_masked: op::fill_region_masked::FillRegionMaskedOp::build(),
  104. rectangle: op::rectangle::RectangleOp::build(),
  105. }
  106. }
  107. }
  108. impl<'l> KahloDispatch<'l> {
  109. pub fn use_lifetime(&self) -> &KahloDispatch<'_> {
  110. unsafe { std::mem::transmute(self) }
  111. }
  112. }
  113. lazy_static::lazy_static! {
  114. pub static ref DISPATCH: KahloDispatch<'static> = KahloDispatch::default();
  115. }
  116. // long but simple implementations
  117. impl<Params> UnaryDispatch<Params> {
  118. pub(crate) fn from_generic<G: GenericUnary<Params>>() -> Self {
  119. Self {
  120. op_array: [
  121. G::perform::<formats::A8>,
  122. G::perform::<formats::Abgr32>,
  123. G::perform::<formats::Bgr24>,
  124. G::perform::<formats::Bgr32>,
  125. G::perform::<formats::Rgb24>,
  126. G::perform::<formats::Rgb32>,
  127. G::perform::<formats::Rgba32>,
  128. ],
  129. changed: Default::default(),
  130. }
  131. }
  132. }
  133. impl<Params> BinaryDispatch<Params> {
  134. pub(crate) fn from_generic<G: GenericBinary<Params>>() -> Self {
  135. Self {
  136. op_array: [
  137. [
  138. G::perform::<formats::A8, formats::A8>,
  139. G::perform::<formats::A8, formats::Abgr32>,
  140. G::perform::<formats::A8, formats::Bgr24>,
  141. G::perform::<formats::A8, formats::Bgr32>,
  142. G::perform::<formats::A8, formats::Rgb24>,
  143. G::perform::<formats::A8, formats::Rgb32>,
  144. G::perform::<formats::A8, formats::Rgba32>,
  145. ],
  146. [
  147. G::perform::<formats::Abgr32, formats::A8>,
  148. G::perform::<formats::Abgr32, formats::Abgr32>,
  149. G::perform::<formats::Abgr32, formats::Bgr24>,
  150. G::perform::<formats::Abgr32, formats::Bgr32>,
  151. G::perform::<formats::Abgr32, formats::Rgb24>,
  152. G::perform::<formats::Abgr32, formats::Rgb32>,
  153. G::perform::<formats::Abgr32, formats::Rgba32>,
  154. ],
  155. [
  156. G::perform::<formats::Bgr24, formats::A8>,
  157. G::perform::<formats::Bgr24, formats::Abgr32>,
  158. G::perform::<formats::Bgr24, formats::Bgr24>,
  159. G::perform::<formats::Bgr24, formats::Bgr32>,
  160. G::perform::<formats::Bgr24, formats::Rgb24>,
  161. G::perform::<formats::Bgr24, formats::Rgb32>,
  162. G::perform::<formats::Bgr24, formats::Rgba32>,
  163. ],
  164. [
  165. G::perform::<formats::Bgr32, formats::A8>,
  166. G::perform::<formats::Bgr32, formats::Abgr32>,
  167. G::perform::<formats::Bgr32, formats::Bgr24>,
  168. G::perform::<formats::Bgr32, formats::Bgr32>,
  169. G::perform::<formats::Bgr32, formats::Rgb24>,
  170. G::perform::<formats::Bgr32, formats::Rgb32>,
  171. G::perform::<formats::Bgr32, formats::Rgba32>,
  172. ],
  173. [
  174. G::perform::<formats::Rgb24, formats::A8>,
  175. G::perform::<formats::Rgb24, formats::Abgr32>,
  176. G::perform::<formats::Rgb24, formats::Bgr24>,
  177. G::perform::<formats::Rgb24, formats::Bgr32>,
  178. G::perform::<formats::Rgb24, formats::Rgb24>,
  179. G::perform::<formats::Rgb24, formats::Rgb32>,
  180. G::perform::<formats::Rgb24, formats::Rgba32>,
  181. ],
  182. [
  183. G::perform::<formats::Rgb32, formats::A8>,
  184. G::perform::<formats::Rgb32, formats::Abgr32>,
  185. G::perform::<formats::Rgb32, formats::Bgr24>,
  186. G::perform::<formats::Rgb32, formats::Bgr32>,
  187. G::perform::<formats::Rgb32, formats::Rgb24>,
  188. G::perform::<formats::Rgb32, formats::Rgb32>,
  189. G::perform::<formats::Rgb32, formats::Rgba32>,
  190. ],
  191. [
  192. G::perform::<formats::Rgba32, formats::A8>,
  193. G::perform::<formats::Rgba32, formats::Abgr32>,
  194. G::perform::<formats::Rgba32, formats::Bgr24>,
  195. G::perform::<formats::Rgba32, formats::Bgr32>,
  196. G::perform::<formats::Rgba32, formats::Rgb24>,
  197. G::perform::<formats::Rgba32, formats::Rgb32>,
  198. G::perform::<formats::Rgba32, formats::Rgba32>,
  199. ],
  200. ],
  201. changed: Default::default(),
  202. }
  203. }
  204. }