PSCF v1.2
MaskGenFilmBase.cpp
1/*
2* PSCF - Polymer Self-Consistent Field Theory
3*
4* Copyright 2016 - 2022, The Regents of the University of Minnesota
5* Distributed under the terms of the GNU General Public License.
6*/
7
8#include "MaskGenFilmBase.tpp"
9#include <prdc/crystal/paramIdConversions.h>
10
11namespace Pscf {
12namespace Prdc
13{
14
15 using namespace Util;
16
17 // Explicit Specializations for checkLatticeVectors
18
19 /*
20 * Check that user-defined lattice basis vectors are compatible with
21 * the thin film constraint.
22 *
23 * In 1D, there is nothing to do; the lattice basis vector is correct in
24 * all cases.
25 */
26 template <>
28 {} // do nothing
29
30
31 // In 2D, we require that gamma = 90°.
32 template <>
34 {
35 RealVec<2> a, b;
36 a = systemLatticeVector(0);
37 b = systemLatticeVector(1);
38
39 double gamma = dot(a,b);
40 if (gamma > 1e-8) { // Dot product between a and b should be 0
41 UTIL_THROW("ERROR: Lattice basis vectors must be orthogonal");
42 }
43 }
44
45 /*
46 * In 3D, we require that there be one lattice basis vector that is
47 * orthogonal to the walls (parameter with index normalVecId), and two
48 * that are parallel to the walls.
49 */
50 template <>
52 {
53 RealVec<3> a, b, c;
54 a = systemLatticeVector(0);
55 b = systemLatticeVector(1);
56 c = systemLatticeVector(2);
57
58 double alpha, beta, gamma;
59 gamma = dot(a,b);
60 beta = dot(a,c);
61 alpha = dot(b,c);
62
63 if (normalVecId() == 0) {
64 if (beta > 1e-8 || gamma > 1e-8) {
65 UTIL_THROW("ERROR: beta and gamma must be 90 degrees");
66 }
67 } else if (normalVecId() == 1) {
68 if (alpha > 1e-8 || gamma > 1e-8) {
69 UTIL_THROW("ERROR: alpha and gamma must be 90 degrees");
70 }
71 } else { // normalVecId == 2
72 if (alpha > 1e-8 || beta > 1e-8) {
73 UTIL_THROW("ERROR: alpha and beta must be 90 degrees");
74 }
75 }
76 }
77
78 // Explicit Specializations for modifyFlexibleParams
79
80 /*
81 * Modifies a flexibleParams array to be compatible with this mask. (1D)
82 */
83 template <>
86 UnitCell<1> const & cell) const
87 {
88 UTIL_CHECK(current.size() == cell.nParameter());
89 UTIL_CHECK(cell.nParameter() == 1);
90
91 if ((current[0]) && (!hasFBulk())) {
92 // If user specified that the parameter is flexible but
93 // did not provide a value for fBulk, the parameter will
94 // be changed to rigid
95 current[0] = false;
96 Log::file()
97 << "Warning - The lattice parameter is not allowed "
98 << "to be flexible\n for a 1D thin film system "
99 << "unless fBulk is provided."
100 << std::endl;
101 }
102
103 return current;
104 }
105
106 /*
107 * Modifies a flexibleParams array to be compatible with this mask. (2D)
108 */
109 template <>
112 UnitCell<2> const & cell) const
113 {
114 UTIL_CHECK(current.size() == cell.nParameter());
115
116 FSArray<bool,6> updated = current;
117
118 // In 2D problems, gamma is fixed at 90 degrees, and the length
119 // of the basis vector with index normalVecId is fixed as well,
120 // unless the input parameter fBulk is provided, in which case
121 // it may be flexible. If fBulk is not provided and the lattice
122 // system is square, hexagonal, or rhombic, then no parameters
123 // can be flexible under these conditions. If rectangular or
124 // oblique, then the length of the basis vector parallel to the
125 // film may be flexible.
126
127 // First, set gamma to rigid
128 if (cell.lattice() == UnitCell<2>::Rhombic) updated[1] = false;
129 if (cell.lattice() == UnitCell<2>::Oblique) updated[2] = false;
130
131 // Next, set the parameter corresponding to normalVecId to rigid
132 // (unless fBulk was provided)
133 if (!hasFBulk()) {
134 int reducedId = convertFullParamIdToReduced<2>(normalVecId_,
135 cell.lattice());
136 updated[reducedId] = false;
137 }
138
139 // Check if the number of flexible lattice parameters has changed
140 // during this function, and print a warning if so.
141 bool warn = false;
142 for (int i = 0; i < updated.size(); i++) {
143 if (updated[i] != current[i]) {
144 warn = true;
145 break;
146 }
147 }
148 if (warn) {
149 Log::file()
150 << "***\n"
151 << "Notice - Some lattice parameters will be held constant\n"
152 << "to comply with the thin film constraint.\n"
153 << "***" << std::endl;
154 }
155
156 return updated;
157 }
158
159 /*
160 * Modifies a flexibleParams array to be compatible with this mask. (3D)
161 */
162 template <>
165 UnitCell<3> const & cell) const
166 {
167 UTIL_CHECK(current.size() == cell.nParameter());
168
169 FSArray<bool,6> updated = current;
170
171 // If fBulk is not provided, then there can be up to 3 flexible
172 // lattice parameters in 3D: the length of the two lattice vectors
173 // that are not normalVecId, and the angle between them. The other
174 // two angles must be 90 degrees, and the length of normalVecId is
175 // fixed. The crystal system determines which parameters are flexible.
176 //
177 // If fBulk is provided, then the length of the basis vector with
178 // index normalVecId may also be flexible.
179
180 // First, set angles to rigid if necessary
181 if (cell.lattice() == UnitCell<3>::Rhombohedral) {
182
183 Log::file() << "Rhombohedral lattice systems are not compatible "
184 << "with a thin film constraint.\n"
185 << "See thin film documentation for more details.\n";
186 UTIL_THROW("Cannot use rhombohedral lattice in a thin film system.");
187
188 } else if (cell.lattice() == UnitCell<3>::Monoclinic) {
189
190 // Beta can be flexible in a monoclinic lattice iff normalVecId = 1
191 if (normalVecId_ != 1) {
192 updated[3] = false;
193 }
194
195 } else if (cell.lattice() == UnitCell<3>::Triclinic) {
196
197 // Set all angles to rigid
198 updated[3] = false;
199 updated[4] = false;
200 updated[5] = false;
201
202 // Allow the angle between the two lattice basis vectors that are
203 // not normalVecId to be flexible
204 updated[normalVecId_+3] = current[normalVecId_+3];
205
206 }
207
208 // Next, set the parameter corresponding to normalVecId to rigid
209 // (unless fBulk was provided)
210 if (!hasFBulk()) {
211 int reducedId = convertFullParamIdToReduced<3>(normalVecId_,
212 cell.lattice());
213 updated[reducedId] = false;
214 }
215
216 // Check if the number of flexible lattice parameters has changed
217 // during this function, and print a warning if so.
218 bool warn = false;
219 for (int i = 0; i < updated.size(); i++) {
220 if (updated[i] != current[i]) {
221 warn = true;
222 break;
223 }
224 }
225 if (warn) {
226 Log::file()
227 << "***\n"
228 << "Notice - Some lattice parameters will be held constant\n"
229 << "to comply with the thin film constraint.\n"
230 << "***" << std::endl;
231 }
232
233 return updated;
234 }
235
236 // Class declarations
237 template class MaskGenFilmBase<1>;
238 template class MaskGenFilmBase<2>;
239 template class MaskGenFilmBase<3>;
240}
241}
FSArray< bool, 6 > modifyFlexibleParams(FSArray< bool, 6 > current, UnitCell< D > const &cell) const
Modifies a flexibleParams array to be compatible with this mask.
void checkLatticeVectors() const
Check that lattice vectors are compatible with thin film constraint.
A fixed capacity (static) contiguous array with a variable logical size.
Definition rpg/System.h:28
int size() const
Return logical size of this array (i.e., number of elements).
Definition FSArray.h:207
static std::ostream & file()
Get log ostream by reference.
Definition Log.cpp:57
#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
T dot(Vec< D, T > const &v1, Vec< D, T > const &v2)
Return dot product of two vectors.
Definition Vec.h:285
Utility classes for scientific computation.