PSCF v1.2
rpg/solvers/Propagator.tpp
1#ifndef RPG_PROPAGATOR_TPP
2#define RPG_PROPAGATOR_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 "Propagator.h"
12#include "Block.h"
13#include <prdc/cuda/resources.h>
14#include <pscf/mesh/Mesh.h>
15#include <util/containers/DArray.h>
16
17namespace Pscf {
18namespace Rpg {
19
20 using namespace Util;
21
22 /*
23 * Constructor.
24 */
25 template <int D>
27 : blockPtr_(0),
28 meshPtr_(0),
29 ns_(0),
30 isAllocated_(false)
31 {}
32
33 /*
34 * Destructor.
35 */
36 template <int D>
39
40 /*
41 * Allocate memory used by this propagator.
42 */
43 template <int D>
44 void Propagator<D>::allocate(int ns, const Mesh<D>& mesh)
45 {
46 ns_ = ns;
47 meshPtr_ = &mesh;
48
49 // Allocate memory in qFieldsAll_ using value of ns
50 int meshSize = meshPtr_->size();
51 qFieldsAll_.allocate(ns * meshSize);
52
53 // Set up array of associated RField<D> arrays
54 qFields_.allocate(ns);
55 for (int i = 0; i < ns; ++i) {
56 qFields_[i].associate(qFieldsAll_, i*meshSize, meshPtr_->dimensions());
57 }
58 isAllocated_ = true;
59 }
60
61 /*
62 * Reallocate memory used by this propagator using new ns value.
63 */
64 template <int D>
66 {
67 UTIL_CHECK(isAllocated_);
68 UTIL_CHECK(ns_ != ns);
69 ns_ = ns;
70
71 // Deallocate memory previously used by this propagator.
72 qFields_.deallocate(); // Destroys associated RField<D> objects
73 // but not the underlying data
74 qFieldsAll_.deallocate(); // Destroy actual propagator data
75
76 // Allocate memory in qFieldsAll_ using new value of ns
77 int meshSize = meshPtr_->size();
78 qFieldsAll_.allocate(ns * meshSize);
79
80 // Set up array of associated RField<D> arrays
81 qFields_.allocate(ns);
82 for (int i = 0; i < ns; ++i) {
83 qFields_[i].associate(qFieldsAll_, i*meshSize, meshPtr_->dimensions());
84 }
85
86 setIsSolved(false);
87 }
88
89 /*
90 * Compute initial head QField from final tail QFields of sources.
91 */
92 template <int D>
94 {
95 // Initialize head field (s=0) to 1.0 at all grid points
96 int nx = meshPtr_->size();
97 VecOp::eqS(qFields_[0], 1.0);
98
99 // Multiply head q-field by tail q-fields of all sources
100 if (nSource() > 0) {
101 DArray<DeviceArray<cudaReal> const *> tails;
102 tails.allocate(nSource()+1);
103 tails[0] = &qFields_[0];
104 for (int is = 0; is < nSource(); ++is) {
105 if (!source(is).isSolved()) {
106 UTIL_THROW("Source not solved in computeHead");
107 }
108 tails[is+1] = &(source(is).tail());
109 }
110 VecOp::mulVMany(qFields_[0], tails);
111 }
112 }
113
114 /*
115 * Solve the modified diffusion equation for this block.
116 */
117 template <int D>
119 {
120 UTIL_CHECK(isAllocated());
121
122 computeHead();
123 for (int iStep = 0; iStep < ns_ - 1; ++iStep) {
124 block().step(qFields_[iStep], qFields_[iStep+1]);
125 }
126 setIsSolved(true);
127 }
128
129 /*
130 * Solve the modified diffusion equation with specified initial field.
131 */
132 template <int D>
134 {
135 UTIL_CHECK(isAllocated());
136
137 // Initialize initial (head) field
138 VecOp::eqV(qFields_[0], head);
139
140 for (int iStep = 0; iStep < ns_ - 1; ++iStep) {
141 block().step(qFields_[iStep], qFields_[iStep+1]);
142 }
143 setIsSolved(true);
144 }
145
146 /*
147 * Integrate to calculate monomer concentration for this block
148 */
149 template <int D>
151 {
152 // Preconditions
153 if (!isSolved()) {
154 UTIL_THROW("Propagator is not solved.");
155 }
156 if (!hasPartner()) {
157 UTIL_THROW("Propagator has no partner set.");
158 }
159 if (!partner().isSolved()) {
160 UTIL_THROW("Partner propagator is not solved");
161 }
162
163 // Take inner product of head and partner tail fields
164 // cannot reduce assuming one propagator, qh == 1
165 // polymers are divided into blocks midway through
166 int nx = meshPtr_->size();
167 double Q = Reduce::innerProduct(head(), partner().tail()) / double(nx);
168 return Q;
169 }
170
171}
172}
173#endif
Description of a regular grid of points in a periodic domain.
int size() const
Get total number of grid points.
Definition Mesh.h:229
Field of real double precision values on an FFT mesh.
void solve()
Solve the modified diffusion equation (MDE) for this block.
double computeQ()
Compute and return partition function for the molecule.
void allocate(int ns, const Mesh< D > &mesh)
Allocate propagator arrays.
void computeHead()
Compute initial QField at head from tail QFields of sources.
void reallocate(int ns)
Reallocate memory used by this propagator.
Dynamically allocatable contiguous array template.
void allocate(int capacity)
Allocate the underlying C array.
Definition DArray.h:199
#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
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 mulVMany(DeviceArray< cudaReal > &a, DArray< DeviceArray< cudaReal > > const &vecs)
Multiply an undefined number of vectors pointwise, kernel wrapper.
Definition VecOpMisc.cu:525
void eqS(DeviceArray< cudaReal > &a, cudaReal const b, const int beginIdA, const int n)
Vector assignment, a[i] = b, kernel wrapper (cudaReal).
Definition VecOp.cu:1054
PSCF package top-level namespace.
Definition param_pc.dox:1
Utility classes for scientific computation.