customHbondForce.cl 9.72 KB
Newer Older
1
2
3
4
5
6
7
8
9
10
11
12
13
/**
 * Compute the difference between two vectors, setting the fourth component to the squared magnitude.
 */
float4 delta(float4 vec1, float4 vec2) {
    float4 result = (float4) (vec1.x-vec2.x, vec1.y-vec2.y, vec1.z-vec2.z, 0.0f);
    result.w = result.x*result.x + result.y*result.y + result.z*result.z;
    return result;
}

/**
 * Compute the difference between two vectors, taking periodic boundary conditions into account
 * and setting the fourth component to the squared magnitude.
 */
14
float4 deltaPeriodic(float4 vec1, float4 vec2, float4 periodicBoxSize, float4 invPeriodicBoxSize) {
15
16
    float4 result = (float4) (vec1.x-vec2.x, vec1.y-vec2.y, vec1.z-vec2.z, 0.0f);
#ifdef USE_PERIODIC
17
18
19
    result.x -= floor(result.x*invPeriodicBoxSize.x+0.5f)*periodicBoxSize.x;
    result.y -= floor(result.y*invPeriodicBoxSize.y+0.5f)*periodicBoxSize.y;
    result.z -= floor(result.z*invPeriodicBoxSize.z+0.5f)*periodicBoxSize.z;
20
21
22
23
24
25
26
27
28
#endif
    result.w = result.x*result.x + result.y*result.y + result.z*result.z;
    return result;
}

/**
 * Compute the angle between two vectors.  The w component of each vector should contain the squared magnitude.
 */
float computeAngle(float4 vec1, float4 vec2) {
29
    float dotProduct = vec1.x*vec2.x + vec1.y*vec2.y + vec1.z*vec2.z;
30
    float cosine = dotProduct*RSQRT(vec1.w*vec2.w);
31
32
33
34
    float angle;
    if (cosine > 0.99f || cosine < -0.99f) {
        // We're close to the singularity in acos(), so take the cross product and use asin() instead.

35
        float4 crossProduct = cross(vec1, vec2);
36
        float scale = vec1.w*vec2.w;
37
        angle = asin(SQRT(dot(crossProduct, crossProduct)/scale));
38
        if (cosine < 0.0f)
39
            angle = PI-angle;
40
41
42
43
44
45
46
    }
    else
       angle = acos(cosine);
    return angle;
}

/**
47
 * Compute the cross product of two vectors, setting the fourth component to the squared magnitude.
48
 */
49
50
51
52
53
float4 computeCross(float4 vec1, float4 vec2) {
    float4 result = cross(vec1, vec2);
    result.w = result.x*result.x + result.y*result.y + result.z*result.z;
    return result;
}
54

55
56
57
/**
 * Compute forces on donors.
 */
