Skip to content
Snippets Groups Projects
ReweightWham.cpp 4.37 KiB
Newer Older
  • Learn to ignore specific revisions
  • /* +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
       Copyright (c) 2011-2016 The plumed team
       (see the PEOPLE file at the root of the distribution for a list of names)
    
       See http://www.plumed.org for more information.
    
       This file is part of plumed, version 2.
    
       plumed is free software: you can redistribute it and/or modify
       it under the terms of the GNU Lesser General Public License as published by
       the Free Software Foundation, either version 3 of the License, or
       (at your option) any later version.
    
       plumed 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 Lesser General Public License for more details.
    
       You should have received a copy of the GNU Lesser General Public License
       along with plumed.  If not, see <http://www.gnu.org/licenses/>.
    +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ */
    #include "ReweightWham.h"
    
    Gareth Tribello's avatar
    Gareth Tribello committed
    #include "core/ActionRegister.h"
    
    
    //+PLUMEDOC REWEIGHTING REWEIGHT_WHAM
    /*
    
    \par Examples
    
    */
    //+ENDPLUMEDOC
    
    namespace PLMD {
    namespace bias {
    
    PLUMED_REGISTER_ACTION(ReweightWham,"REWEIGHT_WHAM")
    
    
    void ReweightWham::registerKeywords(Keywords& keys ) {
    
      ReweightBase::registerKeywords( keys ); keys.use("ARG");
    
      keys.add("compulsory","MAXITER","1000","maximum number of iterations for WHAM algorithm");
      keys.add("compulsory","WHAMTOL","1e-10","threshold for convergence of WHAM algorithm");
    }
    
    ReweightWham::ReweightWham(const ActionOptions&ao):
    
      Action(ao),
      ReweightBase(ao),
      weightsCalculated(false)
    
      std::vector<Value*> targ, fagr;
      unsigned nbias = 0; wlists.push_back( 0 );
      for(unsigned i=1;; i++) {
        if( !parseArgumentList("ARG",i,targ ) ) break;
        log.printf("  bias number %d involves :");
        for(unsigned j=0; j<targ.size(); ++j) {
          log.printf("%s ",targ[j]->getName().c_str() );
          fagr.push_back( targ[j] );
        }
        log.printf("\n"); targ.resize(0);
        wlists.push_back( fagr.size() );
        nbias++;
      }
      plumed_assert( wlists.size()==(nbias+1) ); requestArguments( fagr );
      parse("MAXITER",maxiter); parse("WHAMTOL",thresh);
    
    double ReweightWham::getLogWeight() {
      if( getStep()==0 ) return 1.0;  // This is here as first step is ignored in all analyses
      weightsCalculated=false;
      for(unsigned i=0; i<wlists.size()-1; ++i) {
        double total_bias=0;
        for(unsigned j=wlists[i]; j<wlists[i+1]; ++j) total_bias+=getArgument(j);
        stored_biases.push_back( total_bias );
      }
      return 1.0;
    
    void ReweightWham::clearData() {
      stored_biases.resize(0);
    
    void ReweightWham::calculateWeights( const unsigned& nframes ) {
      if( stored_biases.size()!=(wlists.size()-1)*nframes ) error("wrong number of weights stored");
      // Get the minimum value of the bias
      double minv = *min_element(std::begin(stored_biases), std::end(stored_biases));
      // Resize final weights array
      plumed_assert( stored_biases.size()%(wlists.size()-1)==0 );
      final_weights.resize( stored_biases.size() / (wlists.size()-1), 1.0 );
      // Offset and exponential of the bias
      std::vector<double> expv( stored_biases.size() );
      for(unsigned i=0; i<expv.size(); ++i) expv[i] = exp( (-stored_biases[i]+minv) / simtemp );
      // Initialize Z
      std::vector<double> Z( wlists.size()-1, 1.0 ), oldZ( wlists.size()-1 );
      // Now the iterative loop to calculate the WHAM weights
      for(unsigned iter=0; iter<maxiter; ++iter) {
        // Store Z
        for(unsigned j=0; j<Z.size(); ++j) oldZ[j]=Z[j];
        // Recompute weights
        double norm=0;
        for(unsigned j=0; j<final_weights.size(); ++j) {
          double ew=0;
          for(unsigned k=0; k<Z.size(); ++k) ew += expv[j*Z.size()+k]  / Z[k];
          final_weights[j] = 1.0 / ew; norm += final_weights[j];
        }
        // Normalize weights
        for(unsigned j=0; j<final_weights.size(); ++j) final_weights[j] /= norm;
        // Recompute Z
        for(unsigned j=0; j<Z.size(); ++j) Z[j] = 0.0;
        for(unsigned j=0; j<final_weights.size(); ++j) {
          for(unsigned k=0; k<Z.size(); ++k) Z[k] += final_weights[j]*expv[j*Z.size()+k];
        }
        // Normalize Z and compute change in Z
        double change=0; norm=0; for(unsigned k=0; k<Z.size(); ++k) norm+=Z[k];
        for(unsigned k=0; k<Z.size(); ++k) {
          Z[k] /= norm; double d = std::log( Z[k] / oldZ[k] ); change += d*d;
        }
        if( change<thresh ) { weightsCalculated=true; return; }
      }
      error("Too many iterations in WHAM" );
    }