Skip to content

Commit

Permalink
Merge pull request #69 from juanlucasrey/partial_erfc_support
Browse files Browse the repository at this point in the history
erfc partial support
  • Loading branch information
TimSiebert1 authored Nov 8, 2024
2 parents 9a19766 + 03ea6c6 commit 6ababbb
Show file tree
Hide file tree
Showing 23 changed files with 449 additions and 17 deletions.
93 changes: 93 additions & 0 deletions ADOL-C/boost-test/traceOperatorScalar.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -4514,6 +4514,99 @@ BOOST_AUTO_TEST_CASE(ErfOperator_FOS_Reverse)
myfree1(z);
}

BOOST_AUTO_TEST_CASE(ErfcOperator_ZOS_Forward)
{
double a = 7.1, aout;
adouble ad;

trace_on(1);
ad <<= a;

ad = erfc(ad);

ad >>= aout;
trace_off();

a = std::erfc(a);

BOOST_TEST(aout == a, tt::tolerance(tol));

double *x = myalloc1(1);
double *y = myalloc1(1);

*x = 7.1;

zos_forward(1, 1, 1, 0, x, y);

BOOST_TEST(*y == a, tt::tolerance(tol));

myfree1(x);
myfree1(y);
}

BOOST_AUTO_TEST_CASE(ErfcOperator_FOS_Forward)
{
double a = 7.1, aout;
adouble ad;

trace_on(1);
ad <<= a;

ad = erfc(ad);

ad >>= aout;
trace_off();

double aDerivative = -2. / std::sqrt(std::acos(-1.)) * std::exp(-a * a);
a = std::erfc(a);

double *x = myalloc1(1);
double *xd = myalloc1(1);
double *y = myalloc1(1);
double *yd = myalloc1(1);

*x = 7.1;
*xd = 1.;

fos_forward(1, 1, 1, 0, x, xd, y, yd);

BOOST_TEST(*y == a, tt::tolerance(tol));
BOOST_TEST(*yd == aDerivative, tt::tolerance(tol));

myfree1(x);
myfree1(xd);
myfree1(y);
myfree1(yd);
}

BOOST_AUTO_TEST_CASE(ErfcOperator_FOS_Reverse)
{
double a = 7.1, aout;
adouble ad;

trace_on(1, 1);
ad <<= a;

ad = erfc(ad);

ad >>= aout;
trace_off();

double aDerivative = -2. / std::sqrt(std::acos(-1.)) * std::exp(-a * a);

double *u = myalloc1(1);
double *z = myalloc1(1);

*u = 1.;

fos_reverse(1, 1, 1, u, z);

BOOST_TEST(*z == aDerivative, tt::tolerance(tol));

myfree1(u);
myfree1(z);
}

BOOST_AUTO_TEST_CASE(EqOperator_ZOS_Forward)
{
double a = 10.01, aout;
Expand Down
69 changes: 69 additions & 0 deletions ADOL-C/boost-test/traceOperatorVector.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -3394,6 +3394,75 @@ BOOST_AUTO_TEST_CASE(ErfOperator_FOV_Reverse)
myfree2(z);
}

BOOST_AUTO_TEST_CASE(ErfcOperator_FOV_Forward)
{
double a = 7.1, aout;
adouble ad;

trace_on(1);
ad <<= a;

ad = erfc(ad);

ad >>= aout;
trace_off();

double aDerivative = -2. / std::sqrt(std::acos(-1.)) * std::exp(-a*a);
a = std::erfc(a);

double *x = myalloc1(1);
double **xd = myalloc2(1, 2);
double *y = myalloc1(1);
double **yd = myalloc2(1, 2);

x[0] = 7.1;

for (int i = 0; i < 2; i++) {
xd[0][i] = std::pow(3., i*2.);
}

fov_forward(1, 1, 1, 2, x, xd, y, yd);

BOOST_TEST(*y == a, tt::tolerance(tol));
BOOST_TEST(yd[0][0] == aDerivative, tt::tolerance(tol));
BOOST_TEST(yd[0][1] == aDerivative * std::pow(3., 2.), tt::tolerance(tol));

myfree1(x);
myfree2(xd);
myfree1(y);
myfree2(yd);
}

BOOST_AUTO_TEST_CASE(ErfcOperator_FOV_Reverse)
{
double a = 7.1, aout;
adouble ad;

trace_on(1, 1);
ad <<= a;

ad = erfc(ad);

ad >>= aout;
trace_off();

double aDerivative = -2. / std::sqrt(std::acos(-1.)) * std::exp(-a*a);

double **u = myalloc2(2, 1);
double **z = myalloc2(2, 1);

u[0][0] = 1.;
u[1][0] = -1.1;

fov_reverse(1, 1, 1, 2, u, z);

BOOST_TEST(z[0][0] == aDerivative, tt::tolerance(tol));
BOOST_TEST(z[1][0] == -1.1*aDerivative, tt::tolerance(tol));

myfree2(u);
myfree2(z);
}

