logic.py 16.8 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14
#   Copyright (c) 2020 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.

H
huangxu96 已提交
15
from . import to_tensor
Z
Zhen Wang 已提交
16
from ..fluid.layer_helper import LayerHelper
17
from ..fluid.data_feeder import check_type, check_variable_and_dtype
Z
Zhen Wang 已提交
18
from ..fluid.layers.layer_function_generator import templatedoc
W
wawltor 已提交
19
from .. import fluid
20 21
from ..fluid.framework import in_dygraph_mode
from paddle.common_ops_import import *
Z
zhulei 已提交
22
from ..framework import VarBase as Tensor
23

24
# TODO: define logic functions of a tensor  
25 26 27 28 29 30
from ..fluid.layers import is_empty  #DEFINE_ALIAS
from ..fluid.layers import isfinite  #DEFINE_ALIAS
from ..fluid.layers import logical_and  #DEFINE_ALIAS
from ..fluid.layers import logical_not  #DEFINE_ALIAS
from ..fluid.layers import logical_or  #DEFINE_ALIAS
from ..fluid.layers import logical_xor  #DEFINE_ALIAS
31 32
from ..fluid.layers import reduce_all  #DEFINE_ALIAS
from ..fluid.layers import reduce_any  #DEFINE_ALIAS
33

34 35
__all__ = [
    'equal',
W
wawltor 已提交
36
    'equal_all',
37 38 39 40 41 42 43 44 45 46 47
    'greater_equal',
    'greater_than',
    'is_empty',
    'isfinite',
    'less_equal',
    'less_than',
    'logical_and',
    'logical_not',
    'logical_or',
    'logical_xor',
    'not_equal',
Z
Zhen Wang 已提交
48
    'allclose',
Z
zhulei 已提交
49
    'is_tensor'
50
    #       'isnan'
51 52 53
]


W
wawltor 已提交
54
def equal_all(x, y, name=None):
55 56 57
    """
    This OP returns the truth value of :math:`x == y`. True if two inputs have the same elements, False otherwise.

W
wawltor 已提交
58
    **NOTICE**: The output of this OP has no gradient.
59 60

    Args:
W
wawltor 已提交
61 62 63 64
        x(Tensor): Tensor, data type is float32, float64, int32, int64.
        y(Tensor): Tensor, data type is float32, float64, int32, int64.
        name(str, optional): The default value is None.  Normally there is no need for
            user to set this property.  For more information, please refer to :ref:`api_guide_Name`.
65 66

    Returns:
W
wawltor 已提交
67
        Tensor: output Tensor, data type is bool, value is [False] or [True].
68 69 70 71 72

    Examples:
        .. code-block:: python

          import paddle
W
wawltor 已提交
73

74 75 76
          x = paddle.to_tensor([1, 2, 3])
          y = paddle.to_tensor([1, 2, 3])
          z = paddle.to_tensor([1, 4, 3])
W
wawltor 已提交
77
          result1 = paddle.equal_all(x, y)
N
Noel 已提交
78
          print(result1) # result1 = [True ]
W
wawltor 已提交
79
          result2 = paddle.equal_all(x, z)
N
Noel 已提交
80
          print(result2) # result2 = [False ]
81
    """
W
wawltor 已提交
82 83

    helper = LayerHelper("equal_all", **locals())
84 85
    out = helper.create_variable_for_type_inference(dtype='bool')
    helper.append_op(
W
wawltor 已提交
86 87
        type='equal_all', inputs={'X': [x],
                                  'Y': [y]}, outputs={'Out': [out]})
88
    return out
Z
Zhen Wang 已提交
89 90 91


