LCOV - code coverage report
Current view: top level - tools - Vector.h (source / functions) Hit Total Coverage
Test: plumed test coverage Lines: 66 71 93.0 %
Date: 2025-03-25 09:33:27 Functions: 53 68 77.9 %

          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_Vector_h
      23             : #define __PLUMED_tools_Vector_h
      24             : 
      25             : #include <cmath>
      26             : #include <iosfwd>
      27             : #include <array>
      28             : #include "LoopUnroller.h"
      29             : 
      30             : namespace PLMD {
      31             : 
      32             : /**
      33             : \ingroup TOOLBOX
      34             : Class implementing fixed size vectors of doubles
      35             : 
      36             : \tparam n The number of elements of the vector.
      37             : 
      38             : This class implements a vector of doubles with size fixed at
      39             : compile time. It is useful for small fixed size objects (e.g.
      40             : 3d vectors) as it does not waste space to store the vector size.
      41             : Moreover, as the compiler knows the size, it can be completely
      42             : opimized inline.
      43             : All the methods are inlined for better optimization and
      44             : all the loops are explicitly unrolled using PLMD::LoopUnroller class.
      45             : Vector elements are initialized to zero by default. Notice that
      46             : this means that constructor is a bit slow. This point might change
      47             : in future if we find performance issues.
      48             : Accepts both [] and () syntax for access.
      49             : Several functions are declared as friends even if not necessary so as to
      50             : properly appear in Doxygen documentation.
      51             : 
      52             : Aliases are defined to simplify common declarations (Vector, Vector2d, Vector3d, Vector4d).
      53             : Also notice that some operations are only available for 3 dimensional vectors.
      54             : 
      55             : Example of usage
      56             : \verbatim
      57             : #include "Vector.h"
      58             : 
      59             : using namespace PLMD;
      60             : 
      61             : int main(){
      62             :   VectorGeneric<3> v1;
      63             :   v1[0]=3.0;
      64             : // use equivalently () and [] syntax:
      65             :   v1(1)=5.0;
      66             : // initialize with components
      67             :   VectorGeneric<3> v2=VectorGeneric<3>(1.0,2.0,3.0);
      68             :   VectorGeneric<3> v3=crossProduct(v1,v2);
      69             :   double d=dotProduct(v1,v2);
      70             :   v3+=v1;
      71             :   v2=v1+2.0*v3;
      72             : }
      73             : \endverbatim
      74             : 
      75             : */
      76             : 
      77             : 
      78             : template <unsigned n>
      79             : class VectorGeneric {
      80             :   std::array<double,n> d;
      81             : /// Auxiliary private function for constructor
      82             :   void auxiliaryConstructor();
      83             : /// Auxiliary private function for constructor
      84             :   template<typename... Args>
      85             :   void auxiliaryConstructor(double first,Args... arg);
      86             : public:
      87             : /// Constructor accepting n double parameters.
      88             : /// Can be used as Vector<3>(1.0,2.0,3.0) or Vector<2>(2.0,3.0).
      89             : /// In case a wrong number of parameters is given, a static assertion will fail.
      90             :   template<typename... Args>
      91             :   VectorGeneric(double first,Args... arg);
      92             : /// create it null
      93             :   VectorGeneric();
      94             : /// set it to zero
      95             :   void zero();
      96             : /// array-like access [i]
      97             :   double & operator[](unsigned i);
      98             : /// array-like access [i]
      99             :   const double & operator[](unsigned i)const;
     100             : /// parenthesis access (i)
     101             :   double & operator()(unsigned i);
     102             : /// parenthesis access (i)
     103             :   const double & operator()(unsigned i)const;
     104             : /// increment
     105             :   VectorGeneric& operator +=(const VectorGeneric& b);
     106             : /// decrement
     107             :   VectorGeneric& operator -=(const VectorGeneric& b);
     108             : /// multiply
     109             :   VectorGeneric& operator *=(double s);
     110             : /// divide
     111             :   VectorGeneric& operator /=(double s);
     112             : /// sign +
     113             :   VectorGeneric operator +()const;
     114             : /// sign -
     115             :   VectorGeneric operator -()const;
     116             : /// return v1+v2
     117             :   template<unsigned m>
     118             :   friend VectorGeneric<m> operator+(const VectorGeneric<m>&,const VectorGeneric<m>&);
     119             : /// return v1-v2
     120             :   template<unsigned m>
     121             :   friend VectorGeneric<m> operator-(const VectorGeneric<m>&,const VectorGeneric<m>&);
     122             : /// return s*v
     123             :   template<unsigned m>
     124             :   friend VectorGeneric<m> operator*(double,const VectorGeneric<m>&);
     125             : /// return v*s
     126             :   template<unsigned m>
     127             :   friend VectorGeneric<m> operator*(const VectorGeneric<m>&,double);
     128             : /// return v/s
     129             :   template<unsigned m>
     130             :   friend VectorGeneric<m> operator/(const VectorGeneric<m>&,double);
     131             : /// return v2-v1
     132             :   template<unsigned m>
     133             :   friend VectorGeneric<m> delta(const VectorGeneric<m>&v1,const VectorGeneric<m>&v2);
     134             : /// return v1 .scalar. v2
     135             :   template<unsigned m>
     136             :   friend double dotProduct(const VectorGeneric<m>&,const VectorGeneric<m>&);
     137             : /// return v1 .vector. v2
     138             : /// Only available for size 3
     139             :   friend VectorGeneric<3> crossProduct(const VectorGeneric<3>&,const VectorGeneric<3>&);
     140             : /// compute the squared modulo
     141             :   double modulo2()const;
     142             : /// Compute the modulo.
     143             : /// Shortcut for sqrt(v.modulo2())
     144             :   double modulo()const;
     145             : /// friend version of modulo2 (to simplify some syntax)
     146             :   template<unsigned m>
     147             :   friend double modulo2(const VectorGeneric<m>&);
     148             : /// friend version of modulo (to simplify some syntax)
     149             :   template<unsigned m>
     150             :   friend double modulo(const VectorGeneric<m>&);
     151             : /// << operator.
     152             : /// Allows printing vector `v` with `std::cout<<v;`
     153             :   template<unsigned m>
     154             :   friend std::ostream & operator<<(std::ostream &os, const VectorGeneric<m>&);
     155             : };
     156             : 
     157             : template <unsigned n>
     158             : void VectorGeneric<n>::auxiliaryConstructor()
     159             : {}
     160             : 
     161             : template <unsigned n>
     162             : template<typename... Args>
     163   191221918 : void VectorGeneric<n>::auxiliaryConstructor(double first,Args... arg) {
     164   191221918 :   d[n-(sizeof...(Args))-1]=first;
     165   131763012 :   auxiliaryConstructor(arg...);
     166   191221918 : }
     167             : 
     168             : template <unsigned n>
     169             : template<typename... Args>
     170    59458906 : VectorGeneric<n>::VectorGeneric(double first,Args... arg) {
     171             :   static_assert((sizeof...(Args))+1==n,"you are trying to initialize a Vector with the wrong number of arguments");
     172    59458906 :   auxiliaryConstructor(first,arg...);
     173    59458906 : }
     174             : 
     175             : template <unsigned n>
     176    25037430 : void VectorGeneric<n>::zero() {
     177    25037430 :   LoopUnroller<n>::_zero(d.data());
     178    25037430 : }
     179             : 
     180             : template <unsigned n>
     181   646926254 : VectorGeneric<n>::VectorGeneric() {
     182   646343536 :   LoopUnroller<n>::_zero(d.data());
     183   646926254 : }
     184             : 
     185             : template <unsigned n>
     186  2361748037 : double & VectorGeneric<n>::operator[](unsigned i) {
     187  2361748037 :   return d[i];
     188             : }
     189             : 
     190             : template <unsigned n>
     191  2551360111 : const double & VectorGeneric<n>::operator[](unsigned i)const {
     192  2551360111 :   return d[i];
     193             : }
     194             : 
     195             : template <unsigned n>
     196  1581184674 : double & VectorGeneric<n>::operator()(unsigned i) {
     197  1581184674 :   return d[i];
     198             : }
     199             : 
     200             : template <unsigned n>
     201  1531216359 : const double & VectorGeneric<n>::operator()(unsigned i)const {
     202  1531216359 :   return d[i];
     203             : }
     204             : 
     205             : template <unsigned n>
     206  1960946512 : VectorGeneric<n>& VectorGeneric<n>::operator +=(const VectorGeneric<n>& b) {
     207  1960946512 :   LoopUnroller<n>::_add(d.data(),b.d.data());
     208  1960946512 :   return *this;
     209             : }
     210             : 
     211             : template <unsigned n>
     212  1914905212 : VectorGeneric<n>& VectorGeneric<n>::operator -=(const VectorGeneric<n>& b) {
     213  1914905212 :   LoopUnroller<n>::_sub(d.data(),b.d.data());
     214  1914905212 :   return *this;
     215             : }
     216             : 
     217             : template <unsigned n>
     218  2439672640 : VectorGeneric<n>& VectorGeneric<n>::operator *=(double s) {
     219  2439672640 :   LoopUnroller<n>::_mul(d.data(),s);
     220  2439672640 :   return *this;
     221             : }
     222             : 
     223             : template <unsigned n>
     224       57841 : VectorGeneric<n>& VectorGeneric<n>::operator /=(double s) {
     225       57841 :   LoopUnroller<n>::_mul(d.data(),1.0/s);
     226       57841 :   return *this;
     227             : }
     228             : 
     229             : template <unsigned n>
     230             : VectorGeneric<n>  VectorGeneric<n>::operator +()const {
     231             :   return *this;
     232             : }
     233             : 
     234             : template <unsigned n>
     235   206458710 : VectorGeneric<n> VectorGeneric<n>::operator -()const {
     236   206458710 :   VectorGeneric<n> r;
     237   206458710 :   LoopUnroller<n>::_neg(r.d.data(),d.data());
     238   206458710 :   return r;
     239             : }
     240             : 
     241             : template <unsigned n>
     242  1319412804 : VectorGeneric<n> operator+(const VectorGeneric<n>&v1,const VectorGeneric<n>&v2) {
     243  1319412804 :   VectorGeneric<n> v(v1);
     244  1319412804 :   return v+=v2;
     245             : }
     246             : 
     247             : template <unsigned n>
     248  1371788948 : VectorGeneric<n> operator-(const VectorGeneric<n>&v1,const VectorGeneric<n>&v2) {
     249  1371788948 :   VectorGeneric<n> v(v1);
     250  1371788948 :   return v-=v2;
     251             : }
     252             : 
     253             : template <unsigned n>
     254  2437590445 : VectorGeneric<n> operator*(double s,const VectorGeneric<n>&v) {
     255  2437590445 :   VectorGeneric<n> vv(v);
     256  2437590445 :   return vv*=s;
     257             : }
     258             : 
     259             : template <unsigned n>
     260   947466513 : VectorGeneric<n> operator*(const VectorGeneric<n>&v,double s) {
     261   947466513 :   return s*v;
     262             : }
     263             : 
     264             : template <unsigned n>
     265   359170206 : VectorGeneric<n> operator/(const VectorGeneric<n>&v,double s) {
     266   359170206 :   return v*(1.0/s);
     267             : }
     268             : 
     269             : template <unsigned n>
     270   993455922 : VectorGeneric<n> delta(const VectorGeneric<n>&v1,const VectorGeneric<n>&v2) {
     271   993455922 :   return v2-v1;
     272             : }
     273             : 
     274             : template <unsigned n>
     275  1109907003 : double VectorGeneric<n>::modulo2()const {
     276  1109907003 :   return LoopUnroller<n>::_sum2(d.data());
     277             : }
     278             : 
     279             : template <unsigned n>
     280   265375907 : double dotProduct(const VectorGeneric<n>& v1,const VectorGeneric<n>& v2) {
     281   265375907 :   return LoopUnroller<n>::_dot(v1.d.data(),v2.d.data());
     282             : }
     283             : 
     284             : inline
     285     5459803 : VectorGeneric<3> crossProduct(const VectorGeneric<3>& v1,const VectorGeneric<3>& v2) {
     286             :   return VectorGeneric<3>(
     287     5459803 :            v1[1]*v2[2]-v1[2]*v2[1],
     288     5459803 :            v1[2]*v2[0]-v1[0]*v2[2],
     289     5459803 :            v1[0]*v2[1]-v1[1]*v2[0]);
     290             : }
     291             : 
     292             : template<unsigned n>
     293   408021831 : double VectorGeneric<n>::modulo()const {
     294   408021831 :   return sqrt(modulo2());
     295             : }
     296             : 
     297             : template<unsigned n>
     298   203397173 : double modulo2(const VectorGeneric<n>&v) {
     299   203397173 :   return v.modulo2();
     300             : }
     301             : 
     302             : template<unsigned n>
     303    34471352 : double modulo(const VectorGeneric<n>&v) {
     304    34471352 :   return v.modulo();
     305             : }
     306             : 
     307             : template<unsigned n>
     308           0 : std::ostream & operator<<(std::ostream &os, const VectorGeneric<n>& v) {
     309           0 :   for(unsigned i=0; i<n-1; i++) {
     310           0 :     os<<v(i)<<" ";
     311             :   }
     312           0 :   os<<v(n-1);
     313           0 :   return os;
     314             : }
     315             : 
     316             : 
     317             : /// \ingroup TOOLBOX
     318             : /// Alias for one dimensional vectors
     319             : typedef VectorGeneric<1> Vector1d;
     320             : /// \ingroup TOOLBOX
     321             : /// Alias for two dimensional vectors
     322             : typedef VectorGeneric<2> Vector2d;
     323             : /// \ingroup TOOLBOX
     324             : /// Alias for three dimensional vectors
     325             : typedef VectorGeneric<3> Vector3d;
     326             : /// \ingroup TOOLBOX
     327             : /// Alias for four dimensional vectors
     328             : typedef VectorGeneric<4> Vector4d;
     329             : /// \ingroup TOOLBOX
     330             : /// Alias for five dimensional vectors
     331             : typedef VectorGeneric<5> Vector5d;
     332             : /// \ingroup TOOLBOX
     333             : /// Alias for three dimensional vectors
     334             : typedef Vector3d Vector;
     335             : 
     336             : static_assert(sizeof(VectorGeneric<2>)==2*sizeof(double), "code cannot work if this is not satisfied");
     337             : static_assert(sizeof(VectorGeneric<3>)==3*sizeof(double), "code cannot work if this is not satisfied");
     338             : static_assert(sizeof(VectorGeneric<4>)==4*sizeof(double), "code cannot work if this is not satisfied");
     339             : 
     340             : }
     341             : 
     342             : #endif
     343             : 

Generated by: LCOV version 1.16