LCOV - code coverage report
Current view: top level - tools - RMSD.h (source / functions) Hit Total Coverage
Test: plumed test coverage Lines: 51 63 81.0 %
Date: 2025-03-25 09:33:27 Functions: 7 9 77.8 %

          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             : #ifndef __PLUMED_tools_RMSD_h
      23             : #define __PLUMED_tools_RMSD_h
      24             : 
      25             : #include "Vector.h"
      26             : #include "Matrix.h"
      27             : #include "Tensor.h"
      28             : #include <vector>
      29             : #include <string>
      30             : #include <array>
      31             : 
      32             : namespace PLMD {
      33             : 
      34             : class Log;
      35             : class PDB;
      36             : 
      37             : /** \ingroup TOOLBOX
      38             : A class that implements RMSD calculations
      39             : This is a class that implements the various infrastructure to calculate the
      40             : RMSD or MSD respect a given frame. It can be done through an optimal alignment scheme
      41             : as Kearsley or, more simply, by resetting the center of mass.
      42             : This is the class that decides this. A very simple use is
      43             : \verbatim
      44             : #include "tools/PDB.h"
      45             : #include "tools/RMSD.h"
      46             : #include "tools/Vector.h"
      47             : using namespace PLMD;
      48             : RMSD rmsd;
      49             : PDB pdb;
      50             : // get the pdb (see PDB documentation)
      51             : pdb.read("file.pdb",true,1.0);
      52             : string type;
      53             : type.assign("OPTIMAL");
      54             : // set the reference and the type
      55             : rmsd.set(pdb,type);
      56             : // this calculates the rmsd and the derivatives
      57             : vector<Vector> derivs;
      58             : double val;
      59             : val=rmsd.calculate(getPositions(),derivs,true);
      60             : \endverbatim
      61             : 
      62             : **/
      63             : 
      64             : class RMSD {
      65             :   enum AlignmentMethod {SIMPLE, OPTIMAL, OPTIMAL_FAST};
      66             :   AlignmentMethod alignmentMethod;
      67             : // Reference coordinates
      68             :   std::vector<Vector> reference;
      69             : // Weights for alignment
      70             :   std::vector<double> align;
      71             : // Weights for deviation
      72             :   std::vector<double> displace;
      73             : // Center for reference and flag for its calculation
      74             :   Vector reference_center;
      75             :   bool reference_center_is_calculated;
      76             :   bool reference_center_is_removed;
      77             : // Center for running position (not used in principle but here to reflect reference/positio symmetry
      78             :   Vector positions_center;
      79             :   bool positions_center_is_calculated;
      80             :   bool positions_center_is_removed;
      81             : // calculates the center from the position provided
      82       13180 :   Vector calculateCenter(const std::vector<Vector> &p,const std::vector<double> &w) {
      83       13180 :     plumed_massert(p.size()==w.size(),"mismatch in dimension of position/align arrays while calculating the center");
      84             :     unsigned n;
      85       13180 :     n=p.size();
      86       13180 :     Vector c;
      87       13180 :     c.zero();
      88      209798 :     for(unsigned i=0; i<n; i++) {
      89      196618 :       c+=p[i]*w[i];
      90             :     }
      91       13180 :     return c;
      92             :   };
      93             : // removes the center for the position provided
      94       26355 :   void removeCenter(std::vector<Vector> &p, const Vector &c) {
      95             :     unsigned n;
      96       26355 :     n=p.size();
      97      419566 :     for(unsigned i=0; i<n; i++) {
      98      393211 :       p[i]-=c;
      99             :     }
     100       26355 :   };
     101             : // add center
     102       13180 :   void addCenter(std::vector<Vector> &p, const Vector &c) {
     103       13180 :     Vector cc=c*-1.;
     104       13180 :     removeCenter(p,cc);
     105       13180 :   };
     106             : 
     107             : public:
     108             : /// Constructor
     109             :   RMSD();
     110             : /// clear the structure
     111             :   void clear();
     112             : /// set reference, align and displace from input pdb structure: evtl remove com from the initial structure and normalize the input weights from the pdb
     113             :   void set(const PDB&,const std::string & mytype, bool remove_center=true, bool normalize_weights=true);
     114             : /// set align displace reference and type from input vectors
     115             :   void set(const std::vector<double> & align, const std::vector<double> & displace, const std::vector<Vector> & reference,const std::string & mytype, bool remove_center=true, bool normalize_weights=true );
     116             : /// set the type of alignment we are doing
     117             :   void setType(const std::string & mytype);
     118             : /// set reference coordinates, remove the com by using uniform weights
     119             :   void setReference(const std::vector<Vector> & reference);
     120             :   const std::vector<Vector>& getReference() const ;
     121             : /// set weights and remove the center from reference with normalized weights. If the com has been removed, it resets to the new value
     122             :   void setAlign(const std::vector<double> & align, bool normalize_weights=true, bool remove_center=true);
     123             :   std::vector<double> getAlign();
     124             : /// set align
     125             :   void setDisplace(const std::vector<double> & displace, bool normalize_weights=true);
     126             :   std::vector<double> getDisplace();
     127             : ///
     128             :   std::string getMethod();
     129             : /// workhorses
     130             :   double simpleAlignment(const  std::vector<double>  & align,
     131             :                          const  std::vector<double>  & displace,
     132             :                          const std::vector<Vector> & positions,
     133             :                          const std::vector<Vector> & reference,
     134             :                          std::vector<Vector>  & derivatives,
     135             :                          std::vector<Vector>  & displacement,
     136             :                          bool squared=false)const;
     137             :   template <bool safe,bool alEqDis>
     138             :   double optimalAlignment(const  std::vector<double>  & align,
     139             :                           const  std::vector<double>  & displace,
     140             :                           const std::vector<Vector> & positions,
     141             :                           const std::vector<Vector> & reference,
     142             :                           std::vector<Vector>  & DDistDPos, bool squared=false)const;
     143             : 
     144             :   template <bool safe, bool alEqDis>
     145             :   double optimalAlignmentWithCloseStructure(const  std::vector<double>  & align,
     146             :       const  std::vector<double>  & displace,
     147             :       const std::vector<Vector> & positions,
     148             :       const std::vector<Vector> & reference,
     149             :       std::vector<Vector>  & derivatives,
     150             :       const Tensor & rotationPosClose,
     151             :       const Tensor & rotationRefClose,
     152             :       std::array<std::array<Tensor,3>,3> & drotationPosCloseDrr01,
     153             :       bool squared=false)const;
     154             : 
     155             :   template <bool safe, bool alEqDis>
     156             :   double optimalAlignment_Rot_DRotDRr01(const  std::vector<double>  & align,
     157             :                                         const  std::vector<double>  & displace,
     158             :                                         const std::vector<Vector> & positions,
     159             :                                         const std::vector<Vector> & reference,
     160             :                                         Tensor & Rotation,
     161             :                                         std::array<std::array<Tensor,3>,3> & drotationPosCloseDrr01,
     162             :                                         bool squared=false)const;
     163             : 
     164             :   template <bool safe, bool alEqDis>
     165             :   double optimalAlignment_Rot(const  std::vector<double>  & align,
     166             :                               const  std::vector<double>  & displace,
     167             :                               const std::vector<Vector> & positions,
     168             :                               const std::vector<Vector> & reference,
     169             :                               std::vector<Vector>  & derivatives,
     170             :                               Tensor & Rotation,
     171             :                               bool squared=false)const;
     172             : 
     173             :   template <bool safe,bool alEqDis>
     174             :   double optimalAlignment_DDistDRef(const  std::vector<double>  & align,
     175             :                                     const  std::vector<double>  & displace,
     176             :                                     const std::vector<Vector> & positions,
     177             :                                     const std::vector<Vector> & reference,
     178             :                                     std::vector<Vector>  & DDistDPos,
     179             :                                     std::vector<Vector> &  DDistDRef,
     180             :                                     bool squared=false) const;
     181             : 
     182             :   template <bool safe,bool alEqDis>
     183             :   double optimalAlignment_SOMA(const  std::vector<double>  & align,
     184             :                                const  std::vector<double>  & displace,
     185             :                                const std::vector<Vector> & positions,
     186             :                                const std::vector<Vector> & reference,
     187             :                                std::vector<Vector>  & DDistDPos,
     188             :                                std::vector<Vector> &  DDistDRef,
     189             :                                bool squared=false) const;
     190             : 
     191             :   template <bool safe,bool alEqDis>
     192             :   double optimalAlignment_DDistDRef_Rot_DRotDPos(const  std::vector<double>  & align,
     193             :       const  std::vector<double>  & displace,
     194             :       const std::vector<Vector> & positions,
     195             :       const std::vector<Vector> & reference,
     196             :       std::vector<Vector>  & DDistDPos,
     197             :       std::vector<Vector> &  DDistDRef,
     198             :       Tensor & Rotation,
     199             :       Matrix<std::vector<Vector> > &DRotDPos,
     200             :       bool squared=false) const;
     201             : 
     202             :   template <bool safe,bool alEqDis>
     203             :   double optimalAlignment_DDistDRef_Rot_DRotDPos_DRotDRef(const  std::vector<double>  & align,
     204             :       const  std::vector<double>  & displace,
     205             :       const std::vector<Vector> & positions,
     206             :       const std::vector<Vector> & reference,
     207             :       std::vector<Vector>  & DDistDPos,
     208             :       std::vector<Vector> &  DDistDRef,
     209             :       Tensor & Rotation,
     210             :       Matrix<std::vector<Vector> > &DRotDPos,
     211             :       Matrix<std::vector<Vector> > &DRotDRef,
     212             :       bool squared=false) const;
     213             : 
     214             :   template <bool safe,bool alEqDis>
     215             :   double optimalAlignment_PCA(const  std::vector<double>  & align,
     216             :                               const  std::vector<double>  & displace,
     217             :                               const std::vector<Vector> & positions,
     218             :                               const std::vector<Vector> & reference,
     219             :                               std::vector<Vector> & alignedpositions,
     220             :                               std::vector<Vector> & centeredpositions,
     221             :                               std::vector<Vector> & centeredreference,
     222             :                               Tensor & Rotation,
     223             :                               std::vector<Vector> & DDistDPos,
     224             :                               Matrix<std::vector<Vector> > & DRotDPos,
     225             :                               bool squared=false) const ;
     226             : 
     227             :   template <bool safe,bool alEqDis>
     228             :   double optimalAlignment_Fit(const  std::vector<double>  & align,
     229             :                               const  std::vector<double>  & displace,
     230             :                               const std::vector<Vector> & positions,
     231             :                               const std::vector<Vector> & reference,
     232             :                               Tensor & Rotation,
     233             :                               Matrix<std::vector<Vector> > & DRotDPos,
     234             :                               std::vector<Vector> & centeredpositions,
     235             :                               Vector & center_positions,
     236             :                               bool squared=false);
     237             : 
     238             : 
     239             : /// Compute rmsd: note that this is an intermediate layer which is kept in order to evtl expand with more alignment types/user options to be called while keeping the workhorses separated
     240             :   double calculate(const std::vector<Vector> & positions,std::vector<Vector> &derivatives, bool squared=false)const;
     241             : /// Other convenience methods:
     242             : /// calculate the derivative of distance respect to position(DDistDPos) and reference (DDistDPos)
     243             :   double calc_DDistDRef( const std::vector<Vector>& positions, std::vector<Vector> &DDistDPos, std::vector<Vector>& DDistDRef, const bool squared=false   );
     244             : /// calculate the derivative for SOMA (i.e. derivative respect to reference frame without the matrix derivative)
     245             :   double calc_SOMA( const std::vector<Vector>& positions, std::vector<Vector> &DDistDPos, std::vector<Vector>& DDistDRef, const bool squared=false   );
     246             : ///
     247             :   double calc_DDistDRef_Rot_DRotDPos( const std::vector<Vector>& positions, std::vector<Vector> &DDistDPos, std::vector<Vector>& DDistDRef, Tensor & Rotation,Matrix<std::vector<Vector> > &DRotDPos, const bool squared=false   );
     248             :   double calc_DDistDRef_Rot_DRotDPos_DRotDRef( const std::vector<Vector>& positions, std::vector<Vector> &DDistDPos, std::vector<Vector>& DDistDRef, Tensor & Rotation,Matrix<std::vector<Vector> > &DRotDPos,Matrix<std::vector<Vector> > &DRotDRef, const bool squared=false   );
     249             : /// convenience method to retrieve all the bits and pieces for PCA
     250             :   double calc_PCAelements( const std::vector<Vector>& pos, std::vector<Vector> &DDistDPos, Tensor & Rotation, Matrix<std::vector<Vector> > & DRotDPos,std::vector<Vector>  & alignedpositions, std::vector<Vector> & centeredpositions, std::vector<Vector> &centeredreference, const bool& squared=false) const ;
     251             : /// convenience method to retrieve all the bits and pieces needed by FitToTemplate
     252             :   double calc_FitElements( const std::vector<Vector>& pos, Tensor & Rotation, Matrix<std::vector<Vector> > & DRotDPos,std::vector<Vector> & centeredpositions,Vector & center_positions, const bool& squared=false );
     253             : ///calculate rotation matrix, derivative of rotation matrix w.r.t. positions, derivative of rotation matrix w.r.t. rr01
     254             :   double calc_Rot_DRotDRr01( const std::vector<Vector>& positions, Tensor & Rotation, std::array<std::array<Tensor,3>,3> & DRotDRr01, const bool squared=false   );
     255             : ///calculate rotation matrix, derivative of rotation matrix w.r.t. positions
     256             :   double calc_Rot( const std::vector<Vector>& positions, std::vector<Vector> &DDistDPos, Tensor & Rotation, const bool squared=false   );
     257             : ///calculate with close structure, i.e. approximate the RMSD without expensive computation of rotation matrix by reusing saved rotation matrices from previous iterations
     258             :   double calculateWithCloseStructure( const std::vector<Vector>& positions, std::vector<Vector> &DDistDPos, const Tensor & rotationPosClose, const Tensor & rotationRefClose, std::array<std::array<Tensor,3>,3> & drotationPosCloseDrr01, const bool squared=false   );
     259             : /// static convenience method to get the matrix i,a from drotdpos (which is a bit tricky)
     260         900 :   static  Tensor getMatrixFromDRot(const Matrix< std::vector<Vector> > &drotdpos, const unsigned &i, const unsigned &a) {
     261         900 :     Tensor t;
     262         900 :     t[0][0]=drotdpos(0,0)[i][a];
     263         900 :     t[0][1]=drotdpos(0,1)[i][a];
     264         900 :     t[0][2]=drotdpos(0,2)[i][a];
     265         900 :     t[1][0]=drotdpos(1,0)[i][a];
     266         900 :     t[1][1]=drotdpos(1,1)[i][a];
     267         900 :     t[1][2]=drotdpos(1,2)[i][a];
     268         900 :     t[2][0]=drotdpos(2,0)[i][a];
     269         900 :     t[2][1]=drotdpos(2,1)[i][a];
     270         900 :     t[2][2]=drotdpos(2,2)[i][a];
     271         900 :     return t;
     272             :   };
     273             : };
     274             : 
     275             : /// this is a class which is needed to share information across the various non-threadsafe routines
     276             : /// so that the public function of rmsd are threadsafe while the inner core can safely share information
     277      103848 : class RMSDCoreData {
     278             : private:
     279             :   bool alEqDis;
     280             :   bool distanceIsMSD; // default is RMSD but can deliver the MSD
     281             :   bool hasDistance;  // distance is already calculated
     282             :   bool isInitialized;
     283             :   bool safe;
     284             : 
     285             :   // this need to be copied and they are small, should not affect the performance
     286             :   Vector creference;
     287             :   bool creference_is_calculated;
     288             :   bool creference_is_removed;
     289             :   Vector cpositions;
     290             :   bool cpositions_is_calculated;
     291             :   bool cpositions_is_removed;
     292             :   bool retrieve_only_rotation;
     293             : 
     294             :   // use reference assignment to speed up instead of copying
     295             :   const std::vector<Vector> &positions;
     296             :   const std::vector<Vector> &reference;
     297             :   const std::vector<double> &align;
     298             :   const std::vector<double> &displace;
     299             : 
     300             :   // the needed stuff for distance and more (one could use eigenvecs components and eigenvals for some reason)
     301             :   double dist;
     302             :   Vector4d eigenvals;
     303             :   Tensor4d eigenvecs;
     304             :   double rr00; //  sum of positions squared (needed for dist calc)
     305             :   double rr11; //  sum of reference squared (needed for dist calc)
     306             :   Tensor rotation; // rotation derived from the eigenvector having the smallest eigenvalue
     307             :   std::array<std::array<Tensor,3>,3> drotation_drr01; // derivative of the rotation only available when align!=displace
     308             :   Tensor ddist_drr01;
     309             :   Tensor ddist_drotation;
     310             :   std::vector<Vector> d; // difference of components
     311             : public:
     312             :   /// the constructor (note: only references are passed, therefore is rather fast)
     313             :   /// note: this aligns the reference onto the positions
     314             :   ///
     315             :   /// this method assumes that the centers are already calculated and subtracted
     316             :   RMSDCoreData(const std::vector<double> &a,const std::vector<double> &d,const std::vector<Vector> &p, const std::vector<Vector> &r, Vector &cp, Vector &cr ):
     317             :     alEqDis(false),distanceIsMSD(false),hasDistance(false),isInitialized(false),safe(false),
     318             :     creference(cr),creference_is_calculated(true),creference_is_removed(true),
     319             :     cpositions(cp),cpositions_is_calculated(true),cpositions_is_removed(true),retrieve_only_rotation(false),positions(p),reference(r),align(a),displace(d),dist(0.0),rr00(0.0),rr11(0.0) {};
     320             : 
     321             :   // this constructor does not assume that the positions and reference have the center subtracted
     322      103848 :   RMSDCoreData(const std::vector<double> &a,const std::vector<double> &d,const std::vector<Vector> &p, const std::vector<Vector> &r):
     323      103848 :     alEqDis(false),distanceIsMSD(false),hasDistance(false),isInitialized(false),safe(false),
     324      103848 :     creference_is_calculated(false),creference_is_removed(false),
     325      103848 :     cpositions_is_calculated(false),cpositions_is_removed(false),retrieve_only_rotation(false),positions(p),reference(r),align(a),displace(d),dist(0.0),rr00(0.0),rr11(0.0) {
     326      103848 :     cpositions.zero();
     327      103848 :     creference.zero();
     328      103848 :   };
     329             : 
     330             :   // set the center on the fly without subtracting
     331      103848 :   void calcPositionsCenter() {
     332      103848 :     plumed_massert(!cpositions_is_calculated,"the center was already calculated");
     333      103848 :     cpositions.zero();
     334     1939353 :     for(unsigned i=0; i<positions.size(); i++) {
     335     1835505 :       cpositions+=positions[i]*align[i];
     336             :     }
     337      103848 :     cpositions_is_calculated=true;
     338      103848 :   }
     339           0 :   void calcReferenceCenter() {
     340           0 :     plumed_massert(!creference_is_calculated,"the center was already calculated");
     341           0 :     creference.zero();
     342           0 :     for(unsigned i=0; i<reference.size(); i++) {
     343           0 :       creference+=reference[i]*align[i];
     344             :     }
     345           0 :     creference_is_calculated=true;
     346           0 :   };
     347             :   // assume the center is given externally
     348             :   // cppcheck-suppress passedByValue
     349           0 :   void setPositionsCenter(Vector v) {
     350           0 :     plumed_massert(!cpositions_is_calculated,"You are setting the center two times!");
     351           0 :     cpositions=v;
     352           0 :     cpositions_is_calculated=true;
     353           0 :   };
     354             :   // cppcheck-suppress passedByValue
     355      103848 :   void setReferenceCenter(Vector v) {
     356      103848 :     plumed_massert(!creference_is_calculated,"You are setting the center two times!");
     357      103848 :     creference=v;
     358      103848 :     creference_is_calculated=true;
     359      103848 :   };
     360             :   // the center is already removed
     361             :   void setPositionsCenterIsRemoved(bool t) {
     362      103848 :     cpositions_is_removed=t;
     363             :   };
     364             :   void setReferenceCenterIsRemoved(bool t) {
     365      103848 :     creference_is_removed=t;
     366             :   };
     367             :   bool getPositionsCenterIsRemoved() {
     368             :     return cpositions_is_removed;
     369             :   };
     370             :   bool getReferenceCenterIsRemoved() {
     371             :     return creference_is_removed;
     372             :   };
     373             :   //  does the core calc : first thing to call after the constructor:
     374             :   // only_rotation=true does not retrieve the derivatives, just retrieve the optimal rotation (the same calc cannot be exploit further)
     375             :   void doCoreCalc(bool safe,bool alEqDis, bool only_rotation=false);
     376             :   // do calculation with close structure data structures
     377             :   void doCoreCalcWithCloseStructure(bool safe,bool alEqDis, const Tensor & rotationPosClose, const Tensor & rotationRefClose, std::array<std::array<Tensor,3>,3> & drotationPosCloseDrr01);
     378             :   // retrieve the distance if required after doCoreCalc
     379             :   double getDistance(bool squared);
     380             :   // retrieve the derivative of the distance respect to the position
     381             :   std::vector<Vector> getDDistanceDPositions();
     382             :   // retrieve the derivative of the distance respect to the reference
     383             :   std::vector<Vector> getDDistanceDReference();
     384             :   // specific version for SOMA calculation (i.e. does not need derivative respect to rotation matrix)
     385             :   std::vector<Vector> getDDistanceDReferenceSOMA();
     386             :   // get aligned reference onto position
     387             :   std::vector<Vector> getAlignedReferenceToPositions();
     388             :   // get aligned position onto reference
     389             :   std::vector<Vector> getAlignedPositionsToReference();
     390             :   // get centered positions
     391             :   std::vector<Vector> getCenteredPositions();
     392             :   // get centered reference
     393             :   std::vector<Vector> getCenteredReference();
     394             :   // get center of positions
     395             :   Vector getPositionsCenter();
     396             :   // get center of reference
     397             :   Vector getReferenceCenter();
     398             :   // get rotation matrix (reference ->positions)
     399             :   Tensor getRotationMatrixReferenceToPositions();
     400             :   // get rotation matrix (positions -> reference)
     401             :   Tensor getRotationMatrixPositionsToReference();
     402             :   // get the derivative of the rotation matrix respect to positions
     403             :   // note that the this transformation overlap the  reference onto position
     404             :   // if inverseTransform=true then aligns the positions onto reference
     405             :   Matrix<std::vector<Vector> > getDRotationDPositions( bool inverseTransform=false );
     406             :   // get the derivative of the rotation matrix respect to reference
     407             :   // note that the this transformation overlap the  reference onto position
     408             :   // if inverseTransform=true then aligns the positions onto reference
     409             :   Matrix<std::vector<Vector> >  getDRotationDReference(bool inverseTransform=false );
     410             :   const std::array<std::array<Tensor,3>,3> & getDRotationDRr01() const;
     411             : };
     412             : 
     413             : }
     414             : 
     415             : #endif
     416             : 

Generated by: LCOV version 1.16