149 lines
4.8 KiB
C++
149 lines
4.8 KiB
C++
/* ----------------------------------------------------------------------
|
|
LAMMPS - Large-scale Atomic/Molecular Massively Parallel Simulator
|
|
http://lammps.sandia.gov, Sandia National Laboratories
|
|
Steve Plimpton, sjplimp@sandia.gov
|
|
|
|
Copyright (2003) Sandia Corporation. Under the terms of Contract
|
|
DE-AC04-94AL85000 with Sandia Corporation, the U.S. Government retains
|
|
certain rights in this software. This software is distributed under
|
|
the GNU General Public License.
|
|
|
|
See the README file in the top-level LAMMPS directory.
|
|
------------------------------------------------------------------------- */
|
|
|
|
/* ----------------------------------------------------------------------
|
|
Contributing author: Axel Kohlmeyer (Temple U)
|
|
------------------------------------------------------------------------- */
|
|
|
|
#include "math.h"
|
|
#include "fix_wall_gran_omp.h"
|
|
#include "atom.h"
|
|
#include "update.h"
|
|
|
|
using namespace LAMMPS_NS;
|
|
using namespace FixConst;
|
|
|
|
enum{XPLANE=0,YPLANE=1,ZPLANE=2,ZCYLINDER}; // XYZ PLANE need to be 0,1,2
|
|
enum{HOOKE,HOOKE_HISTORY,HERTZ_HISTORY};
|
|
|
|
#define BIG 1.0e20
|
|
|
|
/* ---------------------------------------------------------------------- */
|
|
|
|
FixWallGranOMP::FixWallGranOMP(LAMMPS *lmp, int narg, char **arg) :
|
|
FixWallGran(lmp, narg, arg) { }
|
|
|
|
/* ---------------------------------------------------------------------- */
|
|
|
|
void FixWallGranOMP::post_force(int vflag)
|
|
{
|
|
double vwall[3];
|
|
|
|
// set position of wall to initial settings and velocity to 0.0
|
|
// if wiggle or shear, set wall position and velocity accordingly
|
|
|
|
double wlo = lo;
|
|
double whi = hi;
|
|
vwall[0] = vwall[1] = vwall[2] = 0.0;
|
|
if (wiggle) {
|
|
double arg = omega * (update->ntimestep - time_origin) * dt;
|
|
if (wallstyle == axis) {
|
|
wlo = lo + amplitude - amplitude*cos(arg);
|
|
whi = hi + amplitude - amplitude*cos(arg);
|
|
}
|
|
vwall[axis] = amplitude*omega*sin(arg);
|
|
} else if (wshear) vwall[axis] = vshear;
|
|
|
|
// loop over all my atoms
|
|
// rsq = distance from wall
|
|
// dx,dy,dz = signed distance from wall
|
|
// for rotating cylinder, reset vwall based on particle position
|
|
// skip atom if not close enough to wall
|
|
// if wall was set to NULL, it's skipped since lo/hi are infinity
|
|
// compute force and torque on atom if close enough to wall
|
|
// via wall potential matched to pair potential
|
|
// set shear if pair potential stores history
|
|
|
|
double * const * const x = atom->x;
|
|
double * const * const v = atom->v;
|
|
double * const * const f = atom->f;
|
|
double * const * const omega = atom->omega;
|
|
double * const * const torque = atom->torque;
|
|
double * const radius = atom->radius;
|
|
double * const rmass = atom->rmass;
|
|
const int * const mask = atom->mask;
|
|
const int nlocal = atom->nlocal;
|
|
|
|
shearupdate = (update->setupflag) ? 0 : 1;
|
|
|
|
int i;
|
|
#if defined(_OPENMP)
|
|
#pragma omp parallel for private(i) default(none) firstprivate(vwall,wlo,whi)
|
|
#endif
|
|
for (i = 0; i < nlocal; i++) {
|
|
|
|
if (mask[i] & groupbit) {
|
|
double dx,dy,dz,del1,del2,delxy,delr,rsq;
|
|
|
|
dx = dy = dz = 0.0;
|
|
|
|
if (wallstyle == XPLANE) {
|
|
del1 = x[i][0] - wlo;
|
|
del2 = whi - x[i][0];
|
|
if (del1 < del2) dx = del1;
|
|
else dx = -del2;
|
|
} else if (wallstyle == YPLANE) {
|
|
del1 = x[i][1] - wlo;
|
|
del2 = whi - x[i][1];
|
|
if (del1 < del2) dy = del1;
|
|
else dy = -del2;
|
|
} else if (wallstyle == ZPLANE) {
|
|
del1 = x[i][2] - wlo;
|
|
del2 = whi - x[i][2];
|
|
if (del1 < del2) dz = del1;
|
|
else dz = -del2;
|
|
} else if (wallstyle == ZCYLINDER) {
|
|
delxy = sqrt(x[i][0]*x[i][0] + x[i][1]*x[i][1]);
|
|
delr = cylradius - delxy;
|
|
if (delr > radius[i]) dz = cylradius;
|
|
else {
|
|
dx = -delr/delxy * x[i][0];
|
|
dy = -delr/delxy * x[i][1];
|
|
if (wshear && axis != 2) {
|
|
vwall[0] = vshear * x[i][1]/delxy;
|
|
vwall[1] = -vshear * x[i][0]/delxy;
|
|
vwall[2] = 0.0;
|
|
}
|
|
}
|
|
}
|
|
|
|
rsq = dx*dx + dy*dy + dz*dz;
|
|
|
|
if (rsq > radius[i]*radius[i]) {
|
|
if (pairstyle != HOOKE) {
|
|
shear[i][0] = 0.0;
|
|
shear[i][1] = 0.0;
|
|
shear[i][2] = 0.0;
|
|
}
|
|
} else {
|
|
if (pairstyle == HOOKE)
|
|
hooke(rsq,dx,dy,dz,vwall,v[i],f[i],omega[i],torque[i],
|
|
radius[i],rmass[i]);
|
|
else if (pairstyle == HOOKE_HISTORY)
|
|
hooke_history(rsq,dx,dy,dz,vwall,v[i],f[i],omega[i],torque[i],
|
|
radius[i],rmass[i],shear[i]);
|
|
else if (pairstyle == HERTZ_HISTORY)
|
|
hertz_history(rsq,dx,dy,dz,vwall,v[i],f[i],omega[i],torque[i],
|
|
radius[i],rmass[i],shear[i]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* ---------------------------------------------------------------------- */
|
|
|
|
void FixWallGranOMP::post_force_respa(int vflag, int ilevel, int iloop)
|
|
{
|
|
if (ilevel == nlevels_respa-1) post_force(vflag);
|
|
}
|