Simpatico  v1.10
DCDTrajectoryReader.cpp
1 /*
2 * Simpatico - Simulation Package for Polymeric and Molecular Liquids
3 *
4 * Copyright 2010 - 2017, The Regents of the University of Minnesota
5 * Distributed under the terms of the GNU General Public License.
6 */
7 
8 #include "DCDTrajectoryReader.h"
9 #include <mcMd/simulation/System.h>
10 #include <mcMd/simulation/Simulation.h>
11 #include <simp/species/Species.h>
12 #include <mcMd/chemistry/Molecule.h>
13 #include <mcMd/chemistry/Atom.h>
14 #include <util/space/Vector.h>
15 #include <util/space/IntVector.h>
16 
17 #include <vector>
18 #include <sstream>
19 
21 #define NFILE_POS 8L
22 #define NATOMS_POS 268L
24 #define FRAMEDATA_POS 276L
26 
27 namespace McMd
28 {
29 
30  using namespace Util;
31 
32  /*
33  * Constructor.
34  */
36  : TrajectoryReader(system),
37  nAtoms_(0),
38  nFrames_(0),
39  frameId_(0)
40  {}
41 
42  /*
43  * Destructor.
44  */
46  {}
47 
48  static unsigned int read_int(std::fstream &file)
49  {
50  unsigned int val;
51  file.read((char *)&val, sizeof(unsigned int));
52  return val;
53  }
54 
55  void DCDTrajectoryReader::open(std::string filename)
56  {
57  // Open trajectory file
58  file_.open(filename.c_str(), std::ios::in | std::ios::binary);
59  if (file_.fail()) {
60  std::string message;
61  message= "Error opening trajectory file. Filename: " + filename;
62  UTIL_THROW(message.c_str());
63  }
64 
65  // Read number of frames
66  file_.seekp(NFILE_POS);
67  nFrames_ = read_int(file_);
68  frameId_ = 0;
69 
70  // Read number of atoms
71  file_.seekp(NATOMS_POS);
72  nAtoms_ = read_int(file_);
73 
74  // Add all molecules and check consistency
75  addMolecules();
76 
77  if (nAtoms_ != nAtomTotal_) {
78  std::ostringstream oss;
79  oss << "Number of atoms in DCD file (" << nAtoms_ << ") does not "
80  << "match allocated number of atoms (" << nAtomTotal_ << ")!";
81  UTIL_THROW(oss.str().c_str());
82  }
83 
84  file_.seekp(FRAMEDATA_POS);
85 
86  xBuffer_.allocate(nAtoms_);
87  yBuffer_.allocate(nAtoms_);
88  zBuffer_.allocate(nAtoms_);
89  }
90 
92  {
93  // Check if the last frame was already read
94  if (frameId_ >= nFrames_) {
95  return false;
96  }
97 
98  double lx, ly, lz;
99  double angle0, angle1, angle2;
100 
101  Vector lengths;
102 
103  // Read frame header
104  int headerSize;
105  headerSize=read_int(file_);
106  if (headerSize != 6*sizeof(double)) {
107  std::ostringstream oss;
108  oss << "Unknown file format!";
109  UTIL_THROW(oss.str().c_str());
110  }
111 
112  file_.read((char *)&lx, sizeof(double));
113  file_.read((char *)&angle0, sizeof(double));
114  file_.read((char *)&ly, sizeof(double));
115  file_.read((char *)&angle1, sizeof(double));
116  file_.read((char *)&angle2, sizeof(double));
117  file_.read((char *)&lz, sizeof(double));
118 
119  headerSize=read_int(file_);
120  if (headerSize != 6*sizeof(double)) {
121  std::ostringstream oss;
122  oss << "Unkown file format!";
123  UTIL_THROW(oss.str().c_str());
124  }
125 
126  if (!file_.good()) {
127  std::ostringstream oss;
128  oss << "Error reading trajectory file!";
129  UTIL_THROW(oss.str().c_str());
130  }
131 
132  lengths[0]=lx;
133  lengths[1]=ly;
134  lengths[2]=lz;
135  boundary().setOrthorhombic(lengths);
136 
137  // Read frame data
138  int blockSize;
139 
140  // read coords
141  blockSize = read_int(file_);
142  file_.read((char *) xBuffer_.cArray(), sizeof(float) * nAtoms_);
143  read_int(file_); // blockSize
144 
145  if (blockSize != (int)sizeof(float)*nAtoms_) {
146  std::ostringstream oss;
147  oss << "Invalid frame size (got " << blockSize
148  << ", expected " << nAtoms_ << ")";
149  UTIL_THROW(oss.str().c_str());
150  }
151 
152  blockSize = read_int(file_);
153  file_.read((char *) yBuffer_.cArray(), sizeof(float) * nAtoms_);
154  read_int(file_); // blockSize
155 
156  if (blockSize != (int)sizeof(float)*nAtoms_) {
157  std::ostringstream oss;
158  oss << "Invalid frame size (got " << blockSize
159  << ", expected " << nAtoms_ << ")";
160  UTIL_THROW(oss.str().c_str());
161  }
162 
163  blockSize = read_int(file_);
164  file_.read((char *) zBuffer_.cArray(), sizeof(float) * nAtoms_);
165  read_int(file_); // blockSize
166 
167  if (blockSize != (int)sizeof(float)* nAtoms_) {
168  std::ostringstream oss;
169  oss << "Invalid frame size (got " << blockSize
170  << ", expected " << nAtoms_ << ")";
171  UTIL_THROW(oss.str().c_str());
172  }
173 
174  // Load positions, assume they are ordered according to species
175  int iSpecies,iMol;
176  int bufferIdx=0;
177  Species *speciesPtr;
178  Molecule::AtomIterator atomIter;
179  Molecule *molPtr;
180 
181  for (iSpecies = 0; iSpecies < simulation().nSpecies(); ++iSpecies) {
182  speciesPtr = &simulation().species(iSpecies);
183  for (iMol = 0; iMol < speciesPtr->capacity(); ++iMol) {
184  molPtr = &system().molecule(iSpecies, iMol);
185  for (molPtr->begin(atomIter); atomIter.notEnd(); ++atomIter) {
186  atomIter->position()[0] = (double) xBuffer_[bufferIdx];
187  atomIter->position()[1] = (double) yBuffer_[bufferIdx];
188  atomIter->position()[2] = (double) zBuffer_[bufferIdx];
189 
190  // shift into simulation cell
191  boundary().shift(atomIter->position());
192 
193  bufferIdx++;
194  }
195  }
196  }
197 
198  // Increment frame counter
199  ++frameId_;
200 
201  // Indicate successful completion
202  return true;
203  }
204 
206  { file_.close(); }
207 
208 }
System & system() const
Get a reference to the parent System.
A Vector is a Cartesian vector.
Definition: Vector.h:75
bool notEnd() const
Is the current pointer not at the end of the array?
Definition: ArrayIterator.h:83
Data * cArray()
Return pointer to underlying C array.
Definition: Array.h:208
void setOrthorhombic(const Vector &lengths)
Set unit cell dimensions for orthorhombic boundary.
void close()
Close trajectory file.
bool readFrame()
Read a single frame.
virtual void addMolecules()
Add all molecules to system.
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
Simulation & simulation() const
Get a reference to the parent Simulation.
Molecule & molecule(int speciesId, int moleculeId)
Get a specific Molecule in this System, by integer index.
Definition: System.h:1137
#define UTIL_THROW(msg)
Macro for throwing an Exception, reporting function, file and line number.
Definition: global.h:51
Utility classes for scientific computation.
Definition: accumulators.mod:1
void shift(Vector &r) const
Shift Cartesian Vector r to its primary image.
DCDTrajectoryReader(System &system)
Constructor.
Trajectory file reader (base class).
Forward iterator for an Array or a C array.
Definition: ArrayIterator.h:39
int nAtomTotal_
Total number of atoms (all species)
Boundary & boundary() const
Get the Boundary.
Single-processor Monte Carlo (MC) and molecular dynamics (MD).
int nSpecies() const
Get the number of Species in this Simulation.
int capacity() const
Maximum allowed number of molecules for this Species.
void allocate(int capacity)
Allocate the underlying C array.
Definition: DArray.h:191
virtual ~DCDTrajectoryReader()
Destructor.
void open(std::string filename)
Read trajectory file header and initialize simulation parameters.
A physical molecule (a set of covalently bonded Atoms).
A Species represents a set of chemically similar molecules.
Species & species(int i)
Get a specific Species by reference.