PSCF v1.2
PolymerTmpl.h
1#ifndef PSCF_POLYMER_TMPL_H
2#define PSCF_POLYMER_TMPL_H
3
4/*
5* PSCF - Polymer Self-Consistent Field Theory
6*
7* Copyright 2016 - 2022, The Regents of the University of Minnesota
8* Distributed under the terms of the GNU General Public License.
9*/
10
11#include <pscf/chem/Species.h> // base class
12#include <util/param/ParamComposite.h> // base class
13
14#include <pscf/chem/Monomer.h> // member template argument
15#include <pscf/chem/Vertex.h> // member template argument
16#include <pscf/chem/PolymerType.h> // member
17#include <util/containers/Pair.h> // member template
18#include <util/containers/DArray.h> // member template
19#include <util/containers/DMatrix.h>
20
21#include <cmath>
22
23namespace Pscf
24{
25
26 class Block;
27 using namespace Util;
28
40 template <class Block>
41 class PolymerTmpl : public Species, public ParamComposite
42 {
43
44 public:
45
46 // Modified diffusion equation solver for one block.
47 typedef typename Block::Propagator Propagator;
48
49 // Monomer concentration field.
50 typedef typename Propagator::CField CField;
51
52 // Chemical potential field.
53 typedef typename Propagator::WField WField;
54
59
64
70 virtual void readParameters(std::istream& in);
71
98 virtual void solve(double phiTot = 1.0);
99
102
108 Block& block(int id);
109
115 Block const& block(int id) const ;
116
124 const Vertex& vertex(int id) const;
125
132 Propagator& propagator(int blockId, int directionId);
133
140 Propagator const & propagator(int blockId, int directionId) const;
141
149 Propagator& propagator(int id);
150
160 const Pair<int>& propagatorId(int id) const;
161
165
169 int nBlock() const;
170
177 int nVertex() const;
178
182 int nPropagator() const; //
183
187 double length() const;
188
192 PolymerType::Enum type() const;
193
195
196 protected:
197
208 virtual void makePlan();
209
210 private:
211
213 DArray<Block> blocks_;
214
216 DArray<Vertex> vertices_;
217
219 DArray< Pair<int> > propagatorIds_;
220
222 int nBlock_;
223
225 int nVertex_;
226
228 int nPropagator_;
229
231 PolymerType::Enum type_;
232
233 };
234
235 /*
236 * Number of vertices (ends and/or junctions)
237 */
238 template <class Block>
240 { return nVertex_; }
241
242 /*
243 * Number of blocks.
244 */
245 template <class Block>
246 inline int PolymerTmpl<Block>::nBlock() const
247 { return nBlock_; }
248
249 /*
250 * Number of propagators.
251 */
252 template <class Block>
254 { return nPropagator_; }
255
256 /*
257 * Total length of all blocks = volume / reference volume
258 */
259 template <class Block>
260 inline double PolymerTmpl<Block>::length() const
261 {
262 double value = 0.0;
263 for (int blockId = 0; blockId < nBlock_; ++blockId) {
264 value += blocks_[blockId].length();
265 }
266 return value;
267 }
268
269 /*
270 * Get a specified Vertex.
271 */
272 template <class Block>
273 inline
275 { return vertices_[id]; }
276
277 /*
278 * Get a specified Block.
279 */
280 template <class Block>
281 inline Block& PolymerTmpl<Block>::block(int id)
282 { return blocks_[id]; }
283
284 /*
285 * Get a specified Block by const reference.
286 */
287 template <class Block>
288 inline Block const & PolymerTmpl<Block>::block(int id) const
289 { return blocks_[id]; }
290
291 /*
292 * Get a propagator id, indexed in order of computation.
293 */
294 template <class Block>
295 inline
297 {
298 UTIL_CHECK(id >= 0);
299 UTIL_CHECK(id < nPropagator_);
300 return propagatorIds_[id];
301 }
302
303 /*
304 * Get a propagator indexed by block and direction.
305 */
306 template <class Block>
307 inline
308 typename Block::Propagator&
309 PolymerTmpl<Block>::propagator(int blockId, int directionId)
310 { return block(blockId).propagator(directionId); }
311
312 /*
313 * Get a const propagator indexed by block and direction.
314 */
315 template <class Block>
316 inline
317 typename Block::Propagator const &
318 PolymerTmpl<Block>::propagator(int blockId, int directionId) const
319 { return block(blockId).propagator(directionId); }
320
321 /*
322 * Get a propagator indexed in order of computation.
323 */
324 template <class Block>
325 inline
326 typename Block::Propagator& PolymerTmpl<Block>::propagator(int id)
327 {
328 Pair<int> propId = propagatorId(id);
329 return propagator(propId[0], propId[1]);
330 }
331
332 /*
333 * Get Polymer type (Branched or Linear).
334 */
335 template <class Block>
336 inline PolymerType::Enum PolymerTmpl<Block>::type() const
337 { return type_; }
338
339 // Non-inline functions
340
341 /*
342 * Constructor.
343 */
344 template <class Block>
346 : Species(),
347 blocks_(),
348 vertices_(),
349 propagatorIds_(),
350 nBlock_(0),
351 nVertex_(0),
352 nPropagator_(0)
353 { setClassName("PolymerTmpl"); }
354
355 /*
356 * Destructor.
357 */
358 template <class Block>
361
362 template <class Block>
364 {
365 // Read polymer type (linear by default)
366 type_ = PolymerType::Linear;
367 readOptional<PolymerType::Enum>(in, "type", type_);
368
369 read<int>(in, "nBlock", nBlock_);
370
371 // Note: For any acyclic graph, nVertex = nBlock + 1
372 nVertex_ = nBlock_ + 1;
373
374 // Allocate all arrays (blocks_, vertices_ and propagatorIds_)
375 blocks_.allocate(nBlock_);
376 vertices_.allocate(nVertex_);
377 propagatorIds_.allocate(2*nBlock_);
378
379 // Set block id and polymerType for all blocks
380 for (int blockId = 0; blockId < nBlock_; ++blockId) {
381 blocks_[blockId].setId(blockId);
382 blocks_[blockId].setPolymerType(type_);
383 }
384
385 // Set all vertex ids
386 for (int vertexId = 0; vertexId < nVertex_; ++vertexId) {
387 vertices_[vertexId].setId(vertexId);
388 }
389
390 // If polymer is linear polymer, set all block vertex Ids:
391 // In a linear chain, block i connects vertex i and vertex i+1.
392 if (type_ == PolymerType::Linear) {
393 for (int blockId = 0; blockId < nBlock_; ++blockId) {
394 blocks_[blockId].setVertexIds(blockId, blockId + 1);
395 }
396 }
397
398 // Read all other required block data
399 readDArray<Block>(in, "blocks", blocks_, nBlock_);
400
401 /*
402 * The parameter file format for each block in the array blocks_
403 * is different for branched and linear polymer. These formats
404 * are defined in >> and << stream io operators for a Pscf::Block.
405 * The choice of format is controlled by the Block::polymerType.
406 *
407 * For a branched polymer, the text format for each block is:
408 *
409 * monomerId length vertexId(0) vertexId(1)
410 *
411 * where monomerId is the index of the block monomer type, length
412 * is the block length, and vertexId(0) and vertexId(1) are the
413 * indices of the two vertices to which the block is attached.
414 *
415 * For a linear polymer, blocks must be entered sequentially, in
416 * their order along the chain, and block vertex id values are
417 * set automatically. In this case, the format for one block is:
418 *
419 * monomerId length
420 *
421 * with no need for explicit vertex ids.
422 */
423
424 // Add blocks to attached vertices
425 int vertexId0, vertexId1;
426 Block* blockPtr;
427 for (int blockId = 0; blockId < nBlock_; ++blockId) {
428 blockPtr = &(blocks_[blockId]);
429 vertexId0 = blockPtr->vertexId(0);
430 vertexId1 = blockPtr->vertexId(1);
431 vertices_[vertexId0].addBlock(*blockPtr);
432 vertices_[vertexId1].addBlock(*blockPtr);
433 }
434
435 // Polymer topology is now fully specified.
436
437 // Construct a plan for the order in which block propagators
438 // should be computed when solving the MDE.
439 makePlan();
440
441 // Read phi or mu (but not both)
442 bool hasPhi = readOptional(in, "phi", phi_).isActive();
443 if (hasPhi) {
444 ensemble_ = Species::Closed;
445 } else {
446 ensemble_ = Species::Open;
447 read(in, "mu", mu_);
448 }
449
450 // Set sources for all propagators
451 Vertex const * vertexPtr = 0;
452 Propagator const * sourcePtr = 0;
453 Propagator * propagatorPtr = 0;
454 Pair<int> propagatorId;
455 int blockId, directionId, vertexId, i;
456 for (blockId = 0; blockId < nBlock(); ++blockId) {
457 // Add sources
458 for (directionId = 0; directionId < 2; ++directionId) {
459 vertexId = block(blockId).vertexId(directionId);
460 vertexPtr = &vertex(vertexId);
461 propagatorPtr = &block(blockId).propagator(directionId);
462 for (i = 0; i < vertexPtr->size(); ++i) {
463 propagatorId = vertexPtr->inPropagatorId(i);
464 if (propagatorId[0] == blockId) {
465 UTIL_CHECK(propagatorId[1] != directionId);
466 } else {
467 sourcePtr =
468 &block(propagatorId[0]).propagator(propagatorId[1]);
469 propagatorPtr->addSource(*sourcePtr);
470 }
471 }
472 }
473
474 }
475
476 }
477
478 /*
479 * Make a plan for the order of computation of block propagators.
480 */
481 template <class Block>
483 {
484 if (nPropagator_ != 0) {
485 UTIL_THROW("nPropagator !=0 on entry");
486 }
487
488 // Allocate and initialize isFinished matrix
489 DMatrix<bool> isFinished;
490 isFinished.allocate(nBlock_, 2);
491 for (int iBlock = 0; iBlock < nBlock_; ++iBlock) {
492 for (int iDirection = 0; iDirection < 2; ++iDirection) {
493 isFinished(iBlock, iDirection) = false;
494 }
495 }
496
497 Pair<int> propagatorId;
498 Vertex* inVertexPtr = 0;
499 int inVertexId = -1;
500 bool isReady;
501 while (nPropagator_ < nBlock_*2) {
502 for (int iBlock = 0; iBlock < nBlock_; ++iBlock) {
503 for (int iDirection = 0; iDirection < 2; ++iDirection) {
504 if (isFinished(iBlock, iDirection) == false) {
505 inVertexId = blocks_[iBlock].vertexId(iDirection);
506 inVertexPtr = &vertices_[inVertexId];
507 isReady = true;
508 for (int j = 0; j < inVertexPtr->size(); ++j) {
509 propagatorId = inVertexPtr->inPropagatorId(j);
510 if (propagatorId[0] != iBlock) {
511 if (!isFinished(propagatorId[0], propagatorId[1])) {
512 isReady = false;
513 break;
514 }
515 }
516 }
517 if (isReady) {
518 propagatorIds_[nPropagator_][0] = iBlock;
519 propagatorIds_[nPropagator_][1] = iDirection;
520 isFinished(iBlock, iDirection) = true;
521 ++nPropagator_;
522 }
523 }
524 }
525 }
526 }
527
528 }
529
530 /*
531 * Compute a solution to the MDE and block concentrations.
532 */
533 template <class Block>
534 void PolymerTmpl<Block>::solve(double phiTot)
535 {
536
537 // Clear all propagators
538 for (int j = 0; j < nPropagator(); ++j) {
539 propagator(j).setIsSolved(false);
540 }
541
542 // Solve modified diffusion equation for all propagators in
543 // the order specified by function makePlan.
544 for (int j = 0; j < nPropagator(); ++j) {
545 UTIL_CHECK(propagator(j).isReady());
546 propagator(j).solve();
547 }
548
549 // Compute molecular partition function q_
550 q_ = block(0).propagator(0).computeQ();
551
552 // The Propagator::computeQ function returns a spatial average.
553 // Correct for partial occupation of the unit cell.
554 q_ = q_/phiTot;
555
556 // Compute mu_ or phi_, depending on ensemble
557 if (ensemble() == Species::Closed) {
558 mu_ = log(phi_/q_);
559 } else
560 if (ensemble() == Species::Open) {
561 phi_ = exp(mu_)*q_;
562 }
563
564 // Compute block concentration fields
565 double prefactor = phi_ / ( q_ * length() );
566 for (int i = 0; i < nBlock(); ++i) {
567 block(i).computeConcentration(prefactor);
568 }
569
570 }
571
572}
573#endif
Descriptor and MDE solver for an acyclic block polymer.
Definition PolymerTmpl.h:42
Block const & block(int id) const
Get a specified Block by const reference.
virtual void readParameters(std::istream &in)
Read and initialize.
virtual void makePlan()
Make a plan for order in which propagators should be computed.
Block & block(int id)
Get a specified Block.
PolymerTmpl()
Constructor.
const Pair< int > & propagatorId(int id) const
Get propagator identifier, indexed by order of computation.
int nBlock() const
Number of blocks.
Propagator & propagator(int blockId, int directionId)
Get propagator for a specific block and direction.
int nPropagator() const
Number of propagators (nBlock*2).
Propagator const & propagator(int blockId, int directionId) const
Get a const propagator for a specific block and direction.
~PolymerTmpl()
Destructor.
Propagator & propagator(int id)
Get propagator indexed in order of computation.
int nVertex() const
Number of vertices (junctions and chain ends).
virtual void solve(double phiTot=1.0)
Solve modified diffusion equation.
double length() const
Sum of the lengths of all blocks in the polymer.
const Vertex & vertex(int id) const
Get a specified Vertex by const reference.
PolymerType::Enum type() const
Get Polymer type (Branched or Linear)
Base class for a molecular species (polymer or solvent).
Definition Species.h:30
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
Dynamically allocatable contiguous array template.
Dynamically allocated Matrix.
Definition DMatrix.h:25
void allocate(int capacity1, int capacity2)
Allocate memory for a matrix.
Definition DMatrix.h:170
An array of exactly 2 objects.
Definition Pair.h:24
An object that can read multiple parameters from file.
void setClassName(const char *className)
Set class name string.
#define UTIL_CHECK(condition)
Assertion macro suitable for serial or parallel production code.
Definition global.h:68
#define UTIL_THROW(msg)
Macro for throwing an Exception, reporting function, file and line number.
Definition global.h:51
PSCF package top-level namespace.
Definition param_pc.dox:1
Utility classes for scientific computation.