OpenMD 3.0
Molecular Dynamics in the Open
Loading...
Searching...
No Matches
ThermoIntegrationForceModifier.cpp
1/*
2 * Copyright (c) 2004-present, The University of Notre Dame. All rights
3 * reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions are met:
7 *
8 * 1. Redistributions of source code must retain the above copyright notice,
9 * this list of conditions and the following disclaimer.
10 *
11 * 2. Redistributions in binary form must reproduce the above copyright notice,
12 * this list of conditions and the following disclaimer in the documentation
13 * and/or other materials provided with the distribution.
14 *
15 * 3. Neither the name of the copyright holder nor the names of its
16 * contributors may be used to endorse or promote products derived from
17 * this software without specific prior written permission.
18 *
19 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
20 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
21 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
22 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
23 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
24 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
25 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
26 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
27 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
28 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29 * POSSIBILITY OF SUCH DAMAGE.
30 *
31 * SUPPORT OPEN SCIENCE! If you use OpenMD or its source code in your
32 * research, please cite the appropriate papers when you publish your
33 * work. Good starting points are:
34 *
35 * [1] Meineke, et al., J. Comp. Chem. 26, 252-271 (2005).
36 * [2] Fennell & Gezelter, J. Chem. Phys. 124, 234104 (2006).
37 * [3] Sun, Lin & Gezelter, J. Chem. Phys. 128, 234107 (2008).
38 * [4] Vardeman, Stocker & Gezelter, J. Chem. Theory Comput. 7, 834 (2011).
39 * [5] Kuang & Gezelter, Mol. Phys., 110, 691-701 (2012).
40 * [6] Lamichhane, Gezelter & Newman, J. Chem. Phys. 141, 134109 (2014).
41 * [7] Lamichhane, Newman & Gezelter, J. Chem. Phys. 141, 134110 (2014).
42 * [8] Bhattarai, Newman & Gezelter, Phys. Rev. B 99, 094106 (2019).
43 */
44
45#include "restraints/ThermoIntegrationForceModifier.hpp"
46
47#ifdef IS_MPI
48#include <mpi.h>
49#endif
50
51#include "brains/SimInfo.hpp"
53
54namespace OpenMD {
55
56 ThermoIntegrationForceModifier::ThermoIntegrationForceModifier(
57 SimInfo* info) :
58 RestraintForceModifier {info} {
59 currSnapshot_ = info_->getSnapshotManager()->getCurrentSnapshot();
60 simParam_ = info_->getSimParams();
61
62 RealType tIntLambda, tIntK;
63
64 if (simParam_->haveThermodynamicIntegrationLambda()) {
65 tIntLambda = simParam_->getThermodynamicIntegrationLambda();
66 } else {
67 tIntLambda = 1.0;
68 snprintf(painCave.errMsg, MAX_SIM_ERROR_MSG_LENGTH,
69 "ThermoIntegration error: the transformation parameter\n"
70 "\t(lambda) was not specified. OpenMD will use a default\n"
71 "\tvalue of %f. To set lambda, use the \n"
72 "\tthermodynamicIntegrationLambda variable.\n",
73 tIntLambda);
74 painCave.isFatal = 0;
75 simError();
76 }
77
78 if (simParam_->haveThermodynamicIntegrationK()) {
79 tIntK = simParam_->getThermodynamicIntegrationK();
80 } else {
81 tIntK = 1.0;
82 snprintf(painCave.errMsg, MAX_SIM_ERROR_MSG_LENGTH,
83 "ThermoIntegration Warning: the tranformation parameter\n"
84 "\texponent (k) was not specified. OpenMD will use a default\n"
85 "\tvalue of %f. To set k, use the thermodynamicIntegrationK\n"
86 "\tvariable.\n",
87 tIntK);
88 painCave.isFatal = 0;
89 simError();
90 }
91
92 // build the scaling factor used to modulate the forces and torques
93 factor_ = pow(tIntLambda, tIntK);
94 }
95
96 void ThermoIntegrationForceModifier::modifyForces() {
97 Snapshot* curSnapshot;
98 SimInfo::MoleculeIterator mi;
99 Molecule* mol;
100 Molecule::IntegrableObjectIterator ii;
101 StuntDouble* sd;
102 Vector3d frc;
103 Vector3d trq;
104 Mat3x3d tempTau;
105
106 // now scale forces and torques of all the sds
107 for (mol = info_->beginMolecule(mi); mol != NULL;
108 mol = info_->nextMolecule(mi)) {
109 for (sd = mol->beginIntegrableObject(ii); sd != NULL;
110 sd = mol->nextIntegrableObject(ii)) {
111 frc = sd->getFrc();
112 frc *= factor_;
113 sd->setFrc(frc);
114
115 if (sd->isDirectional()) {
116 trq = sd->getTrq();
117 trq *= factor_;
118 sd->setTrq(trq);
119 }
120 }
121 }
122
123 curSnapshot = info_->getSnapshotManager()->getCurrentSnapshot();
124
125 // set rawPotential to be the unscaled potential energy
126 RealType rawPot = curSnapshot->getPotentialEnergy();
127 curSnapshot->setRawPotential(rawPot);
128
129 RealType rp = curSnapshot->getRestraintPotential();
130
131 // scale the potential and update the snapshot
132 rawPot *= factor_;
133 curSnapshot->setPotentialEnergy(rawPot);
134
135 // scale the virial tensor
136 tempTau = curSnapshot->getVirialTensor();
137 tempTau *= factor_;
138 curSnapshot->setVirialTensor(tempTau);
139
140 // now, on to the applied restraining potentials (if needed):
141 RealType scaledRestPot(0.0);
142 RealType restPot(0.0);
143
144 if (simParam_->getUseRestraints()) {
145 // do restraints from RestraintForceModifier:
146 scaledRestPot = doRestraints(1.0 - factor_);
147 restPot = getUnscaledPotential();
148 }
149
150#ifdef IS_MPI
151 MPI_Allreduce(MPI_IN_PLACE, &scaledRestPot, 1, MPI_REALTYPE, MPI_SUM,
152 MPI_COMM_WORLD);
153 MPI_Allreduce(MPI_IN_PLACE, &restPot, 1, MPI_REALTYPE, MPI_SUM,
154 MPI_COMM_WORLD);
155#endif
156
157 // give the final values to the snapshot
158 curSnapshot->setPotentialEnergy(rawPot + scaledRestPot);
159 curSnapshot->setRestraintPotential(rp + restPot);
160 }
161} // namespace OpenMD
This basic Periodic Table class was originally taken from the data.cpp file in OpenBabel.