58
59
__kernel void computeDonorForces(__global float4* restrict forceBuffers, __global float* restrict energyBuffer, __global const float4* restrict posq, __global const int4* restrict exclusions,
        __global const int4* restrict donorAtoms, __global const int4* restrict acceptorAtoms, __global const int4* restrict donorBufferIndices, __local float4* posBuffer, float4 periodicBoxSize, float4 invPeriodicBoxSize
60
61
        PARAMETER_ARGUMENTS) {
    float energy = 0.0f;
62
63
64
    float4 f1 = (float4) 0;
    float4 f2 = (float4) 0;
    float4 f3 = (float4) 0;
65
    for (int donorStart = 0; donorStart < NUM_DONORS; donorStart += get_global_size(0)) {
66
67
        // Load information about the donor this thread will compute forces on.

68
        int donorIndex = donorStart+get_global_id(0);
69
        int4 atoms, exclusionIndices;
70
71
72
        float4 d1, d2, d3;
        if (donorIndex < NUM_DONORS) {
            atoms = donorAtoms[donorIndex];
73
74
75
            d1 = (atoms.x > -1 ? posq[atoms.x] : (float4) 0);
            d2 = (atoms.y > -1 ? posq[atoms.y] : (float4) 0);
            d3 = (atoms.z > -1 ? posq[atoms.z] : (float4) 0);
76
77
78
#ifdef USE_EXCLUSIONS
            exclusionIndices = exclusions[donorIndex];
#endif
79
80
81
        }
        else
            atoms = (int4) (-1, -1, -1, -1);
82
83
84
85
        for (int acceptorStart = 0; acceptorStart < NUM_ACCEPTORS; acceptorStart += get_local_size(0)) {
            // Load the next block of acceptors into local memory.

            int blockSize = min((int) get_local_size(0), NUM_ACCEPTORS-acceptorStart);
86
            if (get_local_id(0) < blockSize) {
87
                int4 atoms2 = acceptorAtoms[acceptorStart+get_local_id(0)];
88
89
90
                posBuffer[3*get_local_id(0)] = (atoms2.x > -1 ? posq[atoms2.x] : (float4) 0);
                posBuffer[3*get_local_id(0)+1] = (atoms2.y > -1 ? posq[atoms2.y] : (float4) 0);
                posBuffer[3*get_local_id(0)+2] = (atoms2.z > -1 ? posq[atoms2.z] : (float4) 0);
91
92
            }
            barrier(CLK_LOCAL_MEM_FENCE);
93
94
            if (donorIndex < NUM_DONORS) {
                for (int index = 0; index < blockSize; index++) {
95
96
97
98
99
#ifdef USE_EXCLUSIONS
                    int acceptorIndex = acceptorStart+index;
                    if (acceptorIndex == exclusionIndices.x || acceptorIndex == exclusionIndices.y || acceptorIndex == exclusionIndices.z || acceptorIndex == exclusionIndices.w)
                        continue;
#endif
100
101
102
103
104
                    // Compute the interaction between a donor and an acceptor.

                    float4 a1 = posBuffer[3*index];
                    float4 a2 = posBuffer[3*index+1];
                    float4 a3 = posBuffer[3*index+2];
105
                    float4 deltaD1A1 = deltaPeriodic(d1, a1, periodicBoxSize, invPeriodicBoxSize);
106
#ifdef USE_CUTOFF
107
                    if (deltaD1A1.w < CUTOFF_SQUARED) {
108
#endif
109
110
111
                        COMPUTE_DONOR_FORCE
#ifdef USE_CUTOFF
                    }
112
#endif
113
114
115
                }
            }
        }
116

117
        // Write results
118

119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
        if (donorIndex < NUM_DONORS) {
            int4 bufferIndices = donorBufferIndices[donorIndex];
            if (atoms.x > -1) {
                unsigned int offset = atoms.x+bufferIndices.x*PADDED_NUM_ATOMS;
                float4 force = forceBuffers[offset];
                force.xyz += f1.xyz;
                forceBuffers[offset] = force;
            }
            if (atoms.y > -1) {
                unsigned int offset = atoms.y+bufferIndices.y*PADDED_NUM_ATOMS;
                float4 force = forceBuffers[offset];
                force.xyz += f2.xyz;
                forceBuffers[offset] = force;
            }
            if (atoms.z > -1) {
                unsigned int offset = atoms.z+bufferIndices.z*PADDED_NUM_ATOMS;
                float4 force = forceBuffers[offset];
                force.xyz += f3.xyz;
                forceBuffers[offset] = force;
            }
139
140
141
142
143
144
145
        }
    }
    energyBuffer[get_global_id(0)] += energy;
}
/**
 * Compute forces on acceptors.
 */