@templatedoc()
92
def allclose(x, y, rtol=1e-05, atol=1e-08, equal_nan=False, name=None):
Z
Zhen Wang 已提交
93 94 95 96
    """
    ${comment}

    Args:
97 98
        x(Tensor): ${input_comment}.
        y(Tensor): ${other_comment}.
H
huangxu96 已提交
99 100
        rtol(rtoltype, optional): The relative tolerance. Default: :math:`1e-5` .
        atol(atoltype, optional): The absolute tolerance. Default: :math:`1e-8` .
101 102 103
        equal_nan(equalnantype, optional): ${equal_nan_comment}.
        name (str, optional): Name for the operation. For more information, please
            refer to :ref:`api_guide_Name`. Default: None.
Z
Zhen Wang 已提交
104 105

    Returns:
106 107 108 109 110 111 112 113
        Tensor: ${out_comment}.

    Raises:
        TypeError: The data type of ``x`` must be one of float32, float64.
        TypeError: The data type of ``y`` must be one of float32, float64.
        TypeError: The type of ``rtol`` must be float.
        TypeError: The type of ``atol`` must be float.
        TypeError: The type of ``equal_nan`` must be bool.
Z
Zhen Wang 已提交
114 115 116 117 118 119

    Examples:
        .. code-block:: python

          import paddle

120 121
          x = paddle.to_tensor([10000., 1e-07])
          y = paddle.to_tensor([10000.1, 1e-08])
122
          result1 = paddle.allclose(x, y, rtol=1e-05, atol=1e-08,
Z
Zhen Wang 已提交
123
                                  equal_nan=False, name="ignore_nan")
124 125 126
          np_result1 = result1.numpy()
          # [False]
          result2 = paddle.allclose(x, y, rtol=1e-05, atol=1e-08,
Z
Zhen Wang 已提交
127
                                      equal_nan=True, name="equal_nan")
128 129 130
          np_result2 = result2.numpy()
          # [False]

131 132
          x = paddle.to_tensor([1.0, float('nan')])
          y = paddle.to_tensor([1.0, float('nan')])
133 134 135 136 137 138 139 140
          result1 = paddle.allclose(x, y, rtol=1e-05, atol=1e-08,
                                  equal_nan=False, name="ignore_nan")
          np_result1 = result1.numpy()
          # [False]
          result2 = paddle.allclose(x, y, rtol=1e-05, atol=1e-08,
                                      equal_nan=True, name="equal_nan")
          np_result2 = result2.numpy()
          # [True]
Z
Zhen Wang 已提交
141 142
    """

143
    if in_dygraph_mode():
H
huangxu96 已提交
144 145 146
        rtol_tensor = to_tensor(rtol, dtype='float64')
        atol_tensor = to_tensor(atol, dtype='float64')
        return core.ops.allclose(x, y, rtol_tensor, atol_tensor, 'equal_nan',
147 148 149 150
                                 equal_nan)

    check_variable_and_dtype(x, "input", ['float32', 'float64'], 'allclose')
    check_variable_and_dtype(y, "input", ['float32', 'float64'], 'allclose')
Z
Zhen Wang 已提交
151 152 153 154 155
    check_type(rtol, 'rtol', float, 'allclose')
    check_type(atol, 'atol', float, 'allclose')
    check_type(equal_nan, 'equal_nan', bool, 'allclose')

    helper = LayerHelper("allclose", **locals())
H
huangxu96 已提交
156 157 158 159 160 161 162 163 164 165 166 167 168 169 170
    rtol_var = helper.create_global_variable(
        name=fluid.unique_name.generate('rtol'),
        persistable=True,
        dtype='float64',
        shape=[1])
    helper.set_variable_initializer(
        rtol_var, initializer=fluid.initializer.ConstantInitializer(rtol))
    atol_var = helper.create_variable(
        name=fluid.unique_name.generate('atol'),
        persistable=True,
        dtype='float64',
        shape=[1])
    helper.set_variable_initializer(
        atol_var, initializer=fluid.initializer.ConstantInitializer(atol))

Z
Zhen Wang 已提交
171 172
    out = helper.create_variable_for_type_inference(dtype='bool')

H
huangxu96 已提交
173
    inputs = {'Input': x, 'Other': y, 'Rtol': rtol_var, 'Atol': atol_var}
Z
Zhen Wang 已提交
174
    outputs = {'Out': out}
H
huangxu96 已提交
175
    attrs = {'equal_nan': equal_nan}
Z
Zhen Wang 已提交
176 177 178 179
    helper.append_op(
        type='allclose', inputs=inputs, outputs=outputs, attrs=attrs)

    return out
180 181


