coords_file.cpp
1 /*
2 Copyright (c) 2015, Florian Sittel (www.lettis.net)
3 All rights reserved.
4 
5 Redistribution and use in source and binary forms, with or without modification,
6 are permitted provided that the following conditions are met:
7 
8 1. Redistributions of source code must retain the above copyright notice,
9  this list of conditions and the following disclaimer.
10 
11 2. Redistributions in binary form must reproduce the above copyright notice,
12  this list of conditions and the following disclaimer in the documentation
13  and/or other materials provided with the distribution.
14 
15 THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY
16 EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
17 OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT
18 SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
19 SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT
20 OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21 HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR
22 TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
23 EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
24 */
25 #include "coords_file.hpp"
26 
27 #include <iostream>
28 
29 namespace CoordsFile {
30 
31 template <typename T>
32 std::vector<T>
33 split(std::string s) {
34  std::vector<T> v;
35  std::istringstream iss(s);
36  while(iss.good()) {
37  T buf;
38  iss >> buf;
39  v.push_back(buf);
40  }
41  return v;
42 }
43 
45 
46 AsciiHandler::AsciiHandler(std::string fname, std::string mode)
47  : _eof(false)
48  , _mode(mode) {
49  if (mode == "r") {
50  this->_ifs.open(fname);
51  } else if (mode == "w") {
52  this->_ofs.open(fname);
53  } else {
54  std::cerr << "unknown mode: " << mode << std::endl;
55  exit(EXIT_FAILURE);
56  }
57 }
58 
59 std::vector<float>
60 AsciiHandler::next() {
61  if (_ifs.is_open() && _ifs.good()) {
62  std::string s;
63  std::getline(_ifs, s);
64  if (_ifs.good()) {
65  if (s == "") {
66  // skip empty lines
67  return this->next();
68  } else {
69  return split<float>(s);
70  }
71  }
72  }
73  _eof = true;
74  return {};
75 }
76 
77 void
78 AsciiHandler::write(std::vector<float> row) {
79  if (_ofs.is_open() && _ofs.good()) {
80  for (float f: row) {
81  _ofs << " " << f;
82  }
83  _ofs << std::endl;
84  }
85 }
86 
87 bool
88 AsciiHandler::eof() {
89  return _eof;
90 }
91 
92 
94 
95 XtcHandler::XtcHandler(std::string fname, std::string mode)
96  : _eof(false)
97  , _mode(mode)
98  , _nrow(0) {
99  if (_mode == "r") {
100  read_xtc_natoms(fname.c_str(), &_natoms);
101  _coord_buf = static_cast<rvec*>(calloc(_natoms, sizeof(_coord_buf[0])));
102  }
103  _xdr = xdrfile_open(fname.c_str(), mode.c_str());
104 }
105 
106 XtcHandler::~XtcHandler() {
107  xdrfile_close(_xdr);
108  if (_mode == "r") {
109  free(_coord_buf);
110  }
111 }
112 
113 std::vector<float>
114 XtcHandler::next() {
115  if (_mode == "r") {
116  int step;
117  float time_step;
118  float prec;
119  matrix box;
120  int err = read_xtc(_xdr, _natoms, &step, &time_step, box, _coord_buf, &prec);
121  if (err == exdrOK) {
122  std::vector<float> v(_natoms*3);
123  for (int i=0; i < _natoms; ++i) {
124  v[3*i] = _coord_buf[i][0];
125  v[3*i+1] = _coord_buf[i][1];
126  v[3*i+2] = _coord_buf[i][2];
127  }
128  return v;
129  }
130  }
131  _eof = true;
132  return {};
133 }
134 
135 void
136 XtcHandler::write(std::vector<float> row) {
137  if (_mode == "w") {
138  float fake_box_matrix[3][3] = {{0,0,0}, {0,0,0}, {0,0,0}};
139  int natoms = row.size() / 3;
140  rvec* x = static_cast<rvec*>(calloc(natoms, sizeof(rvec)));
141  for (int i=0; i < natoms; ++i) {
142  x[i][0] = row[3*i];
143  x[i][1] = row[3*i+1];
144  x[i][2] = row[3*i+2];
145  }
146  write_xtc(_xdr, natoms, _nrow, _nrow*1.0f, fake_box_matrix, x, 1000.0f);
147  free(x);
148  ++_nrow;
149  }
150 }
151 
152 bool
153 XtcHandler::eof() {
154  return _eof;
155 }
156 
157 
159 
160 FilePointer
161 open(std::string fname, std::string mode) {
162  if ((fname.size() > 4)
163  && (fname.compare(fname.size()-4, 4, ".xtc") == 0)) {
164  return FilePointer(new XtcHandler(fname, mode));
165  } else {
166  return FilePointer(new AsciiHandler(fname, mode));
167  }
168 }
169 
170 } // end namespace 'CoordsFile'
171