From 0cb4e830fae4df7d92ee4c882a926a2bd1eee7e5 Mon Sep 17 00:00:00 2001
From: Gareth Tribello <gareth.tribello@gmail.com>
Date: Sat, 19 Dec 2015 13:31:21 +0000
Subject: [PATCH] First go at new grid implementation

---
 src/vesselbase/GridVessel.cpp | 208 ++++++++++++++++++++++++++++++++++
 src/vesselbase/GridVessel.h   | 136 ++++++++++++++++++++++
 2 files changed, 344 insertions(+)
 create mode 100644 src/vesselbase/GridVessel.cpp
 create mode 100644 src/vesselbase/GridVessel.h

diff --git a/src/vesselbase/GridVessel.cpp b/src/vesselbase/GridVessel.cpp
new file mode 100644
index 000000000..01416fbfc
--- /dev/null
+++ b/src/vesselbase/GridVessel.cpp
@@ -0,0 +1,208 @@
+/* +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
+   Copyright (c) 2012 The plumed team
+   (see the PEOPLE file at the root of the distribution for a list of names)
+
+   See http://www.plumed-code.org for more information.
+
+   This file is part of plumed, version 2.0.
+
+   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 "GridVessel.h"
+#include "ActionWithVessel.h"
+#include "tools/Tools.h"
+
+namespace PLMD {
+namespace vesselbase {
+
+void GridVessel::registerKeywords( Keywords& keys ){
+  Vessel::registerKeywords( keys );
+  keys.add("compulsory","COMPONENTS","the names of the components in the vector");
+  keys.add("compulsory","COORDINATES","the names of the coordinates of the grid");
+  keys.add("compulsory","PBC","is the grid periodic in each direction or not");
+  keys.add("compulsory","MIN","minimum values for the grid");
+  keys.add("compulsory","MAX","maximum values for the grid");
+  keys.add("compulsory","NBIN","number of bins in each direction for the grid");
+}
+
+GridVessel::GridVessel( const VesselOptions& da ):
+Vessel(da),
+bold(0)
+{
+  std::vector<std::string> compnames; parseVector("COMPONENTS",compnames);
+  std::vector<std::string> coordnames; parseVector("COORDINATES",coordnames);
+  dimension=coordnames.size();
+  std::vector<std::string> spbc( dimension ); parseVector("PBC",spbc); 
+  str_min.resize( dimension); parseVector("MIN",str_min); 
+  str_max.resize( dimension ); parseVector("MAX",str_max);
+  nbin.resize( dimension ); parseVector("NBIN",nbin);
+
+  tmp_indices.resize( str_min.size() );
+  current_neigh.resize( static_cast<unsigned>(pow(2.,dimension)) );
+
+  unsigned n=0; nper=compnames.size()*( 1 + coordnames.size() );
+  arg_names.resize( compnames.size()*( 1 + coordnames.size() ) );
+  for(unsigned i=0;i<compnames.size();++i){
+      arg_names[n]=compnames[i]; n++;
+      for(unsigned j=0;j<coordnames.size();++j){ arg_names[n] = "d" + compnames[i] + "_" + coordnames[j]; n++; }
+  }
+
+  npoints=1; dx.resize( dimension ); min.resize( dimension ); stride.resize( dimension );
+  max.resize( dimension ); arg_names.resize( nper + dimension );
+  for(unsigned i=0;i<dimension;++i){
+      Tools::convert( str_min[i], min[i] );
+      Tools::convert( str_max[i], max[i] );
+      dx[i] = ( max[i] - min[i] ) / static_cast<double>( nbin[i] );
+      if( spbc[i]=="F" ){ max[i] +=dx[i]; nbin[i]+=1; pbc[i]=false; } 
+      else if( spbc[i]=="T" ) pbc[i]=true;
+      else plumed_error();
+      stride[i]=npoints;
+      npoints*=nbin[i];
+  }
+}
+
+std::string GridVessel::getGridDescription() const {
+  std::string des="grid of "; std::string num;
+  for(unsigned i=0;i<dimension-1;++i){
+      Tools::convert( nbin[i], num );
+      des += num + " X ";
+  }
+  Tools::convert( nbin[dimension-1], num );
+  des += num + " equally spaced points between (";
+  for(unsigned i=0;i<dimension-1;++i) des += str_min[i] + ",";
+  Tools::convert( nbin[dimension-1], num );
+  des += str_min[dimension-1] + ") and (";
+  for(unsigned i=0;i<dimension-1;++i) des += str_max[i] + ",";
+  des += str_max[dimension-1] + ")";
+  return des;
+}
+
+void GridVessel::resize(){
+  plumed_massert( nper>0, "Number of datapoints at each grid point has not been set");
+  data.resize( npoints*nper );
+}
+
+unsigned GridVessel::getIndex( const std::vector<unsigned>& indices ) const {
+  plumed_dbg_assert( indices.size()==dimension );
+  // indices are flattended using a column-major order
+  unsigned index=indices[dimension-1];
+  for(unsigned i=dimension-1;i>0;--i){
+    index=index*nbin[i-1]+indices[i-1];
+  } 
+  return index;
+}
+
+void GridVessel::getIndices( const unsigned& index, std::vector<unsigned>& indices ) const {
+ unsigned kk=index;
+ indices[0]=index%nbin[0];
+ for(unsigned i=1;i<dimension-1;++i){
+    kk=(kk-indices[i-1])/nbin[i-1];
+    indices[i]=kk%nbin[i];
+ } 
+ if(dimension>=2){  // I think this is wrong
+    indices[dimension-1]=(kk-indices[dimension-2])/nbin[dimension-2];
+ }
+}
+
+void GridVessel::getGridPointCoordinates( const unsigned& ipoint , std::vector<double>& x ){
+  plumed_dbg_assert( x.size()==dimension && ipoint<npoints );
+  currentGridPoint=ipoint; getIndices( ipoint, tmp_indices ); 
+  for(unsigned i=0;i<dimension;++i) x[i] = min[i] + dx[i]*tmp_indices[i];
+}
+
+unsigned GridVessel::getLocationOnGrid( const std::vector<double>& x, std::vector<double>& dd ){
+  plumed_dbg_assert( x.size()==dimension && dd.size()==dimension );
+  getIndices( bold, tmp_indices ); bool changebox=false;
+  for(unsigned i=0;i<dimension;++i){
+     double bb = x[i] - min[i];
+     if ( bb<0.0 || bb>dx[i]*nbin[i] ){
+        getAction()->error("Extrapolation of function is not allowed");
+     } else if( bb<tmp_indices[i]*dx[i] || bb>(tmp_indices[i]+1)*dx[i] ){
+        tmp_indices[i]=static_cast<unsigned>( std::floor(bb/dx[i]) );
+        changebox=true;
+     }
+     dd[i] = bb/dx[i] - static_cast<double>(tmp_indices[i]);
+  }
+  if(changebox) bold = getIndex( tmp_indices );
+  return bold;
+}
+
+void GridVessel::getSplineNeighbors( const unsigned& mybox, std::vector<unsigned>& mysneigh ){
+  if( mysneigh.size()!=current_neigh.size() ) mysneigh.resize( current_neigh.size() );   
+
+  if( bold!=mybox ){
+     std::vector<unsigned> my_indices( dimension );
+     getIndices( mybox, my_indices );
+     for(unsigned i=0;i<current_neigh.size();++i){
+        unsigned tmp=i;
+        for(unsigned j=0;j<dimension;++j){
+           unsigned i0=tmp%2+my_indices[j]; tmp/=2;
+           if(!pbc[j] && i0==nbin[j]) getAction()->error("Extrapolating function on grid");
+           if( pbc[j] && i0==nbin[j]) i0=0;
+           tmp_indices[j]=i0;
+        }
+        current_neigh[i]=getIndex( tmp_indices );
+     }
+     bold=mybox;
+  }
+  for(unsigned i=0;i<current_neigh.size();++i) mysneigh[i]=current_neigh[i];
+}
+
+double GridVessel::getGridElement( const unsigned& ipoint, const unsigned& jelement ) const {
+  plumed_dbg_assert( ipoint<npoints && jelement<nper );
+  return data[ nper*ipoint + jelement ];
+}
+
+void GridVessel::setGridElement( const unsigned& ipoint, const unsigned& jelement, const double& value ){
+  plumed_dbg_assert( ipoint<npoints && jelement<nper );
+  data[ nper*ipoint + jelement ] = value;
+}
+
+void GridVessel::addToGridElement( const unsigned& ipoint, const unsigned& jelement, const double& value ){
+  plumed_dbg_assert( ipoint<npoints && jelement<nper );
+  data[ nper*ipoint + jelement ] += value;
+}
+
+double GridVessel::getGridElement( const std::vector<unsigned>& indices, const unsigned& jelement ) const {
+  return getGridElement( getIndex( indices ), jelement );
+}
+
+void GridVessel::setGridElement( const std::vector<unsigned>& indices, const unsigned& jelement, const double& value ){
+  setGridElement( getIndex( indices ), jelement, value );
+}
+
+void GridVessel::addToGridElement( const std::vector<unsigned>& indices, const unsigned& jelement, const double& value ){
+  addToGridElement( getIndex( indices ), jelement, value );
+}
+
+std::vector<std::string> GridVessel::getMin() const {
+  return str_min;
+}
+  
+std::vector<std::string> GridVessel::getMax() const {
+  return str_max;
+}
+
+std::vector<unsigned> GridVessel::getNbin() const {
+  std::vector<unsigned> ngrid( dimension );
+  for(unsigned i=0;i<dimension;++i){
+      if( !pbc[i] ) ngrid[i]=nbin[i] - 1;
+      else ngrid[i]=nbin[i];
+  }
+  return ngrid;
+}
+
+}
+}
+
diff --git a/src/vesselbase/GridVessel.h b/src/vesselbase/GridVessel.h
new file mode 100644
index 000000000..166fa668e
--- /dev/null
+++ b/src/vesselbase/GridVessel.h
@@ -0,0 +1,136 @@
+/* +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
+   Copyright (c) 2012 The plumed team
+   (see the PEOPLE file at the root of the distribution for a list of names)
+
+   See http://www.plumed-code.org for more information.
+
+   This file is part of plumed, version 2.0.
+
+   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/>.
++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ */
+#ifndef __PLUMED_vesselbase_GridVessel_h
+#define __PLUMED_vesselbase_GridVessel_h
+
+#include <string>
+#include <cstring>
+#include <vector>
+#include "Vessel.h"
+
+namespace PLMD {
+namespace vesselbase {
+
+class GridVessel : public Vessel {
+private:
+/// These two variables are used to 
+/// remember the box we were in on the previous call
+ unsigned bold;
+/// The number of points in the grid
+ unsigned npoints;
+/// Remember the neighbors that were used last time
+ std::vector<unsigned> current_neigh; 
+/// The names of the various columns in the grid file
+ std::vector<std::string> arg_names;
+/// The minimum and maximum in the grid stored as strings 
+ std::vector<std::string> str_min, str_max;
+/// The minimum and maximum of the grid stored as doubles
+ std::vector<double> min, max;
+/// The spacing between grid points
+ std::vector<double> dx;
+/// The numerical distance between adjacent grid points
+ std::vector<unsigned> stride;
+/// The number of bins in each grid direction
+ std::vector<unsigned> nbin;
+/// Is this direction periodic
+ std::vector<bool> pbc;
+/// A tempory array that can be used to store indices for a point
+ std::vector<unsigned> tmp_indices;
+/// The grid with all the data values on it
+ std::vector<double> data;
+///  Flatten the grid and get the grid index for a point
+ unsigned getIndex( const std::vector<unsigned>& indices ) const ;
+/// The number of pieces of information we are storing for each 
+/// point in the grid
+ unsigned nper;
+/// The dimensionality of the grid
+ unsigned dimension;
+/// The grid point that was requested last by getGridPointCoordinates
+ unsigned currentGridPoint;
+public:
+/// keywords
+  static void registerKeywords( Keywords& keys );
+/// Constructor
+  GridVessel( const VesselOptions& );
+/// Get a description of the grid to output to the log
+ std::string getGridDescription() const ;
+/// Get the indices fof a point
+ void getIndices( const unsigned& index, std::vector<unsigned>& indices ) const ;
+
+/// Operations on one of the elements of grid point i
+ void setGridElement( const unsigned&, const unsigned&, const double& );
+ void addToGridElement( const unsigned&, const unsigned&, const double& );
+
+/// Operations on one of the elements of grid point specified by vector
+ double getGridElement( const std::vector<unsigned>&, const unsigned& ) const ;
+ void setGridElement( const std::vector<unsigned>&, const unsigned&, const double& );
+ void addToGridElement( const std::vector<unsigned>&, const unsigned&, const double& );
+/// Set the size of the buffer equal to nper*npoints
+  virtual void resize();
+/// Get the number of points in the grid
+  unsigned getNumberOfPoints() const;
+/// Get the coordinates for a point in the grid
+  void getGridPointCoordinates( const unsigned& , std::vector<double>& );
+/// Get the dimensionality of the function
+  unsigned getDimension() const ;
+/// Get the number of grid points for each dimension
+  std::vector<unsigned> getNbin() const ;
+/// Get the vector containing the minimum value of the grid in each dimension
+  std::vector<std::string> getMin() const ;
+/// Get the vector containing the maximum value of the grid in each dimension
+  std::vector<std::string> getMax() const ;
+/// Return the volume of one of the grid cells
+  double getCellVolume() const ;
+/// Get the value of the ith grid element 
+  double getGridElement( const unsigned&, const unsigned& ) const ;
+/// Get the numerical index for the box that contains a particular point
+ unsigned getLocationOnGrid( const std::vector<double>& x, std::vector<double>& dd );
+/// Get the points neighboring a particular spline point
+ void getSplineNeighbors( const unsigned& mybox, std::vector<unsigned>& mysneigh );
+/// Get the spacing between grid points
+  const std::vector<double>& getGridSpacing() const ;
+};
+
+inline
+unsigned GridVessel::getNumberOfPoints() const {
+  return npoints;
+}
+
+inline
+const std::vector<double>& GridVessel::getGridSpacing() const {
+  return dx;
+}
+
+inline
+double GridVessel::getCellVolume() const {
+  double myvol=1.0; for(unsigned i=0;i<dimension;++i) myvol *= dx[i];
+  return myvol;
+}
+
+inline
+unsigned GridVessel::getDimension() const {
+  return dimension;
+}
+
+}
+}
+#endif
-- 
GitLab