TestGayBerneForce.h 8.89 KB
Newer Older
1
2
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
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
/* -------------------------------------------------------------------------- *
 *                                   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.               *
 *                                                                            *
 * Portions copyright (c) 2016 Stanford University and the Authors.           *
 * 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.                                     *
 * -------------------------------------------------------------------------- */

#ifdef WIN32
  #define _USE_MATH_DEFINES // Needed to get M_PI
#endif
#include "openmm/internal/AssertionUtilities.h"
#include "openmm/Context.h"
#include "openmm/GayBerneForce.h"
#include "openmm/NonbondedForce.h"
#include "openmm/System.h"
#include "openmm/VerletIntegrator.h"
#include "sfmt/SFMT.h"
#include <iostream>
#include <vector>

using namespace OpenMM;
using namespace std;

const double TOL = 1e-5;

void testPointParticles() {
    // For point particles, it should be identical to a standard Lennard-Jones force.

    const int numParticles = 10;
    const double sigma = 0.5;
    const double epsilon = 1.5;
    System system;
    GayBerneForce* gb = new GayBerneForce();
    NonbondedForce* nb = new NonbondedForce();
    system.addForce(gb);
    system.addForce(nb);
    gb->setForceGroup(1);
    vector<Vec3> positions;
    OpenMM_SFMT::SFMT sfmt;
    init_gen_rand(0, sfmt);
    for (int i = 0; i < numParticles; i++) {
        system.addParticle(1.0);
67
        gb->addParticle(sigma, epsilon, -1, -1, sigma, sigma, sigma, 1, 1, 1);
68
69
70
71
72
73
74
75
76
77
78
79
        nb->addParticle(0, sigma, epsilon);
        positions.push_back(Vec3(2.0*genrand_real2(sfmt), 2.0*genrand_real2(sfmt), 2.0*genrand_real2(sfmt)));
    }
    VerletIntegrator integ(0.001);

    // Compute forces and energy with each one and compare them.

    Context context(system, integ, platform);
    context.setPositions(positions);
    State state1 = context.getState(State::Forces | State::Energy, false, 1);
    State state2 = context.getState(State::Forces | State::Energy, false, 2);
    ASSERT_EQUAL_TOL(state1.getPotentialEnergy(), state2.getPotentialEnergy(), 1e-5);
80
81
82
83
84
85
86
87
88
89
90
91
92
93
    for (int i = 0; i < numParticles; i++)
        ASSERT_EQUAL_VEC(state1.getForces()[i], state2.getForces()[i], 1e-5);
}

void testEnergyScales() {
    // Create two Lennard-Jones particles for which the energy scale factors vary.

    const double sigma = 0.5;
    const double epsilon = 1.5;
    System system;
    for (int i = 0; i < 6; i++)
        system.addParticle(1.0);
    GayBerneForce* gb = new GayBerneForce();
    system.addForce(gb);
94
    gb->addParticle(sigma, epsilon, 1, 2, sigma, sigma, sigma, 1.1, 1.5, 1.8);
95
96
    gb->addParticle(1, 0, -1, -1, 1, 1, 1, 1, 1, 1);
    gb->addParticle(1, 0, -1, -1, 1, 1, 1, 1, 1, 1);
97
    gb->addParticle(sigma, epsilon, 4, 5, sigma, sigma, sigma, 1.2, 1.6, 1.7);
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
    gb->addParticle(1, 0, -1, -1, 1, 1, 1, 1, 1, 1);
    gb->addParticle(1, 0, -1, -1, 1, 1, 1, 1, 1, 1);
    vector<Vec3> positions(6);
    positions[0] = Vec3(0, 0, 0);
    positions[1] = Vec3(1, 0, 0);
    positions[2] = Vec3(0, 1, 0);
    positions[3] = Vec3(1, 0, 0);
    positions[4] = Vec3(2, 0, 0);
    positions[5] = Vec3(1, 1, 0);
    VerletIntegrator integ(0.001);
    Context context(system, integ, platform);
    context.setPositions(positions);

    // Depending on their relative orientations, the interaction should be equivalent
    // to LJ with different values of epsilon.

    double expectedEnergy = 4*epsilon*(pow(sigma, 12.0)-pow(sigma, 6.0));
    double expectedForce = 4*epsilon*(12*pow(sigma, 12.0)-6*pow(sigma, 6.0));
116
    double expectedScale = pow(2.0/(1/sqrt(1.1) + 1/sqrt(1.2)), 2.0);
117
118
119
120
121
122
123
    State state = context.getState(State::Forces | State::Energy);
    ASSERT_EQUAL_TOL(expectedEnergy*expectedScale, state.getPotentialEnergy(), 1e-5);
    ASSERT_EQUAL_VEC(Vec3(expectedForce*expectedScale, 0, 0), state.getForces()[3], 1e-5);
    positions[3] = Vec3(0, 1, 0);
    positions[4] = Vec3(1, 1, 0);
    positions[5] = Vec3(0, 2, 0);
    context.setPositions(positions);
124
    expectedScale = pow(2.0/(1/sqrt(1.5) + 1/sqrt(1.6)), 2.0);
125
126
127
128
129
130
131
    state = context.getState(State::Forces | State::Energy);
    ASSERT_EQUAL_TOL(expectedEnergy*expectedScale, state.getPotentialEnergy(), 1e-5);
    ASSERT_EQUAL_VEC(Vec3(0, expectedForce*expectedScale, 0), state.getForces()[3], 1e-5);
    positions[3] = Vec3(0, 1, 0);
    positions[4] = Vec3(1, 1, 0);
    positions[5] = Vec3(0, 1, 1);
    context.setPositions(positions);
132
    expectedScale = pow(2.0/(1/sqrt(1.5) + 1/sqrt(1.7)), 2.0);
133
134
135
    state = context.getState(State::Forces | State::Energy);
    ASSERT_EQUAL_TOL(expectedEnergy*expectedScale, state.getPotentialEnergy(), 1e-5);
    ASSERT_EQUAL_VEC(Vec3(0, expectedForce*expectedScale, 0), state.getForces()[3], 1e-5);
136
137
}

