LCOV - code coverage report
Current view: top level - tools - Minimise1DBrent.h (source / functions) Hit Total Coverage
Test: plumed test coverage Lines: 111 112 99.1 %
Date: 2025-03-25 09:33:27 Functions: 12 12 100.0 %

          Line data    Source code
       1             : /* +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
       2             :    Copyright (c) 2015-2023 The plumed team
       3             :    (see the PEOPLE file at the root of the distribution for a list of names)
       4             : 
       5             :    See http://www.plumed.org for more information.
       6             : 
       7             :    This file is part of plumed, version 2.
       8             : 
       9             :    plumed is free software: you can redistribute it and/or modify
      10             :    it under the terms of the GNU Lesser General Public License as published by
      11             :    the Free Software Foundation, either version 3 of the License, or
      12             :    (at your option) any later version.
      13             : 
      14             :    plumed is distributed in the hope that it will be useful,
      15             :    but WITHOUT ANY WARRANTY; without even the implied warranty of
      16             :    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
      17             :    GNU Lesser General Public License for more details.
      18             : 
      19             :    You should have received a copy of the GNU Lesser General Public License
      20             :    along with plumed.  If not, see <http://www.gnu.org/licenses/>.
      21             : +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ */
      22             : #ifndef __PLUMED_tools_Minimise1DBrent_h
      23             : #define __PLUMED_tools_Minimise1DBrent_h
      24             : 
      25             : #include "Tools.h"
      26             : 
      27             : #include <vector>
      28             : #include <string>
      29             : 
      30             : namespace PLMD {
      31             : 
      32             : /// A class for doing parabolic interpolation and minimisation of
      33             : /// 1D functions using Brent's method.
      34             : template <class FCLASS>
      35        1192 : class Minimise1DBrent {
      36             : private:
      37             : /// Has the minimum been bracketed
      38             :   bool bracketed;
      39             : /// Has the function been minimised
      40             :   bool minimised;
      41             : /// The tolerance for the line minimiser
      42             :   double tol;
      43             : /// The default ration by which successive intervals are magnified
      44             :   const double GOLD;
      45             : /// The maximum magnification allowed for a parabolic fit step
      46             :   const double GLIMIT;
      47             : /// Use to prevent any possible division by zero
      48             :   const double TINY;
      49             : /// Maximum number of interactions in line minimiser
      50             :   const unsigned ITMAX;
      51             : /// The value of the golden ratio
      52             :   const double CGOLD;
      53             : /// A small number that protects against trying to achieve fractional
      54             : /// accuracy for a minimum that happens to be exactly zero
      55             :   const double ZEPS;
      56             : /// This is the type specifier for the function to minimise
      57             :   typedef double(FCLASS::*eng_pointer)( const double& val );
      58             : /// Three points bracketting the minimum and the corresponding function values
      59             :   double ax,bx,cx,fa,fb,fc,fmin;
      60             : /// The class containing the function we are trying to minimise
      61             :   FCLASS myclass_func;
      62             : public:
      63        1192 :   explicit Minimise1DBrent( const FCLASS& pf,  const double& t=3.0E-8 );
      64             : /// Bracket the minium
      65             :   void bracket( const double& ax, const double& xx, eng_pointer eng );
      66             : /// Find the minimum between two brackets
      67             :   double minimise( eng_pointer eng );
      68             : /// Return the value of the function at the minimum
      69             :   double getMinimumValue() const ;
      70             : };
      71             : 
      72             : template <class FCLASS>
      73        1192 : Minimise1DBrent<FCLASS>::Minimise1DBrent( const FCLASS& pf, const double& t ):
      74        1192 :   bracketed(false),
      75        1192 :   minimised(false),
      76        1192 :   tol(t),
      77        1192 :   GOLD(1.618034),
      78        1192 :   GLIMIT(100.0),
      79        1192 :   TINY(1.0E-20),
      80        1192 :   ITMAX(100),
      81        1192 :   CGOLD(0.3819660),
      82        1192 :   ZEPS(epsilon*1.0E-3),
      83        1192 :   ax(0),bx(0),cx(0),
      84        1192 :   fa(0),fb(0),fc(0),
      85        1192 :   fmin(0),
      86        1192 :   myclass_func(pf) {
      87        1192 : }
      88             : 
      89             : template <class FCLASS>
      90        1192 : void Minimise1DBrent<FCLASS>::bracket( const double& a, const double& b, eng_pointer eng ) {
      91        1192 :   ax=a;
      92        1192 :   bx=b;
      93             :   double fu;
      94        1192 :   fa=(myclass_func.*eng)(ax);
      95        1192 :   fb=(myclass_func.*eng)(bx);
      96        1192 :   if( fb>fa ) {
      97             :     double tmp;
      98         155 :     tmp=ax;
      99         155 :     ax=bx;
     100         155 :     bx=tmp;
     101             :     tmp=fa;
     102         155 :     fa=fb;
     103         155 :     fb=tmp;
     104             :   }
     105        1192 :   cx=bx+GOLD*(bx-ax);
     106        1192 :   fc=(myclass_func.*eng)(cx);
     107        2638 :   while ( fb > fc ) {
     108        1521 :     double r=(bx-ax)*(fb-fc);
     109        1521 :     double q=(bx-cx)*(fb-fa);
     110        1521 :     double u=bx-((bx-cx)*q-(bx-ax)*r)/(2.0*(std::fabs(q-r)>TINY?std::fabs(q-r):TINY)*(q-r>=0?1:-1));
     111        1521 :     double ulim=bx+GLIMIT*(cx-bx);
     112        1521 :     if((bx-u)*(u-cx) > 0.0 ) {
     113         151 :       fu=(myclass_func.*eng)(u);
     114         151 :       if( fu < fc ) {
     115          67 :         ax=bx;
     116          67 :         bx=u;
     117          67 :         fa=fb;
     118          67 :         fb=fu;
     119          67 :         bracketed=true;
     120          75 :         return;
     121          84 :       } else if( fu > fb ) {
     122           8 :         cx=u;
     123           8 :         fc=fu;
     124           8 :         bracketed=true;
     125           8 :         return;
     126             :       }
     127          76 :       u=cx+GOLD*(cx-bx);
     128          76 :       fu=(myclass_func.*eng)(u);
     129        1370 :     } else if((cx-u)*(u-ulim) > 0.0 ) {
     130        1052 :       fu=(myclass_func.*eng)(u);
     131        1052 :       if( fu<fc ) {
     132        1041 :         bx=cx;
     133        1041 :         cx=u;
     134        1041 :         u+=GOLD*(u-bx);
     135        1041 :         fb=fc;
     136        1041 :         fc=fu;
     137        1041 :         fu=(myclass_func.*eng)(u);
     138             :       }
     139         318 :     } else if( (u-ulim)*(ulim-cx) >= 0.0 ) {
     140         283 :       u=ulim;
     141         283 :       fu=(myclass_func.*eng)(u);
     142             :     } else {
     143          35 :       u=cx+GOLD*(cx-bx);
     144          35 :       fu=(myclass_func.*eng)(u);
     145             :     }
     146        1446 :     ax=bx;
     147        1446 :     bx=cx;
     148        1446 :     cx=u;
     149        1446 :     fa=fb;
     150        1446 :     fb=fc;
     151        1446 :     fc=fu;
     152             :   }
     153        1117 :   bracketed=true;
     154             : }
     155             : 
     156             : template <class FCLASS>
     157        1192 : double Minimise1DBrent<FCLASS>::minimise( eng_pointer eng ) {
     158             :   plumed_dbg_assert( bracketed );
     159             : 
     160             :   double a,b,d=0.0,etemp,fu,fv,fw,fx;
     161             :   double p,q,r,tol1,tol2,u,v,w,x,xm;
     162             :   double e=0.0;
     163             : 
     164        1192 :   a=(ax < cx ? ax : cx );
     165        1192 :   b=(ax >= cx ? ax : cx );
     166        1192 :   x=w=v=bx;
     167        1192 :   fw=fv=fx=(myclass_func.*eng)(x);
     168       18942 :   for(unsigned iter=0; iter<ITMAX; ++iter) {
     169       18942 :     xm=0.5*(a+b);
     170       18942 :     tol2=2.0*(tol1=tol*std::fabs(x)+ZEPS);
     171       18942 :     if( std::fabs(x-xm) <= (tol2-0.5*(b-a))) {
     172        1192 :       fmin=fx;
     173        1192 :       minimised=true;
     174        1192 :       return x;
     175             :     }
     176       17750 :     if( std::fabs(e) > tol1 ) {
     177       16377 :       r=(x-w)*(fx-fv);
     178       16377 :       q=(x-v)*(fx-fw);
     179       16377 :       p=(x-v)*q-(x-w)*r;
     180       16377 :       q=2.0*(q-r);
     181       16377 :       if( q > 0.0 ) {
     182        7818 :         p = -p;
     183             :       }
     184       16377 :       q=std::fabs(q);
     185             :       etemp=e;
     186             :       e=d;
     187       16377 :       if( std::fabs(p) >= std::fabs(0.5*q*etemp) || p <= q*(a-x) || p >= q*(b-x) ) {
     188        7607 :         d = CGOLD*(e=(x >= xm ? a-x : b-x ));
     189             :       } else {
     190        8770 :         d=p/q;
     191        8770 :         u=x+d;
     192        8770 :         if(u-a < tol2 || b-u < tol2 ) {
     193         705 :           d=(xm-x>=0?std::fabs(tol1):-std::fabs(tol1));
     194             :         }
     195             :       }
     196             :     } else {
     197        1373 :       d=CGOLD*(e=( x >= xm ? a-x : b-x ));
     198             :     }
     199       17750 :     if( std::fabs(d)>=tol1) {
     200       16896 :       u=x+d;
     201             :     } else {
     202         854 :       u=x+(d>=0?std::fabs(tol1):-std::fabs(tol1));
     203             :     }
     204       17750 :     fu=(myclass_func.*eng)(u);
     205       17750 :     if( fu <= fx ) {
     206        5911 :       if( u >= x ) {
     207             :         a=x;
     208             :       } else {
     209             :         b=x;
     210             :       }
     211             :       v=w;
     212             :       fv=fw;
     213             :       w=x;
     214             :       fw=fx;
     215        5911 :       x=u;
     216             :       fx=fu;
     217             :     } else {
     218       11839 :       if( u < x ) {
     219             :         a=u;
     220             :       } else {
     221             :         b=u;
     222             :       }
     223       11839 :       if( fu <=fw || w==x ) {
     224             :         v=w;
     225             :         w=u;
     226             :         fv=fw;
     227             :         fw=fu;
     228        7526 :       } else if( fu <= fv || v==x || v==w ) {
     229             :         v=u;
     230             :         fv=fu;
     231             :       }
     232             :     }
     233             :   }
     234           0 :   plumed_merror("Too many interactions in brent");
     235             : }
     236             : 
     237             : template <class FCLASS>
     238        1192 : double Minimise1DBrent<FCLASS>::getMinimumValue() const {
     239             :   plumed_dbg_assert( minimised );
     240        1192 :   return fmin;
     241             : }
     242             : 
     243             : }
     244             : #endif

Generated by: LCOV version 1.16