W
wawltor 已提交
182 183
@templatedoc()
def equal(x, y, name=None):
184
    """
S
swtkiwi 已提交
185

186
    This layer returns the truth value of :math:`x == y` elementwise.
N
Noel 已提交
187

W
wawltor 已提交
188
    **NOTICE**: The output of this OP has no gradient.
189 190

    Args:
W
wawltor 已提交
191 192
        x(Tensor): Tensor, data type is float32, float64, int32, int64.
        y(Tensor): Tensor, data type is float32, float64, int32, int64.
193 194 195 196
        name(str, optional): The default value is None.  Normally there is no need for
            user to set this property.  For more information, please refer to :ref:`api_guide_Name`.

    Returns:
W
wawltor 已提交
197
        Tensor: output Tensor, it's shape is the same as the input's Tensor,
198 199 200 201 202
        and the data type is bool. The result of this op is stop_gradient. 

    Examples:
        .. code-block:: python

W
wawltor 已提交
203 204
          import paddle

205 206
          x = paddle.to_tensor([1, 2, 3])
          y = paddle.to_tensor([1, 3, 2])
W
wawltor 已提交
207
          result1 = paddle.equal(x, y)
N
Noel 已提交
208
          print(result1)  # result1 = [True False False]
209
    """
210 211 212 213 214 215 216 217 218 219 220 221 222 223
    if in_dygraph_mode():
        return core.ops.equal(x, y)

    check_variable_and_dtype(x, "x", ["float32", "float64", "int32", "int64"],
                             "equal")
    check_variable_and_dtype(y, "y", ["float32", "float64", "int32", "int64"],
                             "equal")
    helper = LayerHelper("equal", **locals())
    out = helper.create_variable_for_type_inference(dtype='bool')
    out.stop_gradient = True

    helper.append_op(
        type='equal', inputs={'X': [x],
                              'Y': [y]}, outputs={'Out': [out]})
W
wawltor 已提交
224
    return out
225

W
wawltor 已提交
226 227 228 229 230

@templatedoc()
def greater_equal(x, y, name=None):
    """
    This OP returns the truth value of :math:`x >= y` elementwise, which is equivalent function to the overloaded operator `>=`.
N
Noel 已提交
231

W
wawltor 已提交
232 233 234 235 236 237 238 239 240 241 242 243
    **NOTICE**: The output of this OP has no gradient.

    Args:
        x(Tensor): First input to compare which is N-D tensor. The input data type should be float32, float64, int32, int64.
        y(Tensor): Second input to compare which is N-D tensor. The input data type should be float32, float64, int32, int64.
        name(str, optional): The default value is None.  Normally there is no need for
            user to set this property.  For more information, please refer to :ref:`api_guide_Name`.
    Returns:
        Tensor, the output data type is bool: The tensor storing the output, the output shape is same as input :attr:`x`.

    Examples:
        .. code-block:: python
N
Noel 已提交
244

W
wawltor 已提交
245 246
            import paddle

247 248
            x = paddle.to_tensor([1, 2, 3])
            y = paddle.to_tensor([1, 3, 2])
W
wawltor 已提交
249
            result1 = paddle.greater_equal(x, y)
N
Noel 已提交
250
            print(result1)  # result1 = [True False True]
W
wawltor 已提交
251
    """
252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267
    if in_dygraph_mode():
        return core.ops.greater_equal(x, y)

    check_variable_and_dtype(x, "x", ["float32", "float64", "int32", "int64"],
                             "greater_equal")
    check_variable_and_dtype(y, "y", ["float32", "float64", "int32", "int64"],
                             "greater_equal")
    helper = LayerHelper("greater_equal", **locals())
    out = helper.create_variable_for_type_inference(dtype='bool')
    out.stop_gradient = True

    helper.append_op(
        type='greater_equal',
        inputs={'X': [x],
                'Y': [y]},
        outputs={'Out': [out]})
W
wawltor 已提交
268 269 270 271 272 273 274
    return out