138
139
140
141
142
143
144
145
146
147
148
void testEnergyConservation() {
    // Create a box of ellipsoids and make sure a simulation conserves energy.
    // That verifies that forces and energies are consistent.

    const double boxSize = 3.0;
    System system;
    system.setDefaultPeriodicBoxVectors(Vec3(boxSize, 0, 0), Vec3(0, boxSize, 0), Vec3(0, 0, boxSize));
    GayBerneForce* gb = new GayBerneForce();
    system.addForce(gb);
    gb->setNonbondedMethod(GayBerneForce::CutoffPeriodic);
    gb->setCutoffDistance(1.0);
149
150
    gb->setUseSwitchingFunction(true);
    gb->setSwitchingDistance(0.9);
151
152
153
154
155
156
157
158
    vector<Vec3> positions;
    for (int x = 0; x < 3; x++) {
        for (int y = 0; y < 3; y++) {
            for (int z = 0; z < 3; z++) {
                int first = system.getNumParticles();
                system.addParticle(10.0);
                system.addParticle(1.0);
                system.addParticle(1.0);
159
                gb->addParticle(0.2, 10.0, first+1, first+2, 0.2, 0.25, 0.3, 0.9, 1.0, 1.1);
160
161
162
163
164
165
166
167
168
169
170
                gb->addParticle(1.0, 0.0, -1, -1, 1, 1, 1, 1, 1, 1);
                gb->addParticle(1.0, 0.0, -1, -1, 1, 1, 1, 1, 1, 1);
                positions.push_back(Vec3(x, y, z));
                positions.push_back(Vec3(x+0.1, y, z));
                positions.push_back(Vec3(x, y+0.1, z));
                system.addConstraint(first, first+1, 0.1);
                system.addConstraint(first, first+2, 0.1);
                system.addConstraint(first+1, first+2, 0.1*sqrt(2.0));
            }
        }
    }
171
    VerletIntegrator integ(0.0005);
172
173
174
    Context context(system, integ, platform);
    context.setPositions(positions);
    context.setVelocitiesToTemperature(300.0);
175
    double initialEnergy;
176
    for (int i = 0; i < 100; i++) {
177
        integ.step(5);
178
        State state = context.getState(State::Energy);
179
180
181
182
        double energy = state.getPotentialEnergy()+state.getKineticEnergy();
        if (i == 0)
            initialEnergy = energy;
        else
183
            ASSERT_EQUAL_TOL(initialEnergy, energy, 1e-4);
184
185
186
    }
}

187
188
189
190
191
192
void runPlatformTests();

int main(int argc, char* argv[]) {
    try {
        initializeTests(argc, argv);
        testPointParticles();
193
        testEnergyScales();
194
        testEnergyConservation();
195
196
197
198
199
200
201
202
203
        runPlatformTests();
    }
    catch(const exception& e) {
        cout << "exception: " << e.what() << endl;
        return 1;
    }
    cout << "Done" << endl;
    return 0;
}