ReferenceVirtualSites.cpp 12.2 KB
Newer Older
1
2
3
4
5
6
7
8
/* -------------------------------------------------------------------------- *
 *                                   OpenMM                                   *
 * -------------------------------------------------------------------------- *
 * This is part of the OpenMM molecular simulation toolkit originating from   *
 * Simbios, the NIH National Center for Physics-Based Simulation of           *
 * Biological Structures at Stanford, funded under the NIH Roadmap for        *
 * Medical Research, grant U54 GM072970. See https://simtk.org.               *
 *                                                                            *
9
 * Portions copyright (c) 2012-2017 Stanford University and the Authors.      *
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
 * Authors: Peter Eastman                                                     *
 * Contributors:                                                              *
 *                                                                            *
 * 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 "ReferenceVirtualSites.h"
#include "openmm/VirtualSite.h"
peastman's avatar
peastman committed
34
#include <cmath>
35
36
37
38

using namespace OpenMM;
using namespace std;

peastman's avatar
peastman committed
39
void ReferenceVirtualSites::computePositions(const OpenMM::System& system, vector<OpenMM::Vec3>& atomCoordinates) {
40
41
    for (int i = 0; i < system.getNumParticles(); i++)
        if (system.isVirtualSite(i)) {
42
            if (dynamic_cast<const TwoParticleAverageSite*>(&system.getVirtualSite(i)) != NULL) {
43
44
                // A two particle average.
                
45
                const TwoParticleAverageSite& site = dynamic_cast<const TwoParticleAverageSite&>(system.getVirtualSite(i));
46
                int p1 = site.getParticle(0), p2 = site.getParticle(1);
peastman's avatar
peastman committed
47
                double w1 = site.getWeight(0), w2 = site.getWeight(1);
48
49
                atomCoordinates[i] = atomCoordinates[p1]*w1 + atomCoordinates[p2]*w2;
            }
50
            else if (dynamic_cast<const ThreeParticleAverageSite*>(&system.getVirtualSite(i)) != NULL) {
51
52
                // A three particle average.
                
53
                const ThreeParticleAverageSite& site = dynamic_cast<const ThreeParticleAverageSite&>(system.getVirtualSite(i));
54
                int p1 = site.getParticle(0), p2 = site.getParticle(1), p3 = site.getParticle(2);
peastman's avatar
peastman committed
55
                double w1 = site.getWeight(0), w2 = site.getWeight(1), w3 = site.getWeight(2);
56
57
                atomCoordinates[i] = atomCoordinates[p1]*w1 + atomCoordinates[p2]*w2 + atomCoordinates[p3]*w3;
            }
58
            else if (dynamic_cast<const OutOfPlaneSite*>(&system.getVirtualSite(i)) != NULL) {
59
60
                // An out of plane site.
                
61
                const OutOfPlaneSite& site = dynamic_cast<const OutOfPlaneSite&>(system.getVirtualSite(i));
62
                int p1 = site.getParticle(0), p2 = site.getParticle(1), p3 = site.getParticle(2);
peastman's avatar
peastman committed
63
64
65
66
                double w12 = site.getWeight12(), w13 = site.getWeight13(), wcross = site.getWeightCross();
                Vec3 v12 = atomCoordinates[p2]-atomCoordinates[p1];
                Vec3 v13 = atomCoordinates[p3]-atomCoordinates[p1];
                Vec3 cross = v12.cross(v13);
67
68
                atomCoordinates[i] = atomCoordinates[p1] + v12*w12 + v13*w13 + cross*wcross;
            }
69
70
71
72
            else if (dynamic_cast<const LocalCoordinatesSite*>(&system.getVirtualSite(i)) != NULL) {
                // A local coordinates site.
                
                const LocalCoordinatesSite& site = dynamic_cast<const LocalCoordinatesSite&>(system.getVirtualSite(i));
73
74
75
76
77
78
79
80
81
82
83
84
                int numParticles = site.getNumParticles();
                vector<double> originWeights, xWeights, yWeights;
                site.getOriginWeights(originWeights);
                site.getXWeights(xWeights);
                site.getYWeights(yWeights);
                Vec3 origin, xdir, ydir;
                for (int j = 0; j < numParticles; j++) {
                    Vec3 pos = atomCoordinates[site.getParticle(j)];
                    origin += pos*originWeights[j];
                    xdir += pos*xWeights[j];
                    ydir += pos*yWeights[j];
                }
peastman's avatar
peastman committed
85
86
                Vec3 localPosition = site.getLocalPosition();
                Vec3 zdir = xdir.cross(ydir);
87
88
89
90
91
92
                double normXdir = sqrt(xdir.dot(xdir));
                double normZdir = sqrt(zdir.dot(zdir));
                if (normXdir > 0.0)
                    xdir /= normXdir;
                if (normZdir > 0.0)
                    zdir /= normZdir;
93
94
95
                ydir = zdir.cross(xdir);
                atomCoordinates[i] = origin + xdir*localPosition[0] + ydir*localPosition[1] + zdir*localPosition[2];
            }
96
97
98
        }
}

peastman's avatar
peastman committed
99
void ReferenceVirtualSites::distributeForces(const OpenMM::System& system, const vector<OpenMM::Vec3>& atomCoordinates, vector<OpenMM::Vec3>& forces) {
100
101
    for (int i = 0; i < system.getNumParticles(); i++)
        if (system.isVirtualSite(i)) {
peastman's avatar
peastman committed
102
            Vec3 f = forces[i];
103
            if (dynamic_cast<const TwoParticleAverageSite*>(&system.getVirtualSite(i)) != NULL) {
104
105
                // A two particle average.
                
106
                const TwoParticleAverageSite& site = dynamic_cast<const TwoParticleAverageSite&>(system.getVirtualSite(i));
107
                int p1 = site.getParticle(0), p2 = site.getParticle(1);
peastman's avatar
peastman committed
108
                double w1 = site.getWeight(0), w2 = site.getWeight(1);
109
110
111
                forces[p1] += f*w1;
                forces[p2] += f*w2;
            }
112
            else if (dynamic_cast<const ThreeParticleAverageSite*>(&system.getVirtualSite(i)) != NULL) {
113
114
                // A three particle average.
                
115
                const ThreeParticleAverageSite& site = dynamic_cast<const ThreeParticleAverageSite&>(system.getVirtualSite(i));
116
                int p1 = site.getParticle(0), p2 = site.getParticle(1), p3 = site.getParticle(2);
peastman's avatar
peastman committed
117
                double w1 = site.getWeight(0), w2 = site.getWeight(1), w3 = site.getWeight(2);
118
119
120
121
                forces[p1] += f*w1;
                forces[p2] += f*w2;
                forces[p3] += f*w3;
            }
122
            else if (dynamic_cast<const OutOfPlaneSite*>(&system.getVirtualSite(i)) != NULL) {
123
124
                // An out of plane site.
                
125
                const OutOfPlaneSite& site = dynamic_cast<const OutOfPlaneSite&>(system.getVirtualSite(i));
126
                int p1 = site.getParticle(0), p2 = site.getParticle(1), p3 = site.getParticle(2);
peastman's avatar
peastman committed
127
128
129
130
131
132
133
134
135
                double w12 = site.getWeight12(), w13 = site.getWeight13(), wcross = site.getWeightCross();
                Vec3 v12 = atomCoordinates[p2]-atomCoordinates[p1];
                Vec3 v13 = atomCoordinates[p3]-atomCoordinates[p1];
                Vec3 f2(w12*f[0] - wcross*v13[2]*f[1] + wcross*v13[1]*f[2],
                        wcross*v13[2]*f[0] + w12*f[1] - wcross*v13[0]*f[2],
                       -wcross*v13[1]*f[0] + wcross*v13[0]*f[1] + w12*f[2]);
                Vec3 f3(w13*f[0] + wcross*v12[2]*f[1] - wcross*v12[1]*f[2],
                       -wcross*v12[2]*f[0] + w13*f[1] + wcross*v12[0]*f[2],
                        wcross*v12[1]*f[0] - wcross*v12[0]*f[1] + w13*f[2]);
136
137
138
139
                forces[p1] += f-f2-f3;
                forces[p2] += f2;
                forces[p3] += f3;
            }
140
141
142
143
            else if (dynamic_cast<const LocalCoordinatesSite*>(&system.getVirtualSite(i)) != NULL) {
                // A local coordinates site.
                
                const LocalCoordinatesSite& site = dynamic_cast<const LocalCoordinatesSite&>(system.getVirtualSite(i));
144
145
146
147
148
149
150
151
152
153
154
                int numParticles = site.getNumParticles();
                vector<double> originWeights, wx, wy;
                site.getOriginWeights(originWeights);
                site.getXWeights(wx);
                site.getYWeights(wy);
                Vec3 xdir, ydir;
                for (int j = 0; j < numParticles; j++) {
                    Vec3 pos = atomCoordinates[site.getParticle(j)];
                    xdir += pos*wx[j];
                    ydir += pos*wy[j];
                }
peastman's avatar
peastman committed
155
156
                Vec3 localPosition = site.getLocalPosition();
                Vec3 zdir = xdir.cross(ydir);
157
158
159
160
                double normXdir = sqrt(xdir.dot(xdir));
                double normZdir = sqrt(zdir.dot(zdir));
                double invNormXdir = (normXdir > 0.0 ? 1.0/normXdir : 0.0);
                double invNormZdir = (normZdir > 0.0 ? 1.0/normZdir : 0.0);
peastman's avatar
peastman committed
161
162
163
                Vec3 dx = xdir*invNormXdir;
                Vec3 dz = zdir*invNormZdir;
                Vec3 dy = dz.cross(dx);
164
165
166
                
                // The derivatives for this case are very complicated.  They were computed with SymPy then simplified by hand.
                
167
168
169
                vector<double> wxScaled(numParticles);
                for (int j = 0; j < numParticles; j++)
                    wxScaled[j] = wx[j]*invNormXdir;
peastman's avatar
peastman committed
170
171
172
                Vec3 fp1 = localPosition*f[0];
                Vec3 fp2 = localPosition*f[1];
                Vec3 fp3 = localPosition*f[2];
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
                for (int j = 0; j < numParticles; j++) {
                    double t1 = (wx[j]*ydir[0]-wy[j]*xdir[0])*invNormZdir;
                    double t2 = (wx[j]*ydir[1]-wy[j]*xdir[1])*invNormZdir;
                    double t3 = (wx[j]*ydir[2]-wy[j]*xdir[2])*invNormZdir;
                    double sx = t3*dz[1]-t2*dz[2];
                    double sy = t1*dz[2]-t3*dz[0];
                    double sz = t2*dz[0]-t1*dz[1];
                    int p = site.getParticle(j);
                    forces[p][0] += fp1[0]*wxScaled[j]*(1-dx[0]*dx[0]) + fp1[2]*(dz[0]*sx   ) + fp1[1]*((-dx[0]*dy[0]      )*wxScaled[j] + dy[0]*sx - dx[1]*t2 - dx[2]*t3) + f[0]*originWeights[j];
                    forces[p][1] += fp1[0]*wxScaled[j]*( -dx[0]*dx[1]) + fp1[2]*(dz[0]*sy+t3) + fp1[1]*((-dx[1]*dy[0]-dz[2])*wxScaled[j] + dy[0]*sy + dx[1]*t1);
                    forces[p][2] += fp1[0]*wxScaled[j]*( -dx[0]*dx[2]) + fp1[2]*(dz[0]*sz-t2) + fp1[1]*((-dx[2]*dy[0]+dz[1])*wxScaled[j] + dy[0]*sz + dx[2]*t1);
                    forces[p][0] += fp2[0]*wxScaled[j]*( -dx[1]*dx[0]) + fp2[2]*(dz[1]*sx-t3) - fp2[1]*(( dx[0]*dy[1]-dz[2])*wxScaled[j] - dy[1]*sx - dx[0]*t2);
                    forces[p][1] += fp2[0]*wxScaled[j]*(1-dx[1]*dx[1]) + fp2[2]*(dz[1]*sy   ) - fp2[1]*(( dx[1]*dy[1]      )*wxScaled[j] - dy[1]*sy + dx[0]*t1 + dx[2]*t3) + f[1]*originWeights[j];
                    forces[p][2] += fp2[0]*wxScaled[j]*( -dx[1]*dx[2]) + fp2[2]*(dz[1]*sz+t1) - fp2[1]*(( dx[2]*dy[1]+dz[0])*wxScaled[j] - dy[1]*sz - dx[2]*t2);
                    forces[p][0] += fp3[0]*wxScaled[j]*( -dx[2]*dx[0]) + fp3[2]*(dz[2]*sx+t2) + fp3[1]*((-dx[0]*dy[2]-dz[1])*wxScaled[j] + dy[2]*sx + dx[0]*t3);
                    forces[p][1] += fp3[0]*wxScaled[j]*( -dx[2]*dx[1]) + fp3[2]*(dz[2]*sy-t1) + fp3[1]*((-dx[1]*dy[2]+dz[0])*wxScaled[j] + dy[2]*sy + dx[1]*t3);
                    forces[p][2] += fp3[0]*wxScaled[j]*(1-dx[2]*dx[2]) + fp3[2]*(dz[2]*sz   ) + fp3[1]*((-dx[2]*dy[2]      )*wxScaled[j] + dy[2]*sz - dx[0]*t1 - dx[1]*t2) + f[2]*originWeights[j];
                }
191
           }
192
193
        }
}