PSCF v1.2
SweepParameter.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 <r1d/sweep/SweepParameter.h>
9#include <r1d/solvers/Block.h>
10#include <r1d/solvers/Mixture.h>
11#include <r1d/solvers/Polymer.h>
12#include <r1d/System.h>
13#include <pscf/inter/Interaction.h>
14#include <pscf/sweep/ParameterModifier.h>
15#include <util/containers/FSArray.h>
16#include <util/global.h>
17
18#include <algorithm>
19#include <iomanip>
20
21namespace Pscf {
22namespace R1d {
23
24 using namespace Util;
25
26 /*
27 * Default constructor.
28 */
30 : type_(SweepParameter::Null),
31 nId_(0),
32 id_(),
33 initial_(0.0),
34 change_(0.0),
35 systemPtr_(0),
36 parameterTypesPtr_(0),
37 parameterTypeId_(-1)
38 {}
39
40 /*
41 * Constructor, creates association with system.
42 */
44 : type_(SweepParameter::Null),
45 nId_(0),
46 id_(),
47 initial_(0.0),
48 change_(0.0),
49 systemPtr_(&system),
50 parameterTypesPtr_(0),
51 parameterTypeId_(-1)
52 {}
53
54 /*
55 * Read type, set nId and allocate id_ array.
56 */
57 void SweepParameter::readParamType(std::istream& in)
58 {
59 std::string buffer;
60 in >> buffer;
61 std::transform(buffer.begin(), buffer.end(),
62 buffer.begin(), ::tolower);
63
64 if (buffer == "block" || buffer == "block_length") {
65 type_ = Block;
66 nId_ = 2; // polymer and block identifiers
67 } else if (buffer == "chi") {
68 type_ = Chi;
69 nId_ = 2; // two monomer type identifiers
70 } else if (buffer == "kuhn") {
71 type_ = Kuhn;
72 nId_ = 1; // monomer type identifier
73 } else if (buffer == "phi_polymer") {
74 type_ = Phi_Polymer;
75 nId_ = 1; //species identifier.
76 } else if (buffer == "phi_solvent") {
77 type_ = Phi_Solvent;
78 nId_ = 1; //species identifier.
79 } else if (buffer == "mu_polymer") {
80 type_ = Mu_Polymer;
81 nId_ = 1; //species identifier.
82 } else if (buffer == "mu_solvent") {
83 type_ = Mu_Solvent;
84 nId_ = 1; //species identifier.
85 } else if (buffer == "solvent" || buffer == "solvent_size") {
86 type_ = Solvent;
87 nId_ = 1; //species identifier.
88 } else if (buffer == "cell_param") {
89 type_ = Cell_Param;
90 nId_ = 1; //lattice parameter identifier.
91 } else {
92 // Search in parameterTypes array for this sweep parameter
93 bool found = false;
94 for (int i = 0; i < parameterTypesPtr_->size(); i++) {
95 ParameterType& pType = (*parameterTypesPtr_)[i];
96 if (buffer == pType.name) {
97 type_ = Special;
98 nId_ = pType.nId;
99 parameterTypeId_ = i;
100 found = true;
101 break;
102 }
103 }
104 if (!found) {
105 std::string msg;
106 msg = "Invalid SweepParameter::ParamType value: " + buffer;
107 UTIL_THROW(msg.c_str());
108 }
109 }
110
111 if (id_.isAllocated()) id_.deallocate();
112 id_.allocate(nId_);
113 }
114
115 /*
116 * Write type enum value
117 */
118 void SweepParameter::writeParamType(std::ostream& out) const
119 {
120 out << type();
121 }
122
123 /*
124 * Get the ParameterType object for a specialized sweep parameter
125 */
127 {
129 return (*parameterTypesPtr_)[parameterTypeId_];
130 }
131
132 /*
133 * Get the current value from the parent system.
134 */
136 {
137 initial_ = get_();
138 }
139
140 /*
141 * Set a new value in the parent system.
142 */
143 void SweepParameter::update(double newVal)
144 {
145 set_(newVal);
146 }
147
148 /*
149 * Get string representation of type enum value.
150 */
151 std::string SweepParameter::type() const
152 {
153 if (type_ == Block) {
154 return "block";
155 } else if (type_ == Chi) {
156 return "chi";
157 } else if (type_ == Kuhn) {
158 return "kuhn";
159 } else if (type_ == Phi_Polymer) {
160 return "phi_polymer";
161 } else if (type_ == Phi_Solvent) {
162 return "phi_solvent";
163 } else if (type_ == Mu_Polymer) {
164 return "mu_polymer";
165 } else if (type_ == Mu_Solvent) {
166 return "mu_solvent";
167 } else if (type_ == Solvent) {
168 return "solvent_size";
169 } else if (type_ == Special) {
170 return parameterType().name;
171 } else {
172 UTIL_THROW("Invalid type_ in accessor SweepParameter::type().");
173 }
174 }
175
176 double SweepParameter::get_()
177 {
178 if (type_ == Block) {
179 return systemPtr_->mixture().polymer(id(0)).block(id(1)).length();
180 } else if (type_ == Chi) {
181 return systemPtr_->interaction().chi(id(0), id(1));
182 } else if (type_ == Kuhn) {
183 return systemPtr_->mixture().monomer(id(0)).kuhn();
184 } else if (type_ == Phi_Polymer) {
185 return systemPtr_->mixture().polymer(id(0)).phi();
186 } else if (type_ == Phi_Solvent) {
187 return systemPtr_->mixture().solvent(id(0)).phi();
188 } else if (type_ == Mu_Polymer) {
189 return systemPtr_->mixture().polymer(id(0)).mu();
190 } else if (type_ == Mu_Solvent) {
191 return systemPtr_->mixture().solvent(id(0)).mu();
192 } else if (type_ == Solvent) {
193 return systemPtr_->mixture().solvent(id(0)).size();
194 } else if (type_ == Special) {
195 ParameterModifier* modifier = parameterType().modifierPtr;
196 std::string name = parameterType().name;
197 return modifier->getParameter(name,id_);
198 } else {
199 UTIL_THROW("Invalid type_ in SweepParameter::get_.");
200 }
201 }
202
203 void SweepParameter::set_(double newVal)
204 {
205 if (type_ == Block) {
206 systemPtr_->mixture().polymer(id(0)).block(id(1)).setLength(newVal);
207 } else if (type_ == Chi) {
208 systemPtr_->interaction().setChi(id(0), id(1), newVal);
209 } else if (type_ == Kuhn) {
210 systemPtr_->mixture().setKuhn(id(0), newVal);
211 } else if (type_ == Phi_Polymer) {
212 systemPtr_->mixture().polymer(id(0)).setPhi(newVal);
213 } else if (type_ == Phi_Solvent) {
214 systemPtr_->mixture().solvent(id(0)).setPhi(newVal);
215 } else if (type_ == Mu_Polymer) {
216 systemPtr_->mixture().polymer(id(0)).setMu(newVal);
217 } else if (type_ == Mu_Solvent) {
218 systemPtr_->mixture().solvent(id(0)).setMu(newVal);
219 } else if (type_ == Solvent) {
220 systemPtr_->mixture().solvent(id(0)).setSize(newVal);
221 } else if (type_ == Special) {
222 ParameterModifier* modifier = parameterType().modifierPtr;
223 std::string name = parameterType().name;
224 return modifier->setParameter(name,id_,newVal);
225 } else {
226 UTIL_THROW("Invalid type_ in SweepParameter::set_.");
227 }
228 }
229
230 // Definitions of operators, with no explicit instantiations.
231
238 std::istream& operator >> (std::istream& in,
239 SweepParameter& param)
240 {
241 // Read the parameter type.
242 param.readParamType(in);
243 // Read the identifiers associated with this parameter type.
244 for (int i = 0; i < param.nId_; ++i) {
245 in >> param.id_[i];
246 }
247 // Read in the range in the parameter to sweep over
248 in >> param.change_;
249
250 return in;
251 }
252
259 std::ostream& operator << (std::ostream& out,
260 SweepParameter const & param)
261 {
262 param.writeParamType(out);
263 out << " ";
264 for (int i = 0; i < param.nId_; ++i) {
265 out << param.id(i);
266 out << " ";
267 }
268 out << param.change_;
269
270 return out;
271 }
272
273}
274}
DMatrix< double > const & chi() const
Return the chi matrix by const reference.
void setChi(int i, int j, double chi)
Change one element of the chi matrix.
Polymer & polymer(int id)
Get a polymer object.
Monomer const & monomer(int id) const
Get a Monomer type descriptor (const reference).
Solvent & solvent(int id)
Set a solvent solver object.
double kuhn() const
Statistical segment length (random walk step size).
Definition Monomer.h:125
Block within a branched polymer.
void setKuhn(int monomerId, double kuhn)
Reset statistical segment length for one monomer type.
Solver and descriptor for a solvent species.
Class for storing data about an individual sweep parameter.
std::string type() const
Return a string representation of the parameter type.
void writeParamType(std::ostream &out) const
Write the parameter type to an output stream.
SweepParameter()
Default constructor.
bool isSpecialized() const
Is this SweepParameter a specialized parameter type?
void getInitial()
Store the pre-sweep value of the corresponding parameter.
int id(int i) const
Get a id for a sub-object or element to which this is applied.
ParameterType & parameterType() const
Get the ParameterType object for a specialized sweep parameter.
void update(double newVal)
Update the corresponding parameter value in the system.
Main class in SCFT simulation of one system.
Definition r1d/System.h:65
Interaction & interaction()
Get interaction (i.e., excess free energy) by reference.
Definition r1d/System.h:579
Mixture & mixture()
Get Mixture by reference.
Definition r1d/System.h:567
void allocate(int capacity)
Allocate the underlying C array.
Definition DArray.h:199
void deallocate()
Dellocate the underlying C array.
Definition DArray.h:217
bool isAllocated() const
Return true if this DArray has been allocated, false otherwise.
Definition DArray.h:247
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
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.