1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57
use crate::internal::*; #[derive(Debug, Clone, new, Default, Hash)] pub struct MultiBroadcastTo { pub shape: TVec<TDim>, } tract_linalg::impl_dyn_hash!(MultiBroadcastTo); impl MultiBroadcastTo { pub fn eval_t<T: Datum>(input: &Tensor, shape: &[usize]) -> TractResult<TVec<Arc<Tensor>>> { let input = input.to_array_view::<T>()?; let output = input.broadcast(&*shape).ok_or("incompatible shapes")?; Ok(tvec![output.to_owned().into_arc_tensor()]) } } impl Op for MultiBroadcastTo { fn name(&self) -> Cow<str> { "MultiBroadcastTo".into() } canonic!(); op_core_mir!(); op_as_typed_op!(); not_a_pulsed_op!(); } impl StatelessOp for MultiBroadcastTo { fn eval(&self, mut inputs: TVec<Arc<Tensor>>) -> TractResult<TVec<Arc<Tensor>>> { let input = args_1!(inputs); let dims: Vec<usize> = self.shape.iter().map(|d| Ok(d.to_integer()? as usize)).collect::<TractResult<_>>()?; dispatch_datum!(Self::eval_t(input.datum_type())(&*input, &*dims)) } } impl TypedOp for MultiBroadcastTo { fn output_facts(&self, inputs: &[&TypedFact]) -> TractResult<TVec<TypedFact>> { Ok(tvec!(TypedFact::dt_shape(inputs[0].datum_type, &*self.shape)?)) } fn concretize_stream_dim( &self, _source: &TypedModel, node: &TypedNode, target: &mut TypedModel, mapping: &HashMap<OutletId, OutletId>, stream_dim: usize, ) -> TractResult<TVec<OutletId>> { let input = mapping[&node.inputs[0]]; let op = Self { shape: self.shape.iter().map(|d| d.eval(stream_dim as _).unwrap().to_dim()).collect(), }; target.wire_node(&node.name, op, &[input]) } as_op!(); }