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 88 89 90 91 92 93 94 95 96 97 98 99 100 101
|
// 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
// ----------------------------------------------------------------------------
// adolc examples should suppress conversion warnings
# include <cppad/wno_conversion.hpp>
# include <adolc/adouble.h>
# include <adolc/taping.h>
# include <adolc/interfaces.h>
// adouble definitions not in Adolc distribution and
// required in order to use CppAD::AD<adouble>
# include <cppad/example/base_adolc.hpp>
# include <cppad/cppad.hpp>
bool base_adolc(void)
{ bool ok = true; // initialize test result
typedef adouble ADdouble; // for first level of taping
typedef CppAD::AD<ADdouble> ADDdouble; // for second level of taping
size_t n = 4; // number independent variables
CPPAD_TESTVECTOR(ADdouble) a_x(n);
CPPAD_TESTVECTOR(ADDdouble) aa_x(n);
// value of the independent variables
short tag = 0; // Adolc setup
int keep = 1;
trace_on(tag, keep);
size_t j;
for(j = 0; j < n; j++)
a_x[j] <<= double(n - j); // a_x is independent for ADdouble
for(j = 0; j < n; j++)
aa_x[j] = a_x[j]; // track how aa_x depends on a_x
CppAD::Independent(aa_x); // aa_x is independent for ADDdouble
// compute function
size_t m = 5;
CPPAD_TESTVECTOR(ADDdouble) aa_f(m); // dependent variables
// different cases of conditional expressions
aa_f[0] = CondExpLt(aa_x[0], aa_x[1], aa_x[2], aa_x[3]);
aa_f[1] = CondExpLe(aa_x[0], aa_x[1], aa_x[2], aa_x[3]);
aa_f[2] = CondExpEq(aa_x[0], aa_x[1], aa_x[2], aa_x[3]);
aa_f[3] = CondExpGe(aa_x[0], aa_x[1], aa_x[2], aa_x[3]);
aa_f[4] = CondExpGt(aa_x[0], aa_x[1], aa_x[2], aa_x[3]);
// declare inner function (just to stop inner taping)
CppAD::ADFun<ADdouble> a_F(aa_x, aa_f);
// set values for outer function same as inner function
// (corresponding to the tape of adobule operations)
double f_j;
for(j = 0; j < m; j++)
Value(aa_f[j]) >>= f_j;
trace_off();
// arrays for Adolc zos_forward
double *x = nullptr, *y = nullptr;
x = CPPAD_TRACK_NEW_VEC(n, x);
y = CPPAD_TRACK_NEW_VEC(m, y);
// switch order of arguments from when taping was done
for(j = 0; j < n; j++)
x[j] = double(j);
zos_forward(tag, int(m), int(n), keep, x, y);
// CondExpLt(0, 1, 2, 3)
ok &= (y[0] == double(2));
// CondExpLe(0, 1, 2, 3)
ok &= (y[1] == double(2));
// CondExpEq(0, 1, 2, 3)
ok &= (y[2] == double(3));
// CondExpGe(0, 1, 2, 3)
ok &= (y[3] == double(3));
// CondExpGt(0, 1, 2, 3)
ok &= (y[4] == double(3));
// set left = right and true < false
x[0] = x[1] = 1.;
x[2] = 2.;
x[3] = 3.;
zos_forward(tag, int(m), int(n), keep, x, y);
// CondExpLt(1, 1, 2, 3)
ok &= (y[0] == double(3));
// CondExpLe(1, 1, 2, 3)
ok &= (y[1] == double(2));
// CondExpEq(1, 1, 2, 3)
ok &= (y[2] == double(2));
// CondExpGe(1, 1, 2, 3)
ok &= (y[3] == double(2));
// CondExpGt(1, 1, 2, 3)
ok &= (y[4] == double(3));
CPPAD_TRACK_DEL_VEC(x);
CPPAD_TRACK_DEL_VEC(y);
return ok;
}
// END PROGRAM
|