PSCF v1.2
rpc/fts/compressor/LrCompressor.tpp
1#ifndef RPC_LR_COMPRESSOR_TPP
2#define RPC_LR_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 "LrCompressor.h"
12#include <rpc/System.h>
13#include <rpc/fts/compressor/intra/IntraCorrelation.h>
14#include <util/global.h>
15#include <util/format/Dbl.h>
16#include <pscf/mesh/MeshIterator.h>
17#include <pscf/iterator/NanException.h>
18#include <prdc/crystal/shiftToMinimum.h>
19
20
21namespace Pscf {
22namespace Rpc{
23
24 using namespace Util;
25
26 // Constructor
27 template <int D>
29 : Compressor<D>(system),
30 epsilon_(0.0),
31 itr_(0),
32 maxItr_(0),
33 totalItr_(0),
34 errorType_("rmsResid"),
35 verbose_(0),
36 isAllocated_(false),
37 intraCorrelation_(system)
38 { setClassName("LrCompressor"); }
39
40 // Destructor
41 template <int D>
44
45 // Read parameters from file
46 template <int D>
47 void LrCompressor<D>::readParameters(std::istream& in)
48 {
49 maxItr_ = 60;
50 read(in, "epsilon", epsilon_);
51 readOptional(in, "maxItr", maxItr_);
52 readOptional(in, "verbose", verbose_);
53 readOptional(in, "errorType", errorType_);
54 }
55
56 // Initialize just before entry to iterative loop.
57 template <int D>
59 {
60 const int nMonomer = system().mixture().nMonomer();
61 IntVec<D> const & dimensions = system().domain().mesh().dimensions();
62 for (int i = 0; i < D; ++i) {
63 if (i < D - 1) {
64 kMeshDimensions_[i] = dimensions[i];
65 } else {
66 kMeshDimensions_[i] = dimensions[i]/2 + 1;
67 }
68 }
69
70 // Allocate memory required by AM algorithm if not done earlier.
71 if (!isAllocated_){
72 resid_.allocate(dimensions);
73 residK_.allocate(dimensions);
74 wFieldTmp_.allocate(nMonomer);
75 intraCorrelationK_.allocate(kMeshDimensions_);
76 for (int i = 0; i < nMonomer; ++i) {
77 wFieldTmp_[i].allocate(dimensions);
78 }
79 isAllocated_ = true;
80 }
81
82 // Compute intraCorrelation (homopolymer)
83 intraCorrelationK_ = intraCorrelation_.computeIntraCorrelations();
84 }
85
86 template <int D>
88 {
89 // Initialization and allocate operations on entry to loop.
90 setup();
91 UTIL_CHECK(isAllocated_);
92
93 // Start overall timer
94 timerTotal_.start();
95
96 // Solve MDE
97 timerMDE_.start();
98 system().compute();
99 ++mdeCounter_;
100 timerMDE_.stop();
101
102 // Iterative loop
103 for (itr_ = 0; itr_ < maxItr_; ++itr_) {
104
105 if (verbose_ > 2) {
106 Log::file() << "------------------------------- \n";
107 }
108
109 if (verbose_ > 0){
110 Log::file() << " Iteration " << Int(itr_,5);
111 }
112
113 // Compute residual vector
114 getResidual();
115 double error;
116 try {
117 error = computeError(verbose_);
118 } catch (const NanException&) {
119 Log::file() << ", error = NaN" << std::endl;
120 break; // Exit loop if a NanException is caught
121 }
122 if (verbose_ > 0) {
123 Log::file() << ", error = " << Dbl(error, 15) << std::endl;
124 }
125
126 // Check for convergence
127 if (error < epsilon_) {
128
129 // Successful completion (i.e., converged within tolerance)
130 timerTotal_.stop();
131
132 if (verbose_ > 2) {
133 Log::file() << "-------------------------------\n";
134 }
135 if (verbose_ > 0) {
136 Log::file() << " Converged\n";
137 }
138 if (verbose_ == 2) {
139 Log::file() << "\n";
140 computeError(2);
141 }
142 //mdeCounter_ += itr_;
143 totalItr_ += itr_;
144
145 return 0; // Success
146
147 } else{
148
149 // Not yet converged.
150 updateWFields();
151 timerMDE_.start();
152 system().compute();
153 ++mdeCounter_;
154 timerMDE_.stop();
155
156 }
157
158 }
159
160 // Failure: iteration counter itr reached maxItr without converging
161 timerTotal_.stop();
162
163 Log::file() << "Iterator failed to converge.\n";
164 return 1;
165
166 }
167
168 // Compute the residual for the current system state
169 template <int D>
171 {
172 const int nMonomer = system().mixture().nMonomer();
173 const int meshSize = system().domain().mesh().size();
174
175 // Initialize residuals
176 for (int i = 0 ; i < meshSize; ++i) {
177 resid_[i] = -1.0;
178 }
179
180 // Compute SCF residual vector elements
181 for (int j = 0; j < nMonomer; ++j) {
182 for (int k = 0; k < meshSize; ++k) {
183 resid_[k] += system().c().rgrid(j)[k];
184 }
185 }
186 }
187
188 // update system w field using linear response approximation
189 template <int D>
190 void LrCompressor<D>::updateWFields()
191 {
192 const int nMonomer = system().mixture().nMonomer();
193 const int meshSize = system().domain().mesh().size();
194 const double vMonomer = system().mixture().vMonomer();
195
196 // Convert residual to Fourier Space
197 system().domain().fft().forwardTransform(resid_, residK_);
198 MeshIterator<D> iter;
199 iter.setDimensions(residK_.dftDimensions());
200 for (iter.begin(); !iter.atEnd(); ++iter) {
201 residK_[iter.rank()][0] *= 1.0 / (vMonomer * intraCorrelationK_[iter.rank()]);
202 residK_[iter.rank()][1] *= 1.0 / (vMonomer * intraCorrelationK_[iter.rank()]);
203 }
204
205 // Convert back to real Space (destroys residK_)
206 system().domain().fft().inverseTransformUnsafe(residK_, resid_);
207
208 for (int i = 0; i < nMonomer; i++){
209 for (int k = 0; k < meshSize; k++){
210 wFieldTmp_[i][k] = system().w().rgrid(i)[k] + resid_[k];
211 }
212 }
213 system().setWRGrid(wFieldTmp_);
214 }
215
216 template<int D>
217 void LrCompressor<D>::outputToLog()
218 {}
219
220 template<int D>
221 void LrCompressor<D>::outputTimers(std::ostream& out)
222 {
223 // Output timing results, if requested.
224 double total = timerTotal_.time();
225 out << "\n";
226 out << "LrCompressor time contributions:\n";
227 out << " ";
228 out << "Total" << std::setw(22)<< "Per Iteration"
229 << std::setw(9) << "Fraction" << "\n";
230 out << "MDE solution: "
231 << Dbl(timerMDE_.time(), 9, 3) << " s, "
232 << Dbl(timerMDE_.time()/totalItr_, 9, 3) << " s, "
233 << Dbl(timerMDE_.time()/total, 9, 3) << "\n";
234 out << "\n";
235 }
236
237 template<int D>
239 {
240 timerTotal_.clear();
241 timerMDE_.clear();
242 mdeCounter_ = 0;
243 totalItr_ = 0;
244 }
245
246 template <int D>
247 double LrCompressor<D>::maxAbs(RField<D> const & a)
248 {
249 const int n = a.capacity();
250 double max = 0.0;
251 double value;
252 for (int i = 0; i < n; i++) {
253 value = a[i];
254 if (std::isnan(value)) { // if value is NaN, throw NanException
255 throw NanException("LrCompressor::dotProduct",__FILE__,__LINE__,0);
256 }
257 if (fabs(value) > max) {
258 max = fabs(value);
259 }
260 }
261 return max;
262 }
263
264 // Compute and return inner product of two vectors.
265 template <int D>
266 double LrCompressor<D>::dotProduct(RField<D> const & a,
267 RField<D> const & b)
268 {
269 const int n = a.capacity();
270 UTIL_CHECK(b.capacity() == n);
271 double product = 0.0;
272 for (int i = 0; i < n; i++) {
273 // if either value is NaN, throw NanException
274 if (std::isnan(a[i]) || std::isnan(b[i])) {
275 throw NanException("AmCompressor::dotProduct",__FILE__,__LINE__,0);
276 }
277 product += a[i] * b[i];
278 }
279 return product;
280 }
281
282 // Compute L2 norm of an RField
283 template <int D>
284 double LrCompressor<D>::norm(RField<D> const & a)
285 {
286 double normSq = dotProduct(a, a);
287 return sqrt(normSq);
288 }
289
290 // Compute and return the scalar error
291 template <int D>
292 double LrCompressor<D>::computeError(int verbose)
293 {
294 const int meshSize = system().domain().mesh().size();
295 double error = 0;
296
297 // Find max residual vector element
298 double maxRes = maxAbs(resid_);
299 // Find norm of residual vector
300 double normRes = norm(resid_);
301 // Find root-mean-squared residual element value
302 double rmsRes = normRes/sqrt(meshSize);
303 if (errorType_ == "maxResid") {
304 error = maxRes;
305 } else if (errorType_ == "normResid") {
306 error = normRes;
307 } else if (errorType_ == "rmsResid") {
308 error = rmsRes;
309 } else {
310 UTIL_THROW("Invalid iterator error type in parameter file.");
311 }
312
313 if (verbose > 1) {
314 Log::file() << "\n";
315 Log::file() << "Max Residual = " << Dbl(maxRes,15) << "\n";
316 Log::file() << "Residual Norm = " << Dbl(normRes,15) << "\n";
317 Log::file() << "RMS Residual = " << Dbl(rmsRes,15) << "\n";
318
319 // Check if calculation has diverged (normRes will be NaN)
320 UTIL_CHECK(!std::isnan(normRes));
321 }
322 return error;
323 }
324
325}
326}
327#endif
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.
Field of real double precision values on an FFT mesh.
Base class for iterators that impose incompressibility.
void setClassName(const char *className)
Set class name string.
LrCompressor(System< D > &system)
Constructor.
int compress()
Iterate pressure field to obtain partial saddle point.
void outputTimers(std::ostream &out)
Return compressor times contributions.
void readParameters(std::istream &in)
Read all parameters and initialize.
void setup()
Initialize just before entry to iterative loop.
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
Wrapper for a double precision number, for formatted ostream output.
Definition Dbl.h:40
Wrapper for an int, for formatted ostream output.
Definition Int.h:37
static std::ostream & file()
Get log ostream by reference.
Definition Log.cpp:57
File containing preprocessor macros for error handling.
#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
cudaReal maxAbs(DeviceArray< cudaReal > const &in)
Get maximum absolute magnitude of array elements (GPU kernel wrapper).
Definition Reduce.cu:648
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