BOOST_AUTO_TEST_CASE(EqPlusOperator_FOV_Forward)
{
double a = 5.132, aout;
Expand Down
26 changes: 26 additions & 0 deletions ADOL-C/boost-test/tracelessCompositeTests.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -282,6 +282,32 @@ BOOST_AUTO_TEST_CASE(CompositeErfFabs_Traceless)
BOOST_TEST(ax1.getADValue(1) == x2Derivative, tt::tolerance(tol));
}

BOOST_AUTO_TEST_CASE(CompositeErfcFabs_Traceless)
{
double x1 = 4.56, x2 = 6.45;
adouble ax1 = x1, ax2 = x2;

ax1.setADValue(0, 1.);
ax2.setADValue(1, 1.);

ax1 = adtl::erfc(adtl::fabs(ax1 - 2.)*adtl::sinh(ax2));

double x1Derivative = -2. / std::sqrt(std::acos(-1.))
* std::exp(-std::pow(std::fabs(x1 - 2.)
* std::sinh(x2), 2))
* std::sinh(x2);
double x2Derivative = -2. / std::sqrt(std::acos(-1.))
* std::exp(-std::pow(std::fabs(x1 - 2.)
* std::sinh(x2), 2))
* std::fabs(x1 - 2.) * std::cosh(x2);

x1 = std::erfc(std::fabs(x1 - 2.)*std::sinh(x2));

BOOST_TEST(ax1.getValue() == x1, tt::tolerance(tol));
BOOST_TEST(ax1.getADValue(0) == x1Derivative, tt::tolerance(tol));
BOOST_TEST(ax1.getADValue(1) == x2Derivative, tt::tolerance(tol));
}

BOOST_AUTO_TEST_CASE(ExpTrigSqrt_Traceless)
{
double x1 = 1.2, x2 = 2.1;
Expand Down
30 changes: 30 additions & 0 deletions ADOL-C/boost-test/tracelessOperatorScalar.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -1436,6 +1436,36 @@ BOOST_AUTO_TEST_CASE(ErfOperatorDerivative)
BOOST_TEST(ad.getADValue(0) == aDerivative, tt::tolerance(tol));
}

/* The complementary error function erfc(a) is defined as
* 1.0 - erf(a).
*/
BOOST_AUTO_TEST_CASE(ErfcOperatorPrimal)
{
double a = 7.1;
adouble ad = a;

a = std::erfc(a);
ad = adtl::erfc(ad);

BOOST_TEST(ad.getValue() == a, tt::tolerance(tol));
}

BOOST_AUTO_TEST_CASE(ErfcOperatorDerivative)
{
double a = 7.1;
adouble ad = a;

/* The inverse cosine is used for pi so that no additional math library
* import is necessary, see also the implementation in adtl.h.
*/
double aDerivative = -2. / std::sqrt(std::acos(-1.)) * std::exp(-a * a);

ad.setADValue(0, 1.);
ad = adtl::erfc(ad);

BOOST_TEST(ad.getADValue(0) == aDerivative, tt::tolerance(tol));
}

/* Test the primitive non-temporary operations =, +=, -=, *=, /=. */

BOOST_AUTO_TEST_CASE(EqOperatorPrimal_Derivative)
Expand Down
15 changes: 15 additions & 0 deletions ADOL-C/boost-test/tracelessOperatorVector.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -868,6 +868,21 @@ BOOST_AUTO_TEST_CASE(ErfOperatorDerivativeVectorMode)
BOOST_TEST(ad.getADValue(1) == aDerivative*(-2.5), tt::tolerance(tol));
}

BOOST_AUTO_TEST_CASE(ErfcOperatorDerivativeVectorMode)
{
double a = 7.1;
adouble ad = a;

double aDerivative = -2. / std::sqrt(std::acos(-1.)) * std::exp(-a * a);

ad.setADValue(0, 1.);
ad.setADValue(1, -2.5);
ad = adtl::erfc(ad);

BOOST_TEST(ad.getADValue(0) == aDerivative, tt::tolerance(tol));
BOOST_TEST(ad.getADValue(1) == aDerivative*(-2.5), tt::tolerance(tol));
}

BOOST_AUTO_TEST_CASE(EqOperatorDerivativeVectorMode)
{
double a = 10.01;
Expand Down
4 changes: 4 additions & 0 deletions ADOL-C/c_interface/ADOLC_TB_interface.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -261,6 +261,10 @@ extern "C"
{
return new adouble(erf(*static_cast<adouble *>(a)));
}
TBAdoubleHandle tb_erfc(TBAdoubleHandle a)
{
return new adouble(erfc(*static_cast<adouble *>(a)));
}
}

