PSCF v1.2
rpg/solvers/Mixture.tpp
1#ifndef RPG_MIXTURE_TPP
2#define RPG_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/cuda/resources.h>
13
14#include <cmath>
15
16namespace Pscf {
17namespace Rpg
18{
19
20 /*
21 * Constructor.
22 */
23 template <int D>
25 : stress_(),
26 ds_(-1.0),
27 meshPtr_(nullptr),
28 nParams_(0),
29 hasStress_(false),
30 useBatchedFFT_(true)
31 { setClassName("Mixture"); }
32
33 // Destructor
34 template <int D>
37
38 /*
39 * Read all parameters and initialize.
40 */
41 template <int D>
42 void Mixture<D>::readParameters(std::istream& in)
43 {
44 MixtureTmpl< Polymer<D>, Solvent<D> >::readParameters(in);
45 read(in, "ds", ds_);
46 readOptional(in, "useBatchedFFT", useBatchedFFT_);
47
48 UTIL_CHECK(nMonomer() > 0);
49 UTIL_CHECK(nPolymer()+ nSolvent() > 0);
50 UTIL_CHECK(ds_ > 0);
51 }
52
53 /*
54 * Create associations with a mesh, FFT, UnitCell, and WaveList object.
55 */
56 template <int D>
57 void Mixture<D>::associate(Mesh<D> const & mesh, FFT<D> const & fft,
58 UnitCell<D> const & cell, WaveList<D>& wavelist)
59 {
60 UTIL_CHECK(nMonomer() > 0);
61 UTIL_CHECK(nPolymer() + nSolvent() > 0);
62 UTIL_CHECK(mesh.size() > 0);
63 UTIL_CHECK(fft.isSetup());
64 UTIL_CHECK(fft.meshDimensions() == mesh.dimensions());
65 UTIL_CHECK(cell.nParameter() > 0);
66
67 // Assign internal pointers and variables
68 meshPtr_ = &mesh;
69 nParams_ = cell.nParameter();
70
71 // Create associations for all blocks, set nParams in Polymer objects
72 int i, j;
73 for (i = 0; i < nPolymer(); ++i) {
74 polymer(i).setNParams(nParams_);
75 for (j = 0; j < polymer(i).nBlock(); ++j) {
76 polymer(i).block(j).associate(mesh, fft, cell, wavelist);
77 }
78 }
79
80 // Create associations for all solvents
81 if (nSolvent() > 0) {
82 for (int i = 0; i < nSolvent(); ++i) {
83 solvent(i).associate(mesh);
84 }
85 }
86
87 }
88
89 /*
90 * Allocate internal data containers in all solvers.
91 */
92 template <int D>
94 {
95 UTIL_CHECK(nMonomer() > 0);
96 UTIL_CHECK(nPolymer()+ nSolvent() > 0);
97 UTIL_CHECK(ds_ > 0);
98 UTIL_CHECK(mesh().size() > 0);
99
100 // Allocate memory in all Block objects
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_, useBatchedFFT_);
105 }
106 }
107
108 // Allocate memory in all Solvent objects
109 if (nSolvent() > 0) {
110 for (int i = 0; i < nSolvent(); ++i) {
111 solvent(i).allocate();
112 }
113 }
114 }
115
116 /*
117 * Clear data that depends on lattice parameters in all solvers.
118 */
119 template <int D>
121 {
122 for (int i = 0; i < nPolymer(); ++i) {
123 for (int j = 0; j < polymer(i).nBlock(); ++j) {
124 polymer(i).block(j).clearUnitCellData();
125 }
126 }
127 hasStress_ = false;
128 }
129
130 /*
131 * Reset statistical segment length for one monomer type.
132 */
133 template <int D>
134 void Mixture<D>::setKuhn(int monomerId, double kuhn)
135 {
136 // Set new Kuhn length for relevant Monomer object
137 monomer(monomerId).setKuhn(kuhn);
138
139 // Update kuhn length for all blocks of this monomer type
140 for (int i = 0; i < nPolymer(); ++i) {
141 for (int j = 0; j < polymer(i).nBlock(); ++j) {
142 if (monomerId == polymer(i).block(j).monomerId()) {
143 polymer(i).block(j).setKuhn(kuhn);
144 }
145 }
146 }
147 hasStress_ = false;
148 }
149
150 /*
151 * Compute concentrations (but not total free energy).
152 */
153 template <int D>
154 void Mixture<D>::compute(DArray< RField<D> > const & wFields,
155 DArray< RField<D> > & cFields,
156 double phiTot)
157 {
158 UTIL_CHECK(meshPtr_);
159 UTIL_CHECK(mesh().size() > 0);
160 UTIL_CHECK(nMonomer() > 0);
161 UTIL_CHECK(nPolymer() + nSolvent() > 0);
162 UTIL_CHECK(wFields.capacity() == nMonomer());
163 UTIL_CHECK(cFields.capacity() == nMonomer());
164
165 int nMesh = mesh().size();
166 int nm = nMonomer();
167 int i, j;
168
169 // Clear all monomer concentration fields
170 for (i = 0; i < nm; ++i) {
171 UTIL_CHECK(cFields[i].capacity() == nMesh);
172 UTIL_CHECK(wFields[i].capacity() == nMesh);
173 VecOp::eqS(cFields[i], 0.0);
174 }
175
176 // Solve MDE for all polymers
177 for (i = 0; i < nPolymer(); ++i) {
178 polymer(i).compute(wFields, phiTot);
179 }
180
181 // Accumulate monomer concentration fields
182 int monomerId;
183 for (i = 0; i < nPolymer(); ++i) {
184 for (j = 0; j < polymer(i).nBlock(); ++j) {
185 monomerId = polymer(i).block(j).monomerId();
186 UTIL_CHECK(monomerId >= 0);
187 UTIL_CHECK(monomerId < nm);
188 RField<D>& monomerField = cFields[monomerId];
189 RField<D>& blockField = polymer(i).block(j).cField();
190 UTIL_CHECK(blockField.capacity() == nMesh);
191 VecOp::addEqV(monomerField, blockField);
192 }
193 }
194
195 // Process solvent species
196 for (i = 0; i < nSolvent(); i++) {
197 monomerId = solvent(i).monomerId();
198 UTIL_CHECK(monomerId >= 0);
199 UTIL_CHECK(monomerId < nm);
200
201 // Compute solvent concentration
202 solvent(i).compute(wFields[monomerId], phiTot);
203
204 // Add solvent contribution to relevant monomer concentration
205 RField<D>& monomerField = cFields[monomerId];
206 RField<D> const & solventField = solvent(i).cField();
207 UTIL_CHECK(solventField.capacity() == nMesh);
208 VecOp::addEqV(monomerField, solventField);
209 }
210
211 hasStress_ = false;
212 }
213
214 /*
215 * Compute total stress.
216 */
217 template <int D>
218 void Mixture<D>::computeStress(double phiTot)
219 {
220 UTIL_CHECK(nParams_ > 0);
221
222 int i, j;
223
224 // Compute stress for each polymer.
225 for (i = 0; i < nPolymer(); ++i) {
226 polymer(i).computeStress();
227 }
228
229 // Accumulate total stress
230 for (i = 0; i < nParams_; ++i) {
231 stress_[i] = 0.0;
232 for (j = 0; j < nPolymer(); ++j) {
233 stress_[i] += polymer(j).stress(i);
234 }
235 }
236
237 // Correct for partial occupation of the unit cell
238 for (i = 0; i < nParams_; ++i) {
239 stress_[i] /= phiTot;
240 }
241
242 // Note: Solvent does not contribute to derivatives of f_Helmholtz
243 // with respect to unit cell parameters at fixed volume fractions.
244
245 hasStress_ = true;
246 }
247
248 /*
249 * Is the ensemble canonical (i.e, closed for all species)?
250 */
251 template <int D>
253 {
254 // Check ensemble of all polymers
255 for (int i = 0; i < nPolymer(); ++i) {
256 if (polymer(i).ensemble() == Species::Open) {
257 return false;
258 }
259 }
260 // Check ensemble of all solvents
261 for (int i = 0; i < nSolvent(); ++i) {
262 if (solvent(i).ensemble() == Species::Open) {
263 return false;
264 }
265 }
266 // Returns true if false was never returned
267 return true;
268 }
269
270 /*
271 * Combine cFields for each block (and solvent) into one DArray
272 */
273 template <int D>
275 const
276 {
277 UTIL_CHECK(nMonomer() > 0);
278 UTIL_CHECK(nBlock() + nSolvent() > 0);
279
280 int np = nSolvent() + nBlock();
281 int nx = mesh().size();
282 int i, j;
283
284 UTIL_CHECK(blockCFields.capacity() == nBlock() + nSolvent());
285
286 // Clear all monomer concentration fields, check capacities
287 for (i = 0; i < np; ++i) {
288 UTIL_CHECK(blockCFields[i].capacity() == nx);
289 VecOp::eqS(blockCFields[i], 0.0);
290 }
291
292 // Process polymer species
293 int sectionId = -1;
294
295 if (nPolymer() > 0) {
296
297 // Write each block's r-grid data to blockCFields
298 for (i = 0; i < nPolymer(); ++i) {
299 for (j = 0; j < polymer(i).nBlock(); ++j) {
300 sectionId++;
301
302 UTIL_CHECK(sectionId >= 0);
303 UTIL_CHECK(sectionId < np);
304 UTIL_CHECK(blockCFields[sectionId].capacity() == nx);
305
306 blockCFields[sectionId] = polymer(i).block(j).cField();
307 }
308 }
309 }
310
311 // Process solvent species
312 if (nSolvent() > 0) {
313
314 // Write each solvent's r-grid data to blockCFields
315 for (i = 0; i < nSolvent(); ++i) {
316 sectionId++;
317
318 UTIL_CHECK(sectionId >= 0);
319 UTIL_CHECK(sectionId < np);
320 UTIL_CHECK(blockCFields[sectionId].capacity() == nx);
321
322 blockCFields[sectionId] = solvent(i).cField();
323 }
324 }
325 }
326
327} // namespace Rpg
328} // namespace Pscf
329#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
void compute(DArray< RField< D > > const &wFields, DArray< RField< D > > &cFields, double phiTot=1.0)
Compute partition functions and concentrations.
void allocate()
Allocate internal data containers in all solvers.
void createBlockCRGrid(DArray< RField< D > > &blockCFields) const
Combine cFields for each block/solvent into one DArray.
void computeStress(double phiTot=1.0)
Compute derivatives of free energy w/ respect to cell parameters.
void associate(Mesh< D > const &mesh, FFT< D > const &fft, UnitCell< D > const &cell, WaveList< D > &wavelist)
Create associations with a mesh, FFT, UnitCell, and WaveList object.
void readParameters(std::istream &in)
Read all parameters and initialize.
void clearUnitCellData()
Clear data that depends on lattice parameters in all solvers.
bool isCanonical()
Is the ensemble canonical (i.e, closed for all species)?
void setKuhn(int monomerId, double kuhn)
Reset statistical segment length for one monomer type.
Class to calculate and store properties of wavevectors.
Definition WaveList.h:39
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
void addEqV(DeviceArray< cudaReal > &a, DeviceArray< cudaReal > const &b, const int beginIdA, const int beginIdB, const int n)
Vector addition in-place, a[i] += b[i], kernel wrapper (cudaReal).
Definition VecOp.cu:1632
void eqS(DeviceArray< cudaReal > &a, cudaReal const b, const int beginIdA, const int n)
Vector assignment, a[i] = b, kernel wrapper (cudaReal).
Definition VecOp.cu:1054
PSCF package top-level namespace.
Definition param_pc.dox:1