PSCF v1.2
rpc/solvers/Propagator.tpp
1#ifndef RPC_PROPAGATOR_TPP
2#define RPC_PROPAGATOR_TPP
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 "Propagator.h"
12#include "Block.h"
13
14#include <pscf/mesh/Mesh.h>
15
16namespace Pscf {
17namespace Rpc {
18
19 using namespace Util;
20
21 /*
22 * Constructor.
23 */
24 template <int D>
26 : blockPtr_(0),
27 meshPtr_(0),
28 ns_(0),
29 isAllocated_(false)
30 {}
31
32 /*
33 * Destructor.
34 */
35 template <int D>
38
39 /*
40 * Allocate memory used by this propagator.
41 */
42 template <int D>
43 void Propagator<D>::allocate(int ns, const Mesh<D>& mesh)
44 {
45 UTIL_CHECK(!isAllocated_);
46 ns_ = ns;
47 meshPtr_ = &mesh;
48
49 qFields_.allocate(ns);
50 for (int i = 0; i < ns; ++i) {
51 qFields_[i].allocate(mesh.dimensions());
52 }
53 isAllocated_ = true;
54 }
55
56 /*
57 * Reallocate memory used by this propagator using new ns value.
58 */
59 template <int D>
61 {
62 UTIL_CHECK(isAllocated_);
63 UTIL_CHECK(ns_ != ns);
64 ns_ = ns;
65
66 // Deallocate all memory previously used by this propagator.
67 qFields_.deallocate();
68
69 // NOTE: The qFields_ container is a DArray<QField>, where QField
70 // is a typedef for DFields<D>. The DArray::deallocate() function
71 // calls "delete [] ptr", where ptr is a pointer to the underlying
72 // C array. The C++ delete [] command calls the destructor for
73 // each element in an array before deleting the array itself.
74 // The RField<D> destructor deletes the the double* array that
75 // stores the field associated with each slice of the propagator.
76
77 // Allocate new memory for qFields_ using new value of ns
78 qFields_.allocate(ns);
79 for (int i = 0; i < ns; ++i) {
80 qFields_[i].allocate(meshPtr_->dimensions());
81 }
82
83 setIsSolved(false);
84 }
85
86 /*
87 * Compute initial head QField from final tail QFields of sources.
88 */
89 template <int D>
91 {
92
93 // Reference to head of this propagator
94 QField& qh = qFields_[0];
95
96 // Initialize qh field to 1.0 at all grid points
97 int ix;
98 int nx = meshPtr_->size();
99 for (ix = 0; ix < nx; ++ix) {
100 qh[ix] = 1.0;
101 }
102
103 // Pointwise multiply tail QFields of all sources
104 for (int is = 0; is < nSource(); ++is) {
105 if (!source(is).isSolved()) {
106 UTIL_THROW("Source not solved in computeHead");
107 }
108 QField const& qt = source(is).tail();
109 for (ix = 0; ix < nx; ++ix) {
110 qh[ix] *= qt[ix];
111 }
112 }
113 }
114
115 /*
116 * Solve the modified diffusion equation for this block.
117 */
118 template <int D>
120 {
121 UTIL_CHECK(isAllocated());
122 computeHead();
123 for (int iStep = 0; iStep < ns_ - 1; ++iStep) {
124 block().step(qFields_[iStep], qFields_[iStep + 1]);
125 }
126 setIsSolved(true);
127 }
128
129 /*
130 * Solve the modified diffusion equation with a specified initial condition.
131 */
132 template <int D>
133 void Propagator<D>::solve(QField const & head)
134 {
135 int nx = meshPtr_->size();
136 UTIL_CHECK(head.capacity() == nx);
137
138 // Initialize initial (head) field
139 QField& qh = qFields_[0];
140 for (int i = 0; i < nx; ++i) {
141 qh[i] = head[i];
142 }
143
144 // Setup solver and solve
145 for (int iStep = 0; iStep < ns_ - 1; ++iStep) {
146 block().step(qFields_[iStep], qFields_[iStep + 1]);
147 }
148 setIsSolved(true);
149 }
150
151 /*
152 * Compute spatial average of product of head and tail of partner.
153 */
154 template <int D>
156 {
157 // Preconditions
158 if (!isSolved()) {
159 UTIL_THROW("Propagator is not solved.");
160 }
161 if (!hasPartner()) {
162 UTIL_THROW("Propagator has no partner set.");
163 }
164 if (!partner().isSolved()) {
165 UTIL_THROW("Partner propagator is not solved");
166 }
167 QField const& qh = head();
168 QField const& qt = partner().tail();
169 int nx = meshPtr_->size();
170 UTIL_CHECK(qt.capacity() == nx);
171 UTIL_CHECK(qh.capacity() == nx);
172
173 // Take inner product of head and partner tail fields
174 double Q = 0;
175 for (int i =0; i < nx; ++i) {
176 Q += qh[i]*qt[i];
177 }
178 Q /= double(nx);
179 return Q;
180 }
181
182}
183}
184#endif
Description of a regular grid of points in a periodic domain.
IntVec< D > dimensions() const
Get an IntVec<D> of the grid dimensions.
Definition Mesh.h:217
Field of real double precision values on an FFT mesh.
void solve()
Solve the modified diffusion equation (MDE) for this block.
void allocate(int ns, const Mesh< D > &mesh)
Allocate memory used by this propagator.
void reallocate(int ns)
Reallocate memory used by this propagator.
void computeHead()
Compute initial QField at head from tail QFields of sources.
double computeQ()
Compute and return partition function for the polymer.
int capacity() const
Return allocated size.
Definition Array.h:159
#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.