PSCF v1.2
rpc/fts/compressor/LrAmCompressor.tpp
1#ifndef RPC_LR_POST_AM_COMPRESSOR_TPP
2#define RPC_LR_POST_AM_COMPRESSOR_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 "LrAmCompressor.h"
12#include <rpc/System.h>
13#include <rpc/fts/compressor/intra/IntraCorrelation.h>
14#include <pscf/mesh/MeshIterator.h>
15#include <util/global.h>
16
17namespace Pscf {
18namespace Rpc{
19
20 using namespace Util;
21
22 // Constructor
23 template <int D>
25 : Compressor<D>(system),
26 isAllocated_(false),
27 intraCorrelation_(system)
28 { setClassName("LrAmCompressor"); }
29
30 // Destructor
31 template <int D>
34
35 // Read parameters from file
36 template <int D>
37 void LrAmCompressor<D>::readParameters(std::istream& in)
38 {
39 // Call parent class readParameters
40 AmIteratorTmpl<Compressor<D>, DArray<double> >::readParameters(in);
41 AmIteratorTmpl<Compressor<D>, DArray<double> >::readErrorType(in);
42 }
43
44 // Initialize just before entry to iterative loop.
45 template <int D>
46 void LrAmCompressor<D>::setup(bool isContinuation)
47 {
48 const int nMonomer = system().mixture().nMonomer();
49 const int meshSize = system().domain().mesh().size();
50 IntVec<D> const & dimensions = system().domain().mesh().dimensions();
51
52 // Allocate memory required by AM algorithm if not done earlier.
53 AmIteratorTmpl<Compressor<D>, DArray<double> >::setup(isContinuation);
54
55 // Compute Fourier space kMeshDimensions_
56 for (int i = 0; i < D; ++i) {
57 if (i < D - 1) {
58 kMeshDimensions_[i] = dimensions[i];
59 } else {
60 kMeshDimensions_[i] = dimensions[i]/2 + 1;
61 }
62 }
63
64 // Allocate memory required by compressor, if not done previously
65 if (!isAllocated_){
66 newBasis_.allocate(meshSize);
67 w0_.allocate(nMonomer);
68 wFieldTmp_.allocate(nMonomer);
69 resid_.allocate(dimensions);
70 residK_.allocate(dimensions);
71 intraCorrelationK_.allocate(kMeshDimensions_);
72 for (int i = 0; i < nMonomer; ++i) {
73 w0_[i].allocate(dimensions);
74 wFieldTmp_[i].allocate(dimensions);
75 }
76
77 isAllocated_ = true;
78 }
79
80 // Store initial values of monomer chemical potential fields in w0_
81 for (int i = 0; i < nMonomer; ++i) {
82 for (int j = 0; j< meshSize; ++j){
83 w0_[i][j] = system().w().rgrid(i)[j];
84 }
85 }
86
87 // Compute homopolymer intraCorrelation
88 intraCorrelationK_ = intraCorrelation_.computeIntraCorrelations();
89 }
90
91 /*
92 * Apply the Anderson-Mixing algorithm (main function).
93 */
94 template <int D>
96 {
97 int solve = AmIteratorTmpl<Compressor<D>, DArray<double> >::solve();
98 return solve;
99 }
100
101 /*
102 * Assign one array to another.
103 */
104 template <int D>
106 DArray<double> const & b)
107 { a = b; }
108
109 /*
110 * Compute and return inner product of two vectors.
111 */
112 template <int D>
113 double
114 LrAmCompressor<D>::dotProduct(DArray<double> const & a,
115 DArray<double> const & b)
116 {
117 const int n = a.capacity();
118 UTIL_CHECK(b.capacity() == n);
119 double product = 0.0;
120 for (int i = 0; i < n; i++) {
121 // if either value is NaN, throw NanException
122 if (std::isnan(a[i]) || std::isnan(b[i])) {
123 throw NanException("LrAmCompressor::dotProduct",
124 __FILE__,__LINE__,0);
125 }
126 product += a[i] * b[i];
127 }
128 return product;
129 }
130
131 /*
132 * Compute and return the maximum element of a vector.
133 */
134 template <int D>
135 double LrAmCompressor<D>::maxAbs(DArray<double> const & a)
136 {
137 const int n = a.capacity();
138 double max = 0.0;
139 double value;
140 for (int i = 0; i < n; i++) {
141 value = a[i];
142 if (std::isnan(value)) { // if value is NaN, throw NanException
143 throw NanException("LrAmCompressor::dotProduct",
144 __FILE__, __LINE__, 0);
145 }
146 if (fabs(value) > max) {
147 max = fabs(value);
148 }
149 }
150 return max;
151 }
152
153 /*
154 * Update basis by adding one new basis vector.
155 */
156 template <int D>
157 void
158 LrAmCompressor<D>::updateBasis(
159 RingBuffer< DArray<double> > & basis,
160 RingBuffer< DArray<double> > const & hists)
161 {
162 // Make sure at least two histories are stored
163 UTIL_CHECK(hists.size() >= 2);
164
165 const int n = hists[0].capacity();
166
167 // New basis vector is difference between two most recent states
168 for (int i = 0; i < n; i++) {
169 newBasis_[i] = hists[0][i] - hists[1][i];
170 }
171
172 basis.append(newBasis_);
173 }
174
175 template <int D>
176 void
177 LrAmCompressor<D>::addHistories(DArray<double>& trial,
178 RingBuffer<DArray<double> > const & basis,
179 DArray<double> coeffs,
180 int nHist)
181 {
182 int n = trial.capacity();
183 for (int i = 0; i < nHist; i++) {
184 for (int j = 0; j < n; j++) {
185 // Not clear on the origin of the -1 factor
186 trial[j] += coeffs[i] * -1 * basis[i][j];
187 }
188 }
189 }
190
191 /*
192 * Returns mixing value parameter lambda = 1.0
193 */
194 template<int D>
195 double LrAmCompressor<D>::computeLambda(double r)
196 { return 1.0; }
197
198 /*
199 * Correction step (second step of Anderson mixing)
200 *
201 * This LrAM algorithm uses a quasi-Newton correction step with an
202 * approximate Jacobian given by the Jacobian in a homogeneous state.
203 */
204 template <int D>
205 void
206 LrAmCompressor<D>::addPredictedError(DArray<double>& fieldTrial,
207 DArray<double> const & resTrial,
208 double lambda)
209 {
210 // Local constants
211 const int n = fieldTrial.capacity();
212 const double vMonomer = system().mixture().vMonomer();
213
214 // Copy DArray<double> resTrial into RField<D> resid_
215 // Allows use of FFT functions that take an RField container
216 for (int i = 0 ; i < n; ++i) {
217 resid_[i] = resTrial[i];
218 }
219
220 // Fourier transform r-grid residual resid_ to k-grid form residK_
221 system().domain().fft().forwardTransform(resid_, residK_);
222
223 // Compute update on a k-grid, using quasi-Newton algorithm
224 double factor;
225 MeshIterator<D> iter;
226 iter.setDimensions(kMeshDimensions_);
227 for (iter.begin(); !iter.atEnd(); ++iter) {
228 factor = 1.0 / (vMonomer * intraCorrelationK_[iter.rank()]);
229 residK_[iter.rank()][0] *= factor;
230 residK_[iter.rank()][1] *= factor;
231 }
232 // Field residK_ now stores k-grid update rather than residual
233
234 // Convert field update back to r-grid (real space) form
235 // On return, resid_ contains r-grid field update, residK_ is destroyed
236 system().domain().fft().inverseTransformUnsafe(residK_, resid_);
237
238 // Add update to obtain new fieldTrial
239 for (int i = 0; i < n; i++) {
240 fieldTrial[i] += lambda * resid_[i];
241 }
242 }
243
244 /*
245 * Does the associated system have initialized w fields?
246 */
247 template <int D>
248 bool LrAmCompressor<D>::hasInitialGuess()
249 { return system().w().hasData(); }
250
251 /*
252 * Compute and return the number of elements in a field vector.
253 */
254 template <int D>
255 int LrAmCompressor<D>::nElements()
256 { return system().domain().mesh().size(); }
257
258 /*
259 * Get the current field variable from the system.
260 *
261 * The field variable is the change in the Lagrange multiplier field.
262 * relative to that used in initial array of monomer fields, w0_.
263 */
264 template <int D>
265 void LrAmCompressor<D>::getCurrent(DArray<double>& curr)
266 {
267 // Straighten out fields into linear arrays
268 const int meshSize = system().domain().mesh().size();
269 const DArray< RField<D> > * currSys = &system().w().rgrid();
270
271 /*
272 * The field that we are adjusting is the Langrange multiplier field
273 * with number of grid pts components. The current value is the
274 * difference between w and w0_ for the first monomer (any monomer
275 * should give the same answer)
276 */
277 for (int i = 0; i < meshSize; i++){
278 curr[i] = (*currSys)[0][i] - w0_[0][i];
279 }
280
281 }
282
283 /*
284 * Perform the main system computation (solve the MDE).
285 */
286 template <int D>
287 void LrAmCompressor<D>::evaluate()
288 {
289 system().compute();
290 ++mdeCounter_;
291 }
292
293 /*
294 * Compute the residual vector for the current system state
295 */
296 template <int D>
297 void LrAmCompressor<D>::getResidual(DArray<double>& resid)
298 {
299 const int n = nElements();
300 const int nMonomer = system().mixture().nMonomer();
301 const int meshSize = system().domain().mesh().size();
302
303 // Initialize residuals
304 for (int i = 0 ; i < n; ++i) {
305 resid[i] = -1.0;
306 }
307
308 // Compute SCF residual vector elements
309 for (int j = 0; j < nMonomer; ++j) {
310 for (int k = 0; k < meshSize; ++k) {
311 resid[k] += system().c().rgrid(j)[k];
312 }
313 }
314
315 }
316
317 /*
318 * Update the w field values stored in the system
319 */
320 template <int D>
321 void LrAmCompressor<D>::update(DArray<double>& newGuess)
322 {
323 // Local constant values
324 const int nMonomer = system().mixture().nMonomer();
325 const int meshSize = system().domain().mesh().size();
326
327 // Locally update monomer w fields in array wFieldTmp_
328 // Set wFieldTmp = w0_ + newGuess
329 for (int i = 0; i < nMonomer; i++){
330 for (int k = 0; k < meshSize; k++){
331 wFieldTmp_[i][k] = w0_[i][k] + newGuess[k];
332 }
333 }
334
335 // Update w fields in the system WContainer
336 system().setWRGrid(wFieldTmp_);
337 }
338
339 template<int D>
340 void LrAmCompressor<D>::outputToLog()
341 {}
342
343 /*
344 * Output timing results to an output stream (file)
345 */
346 template<int D>
347 void LrAmCompressor<D>::outputTimers(std::ostream& out)
348 {
349 // Output timing results, if requested.
350 out << "\n";
351 out << "LrAmCompressor time contributions:\n";
352 AmIteratorTmpl<Compressor<D>, DArray<double> >::outputTimers(out);
353 }
354
355 /*
356 * Clear all timers and the MDE counter
357 */
358 template<int D>
360 {
362 mdeCounter_ = 0;
363 }
364
365}
366}
367#endif
Template for Anderson mixing iterator algorithm.
An IntVec<D, T> is a D-component vector of elements of integer type T.
Definition IntVec.h:27
Exception thrown when not-a-number (NaN) is encountered.
Base class for iterators that impose incompressibility.
Anderson Mixing compressor with linear-response mixing step.
void readParameters(std::istream &in)
Read all parameters and initialize.
LrAmCompressor(System< D > &system)
Constructor.
void outputTimers(std::ostream &out)
Return compressor times contributions.
void clearTimers()
Clear all timers (reset accumulated time to zero).
void setup(bool isContinuation)
Initialize just before entry to iterative loop.
int compress()
Compress to obtain partial saddle point w+.
Main class for SCFT or PS-FTS simulation of one system.
Definition rpc/System.h:100
int capacity() const
Return allocated size.
Definition Array.h:159
Dynamically allocatable contiguous array template.
Class for storing history of previous values in an array.
Definition RingBuffer.h:27
File containing preprocessor macros for error handling.
#define UTIL_CHECK(condition)
Assertion macro suitable for serial or parallel production code.
Definition global.h:68
cudaReal max(DeviceArray< cudaReal > const &in)
Get maximum of array elements (GPU kernel wrapper).
Definition Reduce.cu:569
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