/*
Copyright 2013 SINTEF ICT, Applied Mathematics.
Copyright 2013 Statoil ASA.
This file is part of the Open Porous Media Project (OPM).
OPM is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
OPM is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with OPM. If not, see .
*/
#ifndef OPM_AUTODIFF_HPP_HEADER
#define OPM_AUTODIFF_HPP_HEADER
#include
namespace Opm
{
/// A simple class for forward-mode automatic differentiation.
///
/// The class represents a single value and a single derivative.
/// Only basic arithmetic operators and a few functions are
/// implemented for it, it is mostly intended for simple
/// experimentation.
template
class AutoDiff
{
public:
/// Create an AutoDiff object representing a constant, that
/// is, its derivative is zero.
static AutoDiff
constant(const Scalar x)
{
return function(x, Scalar(0));
}
/// Create an AutoDiff object representing a primary variable,
/// that is, its derivative is one.
static AutoDiff
variable(const Scalar x)
{
return function(x, Scalar(1));
}
/// Create an AutoDiff object representing a function value
/// and its derivative.
static AutoDiff
function(const Scalar x, const Scalar dx)
{
return AutoDiff(x, dx);
}
void
operator +=(const Scalar& rhs)
{
val_ += rhs;
}
void
operator +=(const AutoDiff& rhs)
{
val_ += rhs.val_;
der_ += rhs.der_;
}
void
operator -=(const Scalar& rhs)
{
val_ -= rhs;
}
void
operator -=(const AutoDiff& rhs)
{
val_ -= rhs.val_;
der_ -= rhs.der_;
}
void
operator *=(const Scalar& rhs)
{
val_ *= rhs;
der_ *= rhs;
}
void
operator *=(const AutoDiff& rhs)
{
der_ = der_*rhs.val_ + val_*rhs.der_;
val_ *= rhs.val_;
}
void
operator /=(const Scalar& rhs)
{
val_ /= rhs;
der_ /= rhs;
}
void
operator /=(const AutoDiff& rhs)
{
der_ = (der_*rhs.val_ - val_*rhs.der_) / (rhs.val_ * rhs.val_);
val_ /= rhs.val_;
}
template
Ostream&
print(Ostream& os) const
{
os << "(x,dx) = (" << val_ << ',' << der_ << ")";
return os;
}
const Scalar val() const { return val_; }
const Scalar der() const { return der_; }
private:
AutoDiff(const Scalar x, const Scalar dx)
: val_(x), der_(dx)
{}
Scalar val_;
Scalar der_;
};
template
Ostream&
operator<<(Ostream& os, const AutoDiff& fw)
{
return fw.print(os);
}
template
AutoDiff
operator +(const AutoDiff& lhs,
const AutoDiff& rhs)
{
AutoDiff ret = lhs;
ret += rhs;
return ret;
}
template
AutoDiff
operator +(const T lhs,
const AutoDiff& rhs)
{
AutoDiff ret = rhs;
ret += Scalar(lhs);
return ret;
}
template
AutoDiff
operator +(const AutoDiff& lhs,
const T rhs)
{
AutoDiff ret = lhs;
ret += Scalar(rhs);
return ret;
}
template
AutoDiff
operator -(const AutoDiff& lhs,
const AutoDiff& rhs)
{
AutoDiff ret = lhs;
ret -= rhs;
return ret;
}
template
AutoDiff
operator -(const T lhs,
const AutoDiff& rhs)
{
return AutoDiff::function(Scalar(lhs) - rhs.val(), -rhs.der());
}
template
AutoDiff
operator -(const AutoDiff& lhs,
const T rhs)
{
AutoDiff ret = lhs;
ret -= Scalar(rhs);
return ret;
}
template
AutoDiff
operator *(const AutoDiff& lhs,
const AutoDiff& rhs)
{
AutoDiff ret = lhs;
ret *= rhs;
return ret;
}
template
AutoDiff
operator *(const T lhs,
const AutoDiff& rhs)
{
AutoDiff ret = rhs;
ret *= Scalar(lhs);
return ret;
}
template
AutoDiff
operator *(const AutoDiff& lhs,
const T rhs)
{
AutoDiff ret = lhs;
ret *= Scalar(rhs);
return ret;
}
template
AutoDiff
operator /(const AutoDiff& lhs,
const AutoDiff& rhs)
{
AutoDiff ret = lhs;
ret /= rhs;
return ret;
}
template
AutoDiff
operator /(const T lhs,
const AutoDiff& rhs)
{
Scalar a = Scalar(lhs) / rhs.val();
Scalar b = (-Scalar(lhs) / (rhs.val() * rhs.val())) * rhs.der();
return AutoDiff::function(a, b);
}
template
AutoDiff
operator /(const AutoDiff& lhs,
const T rhs)
{
Scalar a = lhs.val() / Scalar(rhs);
Scalar b = lhs.der() / Scalar(rhs);
return AutoDiff::function(a, b);
}
template
AutoDiff
cos(const AutoDiff& x)
{
Scalar a = std::cos(x.val());
Scalar b = -std::sin(x.val()) * x.der();
return AutoDiff::function(a, b);
}
template
AutoDiff
sqrt(const AutoDiff& x)
{
Scalar a = std::sqrt(x.val());
Scalar b = (Scalar(1.0) / (Scalar(2.0) * a)) * x.der();
return AutoDiff::function(a, b);
}
} // namespace Opm
namespace std {
using Opm::cos;
using Opm::sqrt;
}
#endif /* OPM_AUTODIFF_HPP_HEADER */