Source code for mindspore.nn.layer.padding

# Copyright 2022 Huawei Technologies Co., Ltd
#
# 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.
# ============================================================================
""" padding """
from __future__ import absolute_import

from mindspore.common import Tensor
from mindspore import ops
from mindspore.ops.primitive import constexpr
from mindspore.nn.cell import Cell

__all__ = ['ConstantPad1d', 'ConstantPad2d', 'ConstantPad3d', 'ReflectionPad1d', 'ReflectionPad2d', 'ZeroPad2d']


@constexpr
def _check_padding_dimension(dimension, padding):
    r"""
    Validate the input padding and add place holders if needed.
    Note: the input 'padding' in this function is already converted to list of lists to match MirrorPad
    """
    if dimension < len(padding):
        raise ValueError(f"For padding with length {len(padding) * 2}, the dimension of the tensor should be at least "
                         f"{len(padding)}, but got {dimension}")
    # add place holders
    if dimension > len(padding):
        padding = [(0, 0) for _ in range(dimension - len(padding))] + [x for x in padding]
    return padding


def _swap_to_ms_padding_order(padding):
    r"""
    Check whether the input padding is a tuple or a int converted to a tuple.
    Check if the length of padding is divisible by 2.
    Convert the input padding to the format that MirrorPad would understand.
    """
    number_of_paddings = len(padding) // 2
    new_padding = [[0, 0]] * number_of_paddings
    for i in range(number_of_paddings):
        new_padding[i] = [padding[2 * i], padding[2 * i + 1]]
    # reverse the padding list to match the order of paddings for MirrorPad
    new_padding.reverse()
    return new_padding


@constexpr
def _check(input_shape, padding, name):
    """
    Check relationship between input shape and padding to make sure after negative dimension padding the out is
    positive.
    """
    if len(input_shape) < len(padding):
        msg = "For '{}', the dimension of input must more than or equal to len(padding)/2, " \
              "but got {}".format(name, len(input_shape))
        raise ValueError(msg)
    if len(input_shape) > len(padding):
        if len(padding) == 2 and isinstance(padding[0], int):
            padding = [(0, 0) for i in range(len(input_shape) - 1)] + [padding]
        else:
            padding = [(0, 0) for i in range(len(input_shape) - len(padding))] + [x for x in padding]
    for index, item in enumerate(padding):
        if index == 0:
            dim_name = '1st'
        elif index == 1:
            dim_name = '2nd'
        elif index == 2:
            dim_name = '3rd'
        else:
            dim_name = str(index + 1) + 'th'

        if item[0] < -input_shape[index]:
            msg = "For '{}', the shape of input after padding must be positive, the input shape is {}, " \
                  "value of parameter 'padding' applied to the {} dimension of input must " \
                  "no less than -{}, but got {}".format(name, input_shape, dim_name, input_shape[index], item[0])
            raise ValueError(msg)
        if item[1] < -input_shape[index]:
            msg = "For '{}', the shape of input after padding must be positive, the input shape is {}, " \
                  "value of parameter 'padding' applied to the {} dimension of input must " \
                  "no less than -{}, but got {}".format(name, input_shape, dim_name, input_shape[index], item[1])
            raise ValueError(msg)
        if input_shape[index] + item[0] + item[1] <= 0:
            msg = "For '{}', the shape of input after padding must be positive, the input shape is {}, " \
                  "but the {} dimension of input shape {} plus padding {} and {} resulted in a non-positive output " \
                  "shape.".format(name, input_shape, dim_name, input_shape[index], item[0], item[1])
            raise ValueError(msg)
    return padding


@constexpr
def _get_new_padding(padding):
    """get non-negative padding and make negative position."""
    new_padding = [[item[0], item[1]] for item in padding]
    start = [0 for i in range(len(new_padding))]
    end = [0 for i in range(len(new_padding))]
    for index, item in enumerate(new_padding):
        if item[0] < 0:
            start[index] = item[0]
            new_padding[index][0] = 0
        if item[1] < 0:
            end[index] = item[1]
            new_padding[index][1] = 0
    new_padding = tuple(new_padding)
    return new_padding, start, end


@constexpr
def _get_begin_size(shape, begin, end):
    """Calculate begin and size for ops.Slice."""
    size = tuple([shape[i] + begin[i] + end[i] for i in range(len(shape))])
    begin = tuple([int(-i) for i in begin])
    return begin, size


