#include "./grad.h" #include "megbrain/imperative/ops/autogen.h" #include "megbrain/imperative/transformations/grad.h" namespace mgb::imperative::python { class CustomGradMaker { bool output_size_set = false, input_has_grad_initialized = false; CustomBackward& target; size_t nr_inputs; void init_input_has_grad() { if (!input_has_grad_initialized) { input_has_grad_initialized = true; target.m_input_has_grad.resize(nr_inputs, true); } } public: CustomGradMaker(CustomBackward& target, size_t nr_inputs) : target(target), nr_inputs(nr_inputs) {} CustomGradMaker& backward(CustomBackward::BackwardFn f) { mgb_assert(!target.m_backward); target.m_backward = f; return *this; } // mandatory CustomGradMaker& output_size(size_t sz) { mgb_assert(!output_size_set); output_size_set = true; target.m_output_attrs.resize(sz); return *this; } // optional, defaults to all true CustomGradMaker& input_has_grad(size_t i, bool v) { init_input_has_grad(); target.m_input_has_grad.at(i) = v; return *this; } // optional, defaults to all true CustomGradMaker& output_requires_grad(size_t i, bool v) { target.m_output_attrs.at(i).requires_grad = v; return *this; } // optional, defaults to all true CustomGradMaker& output_captured(size_t i, bool v) { target.m_output_attrs.at(i).captured = v; return *this; } void finalize() { mgb_assert(output_size_set); init_input_has_grad(); } }; namespace { ValueRef get_shape(ValueRef x) { static auto op = GetVarShape::make(); return imperative::apply(*op, x)[0]; } ValueRef reduce_to(ValueRef x, ValueRef s) { static auto op = Reduce::make(); return imperative::apply(*op, x, s)[0]; } ValueRef reshape_to(ValueRef x, ValueRef s) { static auto op = Reshape::make(); return imperative::apply(*op, x, s)[0]; } ValueRef broadcast_to(ValueRef x, ValueRef s) { static auto op = Broadcast::make(); return imperative::apply(*op, x, s)[0]; } ValueRef make_empty_tensor( CompNodeValue::ref_t device, ValueRef shape, DTypeValue::ref_t dtype) { HostTensorStorage storage(*device); storage.ensure_size(dtype->size()); std::memset(storage.ptr(), 0, dtype->size()); auto t = imperative::apply( CreateTensor(CreateTensor::Unique, *device, *dtype, ValueShape()), HostStorage::make(storage))[0]; auto res = broadcast_to(t, shape); return res; } std::optional matrix_mul_grad_rule( const OpDef& op, Span inputs, Span inputs_require_grad, CustomBackward& backward) { auto&& matmul = op.cast_final_safe(); size_t dimA = matmul.dimA; size_t dimB = matmul.dimB; auto&& param = matmul.param(); auto&& policy = matmul.policy(); mgb_assert(inputs.size() == 2); std::array inps, input_shapes; for (size_t i = 0; i < 2; ++i) { if (inputs_require_grad[i ^ 1]) { inps[i] = inputs[i]; input_shapes[i] = get_shape(inputs[i]); } } auto maker = CustomGradMaker(backward, inputs.size()); maker.output_size(1).output_captured(0, false); maker.backward([inps_ = std::move(inps), input_shapes_ = std::move(input_shapes), param, policy, dimA, dimB](Span grads) { mgb_assert(grads.size() == 1); ValueRef grad = grads[0]; SmallVector ret(2); if (!grad) { return ret; } size_t dimG = std::max(dimA, dimB); if (inps_[1]) { if (param.transposeA) { // A^T(2) @ B(2) = G(2), A'(2) = B'(2) @ G'^T(2) -> MatrixMul auto&& grad_op = MatrixMul::make( param.transposeB, true, param.compute_mode, param.format, policy.strategy, policy.workspace_limit, dimB, dimG); ret[0] = imperative::apply(*grad_op, inps_[1], grad)[0]; } else { // A(>=2) @ B(2) = G(>=2), A'(>=2) = G'(>=2) @ B(2) -> MatrixMul auto&& grad_op = MatrixMul::make( false, !param.transposeB, param.compute_mode, param.format, policy.strategy, policy.workspace_limit, dimG, dimB); ret[0] = imperative::apply(*grad_op, grad, inps_[1])[0]; } } if (inps_[0]) { if (param.transposeB) { // A(>=2) @ B^T(2) = G(>=2), B'(2) = G'^T(>=2) @ A(>=2) -> MatrixMul // (specialized) auto&& grad_op = MatrixMul::make( true, param.transposeA, param.compute_mode, param.format, policy.strategy, policy.workspace_limit, dimG, dimA); ret[1] = imperative::apply(*grad_op, grad, inps_[0])[0]; } else { // A(>=2) @ B(2) = G(>=2), B'(2) = G'(>=2) @ A(>=2) -> MatrixMul // (specialized) auto&& grad_op = MatrixMul::make( !param.transposeA, false, param.compute_mode, param.format, policy.strategy, policy.workspace_limit, dimA, dimG); ret[1] = imperative::apply(*grad_op, inps_[0], grad)[0]; } } return ret; }); maker.finalize(); return imperative::apply(ApplyOp(op), inputs); } std::optional batched_matrix_mul_grad_rule( const OpDef& op, Span inputs, Span inputs_require_grad, CustomBackward& backward) { auto&& bmm = op.cast_final_safe(); size_t dimA = bmm.dimA; size_t dimB = bmm.dimB; auto&& param = bmm.param(); auto&& policy = bmm.policy(); mgb_assert(inputs.size() == 2); std::array inps, input_shapes; for (size_t i = 0; i < 2; ++i) { if (inputs_require_grad[i ^ 1]) { inps[i] = inputs[i]; input_shapes[i] = get_shape(inputs[i]); } } auto maker = CustomGradMaker(backward, inputs.size()); maker.output_size(1).output_captured(0, false); maker.backward([inps_ = std::move(inps), input_shapes_ = std::move(input_shapes), param, policy, dimA, dimB](Span grads) { mgb_assert(grads.size() == 1); ValueRef grad = grads[0]; SmallVector ret(2); if (!grad) { return ret; } size_t dimG = std::max(dimA, dimB); if (inps_[1]) { if (param.transposeA) { auto&& grad_op = BatchedMatrixMul::make( param.transposeB, true, param.compute_mode, param.format, policy.strategy, policy.workspace_limit, dimB, dimG); ret[0] = imperative::apply(*grad_op, inps_[1], grad)[0]; } else { auto&& grad_op = BatchedMatrixMul::make( false, !param.transposeB, param.compute_mode, param.format, policy.strategy, policy.workspace_limit, dimG, dimB); ret[0] = imperative::apply(*grad_op, grad, inps_[1])[0]; } if (dimG != dimA) { ret[0] = reduce_to(ret[0], input_shapes_[0]); } } if (inps_[0]) { if (param.transposeB) { auto&& grad_op = BatchedMatrixMul::make( true, param.transposeA, param.compute_mode, param.format, policy.strategy, policy.workspace_limit, dimG, dimA); ret[1] = imperative::apply(*grad_op, grad, inps_[0])[0]; } else { auto&& grad_op = BatchedMatrixMul::make( !param.transposeA, false, param.compute_mode, param.format, policy.strategy, policy.workspace_limit, dimA, dimG); ret[1] = imperative::apply(*grad_op, inps_[0], grad)[0]; } if (dimG != dimB) { ret[1] = reduce_to(ret[1], input_shapes_[1]); } } return ret; }); maker.finalize(); return imperative::apply(ApplyOp(op), inputs); } std::optional elemwise_grad_rule( const OpDef& op, Span inputs, Span inputs_require_grad, CustomBackward& backward) { auto& elemwise = op.cast_final_safe(); if (elemwise.mode != Elemwise::Mode::ADD) { return {}; } mgb_assert(inputs.size() == 2); std::array input_shapes; for (size_t i = 0; i < 2; ++i) { if (inputs_require_grad[i]) { input_shapes[i] = get_shape(inputs[i]); } } auto maker = CustomGradMaker(backward, inputs.size()); maker.output_size(1).output_captured(0, false); maker.backward([shapes = std::move(input_shapes)](Span grads) { mgb_assert(grads.size() == 1); ValueRef grad = grads[0]; SmallVector ret(2); if (!grad) { return ret; } for (size_t i = 0; i < 2; ++i) { if (shapes[i]) { ret[i] = reduce_to(grad, shapes[i]); } } return ret; }); maker.finalize(); return imperative::apply(ApplyOp(op), inputs); } std::optional reshape_grad_rule( const OpDef& op, Span inputs, Span inputs_require_grad, CustomBackward& backward) { mgb_assert(inputs.size() == 1 || inputs.size() == 2); size_t nr_inp = inputs.size(); std::array input_shapes; for (size_t i = 0; i < nr_inp; ++i) { if (inputs_require_grad[i]) { input_shapes[i] = get_shape(inputs[i]); } } auto maker = CustomGradMaker(backward, inputs.size()); maker.output_size(1).output_captured(0, false); maker.backward([shapes = std::move(input_shapes), nr_inp](Span grads) { mgb_assert(grads.size() == 1); ValueRef grad = grads[0]; SmallVector ret(nr_inp); if (!grad) { return ret; } for (size_t i = 0; i < nr_inp; ++i) { if (shapes[i]) { ret[i] = reshape_to(grad, shapes[i]); } } return ret; }); maker.finalize(); return imperative::apply(ApplyOp(op), inputs); } std::optional broadcast_grad_rule( const OpDef& op, Span inputs, Span inputs_require_grad, CustomBackward& backward) { mgb_assert(inputs.size() == 1 || inputs.size() == 2); size_t nr_inp = inputs.size(); std::array input_shapes; for (size_t i = 0; i < nr_inp; ++i) { if (inputs_require_grad[i]) { input_shapes[i] = get_shape(inputs[i]); } } auto maker = CustomGradMaker(backward, inputs.size()); maker.output_size(1).output_captured(0, false); maker.backward([shapes = std::move(input_shapes), nr_inp](Span grads) { mgb_assert(grads.size() == 1); ValueRef grad = grads[0]; SmallVector ret(nr_inp); if (!grad) { return ret; } for (size_t i = 0; i < nr_inp; ++i) { if (shapes[i]) { ret[i] = reduce_to(grad, shapes[i]); } } return ret; }); maker.finalize(); return imperative::apply(ApplyOp(op), inputs); } std::optional subtensor_grad_rule( const OpDef& op, Span inputs, Span inputs_require_grad, CustomBackward& backward) { auto&& subtensor = op.cast_final_safe(); auto&& grad_op = SetSubtensor::make(subtensor.items); SmallVector inputs2; if (inputs_require_grad[0]) { inputs2.push_back(get_shape(inputs[0])); for (size_t i = 1; i < inputs.size(); ++i) { inputs2.push_back(inputs[i]); } } CompNodeValue::ref_t device = inputs[0].device(); auto get_subtensor_index = [&](int idx) { HostTensorStorage storage(*device); storage.ensure_size(dtype::Int32().size()); auto* ptr = reinterpret_cast(storage.ptr()); ptr[0] = idx; return imperative::apply( CreateTensor( CreateTensor::Unique, *device, dtype::Int32(), ValueShape({1})), HostStorage::make(storage))[0]; }; auto slice_items = subtensor.slice_items; auto items = subtensor.items; for (int i = 0; i < slice_items.size(); i++) { auto&& [axis, b_flag, e_flag, s_flag, idx_flag] = items[i]; auto&& [b_val, e_val, s_val, ax_val] = slice_items[i]; if (b_flag) { inputs2.push_back(get_subtensor_index(b_val)); }; if (e_flag) { inputs2.push_back(get_subtensor_index(e_val)); }; if (s_flag) { inputs2.push_back(get_subtensor_index(s_val)); }; if (idx_flag) { inputs2.push_back(get_subtensor_index(ax_val)); }; }; auto maker = CustomGradMaker(backward, inputs.size()); maker.output_size(1).output_captured(0, false); maker.backward([inputs = std::move(inputs2), grad_op_ = std::move(grad_op)](Span grads) { mgb_assert(grads.size() == 1); ValueRef grad = grads[0]; SmallVector ret(1); if (grad && inputs[0]) { ValueRefList args_(inputs.size() + 1); auto&& zeros = make_empty_tensor(grad.device(), inputs[0], grad.dtype()); args_[0] = zeros; args_[1] = grad; for (size_t i = 1; i < inputs.size(); ++i) { args_[i + 1] = inputs[i]; } ret[0] = imperative::apply(ApplyOp(*grad_op_), args_)[0]; } return ret; }); maker.finalize(); return imperative::apply(ApplyOp(op), inputs); } std::optional indexingMultiAxisVec_grad_rule( const OpDef& op, Span inputs, Span inputs_require_grad, CustomBackward& backward) { auto&& indexingMultiAxisVec = op.cast_final_safe(); auto&& grad_op = IndexingIncrMultiAxisVec::make(indexingMultiAxisVec.items); SmallVector inputs2; if (inputs_require_grad[0]) { inputs2.push_back(get_shape(inputs[0])); for (size_t i = 1; i < inputs.size(); ++i) { inputs2.push_back(inputs[i]); } } auto maker = CustomGradMaker(backward, inputs.size()); maker.output_size(1).output_captured(0, false); maker.backward([inputs = std::move(inputs2), grad_op_ = std::move(grad_op)](Span grads) { mgb_assert(grads.size() == 1); ValueRef grad = grads[0]; SmallVector ret(1); if (grad && inputs[0]) { ValueRefList args_(inputs.size() + 1); auto&& zeros = make_empty_tensor(grad.device(), inputs[0], grad.dtype()); args_[0] = zeros; args_[1] = grad; for (size_t i = 1; i < inputs.size(); ++i) { args_[i + 1] = inputs[i]; } ret[0] = imperative::apply(ApplyOp(*grad_op_), args_)[0]; } return ret; }); maker.finalize(); return imperative::apply(ApplyOp(op), inputs); } std::optional reduce_grad_rule( const OpDef& op, Span inputs, Span inputs_require_grad, CustomBackward& backward) { auto& reduce = op.cast_final_safe(); if (reduce.mode != Reduce::Mode::SUM) { return {}; } auto axis = reduce.axis; if (inputs.size() != 1 || axis == INT_MAX) { return {}; } std::array input_shapes; if (inputs_require_grad[0]) { input_shapes[0] = get_shape(inputs[0]); } if (axis < 0) { axis = (*inputs[0].shape()).ndim + axis; } auto maker = CustomGradMaker(backward, inputs.size()); auto keepdim = reduce.keepdim || axis == INT_MAX; maker.output_size(1).output_captured(0, false); maker.backward( [shapes = std::move(input_shapes), axis, keepdim](Span grads) { mgb_assert(grads.size() == 1); ValueRef grad = grads[0]; if (!keepdim && grad) { auto&& grad_op = AddAxis::make(std::vector({axis})); grad = imperative::apply(*grad_op, grad)[0]; } SmallVector ret(1); if (grad && shapes[0]) { ret[0] = broadcast_to(grad, shapes[0]); } return ret; }); maker.finalize(); return imperative::apply(ApplyOp(op), inputs); } std::optional addAxis_grad_rule( const OpDef& op, Span inputs, Span inputs_require_grad, CustomBackward& backward) { auto&& addAxis = op.cast_final_safe(); mgb_assert(inputs.size() == 1); bool flag = inputs_require_grad[0]; auto&& grad_op = RemoveAxis::make(addAxis.axis); std::sort(grad_op->axis.begin(), grad_op->axis.end(), std::greater()); auto maker = CustomGradMaker(backward, inputs.size()); maker.output_size(1).output_captured(0, false); maker.backward([grad_op_ = std::move(grad_op), flag_ = flag](Span grads) { mgb_assert(grads.size() == 1); ValueRef grad = grads[0]; SmallVector ret(1); if (grad && flag_) { ret[0] = imperative::apply(*grad_op_, grad)[0]; } return ret; }); maker.finalize(); return imperative::apply(op, inputs); } std::optional removeAxis_grad_rule( const OpDef& op, Span inputs, Span inputs_require_grad, CustomBackward& backward) { auto&& removeAxis = op.cast_final_safe(); mgb_assert(inputs.size() == 1); bool flag = inputs_require_grad[0]; auto&& grad_op = AddAxis::make(removeAxis.axis); std::sort(grad_op->axis.begin(), grad_op->axis.end()); auto maker = CustomGradMaker(backward, inputs.size()); maker.output_size(1).output_captured(0, false); maker.backward([grad_op_ = std::move(grad_op), flag_ = flag](Span grads) { mgb_assert(grads.size() == 1); ValueRef grad = grads[0]; SmallVector ret(1); if (grad && flag_) { ret[0] = imperative::apply(*grad_op_, grad)[0]; } return ret; }); maker.finalize(); return imperative::apply(op, inputs); } std::optional pixelShuffle_grad_rule( const OpDef& op, Span inputs, Span inputs_require_grad, CustomBackward& backward) { auto&& pixelShuffle = op.cast_final_safe(); mgb_assert(inputs.size() == 1); bool flag = inputs_require_grad[0]; auto&& grad_op = PixelShuffleBackward::make(pixelShuffle.factor); auto maker = CustomGradMaker(backward, inputs.size()); maker.output_size(1).output_captured(0, false); maker.backward([grad_op_ = std::move(grad_op), flag_ = flag](Span grads) { mgb_assert(grads.size() == 1); ValueRef grad = grads[0]; SmallVector ret(1); if (grad && flag_) { ret[0] = imperative::apply(*grad_op_, grad)[0]; } return ret; }); maker.finalize(); return imperative::apply(op, inputs); } std::optional indexing_grad_rule( const OpDef& op, Span inputs, Span inputs_require_grad, CustomBackward& backward) { auto&& indexing = op.cast_final_safe(); mgb_assert(inputs.size() == 2); bool flag = inputs_require_grad[0]; auto&& grad_op = IndexingSetOneHot::make(indexing.axis, indexing.ndim); SmallVector inputs2; if (flag) { inputs2.push_back(get_shape(inputs[0])); for (size_t i = 1; i < inputs.size(); ++i) { inputs2.push_back(inputs[i]); } } auto maker = CustomGradMaker(backward, inputs.size()); maker.output_size(1).output_captured(0, false); maker.backward([inputs = std::move(inputs2), grad_op_ = std::move(grad_op)](Span grads) { mgb_assert(grads.size() == 1); ValueRef grad = grads[0]; SmallVector ret(1); if (grad && inputs[0]) { ValueRefList args_(inputs.size() + 1); auto&& zeros = make_empty_tensor(grad.device(), inputs[0], grad.dtype()); args_[0] = zeros; args_[1] = inputs[1]; args_[2] = grads[0]; ret[0] = imperative::apply(*grad_op_, args_)[0]; } return ret; }); maker.finalize(); return imperative::apply(op, inputs); } std::optional indexing_set_one_hot_grad_rule( const OpDef& op, Span inputs, Span inputs_require_grad, CustomBackward& backward) { auto&& indexingSetOneHot = op.cast_final_safe(); mgb_assert(inputs.size() == 3); SmallVector inputs2; inputs2.push_back(get_shape(inputs[0])); inputs2.push_back(inputs[1]); inputs2.push_back(get_shape(inputs[2])); auto maker = CustomGradMaker(backward, inputs.size()); maker.output_size(1).output_captured(0, false); maker.backward([inputs = std::move(inputs2), &indexingSetOneHot](Span grads) { mgb_assert(grads.size() == 1); ValueRef grad = grads[0]; SmallVector ret(3); if (!grad) { return ret; } if (inputs[0]) { auto&& grad_op = IndexingSetOneHot::make( indexingSetOneHot.axis, indexingSetOneHot.ndim); ValueRefList args_(inputs.size()); auto&& zeros = make_empty_tensor(grad.device(), inputs[2], grad.dtype()); args_[0] = grads[0]; args_[1] = inputs[1]; args_[2] = zeros; ret[0] = imperative::apply(*grad_op, args_)[0]; } if (inputs[2]) { auto&& grad_op = IndexingOneHot::make( indexingSetOneHot.axis, indexingSetOneHot.ndim); ValueRefList args_(inputs.size() - 1); args_[0] = grads[0]; args_[1] = inputs[1]; ret[2] = imperative::apply(*grad_op, args_)[0]; } return ret; }); maker.finalize(); return imperative::apply(op, inputs); } std::optional fastpathcopy_grad_rule( const OpDef& op, Span inputs, Span inputs_require_grad, CustomBackward& backward) { mgb_assert(inputs.size() == 1); auto maker = CustomGradMaker(backward, inputs.size()); maker.output_size(1).output_captured(0, false); maker.backward([](Span grads) { mgb_assert(grads.size() == 1); ValueRef grad = grads[0]; SmallVector ret(1); if (grad) { ret[0] = grad; } return ret; }); maker.finalize(); return imperative::apply(op, inputs); } std::optional warp_affine_grad_rule( const OpDef& op, Span inputs, Span inputs_require_grad, CustomBackward& backward) { auto&& warp_affine = op.cast_final_safe(); auto&& param = warp_affine.param(); mgb_assert(inputs.size() == 3); SmallVector inps; if (inputs_require_grad[0] || inputs_require_grad[1]) { for (size_t i = 0; i < inputs.size(); ++i) { inps.push_back(inputs[i]); } } auto maker = CustomGradMaker(backward, inputs.size()); maker.output_size(1).output_captured(0, false); maker.backward([inputs = std::move(inps), &warp_affine, param](Span grads) { mgb_assert(grads.size() == 1); ValueRef grad = grads[0]; SmallVector ret(2); if (!grad) { return ret; } CompNodeValue::ref_t device = inputs[0].device(); DTypeValue::ref_t dtype = inputs[0].dtype(); HostTensorStorage storage(*device); storage.ensure_size(3 * (dtype->size())); auto* ptr = reinterpret_cast(storage.ptr()); ptr[0] = 0; ptr[1] = 0; ptr[2] = 1; auto t = imperative::apply( CreateTensor( CreateTensor::Unique, *device, dtype::Float32(), ValueShape({1, 1, 3})), HostStorage::make(storage))[0]; auto mat = inputs[1]; auto&& concat = Concat::make(); concat->axis = 1; mat = imperative::apply(*concat, inputs[1], t)[0]; if (inputs[0]) { auto&& grad_op = WarpPerspectiveBackwardData::make( param.imode, param.border_mode, param.format, param.border_val); ValueRefList args_(inputs.size()); args_[0] = mat; args_[1] = grads[0]; args_[2] = inputs[0]; ret[0] = imperative::apply(*grad_op, args_)[0]; } if (inputs[1]) { auto&& grad_op = WarpPerspectiveBackwardMat::make( param.imode, param.border_mode, param.format, param.border_val); ValueRefList args_(inputs.size()); args_[0] = inputs[0]; args_[1] = mat; args_[2] = grads[0]; ret[1] = imperative::apply(*grad_op, args_)[0]; std::vector> items; std::vector> slice_items; items.push_back(std::make_tuple(1, true, true, false, false)); auto&& subtensor = Subtensor::make(items, slice_items); CompNodeValue::ref_t device = inputs[0].device(); DTypeValue::ref_t dtype = inputs[0].dtype(); int start_idx = 0; int stop_idx = 2; auto get_subtensor_index = [&](int idx) { HostTensorStorage storage(*device); storage.ensure_size(dtype::Int32().size()); auto* ptr = reinterpret_cast(storage.ptr()); ptr[0] = idx; return imperative::apply( CreateTensor( CreateTensor::Unique, *device, dtype::Int32(), ValueShape({1})), HostStorage::make(storage))[0]; }; auto start = get_subtensor_index(start_idx); auto stop = get_subtensor_index(stop_idx); auto data = ret[1]; mgb_assert(data); ret[1] = imperative::apply(*subtensor, data, start, stop)[0]; } return ret; }); maker.finalize(); return imperative::apply(ApplyOp(op), inputs); } struct Init { Init() { CustomBackward::register_grad_rule(Elemwise::typeinfo(), elemwise_grad_rule); CustomBackward::register_grad_rule(Reshape::typeinfo(), reshape_grad_rule); CustomBackward::register_grad_rule(Broadcast::typeinfo(), broadcast_grad_rule); CustomBackward::register_grad_rule(Subtensor::typeinfo(), subtensor_grad_rule); CustomBackward::register_grad_rule( IndexingMultiAxisVec::typeinfo(), indexingMultiAxisVec_grad_rule); CustomBackward::register_grad_rule(Reduce::typeinfo(), reduce_grad_rule); CustomBackward::register_grad_rule(AddAxis::typeinfo(), addAxis_grad_rule); CustomBackward::register_grad_rule( RemoveAxis::typeinfo(), removeAxis_grad_rule); CustomBackward::register_grad_rule( IndexingOneHot::typeinfo(), indexing_grad_rule); CustomBackward::register_grad_rule( IndexingSetOneHot::typeinfo(), indexing_set_one_hot_grad_rule); CustomBackward::register_grad_rule( FastpathCopy::typeinfo(), fastpathcopy_grad_rule); CustomBackward::register_grad_rule( PixelShuffle::typeinfo(), pixelShuffle_grad_rule); CustomBackward::register_grad_rule(MatrixMul::typeinfo(), matrix_mul_grad_rule); CustomBackward::register_grad_rule( BatchedMatrixMul::typeinfo(), batched_matrix_mul_grad_rule); CustomBackward::register_grad_rule( WarpAffine::typeinfo(), warp_affine_grad_rule); } } _; } // namespace } // namespace mgb::imperative::python