1#ifndef RPG_LR_COMPRESSOR_TPP
2#define RPG_LR_COMPRESSOR_TPP
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>
19#include <util/format/Dbl.h>
35 errorType_(
"rmsResid"),
38 intraCorrelation_(system)
52 read(in,
"epsilon", epsilon_);
53 readOptional(in,
"maxItr", maxItr_);
54 readOptional(in,
"verbose", verbose_);
55 readOptional(in,
"errorType", errorType_);
62 const int nMonomer = system().mixture().nMonomer();
63 IntVec<D> const & dimensions = system().mesh().dimensions();
64 for (
int i = 0; i < D; ++i) {
66 kMeshDimensions_[i] = dimensions[i];
68 kMeshDimensions_[i] = dimensions[i]/2 + 1;
74 for (
int i = 0; i < D; ++i) {
75 kSize_ *= kMeshDimensions_[i];
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);
91 intraCorrelationK_ = intraCorrelation_.computeIntraCorrelations();
111 for (itr_ = 0; itr_ < maxItr_; ++itr_) {
114 Log::file() <<
"------------------------------- \n";
125 error = computeError(verbose_);
127 Log::file() <<
", error = NaN" << std::endl;
131 Log::file() <<
", error = " <<
Dbl(error, 15) << std::endl;
135 if (error < epsilon_) {
141 Log::file() <<
"-------------------------------\n";
171 Log::file() <<
"Iterator failed to converge.\n";
180 const int nMonomer = system().mixture().nMonomer();
186 for (
int i = 1; i < nMonomer; i++) {
193 void LrCompressor<D>::updateWFields(){
194 const int nMonomer = system().mixture().nMonomer();
195 const double vMonomer = system().mixture().vMonomer();
198 system().fft().forwardTransform(resid_, residK_);
204 system().fft().inverseTransformUnsafe(residK_, resid_);
207 for (
int i = 0; i < nMonomer; i++) {
208 VecOp::addVV(wFieldTmp_[i], system().w().rgrid(i), resid_);
212 system().setWRGrid(wFieldTmp_);
216 void LrCompressor<D>::outputToLog()
223 double total = timerTotal_.time();
225 out <<
"LrCompressor time contributions:\n";
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";
253 double LrCompressor<D>::dotProduct(RField<D>
const & a,
262 double LrCompressor<D>::norm(RField<D>
const & a)
264 double normSq = dotProduct(a, a);
270 double LrCompressor<D>::computeError(
int verbose)
272 const int meshSize = system().domain().mesh().size();
276 double maxRes =
maxAbs(resid_);
278 double normRes = norm(resid_);
280 double rmsRes = normRes/sqrt(meshSize);
281 if (errorType_ ==
"maxResid") {
283 }
else if (errorType_ ==
"normResid") {
285 }
else if (errorType_ ==
"rmsResid") {
288 UTIL_THROW(
"Invalid iterator error type in parameter file.");
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";
An IntVec<D, T> is a D-component vector of elements of integer type T.
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.
Linear response compressor.
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.
void clearTimers()
Clear timers.
~LrCompressor()
Destructor.
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.
Wrapper for a double precision number, for formatted ostream output.
Wrapper for an int, for formatted ostream output.
static std::ostream & file()
Get log ostream by reference.
File containing preprocessor macros for error handling.
#define UTIL_CHECK(condition)
Assertion macro suitable for serial or parallel production code.
#define UTIL_THROW(msg)
Macro for throwing an Exception, reporting function, file and line number.
cudaReal innerProduct(DeviceArray< cudaReal > const &a, DeviceArray< cudaReal > const &b)
Compute inner product of two real arrays (GPU kernel wrapper).
cudaReal maxAbs(DeviceArray< cudaReal > const &in)
Get maximum absolute magnitude of array elements (GPU kernel wrapper).
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).
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).
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).
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.
PSCF package top-level namespace.
Utility classes for scientific computation.