From 9246b93c6e6281433aca81c1bb893dd2d31683df Mon Sep 17 00:00:00 2001 From: Sanbu <96160062+sanbuphy@users.noreply.github.com> Date: Thu, 13 Apr 2023 15:41:14 +0800 Subject: [PATCH] Support static graph code-gen for temporal_shift (#52686) --- paddle/fluid/operators/temporal_shift_op.cc | 164 -------------------- paddle/phi/api/yaml/backward.yaml | 11 ++ paddle/phi/api/yaml/legacy_backward.yaml | 10 -- paddle/phi/api/yaml/legacy_ops.yaml | 9 -- paddle/phi/api/yaml/op_compat.yaml | 7 + paddle/phi/api/yaml/ops.yaml | 10 ++ paddle/phi/ops/compat/temporal_shift_sig.cc | 39 ----- python/paddle/nn/functional/extension.py | 36 ++++- 8 files changed, 61 insertions(+), 225 deletions(-) delete mode 100644 paddle/fluid/operators/temporal_shift_op.cc delete mode 100644 paddle/phi/ops/compat/temporal_shift_sig.cc diff --git a/paddle/fluid/operators/temporal_shift_op.cc b/paddle/fluid/operators/temporal_shift_op.cc deleted file mode 100644 index 932e6c1f5ea..00000000000 --- a/paddle/fluid/operators/temporal_shift_op.cc +++ /dev/null @@ -1,164 +0,0 @@ -/* Copyright (c) 2019 PaddlePaddle Authors. All Rights Reserve. - Licensed under the Apache License, Version 2.0 (the "License"); - you may not use this file except in compliance with the License. - You may obtain a copy of the License at - http://www.apache.org/licenses/LICENSE-2.0 - Unless required by applicable law or agreed to in writing, software - distributed under the License is distributed on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - See the License for the specific language governing permissions and - limitations under the License. */ - -#include "paddle/fluid/operators/temporal_shift_op.h" - -#include -#include -#include - -#include "paddle/fluid/framework/infershape_utils.h" -#include "paddle/fluid/framework/op_registry.h" -#include "paddle/phi/core/infermeta_utils.h" -#include "paddle/phi/infermeta/unary.h" - -namespace paddle { -namespace operators { - -class TemporalShiftOp : public framework::OperatorWithKernel { - public: - using framework::OperatorWithKernel::OperatorWithKernel; - - protected: - phi::KernelKey GetExpectedKernelType( - const framework::ExecutionContext& ctx) const override { - return phi::KernelKey(OperatorWithKernel::IndicateVarDataType(ctx, "X"), - ctx.GetPlace()); - } -}; - -class TemporalShiftOpMaker : public framework::OpProtoAndCheckerMaker { - public: - void Make() override { - AddInput("X", - "The input tensor of temporal shift operator. " - "This is a 4-D tensor with shape of [N*T, C, H, W] " - "or [N*T, H, W, C]. " - "While N is the batch size, T is the temporal segment " - "number, C is the channel number, H is the height of " - "features and W is the width of features. " - "The data type is float16, float32 and float64"); - AddOutput("Out", - "The output tensor of temporal shift operator. " - "This is a 4-D tensor in the same shape with Input(X)."); - - AddAttr("seg_num", - "The temporal segment number, this should be a positive " - "integer."); - AddAttr( - "shift_ratio", - "The shift ratio of the channels, the first :attr:`shift_ratio` part " - "of channels will be shifted by -1 along the temporal dimension, " - "and the second :attr:`shift_ratio` part of channels will be shifted " - "by 1 along the temporal dimension. :attr:`shift_ratio` should be in " - "range [0, 0.5]. Default 0.25.") - .SetDefault(0.25); - AddAttr( - "data_format", - "(string, default NCHW) Only used in " - "an optional string from: \"NHWC\", \"NCHW\". " - "Specify that the data format of the input and output data is " - "channel_first or channel_last.") - .SetDefault("NCHW"); - - AddComment(R"DOC( - This operator calculates the temporal shifting features for Input(X). - - Input(X) should be in shape of [N*T, C, H, W] or [N*T, H, W, C], while - N is the batch size, T is the temporal segment number specified by - :attr:`seg_num`, C is the channel number, H and W is the height and - width of features. - - Temporal Shifting is calculated as follows when data format is NCHW: - - Step 1: Reshape Input(X) to [N, T, C, H, W]. - - Step 2: Pad 0 to reshaping result in the 2nd(T) dimension with - padding width as 1 on each side, padding result will be in shape - of [N, T+2, C, H, W]. - - Step 3: Assume :attr:`shift_ratio` is :math:`1/4`, slice padding - result as follows: - - $$ - slice1 = x[:, :T, :C/4, :, :] - $$ - $$ - slice2 = x[:, 2:T+2, C/4:C/2, :, :] - $$ - $$ - slice3 = x[:, 1:T+1, C/2:, :, :] - $$ - - Step 4: Concatenate three slices along the 3rd(C) dimension and - reshape result to [N*T, C, H, W]. - - For details of temporal shifting, please refer to paper: - `Temporal Shift Module `_ . - - )DOC"); - } -}; - -class TemporalShiftOpGrad : public framework::OperatorWithKernel { - public: - using framework::OperatorWithKernel::OperatorWithKernel; - - protected: - void InferShape(framework::InferShapeContext* ctx) const override { - if (ctx->HasOutput(framework::GradVarName("X"))) { - ctx->SetOutputDim(framework::GradVarName("X"), - ctx->GetInputDim(framework::GradVarName("Out"))); - } - } - - phi::KernelKey GetExpectedKernelType( - const framework::ExecutionContext& ctx) const override { - return phi::KernelKey(OperatorWithKernel::IndicateVarDataType( - ctx, framework::GradVarName("Out")), - ctx.GetPlace()); - } -}; - -template -class TemporalShiftGradOpMaker : public framework::SingleGradOpMaker { - public: - using framework::SingleGradOpMaker::SingleGradOpMaker; - - protected: - void Apply(GradOpPtr op) const override { - op->SetType("temporal_shift_grad"); - op->SetInput(framework::GradVarName("Out"), this->OutputGrad("Out")); - op->SetOutput(framework::GradVarName("X"), this->InputGrad("X")); - op->SetAttrMap(this->Attrs()); - } -}; - -} // namespace operators -} // namespace paddle - -namespace ops = paddle::operators; -DECLARE_INFER_SHAPE_FUNCTOR(temporal_shift, - TemporalShiftInferShapeFunctor, - PD_INFER_META(phi::TemporalShiftInferMeta)); -REGISTER_OPERATOR(temporal_shift, - ops::TemporalShiftOp, - ops::TemporalShiftOpMaker, - ops::TemporalShiftGradOpMaker, - ops::TemporalShiftGradOpMaker, - TemporalShiftInferShapeFunctor); -REGISTER_OPERATOR(temporal_shift_grad, ops::TemporalShiftOpGrad); -REGISTER_OP_CPU_KERNEL(temporal_shift, - ops::TemporalShiftKernel, - ops::TemporalShiftKernel); -REGISTER_OP_CPU_KERNEL(temporal_shift_grad, - ops::TemporalShiftGradKernel, - ops::TemporalShiftGradKernel); diff --git a/paddle/phi/api/yaml/backward.yaml b/paddle/phi/api/yaml/backward.yaml index 6415dc4fc8c..6c0184d7d0c 100644 --- a/paddle/phi/api/yaml/backward.yaml +++ b/paddle/phi/api/yaml/backward.yaml @@ -1863,6 +1863,17 @@ inplace : (grad_x_grad_forward -> grad_out_forward_grad) optional : grad_out_new_grad, grad_out_grad_grad +- backward_op : temporal_shift_grad + forward : temporal_shift(Tensor x, int seg_num, float shift_ratio = 0.25f, str data_format = "NCHW") -> Tensor(out) + args : (Tensor out_grad, int seg_num, float shift_ratio, str data_format) + output : Tensor(x_grad) + infer_meta : + func : UnchangedInferMeta + param : [out_grad] + kernel : + func : temporal_shift_grad + data_type : out_grad + - backward_op : thresholded_relu_grad forward : thresholded_relu (Tensor x, float threshold) -> Tensor(out) args : (Tensor x, Tensor out_grad, float threshold) diff --git a/paddle/phi/api/yaml/legacy_backward.yaml b/paddle/phi/api/yaml/legacy_backward.yaml index 178e8fcda94..d810ad8bd9f 100755 --- a/paddle/phi/api/yaml/legacy_backward.yaml +++ b/paddle/phi/api/yaml/legacy_backward.yaml @@ -1031,16 +1031,6 @@ data_type : out_grad optional : reserve_space -- backward_op : temporal_shift_grad - forward : temporal_shift(Tensor x, int seg_num, float shift_ratio, str data_format_str) -> Tensor(out) - args : (Tensor out_grad, int seg_num, float shift_ratio, str data_format_str) - output : Tensor(x_grad) - infer_meta : - func : UnchangedInferMeta - param : [out_grad] - kernel : - func : temporal_shift_grad - - backward_op : tile_double_grad forward : tile_grad (Tensor x, Tensor grad_out, IntArray repeat_times) -> Tensor(grad_x) args : (Tensor grad_x_grad, IntArray repeat_times) diff --git a/paddle/phi/api/yaml/legacy_ops.yaml b/paddle/phi/api/yaml/legacy_ops.yaml index 9ffbaa07bd0..05abaf30445 100755 --- a/paddle/phi/api/yaml/legacy_ops.yaml +++ b/paddle/phi/api/yaml/legacy_ops.yaml @@ -1229,15 +1229,6 @@ backward : sync_batch_norm_grad inplace : (mean -> mean_out), (variance -> variance_out) -- op : temporal_shift - args : (Tensor x, int seg_num, float shift_ratio, str data_format_str) - output : Tensor - infer_meta : - func : TemporalShiftInferMeta - kernel : - func : temporal_shift - backward : temporal_shift_grad - - op : tile args : (Tensor x, IntArray repeat_times = {}) output : Tensor(out) diff --git a/paddle/phi/api/yaml/op_compat.yaml b/paddle/phi/api/yaml/op_compat.yaml index 471e257a6af..ad183913832 100644 --- a/paddle/phi/api/yaml/op_compat.yaml +++ b/paddle/phi/api/yaml/op_compat.yaml @@ -2347,3 +2347,10 @@ x : X outputs : out : Out + +- op: temporal_shift + backward: temporal_shift_grad + inputs : + x : X + outputs : + out : Out diff --git a/paddle/phi/api/yaml/ops.yaml b/paddle/phi/api/yaml/ops.yaml index 328b0211450..b482fcbcdcc 100644 --- a/paddle/phi/api/yaml/ops.yaml +++ b/paddle/phi/api/yaml/ops.yaml @@ -1885,6 +1885,16 @@ func : tanh_shrink backward : tanh_shrink_grad +- op : temporal_shift + args : (Tensor x, int seg_num, float shift_ratio = 0.25f, str data_format = "NCHW") + output : Tensor(out) + infer_meta : + func : TemporalShiftInferMeta + kernel : + func : temporal_shift + data_type : x + backward : temporal_shift_grad + - op : thresholded_relu args : (Tensor x, float threshold = 1.0) output : Tensor diff --git a/paddle/phi/ops/compat/temporal_shift_sig.cc b/paddle/phi/ops/compat/temporal_shift_sig.cc deleted file mode 100644 index a6eed22716c..00000000000 --- a/paddle/phi/ops/compat/temporal_shift_sig.cc +++ /dev/null @@ -1,39 +0,0 @@ -// Copyright (c) 2022 PaddlePaddle Authors. All Rights Reserved. -// -// Licensed under the Apache License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at -// -// http://www.apache.org/licenses/LICENSE-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. - -#include "paddle/phi/core/compat/op_utils.h" - -namespace phi { - -KernelSignature TemporalShiftOpArgumentMapping( - const ArgumentMappingContext& ctx) { - return KernelSignature("temporal_shift", - {"X"}, - {"seg_num", "shift_ratio", "data_format"}, - {"Out"}); -} - -KernelSignature TemporalShiftGradOpArgumentMapping( - const ArgumentMappingContext& ctx) { - return KernelSignature("temporal_shift_grad", - {"Out@GRAD"}, - {"seg_num", "shift_ratio", "data_format"}, - {"X@GRAD"}); -} - -} // namespace phi - -PD_REGISTER_ARG_MAPPING_FN(temporal_shift, phi::TemporalShiftOpArgumentMapping); -PD_REGISTER_ARG_MAPPING_FN(temporal_shift_grad, - phi::TemporalShiftGradOpArgumentMapping); diff --git a/python/paddle/nn/functional/extension.py b/python/paddle/nn/functional/extension.py index 67bc16ccddc..c0e86267f34 100644 --- a/python/paddle/nn/functional/extension.py +++ b/python/paddle/nn/functional/extension.py @@ -28,7 +28,6 @@ from ...fluid.framework import in_dygraph_mode from ...fluid.layer_helper import LayerHelper from ...framework import convert_np_dtype_to_dtype_, core from ...tensor.creation import assign -from ...tensor.layer_function_generator import templatedoc __all__ = [] @@ -338,13 +337,44 @@ def gather_tree(ids, parents): return out -@templatedoc() def temporal_shift(x, seg_num, shift_ratio=0.25, name=None, data_format="NCHW"): """ **Temporal Shift Operator** - ${comment} + Calculate the temporal shifting features for Input(X). + + Input(X) should be in shape of [N*T, C, H, W] or [N*T, H, W, C], while + N is the batch size, T is the temporal segment number specified by + :attr:`seg_num`, C is the channel number, H and W is the height and + width of features. + + Temporal Shifting is calculated as follows when data format is NCHW: + + Step 1: Reshape Input(X) to [N, T, C, H, W]. + + Step 2: Pad 0 to reshaping result in the 2nd(T) dimension with + padding width as 1 on each side, padding result will be in shape + of [N, T+2, C, H, W]. + + Step 3: Assume :attr:`shift_ratio` is :math:`1/4`, slice padding + result as follows: + + $$ + slice1 = x[:, :T, :C/4, :, :] + $$ + $$ + slice2 = x[:, 2:T+2, C/4:C/2, :, :] + $$ + $$ + slice3 = x[:, 1:T+1, C/2:, :, :] + $$ + + Step 4: Concatenate three slices along the 3rd(C) dimension and + reshape result to [N*T, C, H, W]. + + For details of temporal shifting, please refer to paper: + `Temporal Shift Module `_ . Args: x(Tensor): ${x_comment} -- GitLab