PSCF v1.2
rpg/fts/compressor/AmCompressor.tpp
1#ifndef RPG_AM_COMPRESSOR_TPP
2#define RPG_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 "AmCompressor.h"
12#include <rpg/System.h>
13#include <prdc/cuda/RField.h>
14#include <prdc/cuda/resources.h>
15#include <pscf/math/IntVec.h>
16#include <util/global.h>
17
18namespace Pscf {
19namespace Rpg{
20
21 using namespace Util;
22
23 // Constructor
24 template <int D>
26 : Compressor<D>(system),
27 isAllocated_(false)
28 { setClassName("AmCompressor"); }
29
30 // Destructor
31 template <int D>
34
35 // Read parameters from file
36 template <int D>
37 void AmCompressor<D>::readParameters(std::istream& in)
38 {
39 // Call parent class readParameters
41 DeviceArray<cudaReal> >::readParameters(in);
43 DeviceArray<cudaReal> >::readErrorType(in);
44 }
45
46
47 // Initialize just before entry to iterative loop.
48 template <int D>
49 void AmCompressor<D>::setup(bool isContinuation)
50 {
51 const int nMonomer = system().mixture().nMonomer();
52 const int meshSize = system().domain().mesh().size();
53 const IntVec<D> dimensions = system().domain().mesh().dimensions();
54
55 // Allocate memory required by AM algorithm if not done earlier.
57 DeviceArray<cudaReal> >::setup(isContinuation);
58
59 // Allocate memory required by compressor if not done earlier.
60 if (!isAllocated_) {
61 w0_.allocate(nMonomer);
62 wFieldTmp_.allocate(nMonomer);
63 for (int i = 0; i < nMonomer; ++i) {
64 w0_[i].allocate(dimensions);
65 wFieldTmp_[i].allocate(dimensions);
66 }
67 isAllocated_ = true;
68 }
69
70 // Store value of initial guess chemical potential fields
71 for (int i = 0; i < nMonomer; ++i) {
72 VecOp::eqV(w0_[i], system().w().rgrid(i));
73 }
74 }
75
76 template <int D>
78 {
80 DeviceArray<cudaReal> >::solve();
81 //mdeCounter_ = AmIteratorTmpl<Compressor<D>, DeviceArray<cudaReal>>::totalItr();
82 return solve;
83 }
84
85 // Assign one array to another
86 template <int D>
88 DeviceArray<cudaReal> const & b)
89 {
90 UTIL_CHECK(b.capacity() == a.capacity());
91 VecOp::eqV(a, b);
92 }
93
94 // Compute and return inner product of two vectors.
95 template <int D>
96 double AmCompressor<D>::dotProduct(DeviceArray<cudaReal> const & a,
97 DeviceArray<cudaReal> const & b)
98 {
99 UTIL_CHECK(a.capacity() == b.capacity());
100 return Reduce::innerProduct(a, b);
101 }
102
103 // Compute and return maximum element of a vector.
104 template <int D>
105 double AmCompressor<D>::maxAbs(DeviceArray<cudaReal> const & a)
106 {
107 return Reduce::maxAbs(a);
108 }
109
110 // Update basis
111 template <int D>
112 void
113 AmCompressor<D>::updateBasis(RingBuffer< DeviceArray<cudaReal> > & basis,
114 RingBuffer< DeviceArray<cudaReal> > const & hists)
115 {
116 // Make sure at least two histories are stored
117 UTIL_CHECK(hists.size() >= 2);
118
119 basis.advance();
120 if (basis[0].isAllocated()) {
121 UTIL_CHECK(basis[0].capacity() == hists[0].capacity());
122 } else {
123 basis[0].allocate(hists[0].capacity());
124 }
125
126 VecOp::subVV(basis[0], hists[0], hists[1]);
127 }
128
129 template <int D>
130 void
131 AmCompressor<D>::addHistories(DeviceArray<cudaReal>& trial,
132 RingBuffer<DeviceArray<cudaReal> > const & basis,
133 DArray<double> coeffs,
134 int nHist)
135 {
136 for (int i = 0; i < nHist; i++) {
137 VecOp::addEqVc(trial, basis[i], -1.0 * coeffs[i]);
138 }
139 }
140
141 template <int D>
142 void AmCompressor<D>::addPredictedError(DeviceArray<cudaReal>& fieldTrial,
143 DeviceArray<cudaReal> const & resTrial,
144 double lambda)
145 {
146 VecOp::addEqVc(fieldTrial, resTrial, lambda);
147 }
148
149 // Does the system have an initial field guess?
150 template <int D>
151 bool AmCompressor<D>::hasInitialGuess()
152 { return system().w().hasData(); }
153
154 // Compute and return the number of elements in a field vector
155 template <int D>
156 int AmCompressor<D>::nElements()
157 { return system().domain().mesh().size(); }
158
159 // Get the current field from the system
160 template <int D>
161 void AmCompressor<D>::getCurrent(DeviceArray<cudaReal>& curr)
162 {
163 /*
164 * The field that we are adjusting is the Langrange multiplier field
165 * with number of grid pts components.The current value is the difference
166 * between w and w0_ for the first monomer (any monomer should give the
167 * same answer)
168 */
169 VecOp::subVV(curr, system().w().rgrid(0), w0_[0]);
170 }
171
172 // Perform the main system computation (solve the MDE)
173 template <int D>
174 void AmCompressor<D>::evaluate()
175 {
176 system().compute();
177 ++mdeCounter_;
178 }
179
180 // Compute the residual for the current system state
181 template <int D>
182 void AmCompressor<D>::getResidual(DeviceArray<cudaReal>& resid)
183 {
184 const int nMonomer = system().mixture().nMonomer();
185
186 // Initialize resid to c field of species 0 minus 1
187 VecOp::subVS(resid, system().c().rgrid(0), 1.0);
188
189 // Add other c fields to get SCF residual vector elements
190 for (int i = 1; i < nMonomer; i++) {
191 VecOp::addEqV(resid, system().c().rgrid(i));
192 }
193 }
194
195 // Update the current system field coordinates
196 template <int D>
197 void AmCompressor<D>::update(DeviceArray<cudaReal>& newGuess)
198 {
199 // Convert back to field format
200 const int nMonomer = system().mixture().nMonomer();
201
202 // New field is the w0_ + the newGuess for the Lagrange multiplier field
203 for (int i = 0; i < nMonomer; i++) {
204 VecOp::addVV(wFieldTmp_[i], w0_[i], newGuess);
205 }
206
207 // set system r grid
208 system().setWRGrid(wFieldTmp_);
209 }
210
211 template<int D>
213 {
214 return 1.0;
215 }
216
217 template<int D>
219 {}
220
221 template<int D>
222 void AmCompressor<D>::outputTimers(std::ostream& out)
223 {
224 // Output timing results, if requested.
225 out << "\n";
226 out << "Compressor times contributions:\n";
228 }
229
230 template<int D>
232 {
234 mdeCounter_ = 0;
235 }
236
237}
238}
239#endif
Template for Anderson mixing iterator algorithm.
Dynamic array on the GPU device with aligned data.
Definition rpg/System.h:32
int capacity() const
Return allocated capacity.
An IntVec<D, T> is a D-component vector of elements of integer type T.
Definition IntVec.h:27
Rpg implementation of the Anderson Mixing compressor.
int compress()
Compress to obtain partial saddle point w+.
void outputTimers(std::ostream &out)
Return compressor times contributions.
AmCompressor(System< D > &system)
Constructor.
void setup(bool isContinuation)
Initialize just before entry to iterative loop.
double computeLambda(double r)
Compute mixing parameter lambda.
void clearTimers()
Clear all timers (reset accumulated time to zero).
void readParameters(std::istream &in)
Read all parameters and initialize.
Base class for iterators that impose incompressibility.
Main class for calculations that represent one system.
Definition rpg/System.h:107
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 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 eqV(DeviceArray< cudaReal > &a, DeviceArray< cudaReal > const &b, const int beginIdA, const int beginIdB, const int n)
Vector assignment, a[i] = b[i], kernel wrapper (cudaReal).
Definition VecOp.cu:1020
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 subVV(DeviceArray< cudaReal > &a, DeviceArray< cudaReal > const &b, DeviceArray< cudaReal > const &c, const int beginIdA, const int beginIdB, const int beginIdC, const int n)
Vector subtraction, a[i] = b[i] - c[i], kernel wrapper (cudaReal).
Definition VecOp.cu:1228
void addEqVc(DeviceArray< cudaReal > &a, DeviceArray< cudaReal > const &b, cudaReal const c)
Vector addition in-place w/ coefficient, a[i] += b[i] * c, kernel wrapper.
Definition VecOpMisc.cu:343
PSCF package top-level namespace.
Definition param_pc.dox:1
Utility classes for scientific computation.