|  | #include "caffe2/operators/conv_transpose_op.h" | 
|  | #include "caffe2/operators/conv_transpose_op_impl.h" | 
|  |  | 
|  | namespace caffe2 { | 
|  |  | 
|  | REGISTER_CPU_OPERATOR(ConvTranspose, ConvTransposeOp<float, CPUContext>); | 
|  |  | 
|  | OPERATOR_SCHEMA(ConvTranspose) | 
|  | .NumInputs(2, 3) | 
|  | .NumOutputs(1) | 
|  | .SetDoc(R"DOC( | 
|  | The ConvTranspose op takes an input data tensor $X$, an input weight tensor $filter$, and optionally an input bias tensor $bias$. It then computes the transposed convolution, sometimes referred to as deconvolution, and produces a single output tensor $Y$. The hyperparameters of the op such as kernel size, stride, and padding are specified as args. At each stride, the filter is deconvolved with a subset of $X$ and the $bias$ is added. This is done throughout the input data until the output computation is complete. | 
|  |  | 
|  | The output shapes are computed as follows. The number of channels in the output feature map is the number of kernels specified in the filter blob. The spatial height and width are computed as: | 
|  |  | 
|  | $$H_{out} = (H_{in}-1)*strides[0] - 2*pads[0] + kernels[0]$$ | 
|  |  | 
|  |  | 
|  | $$W_{out} = (W_{in}-1)*strides[1] - 2*pads[1] + kernels[1]$$ | 
|  |  | 
|  | Note on the implementation layout: conv_transpose_op_impl.h is the templated implementation of the conv_transpose_op.h file, which is why they are separate files. Also, in the implementation this operator inherits from the *ConvTransposeUnpoolOpBase* operator. | 
|  |  | 
|  | Github Links: | 
|  | - https://github.com/pytorch/pytorch/tree/master/caffe2/operators/conv_transpose_op.h | 
|  | - https://github.com/pytorch/pytorch/tree/master/caffe2/operators/conv_transpose_op.cc | 
|  | - https://github.com/pytorch/pytorch/tree/master/caffe2/operators/conv_transpose_unpool_op_base.h | 
|  |  | 
|  | <details> | 
|  |  | 
|  | <summary> <b>Example</b> </summary> | 
|  |  | 
|  | **Code** | 
|  |  | 
|  | ``` | 
|  |  | 
|  | workspace.ResetWorkspace() | 
|  |  | 
|  | op = core.CreateOperator( | 
|  | "ConvTranspose", | 
|  | ["X", "filter", "bias"], | 
|  | ["Y"], | 
|  | kernels=[2,2], | 
|  | pads=[4,4,4,4], | 
|  | strides=[2,2] | 
|  | ) | 
|  |  | 
|  | # Create X: (N,C,H,W) | 
|  | data = np.random.randn(2,3,5,5).astype(np.float32) | 
|  | print("Data shape: ",data.shape) | 
|  |  | 
|  | # Create filter: (M,C,Kh,Kw) | 
|  | filters = np.random.randn(3,1,2,2).astype(np.float32) | 
|  | print("Filter shape: ",filters.shape) | 
|  |  | 
|  | # Create b: M | 
|  | bias = np.array([1.]).astype(np.float32) | 
|  | print("Bias shape: ",bias.shape) | 
|  |  | 
|  | # Put the inputs into the workspace | 
|  | workspace.FeedBlob("X", data) | 
|  | workspace.FeedBlob("filter", filters) | 
|  | workspace.FeedBlob("bias", bias) | 
|  |  | 
|  | # Run the operator | 
|  | workspace.RunOperatorOnce(op) | 
|  | print("Y:\n", workspace.FetchBlob("Y")) | 
|  |  | 
|  | ``` | 
|  |  | 
|  | **Result** | 
|  |  | 
|  | ``` | 
|  |  | 
|  | Data shape:  (2, 3, 5, 5) | 
|  | Filter shape:  (3, 1, 2, 2) | 
|  | Bias shape:  (1,) | 
|  | Y: | 
|  | [[[[0.53606427 0.5775447 ] | 
|  | [0.40148795 1.5188271 ]]] | 
|  |  | 
|  |  | 
|  | [[[1.9903406  3.2794335 ] | 
|  | [0.09960175 0.31917763]]]] | 
|  |  | 
|  | ``` | 
|  |  | 
|  | </details> | 
|  |  | 
|  | )DOC") | 
|  | .Input( | 
|  | 0, | 
|  | "X", | 
|  | "Input data blob, of shape $(N, C_{in}, H_{in}, W_{in})$, to be operated on.") | 
|  | .Input( | 
|  | 1, | 
|  | "filter", | 
|  | "The filter blob, of shape $(M, C_{out}, K_H, K_W)$, containing the filters to be used in the transposed convolution.") | 
|  | .Input( | 
|  | 2, | 
|  | "bias", | 
|  | "The bias blob, of length $C_{out}$, containing the biases for the operation, one bias per output channel. If not passed, biases assumed to be zeros.") | 
|  | .Output( | 
|  | 0, | 
|  | "Y", | 
|  | "Output data blob, of shape $(N, C_{out}, H_{out}, W_{out})$, that contains the result of the operation.") | 
|  | .Arg( | 
|  | "legacy_pad", | 
|  | "*(type: int; optional)* Should the legacy padding be VALID or SAME. When used, pads should not be used.") | 
|  | .Arg( | 
|  | "kernels", | 
|  | "*(type: [int]; default: [])* Desired kernel size. If left at default the kernel size will be inferred from the input $filter$ blob.") | 
|  | .Arg( | 
|  | "strides", | 
|  | "*(type: [int]; default: [])* Controls the stride of the kernel as it traverses the input blob.") | 
|  | .Arg( | 
|  | "pads", | 
|  | "*(type: [int]; default: [])* Controls the amount of padding applied to the input feature map before computation.") | 
|  | .Arg( | 
|  | "adjs", | 
|  | "*(type: [int]; default: [])*") | 
|  | .Arg( | 
|  | "order", | 
|  | "*(type: string; default: \"NCHW\")* Specifies the order of the input data blob, where $N$ is batch size, $C$ is number of channels, $H$ is spatial height, and $W$ is spatial width. The only other valid option is \"NHWC\".") | 
|  | .Arg( | 
|  | "shared_buffer", | 
|  | "*(type: int; default: 0)*") | 
|  | .Arg( | 
|  | "no_bias", | 
|  | "*(type: bool; default: False)* ") | 
|  | .InheritOnnxSchema("ConvTranspose"); | 
|  |  | 
|  | } // namespace caffe2 |