2012-10-08 23:29:55 +08:00
|
|
|
/* ----------------------------------------------------------------------
|
|
|
|
LAMMPS - Large-scale Atomic/Molecular Massively Parallel Simulator
|
|
|
|
|
|
|
|
Original Version:
|
|
|
|
http://lammps.sandia.gov, Sandia National Laboratories
|
|
|
|
Steve Plimpton, sjplimp@sandia.gov
|
|
|
|
|
|
|
|
See the README file in the top-level LAMMPS directory.
|
|
|
|
|
|
|
|
-----------------------------------------------------------------------
|
|
|
|
|
|
|
|
USER-CUDA Package and associated modifications:
|
|
|
|
https://sourceforge.net/projects/lammpscuda/
|
|
|
|
|
|
|
|
Christian Trott, christian.trott@tu-ilmenau.de
|
|
|
|
Lars Winterfeld, lars.winterfeld@tu-ilmenau.de
|
|
|
|
Theoretical Physics II, University of Technology Ilmenau, Germany
|
|
|
|
|
|
|
|
See the README file in the USER-CUDA directory.
|
|
|
|
|
|
|
|
This software is distributed under the GNU General Public License.
|
|
|
|
------------------------------------------------------------------------- */
|
|
|
|
#define Pi F_F(3.1415926535897932384626433832795)
|
|
|
|
#define PI Pi
|
|
|
|
#define PI2 F_F(0.5)*Pi
|
|
|
|
#define PI4 F_F(0.25)*Pi
|
|
|
|
template <const int eflag, const int vflag>
|
|
|
|
static inline __device__ void PairVirialCompute_A_Kernel_Template()
|
|
|
|
{
|
|
|
|
__syncthreads();
|
2014-10-07 06:59:05 +08:00
|
|
|
ENERGY_CFLOAT* shared = sharedmem;
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
if(eflag) {
|
|
|
|
reduceBlock(shared);
|
|
|
|
shared += blockDim.x;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(vflag) {
|
|
|
|
reduceBlock(shared + 0 * blockDim.x);
|
|
|
|
reduceBlock(shared + 1 * blockDim.x);
|
|
|
|
reduceBlock(shared + 2 * blockDim.x);
|
|
|
|
reduceBlock(shared + 3 * blockDim.x);
|
|
|
|
reduceBlock(shared + 4 * blockDim.x);
|
|
|
|
reduceBlock(shared + 5 * blockDim.x);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(threadIdx.x == 0) {
|
|
|
|
shared = sharedmem;
|
2014-10-07 06:59:05 +08:00
|
|
|
ENERGY_CFLOAT* buffer = (ENERGY_CFLOAT*) _buffer;
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
if(eflag) {
|
|
|
|
buffer[blockIdx.x * gridDim.y + blockIdx.y] = ENERGY_F(0.5) * shared[0];
|
|
|
|
shared += blockDim.x;
|
|
|
|
buffer += gridDim.x * gridDim.y;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(vflag) {
|
|
|
|
buffer[blockIdx.x * gridDim.y + blockIdx.y + 0 * gridDim.x * gridDim.y] = ENERGY_F(0.5) * shared[0 * blockDim.x];
|
|
|
|
buffer[blockIdx.x * gridDim.y + blockIdx.y + 1 * gridDim.x * gridDim.y] = ENERGY_F(0.5) * shared[1 * blockDim.x];
|
|
|
|
buffer[blockIdx.x * gridDim.y + blockIdx.y + 2 * gridDim.x * gridDim.y] = ENERGY_F(0.5) * shared[2 * blockDim.x];
|
|
|
|
buffer[blockIdx.x * gridDim.y + blockIdx.y + 3 * gridDim.x * gridDim.y] = ENERGY_F(0.5) * shared[3 * blockDim.x];
|
|
|
|
buffer[blockIdx.x * gridDim.y + blockIdx.y + 4 * gridDim.x * gridDim.y] = ENERGY_F(0.5) * shared[4 * blockDim.x];
|
|
|
|
buffer[blockIdx.x * gridDim.y + blockIdx.y + 5 * gridDim.x * gridDim.y] = ENERGY_F(0.5) * shared[5 * blockDim.x];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
__syncthreads();
|
|
|
|
}
|
|
|
|
|
|
|
|
__global__ void virial_fdotr_compute_kernel(int eflag)
|
|
|
|
{
|
|
|
|
int i = (blockIdx.x * gridDim.y + blockIdx.y) * blockDim.x + threadIdx.x;
|
2014-10-07 06:59:05 +08:00
|
|
|
ENERGY_CFLOAT* sharedE = (ENERGY_CFLOAT*) &sharedmem[0];
|
|
|
|
ENERGY_CFLOAT* sharedVirial = (ENERGY_CFLOAT*) &sharedE[blockDim.x];
|
2012-10-08 23:29:55 +08:00
|
|
|
sharedE += threadIdx.x;
|
|
|
|
sharedVirial += threadIdx.x;
|
|
|
|
|
|
|
|
if(i < _nlocal) {
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT x = _x[i];
|
|
|
|
F_CFLOAT y = _x[i + _nmax];
|
|
|
|
F_CFLOAT z = _x[i + 2 * _nmax];
|
|
|
|
F_CFLOAT fx = _f[i];
|
|
|
|
F_CFLOAT fy = _f[i + _nmax];
|
|
|
|
F_CFLOAT fz = _f[i + 2 * _nmax];
|
2012-10-08 23:29:55 +08:00
|
|
|
//if(fz*z*fz*z>1e-5) printf("V %i %i %e %e %e %e %e %e\n",i,_tag[i],x,y,z,fx,fy,fz);
|
|
|
|
sharedVirial[0] = fx * x;
|
|
|
|
sharedVirial[1 * blockDim.x] = fy * y;
|
|
|
|
sharedVirial[2 * blockDim.x] = fz * z;
|
|
|
|
sharedVirial[3 * blockDim.x] = fy * x;
|
|
|
|
sharedVirial[4 * blockDim.x] = fz * x;
|
|
|
|
sharedVirial[5 * blockDim.x] = fz * y;
|
|
|
|
} else {
|
|
|
|
sharedVirial[0] = 0;
|
|
|
|
sharedVirial[1 * blockDim.x] = 0;
|
|
|
|
sharedVirial[2 * blockDim.x] = 0;
|
|
|
|
sharedVirial[3 * blockDim.x] = 0;
|
|
|
|
sharedVirial[4 * blockDim.x] = 0;
|
|
|
|
sharedVirial[5 * blockDim.x] = 0;
|
|
|
|
}
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
sharedVirial = (ENERGY_CFLOAT*) &sharedmem[0];
|
2012-10-08 23:29:55 +08:00
|
|
|
sharedVirial += blockDim.x;
|
|
|
|
reduceBlockP2(sharedVirial);
|
|
|
|
reduceBlockP2(&sharedVirial[1 * blockDim.x]);
|
|
|
|
reduceBlockP2(&sharedVirial[2 * blockDim.x]);
|
|
|
|
reduceBlockP2(&sharedVirial[3 * blockDim.x]);
|
|
|
|
reduceBlockP2(&sharedVirial[4 * blockDim.x]);
|
|
|
|
reduceBlockP2(&sharedVirial[5 * blockDim.x]);
|
|
|
|
|
|
|
|
if(threadIdx.x < 6) {
|
2014-10-07 06:59:05 +08:00
|
|
|
ENERGY_CFLOAT* buffer = (ENERGY_CFLOAT*) _buffer;
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
if(eflag) buffer = &buffer[gridDim.x * gridDim.y];
|
|
|
|
|
|
|
|
buffer[blockIdx.x * gridDim.y + blockIdx.y + threadIdx.x * gridDim.x * gridDim.y] = sharedVirial[threadIdx.x * blockDim.x];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*#define vec3_scale(K,X,Y) Y.x = K*X.x; Y.y = K*X.y; Y.z = K*X.z;
|
|
|
|
#define vec3_scaleadd(K,X,Y,Z) Z.x = K*X.x+Y.x; Z.y = K*X.y+Y.y; Z.z = K*X.z+Y.z;
|
|
|
|
#define vec3_add(X,Y,Z) Z.x = X.x+Y.x; Z.y = X.y+Y.y; Z.z = X.z+Y.z;
|
|
|
|
#define vec3_dot(X,Y) (X.x*Y.x + X.y*Y.y + X.z*Y.z)*/
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ inline void vec3_scale(F_CFLOAT k, F_CFLOAT3 &x, F_CFLOAT3 &y)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
|
|
|
y.x = k * x.x;
|
|
|
|
y.y = k * x.y;
|
|
|
|
y.z = k * x.z;
|
|
|
|
}
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ inline void vec3_scale(F_CFLOAT k, F_CFLOAT4 &x, F_CFLOAT3 &y)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
|
|
|
y.x = k * x.x;
|
|
|
|
y.y = k * x.y;
|
|
|
|
y.z = k * x.z;
|
|
|
|
}
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ inline void vec3_scale(F_CFLOAT k, F_CFLOAT4 &x, F_CFLOAT4 &y)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
|
|
|
y.x = k * x.x;
|
|
|
|
y.y = k * x.y;
|
|
|
|
y.z = k * x.z;
|
|
|
|
}
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ inline void vec3_scaleadd(F_CFLOAT k, F_CFLOAT3 &x, F_CFLOAT3 &y, F_CFLOAT3 &z)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
|
|
|
z.x = k * x.x + y.x;
|
|
|
|
z.y = k * x.y + y.y;
|
|
|
|
z.z = k * x.z + y.z;
|
|
|
|
}
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ inline void vec3_add(F_CFLOAT3 &x, F_CFLOAT3 &y, F_CFLOAT3 &z)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
|
|
|
z.x = x.x + y.x;
|
|
|
|
z.y = x.y + y.y;
|
|
|
|
z.z = x.z + y.z;
|
|
|
|
}
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ inline F_CFLOAT vec3_dot(F_CFLOAT3 x, F_CFLOAT3 y)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
|
|
|
return x.x * y.x + x.y * y.y + x.z * y.z;
|
|
|
|
}
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ inline F_CFLOAT vec3_dot(F_CFLOAT4 x, F_CFLOAT4 y)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
|
|
|
return x.x * y.x + x.y * y.y + x.z * y.z;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ----------------------------------------------------------------------
|
|
|
|
Fermi-like smoothing function
|
|
|
|
------------------------------------------------------------------------- */
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ inline F_CFLOAT F_fermi(F_CFLOAT &r, int &iparam)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
|
|
|
return F_F(1.0) / (F_F(1.0) + exp(-params[iparam].ZBLexpscale * (r - params[iparam].ZBLcut)));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ----------------------------------------------------------------------
|
|
|
|
Fermi-like smoothing function derivative with respect to r
|
|
|
|
------------------------------------------------------------------------- */
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ inline F_CFLOAT F_fermi_d(F_CFLOAT &r, int &iparam)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
2014-10-07 06:59:05 +08:00
|
|
|
volatile const F_CFLOAT tmp = exp(-params[iparam].ZBLexpscale * (r - params[iparam].ZBLcut));
|
2012-10-08 23:29:55 +08:00
|
|
|
return params[iparam].ZBLexpscale * tmp /
|
|
|
|
((F_F(1.0) + tmp) * (F_F(1.0) + tmp));
|
|
|
|
}
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ inline F_CFLOAT ters_fc(F_CFLOAT r, F_CFLOAT ters_R, F_CFLOAT ters_D)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
|
|
|
return (r < ters_R - ters_D) ? F_F(1.0) : ((r > ters_R + ters_D) ?
|
|
|
|
F_F(0.0) : F_F(0.5) * (F_F(1.0) - sin(PI2 * (r - ters_R) / ters_D)));
|
|
|
|
}
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ inline F_CFLOAT ters_fc_d(F_CFLOAT r, F_CFLOAT ters_R, F_CFLOAT ters_D)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
|
|
|
return ((r < ters_R - ters_D) || (r > ters_R + ters_D)) ?
|
|
|
|
F_F(0.0) : -(PI4 / ters_D) * cos(PI2 * (r - ters_R) / ters_D);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ inline F_CFLOAT ters_gijk(F_CFLOAT &cos_theta, int iparam)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT ters_c = params[iparam].c;
|
|
|
|
F_CFLOAT ters_d = params[iparam].d;
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
return params[iparam].gamma * (F_F(1.0) + pow(params[iparam].c / params[iparam].d, F_F(2.0)) -
|
|
|
|
pow(ters_c, F_F(2.0)) / (pow(ters_d, F_F(2.0)) + pow(params[iparam].h - cos_theta, F_F(2.0))));
|
|
|
|
}
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ F_CFLOAT ters_gijk2(F_CFLOAT &cos_theta, int iparam)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT ters_c = params[iparam].c;
|
|
|
|
F_CFLOAT ters_d = params[iparam].d;
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
return params[iparam].gamma * (F_F(1.0) + pow(ters_c / ters_d, F_F(2.0)) -
|
|
|
|
pow(ters_c, F_F(2.0)) / (pow(ters_d, F_F(2.0)) + pow(params[iparam].h - cos_theta, F_F(2.0))));
|
|
|
|
}
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ inline F_CFLOAT ters_gijk_d(F_CFLOAT costheta, int iparam)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT numerator = -F_F(2.0) * pow(params[iparam].c, F_F(2.0)) * (params[iparam].h - costheta);
|
|
|
|
F_CFLOAT denominator = pow(pow(params[iparam].d, F_F(2.0)) +
|
2012-10-08 23:29:55 +08:00
|
|
|
pow(params[iparam].h - costheta, F_F(2.0)), F_F(2.0));
|
|
|
|
return params[iparam].gamma * numerator / denominator;
|
|
|
|
}
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ inline F_CFLOAT zeta(int iparam, const F_CFLOAT rsqij, const F_CFLOAT rsqik,
|
|
|
|
F_CFLOAT3 &delij, F_CFLOAT3 &delik)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT rij, rik, costheta, arg, ex_delr;
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
rij = sqrt(rsqij);
|
|
|
|
rik = sqrt(rsqik);
|
|
|
|
costheta = vec3_dot(delij, delik) / (rij * rik);
|
|
|
|
|
|
|
|
arg = (params[iparam].powermint == 3) ? (params[iparam].lam3 * (rij - rik) * params[iparam].lam3 * (rij - rik) * params[iparam].lam3 * (rij - rik)) : params[iparam].lam3 * (rij - rik);
|
|
|
|
|
|
|
|
if(arg > F_F(69.0776)) ex_delr = F_F(1.e30);
|
|
|
|
else if(arg < -F_F(69.0776)) ex_delr = F_F(0.0);
|
|
|
|
else ex_delr = exp(arg);
|
|
|
|
|
|
|
|
return ters_fc(rik, params[iparam].bigr, params[iparam].bigd) * ex_delr * params[iparam].gamma * (F_F(1.0) + (params[iparam].c * params[iparam].c / (params[iparam].d * params[iparam].d)) -
|
|
|
|
(params[iparam].c * params[iparam].c) / ((params[iparam].d * params[iparam].d) + (params[iparam].h - costheta) * (params[iparam].h - costheta)));
|
|
|
|
}
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ void repulsive(int iparam, F_CFLOAT rsq, F_CFLOAT &fforce,
|
|
|
|
int eflag, ENERGY_CFLOAT &eng)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT r, tmp_fc, tmp_fc_d, tmp_exp;
|
2012-10-08 23:29:55 +08:00
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT ters_R = params[iparam].bigr;
|
|
|
|
F_CFLOAT ters_D = params[iparam].bigd;
|
2012-10-08 23:29:55 +08:00
|
|
|
r = sqrt(rsq);
|
|
|
|
tmp_fc = ters_fc(r, ters_R, ters_D);
|
|
|
|
tmp_fc_d = ters_fc_d(r, ters_R, ters_D);
|
|
|
|
tmp_exp = exp(-params[iparam].lam1 * r);
|
|
|
|
|
|
|
|
if(!_zbl) {
|
|
|
|
fforce = -params[iparam].biga * tmp_exp * (tmp_fc_d - tmp_fc * params[iparam].lam1) / r;
|
|
|
|
|
|
|
|
if(eflag) eng += tmp_fc * params[iparam].biga * tmp_exp;
|
|
|
|
} else {
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT const fforce_ters = params[iparam].biga * tmp_exp * (tmp_fc_d - tmp_fc * params[iparam].lam1);
|
|
|
|
ENERGY_CFLOAT eng_ters = tmp_fc * params[iparam].biga * tmp_exp;
|
2012-10-08 23:29:55 +08:00
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT r_ov_a = r / params[iparam].a_ij;
|
|
|
|
F_CFLOAT phi = F_F(0.1818) * exp(-F_F(3.2) * r_ov_a) + F_F(0.5099) * exp(-F_F(0.9423) * r_ov_a) +
|
2012-10-08 23:29:55 +08:00
|
|
|
F_F(0.2802) * exp(-F_F(0.4029) * r_ov_a) + F_F(0.02817) * exp(-F_F(0.2016) * r_ov_a);
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT dphi = (F_F(1.0) / params[iparam].a_ij) * (-F_F(3.2) * F_F(0.1818) * exp(-F_F(3.2) * r_ov_a) -
|
2012-10-08 23:29:55 +08:00
|
|
|
F_F(0.9423) * F_F(0.5099) * exp(-F_F(0.9423) * r_ov_a) -
|
|
|
|
F_F(0.4029) * F_F(0.2802) * exp(-F_F(0.4029) * r_ov_a) -
|
|
|
|
F_F(0.2016) * F_F(0.02817) * exp(-F_F(0.2016) * r_ov_a));
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT fforce_ZBL = params[iparam].premult / (-r * r) * phi + params[iparam].premult / r * dphi;
|
|
|
|
ENERGY_CFLOAT eng_ZBL = params[iparam].premult * (F_F(1.0) / r) * phi;
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
fforce = -(-F_fermi_d(r, iparam) * (eng_ZBL - eng_ters) + fforce_ZBL + F_fermi(r, iparam) * (fforce_ters - fforce_ZBL)) / r;
|
|
|
|
|
|
|
|
if(eflag)
|
|
|
|
eng += eng_ZBL + F_fermi(r, iparam) * (eng_ters - eng_ZBL);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ---------------------------------------------------------------------- */
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ inline F_CFLOAT ters_fa(F_CFLOAT r, int iparam, F_CFLOAT ters_R, F_CFLOAT ters_D)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
|
|
|
if(r > ters_R + ters_D) return F_F(0.0);
|
|
|
|
|
|
|
|
if(_zbl)
|
|
|
|
return -params[iparam].bigb * exp(-params[iparam].lam2 * r) * ters_fc(r, ters_R, ters_D) * F_fermi(r, iparam);
|
|
|
|
else
|
|
|
|
return -params[iparam].bigb * exp(-params[iparam].lam2 * r) * ters_fc(r, ters_R, ters_D);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ---------------------------------------------------------------------- */
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ inline F_CFLOAT ters_fa_d(F_CFLOAT r, int iparam, F_CFLOAT ters_R, F_CFLOAT ters_D)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
|
|
|
if(r > ters_R + ters_D) return F_F(0.0);
|
|
|
|
|
|
|
|
if(_zbl)
|
|
|
|
return params[iparam].bigb * exp(-params[iparam].lam2 * r) *
|
|
|
|
((params[iparam].lam2 * ters_fc(r, ters_R, ters_D) - ters_fc_d(r, ters_R, ters_D)) * F_fermi(r, iparam)
|
|
|
|
- ters_fc(r, ters_R, ters_D) * F_fermi_d(r, iparam));
|
|
|
|
else
|
|
|
|
return params[iparam].bigb * exp(-params[iparam].lam2 * r) *
|
|
|
|
(params[iparam].lam2 * ters_fc(r, ters_R, ters_D) - ters_fc_d(r, ters_R, ters_D));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ---------------------------------------------------------------------- */
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ inline F_CFLOAT ters_bij(F_CFLOAT zeta, int iparam)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT tmp = params[iparam].beta * zeta;
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
if(tmp > params[iparam].c1) return F_F(1.0) / sqrt(tmp);
|
|
|
|
|
|
|
|
if(tmp > params[iparam].c2)
|
|
|
|
return (F_F(1.0) - pow(tmp, -params[iparam].powern) / (F_F(2.0) * params[iparam].powern)) / sqrt(tmp);
|
|
|
|
|
|
|
|
if(tmp < params[iparam].c4) return F_F(1.0);
|
|
|
|
|
|
|
|
if(tmp < params[iparam].c3)
|
|
|
|
return F_F(1.0) - pow(tmp, params[iparam].powern) / (F_F(2.0) * params[iparam].powern);
|
|
|
|
|
|
|
|
return pow(F_F(1.0) + pow(tmp, params[iparam].powern), -F_F(1.0) / (F_F(2.0) * params[iparam].powern));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ---------------------------------------------------------------------- */
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ inline F_CFLOAT ters_bij_d(F_CFLOAT zeta, int iparam)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT tmp = params[iparam].beta * zeta;
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
if(tmp > params[iparam].c1) return params[iparam].beta * -F_F(0.5) * pow(tmp, -F_F(1.5));
|
|
|
|
|
|
|
|
if(tmp > params[iparam].c2)
|
|
|
|
return params[iparam].beta * (-F_F(0.5) * pow(tmp, -F_F(1.5)) *
|
|
|
|
(F_F(1.0) - F_F(0.5) * (F_F(1.0) + F_F(1.0) / (F_F(2.0) * params[iparam].powern)) *
|
|
|
|
pow(tmp, -params[iparam].powern)));
|
|
|
|
|
|
|
|
if(tmp < params[iparam].c4) return F_F(0.0);
|
|
|
|
|
|
|
|
if(tmp < params[iparam].c3)
|
|
|
|
return -F_F(0.5) * params[iparam].beta * pow(tmp, params[iparam].powern - F_F(1.0));
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT tmp_n = pow(tmp, params[iparam].powern);
|
2012-10-08 23:29:55 +08:00
|
|
|
return -F_F(0.5) * pow(F_F(1.0) + tmp_n, -F_F(1.0) - (F_F(1.0) / (F_F(2.0) * params[iparam].powern))) * tmp_n / zeta;
|
|
|
|
}
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ void force_zeta(int iparam, F_CFLOAT rsq, F_CFLOAT zeta_ij,
|
|
|
|
F_CFLOAT &fforce, F_CFLOAT &prefactor,
|
|
|
|
int eflag, F_CFLOAT &eng)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT r, fa, fa_d, bij;
|
|
|
|
F_CFLOAT ters_R = params[iparam].bigr;
|
|
|
|
F_CFLOAT ters_D = params[iparam].bigd;
|
2012-10-08 23:29:55 +08:00
|
|
|
r = sqrt(rsq);
|
|
|
|
fa = ters_fa(r, iparam, ters_R, ters_D);
|
|
|
|
fa_d = ters_fa_d(r, iparam, ters_R, ters_D);
|
|
|
|
bij = ters_bij(zeta_ij, iparam);
|
|
|
|
fforce = F_F(0.5) * bij * fa_d / r;
|
|
|
|
prefactor = -F_F(0.5) * fa * ters_bij_d(zeta_ij, iparam);
|
|
|
|
|
|
|
|
if(eflag) eng += bij * fa;
|
|
|
|
}
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ void force_zeta_prefactor_force(int iparam, F_CFLOAT rsq, F_CFLOAT zeta_ij,
|
|
|
|
F_CFLOAT &fforce, F_CFLOAT &prefactor)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT r, fa, fa_d, bij;
|
|
|
|
F_CFLOAT ters_R = params[iparam].bigr;
|
|
|
|
F_CFLOAT ters_D = params[iparam].bigd;
|
2012-10-08 23:29:55 +08:00
|
|
|
r = sqrt(rsq);
|
|
|
|
fa = ters_fa(r, iparam, ters_R, ters_D);
|
|
|
|
fa_d = ters_fa_d(r, iparam, ters_R, ters_D);
|
|
|
|
bij = ters_bij(zeta_ij, iparam);
|
|
|
|
fforce = F_F(0.5) * bij * fa_d / r;
|
|
|
|
prefactor = -F_F(0.5) * fa * ters_bij_d(zeta_ij, iparam);
|
|
|
|
}
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ void force_zeta_prefactor(int iparam, F_CFLOAT rsq, F_CFLOAT zeta_ij,
|
|
|
|
F_CFLOAT &prefactor)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT r, fa;
|
2012-10-08 23:29:55 +08:00
|
|
|
r = sqrt(rsq);
|
|
|
|
fa = ters_fa(r, iparam, params[iparam].bigr, params[iparam].bigd);
|
|
|
|
prefactor = -F_F(0.5) * fa * ters_bij_d(zeta_ij, iparam);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ void costheta_d(F_CFLOAT3 &rij_hat, F_CFLOAT &rij,
|
|
|
|
F_CFLOAT3 &rik_hat, F_CFLOAT &rik,
|
|
|
|
F_CFLOAT3 &dri, F_CFLOAT3 &drj, F_CFLOAT3 &drk)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
|
|
|
// first element is derivative wrt Ri, second wrt Rj, third wrt Rk
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT cos_theta = vec3_dot(rij_hat, rik_hat);
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
vec3_scaleadd(-cos_theta, rij_hat, rik_hat, drj);
|
|
|
|
vec3_scale(F_F(1.0) / rij, drj, drj);
|
|
|
|
vec3_scaleadd(-cos_theta, rik_hat, rij_hat, drk);
|
|
|
|
vec3_scale(F_F(1.0) / rik, drk, drk);
|
|
|
|
vec3_add(drj, drk, dri);
|
|
|
|
vec3_scale(-F_F(1.0), dri, dri);
|
|
|
|
}
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ void ters_zetaterm_d(F_CFLOAT prefactor,
|
|
|
|
F_CFLOAT3 &rij_hat, F_CFLOAT rij,
|
|
|
|
F_CFLOAT3 &rik_hat, F_CFLOAT rik,
|
|
|
|
F_CFLOAT3 &dri, F_CFLOAT3 &drj, F_CFLOAT3 &drk,
|
2012-10-08 23:29:55 +08:00
|
|
|
int iparam)
|
|
|
|
{
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT ex_delr, ex_delr_d, tmp;
|
|
|
|
F_CFLOAT3 dcosdri, dcosdrj, dcosdrk;
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
if(params[iparam].powermint == 3) tmp = (params[iparam].lam3 * (rij - rik) * params[iparam].lam3 * (rij - rik) * params[iparam].lam3 * (rij - rik));
|
|
|
|
else tmp = params[iparam].lam3 * (rij - rik);
|
|
|
|
|
|
|
|
if(tmp > F_F(69.0776)) ex_delr = F_F(1.e30);
|
|
|
|
else if(tmp < -F_F(69.0776)) ex_delr = F_F(0.0);
|
|
|
|
else ex_delr = exp(tmp);
|
|
|
|
|
|
|
|
if(params[iparam].powermint == 3)
|
|
|
|
ex_delr_d = F_F(3.0) * (params[iparam].lam3 * params[iparam].lam3 * params[iparam].lam3) * (rij - rik) * (rij - rik) * ex_delr;
|
|
|
|
else ex_delr_d = params[iparam].lam3 * ex_delr;
|
|
|
|
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
const F_CFLOAT cos_theta = vec3_dot(rij_hat, rik_hat);
|
2012-10-08 23:29:55 +08:00
|
|
|
costheta_d(rij_hat, rij, rik_hat, rik, dcosdri, dcosdrj, dcosdrk);
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
const F_CFLOAT gijk = params[iparam].gamma * (F_F(1.0) + (params[iparam].c * params[iparam].c) / (params[iparam].d * params[iparam].d) -
|
2012-10-08 23:29:55 +08:00
|
|
|
(params[iparam].c * params[iparam].c) / (params[iparam].d * params[iparam].d + (params[iparam].h - cos_theta) * (params[iparam].h - cos_theta)));
|
2014-10-07 06:59:05 +08:00
|
|
|
const F_CFLOAT numerator = -F_F(2.0) * params[iparam].c * params[iparam].c * (params[iparam].h - cos_theta);
|
|
|
|
const F_CFLOAT denominator = (params[iparam].d * params[iparam].d) +
|
2012-10-08 23:29:55 +08:00
|
|
|
(params[iparam].h - cos_theta) * (params[iparam].h - cos_theta);
|
2014-10-07 06:59:05 +08:00
|
|
|
const F_CFLOAT gijk_d = params[iparam].gamma * numerator / (denominator * denominator); // compute the derivative wrt Ri
|
2012-10-08 23:29:55 +08:00
|
|
|
// dri = -dfc*gijk*ex_delr*rik_hat;
|
|
|
|
// dri += fc*gijk_d*ex_delr*dcosdri;
|
|
|
|
// dri += fc*gijk*ex_delr_d*(rik_hat - rij_hat);
|
2014-10-07 06:59:05 +08:00
|
|
|
const F_CFLOAT fc = ters_fc(rik, params[iparam].bigr, params[iparam].bigd);
|
|
|
|
const F_CFLOAT dfc = ters_fc_d(rik, params[iparam].bigr, params[iparam].bigd);
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
|
|
|
|
vec3_scale(-dfc * gijk * ex_delr, rik_hat, dri);
|
|
|
|
vec3_scaleadd(fc * gijk_d * ex_delr, dcosdri, dri, dri);
|
|
|
|
vec3_scaleadd(fc * gijk * ex_delr_d, rik_hat, dri, dri);
|
|
|
|
vec3_scaleadd(-fc * gijk * ex_delr_d, rij_hat, dri, dri);
|
|
|
|
vec3_scale(prefactor, dri, dri);
|
|
|
|
// compute the derivative wrt Rj
|
|
|
|
// drj = fc*gijk_d*ex_delr*dcosdrj;
|
|
|
|
// drj += fc*gijk*ex_delr_d*rij_hat;
|
|
|
|
|
|
|
|
vec3_scale(fc * gijk_d * ex_delr, dcosdrj, drj);
|
|
|
|
vec3_scaleadd(fc * gijk * ex_delr_d, rij_hat, drj, drj);
|
|
|
|
vec3_scale(prefactor, drj, drj);
|
|
|
|
|
|
|
|
// compute the derivative wrt Rk
|
|
|
|
// drk = dfc*gijk*ex_delr*rik_hat;
|
|
|
|
// drk += fc*gijk_d*ex_delr*dcosdrk;
|
|
|
|
// drk += -fc*gijk*ex_delr_d*rik_hat;
|
|
|
|
|
|
|
|
vec3_scale(dfc * gijk * ex_delr, rik_hat, drk);
|
|
|
|
vec3_scaleadd(fc * gijk_d * ex_delr, dcosdrk, drk, drk);
|
|
|
|
vec3_scaleadd(-fc * gijk * ex_delr_d, rik_hat, drk, drk);
|
|
|
|
vec3_scale(prefactor, drk, drk);
|
|
|
|
}
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ void ters_zetaterm_d_fi(F_CFLOAT &prefactor,
|
|
|
|
F_CFLOAT3 &rij_hat, F_CFLOAT &rij,
|
|
|
|
F_CFLOAT3 &rik_hat, F_CFLOAT &rik,
|
|
|
|
F_CFLOAT3 &dri, int &iparam)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT ex_delr, ex_delr_d, tmp;
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
if(params[iparam].powermint == 3) tmp = (params[iparam].lam3 * (rij - rik) * params[iparam].lam3 * (rij - rik) * params[iparam].lam3 * (rij - rik));
|
|
|
|
else tmp = params[iparam].lam3 * (rij - rik);
|
|
|
|
|
|
|
|
if(tmp > F_F(69.0776)) ex_delr = F_F(1.e30);
|
|
|
|
else if(tmp < -F_F(69.0776)) ex_delr = F_F(0.0);
|
|
|
|
else ex_delr = exp(tmp);
|
|
|
|
|
|
|
|
if(params[iparam].powermint == 3)
|
|
|
|
ex_delr_d = F_F(3.0) * (params[iparam].lam3 * params[iparam].lam3 * params[iparam].lam3) * (rij - rik) * (rij - rik) * ex_delr;
|
|
|
|
else ex_delr_d = params[iparam].lam3 * ex_delr;
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
const F_CFLOAT cos_theta = vec3_dot(rij_hat, rik_hat);
|
2012-10-08 23:29:55 +08:00
|
|
|
//costheta_d(rij_hat,rij,rik_hat,rik,dcosdri,dcosdrj,dcosdrk);
|
|
|
|
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT3 dcosdri;
|
2012-10-08 23:29:55 +08:00
|
|
|
vec3_scaleadd(-cos_theta, rij_hat, rik_hat, dri);
|
|
|
|
vec3_scale(F_F(1.0) / rij, dri, dri);
|
|
|
|
vec3_scaleadd(-cos_theta, rik_hat, rij_hat, dcosdri);
|
|
|
|
vec3_scale(F_F(1.0) / rik, dcosdri, dcosdri);
|
|
|
|
vec3_add(dri, dcosdri, dcosdri);
|
|
|
|
vec3_scale(-F_F(1.0), dcosdri, dcosdri);
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
const F_CFLOAT gijk = params[iparam].gamma * (F_F(1.0) + (params[iparam].c * params[iparam].c) / (params[iparam].d * params[iparam].d) -
|
2012-10-08 23:29:55 +08:00
|
|
|
(params[iparam].c * params[iparam].c) / (params[iparam].d * params[iparam].d + (params[iparam].h - cos_theta) * (params[iparam].h - cos_theta)));
|
2014-10-07 06:59:05 +08:00
|
|
|
const F_CFLOAT numerator = -F_F(2.0) * params[iparam].c * params[iparam].c * (params[iparam].h - cos_theta);
|
|
|
|
const F_CFLOAT denominator = (params[iparam].d * params[iparam].d) +
|
2012-10-08 23:29:55 +08:00
|
|
|
(params[iparam].h - cos_theta) * (params[iparam].h - cos_theta);
|
2014-10-07 06:59:05 +08:00
|
|
|
const F_CFLOAT gijk_d = params[iparam].gamma * numerator / (denominator * denominator); // compute the derivative wrt Ri
|
2012-10-08 23:29:55 +08:00
|
|
|
//
|
2014-10-07 06:59:05 +08:00
|
|
|
const F_CFLOAT fc = ters_fc(rik, params[iparam].bigr, params[iparam].bigd);
|
|
|
|
const F_CFLOAT dfc = ters_fc_d(rik, params[iparam].bigr, params[iparam].bigd);
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
vec3_scale(-dfc * gijk * ex_delr, rik_hat, dri);
|
|
|
|
vec3_scaleadd(fc * gijk_d * ex_delr, dcosdri, dri, dri);
|
|
|
|
vec3_scaleadd(fc * gijk * ex_delr_d, rik_hat, dri, dri);
|
|
|
|
vec3_scaleadd(-fc * gijk * ex_delr_d, rij_hat, dri, dri);
|
|
|
|
vec3_scale(prefactor, dri, dri);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ void ters_zetaterm_d_fj(F_CFLOAT &prefactor,
|
|
|
|
F_CFLOAT3 &rij_hat, F_CFLOAT &rij,
|
|
|
|
F_CFLOAT3 &rik_hat, F_CFLOAT &rik,
|
|
|
|
F_CFLOAT3 &drj, int &iparam)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT ex_delr, ex_delr_d, tmp;
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
if(params[iparam].powermint == 3) tmp = (params[iparam].lam3 * (rij - rik) * params[iparam].lam3 * (rij - rik) * params[iparam].lam3 * (rij - rik));
|
|
|
|
else tmp = params[iparam].lam3 * (rij - rik);
|
|
|
|
|
|
|
|
if(tmp > F_F(69.0776)) ex_delr = F_F(1.e30);
|
|
|
|
else if(tmp < -F_F(69.0776)) ex_delr = F_F(0.0);
|
|
|
|
else ex_delr = exp(tmp);
|
|
|
|
|
|
|
|
if(params[iparam].powermint == 3)
|
|
|
|
ex_delr_d = F_F(3.0) * (params[iparam].lam3 * params[iparam].lam3 * params[iparam].lam3) * (rij - rik) * (rij - rik) * ex_delr;
|
|
|
|
else ex_delr_d = params[iparam].lam3 * ex_delr;
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
const F_CFLOAT cos_theta = vec3_dot(rij_hat, rik_hat);
|
2012-10-08 23:29:55 +08:00
|
|
|
vec3_scaleadd(-cos_theta, rij_hat, rik_hat, drj);
|
|
|
|
vec3_scale(F_F(1.0) / rij, drj, drj);
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
const F_CFLOAT gijk = params[iparam].gamma * (F_F(1.0) + (params[iparam].c * params[iparam].c) / (params[iparam].d * params[iparam].d) -
|
2012-10-08 23:29:55 +08:00
|
|
|
(params[iparam].c * params[iparam].c) / (params[iparam].d * params[iparam].d + (params[iparam].h - cos_theta) * (params[iparam].h - cos_theta)));
|
2014-10-07 06:59:05 +08:00
|
|
|
const F_CFLOAT numerator = -F_F(2.0) * params[iparam].c * params[iparam].c * (params[iparam].h - cos_theta);
|
|
|
|
const F_CFLOAT denominator = (params[iparam].d * params[iparam].d) +
|
2012-10-08 23:29:55 +08:00
|
|
|
(params[iparam].h - cos_theta) * (params[iparam].h - cos_theta);
|
2014-10-07 06:59:05 +08:00
|
|
|
const F_CFLOAT gijk_d = params[iparam].gamma * numerator / (denominator * denominator); // compute the derivative wrt Ri
|
2012-10-08 23:29:55 +08:00
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
const F_CFLOAT fc = ters_fc(rik, params[iparam].bigr, params[iparam].bigd);
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
vec3_scale(fc * gijk_d * ex_delr, drj, drj);
|
|
|
|
vec3_scaleadd(fc * gijk * ex_delr_d, rij_hat, drj, drj);
|
|
|
|
vec3_scale(prefactor, drj, drj);
|
|
|
|
}
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ void ters_zetaterm_d_fk(F_CFLOAT &prefactor,
|
|
|
|
F_CFLOAT3 &rij_hat, F_CFLOAT &rij,
|
|
|
|
F_CFLOAT3 &rik_hat, F_CFLOAT &rik,
|
|
|
|
F_CFLOAT3 &drk, int &iparam)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT ex_delr, ex_delr_d, tmp;
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
if(params[iparam].powermint == 3) tmp = (params[iparam].lam3 * (rij - rik) * params[iparam].lam3 * (rij - rik) * params[iparam].lam3 * (rij - rik));
|
|
|
|
else tmp = params[iparam].lam3 * (rij - rik);
|
|
|
|
|
|
|
|
if(tmp > F_F(69.0776)) ex_delr = F_F(1.e30);
|
|
|
|
else if(tmp < -F_F(69.0776)) ex_delr = F_F(0.0);
|
|
|
|
else ex_delr = exp(tmp);
|
|
|
|
|
|
|
|
if(params[iparam].powermint == 3)
|
|
|
|
ex_delr_d = F_F(3.0) * (params[iparam].lam3 * params[iparam].lam3 * params[iparam].lam3) * (rij - rik) * (rij - rik) * ex_delr;
|
|
|
|
else ex_delr_d = params[iparam].lam3 * ex_delr;
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
const F_CFLOAT cos_theta = vec3_dot(rij_hat, rik_hat);
|
2012-10-08 23:29:55 +08:00
|
|
|
vec3_scaleadd(-cos_theta, rik_hat, rij_hat, drk);
|
|
|
|
vec3_scale(F_F(1.0) / rik, drk, drk);
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
const F_CFLOAT gijk = params[iparam].gamma * (F_F(1.0) + (params[iparam].c * params[iparam].c) / (params[iparam].d * params[iparam].d) -
|
2012-10-08 23:29:55 +08:00
|
|
|
(params[iparam].c * params[iparam].c) / (params[iparam].d * params[iparam].d + (params[iparam].h - cos_theta) * (params[iparam].h - cos_theta)));
|
2014-10-07 06:59:05 +08:00
|
|
|
const F_CFLOAT numerator = -F_F(2.0) * params[iparam].c * params[iparam].c * (params[iparam].h - cos_theta);
|
|
|
|
const F_CFLOAT denominator = (params[iparam].d * params[iparam].d) +
|
2012-10-08 23:29:55 +08:00
|
|
|
(params[iparam].h - cos_theta) * (params[iparam].h - cos_theta);
|
2014-10-07 06:59:05 +08:00
|
|
|
const F_CFLOAT gijk_d = params[iparam].gamma * numerator / (denominator * denominator); // compute the derivative wrt Ri
|
2012-10-08 23:29:55 +08:00
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
const F_CFLOAT fc = ters_fc(rik, params[iparam].bigr, params[iparam].bigd);
|
|
|
|
const F_CFLOAT dfc = ters_fc_d(rik, params[iparam].bigr, params[iparam].bigd);
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
vec3_scale(fc * gijk_d * ex_delr, drk, drk);
|
|
|
|
vec3_scaleadd(dfc * gijk * ex_delr, rik_hat, drk, drk);
|
|
|
|
vec3_scaleadd(-fc * gijk * ex_delr_d, rik_hat, drk, drk);
|
|
|
|
vec3_scale(prefactor, drk, drk);
|
|
|
|
}
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ void attractive(int iparam, F_CFLOAT prefactor,
|
|
|
|
F_CFLOAT4 &delij,
|
|
|
|
F_CFLOAT4 &delik,
|
|
|
|
F_CFLOAT3 &fi, F_CFLOAT3 &fj, F_CFLOAT3 &fk)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT3 rij_hat, rik_hat;
|
|
|
|
F_CFLOAT rij, rijinv, rik, rikinv;
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
rij = sqrt(delij.w);
|
|
|
|
rijinv = F_F(1.0) / rij;
|
|
|
|
vec3_scale(rijinv, delij, rij_hat);
|
|
|
|
|
|
|
|
rik = sqrt(delik.w);
|
|
|
|
rikinv = F_F(1.0) / rik;
|
|
|
|
vec3_scale(rikinv, delik, rik_hat);
|
|
|
|
|
|
|
|
ters_zetaterm_d(prefactor, rij_hat, rij, rik_hat, rik, fi, fj, fk, iparam);
|
|
|
|
}
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ void attractive_fi(int &iparam, F_CFLOAT &prefactor,
|
|
|
|
F_CFLOAT4 &delij,
|
|
|
|
F_CFLOAT4 &delik,
|
|
|
|
F_CFLOAT3 &f)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT3 rij_hat, rik_hat;
|
|
|
|
F_CFLOAT rij, rijinv, rik, rikinv;
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
rij = sqrt(delij.w);
|
|
|
|
rijinv = F_F(1.0) / rij;
|
|
|
|
vec3_scale(rijinv, delij, rij_hat);
|
|
|
|
|
|
|
|
rik = sqrt(delik.w);
|
|
|
|
rikinv = F_F(1.0) / rik;
|
|
|
|
vec3_scale(rikinv, delik, rik_hat);
|
|
|
|
|
|
|
|
ters_zetaterm_d_fi(prefactor, rij_hat, rij, rik_hat, rik, f, iparam);
|
|
|
|
}
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ void attractive_fj(int iparam, F_CFLOAT prefactor,
|
|
|
|
F_CFLOAT4 &delij,
|
|
|
|
F_CFLOAT4 &delik,
|
|
|
|
F_CFLOAT3 &f)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT3 rij_hat, rik_hat;
|
|
|
|
F_CFLOAT rij, rijinv, rik, rikinv;
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
rij = sqrt(delij.w);
|
|
|
|
rijinv = F_F(1.0) / rij;
|
|
|
|
vec3_scale(rijinv, delij, rij_hat);
|
|
|
|
|
|
|
|
rik = sqrt(delik.w);
|
|
|
|
rikinv = F_F(1.0) / rik;
|
|
|
|
vec3_scale(rikinv, delik, rik_hat);
|
|
|
|
|
|
|
|
ters_zetaterm_d_fj(prefactor, rij_hat, rij, rik_hat, rik, f, iparam);
|
|
|
|
}
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__device__ void attractive_fk(int iparam, F_CFLOAT prefactor,
|
|
|
|
F_CFLOAT4 &delij,
|
|
|
|
F_CFLOAT4 &delik,
|
|
|
|
F_CFLOAT3 &f)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT3 rij_hat, rik_hat;
|
|
|
|
F_CFLOAT rij, rijinv, rik, rikinv;
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
rij = sqrt(delij.w);
|
|
|
|
rijinv = F_F(1.0) / rij;
|
|
|
|
vec3_scale(rijinv, delij, rij_hat);
|
|
|
|
|
|
|
|
rik = sqrt(delik.w);
|
|
|
|
rikinv = F_F(1.0) / rik;
|
|
|
|
vec3_scale(rikinv, delik, rik_hat);
|
|
|
|
|
|
|
|
ters_zetaterm_d_fk(prefactor, rij_hat, rij, rik_hat, rik, f, iparam);
|
|
|
|
}
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__global__ void Pair_Tersoff_Kernel_TpA_RIJ()//F_CFLOAT4* _glob_r_ij,int* _glob_numneigh_red,int* _glob_neighbors_red,int* _glob_neightype_red)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
|
|
|
int ii = (blockIdx.x * gridDim.y + blockIdx.y) * blockDim.x + threadIdx.x;
|
|
|
|
|
|
|
|
if(ii >= _nall) return;
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
X_CFLOAT4 myxtype;
|
|
|
|
F_CFLOAT4 delij;
|
|
|
|
F_CFLOAT xtmp, ytmp, ztmp;
|
2012-10-08 23:29:55 +08:00
|
|
|
int itype, jnum, i, j;
|
|
|
|
int* jlist;
|
|
|
|
int neigh_red = 0;
|
|
|
|
i = ii;//_ilist[ii];
|
|
|
|
myxtype = fetchXType(i);
|
|
|
|
|
|
|
|
xtmp = myxtype.x;
|
|
|
|
ytmp = myxtype.y;
|
|
|
|
ztmp = myxtype.z;
|
|
|
|
itype = map[(static_cast <int>(myxtype.w))];
|
|
|
|
|
|
|
|
jnum = _numneigh[i];
|
|
|
|
jlist = &_neighbors[i];
|
|
|
|
|
|
|
|
__syncthreads();
|
|
|
|
|
|
|
|
for(int jj = 0; jj < jnum; jj++) {
|
|
|
|
if(jj < jnum) {
|
|
|
|
|
|
|
|
j = jlist[jj * _nall];
|
|
|
|
j &= NEIGHMASK;
|
|
|
|
myxtype = fetchXType(j);
|
|
|
|
delij.x = xtmp - myxtype.x;
|
|
|
|
delij.y = ytmp - myxtype.y;
|
|
|
|
delij.z = ztmp - myxtype.z;
|
|
|
|
int jtype = map[(static_cast <int>(myxtype.w))];
|
|
|
|
int iparam_ij = elem2param[(itype * nelements + jtype) * nelements + jtype];
|
|
|
|
delij.w = vec3_dot(delij, delij);
|
|
|
|
|
|
|
|
if(delij.w < params[iparam_ij].cutsq) {
|
|
|
|
_glob_neighbors_red[i + neigh_red * _nall] = j;
|
|
|
|
_glob_neightype_red[i + neigh_red * _nall] = jtype;
|
|
|
|
_glob_r_ij[i + neigh_red * _nall] = delij;
|
|
|
|
neigh_red++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
_glob_numneigh_red[i] = neigh_red;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
__global__ void Pair_Tersoff_Kernel_TpA_ZetaIJ()//F_CFLOAT* _glob_zeta_ij,F_CFLOAT4* _glob_r_ij,int* _glob_numneigh_red,int* _glob_neighbors_red,int* _glob_neightype_red)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
|
|
|
|
|
|
|
int ii = (blockIdx.x * gridDim.y + blockIdx.y) * blockDim.x + threadIdx.x;
|
|
|
|
|
|
|
|
if(ii >= _nall) return;
|
|
|
|
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT4 delij;
|
|
|
|
F_CFLOAT4 delik;
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
int itype, jnum, i, j;
|
|
|
|
int* jlist;
|
|
|
|
i = ii;
|
|
|
|
itype = map[(static_cast <int>(_type[i]))];
|
|
|
|
|
|
|
|
jnum = _glob_numneigh_red[i];
|
|
|
|
jlist = &_glob_neighbors_red[i];
|
|
|
|
|
|
|
|
__syncthreads();
|
|
|
|
|
|
|
|
for(int jj = 0; jj < jnum; jj++) {
|
|
|
|
if(jj < jnum) {
|
|
|
|
|
|
|
|
j = jlist[jj * _nall];
|
|
|
|
j &= NEIGHMASK;
|
|
|
|
int jtype = _glob_neightype_red[i + jj * _nall];
|
|
|
|
delij = _glob_r_ij[i + jj * _nall];
|
|
|
|
|
|
|
|
int iparam_ij = elem2param[(itype * nelements + jtype) * nelements + jtype];
|
|
|
|
|
|
|
|
if(delij.w < params[iparam_ij].cutsq) {
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT zeta_ij = 0.0;
|
|
|
|
F_CFLOAT3 delij3 = {delij.x, delij.y, delij.z};
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
for(int kk = 0; kk < jnum; kk++) {
|
|
|
|
if(jj == kk) continue;
|
|
|
|
|
|
|
|
int k = jlist[kk * _nall];
|
|
|
|
k &= NEIGHMASK;
|
|
|
|
|
|
|
|
int ktype = _glob_neightype_red[i + kk * _nall];
|
|
|
|
delik = _glob_r_ij[i + kk * _nall];
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT3 delik3 = {delik.x, delik.y, delik.z};
|
2012-10-08 23:29:55 +08:00
|
|
|
int iparam_ijk = elem2param[(itype * nelements + jtype) * nelements + ktype];
|
2014-10-07 06:59:05 +08:00
|
|
|
const F_CFLOAT rsqki = delik.w;
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
if(rsqki <= params[iparam_ijk].cutsq)
|
|
|
|
zeta_ij += zeta(iparam_ijk, delij.w, rsqki, delij3, delik3);
|
|
|
|
}
|
|
|
|
|
|
|
|
_glob_zeta_ij[i + jj * _nall] = zeta_ij;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//back3: num 12 steps 10: ZetaIJ/TPA 0.255/0.106
|
|
|
|
//back5: num 12 steps 10: ZetaIJ/TPA 0.257/0.098
|
|
|
|
//back6: num 12 steps 10: ZetaIJ/TPA 0.027/0.097 /rij berechnung extra
|
|
|
|
//back12: num 12 steps 10: ZetaIJ/TPA 0.026/0.070
|
|
|
|
//back15: num 12 steps 10: ZetaIJ/TPA 0.0137/0.0287 //pow beseitigt
|
|
|
|
// num 12 steps 10: ZetaIJ/TPA 0.0137/0.027
|
|
|
|
template <int eflag, int vflagm>
|
2014-10-07 06:59:05 +08:00
|
|
|
__global__ void Pair_Tersoff_Kernel_TpA(int eflag_atom, int vflag_atom) //,F_CFLOAT* _glob_zeta_ij,F_CFLOAT4* _glob_r_ij,int* _glob_numneigh_red,int* _glob_neighbors_red,int* _glob_neightype_red)
|
2012-10-08 23:29:55 +08:00
|
|
|
{
|
2014-10-07 06:59:05 +08:00
|
|
|
ENERGY_CFLOAT evdwl = ENERGY_F(0.0);
|
2012-10-08 23:29:55 +08:00
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
ENERGY_CFLOAT* sharedE = &sharedmem[threadIdx.x];
|
|
|
|
ENERGY_CFLOAT* sharedV = &sharedmem[threadIdx.x];
|
2012-10-08 23:29:55 +08:00
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT* shared_F_F = (F_CFLOAT*) sharedmem;
|
2012-10-08 23:29:55 +08:00
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
if((eflag || eflag_atom) && (vflagm || vflag_atom)) shared_F_F = (F_CFLOAT*) &sharedmem[7 * blockDim.x];
|
|
|
|
else if(eflag) shared_F_F = (F_CFLOAT*) &sharedmem[blockDim.x];
|
|
|
|
else if(vflagm) shared_F_F = (F_CFLOAT*) &sharedmem[6 * blockDim.x];
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
shared_F_F += threadIdx.x;
|
|
|
|
|
|
|
|
if(eflag_atom || eflag) {
|
|
|
|
sharedE[0] = ENERGY_F(0.0);
|
|
|
|
sharedV += blockDim.x;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(vflagm || vflag_atom) {
|
|
|
|
sharedV[0 * blockDim.x] = ENERGY_F(0.0);
|
|
|
|
sharedV[1 * blockDim.x] = ENERGY_F(0.0);
|
|
|
|
sharedV[2 * blockDim.x] = ENERGY_F(0.0);
|
|
|
|
sharedV[3 * blockDim.x] = ENERGY_F(0.0);
|
|
|
|
sharedV[4 * blockDim.x] = ENERGY_F(0.0);
|
|
|
|
sharedV[5 * blockDim.x] = ENERGY_F(0.0);
|
|
|
|
}
|
|
|
|
|
|
|
|
int jnum_red = 0;
|
|
|
|
#define fxtmp shared_F_F[0]
|
|
|
|
#define fytmp shared_F_F[blockDim.x]
|
|
|
|
#define fztmp shared_F_F[2*blockDim.x]
|
|
|
|
//#define jnum_red (static_cast <int> (shared_F_F[3*blockDim.x]))
|
|
|
|
|
|
|
|
int ii = (blockIdx.x * gridDim.y + blockIdx.y) * blockDim.x + threadIdx.x;
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
X_CFLOAT4 myxtype_i, myxtype_j, myxtype_k;
|
|
|
|
F_CFLOAT4 delij, delik, deljk;
|
|
|
|
F_CFLOAT fpair;
|
|
|
|
F_CFLOAT prefactor_ij, prefactor_ji;
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
int itype, i, j;
|
|
|
|
int* jlist_red;
|
|
|
|
|
|
|
|
if(ii < _inum) {
|
|
|
|
i = _ilist[ii];
|
|
|
|
|
|
|
|
if(vflagm)
|
|
|
|
myxtype_i = fetchXType(i);
|
|
|
|
|
|
|
|
//itype=map[(static_cast <int> (myxtype_i.w))];
|
|
|
|
itype = map[_type[i]];
|
|
|
|
|
|
|
|
|
|
|
|
fxtmp = F_F(0.0);
|
|
|
|
fytmp = F_F(0.0);
|
|
|
|
fztmp = F_F(0.0);
|
|
|
|
|
|
|
|
|
|
|
|
//shared_F_F[3*blockDim.x] = _glob_numneigh_red[i];
|
|
|
|
jnum_red = _glob_numneigh_red[i];
|
|
|
|
jlist_red = &_glob_neighbors_red[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
__syncthreads();
|
|
|
|
|
|
|
|
#pragma unroll 1
|
|
|
|
|
|
|
|
for(int jj = 0; jj < jnum_red; jj++) {
|
|
|
|
if(i < _nlocal) {
|
|
|
|
fpair = F_F(0.0);
|
|
|
|
j = jlist_red[jj * _nall];
|
|
|
|
j &= NEIGHMASK;
|
|
|
|
|
|
|
|
if(vflagm)
|
|
|
|
myxtype_j = fetchXType(j);
|
|
|
|
|
|
|
|
int jtype = _glob_neightype_red[i + jj * _nall];
|
|
|
|
delij = _glob_r_ij[i + jj * _nall];
|
|
|
|
|
|
|
|
volatile int iparam_ij = elem2param[(itype * nelements + jtype) * nelements + jtype];
|
|
|
|
volatile int iparam_ji = elem2param[(jtype * nelements + itype) * nelements + itype];
|
|
|
|
|
|
|
|
if(delij.w < params[iparam_ij].cutsq) {
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT dxfp, dyfp, dzfp;
|
2012-10-08 23:29:55 +08:00
|
|
|
repulsive(iparam_ij, delij.w, fpair, eflag, evdwl);
|
|
|
|
fxtmp += dxfp = delij.x * fpair;
|
|
|
|
fytmp += dyfp = delij.y * fpair;
|
|
|
|
fztmp += dzfp = delij.z * fpair;
|
|
|
|
|
|
|
|
if(vflagm) {
|
|
|
|
sharedV[0 * blockDim.x] += delij.x * dxfp;
|
|
|
|
sharedV[1 * blockDim.x] += delij.y * dyfp;
|
|
|
|
sharedV[2 * blockDim.x] += delij.z * dzfp;
|
|
|
|
sharedV[3 * blockDim.x] += delij.x * dyfp;
|
|
|
|
sharedV[4 * blockDim.x] += delij.x * dzfp;
|
|
|
|
sharedV[5 * blockDim.x] += delij.y * dzfp;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
force_zeta(iparam_ij, delij.w, _glob_zeta_ij[i + jj * _nall], fpair, prefactor_ij, eflag, evdwl);
|
|
|
|
fxtmp -=
|
|
|
|
dxfp = delij.x * fpair;
|
|
|
|
fytmp -=
|
|
|
|
dyfp = delij.y * fpair;
|
|
|
|
fztmp -=
|
|
|
|
dzfp = delij.z * fpair;
|
|
|
|
|
|
|
|
if(vflagm) {
|
|
|
|
sharedV[0 * blockDim.x] -= ENERGY_F(2.0) * delij.x * dxfp;
|
|
|
|
sharedV[1 * blockDim.x] -= ENERGY_F(2.0) * delij.y * dyfp;
|
|
|
|
sharedV[2 * blockDim.x] -= ENERGY_F(2.0) * delij.z * dzfp;
|
|
|
|
sharedV[3 * blockDim.x] -= ENERGY_F(2.0) * delij.x * dyfp;
|
|
|
|
sharedV[4 * blockDim.x] -= ENERGY_F(2.0) * delij.x * dzfp;
|
|
|
|
sharedV[5 * blockDim.x] -= ENERGY_F(2.0) * delij.y * dzfp;
|
|
|
|
}
|
|
|
|
|
|
|
|
int j_jj = 0;
|
|
|
|
|
|
|
|
//#pragma unroll 1
|
|
|
|
for(int kk = 0; kk < _glob_numneigh_red[j]; kk++) {
|
|
|
|
if(_glob_neighbors_red[j + kk * _nall] == i) j_jj = kk;
|
|
|
|
}
|
|
|
|
|
|
|
|
force_zeta_prefactor_force(iparam_ji, delij.w, _glob_zeta_ij[j + j_jj * _nall], fpair, prefactor_ji);
|
|
|
|
|
|
|
|
fxtmp -=
|
|
|
|
dxfp = delij.x * fpair;
|
|
|
|
fytmp -=
|
|
|
|
dyfp = delij.y * fpair;
|
|
|
|
fztmp -=
|
|
|
|
dzfp = delij.z * fpair;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
vec3_scale(F_F(-1.0), delij, delij);
|
|
|
|
|
|
|
|
#pragma unroll 1
|
|
|
|
|
|
|
|
for(int kk = 0; kk < jnum_red; kk++) {
|
|
|
|
if(jj == kk) continue;
|
|
|
|
|
|
|
|
int k = jlist_red[kk * _nall];
|
|
|
|
k &= NEIGHMASK;
|
|
|
|
|
|
|
|
if(vflagm)
|
|
|
|
myxtype_k = fetchXType(k);
|
|
|
|
|
|
|
|
delik = _glob_r_ij[i + kk * _nall];
|
|
|
|
|
|
|
|
int ktype = _glob_neightype_red[i + kk * _nall];
|
|
|
|
int iparam_ijk = elem2param[(itype * nelements + jtype) * nelements + ktype];
|
|
|
|
vec3_scale(F_F(-1.0), delik, delik);
|
|
|
|
|
|
|
|
if(delik.w <= params[iparam_ijk].cutsq) {
|
|
|
|
if(vflagm) {
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT3 fi, fj, fk;
|
2012-10-08 23:29:55 +08:00
|
|
|
attractive(iparam_ijk, prefactor_ij,
|
|
|
|
delij, delik, fi, fj, fk);
|
|
|
|
fxtmp += fi.x;
|
|
|
|
fytmp += fi.y;
|
|
|
|
fztmp += fi.z;
|
|
|
|
|
|
|
|
sharedV[0 * blockDim.x] += ENERGY_F(2.0) * myxtype_i.x * fi.x;
|
|
|
|
sharedV[1 * blockDim.x] += ENERGY_F(2.0) * myxtype_i.y * fi.y;
|
|
|
|
sharedV[2 * blockDim.x] += ENERGY_F(2.0) * myxtype_i.z * fi.z;
|
|
|
|
sharedV[3 * blockDim.x] += ENERGY_F(2.0) * myxtype_i.x * fi.y;
|
|
|
|
sharedV[4 * blockDim.x] += ENERGY_F(2.0) * myxtype_i.x * fi.z;
|
|
|
|
sharedV[5 * blockDim.x] += ENERGY_F(2.0) * myxtype_i.y * fi.z;
|
|
|
|
|
|
|
|
sharedV[0 * blockDim.x] += ENERGY_F(2.0) * myxtype_j.x * fj.x;
|
|
|
|
sharedV[1 * blockDim.x] += ENERGY_F(2.0) * myxtype_j.y * fj.y;
|
|
|
|
sharedV[2 * blockDim.x] += ENERGY_F(2.0) * myxtype_j.z * fj.z;
|
|
|
|
sharedV[3 * blockDim.x] += ENERGY_F(2.0) * myxtype_j.x * fj.y;
|
|
|
|
sharedV[4 * blockDim.x] += ENERGY_F(2.0) * myxtype_j.x * fj.z;
|
|
|
|
sharedV[5 * blockDim.x] += ENERGY_F(2.0) * myxtype_j.y * fj.z;
|
|
|
|
|
|
|
|
sharedV[0 * blockDim.x] += ENERGY_F(2.0) * myxtype_k.x * fk.x;
|
|
|
|
sharedV[1 * blockDim.x] += ENERGY_F(2.0) * myxtype_k.y * fk.y;
|
|
|
|
sharedV[2 * blockDim.x] += ENERGY_F(2.0) * myxtype_k.z * fk.z;
|
|
|
|
sharedV[3 * blockDim.x] += ENERGY_F(2.0) * myxtype_k.x * fk.y;
|
|
|
|
sharedV[4 * blockDim.x] += ENERGY_F(2.0) * myxtype_k.x * fk.z;
|
|
|
|
sharedV[5 * blockDim.x] += ENERGY_F(2.0) * myxtype_k.y * fk.z;
|
|
|
|
} else {
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT3 fi; //local variable
|
2012-10-08 23:29:55 +08:00
|
|
|
attractive_fi(iparam_ijk, prefactor_ij,
|
|
|
|
delij, delik, fi);
|
|
|
|
fxtmp += fi.x;
|
|
|
|
fytmp += fi.y;
|
|
|
|
fztmp += fi.z;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int j_jnum_red = _glob_numneigh_red[j];
|
|
|
|
int* j_jlist_red = &_glob_neighbors_red[j];
|
|
|
|
|
|
|
|
int j_ii = 0;
|
|
|
|
|
|
|
|
//#pragma unroll 1
|
|
|
|
for(int j_kk = 0; j_kk < j_jnum_red; j_kk++) {
|
|
|
|
if(j_jlist_red[j_kk * _nall] == i) j_ii = j_kk;
|
|
|
|
}
|
|
|
|
|
|
|
|
#pragma unroll 1
|
|
|
|
|
|
|
|
for(int kk = 0; kk < j_jnum_red; kk++) {
|
|
|
|
if(j_ii == kk) continue;
|
|
|
|
|
|
|
|
int k = j_jlist_red[kk * _nall];
|
|
|
|
k &= NEIGHMASK;
|
|
|
|
deljk = _glob_r_ij[j + kk * _nall];
|
|
|
|
vec3_scale(F_F(-1.0), deljk, deljk);
|
|
|
|
int ktype = _glob_neightype_red[j + kk * _nall];
|
|
|
|
|
|
|
|
int iparam_jik = elem2param[(jtype * nelements + itype) * nelements + ktype];
|
|
|
|
int iparam_jki = elem2param[(jtype * nelements + ktype) * nelements + itype];
|
|
|
|
|
|
|
|
|
|
|
|
vec3_scale(F_F(-1.0), delij, delij);
|
|
|
|
|
|
|
|
if(deljk.w <= params[iparam_jik].cutsq) {
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT3 ftmp; //local variable
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
attractive_fj(iparam_jik, prefactor_ji,
|
|
|
|
delij, deljk, ftmp);
|
|
|
|
fxtmp += ftmp.x;
|
|
|
|
fytmp += ftmp.y;
|
|
|
|
fztmp += ftmp.z;
|
|
|
|
int iparam_jk = elem2param[(jtype * nelements + ktype) * nelements + ktype];
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT prefactor_jk;
|
2012-10-08 23:29:55 +08:00
|
|
|
force_zeta_prefactor(iparam_jk, deljk.w, _glob_zeta_ij[j + kk * _nall], prefactor_jk);
|
|
|
|
|
|
|
|
attractive_fk(iparam_jki, prefactor_jk,
|
|
|
|
deljk, delij, ftmp);
|
|
|
|
fxtmp += ftmp.x;
|
|
|
|
fytmp += ftmp.y;
|
|
|
|
fztmp += ftmp.z;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
vec3_scale(F_F(-1.0), delij, delij);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
__syncthreads();
|
|
|
|
|
|
|
|
if(ii < _inum) {
|
2014-10-07 06:59:05 +08:00
|
|
|
F_CFLOAT* my_f;
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
if(_collect_forces_later) {
|
2014-10-07 06:59:05 +08:00
|
|
|
ENERGY_CFLOAT* buffer = (ENERGY_CFLOAT*) _buffer;
|
2012-10-08 23:29:55 +08:00
|
|
|
|
|
|
|
if(eflag) {
|
|
|
|
buffer = &buffer[1 * gridDim.x * gridDim.y];
|
|
|
|
}
|
|
|
|
|
|
|
|
if(vflagm) {
|
|
|
|
buffer = &buffer[6 * gridDim.x * gridDim.y];
|
|
|
|
}
|
|
|
|
|
2014-10-07 06:59:05 +08:00
|
|
|
my_f = (F_CFLOAT*) buffer;
|
2012-10-08 23:29:55 +08:00
|
|
|
my_f += i;
|
|
|
|
*my_f = fxtmp;
|
|
|
|
my_f += _nmax;
|
|
|
|
*my_f = fytmp;
|
|
|
|
my_f += _nmax;
|
|
|
|
*my_f = fztmp;
|
|
|
|
} else {
|
|
|
|
my_f = _f + i;
|
|
|
|
*my_f += fxtmp;
|
|
|
|
my_f += _nmax;
|
|
|
|
*my_f += fytmp;
|
|
|
|
my_f += _nmax;
|
|
|
|
*my_f += fztmp;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
__syncthreads();
|
|
|
|
|
|
|
|
if(eflag) {
|
|
|
|
sharedE[0] = evdwl;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(eflag_atom && i < _nlocal) {
|
|
|
|
_eatom[i] = ENERGY_F(0.5) * evdwl;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(vflag_atom && i < _nlocal) {
|
|
|
|
_vatom[i] = ENERGY_F(0.5) * sharedV[0 * blockDim.x];
|
|
|
|
_vatom[i + _nmax] = ENERGY_F(0.5) * sharedV[1 * blockDim.x];
|
|
|
|
_vatom[i + 2 * _nmax] = ENERGY_F(0.5) * sharedV[2 * blockDim.x];
|
|
|
|
_vatom[i + 3 * _nmax] = ENERGY_F(0.5) * sharedV[3 * blockDim.x];
|
|
|
|
_vatom[i + 4 * _nmax] = ENERGY_F(0.5) * sharedV[4 * blockDim.x];
|
|
|
|
_vatom[i + 5 * _nmax] = ENERGY_F(0.5) * sharedV[5 * blockDim.x];
|
|
|
|
}
|
|
|
|
|
|
|
|
if(vflagm && eflag) PairVirialCompute_A_Kernel_Template<1, 1>();
|
|
|
|
else if(eflag) PairVirialCompute_A_Kernel_Template<1, 0>();
|
|
|
|
else if(vflagm) PairVirialCompute_A_Kernel_Template<0, 1>();
|
|
|
|
|
|
|
|
#undef fxtmp
|
|
|
|
#undef fytmp
|
|
|
|
#undef fztmp
|
|
|
|
//#undef jnum_red
|
|
|
|
}
|