PSCF v1.2
rpc/solvers/Mixture.tpp
1#ifndef RPC_MIXTURE_TPP
2#define RPC_MIXTURE_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 "Mixture.h"
12#include <prdc/cpu/RField.h>
13#include <pscf/mesh/Mesh.h>
14
15#include <cmath>
16
17namespace Pscf {
18namespace Rpc
19{
20
21 using namespace Prdc::Cpu;
22
23 template <int D>
25 : stress_(),
26 ds_(-1.0),
27 meshPtr_(nullptr),
28 nParam_(0),
29 hasStress_(false)
30 { setClassName("Mixture"); }
31
32 template <int D>
35
36 template <int D>
37 void Mixture<D>::readParameters(std::istream& in)
38 {
39 MixtureTmpl< Polymer<D>, Solvent<D> >::readParameters(in);
40 read(in, "ds", ds_);
41
42 UTIL_CHECK(nMonomer() > 0);
43 UTIL_CHECK(nPolymer()+ nSolvent() > 0);
44 UTIL_CHECK(ds_ > 0);
45 }
46
47 /*
48 * Create associations with mesh, fft, and unit cell.
49 */
50 template <int D>
51 void Mixture<D>::associate(Mesh<D> const & mesh,
52 FFT<D> const & fft,
53 UnitCell<D> const & cell)
54 {
55 UTIL_CHECK(nMonomer() > 0);
56 UTIL_CHECK(nPolymer()+ nSolvent() > 0);
57 UTIL_CHECK(mesh.size() > 0);
58 UTIL_CHECK(fft.isSetup());
59 UTIL_CHECK(mesh.dimensions() == fft.meshDimensions());
60 UTIL_CHECK(cell.nParameter() > 0);
61
62 // Save addresses of mesh and unit cell
63 meshPtr_ = &mesh;
64 nParam_ = cell.nParameter();
65
66 // Create associations for all polymer blocks
67 if (nPolymer() > 0) {
68 int i, j;
69 for (i = 0; i < nPolymer(); ++i) {
70 polymer(i).setNParams(nParam_);
71 for (j = 0; j < polymer(i).nBlock(); ++j) {
72 polymer(i).block(j).associate(mesh, fft, cell);
73 }
74 }
75 }
76
77 // Create associations for all solvents (if any)
78 if (nSolvent() > 0) {
79 for (int i = 0; i < nSolvent(); ++i) {
80 solvent(i).associate(mesh);
81 }
82 }
83
84 }
85
86
87 /*
88 * Allocate internal data containers for all solvers.
89 */
90 template <int D>
92 {
93 UTIL_CHECK(meshPtr_);
94 UTIL_CHECK(nMonomer() > 0);
95 UTIL_CHECK(nPolymer()+ nSolvent() > 0);
96 UTIL_CHECK(meshPtr_->size() > 0);
97 UTIL_CHECK(ds_ > 0);
98
99 // Allocate memory for all Block objects
100 if (nPolymer() > 0) {
101 int i, j;
102 for (i = 0; i < nPolymer(); ++i) {
103 for (j = 0; j < polymer(i).nBlock(); ++j) {
104 polymer(i).block(j).allocate(ds_);
105 }
106 }
107 }
108
109 // Set spatial discretization for solvents
110 if (nSolvent() > 0) {
111 for (int i = 0; i < nSolvent(); ++i) {
112 solvent(i).allocate();
113 }
114 }
115
116 }
117
118 /*
119 * Clear all data that depends on the unit cell parameters.
120 */
121 template <int D>
123 {
124 if (nPolymer() > 0) {
125 for (int i = 0; i < nPolymer(); ++i) {
126 polymer(i).clearUnitCellData();
127 }
128 }
129 hasStress_ = false;
130 }
131
132 /*
133 * Reset statistical segment length for one monomer type.
134 */
135 template <int D>
136 void Mixture<D>::setKuhn(int monomerId, double kuhn)
137 {
138 // Set new Kuhn length for relevant Monomer object
139 monomer(monomerId).setKuhn(kuhn);
140
141 // Update kuhn length for all blocks of this monomer type
142 for (int i = 0; i < nPolymer(); ++i) {
143 for (int j = 0; j < polymer(i).nBlock(); ++j) {
144 Block<D>& block = polymer(i).block(j);
145 if (monomerId == block.monomerId()) {
146 block.setKuhn(kuhn);
147 }
148 }
149 }
150 hasStress_ = false;
151 }
152
153 /*
154 * Compute concentrations (but not total free energy).
155 */
156 template <int D>
157 void Mixture<D>::compute(DArray< RField<D> > const & wFields,
158 DArray< RField<D> > & cFields,
159 double phiTot)
160 {
161 UTIL_CHECK(meshPtr_);
162 UTIL_CHECK(mesh().size() > 0);
163 UTIL_CHECK(nMonomer() > 0);
164 UTIL_CHECK(nPolymer() + nSolvent() > 0);
165 UTIL_CHECK(wFields.capacity() == nMonomer());
166 UTIL_CHECK(cFields.capacity() == nMonomer());
167
168 int meshSize = mesh().size();
169 int nm = nMonomer();
170 int i, j, k;
171
172 // Clear all monomer concentration fields, check capacities
173 for (i = 0; i < nm; ++i) {
174 UTIL_CHECK(cFields[i].capacity() == meshSize);
175 UTIL_CHECK(wFields[i].capacity() == meshSize);
176 for (j = 0; j < meshSize; ++j) {
177 cFields[i][j] = 0.0;
178 }
179 }
180
181 // Process polymer species
182 // Solve MDE for all polymers
183 for (i = 0; i < nPolymer(); ++i) {
184 polymer(i).compute(wFields, phiTot);
185 }
186
187 // Accumulate block contributions to monomer concentrations
188 int monomerId;
189 for (i = 0; i < nPolymer(); ++i) {
190 for (j = 0; j < polymer(i).nBlock(); ++j) {
191 monomerId = polymer(i).block(j).monomerId();
192 UTIL_CHECK(monomerId >= 0);
193 UTIL_CHECK(monomerId < nm);
194 RField<D>& monomerField = cFields[monomerId];
195 RField<D> const & blockField = polymer(i).block(j).cField();
196 UTIL_CHECK(blockField.capacity() == meshSize);
197 for (k = 0; k < meshSize; ++k) {
198 monomerField[k] += blockField[k];
199 }
200 }
201 }
202
203 // Process solvent species
204 // For each solvent, call compute and accumulate cFields
205 for (i = 0; i < nSolvent(); ++i) {
206 monomerId = solvent(i).monomerId();
207 UTIL_CHECK(monomerId >= 0);
208 UTIL_CHECK(monomerId < nm);
209
210 // Compute solvent concentration
211 solvent(i).compute(wFields[monomerId], phiTot);
212
213 // Add solvent contribution to relevant monomer concentration
214 RField<D>& monomerField = cFields[monomerId];
215 RField<D> const & solventField = solvent(i).cField();
216 UTIL_CHECK(solventField.capacity() == meshSize);
217 for (k = 0; k < meshSize; ++k) {
218 monomerField[k] += solventField[k];
219 }
220
221 }
222
223 hasStress_ = false;
224 }
225
226 /*
227 * Compute total stress for this mixture.
228 */
229 template <int D>
230 void Mixture<D>::computeStress(double phiTot)
231 {
232 int i, j;
233
234 // Initialize all stress components to zero
235 for (i = 0; i < 6; ++i) {
236 stress_[i] = 0.0;
237 }
238
239 if (nPolymer() > 0) {
240
241 // Compute stress for all polymers, after solving MDE
242 for (i = 0; i < nPolymer(); ++i) {
243 polymer(i).computeStress();
244 }
245
246 // Accumulate stress for all the polymer chains
247 for (i = 0; i < nParam_; ++i) {
248 for (j = 0; j < nPolymer(); ++j) {
249 stress_[i] += polymer(j).stress(i);
250 }
251 }
252
253 }
254
255 // Correct for possible partial occupation of the unit cell.
256 // Used in problems that contain a Mask, e.g., thin films.
257 for (i = 0; i < nParam_; ++i) {
258 stress_[i] /= phiTot;
259 }
260
261 // Note: Solvent does not contribute to derivatives of f_Helmholtz
262 // with respect to unit cell parameters at fixed volume fractions.
263
264 hasStress_ = true;
265 }
266
267 template <int D>
269 {
270 // Check ensemble of all polymers
271 for (int i = 0; i < nPolymer(); ++i) {
272 if (polymer(i).ensemble() == Species::Open) {
273 return false;
274 }
275 }
276 // Check ensemble of all solvents
277 for (int i = 0; i < nSolvent(); ++i) {
278 if (solvent(i).ensemble() == Species::Open) {
279 return false;
280 }
281 }
282 // Returns true if false was never returned
283 return true;
284 }
285
286 /*
287 * Combine cFields for all blocks and solvents into one DArray
288 */
289 template <int D>
290 void
292 const
293 {
294 int np = nSolvent() + nBlock();
295 UTIL_CHECK(np > 0);
296 int nx = mesh().size();
297 UTIL_CHECK(nx > 0);
298 int i, j;
299
300 // Check allocation of blockCFields, allocate if necessary
301 // Initialize all concentration values to zero
302 if (!blockCFields.isAllocated()) {
303 blockCFields.allocate(np);
304 }
305 UTIL_CHECK(blockCFields.capacity() == np);
306 for (i = 0; i < np; ++i) {
307 if (!blockCFields[i].isAllocated()) {
308 blockCFields[i].allocate(mesh().dimensions());
309 }
310 UTIL_CHECK(blockCFields[i].capacity() == nx);
311 for (j = 0; j < nx; ++j) {
312 blockCFields[i][j] = 0.0;
313 }
314 }
315
316 // Initialize section (block or solvent) index
317 int sectionId = 0;
318
319 // Process polymer species
320 if (nPolymer() > 0) {
321 for (i = 0; i < nPolymer(); ++i) {
322 for (j = 0; j < polymer(i).nBlock(); ++j) {
323 UTIL_CHECK(sectionId >= 0);
324 UTIL_CHECK(sectionId < np);
325 // Copy block r-grid c-field to blockCFields
326 blockCFields[sectionId] = polymer(i).block(j).cField();
327 sectionId++;
328 }
329 }
330 }
331 UTIL_CHECK(sectionId == nBlock());
332
333 // Process solvent species
334 if (nSolvent() > 0) {
335 for (i = 0; i < nSolvent(); ++i) {
336 UTIL_CHECK(sectionId >= 0);
337 UTIL_CHECK(sectionId < np);
338 // Copy solvent r-grid c-field to blockCFields
339 blockCFields[sectionId] = solvent(i).cField();
340 sectionId++;
341 }
342 }
343 UTIL_CHECK(sectionId == np);
344
345 }
346
347} // namespace Rpc
348} // namespace Pscf
349#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
int size() const
Get total number of grid points.
Definition Mesh.h:229
A mixture of polymer and solvent species.
Definition MixtureTmpl.h:27
Fourier transform wrapper.
bool isSetup() const
Has this FFT object been setup?
Definition cpu/FFT.h:210
IntVec< D > const & meshDimensions() const
Return the dimensions of the grid for which this was setup.
Definition cpu/FFT.h:217
Field of real double precision values on an FFT mesh.
int nParameter() const
Get the number of parameters in the unit cell.
Base template for UnitCell<D> classes, D=1, 2 or 3.
Definition rpg/System.h:34
Block within a branched polymer.
void setKuhn(double kuhn)
Set or reset monomer statistical segment length.
int monomerId() const
Get the monomer type id.
void associate(Mesh< D > const &mesh, FFT< D > const &fft, UnitCell< D > const &cell)
Create an association with a mesh and fft, and allocate memory.
bool isCanonical()
Is this mixture being treated in canonical ensemble?
void compute(DArray< RField< D > > const &wFields, DArray< RField< D > > &cFields, double phiTot=1.0)
Compute partition functions and concentrations.
void allocate()
Allocate required internal memory for all solvers.
void readParameters(std::istream &in)
Read all parameters and initialize.
void setKuhn(int monomerId, double kuhn)
Reset statistical segment length for one monomer type.
void computeStress(double phiTot=1.0)
Compute derivatives of free energy w/ respect to cell parameters.
void createBlockCRGrid(DArray< RField< D > > &blockCFields) const
Get c-fields for all blocks and solvents as array of r-grid fields.
void clearUnitCellData()
Clear all data in solvers that depends on the unit cell parameters.
int capacity() const
Return allocated size.
Definition Array.h:159
Dynamically allocatable contiguous array template.
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
PSCF package top-level namespace.
Definition param_pc.dox:1