@templatedoc()
def greater_than(x, y, name=None):
    """
    This OP returns the truth value of :math:`x > y` elementwise, which is equivalent function to the overloaded operator `>`.
N
Noel 已提交
275

W
wawltor 已提交
276 277 278 279 280 281 282 283 284 285 286 287
    **NOTICE**: The output of this OP has no gradient.

    Args:
        x(Tensor): First input to compare which is N-D tensor. The input data type should be float32, float64, int32, int64.
        y(Tensor): Second input to compare which is N-D tensor. The input data type should be float32, float64, int32, int64.
        name(str, optional): The default value is None.  Normally there is no need for
            user to set this property.  For more information, please refer to :ref:`api_guide_Name`.
    Returns:
        Tensor, the output data type is bool: The tensor storing the output, the output shape is same as input :attr:`x` .

    Examples:
        .. code-block:: python
N
Noel 已提交
288

W
wawltor 已提交
289 290
            import paddle

291 292
            x = paddle.to_tensor([1, 2, 3])
            y = paddle.to_tensor([1, 3, 2])
W
wawltor 已提交
293
            result1 = paddle.greater_than(x, y)
N
Noel 已提交
294
            print(result1)  # result1 = [False False True]
W
wawltor 已提交
295
    """
296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311
    if in_dygraph_mode():
        return core.ops.greater_than(x, y)

    check_variable_and_dtype(x, "x", ["float32", "float64", "int32", "int64"],
                             "greater_than")
    check_variable_and_dtype(y, "y", ["float32", "float64", "int32", "int64"],
                             "greater_than")
    helper = LayerHelper("greater_than", **locals())
    out = helper.create_variable_for_type_inference(dtype='bool')
    out.stop_gradient = True

    helper.append_op(
        type='greater_than',
        inputs={'X': [x],
                'Y': [y]},
        outputs={'Out': [out]})
W
wawltor 已提交
312 313 314 315 316 317 318
    return out


@templatedoc()
def less_equal(x, y, name=None):
    """
    This OP returns the truth value of :math:`x <= y` elementwise, which is equivalent function to the overloaded operator `<=`.
N
Noel 已提交
319

W
wawltor 已提交
320 321 322 323 324 325 326 327 328 329 330 331 332
    **NOTICE**: The output of this OP has no gradient.

    Args:
        x(Tensor): First input to compare which is N-D tensor. The input data type should be float32, float64, int32, int64.
        y(Tensor): Second input to compare which is N-D tensor. The input data type should be float32, float64, int32, int64.
        name(str, optional): The default value is None.  Normally there is no need for
            user to set this property.  For more information, please refer to :ref:`api_guide_Name`.

    Returns:
        Tensor, the output data type is bool: The tensor storing the output, the output shape is same as input :attr:`x`.

    Examples:
        .. code-block:: python
N
Noel 已提交
333

W
wawltor 已提交
334 335
            import paddle

336 337
            x = paddle.to_tensor([1, 2, 3])
            y = paddle.to_tensor([1, 3, 2])
W
wawltor 已提交
338
            result1 = paddle.less_equal(x, y)
N
Noel 已提交
339
            print(result1)  # result1 = [True True False]
W
wawltor 已提交
340
    """
341 342 343 344 345 346 347 348 349 350 351 352 353 354
    if in_dygraph_mode():
        return core.ops.less_equal(x, y)

    check_variable_and_dtype(x, "x", ["float32", "float64", "int32", "int64"],
                             "less_equal")
    check_variable_and_dtype(y, "y", ["float32", "float64", "int32", "int64"],
                             "less_equal")
    helper = LayerHelper("less_equal", **locals())
    out = helper.create_variable_for_type_inference(dtype='bool')
    out.stop_gradient = True

    helper.append_op(
        type='less_equal', inputs={'X': [x],
                                   'Y': [y]}, outputs={'Out': [out]})
W
wawltor 已提交
355 356 357 358 359 360 361
    return out