/*
Expand Down
3 changes: 2 additions & 1 deletion ADOL-C/c_interface/ADOLC_TB_interface.h
Original file line number Diff line number Diff line change
Expand Up @@ -115,6 +115,7 @@ extern "C"
TBAdoubleHandle tb_floor(TBAdoubleHandle a);
TBAdoubleHandle tb_ldexp(TBAdoubleHandle a, int n);
TBAdoubleHandle tb_erf(TBAdoubleHandle a);
TBAdoubleHandle tb_erfc(TBAdoubleHandle a);
#ifdef __cplusplus
}
#endif
Expand All @@ -139,4 +140,4 @@ extern "C"
}
#endif

#endif // ADOLC_TB_INTERFACE_H
#endif // ADOLC_TB_INTERFACE_H
4 changes: 4 additions & 0 deletions ADOL-C/c_interface/ADOLC_TL_interface.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -292,4 +292,8 @@ extern "C"
{
return new adtl::adouble(erf(*static_cast<adtl::adouble *>(a)));
}
TLAdoubleHandle tl_erfc(TLAdoubleHandle a)
{
return new adtl::adouble(erfc(*static_cast<adtl::adouble *>(a)));
}
}
3 changes: 2 additions & 1 deletion ADOL-C/c_interface/ADOLC_TL_interface.h
Original file line number Diff line number Diff line change
Expand Up @@ -122,8 +122,9 @@ extern "C"
TLAdoubleHandle tl_floor(TLAdoubleHandle a);
TLAdoubleHandle tl_ldexp(TLAdoubleHandle a, const int n);
TLAdoubleHandle tl_erf(TLAdoubleHandle a);
TLAdoubleHandle tl_erfc(TLAdoubleHandle a);
#ifdef __cplusplus
}
#endif

#endif // ADOLC_TL_INTERFACE_H
#endif // ADOLC_TL_INTERFACE_H
11 changes: 11 additions & 0 deletions ADOL-C/include/adolc/adoublecuda.h
Original file line number Diff line number Diff line change
Expand Up @@ -152,6 +152,7 @@ class adouble {
CUDADEV friend adouble ldexp (const double v, const adouble &a);
CUDADEV friend double frexp (const adouble &a, int* v);
CUDADEV friend adouble erf (const adouble &a);
CUDADEV friend adouble erfc (const adouble &a);


/******************* nontemporary results ***************************/
Expand Down Expand Up @@ -794,6 +795,16 @@ CUDADEV adouble erf (const adouble &a) {
return tmp;
}

CUDADEV adouble erfc (const adouble &a) {
adouble tmp;
tmp.val=ADOLC_MATH_NSP_ERF::erfc(a.val);
double tmp2 = -2.0 /
ADOLC_MATH_NSP_ERF::sqrt(ADOLC_MATH_NSP::acos(-1.0)) *
ADOLC_MATH_NSP_ERF::exp(-a.val*a.val);
FOR_I_EQ_0_LT_NUMDIR
tmp.ADVAL_I=tmp2*a.ADVAL_I;
return tmp;
}

/******************* nontemporary results *********************************/
CUDADEV void adouble::operator = (const double v) {
Expand Down
12 changes: 12 additions & 0 deletions ADOL-C/include/adolc/adtl.h
Original file line number Diff line number Diff line change
Expand Up @@ -137,6 +137,7 @@ class adouble {
inline friend adouble ldexp (const double v, const adouble &a);
inline friend double frexp (const adouble &a, int* v);
inline friend adouble erf (const adouble &a);
inline friend adouble erfc (const adouble &a);

inline friend void condassign( adouble &res, const adouble &cond,
const adouble &arg1, const adouble &arg2 );
Expand Down Expand Up @@ -866,6 +867,17 @@ inline adouble erf (const adouble &a) {
return tmp;
}

inline adouble erfc (const adouble &a) {
adouble tmp;
tmp.PRIMAL_VALUE=ADOLC_MATH_NSP_ERF::erfc(a.PRIMAL_VALUE);
double tmp2 = -2.0 /
ADOLC_MATH_NSP_ERF::sqrt(ADOLC_MATH_NSP::acos(-1.0)) *
ADOLC_MATH_NSP_ERF::exp(-a.PRIMAL_VALUE*a.PRIMAL_VALUE);
FOR_I_EQ_1_LTEQ_NUMDIR
tmp.ADVAL_I=tmp2*a.ADVAL_I;
return tmp;
}

inline void condassign( adouble &res, const adouble &cond,
const adouble &arg1, const adouble &arg2 ) {
if (cond.getValue() > 0)
Expand Down
Loading

0 comments on commit 6ababbb

Please sign in to comment.