PSCF v1.2
ExtGenFilmBase.tpp
1#ifndef PRDC_EXT_GEN_FILM_BASE_TPP
2#define PRDC_EXT_GEN_FILM_BASE_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 "ExtGenFilmBase.h"
12#include "prdc/crystal/SpaceGroup.h"
13#include <util/param/ParamComponent.h>
14
15namespace Pscf {
16namespace Prdc
17{
18
19 using namespace Util;
20
21 /*
22 * Constructor
23 */
24 template <int D>
27 normalVecCurrent_(),
28 chiBottomCurrent_(),
29 chiTopCurrent_(),
30 normalVecId_(-1),
31 interfaceThickness_(-1.0),
32 excludedThickness_(-1.0),
33 chiBottom_(),
34 chiTop_()
35 {
36 type_ = External;
37 isDependent_ = true;
38 }
39
40 /*
41 * Destructor
42 */
43 template <int D>
46
47 /*
48 * Read and initialize.
49 */
50 template <int D>
51 void ExtGenFilmBase<D>::readParameters(std::istream& in)
52 {
53 // First, read data defining the mask (quietly, with echo = false)
54 bool echo = ParamComponent::echo();
56 read(in, "normalVecId", normalVecId_);
57 read(in, "interfaceThickness", interfaceThickness_);
58 read(in, "excludedThickness", excludedThickness_);
59 double tmp;
60 readOptional(in, "fBulk", tmp); // will not be used
62
63 // Remove all of these parameters from this ParamComposite, since
64 // they are already managed by the MaskGenFilm ParamComposite
66
67 // Make sure inputs are valid
68 if (normalVecId_ > D || normalVecId_ < 0) {
69 UTIL_THROW("bad value for normalVecId, must be in [0,D)");
70 }
71 if (interfaceThickness_ > excludedThickness_) {
72 UTIL_THROW("excludedThickness must be larger than interfaceThickness");
73 }
74 if ((excludedThickness_ <= 0) || (interfaceThickness_ <= 0)) {
75 UTIL_THROW("excludedThickness and interfaceThickness must be >0");
76 }
77
78 // Allocate chiBottom_ and chiTop_
79 int nm = systemNMonomer();
80 chiBottom_.allocate(nm);
81 chiTop_.allocate(nm);
82
83 // Read chiBottom_ and chiTop_ arrays
84 readDArray(in, "chiBottom", chiBottom_, nm);
85 readDArray(in, "chiTop", chiTop_, nm);
86 }
87
88 /*
89 * Check that the system is compatible with this field
90 */
91 template <int D>
93 {
94 // Ensure that space group symmetry is compatible with the fields
95
96 // If chiBottom == chiTop, do nothing. All necessary checks have
97 // already been performed by the mask generator.
98 if (hasSymmetricWalls()) return;
99
100 // Otherwise, walls are asymmetric, so the space group must not
101 // have any operations that swap the two walls
102 std::string groupName = systemSpaceGroup();
103 SpaceGroup<D> group;
104 std::ifstream in;
105
106 // Open and read file containing space group's symmetry operations
107 readGroup(groupName, group);
108
109 // Make sure all symmetry operations are allowed
110 std::string msg = "Space group contains forbidden symmetry operations";
111 for (int i = 0; i < group.size(); i++) {
112 for (int j = 0; j < D; j++) {
113 int r = group[i].R(normalVecId_,j);
114 if (j == normalVecId_) {
115 if (r != 1) {
116 UTIL_THROW(msg.c_str());
117 }
118 } else { // j != normalVecId_
119 if (r != 0) {
120 UTIL_THROW(msg.c_str());
121 }
122 }
123 }
124 if (group[i].t(normalVecId_) != 0) {
125 UTIL_THROW(msg.c_str());
126 }
127 }
128 }
129
130 /*
131 * Check whether system has changed such that the field needs updating
132 */
133 template <int D>
135 {
136 // First, if fields have not yet been generated, decide whether they
137 // need to be generated (if athermal, no need).
138 if (!isGenerated()) {
139 if (isAthermal()) {
140 return false;
141 } else {
142 return true;
143 }
144 }
145
146 UTIL_CHECK(normalVecId_ >= 0);
147
148 // Check if chiBottom or chiTop have been changed
149 for (int i = 0; i < chiBottom_.capacity(); i++) {
150 if ((chiBottom_[i] != chiBottomCurrent_[i]) ||
151 (chiTop_[i] != chiTopCurrent_[i])) {
152 return true;
153 }
154 }
155
156 // If chiTop and chiBottom are unchanged and all zeros, no update needed
157 if (isAthermal()) return false;
158
159 // Check if system normalVec differ from normalVecCurrent_
160 if (normalVecCurrent_ == systemLatticeVector(normalVecId_)) {
161 return false;
162 } else {
163 return true;
164 }
165 }
166
167 /*
168 * Check whether or not the two walls are chemically identical using
169 * the chi array.
170 */
171 template <int D>
173 {
174 int nm = systemNMonomer();
175
176 UTIL_CHECK(nm > 0);
177 UTIL_CHECK(chiBottom_.capacity() == nm);
178 UTIL_CHECK(chiTop_.capacity() == nm);
179
180 for (int i = 0; i < nm; i++) {
181 if (fabs(chiBottom_[i]-chiTop_[i]) > 1e-7) {
182 return false;
183 }
184 }
185 return true;
186 }
187
188 /*
189 * Check whether or not the walls are athermal (only true if all values
190 * in chi array are zero)
191 */
192 template <int D>
194 {
195 int nm = systemNMonomer();
196
197 UTIL_CHECK(nm > 0);
198 UTIL_CHECK(chiBottom_.capacity() == nm);
199 UTIL_CHECK(chiTop_.capacity() == nm);
200
201 for (int i = 0; i < nm; i++) {
202 if ((fabs(chiBottom_[i]) >= 1e-7) || (fabs(chiTop_[i]) >= 1e-7)) {
203 return false;
204 }
205 }
206 return true;
207 }
208
209 /*
210 * Return specialized sweep parameter types to add to the Sweep object.
211 */
212 template <int D>
215 {
217 pTypes.append(ParameterType("chi_top", 1, *this));
218 pTypes.append(ParameterType("chi_bottom", 1, *this));
219 return pTypes;
220 }
221
222 /*
223 * Set the value of a specialized sweep parameter (chi_top or chi_bottom).
224 */
225 template <int D>
226 void ExtGenFilmBase<D>::setParameter(std::string name,
227 DArray<int> ids,
228 double value,
229 bool& success)
230 {
231 success = true;
232 bool wasAthermal = isAthermal();
233 if (name == "chi_top") {
234 chiTop_[ids[0]] = value;
235 } else if (name == "chi_bottom") {
236 chiBottom_[ids[0]] = value;
237 } else {
238 success = false;
239 }
240
241 // If walls were athermal but are not anymore, then the h fields
242 // have not been allocated. Thus, we must call allocate() again.
243 if (wasAthermal && success && (!isAthermal())) {
244 allocate();
245 }
246 }
247
248 /*
249 * Get the value of a specialized sweep parameter (chi_top or chi_bottom).
250 */
251 template <int D>
252 double ExtGenFilmBase<D>::getParameter(std::string name,
253 DArray<int> ids,
254 bool& success)
255 const
256 {
257 success = true;
258 if (name == "chi_top") {
259 return chiTop_[ids[0]];
260 } else if (name == "chi_bottom") {
261 return chiBottom_[ids[0]];
262 } else {
263 success = false;
264 return 0.0;
265 }
266 }
267
268}
269}
270#endif
Abstract base class for objects that generate fields for ImposedFields.
bool isAthermal() const
Are the walls athermal?
GArray< ParameterType > getParameterTypes()
Return specialized sweep parameter types to add to the Sweep object.
bool updateNeeded() const
Check whether system has changed such that the fields needs updating.
void checkCompatibility()
Check that the system is compatible with these fields.
bool hasSymmetricWalls() const
Are the walls chemically identical?
void setParameter(std::string name, DArray< int > ids, double value, bool &success)
Set the value of a specialized sweep parameter.
void readParameters(std::istream &in)
Read and initialize.
double getParameter(std::string name, DArray< int > ids, bool &success) const
Get the value of a specialized sweep parameter.
bool isDependent_
Is this object dependent on the parameters of another FieldGenerator?
Type type_
Type of field (Mask, External, Both, or None)
Crystallographic space group.
Definition FieldIoReal.h:23
int size() const
Return number of elements in group (i.e., the order of the group).
Dynamically allocatable contiguous array template.
An automatically growable array, analogous to a std::vector.
Definition GArray.h:34
void append(Data const &data)
Append an element to the end of the sequence.
Definition GArray.h:306
static bool echo()
Get echo parameter.
static void setEcho(bool echo=true)
Enable or disable echoing for all subclasses of ParamComponent.
void resetParam()
Resets ParamComposite to its empty state.
#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
PSCF package top-level namespace.
Definition param_pc.dox:1
Utility classes for scientific computation.
Declaration of a specialized sweep parameter type.