File: atan2.cpp

package info (click to toggle)
cppad 2025.00.00.2-1
  • links: PTS, VCS
  • area: main
  • in suites: trixie
  • size: 11,552 kB
  • sloc: cpp: 112,594; sh: 5,972; ansic: 179; python: 71; sed: 12; makefile: 10
file content (87 lines) | stat: -rw-r--r-- 2,252 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-or-later
// SPDX-FileCopyrightText: Bradley M. Bell <bradbell@seanet.com>
// SPDX-FileContributor: 2003-22 Bradley M. Bell
// ----------------------------------------------------------------------------

/*
{xrst_begin atan2.cpp}

The AD atan2 Function: Example and Test
#######################################

{xrst_literal
   // BEGIN C++
   // END C++
}

{xrst_end atan2.cpp}
*/
// BEGIN C++

# include <cppad/cppad.hpp>
# define N_THETA 20

bool atan2(void)
{  bool ok = true;
   //
   using CppAD::AD;
   using CppAD::NearEqual;
   double eps99 = 99.0 * std::numeric_limits<double>::epsilon();
   double pi    = 2.0 * std::atan(1.0);
   //
   for(size_t k = 0; k < N_THETA; ++k)
   {  // theta
      double theta =  2.0 * pi * double(k+1) / double(N_THETA) - pi;
      //
      // radius
      double radius = 1.0 + double(k) / double(N_THETA);
      //
      // x, y
      double x = radius * std::cos(theta);
      double y = radius * std::sin(theta);
      //
      // au
      CPPAD_TESTVECTOR(AD<double>) au(2);
      au[0] = x;
      au[1] = y;
      CppAD::Independent(au);
      //
      // av
      CPPAD_TESTVECTOR(AD<double>) av(1);
      av[0] = CppAD::atan2(au[1], au[0]);
      //
      // f(x, y) = atan2(y, x)
      CppAD::ADFun<double> f(au, av);
      //
      // check value
      ok &= NearEqual(av[0] , theta, eps99, eps99);
      //
      // partial_x, partial_y
      // see https://en.wikipedia.org/wiki/Atan2#Derivative
      double partial_x = - y / (radius * radius);
      double partial_y =   x / (radius * radius);
      //
      // check forward mode
      CPPAD_TESTVECTOR(double) du(2), dv(1);
      du[0] = 1.0;
      du[1] = 0.0;
      dv    = f.Forward(1, du);
      ok   &= NearEqual(dv[0], partial_x, eps99, eps99);
      du[0] = 0.0;
      du[1] = 1.0;
      dv    = f.Forward(1, du);
      ok   &= NearEqual(dv[0], partial_y, eps99, eps99);
      //
      // check reverse mode
      CPPAD_TESTVECTOR(double)  w(1);
      CPPAD_TESTVECTOR(double) dw(2);
      w[0]  = 1.;
      dw    = f.Reverse(1, w);
      ok   &= NearEqual(dw[0], partial_x, eps99, eps99);
      ok   &= NearEqual(dw[1], partial_y, eps99, eps99);
      //
   }
   return ok;
}

// END C++