forked from lijiext/lammps
476 lines
14 KiB
C++
476 lines
14 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.
|
|
------------------------------------------------------------------------- */
|
|
|
|
#include "fix_ave_atom.h"
|
|
#include <cstdlib>
|
|
#include <cstring>
|
|
#include "atom.h"
|
|
#include "update.h"
|
|
#include "modify.h"
|
|
#include "compute.h"
|
|
#include "input.h"
|
|
#include "variable.h"
|
|
#include "memory.h"
|
|
#include "error.h"
|
|
#include "force.h"
|
|
|
|
using namespace LAMMPS_NS;
|
|
using namespace FixConst;
|
|
|
|
enum{X,V,F,COMPUTE,FIX,VARIABLE};
|
|
|
|
#define INVOKED_PERATOM 8
|
|
|
|
/* ---------------------------------------------------------------------- */
|
|
|
|
FixAveAtom::FixAveAtom(LAMMPS *lmp, int narg, char **arg) :
|
|
Fix(lmp, narg, arg),
|
|
nvalues(0), which(NULL), argindex(NULL), value2index(NULL),
|
|
ids(NULL), array(NULL)
|
|
{
|
|
if (narg < 7) error->all(FLERR,"Illegal fix ave/atom command");
|
|
|
|
nevery = force->inumeric(FLERR,arg[3]);
|
|
nrepeat = force->inumeric(FLERR,arg[4]);
|
|
peratom_freq = force->inumeric(FLERR,arg[5]);
|
|
|
|
nvalues = narg - 6;
|
|
|
|
// expand args if any have wildcard character "*"
|
|
// this can reset nvalues
|
|
|
|
int expand = 0;
|
|
char **earg;
|
|
nvalues = input->expand_args(nvalues,&arg[6],1,earg);
|
|
|
|
if (earg != &arg[6]) expand = 1;
|
|
arg = earg;
|
|
|
|
// parse values
|
|
|
|
which = new int[nvalues];
|
|
argindex = new int[nvalues];
|
|
ids = new char*[nvalues];
|
|
value2index = new int[nvalues];
|
|
|
|
for (int i = 0; i < nvalues; i++) {
|
|
ids[i] = NULL;
|
|
|
|
if (strcmp(arg[i],"x") == 0) {
|
|
which[i] = X;
|
|
argindex[i] = 0;
|
|
} else if (strcmp(arg[i],"y") == 0) {
|
|
which[i] = X;
|
|
argindex[i] = 1;
|
|
} else if (strcmp(arg[i],"z") == 0) {
|
|
which[i] = X;
|
|
argindex[i] = 2;
|
|
|
|
} else if (strcmp(arg[i],"vx") == 0) {
|
|
which[i] = V;
|
|
argindex[i] = 0;
|
|
} else if (strcmp(arg[i],"vy") == 0) {
|
|
which[i] = V;
|
|
argindex[i] = 1;
|
|
} else if (strcmp(arg[i],"vz") == 0) {
|
|
which[i] = V;
|
|
argindex[i] = 2;
|
|
|
|
} else if (strcmp(arg[i],"fx") == 0) {
|
|
which[i] = F;
|
|
argindex[i] = 0;
|
|
} else if (strcmp(arg[i],"fy") == 0) {
|
|
which[i] = F;
|
|
argindex[i] = 1;
|
|
} else if (strcmp(arg[i],"fz") == 0) {
|
|
which[i] = F;
|
|
argindex[i] = 2;
|
|
|
|
} else if (strncmp(arg[i],"c_",2) == 0 ||
|
|
strncmp(arg[i],"f_",2) == 0 ||
|
|
strncmp(arg[i],"v_",2) == 0) {
|
|
if (arg[i][0] == 'c') which[i] = COMPUTE;
|
|
else if (arg[i][0] == 'f') which[i] = FIX;
|
|
else if (arg[i][0] == 'v') which[i] = VARIABLE;
|
|
|
|
int n = strlen(arg[i]);
|
|
char *suffix = new char[n];
|
|
strcpy(suffix,&arg[i][2]);
|
|
|
|
char *ptr = strchr(suffix,'[');
|
|
if (ptr) {
|
|
if (suffix[strlen(suffix)-1] != ']')
|
|
error->all(FLERR,"Illegal fix ave/atom command");
|
|
argindex[i] = atoi(ptr+1);
|
|
*ptr = '\0';
|
|
} else argindex[i] = 0;
|
|
|
|
n = strlen(suffix) + 1;
|
|
ids[i] = new char[n];
|
|
strcpy(ids[i],suffix);
|
|
delete [] suffix;
|
|
|
|
} else error->all(FLERR,"Illegal fix ave/atom command");
|
|
}
|
|
|
|
// if wildcard expansion occurred, free earg memory from exapnd_args()
|
|
|
|
if (expand) {
|
|
for (int i = 0; i < nvalues; i++) delete [] earg[i];
|
|
memory->sfree(earg);
|
|
}
|
|
|
|
// setup and error check
|
|
// for fix inputs, check that fix frequency is acceptable
|
|
|
|
if (nevery <= 0 || nrepeat <= 0 || peratom_freq <= 0)
|
|
error->all(FLERR,"Illegal fix ave/atom command");
|
|
if (peratom_freq % nevery || nrepeat*nevery > peratom_freq)
|
|
error->all(FLERR,"Illegal fix ave/atom command");
|
|
|
|
for (int i = 0; i < nvalues; i++) {
|
|
if (which[i] == COMPUTE) {
|
|
int icompute = modify->find_compute(ids[i]);
|
|
if (icompute < 0)
|
|
error->all(FLERR,"Compute ID for fix ave/atom does not exist");
|
|
if (modify->compute[icompute]->peratom_flag == 0)
|
|
error->all(FLERR,
|
|
"Fix ave/atom compute does not calculate per-atom values");
|
|
if (argindex[i] == 0 &&
|
|
modify->compute[icompute]->size_peratom_cols != 0)
|
|
error->all(FLERR,"Fix ave/atom compute does not "
|
|
"calculate a per-atom vector");
|
|
if (argindex[i] && modify->compute[icompute]->size_peratom_cols == 0)
|
|
error->all(FLERR,"Fix ave/atom compute does not "
|
|
"calculate a per-atom array");
|
|
if (argindex[i] &&
|
|
argindex[i] > modify->compute[icompute]->size_peratom_cols)
|
|
error->all(FLERR,"Fix ave/atom compute array is accessed out-of-range");
|
|
|
|
} else if (which[i] == FIX) {
|
|
int ifix = modify->find_fix(ids[i]);
|
|
if (ifix < 0)
|
|
error->all(FLERR,"Fix ID for fix ave/atom does not exist");
|
|
if (modify->fix[ifix]->peratom_flag == 0)
|
|
error->all(FLERR,"Fix ave/atom fix does not calculate per-atom values");
|
|
if (argindex[i] == 0 && modify->fix[ifix]->size_peratom_cols != 0)
|
|
error->all(FLERR,
|
|
"Fix ave/atom fix does not calculate a per-atom vector");
|
|
if (argindex[i] && modify->fix[ifix]->size_peratom_cols == 0)
|
|
error->all(FLERR,
|
|
"Fix ave/atom fix does not calculate a per-atom array");
|
|
if (argindex[i] && argindex[i] > modify->fix[ifix]->size_peratom_cols)
|
|
error->all(FLERR,"Fix ave/atom fix array is accessed out-of-range");
|
|
if (nevery % modify->fix[ifix]->peratom_freq)
|
|
error->all(FLERR,
|
|
"Fix for fix ave/atom not computed at compatible time");
|
|
|
|
} else if (which[i] == VARIABLE) {
|
|
int ivariable = input->variable->find(ids[i]);
|
|
if (ivariable < 0)
|
|
error->all(FLERR,"Variable name for fix ave/atom does not exist");
|
|
if (input->variable->atomstyle(ivariable) == 0)
|
|
error->all(FLERR,"Fix ave/atom variable is not atom-style variable");
|
|
}
|
|
}
|
|
|
|
// this fix produces either a per-atom vector or array
|
|
|
|
peratom_flag = 1;
|
|
if (nvalues == 1) size_peratom_cols = 0;
|
|
else size_peratom_cols = nvalues;
|
|
|
|
// perform initial allocation of atom-based array
|
|
// register with Atom class
|
|
|
|
grow_arrays(atom->nmax);
|
|
atom->add_callback(0);
|
|
|
|
// zero the array since dump may access it on timestep 0
|
|
// zero the array since a variable may access it before first run
|
|
|
|
int nlocal = atom->nlocal;
|
|
for (int i = 0; i < nlocal; i++)
|
|
for (int m = 0; m < nvalues; m++)
|
|
array[i][m] = 0.0;
|
|
|
|
// nvalid = next step on which end_of_step does something
|
|
// add nvalid to all computes that store invocation times
|
|
// since don't know a priori which are invoked by this fix
|
|
// once in end_of_step() can set timestep for ones actually invoked
|
|
|
|
irepeat = 0;
|
|
nvalid_last = -1;
|
|
nvalid = nextvalid();
|
|
modify->addstep_compute_all(nvalid);
|
|
}
|
|
|
|
/* ---------------------------------------------------------------------- */
|
|
|
|
FixAveAtom::~FixAveAtom()
|
|
{
|
|
// unregister callback to this fix from Atom class
|
|
|
|
atom->delete_callback(id,0);
|
|
|
|
delete [] which;
|
|
delete [] argindex;
|
|
for (int m = 0; m < nvalues; m++) delete [] ids[m];
|
|
delete [] ids;
|
|
delete [] value2index;
|
|
|
|
memory->destroy(array);
|
|
}
|
|
|
|
/* ---------------------------------------------------------------------- */
|
|
|
|
int FixAveAtom::setmask()
|
|
{
|
|
int mask = 0;
|
|
mask |= END_OF_STEP;
|
|
return mask;
|
|
}
|
|
|
|
/* ---------------------------------------------------------------------- */
|
|
|
|
void FixAveAtom::init()
|
|
{
|
|
// set indices and check validity of all computes,fixes,variables
|
|
|
|
for (int m = 0; m < nvalues; m++) {
|
|
if (which[m] == COMPUTE) {
|
|
int icompute = modify->find_compute(ids[m]);
|
|
if (icompute < 0)
|
|
error->all(FLERR,"Compute ID for fix ave/atom does not exist");
|
|
value2index[m] = icompute;
|
|
|
|
} else if (which[m] == FIX) {
|
|
int ifix = modify->find_fix(ids[m]);
|
|
if (ifix < 0)
|
|
error->all(FLERR,"Fix ID for fix ave/atom does not exist");
|
|
value2index[m] = ifix;
|
|
|
|
} else if (which[m] == VARIABLE) {
|
|
int ivariable = input->variable->find(ids[m]);
|
|
if (ivariable < 0)
|
|
error->all(FLERR,"Variable name for fix ave/atom does not exist");
|
|
value2index[m] = ivariable;
|
|
|
|
} else value2index[m] = -1;
|
|
}
|
|
|
|
// need to reset nvalid if nvalid < ntimestep b/c minimize was performed
|
|
|
|
if (nvalid < update->ntimestep) {
|
|
irepeat = 0;
|
|
nvalid = nextvalid();
|
|
modify->addstep_compute_all(nvalid);
|
|
}
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------
|
|
only does something if nvalid = current timestep
|
|
------------------------------------------------------------------------- */
|
|
|
|
void FixAveAtom::setup(int /*vflag*/)
|
|
{
|
|
end_of_step();
|
|
}
|
|
|
|
/* ---------------------------------------------------------------------- */
|
|
|
|
void FixAveAtom::end_of_step()
|
|
{
|
|
int i,j,m,n;
|
|
|
|
// skip if not step which requires doing something
|
|
// error check if timestep was reset in an invalid manner
|
|
|
|
bigint ntimestep = update->ntimestep;
|
|
if (ntimestep < nvalid_last || ntimestep > nvalid)
|
|
error->all(FLERR,"Invalid timestep reset for fix ave/atom");
|
|
if (ntimestep != nvalid) return;
|
|
nvalid_last = nvalid;
|
|
|
|
// zero if first step
|
|
|
|
int nlocal = atom->nlocal;
|
|
|
|
if (irepeat == 0)
|
|
for (i = 0; i < nlocal; i++)
|
|
for (m = 0; m < nvalues; m++)
|
|
array[i][m] = 0.0;
|
|
|
|
// accumulate results of attributes,computes,fixes,variables to local copy
|
|
// compute/fix/variable may invoke computes so wrap with clear/add
|
|
|
|
modify->clearstep_compute();
|
|
|
|
int *mask = atom->mask;
|
|
|
|
for (m = 0; m < nvalues; m++) {
|
|
n = value2index[m];
|
|
j = argindex[m];
|
|
|
|
if (which[m] == X) {
|
|
double **x = atom->x;
|
|
for (i = 0; i < nlocal; i++)
|
|
if (mask[i] & groupbit) array[i][m] += x[i][j];
|
|
|
|
} else if (which[m] == V) {
|
|
double **v = atom->v;
|
|
for (i = 0; i < nlocal; i++)
|
|
if (mask[i] & groupbit) array[i][m] += v[i][j];
|
|
|
|
} else if (which[m] == F) {
|
|
double **f = atom->f;
|
|
for (i = 0; i < nlocal; i++)
|
|
if (mask[i] & groupbit) array[i][m] += f[i][j];
|
|
|
|
// invoke compute if not previously invoked
|
|
|
|
} else if (which[m] == COMPUTE) {
|
|
Compute *compute = modify->compute[n];
|
|
if (!(compute->invoked_flag & INVOKED_PERATOM)) {
|
|
compute->compute_peratom();
|
|
compute->invoked_flag |= INVOKED_PERATOM;
|
|
}
|
|
|
|
if (j == 0) {
|
|
double *compute_vector = compute->vector_atom;
|
|
for (i = 0; i < nlocal; i++)
|
|
if (mask[i] & groupbit) array[i][m] += compute_vector[i];
|
|
} else {
|
|
int jm1 = j - 1;
|
|
double **compute_array = compute->array_atom;
|
|
for (i = 0; i < nlocal; i++)
|
|
if (mask[i] & groupbit) array[i][m] += compute_array[i][jm1];
|
|
}
|
|
|
|
// access fix fields, guaranteed to be ready
|
|
|
|
} else if (which[m] == FIX) {
|
|
if (j == 0) {
|
|
double *fix_vector = modify->fix[n]->vector_atom;
|
|
for (i = 0; i < nlocal; i++)
|
|
if (mask[i] & groupbit) array[i][m] += fix_vector[i];
|
|
} else {
|
|
int jm1 = j - 1;
|
|
double **fix_array = modify->fix[n]->array_atom;
|
|
for (i = 0; i < nlocal; i++)
|
|
if (mask[i] & groupbit) array[i][m] += fix_array[i][jm1];
|
|
}
|
|
|
|
// evaluate atom-style variable
|
|
// final argument = 1 sums result to array
|
|
|
|
} else if (which[m] == VARIABLE) {
|
|
if (array) input->variable->compute_atom(n,igroup,&array[0][m],nvalues,1);
|
|
else input->variable->compute_atom(n,igroup,NULL,nvalues,1);
|
|
}
|
|
}
|
|
|
|
// done if irepeat < nrepeat
|
|
// else reset irepeat and nvalid
|
|
|
|
irepeat++;
|
|
if (irepeat < nrepeat) {
|
|
nvalid += nevery;
|
|
modify->addstep_compute(nvalid);
|
|
return;
|
|
}
|
|
|
|
irepeat = 0;
|
|
nvalid = ntimestep+peratom_freq - (nrepeat-1)*nevery;
|
|
modify->addstep_compute(nvalid);
|
|
|
|
if (array == NULL) return;
|
|
|
|
// average the final result for the Nfreq timestep
|
|
|
|
double repeat = nrepeat;
|
|
for (i = 0; i < nlocal; i++)
|
|
for (m = 0; m < nvalues; m++)
|
|
array[i][m] /= repeat;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------
|
|
memory usage of local atom-based array
|
|
------------------------------------------------------------------------- */
|
|
|
|
double FixAveAtom::memory_usage()
|
|
{
|
|
double bytes;
|
|
bytes = atom->nmax*nvalues * sizeof(double);
|
|
return bytes;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------
|
|
allocate atom-based array
|
|
------------------------------------------------------------------------- */
|
|
|
|
void FixAveAtom::grow_arrays(int nmax)
|
|
{
|
|
memory->grow(array,nmax,nvalues,"fix_ave/atom:array");
|
|
array_atom = array;
|
|
if (array) vector_atom = array[0];
|
|
else vector_atom = NULL;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------
|
|
copy values within local atom-based array
|
|
------------------------------------------------------------------------- */
|
|
|
|
void FixAveAtom::copy_arrays(int i, int j, int /*delflag*/)
|
|
{
|
|
for (int m = 0; m < nvalues; m++)
|
|
array[j][m] = array[i][m];
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------
|
|
pack values in local atom-based array for exchange with another proc
|
|
------------------------------------------------------------------------- */
|
|
|
|
int FixAveAtom::pack_exchange(int i, double *buf)
|
|
{
|
|
for (int m = 0; m < nvalues; m++) buf[m] = array[i][m];
|
|
return nvalues;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------
|
|
unpack values in local atom-based array from exchange with another proc
|
|
------------------------------------------------------------------------- */
|
|
|
|
int FixAveAtom::unpack_exchange(int nlocal, double *buf)
|
|
{
|
|
for (int m = 0; m < nvalues; m++) array[nlocal][m] = buf[m];
|
|
return nvalues;
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------
|
|
calculate nvalid = next step on which end_of_step does something
|
|
can be this timestep if multiple of nfreq and nrepeat = 1
|
|
else backup from next multiple of nfreq
|
|
------------------------------------------------------------------------- */
|
|
|
|
bigint FixAveAtom::nextvalid()
|
|
{
|
|
bigint nvalid = (update->ntimestep/peratom_freq)*peratom_freq + peratom_freq;
|
|
if (nvalid-peratom_freq == update->ntimestep && nrepeat == 1)
|
|
nvalid = update->ntimestep;
|
|
else
|
|
nvalid -= (nrepeat-1)*nevery;
|
|
if (nvalid < update->ntimestep) nvalid += peratom_freq;
|
|
return nvalid;
|
|
}
|