PSCF v1.2
rpg/scft/sweep/SweepParameter.tpp
1#ifndef RPG_SWEEP_PARAMETER_TPP
2#define RPG_SWEEP_PARAMETER_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 <rpg/System.h>
12#include <rpg/solvers/Mixture.h>
13#include <rpg/solvers/Polymer.h>
14#include <rpg/solvers/Block.h>
15
16#include <prdc/crystal/UnitCell.h>
17
18#include <pscf/inter/Interaction.h>
19#include <pscf/sweep/ParameterModifier.h>
20#include <util/containers/FSArray.h>
21#include <util/global.h>
22
23#include <algorithm>
24#include <iomanip>
25
26namespace Pscf {
27namespace Rpg {
28
29 using namespace Util;
30 using namespace Pscf::Prdc;
31
32 /*
33 * Default constructor.
34 */
35 template <int D>
37 : type_(SweepParameter<D>::Null),
38 nId_(0),
39 id_(),
40 initial_(0.0),
41 change_(0.0),
42 systemPtr_(0),
43 parameterTypesPtr_(0),
44 parameterTypeId_(-1)
45 {}
46
47 /*
48 * Constructor, creates association with system.
49 */
50 template <int D>
52 : type_(SweepParameter<D>::Null),
53 nId_(0),
54 id_(),
55 initial_(0.0),
56 change_(0.0),
57 systemPtr_(0),
58 parameterTypesPtr_(0),
59 parameterTypeId_(-1)
60 {}
61
62 /*
63 * Read type, set nId and allocate id_ array.
64 */
65 template <int D>
66 void SweepParameter<D>::readParamType(std::istream& in)
67 {
68 std::string buffer;
69 in >> buffer;
70 std::transform(buffer.begin(), buffer.end(),
71 buffer.begin(), ::tolower);
72
73 if (buffer == "block" || buffer == "block_length") {
74 type_ = Block;
75 nId_ = 2; // polymer and block identifiers
76 } else if (buffer == "chi") {
77 type_ = Chi;
78 nId_ = 2; // two monomer type identifiers
79 } else if (buffer == "kuhn") {
80 type_ = Kuhn;
81 nId_ = 1; // monomer type identifier
82 } else if (buffer == "phi_polymer") {
83 type_ = Phi_Polymer;
84 nId_ = 1; //species identifier.
85 } else if (buffer == "phi_solvent") {
86 type_ = Phi_Solvent;
87 nId_ = 1; //species identifier.
88 } else if (buffer == "mu_polymer") {
89 type_ = Mu_Polymer;
90 nId_ = 1; //species identifier.
91 } else if (buffer == "mu_solvent") {
92 type_ = Mu_Solvent;
93 nId_ = 1; //species identifier.
94 } else if (buffer == "solvent" || buffer == "solvent_size") {
95 type_ = Solvent;
96 nId_ = 1; //species identifier.
97 } else if (buffer == "cell_param") {
98 type_ = Cell_Param;
99 nId_ = 1; //lattice parameter identifier.
100 } else {
101 // Search in parameterTypes array for this sweep parameter
102 bool found = false;
103 for (int i = 0; i < parameterTypesPtr_->size(); i++) {
104 ParameterType& pType = (*parameterTypesPtr_)[i];
105 if (buffer == pType.name) {
106 type_ = Special;
107 nId_ = pType.nId;
108 parameterTypeId_ = i;
109 found = true;
110 break;
111 }
112 }
113 if (!found) {
114 std::string msg;
115 msg = "Invalid SweepParameter::ParamType value: " + buffer;
116 UTIL_THROW(msg.c_str());
117 }
118 }
119
120 if (id_.isAllocated()) id_.deallocate();
121 id_.allocate(nId_);
122
123 }
124
125 /*
126 * Write type enum value
127 */
128 template <int D>
129 void SweepParameter<D>::writeParamType(std::ostream& out) const
130 {
131 out << type();
132 }
133
134 /*
135 * Get the ParameterType object for a specialized sweep parameter
136 */
137 template <int D>
139 {
140 UTIL_CHECK(isSpecialized());
141 return (*parameterTypesPtr_)[parameterTypeId_];
142 }
143
144 /*
145 * Get the current value from the parent system.
146 */
147 template <int D>
149 {
150 initial_ = get_();
151 }
152
153 /*
154 * Set a new value in the parent system.
155 */
156 template <int D>
157 void SweepParameter<D>::update(double newVal)
158 {
159 set_(newVal);
160 }
161
162 /*
163 * Get string representation of type enum value.
164 */
165 template <int D>
166 std::string SweepParameter<D>::type() const
167 {
168 if (type_ == Block) {
169 return "block";
170 } else if (type_ == Chi) {
171 return "chi";
172 } else if (type_ == Kuhn) {
173 return "kuhn";
174 } else if (type_ == Phi_Polymer) {
175 return "phi_polymer";
176 } else if (type_ == Phi_Solvent) {
177 return "phi_solvent";
178 } else if (type_ == Mu_Polymer) {
179 return "mu_polymer";
180 } else if (type_ == Mu_Solvent) {
181 return "mu_solvent";
182 } else if (type_ == Solvent) {
183 return "solvent_size";
184 } else if (type_ == Cell_Param) {
185 return "cell_param";
186 } else if (type_ == Special) {
187 return parameterType().name;
188 } else {
189 UTIL_THROW("This should never happen.");
190 }
191 }
192
193 template <int D>
195 {
196 if (type_ == Block) {
197 return systemPtr_->mixture().polymer(id(0)).block(id(1)).length();
198 } else if (type_ == Chi) {
199 return systemPtr_->interaction().chi(id(0),id(1));
200 } else if (type_ == Kuhn) {
201 return systemPtr_->mixture().monomer(id(0)).kuhn();
202 } else if (type_ == Phi_Polymer) {
203 return systemPtr_->mixture().polymer(id(0)).phi();
204 } else if (type_ == Phi_Solvent) {
205 return systemPtr_->mixture().solvent(id(0)).phi();
206 } else if (type_ == Mu_Polymer) {
207 return systemPtr_->mixture().polymer(id(0)).mu();
208 } else if (type_ == Mu_Solvent) {
209 return systemPtr_->mixture().solvent(id(0)).mu();
210 } else if (type_ == Solvent) {
211 return systemPtr_->mixture().solvent(id(0)).size();
212 } else if (type_ == Cell_Param) {
213 return systemPtr_->unitCell().parameter(id(0));
214 } else if (type_ == Special) {
215 ParameterModifier* modifier = parameterType().modifierPtr;
216 std::string name = parameterType().name;
217 return modifier->getParameter(name,id_);
218 } else {
219 UTIL_THROW("This should never happen.");
220 }
221 }
222
223 template <int D>
224 void SweepParameter<D>::set_(double newVal)
225 {
226 if (type_ == Block) {
227 systemPtr_->mixture().polymer(id(0)).block(id(1)).setLength(newVal);
228 } else if (type_ == Chi) {
229 systemPtr_->interaction().setChi(id(0), id(1), newVal);
230 } else if (type_ == Kuhn) {
231 systemPtr_->mixture().setKuhn(id(0), newVal);
232 } else if (type_ == Phi_Polymer) {
233 systemPtr_->mixture().polymer(id(0)).setPhi(newVal);
234 } else if (type_ == Phi_Solvent) {
235 systemPtr_->mixture().solvent(id(0)).setPhi(newVal);
236 } else if (type_ == Mu_Polymer) {
237 systemPtr_->mixture().polymer(id(0)).setMu(newVal);
238 } else if (type_ == Mu_Solvent) {
239 systemPtr_->mixture().solvent(id(0)).setMu(newVal);
240 } else if (type_ == Solvent) {
241 systemPtr_->mixture().solvent(id(0)).setSize(newVal);
242 } else if (type_ == Cell_Param) {
243 FSArray<double,6> params = systemPtr_->unitCell().parameters();
244 params[id(0)] = newVal;
245 systemPtr_->setUnitCell(params);
246 } else if (type_ == Special) {
247 ParameterModifier* modifier = parameterType().modifierPtr;
248 std::string name = parameterType().name;
249 return modifier->setParameter(name,id_,newVal);
250 } else {
251 UTIL_THROW("This should never happen.");
252 }
253 }
254
255 template <int D>
256 template <class Archive>
257 void SweepParameter<D>::serialize(Archive ar, const unsigned int version)
258 {
259 serializeEnum(ar, type_, version);
260 ar & nId_;
261 for (int i = 0; i < nId_; ++i) {
262 ar & id_[i];
263 }
264 ar & initial_;
265 ar & change_;
266 }
267
268 // Definitions of operators, with no explicit instantiations.
269
276 template <int D>
277 std::istream& operator >> (std::istream& in,
278 SweepParameter<D>& param)
279 {
280 // Read the parameter type.
281 param.readParamType(in);
282 // Read the identifiers associated with this parameter type.
283 for (int i = 0; i < param.nId_; ++i) {
284 in >> param.id_[i];
285 }
286 // Read in the range in the parameter to sweep over
287 in >> param.change_;
288
289 return in;
290 }
291
298 template <int D>
299 std::ostream& operator << (std::ostream& out,
300 SweepParameter<D> const & param)
301 {
302 param.writeParamType(out);
303 out << " ";
304 for (int i = 0; i < param.nId_; ++i) {
305 out << param.id(i);
306 out << " ";
307 }
308 out << param.change_;
309
310 return out;
311 }
312
313}
314}
315
316#endif
Base class allowing subclasses to define sweepable parameters.
virtual void setParameter(std::string name, DArray< int > ids, double value, bool &success)
Set the value of a specialized sweep parameter.
virtual double getParameter(std::string name, DArray< int > ids, bool &success) const
Get the value of a specialized sweep parameter.
Block within a branched polymer.
Solver and descriptor for a solvent species.
Class for storing data about an individual sweep parameter.
void getInitial()
Store the pre-sweep value of the corresponding parameter.
void serialize(Archive ar, const unsigned int version)
Serialize to or from an archive.
void writeParamType(std::ostream &out) const
Write the parameter type to an output stream.
int id(int i) const
Get a id for a sub-object or element to which this is applied.
void update(double newVal)
Update the corresponding parameter value in the system.
std::string type() const
Return a string representation of the parameter type.
ParameterType & parameterType() const
Get the ParameterType object for a specialized sweep parameter.
Main class for calculations that represent one system.
Definition rpg/System.h:107
A fixed capacity (static) contiguous array with a variable logical size.
Definition rpg/System.h:28
File containing preprocessor macros for error handling.
#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
void serializeEnum(Archive &ar, T &data, const unsigned int version=0)
Serialize an enumeration value.
Definition serialize.h:59
Periodic fields and crystallography.
Definition CField.cpp:11
PSCF package top-level namespace.
Definition param_pc.dox:1
Utility classes for scientific computation.
std::istream & operator>>(std::istream &in, Pair< Data > &pair)
Input a Pair from an istream.
Definition Pair.h:44
std::ostream & operator<<(std::ostream &out, const Pair< Data > &pair)
Output a Pair to an ostream, without line breaks.
Definition Pair.h:57
Declaration of a specialized sweep parameter type.