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