LCOV - code coverage report
Current view: top level - bias - MovingRestraint.cpp (source / functions) Hit Total Coverage
Test: plumed test coverage Lines: 121 124 97.6 %
Date: 2025-03-25 09:33:27 Functions: 3 4 75.0 %

          Line data    Source code
       1             : /* +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
       2             :    Copyright (c) 2011-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             : #include "Bias.h"
      23             : #include "core/ActionRegister.h"
      24             : 
      25             : namespace PLMD {
      26             : namespace bias {
      27             : 
      28             : //+PLUMEDOC BIAS MOVINGRESTRAINT
      29             : /*
      30             : Add a time-dependent, harmonic restraint on one or more variables.
      31             : 
      32             : This form of bias can be used to performed steered MD \cite Grubmuller3
      33             : and Jarzynski sampling \cite jarzynski.
      34             : 
      35             : The harmonic restraint on your system is given by:
      36             : 
      37             : \f[
      38             : V(\vec{s},t) = \frac{1}{2} \kappa(t) ( \vec{s} - \vec{s}_0(t) )^2
      39             : \f]
      40             : 
      41             : The time dependence of \f$\kappa\f$ and \f$\vec{s}_0\f$ are specified by a list of
      42             : STEP, KAPPA and AT keywords.  These keywords tell plumed what values \f$\kappa\f$ and \f$\vec{s}_0\f$
      43             : should have at the time specified by the corresponding STEP keyword.  In between these times
      44             : the values of \f$\kappa\f$ and \f$\vec{s}_0\f$ are linearly interpolated.
      45             : 
      46             : Additional material and examples can be also found in the tutorial \ref belfast-5
      47             : 
      48             : \par Examples
      49             : 
      50             : The following input is dragging the distance between atoms 2 and 4
      51             : from 1 to 2 in the first 1000 steps, then back in the next 1000 steps.
      52             : In the following 500 steps the restraint is progressively switched off.
      53             : \plumedfile
      54             : DISTANCE ATOMS=2,4 LABEL=d
      55             : MOVINGRESTRAINT ...
      56             :   ARG=d
      57             :   STEP0=0    AT0=1.0 KAPPA0=100.0
      58             :   STEP1=1000 AT1=2.0
      59             :   STEP2=2000 AT2=1.0
      60             :   STEP3=2500         KAPPA3=0.0
      61             : ... MOVINGRESTRAINT
      62             : \endplumedfile
      63             : The following input is progressively building restraints
      64             : distances between atoms 1 and 5 and between atoms 2 and 4
      65             : in the first 1000 steps. Afterwards, the restraint is kept
      66             : static.
      67             : \plumedfile
      68             : DISTANCE ATOMS=1,5 LABEL=d1
      69             : DISTANCE ATOMS=2,4 LABEL=d2
      70             : MOVINGRESTRAINT ...
      71             :   ARG=d1,d2
      72             :   STEP0=0    AT0=1.0,1.5 KAPPA0=0.0,0.0
      73             :   STEP1=1000 AT1=1.0,1.5 KAPPA1=1.0,1.0
      74             : ... MOVINGRESTRAINT
      75             : \endplumedfile
      76             : The following input is progressively bringing atoms 1 and 2
      77             : close to each other with an upper wall
      78             : \plumedfile
      79             : DISTANCE ATOMS=1,2 LABEL=d1
      80             : MOVINGRESTRAINT ...
      81             :   ARG=d1
      82             :   VERSE=U
      83             :   STEP0=0    AT0=1.0 KAPPA0=10.0
      84             :   STEP1=1000 AT1=0.0
      85             : ... MOVINGRESTRAINT
      86             : \endplumedfile
      87             : 
      88             : By default the Action is issuing some values which are
      89             : the work on each degree of freedom, the center of the harmonic potential,
      90             : the total bias deposited
      91             : 
      92             : (See also \ref DISTANCE).
      93             : 
      94             : \attention Work is not computed properly when KAPPA is time dependent.
      95             : 
      96             : */
      97             : //+ENDPLUMEDOC
      98             : 
      99             : 
     100             : class MovingRestraint : public Bias {
     101             :   std::vector<std::vector<double> > at;
     102             :   std::vector<std::vector<double> > kappa;
     103             :   std::vector<long long int> step;
     104             :   std::vector<double> oldaa;
     105             :   std::vector<double> oldk;
     106             :   std::vector<double> olddpotdk;
     107             :   std::vector<double> oldf;
     108             :   std::vector<std::string> verse;
     109             :   std::vector<double> work;
     110             :   std::vector<double> kk,aa,f,dpotdk;
     111             :   double tot_work;
     112             :   std::vector<Value*> valueCntr;
     113             :   std::vector<Value*> valueWork;
     114             :   std::vector<Value*> valueKappa;
     115             :   Value* valueTotWork=nullptr;
     116             :   Value* valueForce2=nullptr;
     117             : public:
     118             :   explicit MovingRestraint(const ActionOptions&);
     119             :   void calculate() override;
     120             :   static void registerKeywords( Keywords& keys );
     121             : };
     122             : 
     123             : PLUMED_REGISTER_ACTION(MovingRestraint,"MOVINGRESTRAINT")
     124             : 
     125           6 : void MovingRestraint::registerKeywords( Keywords& keys ) {
     126           6 :   Bias::registerKeywords(keys);
     127           6 :   keys.add("compulsory","VERSE","B","Tells plumed whether the restraint is only acting for CV larger (U) or smaller (L) than "
     128             :            "the restraint or whether it is acting on both sides (B)");
     129           6 :   keys.add("numbered","STEP","This keyword appears multiple times as STEPx with x=0,1,2,...,n. Each value given represents "
     130             :            "the MD step at which the restraint parameters take the values KAPPAx and ATx.");
     131          12 :   keys.reset_style("STEP","compulsory");
     132           6 :   keys.add("numbered","AT","ATx is equal to the position of the restraint at time STEPx. For intermediate times this parameter "
     133             :            "is linearly interpolated. If no ATx is specified for STEPx then the values of AT are kept constant "
     134             :            "during the interval of time between STEP(x-1) and STEPx.");
     135          12 :   keys.reset_style("AT","compulsory");
     136           6 :   keys.add("numbered","KAPPA","KAPPAx is equal to the value of the force constants at time STEPx. For intermediate times this "
     137             :            "parameter is linearly interpolated.  If no KAPPAx is specified for STEPx then the values of KAPPAx "
     138             :            "are kept constant during the interval of time between STEP(x-1) and STEPx.");
     139          12 :   keys.reset_style("KAPPA","compulsory");
     140          12 :   keys.addOutputComponent("work","default","scalar","the total work performed changing this restraint");
     141          12 :   keys.addOutputComponent("force2","default","scalar","the instantaneous value of the squared force due to this bias potential");
     142          12 :   keys.addOutputComponent("_cntr","default","scalar","one or multiple instances of this quantity can be referenced elsewhere in the input file. "
     143             :                           "these quantities will named with  the arguments of the bias followed by "
     144             :                           "the character string _cntr. These quantities give the instantaneous position "
     145             :                           "of the center of the harmonic potential.");
     146          12 :   keys.addOutputComponent("_work","default","scalar","one or multiple instances of this quantity can be referenced elsewhere in the input file. "
     147             :                           "These quantities will named with the arguments of the bias followed by "
     148             :                           "the character string _work. These quantities tell the user how much work has "
     149             :                           "been done by the potential in dragging the system along the various colvar axis.");
     150          12 :   keys.addOutputComponent("_kappa","default","scalar","one or multiple instances of this quantity can be referenced elsewhere in the input file. "
     151             :                           "These quantities will named with the arguments of the bias followed by "
     152             :                           "the character string _kappa. These quantities tell the user the time dependent value of kappa.");
     153           6 : }
     154             : 
     155           4 : MovingRestraint::MovingRestraint(const ActionOptions&ao):
     156             :   PLUMED_BIAS_INIT(ao),
     157           4 :   verse(getNumberOfArguments()),
     158           4 :   kk(getNumberOfArguments()),
     159           4 :   aa(getNumberOfArguments()),
     160           4 :   f(getNumberOfArguments()),
     161           8 :   dpotdk(getNumberOfArguments()) {
     162           4 :   parseVector("VERSE",verse);
     163           4 :   std::vector<long long int> ss(1);
     164           4 :   ss[0]=-1;
     165           4 :   std::vector<double> kk( getNumberOfArguments() ), aa( getNumberOfArguments() );
     166          10 :   for(int i=0;; i++) {
     167             :     // Read in step
     168          28 :     if( !parseNumberedVector("STEP",i,ss) ) {
     169             :       break;
     170             :     }
     171          19 :     for(unsigned j=0; j<step.size(); j++) {
     172           9 :       if(ss[0]<step[j]) {
     173           0 :         error("in moving restraint step number must always increase");
     174             :       }
     175             :     }
     176          10 :     step.push_back(ss[0]);
     177             : 
     178             :     // Try to read kappa
     179          20 :     if( !parseNumberedVector("KAPPA",i,kk) ) {
     180           3 :       kk=kappa[i-1];
     181             :     }
     182          10 :     kappa.push_back(kk);
     183             : 
     184             :     // Now read AT
     185          20 :     if( !parseNumberedVector("AT",i,aa) ) {
     186           1 :       aa=at[i-1];
     187             :     }
     188          10 :     at.push_back(aa);
     189          10 :   }
     190           4 :   checkRead();
     191             : 
     192          14 :   for(unsigned i=0; i<step.size(); i++) {
     193          10 :     log.printf("  step%u %lld\n",i,step[i]);
     194          10 :     log.printf("  at");
     195          22 :     for(unsigned j=0; j<at[i].size(); j++) {
     196          12 :       log.printf(" %f",at[i][j]);
     197             :     }
     198          10 :     log.printf("\n");
     199          10 :     log.printf("  with force constant");
     200          22 :     for(unsigned j=0; j<kappa[i].size(); j++) {
     201          12 :       log.printf(" %f",kappa[i][j]);
     202             :     }
     203          10 :     log.printf("\n");
     204             :   };
     205             : 
     206           8 :   addComponent("force2");
     207           4 :   componentIsNotPeriodic("force2");
     208           4 :   valueForce2=getPntrToComponent("force2");
     209             : 
     210             :   // add the centers of the restraint as additional components that can be retrieved (useful for debug)
     211             : 
     212             :   std::string comp;
     213           9 :   for(unsigned i=0; i< getNumberOfArguments() ; i++) {
     214          10 :     comp=getPntrToArgument(i)->getName()+"_cntr"; // each spring has its own center
     215           5 :     addComponent(comp);
     216           5 :     componentIsNotPeriodic(comp);
     217           5 :     valueCntr.push_back(getPntrToComponent(comp));
     218          10 :     comp=getPntrToArgument(i)->getName()+"_work"; // each spring has its own work
     219           5 :     addComponent(comp);
     220           5 :     componentIsNotPeriodic(comp);
     221           5 :     valueWork.push_back(getPntrToComponent(comp));
     222          10 :     comp=getPntrToArgument(i)->getName()+"_kappa"; // each spring has its own kappa
     223           5 :     addComponent(comp);
     224           5 :     componentIsNotPeriodic(comp);
     225           5 :     valueKappa.push_back(getPntrToComponent(comp));
     226           5 :     work.push_back(0.); // initialize the work value
     227             :   }
     228           8 :   addComponent("work");
     229           4 :   componentIsNotPeriodic("work");
     230           4 :   valueTotWork=getPntrToComponent("work");
     231           4 :   tot_work=0.0;
     232             : 
     233           4 :   log<<"  Bibliography ";
     234           8 :   log<<cite("Grubmuller, Heymann, and Tavan, Science 271, 997 (1996)")<<"\n";
     235             : 
     236           4 : }
     237             : 
     238             : 
     239         566 : void MovingRestraint::calculate() {
     240             :   double ene=0.0;
     241             :   double totf2=0.0;
     242         566 :   unsigned narg=getNumberOfArguments();
     243         566 :   long long int now=getStep();
     244         566 :   if(now<=step[0]) {
     245           3 :     kk=kappa[0];
     246           3 :     aa=at[0];
     247           3 :     oldaa=at[0];
     248           3 :     oldk=kappa[0];
     249           3 :     olddpotdk.resize(narg);
     250           3 :     oldf.resize(narg);
     251         563 :   } else if(now>=step[step.size()-1]) {
     252          47 :     kk=kappa[step.size()-1];
     253          47 :     aa=at[step.size()-1];
     254             :   } else {
     255             :     unsigned i=0;
     256         521 :     for(i=1; i<step.size()-1; i++)
     257          10 :       if(now<step[i]) {
     258             :         break;
     259             :       }
     260         516 :     double c2=(now-step[i-1])/double(step[i]-step[i-1]);
     261         516 :     double c1=1.0-c2;
     262        1040 :     for(unsigned j=0; j<narg; j++) {
     263         524 :       kk[j]=(c1*kappa[i-1][j]+c2*kappa[i][j]);
     264             :     }
     265        1040 :     for(unsigned j=0; j<narg; j++) {
     266         524 :       const double a1=at[i-1][j];
     267         524 :       const double a2=at[i][j];
     268         524 :       aa[j]=(c1*a1+c2*(a1+difference(j,a1,a2)));
     269             :     }
     270             :   }
     271         566 :   tot_work=0.0;
     272        1142 :   for(unsigned i=0; i<narg; ++i) {
     273         576 :     const double cv=difference(i,aa[i],getArgument(i)); // this gives: getArgument(i) - aa[i]
     274         576 :     valueCntr[i]->set(aa[i]);
     275         576 :     const double k=kk[i];
     276         576 :     f[i]=-k*cv;
     277         576 :     if(verse[i]=="U" && cv<0) {
     278           0 :       continue;
     279             :     }
     280         576 :     if(verse[i]=="L" && cv>0) {
     281           0 :       continue;
     282             :     }
     283        1728 :     plumed_assert(verse[i]=="U" || verse[i]=="L" || verse[i]=="B");
     284         576 :     dpotdk[i]=0.5*cv*cv;
     285         576 :     if(oldaa.size()==aa.size() && oldf.size()==f.size()) {
     286         575 :       work[i]+=0.5*(oldf[i]+f[i])*(aa[i]-oldaa[i]) + 0.5*( dpotdk[i]+olddpotdk[i] )*(kk[i]-oldk[i]);
     287             :     }
     288         576 :     valueWork[i]->set(work[i]);
     289         576 :     valueKappa[i]->set(kk[i]);
     290         576 :     tot_work+=work[i];
     291         576 :     ene+=0.5*k*cv*cv;
     292         576 :     setOutputForce(i,f[i]);
     293         576 :     totf2+=f[i]*f[i];
     294             :   };
     295         566 :   valueTotWork->set(tot_work);
     296         566 :   oldf=f;
     297         566 :   oldaa=aa;
     298         566 :   oldk=kk;
     299         566 :   olddpotdk=dpotdk;
     300         566 :   setBias(ene);
     301         566 :   valueForce2->set(totf2);
     302         566 : }
     303             : 
     304             : }
     305             : }
     306             : 
     307             : 

Generated by: LCOV version 1.16