Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

【Hackathon No.8】add api hypot & hypot_ #57295

Merged
merged 12 commits into from
Oct 18, 2023
Next Next commit
add hypot
  • Loading branch information
llyyxx0413 committed Sep 13, 2023
commit dcd3779fed48f560d87601132bd25f5fa1eebfa9
53 changes: 53 additions & 0 deletions python/paddle/tensor/math.py
Original file line number Diff line number Diff line change
Expand Up @@ -6943,3 +6943,56 @@ def ldexp_(x, y, name=None):
y = paddle.cast(y, dtype=out_dtype)
two = paddle.to_tensor(2, dtype=out_dtype)
return paddle.multiply_(x, paddle.pow(two, y))

def hypot(x, y, name=None):
"""
Calculate the length of the hypotenuse of a right-angle triangle. The equation is:

.. math::
out = {\sqrt{x^2 + y^2} }

Args:
x (Tensor): The input Tensor, the data type is float32, float64, int32 or int64.
y (Tensor): The input Tensor, the data type is float32, float64, int32 or int64.
name (str, optional): Name for the operation (optional, default is None).For more information, please refer to :ref:`api_guide_Name`.

Returns:
out (Tensor): An N-D Tensor. If x, y have different shapes and are "broadcastable", the resulting tensor shape is the shape of x and y after broadcasting. If x, y have the same shape, its shape is the same as x and y. And the data type is float32 or float64.

Examples:

.. code-block:: python

>>> import paddle

>>> x = paddle.to_tensor([3], dtype='float32')
>>> y = paddle.to_tensor([4], dtype='int32')
>>> res = paddle.hypot(x, y)
>>> print(res)
Tensor(shape=[1], dtype=float32, place=Place(cpu), stop_gradient=True,
[5.])

"""
if not isinstance(x, (paddle.Tensor, Variable)):
raise TypeError(f"x must be tensor type, but got {type(x)}")
if not isinstance(y, (paddle.Tensor, Variable)):
raise TypeError(f"y must be tensor type, but got {type(y)}")

out = (paddle.square(x) + paddle.square(y)).sqrt()
return out

def hypot_(x, y, name=None):
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

inplace API should be with decorator of @inplace_apis_in_dygraph_only

r"""
Inplace version of ``hypot`` API, the output Tensor will be inplaced with input ``x``.
Please refer to :ref:`api_paddle_hypot`.
"""
if not isinstance(x, (paddle.Tensor, Variable)):
raise TypeError(f"x must be tensor type, but got {type(x)}")
if not isinstance(y, (paddle.Tensor, Variable)):
raise TypeError(f"y must be tensor type, but got {type(y)}")

out = x.pow_(2).add_(y.pow(2)).sqrt_()
return out



101 changes: 101 additions & 0 deletions test/legacy_test/test_hypot.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,101 @@
# Copyright (c) 2019 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.

import unittest

import numpy as np

import paddle
from paddle import base
from paddle.base import core

paddle.enable_static()


class TestHypotAPI(unittest.TestCase):
def setUp(self):
self.x_np = [12]
self.y_np = [20]
self.x_shape = [1]
self.y_shape = [1]

def test_static_graph(self):
startup_program = base.Program()
train_program = base.Program()
with base.program_guard(startup_program, train_program):
x = paddle.static.data(
name='input1', dtype='float32', shape=self.x_shape
)
y = paddle.static.data(
name='input2', dtype='float32', shape=self.y_shape
)
out = paddle.hypot(x, y)

place = (
base.CUDAPlace(0)
if core.is_compiled_with_cuda()
else base.CPUPlace()
)
exe = base.Executor(place)
res = exe.run(
base.default_main_program(),
feed={'input1': self.x_np, 'input2': self.y_np},
fetch_list=[out],
)
self.assertTrue(
(np.array(res[0]) == np.hypot(self.x_np, self.y_np)).all()
)

def test_dygraph(self):
paddle.disable_static()
x = paddle.to_tensor(self.x_np)
y = paddle.to_tensor(self.y_np)
result = paddle.hypot(x, y)
np.testing.assert_allclose(
np.hypot(self.x_np, self.y_np), result.numpy(), rtol=1e-05
)

paddle.enable_static()


class TestHypotAPI2(TestHypotAPI):
def setUp(self):
self.x_np = np.arange(6).astype(np.float32)
self.y_np = np.array([20]).astype(np.float32)
self.x_shape = [6]
self.y_shape = [1]


class TestHypotAPI3(TestHypotAPI):
def setUp(self):
self.x_np = 0
self.y_np = 20
self.x_shape = []
self.y_shape = []


class TestHypotAPI4(TestHypotAPI):
def setUp(self):
self.x_np = [0]
self.y_np = [0]
self.x_shape = [1]
self.y_shape = [1]


class TestHypotAPI5(TestHypotAPI):
def setUp(self):
self.x_np = 12
self.y_np = -20
self.x_shape = []
self.y_shape = []
28 changes: 28 additions & 0 deletions test/legacy_test/test_inplace.py
Original file line number Diff line number Diff line change
Expand Up @@ -838,6 +838,34 @@ def test_error(self):
self.assertRaises(ValueError, paddle.gcd_, x, y)


class TestDygraphInplaceHypot(TestDygraphInplace):
def init_data(self):
self.input_var_numpy = np.random.randint(2, size=200)
self.input_var_numpy = self.input_var_numpy.reshape([10, 20])
self.dtype = "float32"
self.y = paddle.randn(low=-5, high=5, shape=[10, 20], dtype="float32")

def inplace_api_processing(self, var):
return paddle.hypot_(var, self.y)

def non_inplace_api_processing(self, var):
return paddle.hypot(var, self.y)

def test_forward_version(self):
with paddle.base.dygraph.guard():
var = paddle.to_tensor(self.input_var_numpy).astype(self.dtype)
self.assertEqual(var.inplace_version, 0)

inplace_var = self.inplace_api_processing(var)
self.assertEqual(var.inplace_version, 3)

inplace_var[0] = 2.0
self.assertEqual(var.inplace_version, 4)

inplace_var = self.inplace_api_processing(inplace_var)
self.assertEqual(var.inplace_version, 7)


class TestDygraphInplaceNanToNum(TestDygraphInplace):
def init_data(self):
self.input_var_numpy = np.array(
Expand Down