146
147
__kernel void computeAcceptorForces(__global float4* restrict forceBuffers, __global float* restrict energyBuffer, __global const float4* restrict posq, __global const int4* restrict exclusions,
        __global const int4* restrict donorAtoms, __global const int4* restrict acceptorAtoms, __global const int4* restrict acceptorBufferIndices, __local float4* restrict posBuffer, float4 periodicBoxSize, float4 invPeriodicBoxSize
148
        PARAMETER_ARGUMENTS) {
149
150
151
    float4 f1 = (float4) 0;
    float4 f2 = (float4) 0;
    float4 f3 = (float4) 0;
152
153
154
155
    for (int acceptorStart = 0; acceptorStart < NUM_ACCEPTORS; acceptorStart += get_global_size(0)) {
        // Load information about the acceptor this thread will compute forces on.

        int acceptorIndex = acceptorStart+get_global_id(0);
156
        int4 atoms, exclusionIndices;
157
158
159
        float4 a1, a2, a3;
        if (acceptorIndex < NUM_ACCEPTORS) {
            atoms = acceptorAtoms[acceptorIndex];
160
161
162
            a1 = (atoms.x > -1 ? posq[atoms.x] : (float4) 0);
            a2 = (atoms.y > -1 ? posq[atoms.y] : (float4) 0);
            a3 = (atoms.z > -1 ? posq[atoms.z] : (float4) 0);
163
164
165
#ifdef USE_EXCLUSIONS
            exclusionIndices = exclusions[acceptorIndex];
#endif
166
167
168
169
170
171
172
        }
        else
            atoms = (int4) (-1, -1, -1, -1);
        for (int donorStart = 0; donorStart < NUM_DONORS; donorStart += get_local_size(0)) {
            // Load the next block of donors into local memory.

            int blockSize = min((int) get_local_size(0), NUM_DONORS-donorStart);
173
            if (get_local_id(0) < blockSize) {
174
                int4 atoms2 = donorAtoms[donorStart+get_local_id(0)];
175
176
177
                posBuffer[3*get_local_id(0)] = (atoms2.x > -1 ? posq[atoms2.x] : (float4) 0);
                posBuffer[3*get_local_id(0)+1] = (atoms2.y > -1 ? posq[atoms2.y] : (float4) 0);
                posBuffer[3*get_local_id(0)+2] = (atoms2.z > -1 ? posq[atoms2.z] : (float4) 0);
178
179
180
181
            }
            barrier(CLK_LOCAL_MEM_FENCE);
            if (acceptorIndex < NUM_ACCEPTORS) {
                for (int index = 0; index < blockSize; index++) {
182
183
184
185
186
#ifdef USE_EXCLUSIONS
                    int donorIndex = donorStart+index;
                    if (donorIndex == exclusionIndices.x || donorIndex == exclusionIndices.y || donorIndex == exclusionIndices.z || donorIndex == exclusionIndices.w)
                        continue;
#endif
187
188
189
190
191
                    // Compute the interaction between a donor and an acceptor.

                    float4 d1 = posBuffer[3*index];
                    float4 d2 = posBuffer[3*index+1];
                    float4 d3 = posBuffer[3*index+2];
192
                    float4 deltaD1A1 = deltaPeriodic(d1, a1, periodicBoxSize, invPeriodicBoxSize);
193
194
#ifdef USE_CUTOFF
                    if (deltaD1A1.w < CUTOFF_SQUARED) {
195
#endif
196
                        COMPUTE_ACCEPTOR_FORCE
197
#ifdef USE_CUTOFF
198
                    }
199
#endif
200
                }
201
202
203
204
205
            }
        }

        // Write results

206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
        if (acceptorIndex < NUM_ACCEPTORS) {
            int4 bufferIndices = acceptorBufferIndices[acceptorIndex];
            if (atoms.x > -1) {
                unsigned int offset = atoms.x+bufferIndices.x*PADDED_NUM_ATOMS;
                float4 force = forceBuffers[offset];
                force.xyz += f1.xyz;
                forceBuffers[offset] = force;
            }
            if (atoms.y > -1) {
                unsigned int offset = atoms.y+bufferIndices.y*PADDED_NUM_ATOMS;
                float4 force = forceBuffers[offset];
                force.xyz += f2.xyz;
                forceBuffers[offset] = force;
            }
            if (atoms.z > -1) {
                unsigned int offset = atoms.z+bufferIndices.z*PADDED_NUM_ATOMS;
                float4 force = forceBuffers[offset];
                force.xyz += f3.xyz;
                forceBuffers[offset] = force;
            }
226
        }
227
228
    }
}