@templatedoc()
def less_than(x, y, name=None):
    """
    This OP returns the truth value of :math:`x < y` elementwise, which is equivalent function to the overloaded operator `<`.
N
Noel 已提交
362

W
wawltor 已提交
363 364 365 366 367 368 369 370 371 372 373 374 375
    **NOTICE**: The output of this OP has no gradient.

    Args:
        x(Tensor): First input to compare which is N-D tensor. The input data type should be float32, float64, int32, int64.
        y(Tensor): Second input to compare which is N-D tensor. The input data type should be float32, float64, int32, int64.
        name(str, optional): The default value is None.  Normally there is no need for
            user to set this property.  For more information, please refer to :ref:`api_guide_Name`.

    Returns:
        Tensor, the output data type is bool: The tensor storing the output, the output shape is same as input :attr:`x`.

    Examples:
        .. code-block:: python
N
Noel 已提交
376

W
wawltor 已提交
377 378
            import paddle

379 380
            x = paddle.to_tensor([1, 2, 3])
            y = paddle.to_tensor([1, 3, 2])
W
wawltor 已提交
381
            result1 = paddle.less_than(x, y)
N
Noel 已提交
382
            print(result1)  # result1 = [False True False]
W
wawltor 已提交
383
    """
384 385 386 387 388 389 390 391 392 393 394 395 396 397
    if in_dygraph_mode():
        return core.ops.less_than(x, y)

    check_variable_and_dtype(x, "x", ["float32", "float64", "int32", "int64"],
                             "less_than")
    check_variable_and_dtype(y, "y", ["float32", "float64", "int32", "int64"],
                             "less_than")
    helper = LayerHelper("less_than", **locals())
    out = helper.create_variable_for_type_inference(dtype='bool')
    out.stop_gradient = True

    helper.append_op(
        type='less_than', inputs={'X': [x],
                                  'Y': [y]}, outputs={'Out': [out]})
W
wawltor 已提交
398 399 400 401 402 403 404
    return out


@templatedoc()
def not_equal(x, y, name=None):
    """
    This OP returns the truth value of :math:`x != y` elementwise, which is equivalent function to the overloaded operator `!=`.
N
Noel 已提交
405
    
W
wawltor 已提交
406 407 408 409 410 411 412 413 414 415 416 417 418
    **NOTICE**: The output of this OP has no gradient.

    Args:
        x(Tensor): First input to compare which is N-D tensor. The input data type should be float32, float64, int32, int64.
        y(Tensor): Second input to compare which is N-D tensor. The input data type should be float32, float64, int32, int64.
        name(str, optional): The default value is None.  Normally there is no need for
            user to set this property.  For more information, please refer to :ref:`api_guide_Name`.

    Returns:
        Tensor, the output data type is bool: The tensor storing the output, the output shape is same as input :attr:`x`.

    Examples:
        .. code-block:: python
419

W
wawltor 已提交
420 421
            import paddle

422 423
            x = paddle.to_tensor([1, 2, 3])
            y = paddle.to_tensor([1, 3, 2])
W
wawltor 已提交
424
            result1 = paddle.not_equal(x, y)
N
Noel 已提交
425
            print(result1)  # result1 = [False True True]
W
wawltor 已提交
426
    """
427 428 429 430 431 432 433 434 435 436 437 438 439 440
    if in_dygraph_mode():
        return core.ops.not_equal(x, y)

    check_variable_and_dtype(x, "x", ["float32", "float64", "int32", "int64"],
                             "not_equal")
    check_variable_and_dtype(y, "y", ["float32", "float64", "int32", "int64"],
                             "not_equal")
    helper = LayerHelper("not_equal", **locals())
    out = helper.create_variable_for_type_inference(dtype='bool')
    out.stop_gradient = True

    helper.append_op(
        type='not_equal', inputs={'X': [x],
                                  'Y': [y]}, outputs={'Out': [out]})
441
    return out
Z
zhulei 已提交
442 443 444 445 446


def is_tensor(x):
    """

C
chentianyu03 已提交
447
    This function tests whether input object is a paddle.Tensor.
Z
zhulei 已提交
448 449 450 451 452

    Args:
        x (object): Object to test.

    Returns:
C
chentianyu03 已提交
453
        A boolean value. True if 'x' is a paddle.Tensor, otherwise False.
Z
zhulei 已提交
454 455 456 457 458 459 460 461 462 463 464 465 466 467 468

    Examples:
        .. code-block:: python

            import paddle

            input1 = paddle.rand(shape=[2, 3, 5], dtype='float32')
            check = paddle.is_tensor(input1)
            print(check)  #True

            input3 = [1, 4]
            check = paddle.is_tensor(input3)
            print(check)  #False
            
    """
C
chentianyu03 已提交
469
    return isinstance(x, Tensor)