LCOV - code coverage report
Current view: top level - tools - Pbc.cpp (source / functions) Hit Total Coverage
Test: plumed test coverage Lines: 145 152 95.4 %
Date: 2025-03-25 09:33:27 Functions: 12 13 92.3 %

          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 "Pbc.h"
      23             : #include "Tools.h"
      24             : #include "Exception.h"
      25             : #include "LatticeReduction.h"
      26             : #include <iostream>
      27             : #include "Random.h"
      28             : #include <cmath>
      29             : 
      30             : namespace PLMD {
      31             : 
      32       26321 : Pbc::Pbc():
      33      394815 :   type(unset) {
      34       26321 :   box.zero();
      35       26321 :   invBox.zero();
      36       26321 : }
      37             : 
      38       20240 : void Pbc::buildShifts(gch::small_vector<Vector,maxshiftsize> shifts[2][2][2])const {
      39             :   const double small=1e-28;
      40             : 
      41             : // clear all shifts
      42       60720 :   for(int i=0; i<2; i++)
      43      121440 :     for(int j=0; j<2; j++)
      44      242880 :       for(int k=0; k<2; k++) {
      45      161920 :         shifts[i][j][k].clear();
      46             :       }
      47             : 
      48             : // enumerate all possible shifts
      49             : // since box is reduced, only 27 shifts have to be attempted
      50       80960 :   for(int l=-1; l<=1; l++)
      51      242880 :     for(int m=-1; m<=1; m++)
      52      728640 :       for(int n=-1; n<=1; n++) {
      53             : 
      54             : // int/double shift vectors
      55      546480 :         const int ishift[3]= {l,m,n};
      56      546480 :         Vector dshift(l,m,n);
      57             : 
      58             : // count how many components are != 0
      59             :         unsigned count=0;
      60     2185920 :         for(int s=0; s<3; s++)
      61     1639440 :           if(ishift[s]!=0) {
      62     1092960 :             count++;
      63             :           }
      64             : 
      65             : // skips trivial (0,0,0) and cases with three shifts
      66             : // only 18 shifts survive past this point
      67      546480 :         if(count==0 || count==3) {
      68      261680 :           continue;
      69             :         }
      70             : 
      71             : // check if that Wigner-Seitz face is perpendicular to the axis.
      72             : // this allows to eliminate shifts in symmetric cells.
      73             : // e.g., if one lactice vector is orthogonal to the plane spanned
      74             : // by the other two vectors, that shift should never be tried
      75      364320 :         Vector cosdir=matmul(reduced,transpose(reduced),dshift);
      76      364320 :         double dp=dotProduct(dshift,cosdir);
      77      364320 :         double ref=modulo2(dshift)*modulo2(cosdir);
      78      364320 :         if(std::fabs(ref-dp*dp)<small) {
      79       79520 :           continue;
      80             :         }
      81             : 
      82             : // here we start pruning depending on the sign of the scaled coordinate
      83      854400 :         for(int i=0; i<2; i++)
      84     1708800 :           for(int j=0; j<2; j++)
      85     3417600 :             for(int k=0; k<2; k++) {
      86             : 
      87     2278400 :               const int block[3]= {2*i-1,2*j-1,2*k-1};
      88             : 
      89             : // skip cases where shift would bring too far from origin
      90             :               bool skip=false;
      91     9113600 :               for(int s=0; s<3; s++)
      92     6835200 :                 if(ishift[s]*block[s]>0) {
      93             :                   skip=true;
      94             :                 }
      95     2278400 :               if(skip) {
      96     1795510 :                 continue;
      97             :               }
      98             :               skip=true;
      99     3113328 :               for(int s=0; s<3; s++) {
     100             : // check that the components of cosdir along the non-shifted directions
     101             : // have the proper sign
     102     2334996 :                 if(((1-ishift[s]*ishift[s])*block[s])*cosdir[s]<-small) {
     103             :                   skip=false;
     104             :                 }
     105             :               }
     106      778332 :               if(skip) {
     107      295442 :                 continue;
     108             :               }
     109             : 
     110             : // if we arrive to this point, shift is eligible and is added to the list
     111      965780 :               shifts[i][j][k].push_back(matmul(transpose(reduced),dshift));
     112             :             }
     113             :       }
     114       20240 : }
     115             : 
     116      600000 : void Pbc::fullSearch(Vector&d)const {
     117      600000 :   if(type==unset) {
     118       71900 :     return;
     119             :   }
     120      528100 :   Vector s=matmul(invReduced.transpose(),d);
     121     2112400 :   for(int i=0; i<3; i++) {
     122     1584300 :     s[i]=Tools::pbc(s[i]);
     123             :   }
     124      528100 :   d=matmul(reduced.transpose(),s);
     125             :   const int smax=4;
     126      528100 :   Vector a0(reduced.getRow(0));
     127      528100 :   Vector a1(reduced.getRow(1));
     128      528100 :   Vector a2(reduced.getRow(2));
     129      528100 :   Vector best(d);
     130      528100 :   double lbest=d.modulo2();
     131     5281000 :   for(int i=-smax; i<=smax; i++)
     132    47529000 :     for(int j=-smax; j<=smax; j++)
     133   427761000 :       for(int k=-smax; k<=smax; k++) {
     134   384984900 :         Vector trial=d+i*a0+j*a1+k*a2;
     135   384984900 :         double ltrial=trial.modulo2();
     136   384984900 :         if(ltrial<lbest) {
     137       29897 :           best=trial;
     138             :           lbest=ltrial;
     139             :         }
     140             :       }
     141      528100 :   d=best;
     142             : }
     143             : 
     144       93182 : void Pbc::setBox(const Tensor&b) {
     145       93182 :   box=b;
     146             : // detect type:
     147             :   const double epsilon=1e-28;
     148             : 
     149       93182 :   type=unset;
     150       93182 :   double det=box.determinant();
     151       93182 :   if(det*det<epsilon) {
     152             :     return;
     153             :   }
     154             : 
     155             :   bool cxy=false;
     156             :   bool cxz=false;
     157             :   bool cyz=false;
     158       91141 :   if(box(0,1)*box(0,1)<epsilon && box(1,0)*box(1,0)<epsilon) {
     159             :     cxy=true;
     160             :   }
     161       91141 :   if(box(0,2)*box(0,2)<epsilon && box(2,0)*box(2,0)<epsilon) {
     162             :     cxz=true;
     163             :   }
     164       91141 :   if(box(1,2)*box(1,2)<epsilon && box(2,1)*box(2,1)<epsilon) {
     165             :     cyz=true;
     166             :   }
     167             : 
     168       91141 :   invBox=box.inverse();
     169             : 
     170       91141 :   if(cxy && cxz && cyz) {
     171       70901 :     type=orthorombic;
     172             :   } else {
     173       20240 :     type=generic;
     174             :   }
     175             : 
     176       91141 :   if(type==orthorombic) {
     177       70901 :     reduced=box;
     178       70901 :     invReduced=inverse(reduced);
     179      283604 :     for(unsigned i=0; i<3; i++) {
     180      212703 :       diag[i]=box[i][i];
     181      212703 :       hdiag[i]=0.5*box[i][i];
     182      212703 :       mdiag[i]=-0.5*box[i][i];
     183             :     }
     184             :   } else {
     185       20240 :     reduced=box;
     186       20240 :     LatticeReduction::reduce(reduced);
     187       20240 :     invReduced=inverse(reduced);
     188       20240 :     buildShifts(shifts);
     189             :   }
     190             : 
     191             : }
     192             : 
     193           0 : double Pbc::distance( const bool pbc, const Vector& v1, const Vector& v2 ) const {
     194           0 :   if(pbc) {
     195           0 :     return ( distance(v1,v2) ).modulo();
     196             :   } else {
     197           0 :     return ( delta(v1,v2) ).modulo();
     198             :   }
     199             : }
     200             : 
     201      405876 : void Pbc::apply(std::vector<Vector>& dlist, unsigned max_index) const {
     202      405876 :   apply(VectorView(&dlist[0][0],dlist.size()),max_index);
     203      405876 : }
     204             : 
     205      405876 : void Pbc::apply(VectorView dlist, unsigned max_index) const {
     206      405876 :   if (max_index==0) {
     207           0 :     max_index=dlist.size();
     208             :   }
     209      405876 :   if(type==unset) {
     210             :     // do nothing
     211      405872 :   } else if(type==orthorombic) {
     212             : #ifdef __PLUMED_PBC_WHILE
     213             :     for(unsigned k=0; k<max_index; ++k) {
     214             :       while(dlist[k][0]>hdiag[0]) {
     215             :         dlist[k][0]-=diag[0];
     216             :       }
     217             :       while(dlist[k][0]<=mdiag[0]) {
     218             :         dlist[k][0]+=diag[0];
     219             :       }
     220             :       while(dlist[k][1]>hdiag[1]) {
     221             :         dlist[k][1]-=diag[1];
     222             :       }
     223             :       while(dlist[k][1]<=mdiag[1]) {
     224             :         dlist[k][1]+=diag[1];
     225             :       }
     226             :       while(dlist[k][2]>hdiag[2]) {
     227             :         dlist[k][2]-=diag[2];
     228             :       }
     229             :       while(dlist[k][2]<=mdiag[2]) {
     230             :         dlist[k][2]+=diag[2];
     231             :       }
     232             :     }
     233             : #else
     234    26205115 :     for(unsigned k=0; k<max_index; ++k) {
     235   103197924 :       for(int i=0; i<3; i++) {
     236    77398443 :         dlist[k][i]=Tools::pbc(dlist[k][i]*invBox(i,i))*box(i,i);
     237             :       }
     238             :     }
     239             : #endif
     240         238 :   } else if(type==generic) {
     241       21116 :     for(unsigned k=0; k<max_index; ++k) {
     242             :       //Inlining by hand this part of function from distance speeds up by about 20-30%
     243             :       //against the previos version, and 60-80% agains this version non inlined.
     244             :       //I do not think is the `if(nshifts) *nshifts+=myshifts.size();`,
     245             :       //but that the compiler now see how we are juggling with the memory and it
     246             :       //does its magic
     247             : 
     248             :       //I tried writing VectorGeneric<3> matmul(const MemoryView<3UL> a,const TensorGeneric<3,3>&b)
     249             :       // by copy-pasting the original vector-tensor, but slows down this method by 10%... (on gcc9)
     250       20878 :       Vector s=matmul(Vector{dlist[k][0],dlist[k][1],dlist[k][2]},invReduced);
     251             :       // bring to -0.5,+0.5 region in scaled coordinates:
     252       83512 :       for(int i=0; i<3; ++i) {
     253       62634 :         s[i]=Tools::pbc(s[i]);
     254             :       }
     255       20878 :       Vector best(matmul(s,reduced));
     256             :       // check if shifts have to be attempted:
     257       20878 :       if((std::fabs(s[0])+std::fabs(s[1])+std::fabs(s[2])>0.5)) {
     258             :         // list of shifts is specific for that "octant" (depends on signs of s[i]):
     259       34605 :         const auto & myshifts(shifts[(s[0]>0?1:0)][(s[1]>0?1:0)][(s[2]>0?1:0)]);
     260       17298 :         Vector reference = best;
     261       17298 :         double lbest(modulo2(best));
     262             :         // loop over possible shifts:
     263       81547 :         for(unsigned i=0; i<myshifts.size(); ++i) {
     264       64249 :           Vector trial=reference+myshifts[i];
     265       64249 :           double ltrial=modulo2(trial);
     266       64249 :           if(ltrial<lbest) {
     267             :             lbest=ltrial;
     268        1433 :             best=trial;
     269             :           }
     270             :         }
     271             :       }
     272       20878 :       dlist[k][0]  = best[0];
     273       20878 :       dlist[k][1]  = best[1];
     274       20878 :       dlist[k][2]  = best[2];
     275             :     }
     276             :   } else {
     277           0 :     plumed_merror("unknown pbc type");
     278             :   }
     279      405876 : }
     280             : 
     281   559421392 : Vector Pbc::distance(const Vector&v1,const Vector&v2,int*nshifts)const {
     282   559421392 :   Vector d=delta(v1,v2);
     283   559421392 :   if(type==unset) {
     284             :     // do nothing
     285   533228202 :   } else if(type==orthorombic) {
     286             : #ifdef __PLUMED_PBC_WHILE
     287             :     for(unsigned i=0; i<3; i++) {
     288             :       while(d[i]>hdiag[i]) {
     289             :         d[i]-=diag[i];
     290             :       }
     291             :       while(d[i]<=mdiag[i]) {
     292             :         d[i]+=diag[i];
     293             :       }
     294             :     }
     295             : #else
     296  1939966680 :     for(int i=0; i<3; i++) {
     297  1454975010 :       d[i]=Tools::pbc(d[i]*invBox(i,i))*box(i,i);
     298             :     }
     299             : #endif
     300    48236532 :   } else if(type==generic) {
     301    48236532 :     Vector s=matmul(d,invReduced);
     302             : // check if images have to be computed:
     303             : //    if((std::fabs(s[0])+std::fabs(s[1])+std::fabs(s[2])>0.5)){
     304             : // NOTICE: the check in the previous line, albeit correct, is breaking many regtest
     305             : //         since it does not apply Tools::pbc in many cases. Moreover, it does not
     306             : //         introduce a significant gain. I thus leave it out for the moment.
     307             :     if constexpr (true) {
     308             : // bring to -0.5,+0.5 region in scaled coordinates:
     309   192946128 :       for(int i=0; i<3; i++) {
     310   144709596 :         s[i]=Tools::pbc(s[i]);
     311             :       }
     312    48236532 :       d=matmul(s,reduced);
     313             : // check if shifts have to be attempted:
     314    48236532 :       if((std::fabs(s[0])+std::fabs(s[1])+std::fabs(s[2])>0.5)) {
     315             : // list of shifts is specific for that "octant" (depends on signs of s[i]):
     316    78664302 :         const auto & myshifts(shifts[(s[0]>0?1:0)][(s[1]>0?1:0)][(s[2]>0?1:0)]);
     317    39816653 :         Vector best(d);
     318    39816653 :         double lbest(modulo2(best));
     319             : // loop over possible shifts:
     320    39816653 :         if(nshifts) {
     321      109939 :           *nshifts+=myshifts.size();
     322             :         }
     323   179244081 :         for(unsigned i=0; i<myshifts.size(); i++) {
     324   139427428 :           Vector trial=d+myshifts[i];
     325   139427428 :           double ltrial=modulo2(trial);
     326   139427428 :           if(ltrial<lbest) {
     327             :             lbest=ltrial;
     328     3759325 :             best=trial;
     329             :           }
     330             :         }
     331    39816653 :         d=best;
     332             :       }
     333             :     }
     334             :   } else {
     335           0 :     plumed_merror("unknown pbc type");
     336             :   }
     337   559421392 :   return d;
     338             : }
     339             : 
     340     1974153 : Vector Pbc::realToScaled(const Vector&d)const {
     341     1974153 :   return matmul(invBox.transpose(),d);
     342             : }
     343             : 
     344      207347 : Vector Pbc::scaledToReal(const Vector&d)const {
     345      207347 :   return matmul(box.transpose(),d);
     346             : }
     347             : 
     348        8569 : bool Pbc::isOrthorombic()const {
     349        8569 :   return type==orthorombic;
     350             : }
     351             : 
     352       28812 : const Tensor& Pbc::getBox()const {
     353       28812 :   return box;
     354             : }
     355             : 
     356       45554 : const Tensor& Pbc::getInvBox()const {
     357       45554 :   return invBox;
     358             : }
     359             : 
     360             : }

Generated by: LCOV version 1.16