kCalculateCDLJEwaldReciprocal.h 4.81 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
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
/* -------------------------------------------------------------------------- *
 *                                   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) 2009 Stanford University and the Authors.           *
 * Authors: Rossen P. Apostolov, Peter Eastman                                     *
 * Contributors:                                                              *
 *                                                                            *
 * This program is free software: you can redistribute it and/or modify       *
 * it under the terms of the GNU Lesser General Public License as published   *
 * by the Free Software Foundation, either version 3 of the License, or       *
 * (at your option) any later version.                                        *
 *                                                                            *
 * This program is distributed in the hope that it will be useful,            *
 * but WITHOUT ANY WARRANTY; without even the implied warranty of             *
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the              *
 * GNU Lesser General Public License for more details.                        *
 *                                                                            *
 * You should have received a copy of the GNU Lesser General Public License   *
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.      *
 * -------------------------------------------------------------------------- */

/**
 * This file contains the kernel for evaluating nonbonded forces using the
 * Ewald summation method (Reciprocal space summation).
 */


__global__ void kCalculateCDLJEwaldReciprocalForces_kernel()
{
    float alphaEwald       =  3.123413;
    float PI               = 3.14159265358979323846f;
    float TWO_PI           = 2.0 * PI;
    float SQRT_PI          = sqrt(PI);
    float eps0             = 5.72765E-4;

    int numRx              = 20+1;
    int numRy              = 20+1;
    int numRz              = 20+1;

    int lowry, lowrz;

    float kx, ky, kz, k2, ek;
    float Qi, Qj, SinI, SinJ, CosI, CosJ;

    float factorEwald   = -1 / (4*alphaEwald*alphaEwald);

    float recipBoxSizeX = TWO_PI / cSim.periodicBoxSizeX;
    float recipBoxSizeY = TWO_PI / cSim.periodicBoxSizeY;
    float recipBoxSizeZ = TWO_PI / cSim.periodicBoxSizeZ;
    float V = cSim.periodicBoxSizeX * cSim.periodicBoxSizeY * cSim.periodicBoxSizeZ;

    float4 apos1, apos2 ;

    unsigned int atomID1    = threadIdx.x + blockIdx.x * blockDim.x;

    while (atomID1 < cSim.atoms)
    {
        apos1             = cSim.pPosq[atomID1];

        unsigned int atomID2    = 0;
        while (atomID2 < cSim.atoms)
        {
                apos2             = cSim.pPosq[atomID2];

                lowry = 0;
                lowrz = 1;

                for(int rx = 0; rx < numRx; rx++) {

                  kx = rx * recipBoxSizeX;

                  for(int ry = lowry; ry < numRy; ry++) {

                    ky = ry * recipBoxSizeY;

                    for (int rz = lowrz; rz < numRz; rz++) {

                      kz = rz * recipBoxSizeZ;

                      k2 = kx * kx + ky * ky + kz * kz;
                      ek = exp (  k2 * factorEwald);

                      Qi = apos1.w ;
                      Qj = apos2.w ;

                      SinI = sin ( kx * apos1.x + ky * apos1.y + kz * apos1.z );
                      SinJ = sin ( kx * apos2.x + ky * apos2.y + kz * apos2.z );
                      CosI = cos ( kx * apos1.x + ky * apos1.y + kz * apos1.z );
                      CosJ = cos ( kx * apos2.x + ky * apos2.y + kz * apos2.z );

                      cSim.pForce4[atomID1].x -= (2.0 / (V * eps0 ))  * Qi * ( kx/k2) * ek * ( - SinI * Qj * CosJ + CosI * Qj * SinJ);
                      cSim.pForce4[atomID1].y -= (2.0 / (V * eps0 ))  * Qi * ( ky/k2) * ek * ( - SinI * Qj * CosJ + CosI * Qj * SinJ);
                      cSim.pForce4[atomID1].z -= (2.0 / (V * eps0 ))  * Qi * ( kz/k2) * ek * ( - SinI * Qj * CosJ + CosI * Qj * SinJ);

                      lowrz = 1 - numRz;
                    }

                    lowry = 1 - numRy;
                  }
                }

                atomID2++;
       }

       atomID1                            += blockDim.x * gridDim.x;

    }
}