ReferenceCustomNonbondedIxn.cpp 13.3 KB
Newer Older
1

2
/* Portions copyright (c) 2009-2022 Stanford University and Simbios.
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
 * Contributors: Peter Eastman
 *
 * Permission is hereby granted, free of charge, to any person obtaining
 * a copy of this software and associated documentation files (the
 * "Software"), to deal in the Software without restriction, including
 * without limitation the rights to use, copy, modify, merge, publish,
 * distribute, sublicense, and/or sell copies of the Software, and to
 * permit persons to whom the Software is furnished to do so, subject
 * to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included
 * in all copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
 * IN NO EVENT SHALL THE AUTHORS, CONTRIBUTORS OR COPYRIGHT HOLDERS BE
 * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
 * OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
 * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 */

#include <string.h>
#include <sstream>

28
#include "SimTKOpenMMUtilities.h"
29
30
31
32
#include "ReferenceForce.h"
#include "ReferenceCustomNonbondedIxn.h"

using std::map;
33
using std::pair;
34
35
using std::string;
using std::stringstream;
36
using std::set;
37
using std::vector;
38
using namespace OpenMM;
39
40
41
42
43
44
45

/**---------------------------------------------------------------------------------------

   ReferenceCustomNonbondedIxn constructor

   --------------------------------------------------------------------------------------- */

46
ReferenceCustomNonbondedIxn::ReferenceCustomNonbondedIxn(const Lepton::CompiledExpression& energyExpression,
47
        const Lepton::CompiledExpression& forceExpression, const vector<string>& parameterNames,
48
49
        const vector<Lepton::CompiledExpression> energyParamDerivExpressions,
        const std::vector<std::string>& computedValueNames, const std::vector<Lepton::CompiledExpression> computedValueExpressions) :
50
            cutoff(false), useSwitch(false), periodic(false), energyExpression(energyExpression), forceExpression(forceExpression),
51
52
            paramNames(parameterNames), energyParamDerivExpressions(energyParamDerivExpressions), computedValueNames(computedValueNames),
            computedValueExpressions(computedValueExpressions) {
53
54
55
56
    expressionSet.registerExpression(this->energyExpression);
    expressionSet.registerExpression(this->forceExpression);
    for (int i = 0; i < this->energyParamDerivExpressions.size(); i++)
        expressionSet.registerExpression(this->energyParamDerivExpressions[i]);
57
58
    for (int i = 0; i < this->computedValueExpressions.size(); i++)
        expressionSet.registerExpression(this->computedValueExpressions[i]);
59
    rIndex = expressionSet.getVariableIndex("r");
peastman's avatar
peastman committed
60
    for (auto& param : paramNames) {
61
62
        for (int j = 1; j < 3; j++) {
            stringstream name;
peastman's avatar
peastman committed
63
            name << param << j;
64
            particleParamIndex.push_back(expressionSet.getVariableIndex(name.str()));
65
        }
66
    }
67
68
69
70
71
72
73
    for (auto& value : computedValueNames) {
        for (int j = 1; j < 3; j++) {
            stringstream name;
            name << value << j;
            computedValueIndex.push_back(expressionSet.getVariableIndex(name.str()));
        }
    }
74
75
76
77
78
79
80
81
}

/**---------------------------------------------------------------------------------------

   ReferenceCustomNonbondedIxn destructor

   --------------------------------------------------------------------------------------- */

82
ReferenceCustomNonbondedIxn::~ReferenceCustomNonbondedIxn() {
83
84
85
86
87
88
89
90
91
92
93
}

  /**---------------------------------------------------------------------------------------

     Set the force to use a cutoff.

     @param distance            the cutoff distance
     @param neighbors           the neighbor list to use

     --------------------------------------------------------------------------------------- */

peastman's avatar
peastman committed
94
  void ReferenceCustomNonbondedIxn::setUseCutoff(double distance, const OpenMM::NeighborList& neighbors) {
95
96
97
98
99
100

    cutoff = true;
    cutoffDistance = distance;
    neighborList = &neighbors;
  }

101
102
103
104
105
106
107
108
109
110
111
112
113
/**---------------------------------------------------------------------------------------

   Restrict the force to a list of interaction groups.

   @param distance            the cutoff distance
   @param neighbors           the neighbor list to use

   --------------------------------------------------------------------------------------- */

void ReferenceCustomNonbondedIxn::setInteractionGroups(const vector<pair<set<int>, set<int> > >& groups) {
    interactionGroups = groups;
}

114
115
116
117
118
119
120
121
/**---------------------------------------------------------------------------------------

   Set the force to use a switching function.

   @param distance            the switching distance

   --------------------------------------------------------------------------------------- */

