/*=========================================================================== // // File: AutoDiff.hpp // // Created: 2013-04-29 10:51:23+0200 // // Authors: Knut-Andreas Lie // Halvor M. Nilsen // Atgeirr F. Rasmussen // Xavier Raynaud // Bård Skaflestad // //==========================================================================*/ /* 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 . */ #include #ifndef OPM_AUTODIFF_HPP_HEADER #define OPM_AUTODIFF_HPP_HEADER namespace AutoDiff { template class Forward { public: static Forward constant(const Scalar x) { // Constant is function with zero derivative. return function(x, Scalar(0)); } static Forward variable(const Scalar x) { // Variable is function with unit derivative (wrt. itself). return function(x, Scalar(1)); } static Forward function(const Scalar x, const Scalar dx) { return Forward(x, dx); } void operator +=(const Scalar& rhs) { val_ += rhs; } void operator +=(const Forward& rhs) { val_ += rhs.val_; der_ += rhs.der_; } void operator -=(const Scalar& rhs) { val_ -= rhs; } void operator -=(const Forward& rhs) { val_ -= rhs.val_; der_ -= rhs.der_; } void operator *=(const Scalar& rhs) { val_ *= rhs; der_ *= rhs; } void operator *=(const Forward& rhs) { der_ = der_*rhs.val_ + val_*rhs.der_; val_ *= rhs.val_; } void operator /=(const Scalar& rhs) { val_ /= rhs; der_ /= rhs; } void operator /=(const Forward& 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: Forward(const Scalar x, const Scalar dx) : val_(x), der_(dx) {} Scalar val_; Scalar der_; }; template Ostream& operator<<(Ostream& os, const Forward& fw) { return fw.print(os); } template Forward operator +(const Forward& lhs, const Forward& rhs) { Forward ret = lhs; ret += rhs; return ret; } template Forward operator +(const T lhs, const Forward& rhs) { Forward ret = rhs; ret += Scalar(lhs); return ret; } template Forward operator +(const Forward& lhs, const T rhs) { Forward ret = lhs; ret += Scalar(rhs); return ret; } template Forward operator -(const Forward& lhs, const Forward& rhs) { Forward ret = lhs; ret -= rhs; return ret; } template Forward operator -(const T lhs, const Forward& rhs) { return Forward::function(Scalar(lhs) - rhs.val(), -rhs.der()); } template Forward operator -(const Forward& lhs, const T rhs) { Forward ret = lhs; ret -= Scalar(rhs); return ret; } template Forward operator *(const Forward& lhs, const Forward& rhs) { Forward ret = lhs; ret *= rhs; return ret; } template Forward operator *(const T lhs, const Forward& rhs) { Forward ret = rhs; ret *= Scalar(lhs); return ret; } template Forward operator *(const Forward& lhs, const T rhs) { Forward ret = lhs; ret *= Scalar(rhs); return ret; } template Forward operator /(const Forward& lhs, const Forward& rhs) { Forward ret = lhs; ret /= rhs; return ret; } template Forward operator /(const T lhs, const Forward& rhs) { Scalar a = Scalar(lhs) / rhs.val(); Scalar b = -Scalar(lhs) / (rhs.val() * rhs.val()); return Forward::function(a, b); } template Forward operator /(const Forward& lhs, const T rhs) { Scalar a = lhs.val() / Scalar(rhs); Scalar b = lhs.der() / Scalar(rhs); return Forward::function(a, b); } template Forward cos(const Forward& x) { Scalar a = std::cos(x.val()); Scalar b = -std::sin(x.val()) * x.der(); return Forward::function(a, b); } template Forward sqrt(const Forward& x) { Scalar a = std::sqrt(x.val()); Scalar b = (Scalar(1.0) / (Scalar(2.0) * a)) * x.der(); return Forward::function(a, b); } } // namespace AutoDiff namespace std { using AutoDiff::cos; using AutoDiff::sqrt; } #endif /* OPM_AUTODIFF_HPP_HEADER */