class _ConstantPadNd(Cell):
    r"""
    Using a given value to pads the last n dimensions of input tensor.

    Args:
        padding(tuple, list): The padding size to pad the last n dimensions of input tensor. The padding
            sequence is starting from the last dimension and moving forward. The length of padding must be
            a multiple of 2. len(padding)/2 dimensions of input will be padded.
        value(union[int, float]): Padding value.

         padding (union[list, tuple]): The padding size to pad the last n dimensions of input tensor.
            The padding sequence is starting from the last dimension and moving forward.
            The length of padding must be a multiple of 2. If padding is :math:`(padding_0, padding_1, padding_2,
            padding_3, ..., padding_2m, padding_{2m+1}, ...)`. The input is `x`,
            the size of last dimension of output is :math:`padding\_0 + x.shape[-1] + padding\_1`.
            The size of penultimate dimension of output is :math:`padding\_2 + x.shape[-2] + padding\_3`.
            The size of 3rd to last dimension of output is :math:`padding\_4 + x.shape[-3] + padding\_5`.
            The size of i-td to last dimension of output is :math:`padding\_{2m} + x.shape[-m-1] + padding\_{2m+1}`.
            The remaining dimensions of the output are consistent with those of the input.
        value (union[int, float]): Padding value.
        name (str): Name of method, used for positioning error messages in the base class.

    Returns:
        Tensor, the tensor after padding.

    Supported Platforms:
        ``Ascend`` ``GPU`` ``CPU``

    Raises:
        ValueError: If the length of padding is not a multiple of 2.
        ValueError: If the length of input less than len(padding)/2.
        ValueError: If the output shape after padding is not positive.
    """

    def __init__(self, padding, value, name='ConstantPadNd'):
        """Initialize Pad."""
        super(_ConstantPadNd, self).__init__()
        if isinstance(padding, int):
            if name == 'ConstantPad1d':
                padding = (padding, padding)
            elif name in ['ConstantPad2d', 'ZeroPad2d']:
                padding = (padding, padding, padding, padding)
            elif name == 'ConstantPad3d':
                padding = (padding, padding, padding, padding, padding, padding)

        elif isinstance(padding, tuple):
            if len(padding) % 2 != 0:
                msg = "For '{}', the length of parameter 'padding' with tuple type must be a multiple of 2, " \
                      "but got {}".format(name, len(padding))
                raise ValueError(msg)
            if name == 'ConstantPad1d' and len(padding) != 2:
                msg = "For '{}', the length of parameter 'padding' with tuple type must equal to 2." \
                      "but got {}".format(name, len(padding))
                raise ValueError(msg)
            if name in ['ConstantPad2d', 'ZeroPad2d'] and len(padding) > 4:
                msg = "For '{}', the length of parameter 'padding' with tuple type must no more than 4." \
                      "but got {}".format(name, len(padding))
                raise ValueError(msg)
            if name == 'ConstantPad3d' and len(padding) > 6:
                msg = "For '{}', the length of parameter 'padding' with tuple type must no more than 6." \
                      "but got {}".format(name, len(padding))
                raise ValueError(msg)

        else:
            msg = "For '{}', the type of parameter 'padding' must be in [int, float], " \
                  "but got {}".format(name, type(padding))
            raise TypeError(msg)

        if not isinstance(value, (int, float)):
            msg = "For '{}', the type of parameter 'value' must be in [int, float], " \
                  "but got {}".format(name, type(value))
            raise TypeError(msg)

        self.value = value
        self.padding = _swap_to_ms_padding_order(padding)
        self._name = name

    def construct(self, x):
        """Construct the pad net."""
        input_shape = x.shape
        input_type = x.dtype
        padding = _check(input_shape, self.padding, self._name)
        new_padding, start, end = _get_new_padding(padding)
        mask = ops.Ones()(input_shape, input_type)
        output = ops.Pad(new_padding)(x)
        mask = ops.Pad(new_padding)(mask)
        ones = ops.Ones()(output.shape, output.dtype)
        value = ops.Fill()(output.dtype, output.shape, self.value)
        output = ops.Add()(ops.Mul()(mask, output), ops.Mul()(ops.Sub()(ones, mask), value))
        slice_op = ops.Slice()
        begin, size = _get_begin_size(output.shape, start, end)
        output = slice_op(output, begin, size)
        return output


