Simpatico  v1.10
ClusterIdentifier.cpp
1 #ifndef MCMD_CLUSTER_IDENTIFIER_CPP
2 #define MCMD_CLUSTER_IDENTIFIER_CPP
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 "ClusterIdentifier.h"
12 #include <mcMd/simulation/System.h>
13 #include <mcMd/simulation/Simulation.h>
14 #include <mcMd/chemistry/Molecule.h>
15 #include <mcMd/chemistry/Atom.h>
16 #include <simp/species/Species.h>
17 #include <simp/boundary/Boundary.h>
18 
19 namespace McMd
20 {
21 
22  using namespace Util;
23  using namespace Simp;
24 
25  /*
26  * Constructor.
27  */
29  : links_(),
30  clusters_(),
31  workStack_(),
32  cellList_(),
33  systemPtr_(&system),
34  speciesId_(),
35  atomTypeId_(),
36  cutoff_()
37  {}
38 
39  /*
40  * Destructor.
41  */
43  {}
44 
45  /*
46  * Initial setup.
47  */
48  void
49  ClusterIdentifier::initialize(int speciesId, int atomTypeId, double cutoff)
50  {
51  // Set member variables
52  speciesId_ = speciesId;
53  atomTypeId_ = atomTypeId;
54  cutoff_ = cutoff;
55 
56  // Allocate memory
57  Species* speciesPtr = &system().simulation().species(speciesId);
58  int moleculeCapacity = speciesPtr->capacity();
59  links_.allocate(moleculeCapacity);
60  clusters_.reserve(64);
61  int atomCapacity = system().simulation().atomCapacity();
62  cellList_.setAtomCapacity(atomCapacity);
63 
64  // Note: We must set the cellist atom capacity to the total atom capacity,
65  // even though we are only interested in clusters of one species, because
66  // the celllist atom capacity sets the maximum allowed atom index value.
67  }
68 
69  /*
70  * Pop and process the next molecule on the workStack (private).
71  */
72  void ClusterIdentifier::processNextMolecule(Cluster& cluster)
73  {
74  CellList::NeighborArray neighborArray;
75  Molecule::AtomIterator atomIter;
76  ClusterLink* thisLinkPtr;
77  ClusterLink* otherLinkPtr;
78  Atom* otherAtomPtr;
79  Boundary& boundary = system().boundary();
80  double cutoffSq = cutoff_*cutoff_;
81  double rsq;
82  int thisMolId, otherMolId, otherClusterId;
83 
84  // Pop this molecule off the stack
85  thisLinkPtr = &workStack_.pop();
86  thisMolId = thisLinkPtr->molecule().id();
87  if (thisLinkPtr->clusterId() != cluster.id()) {
88  UTIL_THROW("Top ClusterLink not marked with this cluster id");
89  }
90 
91  /*
92  * Loop over atoms of this molecule.
93  * For each atom of type atomTypeId_, find neighboring molecules.
94  * Add each new neighbor to the cluster, and to the workStack.
95  */
96  thisLinkPtr->molecule().begin(atomIter);
97  for ( ; atomIter.notEnd(); ++atomIter) {
98  if (atomIter->typeId() == atomTypeId_) {
99  cellList_.getNeighbors(atomIter->position(), neighborArray);
100  for (int i = 0; i < neighborArray.size(); i++) {
101  otherAtomPtr = neighborArray[i];
102  otherMolId = otherAtomPtr->molecule().id();
103  if (otherMolId != thisMolId) {
104  rsq = boundary.distanceSq(atomIter->position(),
105  otherAtomPtr->position());
106  if (rsq < cutoffSq) {
107  otherLinkPtr = &(links_[otherMolId]);
108  assert(&otherLinkPtr->molecule() ==
109  &otherAtomPtr->molecule());
110  otherClusterId = otherLinkPtr->clusterId();
111  if (otherClusterId == -1) {
112  cluster.addLink(*otherLinkPtr);
113  workStack_.push(*otherLinkPtr);
114  } else
115  if (otherClusterId != cluster.id()) {
116  UTIL_THROW("Cluster Clash!");
117  }
118  }
119  }
120  } // neighbor atoms
121  }
122  } // atoms
123 
124  }
125 
126  /*
127  * Identify all clusters in the system.
128  */
130  {
131 
132  // Initialize all data structures:
133  // Setup a grid of empty cells
134  cellList_.setup(system().boundary(), cutoff_);
135  // Clear clusters array and all links
136  clusters_.clear();
137  for (int i = 0; i < links_.capacity(); ++i) {
138  links_[i].clear();
139  }
140 
141  // Build the cellList, associate Molecule with ClusterLink.
142  // Iterate over molecules of species speciesId_
143  System::MoleculeIterator molIter;
144  Molecule::AtomIterator atomIter;
145  system().begin(speciesId_, molIter);
146  for ( ; molIter.notEnd(); ++molIter) {
147 
148  // Associate this Molecule with a ClusterLink
149  links_[molIter->id()].setMolecule(*molIter.get());
150 
151  // Add atoms of type = atomTypeId_ to the CellList
152  for (molIter->begin(atomIter); atomIter.notEnd(); ++atomIter) {
153  if (atomIter->typeId() == atomTypeId_) {
154  system().boundary().shift(atomIter->position());
155  cellList_.addAtom(*atomIter);
156  }
157 
158  }
159  }
160 
161  // Identify all clusters
162  Cluster* clusterPtr;
163  ClusterLink* linkPtr;
164  int clusterId = 0;
165  system().begin(speciesId_, molIter);
166  for ( ; molIter.notEnd(); ++molIter) {
167 
168  // Find the link with same index as this molecule
169  linkPtr = &(links_[molIter->id()]);
170  assert (&(linkPtr->molecule()) == molIter.get());
171 
172  // If this link is not in a cluster, begin a new cluster
173  if (linkPtr->clusterId() == -1) {
174 
175  // Add a new empty cluster to clusters_ array
176  clusters_.resize(clusterId+1);
177  clusterPtr = &clusters_[clusterId];
178  clusterPtr->clear();
179  clusterPtr->setId(clusterId);
180 
181  // Identify molecules in this cluster
182  clusterPtr->addLink(*linkPtr);
183  workStack_.push(*linkPtr);
184  while (workStack_.size() > 0) {
185  processNextMolecule(*clusterPtr);
186  }
187 
188  clusterId++;
189  }
190 
191  }
192 
193  // Validity check - throws exception on failure.
194  isValid();
195  }
196 
198  {
199  // Check clusters
200  int nCluster = clusters_.size();
201  int nMolecule = 0;
202  for (int i = 0; i < nCluster; ++i) {
203  if (!clusters_[i].isValid()) {
204  UTIL_THROW("Invalid cluster");
205  }
206  nMolecule += clusters_[i].size();
207  }
208  if (nMolecule != systemPtr_->nMolecule(speciesId_)) {
209  UTIL_THROW("Error in number of molecules");
210  }
211 
212  // Check molecules and links
213  ClusterLink const * linkPtr;
215  systemPtr_->begin(speciesId_, molIter);
216  for ( ; molIter.notEnd(); ++molIter) {
217 
218  linkPtr = &(links_[molIter->id()]);
219  if (&(linkPtr->molecule()) != molIter.get()) {
220  UTIL_THROW("Link without correct molecule association");
221  }
222  if (linkPtr->clusterId() == -1) {
223  UTIL_THROW("Unmarked molecule");
224  }
225  }
226 
227  // Normal return (no errors)
228  return true;
229  }
230 
231 }
232 #endif
Molecule & molecule() const
Get the parent Molecule by reference.
Cluster & cluster(int id)
Get a specific cluster, indexed in the order identified.
int nCluster() const
Get number of clusters.
ClusterIdentifier(System &system)
Constructor.
virtual void initialize(int speciesId, int atomTypeId, double cutoff)
Clear accumulator.
void getNeighbors(const Vector &pos, NeighborArray &neighbors) const
Fill a NeighborArray with pointers to atoms near a specified position.
void begin(int speciesId, MoleculeIterator &iterator)
Initialize an iterator for molecules of one species in this System.
Definition: System.h:1147
double distanceSq(const Vector &r1, const Vector &r2) const
Return square distance between positions r1 and r2.
bool notEnd() const
Is the current pointer not at the end of the array?
Definition: ArrayIterator.h:83
const Data * get() const
Return a pointer to const current data.
An orthorhombic periodic unit cell.
int atomCapacity() const
Get the total number of Atoms allocated.
void begin(AtomIterator &iterator)
Set an Molecule::AtomIterator to first Atom in this Molecule.
A set of interacting Molecules enclosed by a Boundary.
Definition: System.h:115
void clear()
Set cluster to empty.
Definition: Cluster.cpp:25
Forward iterator for a PArray.
Classes used by all simpatico molecular simulations.
int id() const
Get the index for this Molecule (unique in species).
int nMolecule(int speciesId) const
Get the number of molecules of one Species in this System.
Definition: System.h:1128
#define UTIL_THROW(msg)
Macro for throwing an Exception, reporting function, file and line number.
Definition: global.h:51
Simulation & simulation() const
Get the parent Simulation by reference.
Definition: System.h:1055
A point particle within a Molecule.
bool notEnd() const
Is the current pointer not at the end of the PArray?
Data * get() const
Return a pointer to the current data.
Utility classes for scientific computation.
Definition: accumulators.mod:1
void setup(const Boundary &boundary, double cutoff)
Setup grid of empty cells.
int id() const
Get the cluster id.
Definition: Cluster.h:66
void shift(Vector &r) const
Shift Cartesian Vector r to its primary image.
Forward iterator for an Array or a C array.
Definition: ArrayIterator.h:39
Forward iterator for a PArray.
Definition: ArraySet.h:19
void identifyClusters()
Identify all clusters (main operation).
bool notEnd() const
Is the current pointer not at the end of the array?
Boundary & boundary() const
Get the Boundary by reference.
Definition: System.h:1064
Cluster of molecules.
Definition: Cluster.h:31
Single-processor Monte Carlo (MC) and molecular dynamics (MD).
void setId(int id)
Set cluster identifier.
Definition: Cluster.cpp:39
int capacity() const
Maximum allowed number of molecules for this Species.
void setAtomCapacity(int atomCapacity)
Set atom capacity.
const Vector & position() const
Get the position Vector by const reference.
A Species represents a set of chemically similar molecules.
bool isValid() const
Return true if valid, or throw Exception otherwise.
Species & species(int i)
Get a specific Species by reference.
int size() const
Return logical size of this array (i.e., number of elements).
Definition: FSArray.h:207
void addLink(ClusterLink &link)
Add a link to the list.
Definition: Cluster.cpp:52
void addAtom(Atom &atom)
Add a Atom to the appropriate cell, based on its position.