cppad/basic.cpp
Go to the documentation of this file.
1 //
2 // Copyright (c) 2018-2019 CNRS INRIA
3 //
4 
6 #include <cppad/speed/det_by_minor.hpp>
7 
8 #include <boost/variant.hpp> // to avoid C99 warnings
9 
10 #include <iostream>
11 
12 #include <boost/test/unit_test.hpp>
13 #include <boost/utility/binary.hpp>
14 
15 BOOST_AUTO_TEST_SUITE(BOOST_TEST_MODULE)
16 
17 BOOST_AUTO_TEST_CASE(test_example1_cppad)
18 {
19  using CppAD::AD;
20  using CppAD::NearEqual;
21  using Eigen::Dynamic;
22  using Eigen::Matrix;
23  //
24  typedef Matrix<AD<double>, Dynamic, 1> eigen_vector;
25  //
26  // some temporary indices
27  size_t i, j;
28 
29  // domain and range space vectors
30  size_t n = 10, m = n;
31  eigen_vector a_x(n), a_y(m);
32 
33  // set and declare independent variables and start tape recording
34  for (j = 0; j < n; j++)
35  {
36  a_x[(Eigen::DenseIndex)j] = double(1 + j);
37  }
38  CppAD::Independent(a_x);
39 
40  // evaluate a component wise function
41  a_y = a_x.array() + a_x.array().sin();
42 
43  // create f: x -> y and stop tape recording
44  CppAD::ADFun<double> f(a_x, a_y);
45 
46  // compute the derivative of y w.r.t x using CppAD
47  CPPAD_TESTVECTOR(double) x(n);
48  for (j = 0; j < n; j++)
49  {
50  x[j] = double(j) + 1.0 / double(j + 1);
51  }
52  CPPAD_TESTVECTOR(double) jac = f.Jacobian(x);
53 
54  // check Jacobian
55  double eps = 100. * CppAD::numeric_limits<double>::epsilon();
56  for (i = 0; i < m; i++)
57  {
58  for (j = 0; j < n; j++)
59  {
60  double check = 1.0 + cos(x[i]);
61  if (i != j)
62  check = 0.0;
63  BOOST_CHECK(NearEqual(jac[i * n + j], check, eps, eps));
64  }
65  }
66 }
67 
68 BOOST_AUTO_TEST_CASE(test_example2_cppad)
69 {
70  using CppAD::AD;
71  using CppAD::NearEqual;
72  using Eigen::Dynamic;
73  using Eigen::Matrix;
74  //
75  typedef Matrix<double, Dynamic, Dynamic> eigen_matrix;
76  typedef Matrix<AD<double>, Dynamic, Dynamic> eigen_ad_matrix;
77  //
78  typedef Matrix<double, Dynamic, 1> eigen_vector;
79  typedef Matrix<AD<double>, Dynamic, 1> eigen_ad_vector;
80  // some temporary indices
81  size_t i, j;
82 
83  // domain and range space vectors
84  size_t size = 3, n = size * size, m = 1;
85  eigen_ad_vector a_x(n), a_y(m);
86  eigen_vector x(n);
87 
88  // set and declare independent variables and start tape recording
89  for (i = 0; i < size; i++)
90  {
91  for (j = 0; j < size; j++)
92  { // lower triangular matrix
93  a_x[(Eigen::DenseIndex)(i * size + j)] = x[(Eigen::DenseIndex)(i * size + j)] = 0.0;
94  if (j <= i)
95  a_x[(Eigen::DenseIndex)(i * size + j)] = x[(Eigen::DenseIndex)(i * size + j)] =
96  double(1 + i + j);
97  }
98  }
99  CppAD::Independent(a_x);
100 
101  // copy independent variable vector to a matrix
102  eigen_ad_matrix a_X(size, size);
103  eigen_matrix X(size, size);
104  for (i = 0; i < size; i++)
105  {
106  for (j = 0; j < size; j++)
107  {
108  X((Eigen::DenseIndex)i, (Eigen::DenseIndex)j) = x[(Eigen::DenseIndex)(i * size + j)];
109  // If we used a_X(i, j) = X(i, j), a_X would not depend on a_x.
110  a_X((Eigen::DenseIndex)i, (Eigen::DenseIndex)j) = a_x[(Eigen::DenseIndex)(i * size + j)];
111  }
112  }
113 
114  // Compute the log of determinant of X
115  a_y[0] = log(a_X.determinant());
116 
117  // create f: x -> y and stop tape recording
118  CppAD::ADFun<double> f(a_x, a_y);
119 
120  // check function value
121  double eps = 100. * CppAD::numeric_limits<double>::epsilon();
122  CppAD::det_by_minor<double> det(size);
123  BOOST_CHECK(NearEqual(Value(a_y[0]), log(det(x)), eps, eps));
124 
125  // compute the derivative of y w.r.t x using CppAD
126  eigen_vector jac = f.Jacobian(x);
127 
128  // check the derivative using the formula
129  // d/dX log(det(X)) = transpose( inv(X) )
130  eigen_matrix inv_X = X.inverse();
131  for (i = 0; i < size; i++)
132  {
133  for (j = 0; j < size; j++)
134  BOOST_CHECK(NearEqual(
135  jac[(Eigen::DenseIndex)(i * size + j)], inv_X((Eigen::DenseIndex)j, (Eigen::DenseIndex)i),
136  eps, eps));
137  }
138 }
139 
141 {
142  using CppAD::AD;
143  using CppAD::NearEqual;
144  double eps99 = 99.0 * std::numeric_limits<double>::epsilon();
145 
146  typedef AD<double> AD_double;
147 
148  double x0 = 1.;
149  CPPAD_TESTVECTOR(AD_double) x(1), y(1), z(1);
150  x[0] = x0;
151  CppAD::Independent(x);
152 
153  y[0] = CppAD::cos(x[0]);
154  BOOST_CHECK(NearEqual(y[0], std::cos(x0), eps99, eps99));
155  CppAD::ADFun<double> fcos(x, y);
156 
157  CPPAD_TESTVECTOR(double) x_eval(1);
158  x_eval[0] = x0;
159  CPPAD_TESTVECTOR(double) dy(1);
160  dy = fcos.Jacobian(x_eval);
161  BOOST_CHECK(NearEqual(dy[0], -std::sin(x0), eps99, eps99));
162 
163  CppAD::Independent(x);
164  z[0] = CppAD::sin(x[0]);
165  BOOST_CHECK(NearEqual(z[0], std::sin(x0), eps99, eps99));
166 
167  CppAD::ADFun<double> fsin(x, z);
168 
169  CPPAD_TESTVECTOR(double) dz(1);
170  dz = fsin.Jacobian(x_eval);
171  BOOST_CHECK(NearEqual(dz[0], std::cos(x0), eps99, eps99));
172 }
173 
174 BOOST_AUTO_TEST_CASE(test_eigen_min)
175 {
176  using CppAD::AD;
177 
178  typedef double Scalar;
179  typedef AD<double> ADScalar;
180  Eigen::Matrix<ADScalar, Eigen::Dynamic, 1> ad_X;
181  Eigen::Matrix<ADScalar, Eigen::Dynamic, 1> ad_Y;
182  ad_X.resize(2);
183  ad_Y.resize(2);
184 
185  Eigen::Vector2d x_test(-1, 1);
186  Eigen::Vector2d y_test = x_test.array().min(Scalar(0.));
187 
188  CppAD::Independent(ad_X);
189  // Function
190  ad_Y = ad_X.array().min(Scalar(0.));
191  CppAD::ADFun<Scalar> ad_fun(ad_X, ad_Y);
192 
193  CPPAD_TESTVECTOR(Scalar) x((size_t)2);
194  Eigen::Map<Eigen::Vector2d>(x.data(), 2, 1) = x_test;
195 
196  CPPAD_TESTVECTOR(Scalar) y = ad_fun.Forward(0, x);
197 
198  BOOST_CHECK(Eigen::Map<Eigen::Vector2d>(y.data(), 2, 1).isApprox(y_test));
199 }
200 
201 BOOST_AUTO_TEST_CASE(test_eigen_max)
202 {
203  using CppAD::AD;
204 
205  typedef double Scalar;
206  typedef AD<double> ADScalar;
207  Eigen::Matrix<ADScalar, Eigen::Dynamic, 1> ad_X;
208  Eigen::Matrix<ADScalar, Eigen::Dynamic, 1> ad_Y;
209  ad_X.resize(2);
210  ad_Y.resize(2);
211 
212  Eigen::Vector2d x_test(-1, 1);
213  Eigen::Vector2d y_test = x_test.array().max(Scalar(0.));
214 
215  CppAD::Independent(ad_X);
216  // Function
217  ad_Y = ad_X.array().max(Scalar(0.));
218  CppAD::ADFun<Scalar> ad_fun(ad_X, ad_Y);
219 
220  CPPAD_TESTVECTOR(Scalar) x((size_t)2);
221  Eigen::Map<Eigen::Vector2d>(x.data(), 2, 1) = x_test;
222 
223  CPPAD_TESTVECTOR(Scalar) y = ad_fun.Forward(0, x);
224 
225  BOOST_CHECK(Eigen::Map<Eigen::Vector2d>(y.data(), 2, 1).isApprox(y_test));
226 }
227 
228 BOOST_AUTO_TEST_CASE(test_eigen_support)
229 {
230  using namespace CppAD;
231 
232  // use a special object for source code generation
233  typedef AD<double> ADScalar;
234 
235  typedef Eigen::Matrix<ADScalar, Eigen::Dynamic, 1> ADVector;
236 
237  ADVector vec_zero(ADVector::Zero(100));
238  BOOST_CHECK(vec_zero.isZero());
239 
240  ADVector vec_ones(100);
241  vec_ones.fill(1);
242  BOOST_CHECK(vec_ones.isOnes());
243 }
244 
246 {
247  CppAD::AD<double> ad_value;
248  ad_value = -1.;
249  abs(ad_value);
250 }
251 
253 {
254  CppAD::AD<double> theta, x, y;
255  x = pinocchio::math::cos(theta);
256  y = pinocchio::math::sin(theta);
257 
258  pinocchio::math::atan2(y, x);
259 }
260 
261 BOOST_AUTO_TEST_SUITE_END()
test-cpp2pybind11.m
m
Definition: test-cpp2pybind11.py:25
simulation-closed-kinematic-chains.dy
dy
Definition: simulation-closed-kinematic-chains.py:141
codegen-rnea.jac
jac
Definition: codegen-rnea.py:41
y
y
inverse-kinematics.i
int i
Definition: inverse-kinematics.py:17
BOOST_AUTO_TEST_CASE
BOOST_AUTO_TEST_CASE(test_example1_cppad)
Definition: cppad/basic.cpp:17
pinocchio::python::Scalar
context::Scalar Scalar
Definition: admm-solver.cpp:29
simulation-closed-kinematic-chains.dz
dz
Definition: simulation-closed-kinematic-chains.py:140
autodiff-rnea.f
f
Definition: autodiff-rnea.py:24
CppAD
Definition: autodiff/cppad.hpp:146
collision-with-point-clouds.X
X
Definition: collision-with-point-clouds.py:36
pinocchio::Dynamic
const int Dynamic
Definition: fwd.hpp:140
size
FCL_REAL size() const
x
x
cppad.hpp
check
bool check(const T &value, const T &other)
ocp.x0
x0
Definition: ocp.py:15
dcrba.eps
int eps
Definition: dcrba.py:458
pinocchio.explog.log
def log(x)
Definition: bindings/python/pinocchio/explog.py:29
X
n
Vec3f n


pinocchio
Author(s):
autogenerated on Sun Dec 22 2024 03:41:06