PSCF v1.2
r1d/misc/FieldIo.cpp
1/*
2* PSCF - Polymer Self-Consistent Field Theory
3*
4* Copyright 2016 - 2022, The Regents of the University of Minnesota
5* Distributed under the terms of the GNU General Public License.
6*/
7
8#include "FieldIo.h"
9
10#include <r1d/domain/Domain.h>
11#include <r1d/solvers/Mixture.h>
12#include <r1d/solvers/Polymer.h>
13
14#include <pscf/chem/Vertex.h>
15#include <pscf/homogeneous/Clump.h>
16//#include <pscf/inter/Interaction.h>
17
18#include <util/misc/FileMaster.h>
19#include <util/format/Str.h>
20#include <util/format/Int.h>
21#include <util/format/Dbl.h>
22
23#include <string>
24
25namespace Pscf {
26namespace R1d
27{
28
29 using namespace Util;
30
31 /*
32 * Constructor.
33 */
36
37 /*
38 * Destructor.
39 */
42
43
44 void FieldIo::associate(Domain const & domain,
45 FileMaster const & fileMaster)
46 {
47 domainPtr_ = &domain;
48 fileMasterPtr_ = &fileMaster;
49 }
50
52 std::string const & filename)
53 {
54 std::ifstream in;
55 fileMaster().openInputFile(filename, in);
56 readFields(fields, in);
57 in.close();
58 }
59
60 void FieldIo::readFields(DArray<Field>& fields, std::istream& in)
61 {
62 // Read grid dimensions
63 std::string label;
64 in >> label;
65 UTIL_CHECK(label == "nx");
66 int nx;
67 in >> nx;
68 UTIL_CHECK(nx > 0);
69 UTIL_CHECK(nx == domain().nx());
70 in >> label;
71 UTIL_CHECK (label == "nm");
72 int nm;
73 in >> nm;
74 UTIL_CHECK(nm > 0);
75
76 // Check dimensions of fields array
77 UTIL_CHECK(nm == fields.capacity());
78 for (int i = 0; i < nm; ++i) {
79 UTIL_CHECK(nx == fields[i].capacity());
80 }
81
82 // Read fields
83 int i, j, idum;
84 for (i = 0; i < nx; ++i) {
85 in >> idum;
86 UTIL_CHECK(idum == i);
87 for (j = 0; j < nm; ++j) {
88 in >> fields[j][i];
89 }
90 }
91 }
92
93 void FieldIo::writeField(Field const & field,
94 std::string const & filename, bool writeHeader) const
95 {
96 std::ofstream out;
97 fileMaster().openOutputFile(filename, out);
98 writeField(field, out, writeHeader);
99 out.close();
100 }
101
102 void FieldIo::writeField(Field const & field, std::ostream& out, bool writeHeader) const
103 {
104 int nx = field.capacity();
105 UTIL_CHECK(nx == domain().nx());
106
107 if (writeHeader){
108 out << "nx " << nx << std::endl;
109 }
110 // Write fields
111 int i;
112 for (i = 0; i < nx; ++i) {
113 out << " " << Dbl(field[i], 18, 11);
114 out << std::endl;
115 }
116 }
117
119 std::string const & filename, bool writeHeader)
120 const
121 {
122 std::ofstream out;
123 fileMaster().openOutputFile(filename, out);
124 writeFields(fields, out, writeHeader);
125 out.close();
126 }
127
129 std::ostream& out, bool writeHeader)
130 const
131 {
132 int nm = fields.capacity();
133 UTIL_CHECK(nm > 0);
134 int nx = fields[0].capacity();
135 UTIL_CHECK(nx == domain().nx());
136 if (nm > 1) {
137 for (int i = 0; i < nm; ++i) {
138 UTIL_CHECK(nx == fields[i].capacity());
139 }
140 }
141 if (writeHeader){
142 out << "nx " << nx << std::endl;
143 out << "nm " << nm << std::endl;
144 }
145 // Write fields
146 int i, j;
147 for (i = 0; i < nx; ++i) {
148 out << Int(i, 5);
149 for (j = 0; j < nm; ++j) {
150 out << " " << Dbl(fields[j][i], 18, 11);
151 }
152 out << std::endl;
153 }
154 }
155
156 void FieldIo::writeBlockCFields(Mixture const & mixture,
157 std::string const & filename) const
158 {
159 std::ofstream out;
160 fileMaster().openOutputFile(filename, out);
161 writeBlockCFields(mixture, out);
162 out.close();
163 }
164
165 /*
166 * Write the concentrations associated with all blocks.
167 */
168 void FieldIo::writeBlockCFields(Mixture const & mixture, std::ostream& out)
169 const
170 {
171 int nx = domain().nx(); // number grid points
172 int np = mixture.nPolymer(); // number of polymer species
173 int nb_tot = mixture.nBlock(); // number of blocks in all polymers
174 int ns = mixture.nSolvent(); // number of solvents
175
176 out << "nx " << nx << std::endl;
177 out << "n_block " << nb_tot << std::endl;
178 out << "n_solvent " << ns << std::endl;
179
180 int nb; // number of blocks per polymer
181 int i, j, k, l; // loop indices
182 double c;
183 for (i = 0; i < nx; ++i) {
184 out << Int(i, 5);
185 for (j = 0; j < np; ++j) {
186 nb = mixture.polymer(j).nBlock();
187 for (k = 0; k < nb; ++k) {
188 c = mixture.polymer(j).block(k).cField()[i];
189 out << " " << Dbl(c, 15, 8);
190 }
191 }
192 for (l = 0; l < ns; ++l) {
193 c = mixture.solvent(l).cField()[i];
194 out << " " << Dbl(c, 15, 8);
195 }
196 out << std::endl;
197 }
198 }
199
200 /*
201 * Write incoming q fields for a specified vertex.
202 */
203 void FieldIo::writeVertexQ(Mixture const & mixture,
204 int polymerId, int vertexId,
205 std::string const & filename) const
206 {
207 std::ofstream out;
208 fileMaster().openOutputFile(filename, out);
209 writeVertexQ(mixture, polymerId, vertexId, out);
210 out.close();
211 }
212
213 /*
214 * Write incoming q fields for a specified vertex.
215 */
216 void FieldIo::writeVertexQ(Mixture const & mixture,
217 int polymerId, int vertexId,
218 std::ostream& out) const
219 {
220 UTIL_CHECK(polymerId >= 0);
221 UTIL_CHECK(polymerId < mixture.nPolymer());
222 Polymer const & polymer = mixture.polymer(polymerId);
223 UTIL_CHECK(vertexId >= 0);
224 UTIL_CHECK(vertexId <= polymer.nBlock());
225 Vertex const & vertex = polymer.vertex(vertexId);
226 int nb = vertex.size(); // number of attached blocks
227 int nx = domain().nx(); // number grid points
228
229 Pair<int> pId;
230 int bId;
231 int dId;
232 int i, j;
233 double c, product;
234 for (i = 0; i < nx; ++i) {
235 out << Int(i, 5);
236 product = 1.0;
237 for (j = 0; j < nb; ++j) {
238 pId = vertex.inPropagatorId(j);
239 bId = pId[0]; // blockId
240 UTIL_CHECK(bId >= 0);
241 dId = pId[1]; // directionId
242 UTIL_CHECK(dId >= 0);
243 UTIL_CHECK(dId <= 1);
244 c = polymer.propagator(bId, dId).tail()[i];
245 out << " " << Dbl(c, 15, 8);
246 product *= c;
247 }
248 out << " " << Dbl(product, 15, 8) << std::endl;
249 }
250
251 }
252
253 /*
254 * Interpolate fields onto new mesh, write result to output file
255 */
256 void FieldIo::remesh(DArray<Field> const & fields, int nx,
257 std::string const & filename)
258 {
259 std::ofstream out;
260 fileMaster().openOutputFile(filename, out);
261 remesh(fields, nx, out);
262 out.close();
263 }
264
265 /*
266 * Interpolate fields onto new mesh, write result to outputstream
267 */
268 void
269 FieldIo::remesh(DArray<Field> const & fields, int nx, std::ostream& out)
270 {
271 // Query and check dimensions of fields array
272 int nm = fields.capacity();
273 UTIL_CHECK(nm > 0);
274 for (int i = 0; i < nm; ++i) {
275 UTIL_CHECK(fields[i].capacity() == domain().nx());
276 }
277
278 // Output new grid dimensions
279 out << "nx " << nx << std::endl;
280 out << "nm " << nm << std::endl;
281
282 // Output first grid point
283 int i, j;
284 i = 0;
285 out << Int(i, 5);
286 for (j = 0; j < nm; ++j) {
287 out << " " << Dbl(fields[j][i]);
288 }
289 out << std::endl;
290
291 // Spacing for new grid
292 double dx = (domain().xMax() - domain().xMin())/double(nx-1);
293
294 // Variables used for interpolation
295 double y; // Grid coordinate in old grid
296 double fu; // fractional part of y
297 double fl; // 1.0 - fl
298 double w; // interpolated field value
299 int yi; // Truncated integer part of y
300
301 // Loop over intermediate points
302 for (i = 1; i < nx -1; ++i) {
303 y = dx*double(i)/domain().dx();
304 yi = y;
305 UTIL_CHECK(yi >= 0);
306 UTIL_CHECK(yi + 1 < domain().nx());
307 fu = y - double(yi);
308 fl = 1.0 - fu;
309
310 out << Int(i, 5);
311 for (j = 0; j < nm; ++j) {
312 w = fl*fields[j][yi] + fu*fields[j][yi+1];
313 out << " " << Dbl(w);
314 }
315 out << std::endl;
316 }
317
318 // Output last grid point
319 out << Int(nx - 1, 5);
320 i = domain().nx() - 1;
321 for (j = 0; j < nm; ++j) {
322 out << " " << Dbl(fields[j][i]);
323 }
324 out << std::endl;
325
326 }
327
328 void
329 FieldIo::extend(DArray<Field> const & fields, int m,
330 std::string const & filename)
331 {
332 std::ofstream out;
333 fileMaster().openOutputFile(filename, out);
334 extend(fields, m, out);
335 out.close();
336 }
337
338 /*
339 * Interpolate fields onto new mesh.
340 */
341 void
342 FieldIo::extend(DArray<Field> const & fields, int m, std::ostream& out)
343 {
344 // Query and check dimensions of fields array
345 int nm = fields.capacity();
346 UTIL_CHECK(nm > 0);
347 int nx = fields[0].capacity();
348 UTIL_CHECK(nx == domain().nx());
349 if (nm > 1) {
350 for (int i = 0; i < nm; ++i) {
351 UTIL_CHECK(nx == fields[i].capacity());
352 }
353 }
354
355 // Output new grid dimensions
356 out << "nx " << nx + m << std::endl;
357 out << "nm " << nm << std::endl;
358
359 // Loop over existing points;
360 int i, j;
361 for (i = 0; i < nx; ++i) {
362 out << Int(i, 5);
363 for (j = 0; j < nm; ++j) {
364 out << " " << Dbl(fields[j][i]);
365 }
366 out << std::endl;
367 }
368
369 // Add m new points, assign each the last value
370 for (i = nx; i < nx + m; ++i) {
371 out << Int(i, 5);
372 for (j = 0; j < nm; ++j) {
373 out << " " << Dbl(fields[j][nx-1]);
374 }
375 out << std::endl;
376 }
377
378 }
379
380} // namespace R1d
381} // namespace Pscf
Polymer & polymer(int id)
Get a polymer object.
int nBlock() const
Get number of total blocks in the mixture across all polymers.
int nSolvent() const
Get number of solvent (point particle) species.
Solvent & solvent(int id)
Set a solvent solver object.
int nPolymer() const
Get number of polymer species.
int nBlock() const
Number of blocks.
Propagator & propagator(int blockId, int directionId)
Get propagator for a specific block and direction.
const Vertex & vertex(int id) const
Get a specified Vertex by const reference.
One-dimensional spatial domain and discretization grid.
int nx() const
Get number of spatial grid points, including both endpoints.
double xMin() const
Get minimum spatial coordinate.
double xMax() const
Get maximum spatial coordinate.
double dx() const
Get spatial grid step size.
void writeFields(DArray< Field > const &fields, std::string const &filename, bool writeHeader=true) const
Write a set of fields, one per monomer type, to a named file.
void extend(DArray< Field > const &fields, int m, std::ostream &out)
Add points to the end of a field mesh and write to stream.
void writeField(Field const &field, std::ostream &out, bool writeHeader=true) const
Write a single field to an output stream.
void writeVertexQ(Mixture const &mixture, int polymerId, int vertexId, std::ostream &out) const
Write product of incoming q fields for one vertex to stream.
void associate(Domain const &domain, FileMaster const &fileMaster)
Get and store addresses of associated objects.
void remesh(DArray< Field > const &fields, int nx, std::ostream &out)
Interpolate an array of fields onto a new mesh and write to stream.
void writeField(Field const &field, std::string const &filename, bool writeHeader=true) const
Write a single field to a file.
void writeFields(DArray< Field > const &fields, std::ostream &out, bool writeHeader=true) const
Write a set of fields, one per monomer type, to an output stream.
void writeBlockCFields(Mixture const &mixture, std::ostream &out) const
Write block concentration fields for all blocks to an output stream.
void readFields(DArray< Field > &fields, std::istream &in)
Read a set of fields, one per monomer type.
Mixture of polymers and solvents.
Descriptor and solver for a branched polymer species.
A junction or chain end in a block polymer.
Definition Vertex.h:26
int size() const
Get the number of attached blocks.
Definition Vertex.h:105
Pair< int > const & inPropagatorId(int i) const
Get the block and direction of an incoming propagator.
Definition Vertex.h:109
int capacity() const
Return allocated size.
Definition Array.h:159
Dynamically allocatable contiguous array template.
Wrapper for a double precision number, for formatted ostream output.
Definition Dbl.h:40
A FileMaster manages input and output files for a simulation.
Definition FileMaster.h:143
void openOutputFile(const std::string &filename, std::ofstream &out, std::ios_base::openmode mode=std::ios_base::out) const
Open an output file.
void openInputFile(const std::string &filename, std::ifstream &in, std::ios_base::openmode mode=std::ios_base::in) const
Open an input file.
Wrapper for an int, for formatted ostream output.
Definition Int.h:37
An array of exactly 2 objects.
Definition Pair.h:24
#define UTIL_CHECK(condition)
Assertion macro suitable for serial or parallel production code.
Definition global.h:68
PSCF package top-level namespace.
Definition param_pc.dox:1
Utility classes for scientific computation.
float product(float a, float b)
Product for float Data.
Definition product.h:22