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