peastman's avatar
peastman committed
122
void ReferenceCustomNonbondedIxn::setUseSwitchingFunction(double distance) {
123
124
125
126
    useSwitch = true;
    switchingDistance = distance;
}

127
128
129
130
131
132
  /**---------------------------------------------------------------------------------------

     Set the force to use periodic boundary conditions.  This requires that a cutoff has
     also been set, and the smallest side of the periodic box is at least twice the cutoff
     distance.

133
     @param vectors    the vectors defining the periodic box
134
135
136

     --------------------------------------------------------------------------------------- */

peastman's avatar
peastman committed
137
  void ReferenceCustomNonbondedIxn::setPeriodic(OpenMM::Vec3* vectors) {
138
139

    assert(cutoff);
140
141
142
    assert(vectors[0][0] >= 2.0*cutoffDistance);
    assert(vectors[1][1] >= 2.0*cutoffDistance);
    assert(vectors[2][2] >= 2.0*cutoffDistance);
143
    periodic = true;
144
145
146
    periodicBoxVectors[0] = vectors[0];
    periodicBoxVectors[1] = vectors[1];
    periodicBoxVectors[2] = vectors[2];
147
148
149
150
151
152
153
154
155
156
157

  }


/**---------------------------------------------------------------------------------------

   Calculate the custom pair ixn

   @param numberOfAtoms    number of atoms
   @param atomCoordinates  atom coordinates
   @param atomParameters   atom parameters                             atomParameters[atomIndex][paramterIndex]
158
159
   @param exclusions       atom exclusion indices
                           exclusions[atomIndex] contains the list of exclusions for that atom
160
   @param globalParameters the values of global parameters
161
162
163
164
165
   @param forces           force array (forces added)
   @param totalEnergy      total energy

   --------------------------------------------------------------------------------------- */

peastman's avatar
peastman committed
166
void ReferenceCustomNonbondedIxn::calculatePairIxn(int numberOfAtoms, vector<Vec3>& atomCoordinates,
167
168
169
                                             vector<vector<double> >& atomParameters, vector<set<int> >& exclusions,
                                             const map<string, double>& globalParameters, vector<Vec3>& forces,
                                             double* totalEnergy, double* energyParamDerivs) {
170

peastman's avatar
peastman committed
171
172
    for (auto& param : globalParameters)
        expressionSet.setVariable(expressionSet.getVariableIndex(param.first), param.second);
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
    
    // Calculate computed values.
    
    vector<int> paramIndex, valueIndex;
    for (auto& name : paramNames)
        paramIndex.push_back(expressionSet.getVariableIndex(name));
    for (auto& name : computedValueNames)
        valueIndex.push_back(expressionSet.getVariableIndex(name));
    vector<vector<double> > computedValues(computedValueNames.size(), vector<double>(numberOfAtoms));
    for (int i = 0; i < numberOfAtoms; i++) {
        for (int j = 0; j < paramNames.size(); j++)
            expressionSet.setVariable(paramIndex[j], atomParameters[i][j]);
        for (int j = 0; j < computedValueNames.size(); j++)
            computedValues[j][i] = computedValueExpressions[j].evaluate();
    }
    
    // Compute the interactions.

    if (interactionGroups.size() > 0) {
192
193
        // The user has specified interaction groups, so compute only the requested interactions.
        
peastman's avatar
peastman committed
194
195
196
        for (auto& group : interactionGroups) {
            const set<int>& set1 = group.first;
            const set<int>& set2 = group.second;
197
198
            for (set<int>::const_iterator atom1 = set1.begin(); atom1 != set1.end(); ++atom1) {
                for (set<int>::const_iterator atom2 = set2.begin(); atom2 != set2.end(); ++atom2) {
peastman's avatar
peastman committed
199
200
201
202
203
                    if (*atom1 == *atom2 || exclusions[*atom1].find(*atom2) != exclusions[*atom1].end())
                        continue; // This is an excluded interaction.
                    if (*atom1 > *atom2 && set1.find(*atom2) != set1.end() && set2.find(*atom1) != set2.end())
                        continue; // Both atoms are in both sets, so skip duplicate interactions.
                    for (int j = 0; j < (int) paramNames.size(); j++) {
204
205
                        expressionSet.setVariable(particleParamIndex[j*2], atomParameters[*atom1][j]);
                        expressionSet.setVariable(particleParamIndex[j*2+1], atomParameters[*atom2][j]);
206
                    }
207
208
209
210
                    for (int j = 0; j < (int) computedValueNames.size(); j++) {
                        expressionSet.setVariable(computedValueIndex[j*2], computedValues[j][*atom1]);
                        expressionSet.setVariable(computedValueIndex[j*2+1], computedValues[j][*atom2]);
                    }
211
                    calculateOneIxn(*atom1, *atom2, atomCoordinates, forces, totalEnergy, energyParamDerivs);
212
213
214
215
216
217
218
                }
            }
        }
    }
    else if (cutoff) {
        // We are using a cutoff, so get the interactions from the neighbor list.
        
peastman's avatar
peastman committed
219
        for (auto& pair : *neighborList) {
220
            for (int j = 0; j < (int) paramNames.size(); j++) {
221
222
                expressionSet.setVariable(particleParamIndex[j*2], atomParameters[pair.first][j]);
                expressionSet.setVariable(particleParamIndex[j*2+1], atomParameters[pair.second][j]);
223
            }
224
225
226
227
            for (int j = 0; j < (int) computedValueNames.size(); j++) {
                expressionSet.setVariable(computedValueIndex[j*2], computedValues[j][pair.first]);
                expressionSet.setVariable(computedValueIndex[j*2+1], computedValues[j][pair.second]);
            }
228
            calculateOneIxn(pair.first, pair.second, atomCoordinates, forces, totalEnergy, energyParamDerivs);
229
230
231
232
233
234
235
236
237
        }
    }
    else {
        // Every particle interacts with every other one.
        
        for (int ii = 0; ii < numberOfAtoms; ii++) {
            for (int jj = ii+1; jj < numberOfAtoms; jj++) {
                if (exclusions[jj].find(ii) == exclusions[jj].end()) {
                    for (int j = 0; j < (int) paramNames.size(); j++) {
238
239
                        expressionSet.setVariable(particleParamIndex[j*2], atomParameters[ii][j]);
                        expressionSet.setVariable(particleParamIndex[j*2+1], atomParameters[jj][j]);
240
                    }
241
242
243
244
                    for (int j = 0; j < (int) computedValueNames.size(); j++) {
                        expressionSet.setVariable(computedValueIndex[j*2], computedValues[j][ii]);
                        expressionSet.setVariable(computedValueIndex[j*2+1], computedValues[j][jj]);
                    }
245
                    calculateOneIxn(ii, jj, atomCoordinates, forces, totalEnergy, energyParamDerivs);
246
247
248
249
                }
            }
        }
    }
250
251
252
253
254
255
256
257
258
259
260
261
262
263
}

  /**---------------------------------------------------------------------------------------

     Calculate one pair ixn between two atoms

     @param ii               the index of the first atom
     @param jj               the index of the second atom
     @param atomCoordinates  atom coordinates
     @param forces           force array (forces added)
     @param totalEnergy      total energy

     --------------------------------------------------------------------------------------- */