[docs]class ConstantPad1d(_ConstantPadNd): r""" Using a given constant value to pads the last dimensions of input tensor. Args: padding (Union[int, tuple]): The padding size to pad the last dimension of input tensor. If is int, uses the same padding in both boundaries of input's last dimension. If a 2-tuple, uses (padding_0, padding_1) to pad. If the input is `x`, the size of last dimension of output is :math:`padding\_0 + x.shape[-1] + padding\_1`. The remaining dimensions of the output are consistent with those of the input. value (Union[int, float]): Padding value. Returns: Tensor, the tensor after padding. Raises: TypeError: If `padding` is not a tuple or int. TypeError: If `value` is not int or float. ValueError: If the length of `padding` with tuple type is not equal to 2. ValueError: If the output shape after padding is not positive. Supported Platforms: ``Ascend`` ``GPU`` ``CPU`` Examples: >>> import numpy as np >>> from mindspore import Tensor >>> from mindspore.nn import ConstantPad1d >>> x = np.ones(shape=(1, 2, 3, 4)).astype(np.float32) >>> x = Tensor(x) >>> # padding is tuple >>> padding = (0, 1) >>> value = 0.5 >>> pad1d = ConstantPad1d(padding, value) >>> out = pad1d(x) >>> print(out) [[[[1. 1. 1. 1. 0.5] [1. 1. 1. 1. 0.5] [1. 1. 1. 1. 0.5]] [[1. 1. 1. 1. 0.5] [1. 1. 1. 1. 0.5] [1. 1. 1. 1. 0.5]]]] >>> print(out.shape) (1, 2, 3, 5) >>> # padding is int >>> padding = 1 >>> value = 0.5 >>> pad1d = ConstantPad1d(padding, value) >>> out = pad1d(x) >>> print(out) [[[[0.5 1. 1. 1. 1. 0.5] [0.5 1. 1. 1. 1. 0.5] [0.5 1. 1. 1. 1. 0.5]] [[0.5 1. 1. 1. 1. 0.5] [0.5 1. 1. 1. 1. 0.5] [0.5 1. 1. 1. 1. 0.5]]]] >>> print(out.shape) (1, 2, 3, 6) >>> # padding is negative >>> padding = (-1, 0) >>> value = 0.5 >>> pad1d = ConstantPad1d(padding, value) >>> out = pad1d(x) >>> print(out) [[[[1. 1. 1.] [1. 1. 1.] [1. 1. 1.]] [[1. 1. 1.] [1. 1. 1.] [1. 1. 1.]]]] >>> print(out.shape) (1, 2, 3, 3) """ def __init__(self, padding, value): super(ConstantPad1d, self).__init__(padding, value, name='ConstantPad1d')
[docs]class ConstantPad2d(_ConstantPadNd): r""" Using a given constant value to pads the last two dimensions of input tensor. Args: padding (Union[int, tuple]): The padding size to pad the last two dimensions of input tensor. If is int, uses the same padding in boundaries of input's last two dimensions. If is tuple and length of padding is 4 uses (padding_0, padding_1, padding_2, padding_3) to pad. If the input is `x`, the size of last dimension of output is :math:`padding\_0 + x.shape[-1] + padding\_1`. The size of penultimate dimension of output is :math:`padding\_2 + x.shape[-2] + padding\_3`. The remaining dimensions of the output are consistent with those of the input. value (Union[int, float]): Padding value. Returns: Tensor, the tensor after padding. Raises: TypeError: If `padding` is not a tuple or int. TypeError: If `value` is not int or float. ValueError: If the length of `padding` is more than 4 or not a multiple of 2. ValueError: If the output shape after padding is not positive. Supported Platforms: ``Ascend`` ``GPU`` ``CPU`` Examples: >>> import numpy as np >>> from mindspore import Tensor >>> from mindspore.nn import ConstantPad2d >>> x = np.ones(shape=(1, 2, 3, 4)).astype(np.float32) >>> x = Tensor(x) >>> padding = (-1, 1, 0, 1) >>> value = 0.5 >>> pad2d = ConstantPad2d(padding, value) >>> out = pad2d(x) >>> print(out) [[[[1. 1. 1. 0.5] [1. 1. 1. 0.5] [1. 1. 1. 0.5] [0.5 0.5 0.5 0.5]] [[1. 1. 1. 0.5] [1. 1. 1. 0.5] [1. 1. 1. 0.5] [0.5 0.5 0.5 0.5]]]] >>> print(out.shape) (1, 2, 4, 4) """ def __init__(self, padding, value): super(ConstantPad2d, self).__init__(padding, value, name='ConstantPad2d')
[docs]class ConstantPad3d(_ConstantPadNd): r""" Using a given constant value to pads the last three dimensions of input tensor. Args: padding (Union[int, tuple]): The padding size to pad the last three dimensions of input tensor. If is int, uses the same padding in boundaries of input's last three dimensions. If is tuple and length of padding is 6 uses (padding_0, padding_1, padding_2, padding_3, padding_4, padding_5) to pad. If the input is `x`, the size of last dimension of output is :math:`padding\_0 + x.shape[-1] + padding\_1`. The size of penultimate dimension of output is :math:`padding\_2 + x.shape[-2] + padding\_3`. The size of 3rd to last dimension of output is :math:`padding\_4 + x.shape[-3] + padding\_5`. The remaining dimensions of the output are consistent with those of the input. value (Union[int, float]): Padding value. Returns: Tensor, the tensor after padding. Raises: TypeError: If `padding` is not a tuple or int. TypeError: If `value` is not int or float. ValueError: If the length of `padding` is more than 6 or not a multiple of 2. ValueError: If the output shape after padding is not positive. Supported Platforms: ``Ascend`` ``GPU`` ``CPU`` Examples: >>> import numpy as np >>> from mindspore import Tensor >>> from mindspore.nn import ConstantPad3d >>> x = np.ones(shape=(1, 2, 3, 4)).astype(np.float32) >>> x = Tensor(x) >>> padding = (-1, 1, 0, 1, 1, 0) >>> value = 0.5 >>> pad3d = ConstantPad3d(padding, value) >>> out = pad3d(x) >>> print(out) [[[[0.5 0.5 0.5 0.5] [0.5 0.5 0.5 0.5] [0.5 0.5 0.5 0.5] [0.5 0.5 0.5 0.5]] [[1. 1. 1. 0.5] [1. 1. 1. 0.5] [1. 1. 1. 0.5] [0.5 0.5 0.5 0.5]] [[1. 1. 1. 0.5] [1. 1. 1. 0.5] [1. 1. 1. 0.5] [0.5 0.5 0.5 0.5]]]] >>> print(out.shape) (1, 3, 4, 4) """ def __init__(self, padding, value): super(ConstantPad3d, self).__init__(padding, value, name='ConstantPad3d')
class _ReflectionPadNd(Cell): r""" Using a given padding to do reflection pad on the given tensor. Work as a parent class, and only accepts tuple as padding input. """ def __init__(self, padding, name="ReflectionPadNd"): super(_ReflectionPadNd, self).__init__() self.name = name # check if padding and its elements are valid if not isinstance(padding, tuple): raise TypeError(f"For '{self.name}' the input 'padding' must be an integer or tuple, " f"but got {type(padding).__name__}") if len(padding) % 2 != 0: raise ValueError(f"For '{self.name}' the length of input 'padding' must be divisible by 2, " f"but got padding of length {len(padding)}. ") if not all(isinstance(i, int) for i in padding): raise TypeError(f"For '{self.name}' every element in 'padding' must be integer, " f"but got {padding}. ") if not all(i >= 0 for i in padding): raise ValueError(f"For '{self.name}' every element in 'padding' must be >= 0. " f"but got {padding}. ") self.padding = _swap_to_ms_padding_order(padding) def construct(self, x): input_shape = x.shape padding = _check_padding_dimension(len(input_shape), self.padding) x = ops.MirrorPad(mode='REFLECT')(x, Tensor(padding)) return x
[docs]class ReflectionPad1d(_ReflectionPadNd): r""" Using a given padding to do reflection pad on the given tensor. Args: padding (union[int, tuple]): The padding size to pad the last dimension of input tensor. If padding is an integer: all directions will be padded with the same size. If padding is a tuple: uses :math:`(pad_{left}, pad_{right})` to pad. Inputs: - **x** (Tensor) - 2D or 3D, shape: :math:`(C, W_{in})` or :math:`(N, C, W_{in})`. Outputs: Tensor, after padding. Shape: :math:`(C, W_{out})` or :math:`(N, C, W_{out})`, where :math:`W_{out} = W_{in} + pad_{left} + pad_{right}`. Raises: TypeError: If 'padding' is not a tuple or int. TypeError: If there is an element in 'padding' that is not int. ValueError: If the length of 'padding' is not divisible by 2. ValueError: If there is an element in 'padding' that is negative. ValueError: If the there is a dimension mismatch between the padding and the tensor. Supported Platforms: ``Ascend`` ``GPU`` ``CPU`` Examples: >>> import numpy as np >>> from mindspore import Tensor >>> from mindspore.nn import ReflectionPad1d >>> x = Tensor(np.array([[[0, 1, 2, 3], [4, 5, 6, 7]]]).astype(np.float32)) >>> # x has shape (1, 2, 4) >>> padding = (3, 1) >>> # The first and the second dimension of x remain the same. >>> # The third dimension of x: W_out = W_in + pad_left + pad_right = 4 + 3 + 1 = 8 >>> pad1d = ReflectionPad1d(padding) >>> out = pad1d(x) >>> # The shape of out is (1, 2, 8) >>> print(out) [[[3. 2. 1. 0. 1. 2. 3. 2.] [7. 6. 5. 4. 5. 6. 7. 6.]]] """ def __init__(self, padding): if isinstance(padding, int): padding = (padding, padding) super(ReflectionPad1d, self).__init__(padding, 'ReflectionPad1d')
[docs]class ReflectionPad2d(_ReflectionPadNd): r""" Using a given padding to do reflection pad the given tensor. Args: padding (union[int, tuple]): The padding size to pad the input tensor. If padding is an integer: all directions will be padded with the same size. If padding is a tuple: uses :math:`(pad_{left}, pad_{right}, pad_{up}, pad_{down})` to pad. Inputs: - **x** (Tensor) - 3D or 4D, shape: :math:`(C, H_{in}, W_{out})` or :math:`(N, C, H_{out}, W_{out})`. Outputs: Tensor, after padding. Shape: :math:`(C, H_{out}, W_{out})` or :math:`(N, C, H_{out}, W_{out})`, where :math:`H_{out} = H_{in} + pad_{up} + pad_{down}`, :math:`W_{out} = W_{in} + pad_{left} + pad_{right}`. Raises: TypeError: If 'padding' is not a tuple or int. TypeError: If there is an element in 'padding' that is not int. ValueError: If the length of 'padding' is not divisible by 2. ValueError: If there is an element in 'padding' that is negative. ValueError: If the there is a dimension mismatch between the padding and the tensor. Supported Platforms: ``Ascend`` ``GPU`` ``CPU`` Examples: >>> import numpy as np >>> from mindspore import Tensor >>> from mindspore.nn import ReflectionPad2d >>> x = Tensor(np.array([[[0, 1, 2], [3, 4, 5], [6, 7, 8]]]).astype(np.float32)) >>> # x has shape (1, 3, 3) >>> padding = (1, 1, 2, 0) >>> pad2d = ReflectionPad2d(padding) >>> # The first dimension of x remains the same. >>> # The second dimension of x: H_out = H_in + pad_up + pad_down = 3 + 1 + 1 = 5 >>> # The third dimension of x: W_out = W_in + pad_left + pad_right = 3 + 2 + 0 = 5 >>> out = pad2d(x) >>> # The shape of out is (1, 5, 5) >>> print(out) [[[7. 6. 7. 8. 7.] [4. 3. 4. 5. 4.] [1. 0. 1. 2. 1.] [4. 3. 4. 5. 4.] [7. 6. 7. 8. 7.]]] """ def __init__(self, padding): if isinstance(padding, int): padding = (padding, padding, padding, padding) super(ReflectionPad2d, self).__init__(padding, 'ReflectionPad2d')
[docs]class ZeroPad2d(_ConstantPadNd): r""" Pads the last two dimensions of input tensor with zero. Args: padding (Union[int, tuple]): The padding size to pad the last two dimensions of input tensor. If is int, uses the same padding in boundaries of input's last two dimensions. If is tuple and length of padding is 4 uses (padding_0, padding_1, padding_2, padding_3) to pad. If the input is `x`, the size of last dimension of output is :math:`padding\_0 + x.shape[-1] + padding\_1`. The size of penultimate dimension of output is :math:`padding\_2 + x.shape[-2] + padding\_3`. The remaining dimensions of the output are consistent with those of the input. Returns: Tensor, the tensor after padding. Raises: TypeError: If `padding` is not a tuple or int. ValueError: If the length of `padding` is more than 4 or not a multiple of 2. ValueError: If the output shape after padding is not positive. Supported Platforms: ``Ascend`` ``GPU`` ``CPU`` Examples: >>> import numpy as np >>> from mindspore import Tensor >>> from mindspore.nn import ZeroPad2d >>> x = np.ones(shape=(1, 2, 3, 4)).astype(np.float32) >>> x = Tensor(x) >>> padding = (-1, 1, 0, 1) >>> pad = ZeroPad2d(padding) >>> out = pad(x) >>> print(out) [[[[1. 1. 1. 0.] [1. 1. 1. 0.] [1. 1. 1. 0.] [0. 0. 0. 0.]] [[1. 1. 1. 0.] [1. 1. 1. 0.] [1. 1. 1. 0.] [0. 0. 0. 0.]]]] >>> print(out.shape) (1, 2, 4, 4) """ def __init__(self, padding): value = 0 super(ZeroPad2d, self).__init__(padding, value, name='ZeroPad2d')