PSCF v1.2
FieldBasisConverter.tpp
1#ifndef PRDC_CPU_FIELD_BASIS_CONVERTER_TPP
2#define PRDC_CPU_FIELD_BASIS_CONVERTER_TPP
3
4/*
5* PSCF Package
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 "FieldBasisConverter.h"
12#include <prdc/cpu/RField.h>
13#include <prdc/cpu/RFieldDft.h>
14#include <util/containers/DArray.h>
15
16namespace Pscf {
17namespace Prdc {
18namespace Cpu {
19
20 using namespace Util;
21
22 /*
23 * Default constructor.
24 */
25 template <int D>
27 : basis_(),
28 nMonomer_(0)
29 {}
30
31 /*
32 * Constructor, creates basis.
33 */
34 template <int D>
36 : basis_(),
37 nMonomer_(0)
38 { setBasis(basis); }
39
40 /*
41 * Destructor.
42 */
43 template <int D>
46
47 /*
48 * Set the basis.
49 */
50 template <int D>
52 {
53 UTIL_CHECK(basis.isAllocated());
54 UTIL_CHECK(basis.capacity1() > 1);
55 UTIL_CHECK(basis.capacity1() == basis.capacity2());
56
57 basis_ = basis;
58 nMonomer_ = basis.capacity1();
59
60 UTIL_CHECK(basis_.capacity1() == nMonomer_);
61 UTIL_CHECK(basis_.capacity2() == nMonomer_);
62 }
63
64 template <int D>
65 double FieldBasisConverter<D>::maxBasisError(double normSq) const
66 {
67 UTIL_CHECK(nMonomer_ > 1);
68 UTIL_CHECK(normSq > 0.0);
69
70 double error = 0.0;
71 double maxError = 0.0;
72 int i, j, k;
73 for (i = 0; i < nMonomer_; ++i) {
74 for (j = 0; j <= i; ++j) {
75 error = 0.0;
76 for (k = 0; k < nMonomer_; ++k) {
77 error += basis_(i,k)*basis_(j,k);
78 }
79 if (i == j) {
80 error = error - normSq;
81 }
82 error = std::abs(error);
83 if (error > maxError) {
84 maxError = error;
85 }
86 }
87 }
88 return maxError;
89 }
90
91 /*
92 * Compute pointwise components of r-grid fields in basis.
93 */
94 template <int D>
95 void
97 DArray< RField<D> > & out,
98 double prefactor) const
99 {
100 // Preconditions
101 UTIL_CHECK(nMonomer_ > 1);
102 UTIL_CHECK(in.isAllocated());
103 UTIL_CHECK(out.isAllocated());
104 UTIL_CHECK(in.capacity() == nMonomer_);
105 UTIL_CHECK(out.capacity() == nMonomer_);
106 int meshSize = in[0].capacity();
107 for (int i=0; i < nMonomer_; ++i) {
108 UTIL_CHECK(in[i].capacity() == meshSize);
109 UTIL_CHECK(out[i].capacity() == meshSize);
110 }
111
112 double vec;
113 int i, j, k;
114
115 // Loop over components in basis
116 for (i = 0; i < nMonomer_; ++i) {
117 RField<D>& wo = out[i];
118
119 // Initialize wo = out[i] to zero
120 for (k = 0; k < meshSize; ++k) {
121 wo[k] = 0.0;
122 }
123
124 // Loop over monomer types and mesh points
125 for (j = 0; j < nMonomer_; ++j) {
126 vec = basis_(i, j)*prefactor;
127 RField<D> const & wi = in[j];
128 for (k = 0; k < meshSize; ++k) {
129 wo[k] += vec*wi[k];
130 }
131 } // for j < nMonomer_
132
133 } // for i < nMonomer_
134
135 }
136
137 /*
138 * Compute r-grid fields from components in basis.
139 */
140 template <int D>
141 void
143 DArray<RField<D>> & out,
144 double prefactor) const
145 {
146 // Preconditions
147 UTIL_CHECK(nMonomer_ > 1);
148 UTIL_CHECK(in.isAllocated());
149 UTIL_CHECK(out.isAllocated());
150 UTIL_CHECK(in.capacity() == nMonomer_);
151 UTIL_CHECK(out.capacity() == nMonomer_);
152 int meshSize = in[0].capacity();
153 for (int i=0; i < nMonomer_; ++i) {
154 UTIL_CHECK(in[i].capacity() == meshSize);
155 UTIL_CHECK(out[i].capacity() == meshSize);
156 }
157
158 double vec;
159 int i, j, k;
160
161 // Loop over monomer types
162 for (i = 0; i < nMonomer_; ++i) {
163 RField<D>& wo = out[i];
164
165 // Initialize wo = out[i] to zero
166 for (k = 0; k < meshSize; ++k) {
167 wo[k] = 0.0;
168 }
169
170 // Loop over components in basis
171 for (j = 0; j < nMonomer_; ++j) {
172 vec = basis_(j, i)*prefactor;
173 RField<D> const & wi = in[j];
174 // Loop over mesh points
175 for (k = 0; k < meshSize; ++k) {
176 wo[k] += vec*wi[k];
177 }
178 } // for j < nMonomer_
179
180 } // for i < nMonomer_
181
182 }
183
184}
185}
186}
187#endif
void convertToBasis(DArray< RField< D > > const &in, DArray< RField< D > > &out, double prefactor=1.0) const
Convert a set of monomer fields to field basis components.
void setBasis(DMatrix< double > basis)
Set or reset the basis after construction.
double maxBasisError(double normSq=1.0) const
Check validity (orthogonality and normalization) of the basis.
void convertFromBasis(DArray< RField< D > > const &in, DArray< RField< D > > &out, double prefactor=1.0) const
Convert a set of field basis components to monomer fields.
Field of real double precision values on an FFT mesh.
Dynamically allocatable contiguous array template.
Dynamically allocated Matrix.
Definition DMatrix.h:25
bool isAllocated() const
Return true if the DMatrix has been allocated, false otherwise.
Definition DMatrix.h:202
int capacity2() const
Get number of columns (range of the second array index).
Definition Matrix.h:143
int capacity1() const
Get number of rows (range of the first array index).
Definition Matrix.h:136
#define UTIL_CHECK(condition)
Assertion macro suitable for serial or parallel production code.
Definition global.h:68
PSCF package top-level namespace.
Definition param_pc.dox:1
Utility classes for scientific computation.