peastman's avatar
peastman committed
264
void ReferenceCustomNonbondedIxn::calculateOneIxn(int ii, int jj, vector<Vec3>& atomCoordinates, vector<Vec3>& forces,
265
                        double* totalEnergy, double* energyParamDerivs) {
266
267
    // get deltaR, R2, and R between 2 atoms

peastman's avatar
peastman committed
268
    double deltaR[ReferenceForce::LastDeltaRIndex];
269
    if (periodic)
270
        ReferenceForce::getDeltaRPeriodic(atomCoordinates[jj], atomCoordinates[ii], periodicBoxVectors, deltaR);
271
    else
272
        ReferenceForce::getDeltaR(atomCoordinates[jj], atomCoordinates[ii], deltaR);
peastman's avatar
peastman committed
273
    double r = deltaR[ReferenceForce::RIndex];
274
    if (cutoff && r >= cutoffDistance)
275
276
277
278
        return;

    // accumulate forces

279
    expressionSet.setVariable(rIndex, r);
peastman's avatar
peastman committed
280
281
282
    double dEdR = forceExpression.evaluate()/(deltaR[ReferenceForce::RIndex]);
    double energy = energyExpression.evaluate();
    double switchValue = 1.0;
283
284
    if (useSwitch) {
        if (r > switchingDistance) {
peastman's avatar
peastman committed
285
            double t = (r-switchingDistance)/(cutoffDistance-switchingDistance);
286
            switchValue = 1+t*t*t*(-10+t*(15-t*6));
peastman's avatar
peastman committed
287
            double switchDeriv = t*t*(-30+t*(60-t*30))/(cutoffDistance-switchingDistance);
288
289
290
291
            dEdR = switchValue*dEdR + energy*switchDeriv/r;
            energy *= switchValue;
        }
    }
292
    for (int kk = 0; kk < 3; kk++) {
peastman's avatar
peastman committed
293
294
295
       double force = -dEdR*deltaR[kk];
       forces[ii][kk] += force;
       forces[jj][kk] -= force;
296
    }
297
298
    for (int i = 0; i < energyParamDerivExpressions.size(); i++)
        energyParamDerivs[i] += switchValue*energyParamDerivExpressions[i].evaluate();
299
300
301

    // accumulate energies

302
303
    if (totalEnergy)
        *totalEnergy += energy;
304
305
306
  }