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