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

          Line data    Source code
       1             : /* +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
       2             :    Copyright (c) 2014-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_MultiValue_h
      23             : #define __PLUMED_tools_MultiValue_h
      24             : 
      25             : #include "Exception.h"
      26             : #include "Vector.h"
      27             : #include "Tensor.h"
      28             : #include <vector>
      29             : #include <cstddef>
      30             : 
      31             : namespace PLMD {
      32             : 
      33             : class MultiValue {
      34             :   friend class ActionWithVector;
      35             : private:
      36             : /// The index of the task we are currently performing
      37             :   std::size_t task_index, task2_index;
      38             : /// Values of quantities
      39             :   std::vector<double> values;
      40             : /// Number of derivatives per value
      41             :   unsigned nderivatives;
      42             : /// Derivatives
      43             :   std::vector<double> derivatives;
      44             : /// Matrix asserting which values have derivatives
      45             :   std::vector<bool> hasderiv;
      46             : /// Tempory value
      47             :   double tmpval;
      48             : /// Lists of active variables
      49             :   std::vector<unsigned> nactive, active_list;
      50             : /// Tempory vector of derivatives (used for calculating quotients
      51             :   std::vector<double> tmpder;
      52             : /// Logical to check if any derivatives were set
      53             :   bool atLeastOneSet;
      54             : /// Are we in this for a call on vectors
      55             :   bool vector_call;
      56             :   unsigned nindices, nsplit;
      57             : /// This allows us to store matrix elements
      58             :   unsigned nmatrix_cols;
      59             :   std::vector<double> matrix_row_stash;
      60             :   std::vector<double> matrix_force_stash;
      61             :   std::vector<unsigned> matrix_bookeeping;
      62             : /// These are used to store the indices that have derivatives wrt to at least one
      63             : /// of the elements in a matrix
      64             :   std::vector<unsigned> matrix_row_nderivatives;
      65             :   std::vector<std::vector<unsigned> > matrix_row_derivative_indices;
      66             : /// This is a fudge to save on vector resizing in MultiColvar
      67             :   std::vector<unsigned> indices;
      68             :   std::vector<Vector> tmp_atoms;
      69             :   std::vector<std::vector<Vector> > tmp_atom_der;
      70             :   std::vector<Tensor> tmp_atom_virial;
      71             :   std::vector<std::vector<double> > tmp_vectors;
      72             : public:
      73       26332 :   MultiValue( const std::size_t& nvals, const std::size_t& nder, const std::size_t& nmat=0, const std::size_t& maxcol=0, const std::size_t& nbook=0 );
      74          36 :   void resize( const std::size_t& nvals, const std::size_t& nder, const std::size_t& nmat=0, const std::size_t& maxcol=0, const std::size_t& nbook=0 );
      75             : /// Set the task index prior to the loop
      76             :   void setTaskIndex( const std::size_t& tindex );
      77             : ///
      78             :   std::size_t getTaskIndex() const ;
      79             : ///
      80             :   void setSecondTaskIndex( const std::size_t& tindex );
      81             : /// Get the task index
      82             :   std::size_t getSecondTaskIndex() const ;
      83             : ///
      84             :   void setSplitIndex( const std::size_t& nat );
      85             :   std::size_t getSplitIndex() const ;
      86             : ///
      87             :   void setNumberOfIndices( const std::size_t& nat );
      88             :   std::size_t getNumberOfIndices() const ;
      89             : ///
      90             :   std::vector<unsigned>& getIndices();
      91             :   std::vector<Vector>& getAtomVector();
      92             : /// Get the number of values in the stash
      93             :   unsigned getNumberOfValues() const ;
      94             : /// Get the number of derivatives in the stash
      95             :   unsigned getNumberOfDerivatives() const ;
      96             : /// Get references to some memory. These vectors allow us to
      97             : /// avoid doing lots of resizing of vectors in MultiColvarTemplate
      98             :   std::vector<Vector>& getFirstAtomVector();
      99             :   std::vector<std::vector<Vector> >& getFirstAtomDerivativeVector();
     100             :   const std::vector<std::vector<Vector> >& getConstFirstAtomDerivativeVector() const ;
     101             :   std::vector<Tensor>& getFirstAtomVirialVector();
     102             :   void resizeTemporyVector(const unsigned& n );
     103             :   std::vector<double>& getTemporyVector(const unsigned& ind );
     104             : ///
     105             :   bool inVectorCall() const ;
     106             : /// Set value numbered
     107             :   void setValue( const std::size_t&,  const double& );
     108             : /// Add value numbered
     109             :   void addValue( const std::size_t&,  const double& );
     110             : /// Add derivative
     111             :   void addDerivative( const std::size_t&, const std::size_t&, const double& );
     112             : /// Set the value of the derivative
     113             :   void setDerivative( const std::size_t& ival, const std::size_t& jder, const double& der);
     114             : /// Return the ith value
     115             :   double get( const std::size_t& ) const ;
     116             : /// Return a derivative value
     117             :   double getDerivative( const std::size_t&, const std::size_t& ) const ;
     118             : /// Clear all values
     119             :   void clearAll();
     120             : /// Clear the derivatives
     121             :   void clearDerivatives( const unsigned& );
     122             : /// Clear a value
     123             :   void clear( const unsigned& );
     124             : /// Functions for accessing active list
     125             :   bool updateComplete();
     126             :   void emptyActiveMembers();
     127             :   void putIndexInActiveArray( const unsigned & );
     128             :   void updateIndex( const unsigned& );
     129             : ///
     130             :   void updateIndex( const std::size_t&, const std::size_t& );
     131             : ///
     132             :   unsigned getActiveIndex( const std::size_t&, const std::size_t& ) const ;
     133             : ///
     134             :   void clearActiveMembers( const std::size_t& ival );
     135             : ///
     136             :   unsigned getNumberActive( const std::size_t& ival ) const ;
     137             : ///
     138             :   unsigned getActiveIndex( const unsigned& ) const ;
     139             : /// Get the matrix bookeeping array
     140             :   const std::vector<unsigned> & getMatrixBookeeping() const ;
     141             :   void stashMatrixElement( const unsigned& nmat, const unsigned& rowstart, const unsigned& jcol, const double& val );
     142             :   double getStashedMatrixElement( const unsigned& nmat, const unsigned& jcol ) const ;
     143             : /// Get the bookeeping stuff for the derivatives wrt to rows of matrix
     144             :   void setNumberOfMatrixRowDerivatives( const unsigned& nmat, const unsigned& nind );
     145             :   unsigned getNumberOfMatrixRowDerivatives( const unsigned& nmat ) const ;
     146             :   std::vector<unsigned>& getMatrixRowDerivativeIndices( const unsigned& nmat );
     147             : /// Stash the forces on the matrix
     148             :   void addMatrixForce( const unsigned& imat, const unsigned& jind, const double& f );
     149             :   double getStashedMatrixForce( const unsigned& imat, const unsigned& jind ) const ;
     150             : };
     151             : 
     152             : inline
     153             : unsigned MultiValue::getNumberOfValues() const {
     154             :   return values.size();
     155             : }
     156             : 
     157             : inline
     158             : unsigned MultiValue::getNumberOfDerivatives() const {
     159      203287 :   return nderivatives; //derivatives.ncols();
     160             : }
     161             : 
     162             : inline
     163             : double MultiValue::get( const std::size_t& ival ) const {
     164             :   plumed_dbg_assert( ival<=values.size() );
     165   291586968 :   return values[ival];
     166             : }
     167             : 
     168             : inline
     169             : void MultiValue::setValue( const std::size_t& ival,  const double& val) {
     170             :   plumed_dbg_assert( ival<=values.size() );
     171    25041332 :   values[ival]=val;
     172             : }
     173             : 
     174             : inline
     175             : void MultiValue::addValue( const std::size_t& ival,  const double& val) {
     176             :   plumed_dbg_assert( ival<=values.size() );
     177   187589266 :   values[ival]+=val;
     178             : }
     179             : 
     180             : inline
     181  1072182154 : void MultiValue::addDerivative( const std::size_t& ival, const std::size_t& jder, const double& der) {
     182             :   plumed_dbg_assert( ival<=values.size() && jder<nderivatives );
     183  1072182154 :   atLeastOneSet=true;
     184  1072182154 :   hasderiv[nderivatives*ival+jder]=true;
     185  1072182154 :   derivatives[nderivatives*ival+jder] += der;
     186  1072182154 : }
     187             : 
     188             : inline
     189             : void MultiValue::setDerivative( const std::size_t& ival, const std::size_t& jder, const double& der) {
     190             :   plumed_dbg_assert( ival<=values.size() && jder<nderivatives );
     191             :   atLeastOneSet=true;
     192             :   hasderiv[nderivatives*ival+jder]=true;
     193             :   derivatives[nderivatives*ival+jder]=der;
     194             : }
     195             : 
     196             : 
     197             : inline
     198             : double MultiValue::getDerivative( const std::size_t& ival, const std::size_t& jder ) const {
     199             :   plumed_dbg_assert( ival<values.size() && jder<nderivatives );
     200   894802724 :   return derivatives[nderivatives*ival+jder];
     201             : }
     202             : 
     203             : inline
     204  1957497618 : void MultiValue::updateIndex( const std::size_t& ival, const std::size_t& jder ) {
     205             :   plumed_dbg_assert( ival<values.size() && jder<nderivatives );
     206             : #ifdef DNDEBUG
     207             :   for(unsigned i=0; i<nactive[ival]; ++i) {
     208             :     plumed_dbg_assert( active_list[nderivatives*ival+nactive[ival]]!=jder );
     209             :   }
     210             : #endif
     211  1957497618 :   if( hasderiv[nderivatives*ival+jder] ) {
     212             :     plumed_dbg_assert( nactive[ival]<nderivatives);
     213   563342204 :     active_list[nderivatives*ival+nactive[ival]]=jder;
     214   563342204 :     nactive[ival]++;
     215             :   }
     216  1957497618 : }
     217             : 
     218             : inline
     219             : unsigned MultiValue::getNumberActive( const std::size_t& ival ) const {
     220             :   plumed_dbg_assert( ival<nactive.size() );
     221  1309644336 :   return nactive[ival];
     222             : }
     223             : 
     224             : inline
     225             : unsigned MultiValue::getActiveIndex( const std::size_t& ival, const std::size_t& ind ) const {
     226             :   plumed_dbg_assert( ind<nactive[ival] );
     227  1110818280 :   return active_list[nderivatives*ival+ind];
     228             : }
     229             : 
     230             : inline
     231             : void MultiValue::setSplitIndex( const std::size_t& nat ) {
     232      663993 :   nsplit = nat;
     233             : }
     234             : 
     235             : inline
     236             : std::size_t MultiValue::getSplitIndex() const {
     237   246931070 :   return nsplit;
     238             : }
     239             : 
     240             : inline
     241             : void MultiValue::setNumberOfIndices( const std::size_t& nat ) {
     242      405876 :   nindices = nat;
     243             : }
     244             : 
     245             : inline
     246             : std::size_t MultiValue::getNumberOfIndices() const {
     247    28540953 :   return nindices;
     248             : }
     249             : 
     250             : 
     251             : inline
     252             : bool MultiValue::inVectorCall() const {
     253             :   return (matrix_row_nderivatives.size()>0 && vector_call);
     254             : }
     255             : 
     256             : inline
     257             : void MultiValue::clearActiveMembers( const std::size_t& ival ) {
     258             :   nactive[ival]=0;
     259             : }
     260             : 
     261             : inline
     262             : void MultiValue::setTaskIndex( const std::size_t& tindex ) {
     263    98409836 :   task_index = tindex;
     264             : }
     265             : 
     266             : inline
     267             : std::size_t MultiValue::getTaskIndex() const {
     268    17449808 :   return task_index;
     269             : }
     270             : 
     271             : inline
     272             : void MultiValue::setSecondTaskIndex( const std::size_t& tindex ) {
     273    88130811 :   task2_index = tindex;
     274             : }
     275             : 
     276             : inline
     277             : std::size_t MultiValue::getSecondTaskIndex() const {
     278     6403164 :   return task2_index;
     279             : }
     280             : 
     281             : inline
     282             : std::vector<unsigned>& MultiValue::getIndices() {
     283     1379763 :   return indices;
     284             : }
     285             : 
     286             : inline
     287             : std::vector<Vector>& MultiValue::getAtomVector() {
     288      405876 :   return tmp_atoms;
     289             : }
     290             : 
     291             : inline
     292             : std::vector<Vector>& MultiValue::getFirstAtomVector() {
     293      670728 :   return tmp_atoms;
     294             : }
     295             : 
     296             : inline
     297             : std::vector<std::vector<Vector> >& MultiValue::getFirstAtomDerivativeVector() {
     298     1076604 :   return tmp_atom_der;
     299             : }
     300             : 
     301             : inline
     302             : const std::vector<std::vector<Vector> >& MultiValue::getConstFirstAtomDerivativeVector() const {
     303             :   return tmp_atom_der;
     304             : }
     305             : 
     306             : inline
     307             : std::vector<Tensor>& MultiValue::getFirstAtomVirialVector() {
     308      490140 :   return tmp_atom_virial;
     309             : }
     310             : 
     311             : inline
     312     8598870 : void MultiValue::stashMatrixElement( const unsigned& nmat, const unsigned& rowstart, const unsigned& jcol, const double& val ) {
     313             :   plumed_dbg_assert( jcol<nmatrix_cols && rowstart + matrix_bookeeping[rowstart]<matrix_bookeeping.size() && nmatrix_cols*nmat + matrix_bookeeping[rowstart]<matrix_row_stash.size() );
     314     8598870 :   matrix_bookeeping[rowstart]++;
     315     8598870 :   matrix_bookeeping[rowstart + matrix_bookeeping[rowstart]]=jcol;
     316     8598870 :   matrix_row_stash[ nmatrix_cols*nmat + jcol] = val;
     317     8598870 : }
     318             : 
     319             : inline
     320             : double MultiValue::getStashedMatrixElement( const unsigned& nmat, const unsigned& jcol ) const {
     321             :   plumed_dbg_assert( nmatrix_cols*nmat + jcol<matrix_row_stash.size() );
     322     7206015 :   return matrix_row_stash[ nmatrix_cols*nmat + jcol ];
     323             : }
     324             : 
     325             : inline
     326             : const std::vector<unsigned> & MultiValue::getMatrixBookeeping() const {
     327             :   return matrix_bookeeping;
     328             : }
     329             : 
     330             : inline
     331             : void MultiValue::setNumberOfMatrixRowDerivatives( const unsigned& nmat, const unsigned& nind ) {
     332             :   plumed_dbg_assert( nmat<matrix_row_nderivatives.size() && nind<=matrix_row_derivative_indices[nmat].size() );
     333    12992400 :   matrix_row_nderivatives[nmat]=nind;
     334             : }
     335             : 
     336             : inline
     337             : unsigned MultiValue::getNumberOfMatrixRowDerivatives( const unsigned& nmat ) const {
     338             :   plumed_dbg_assert( nmat<matrix_row_nderivatives.size() );
     339   108296955 :   return matrix_row_nderivatives[nmat];
     340             : }
     341             : 
     342             : inline
     343             : std::vector<unsigned>& MultiValue::getMatrixRowDerivativeIndices( const unsigned& nmat ) {
     344             :   plumed_dbg_assert( nmat<matrix_row_nderivatives.size() );
     345     1978950 :   return matrix_row_derivative_indices[nmat];
     346             : }
     347             : 
     348             : inline
     349             : void MultiValue::addMatrixForce( const unsigned& imat, const unsigned& jind, const double& f ) {
     350    19244157 :   matrix_force_stash[imat*nderivatives + jind]+=f;
     351             : }
     352             : 
     353             : inline
     354             : double MultiValue::getStashedMatrixForce( const unsigned& imat, const unsigned& jind ) const {
     355   842633458 :   return matrix_force_stash[imat*nderivatives + jind];
     356             : }
     357             : 
     358             : inline
     359      490140 : void MultiValue::resizeTemporyVector(const unsigned& n ) {
     360      490140 :   if( n>tmp_vectors.size() ) {
     361       31670 :     tmp_vectors.resize(n);
     362             :   }
     363      490140 : }
     364             : 
     365             : inline
     366             : std::vector<double>& MultiValue::getTemporyVector(const unsigned& ind ) {
     367             :   plumed_dbg_assert( ind<tmp_vectors.size() );
     368             :   return tmp_vectors[ind];
     369             : }
     370             : 
     371             : }
     372             : #endif

Generated by: LCOV version 1.16