Simpatico  v1.10
tools/neighbor/CellList.h
1 #ifndef TOOLS_CELL_LIST_H
2 #define TOOLS_CELL_LIST_H
3 
4 /*
5 * Simpatico - Simulation Package for Polymeric and Molecular Liquids
6 *
7 * Copyright 2010 - 2017, The Regents of the University of Minnesota
8 * Distributed under the terms of the GNU General Public License.
9 */
10 
11 #include <tools/neighbor/Cell.h>
12 #include <tools/neighbor/CellAtom.h>
13 #include <tools/chemistry/Atom.h>
14 #include <simp/boundary/Boundary.h>
15 #include <util/space/Grid.h>
16 #include <util/containers/DArray.h>
17 #include <util/containers/GArray.h>
18 #include <util/containers/FArray.h>
19 #include <util/global.h>
20 
21 namespace Tools
22 {
23 
24  using namespace Util;
25  using namespace Simp;
26 
83  class CellList
84  {
85 
86  public:
87 
88  // Public methods
89 
93  CellList();
94 
98  virtual ~CellList();
99 
124  void allocate(int atomCapacity, const Vector& lower, const Vector& upper,
125  const Vector& cutoffs);
126 
140  void allocate(int atomCapacity, const Vector& lower, const Vector& upper,
141  double cutoff);
142 
164  void
165  makeGrid(const Vector& lower, const Vector& upper, const Vector& cutoffs);
166 
178  void placeAtom(Atom &atom);
179 
188  void build();
189 
196  void update();
197 
201  void clear();
202 
206  const Grid& grid() const;
207 
213  double cellLength(int i) const;
214 
223  int cellIndexFromPosition(const Vector& position) const;
224 
228  const Cell* begin() const;
229 
235  const Cell& cell(int i) const;
236 
240  int nAtom() const;
241 
245  int nReject() const;
246 
247  #ifdef UTIL_DEBUG
248 
251  int maxNAtomCell() const;
252  #endif
253 
257  int atomCapacity() const;
258 
262  int cellCapacity() const;
263 
267  bool isAllocated() const;
268 
274  bool isBuilt() const;
275 
281  bool isValid() const;
282 
283  private:
284 
285  /*
286  * Temporary storage for atom pointers, before copying to cells.
287  */
288  struct Tag {
289  Atom* ptr;
290  int cellRank;
291  };
292 
294  Grid grid_;
295 
297  DArray<Tag> tags_;
298 
300  DArray<CellAtom> atoms_;
301 
303  GArray<Cell> cells_;
304 
306  Vector lower_;
307 
309  Vector upper_;
310 
312  Vector cellLengths_;
313 
315  Cell* begin_;
316 
318  int nAtom_;
319 
321  int nReject_;
322 
323  #ifdef UTIL_DEBUG
324  int maxNAtomCell_;
326  #endif
327 
329  bool isBuilt_;
330 
332  int gridOffsets_ [3];
333 
345  void setGridDimensions(const Vector& lower, const Vector& upper,
346  const Vector& cutoffs);
347 
351  bool isValidAtomId(int atomId);
352 
362  int calculateAxisOffset(int cellId, int i, int x) const;
363  };
364 
365  // Private inline method definitions:
366 
367  /*
368  * Return value of offset for cell a distance x along axis i shifted to the range
369  * 0 <= x <= grid_.Dimension(i)
370  */
371  inline int CellList::calculateAxisOffset(int cellId, int i, int x) const
372  {
373  // mod the shift so it's within the grid dimension
374  x %= grid_.dimension(i);
375 
376  // calculate cell's position within the grid
377  int p[3]; int j;
378  for (j = 0; j < Dimension -1; ++j) {
379  p[j] = cellId/gridOffsets_[j];
380  cellId -= p[j]*gridOffsets_[j];
381  }
382  p[j] = cellId;
383 
384  if (p[i] + x >= grid_.dimension(i)) return (x - grid_.dimension(i))*gridOffsets_[i];
385  if (p[i] + x < 0) return (grid_.dimension(i) + x)*gridOffsets_[i];
386  return x*gridOffsets_[i];
387  }
388 
389  // Public inline method definitions:
390 
391  /*
392  * Identify the cell within which a position lies.
393  */
394  inline int CellList::cellIndexFromPosition(const Vector& position) const
395  {
396  IntVector r;
397  for (int i = 0; i < Dimension; ++i) {
398  if (position[i] <= lower_[i]) {
399  return -1;
400  }
401  if (position[i] >= upper_[i]) {
402  return -1;
403  }
404  r[i] = int( (position[i] - lower_[i]) / cellLengths_[i] );
405  assert(r[i] < grid_.dimension(i));
406  }
407  return grid_.rank(r);
408  }
409 
410  /*
411  * Add an atom to the appropriate cell, based on its position.
412  */
413  inline void CellList::placeAtom(Atom &atom)
414  {
415  // Preconditon
416  assert(nAtom_ < tags_.capacity());
417 
418  int rank = cellIndexFromPosition(atom.position);
419  if (rank >= 0) {
420  tags_[nAtom_].cellRank = rank;
421  tags_[nAtom_].ptr = &atom;
422  cells_[rank].incrementCapacity();
423  ++nAtom_;
424  } else {
425  ++nReject_;
426  }
427  }
428 
429  /*
430  * Return true iff atomId is valid, i.e., if 0 <= 0 < atomCapacity.
431  */
432  inline bool CellList::isValidAtomId(int atomId)
433  { return ( (0 <= atomId) && (atomId < tags_.capacity()) ); }
434 
435  /*
436  * Return associated Grid object.
437  */
438  inline const Grid& CellList::grid() const
439  { return grid_; }
440 
441  /*
442  * Return length of each cell in direction i.
443  */
444  inline double CellList::cellLength(int i) const
445  { return cellLengths_[i]; }
446 
447  /*
448  * Return reference to cell number i.
449  */
450  inline const Cell& CellList::cell(int i) const
451  { return cells_[i]; }
452 
453  /*
454  * Return pointer to first Cell.
455  */
456  inline const Cell* CellList::begin() const
457  { return begin_; }
458 
459  /*
460  * Is this CellList allocated?
461  */
462  inline bool CellList::isAllocated() const
463  { return (cells_.capacity() > 0); }
464 
465 }
466 #endif
const Cell & cell(int i) const
Return a specified cell by const reference.
const int Dimension
Dimensionality of space.
Definition: Dimension.h:19
An automatically growable array, analogous to a std::vector.
Definition: GArray.h:33
A Vector is a Cartesian vector.
Definition: Vector.h:75
A cell list used only to identify nearby atom pairs.
double cellLength(int i) const
Dimension of each cell in direction i.
File containing preprocessor macros for error handling.
Classes used by all simpatico molecular simulations.
bool isAllocated() const
Has memory been allocated for this CellList?
Utility classes for scientific computation.
Definition: accumulators.mod:1
Single-processor classes for pre- and post-processing MD trajectories.
A point particle in an MD simulation.
const Grid & grid() const
Return Grid object by const reference.
Vector position
Atom position.
An IntVector is an integer Cartesian vector.
Definition: IntVector.h:73
int cellIndexFromPosition(const Vector &position) const
Return the index of the cell that contains a position Vector.
void placeAtom(Atom &atom)
Determine the appropriate cell for an Atom, based on its position.
A grid of points indexed by integer coordinates.
Definition: Grid.h:33
A single cell in a CellList.
const Cell * begin() const
Return pointer to first local cell in linked list.