/* ---------------------------------------------------------------------- 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: Pieter in 't Veld (SNL) ------------------------------------------------------------------------- */ #include "stdlib.h" #include "string.h" #include "unistd.h" #include "fix_ave_spatial.h" #include "atom.h" #include "update.h" #include "force.h" #include "domain.h" #include "region.h" #include "lattice.h" #include "modify.h" #include "compute.h" #include "input.h" #include "variable.h" #include "memory.h" #include "error.h" using namespace LAMMPS_NS; using namespace FixConst; enum{LOWER,CENTER,UPPER,COORD}; enum{V,F,DENSITY_NUMBER,DENSITY_MASS,COMPUTE,FIX,VARIABLE}; enum{SAMPLE,ALL}; enum{BOX,LATTICE,REDUCED}; enum{ONE,RUNNING,WINDOW}; #define INVOKED_PERATOM 8 #define BIG 1000000000 /* ---------------------------------------------------------------------- */ FixAveSpatial::FixAveSpatial(LAMMPS *lmp, int narg, char **arg) : Fix(lmp, narg, arg) { if (narg < 6) error->all(FLERR,"Illegal fix ave/spatial command"); MPI_Comm_rank(world,&me); nevery = force->inumeric(FLERR,arg[3]); nrepeat = force->inumeric(FLERR,arg[4]); nfreq = force->inumeric(FLERR,arg[5]); global_freq = nfreq; no_change_box = 1; ndim = 0; int iarg = 6; while (iarg < narg && ndim < 3) { if (iarg+3 > narg) break; if (strcmp(arg[iarg],"x") == 0) dim[ndim] = 0; else if (strcmp(arg[iarg],"y") == 0) dim[ndim] = 1; else if (strcmp(arg[iarg],"z") == 0) dim[ndim] = 2; else break; if (dim[ndim] == 2 && domain->dimension == 2) error->all(FLERR,"Cannot use fix ave/spatial z for 2 dimensional model"); if (strcmp(arg[iarg+1],"lower") == 0) originflag[ndim] = LOWER; else if (strcmp(arg[iarg+1],"center") == 0) originflag[ndim] = CENTER; else if (strcmp(arg[iarg+1],"upper") == 0) originflag[ndim] = UPPER; else originflag[ndim] = COORD; if (originflag[ndim] == COORD) origin[ndim] = force->numeric(FLERR,arg[iarg+1]); delta[ndim] = force->numeric(FLERR,arg[iarg+2]); ndim++; iarg += 3; } if (!ndim) error->all(FLERR,"Illegal fix ave/spatial command"); if (ndim == 2 && dim[0] == dim[1]) error->all(FLERR,"Same dimension twice in fix ave/spatial"); if (ndim == 3 && (dim[0] == dim[1] || dim[1] == dim[2] || dim[0] == dim[2])) error->all(FLERR,"Same dimension twice in fix ave/spatial"); // parse values until one isn't recognized which = new int[narg-9]; argindex = new int[narg-9]; ids = new char*[narg-9]; value2index = new int[narg-9]; nvalues = 0; while (iarg < narg) { ids[nvalues] = NULL; if (strcmp(arg[iarg],"vx") == 0) { which[nvalues] = V; argindex[nvalues++] = 0; } else if (strcmp(arg[iarg],"vy") == 0) { which[nvalues] = V; argindex[nvalues++] = 1; } else if (strcmp(arg[iarg],"vz") == 0) { which[nvalues] = V; argindex[nvalues++] = 2; } else if (strcmp(arg[iarg],"fx") == 0) { which[nvalues] = F; argindex[nvalues++] = 0; } else if (strcmp(arg[iarg],"fy") == 0) { which[nvalues] = F; argindex[nvalues++] = 1; } else if (strcmp(arg[iarg],"fz") == 0) { which[nvalues] = F; argindex[nvalues++] = 2; } else if (strcmp(arg[iarg],"density/number") == 0) { which[nvalues] = DENSITY_NUMBER; argindex[nvalues++] = 0; } else if (strcmp(arg[iarg],"density/mass") == 0) { which[nvalues] = DENSITY_MASS; argindex[nvalues++] = 0; } else if (strncmp(arg[iarg],"c_",2) == 0 || strncmp(arg[iarg],"f_",2) == 0 || strncmp(arg[iarg],"v_",2) == 0) { if (arg[iarg][0] == 'c') which[nvalues] = COMPUTE; else if (arg[iarg][0] == 'f') which[nvalues] = FIX; else if (arg[iarg][0] == 'v') which[nvalues] = VARIABLE; int n = strlen(arg[iarg]); char *suffix = new char[n]; strcpy(suffix,&arg[iarg][2]); char *ptr = strchr(suffix,'['); if (ptr) { if (suffix[strlen(suffix)-1] != ']') error->all(FLERR,"Illegal fix ave/spatial command"); argindex[nvalues] = atoi(ptr+1); *ptr = '\0'; } else argindex[nvalues] = 0; n = strlen(suffix) + 1; ids[nvalues] = new char[n]; strcpy(ids[nvalues],suffix); nvalues++; delete [] suffix; } else break; iarg++; } // optional args normflag = ALL; scaleflag = LATTICE; regionflag = 0; idregion = NULL; fp = NULL; ave = ONE; nwindow = 0; overwrite = 0; char *title1 = NULL; char *title2 = NULL; char *title3 = NULL; while (iarg < narg) { if (strcmp(arg[iarg],"norm") == 0) { if (iarg+2 > narg) error->all(FLERR,"Illegal fix ave/spatial command"); if (strcmp(arg[iarg+1],"all") == 0) normflag = ALL; else if (strcmp(arg[iarg+1],"sample") == 0) normflag = SAMPLE; else error->all(FLERR,"Illegal fix ave/spatial command"); iarg += 2; } else if (strcmp(arg[iarg],"units") == 0) { if (iarg+2 > narg) error->all(FLERR,"Illegal fix ave/spatial command"); if (strcmp(arg[iarg+1],"box") == 0) scaleflag = BOX; else if (strcmp(arg[iarg+1],"lattice") == 0) scaleflag = LATTICE; else if (strcmp(arg[iarg+1],"reduced") == 0) scaleflag = REDUCED; else error->all(FLERR,"Illegal fix ave/spatial command"); iarg += 2; } else if (strcmp(arg[iarg],"region") == 0) { if (iarg+2 > narg) error->all(FLERR,"Illegal fix ave/spatial command"); int iregion = domain->find_region(arg[iarg+1]); if (iregion == -1) error->all(FLERR,"Region ID for fix ave/spatial does not exist"); int n = strlen(arg[iarg+1]) + 1; idregion = new char[n]; strcpy(idregion,arg[iarg+1]); regionflag = 1; iarg += 2; } else if (strcmp(arg[iarg],"file") == 0) { if (iarg+2 > narg) error->all(FLERR,"Illegal fix ave/spatial command"); if (me == 0) { fp = fopen(arg[iarg+1],"w"); if (fp == NULL) { char str[128]; sprintf(str,"Cannot open fix ave/spatial file %s",arg[iarg+1]); error->one(FLERR,str); } } iarg += 2; } else if (strcmp(arg[iarg],"ave") == 0) { if (iarg+2 > narg) error->all(FLERR,"Illegal fix ave/spatial command"); if (strcmp(arg[iarg+1],"one") == 0) ave = ONE; else if (strcmp(arg[iarg+1],"running") == 0) ave = RUNNING; else if (strcmp(arg[iarg+1],"window") == 0) ave = WINDOW; else error->all(FLERR,"Illegal fix ave/spatial command"); if (ave == WINDOW) { if (iarg+3 > narg) error->all(FLERR,"Illegal fix ave/spatial command"); nwindow = force->inumeric(FLERR,arg[iarg+2]); if (nwindow <= 0) error->all(FLERR,"Illegal fix ave/spatial command"); } iarg += 2; if (ave == WINDOW) iarg++; } else if (strcmp(arg[iarg],"overwrite") == 0) { overwrite = 1; iarg += 1; } else if (strcmp(arg[iarg],"title1") == 0) { if (iarg+2 > narg) error->all(FLERR,"Illegal fix ave/spatial command"); delete [] title1; int n = strlen(arg[iarg+1]) + 1; title1 = new char[n]; strcpy(title1,arg[iarg+1]); iarg += 2; } else if (strcmp(arg[iarg],"title2") == 0) { if (iarg+2 > narg) error->all(FLERR,"Illegal fix ave/spatial command"); delete [] title2; int n = strlen(arg[iarg+1]) + 1; title2 = new char[n]; strcpy(title2,arg[iarg+1]); iarg += 2; } else if (strcmp(arg[iarg],"title3") == 0) { if (iarg+2 > narg) error->all(FLERR,"Illegal fix ave/spatial command"); delete [] title3; int n = strlen(arg[iarg+1]) + 1; title3 = new char[n]; strcpy(title3,arg[iarg+1]); iarg += 2; } else error->all(FLERR,"Illegal fix ave/spatial command"); } // setup and error check if (nevery <= 0 || nrepeat <= 0 || nfreq <= 0) error->all(FLERR,"Illegal fix ave/spatial command"); if (nfreq % nevery || (nrepeat-1)*nevery >= nfreq) error->all(FLERR,"Illegal fix ave/spatial command"); if (delta[0] <= 0.0) error->all(FLERR,"Illegal fix ave/spatial command"); if (ndim >= 2 && delta[1] <= 0.0) error->all(FLERR,"Illegal fix ave/spatial command"); if (ndim == 3 && delta[2] <= 0.0) error->all(FLERR,"Illegal fix ave/spatial command"); if (ave != RUNNING && overwrite) error->all(FLERR,"Illegal fix ave/spatial 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/spatial does not exist"); if (modify->compute[icompute]->peratom_flag == 0) error->all(FLERR,"Fix ave/spatial compute does not " "calculate per-atom values"); if (argindex[i] == 0 && modify->compute[icompute]->size_peratom_cols != 0) error->all(FLERR,"Fix ave/spatial compute does not " "calculate a per-atom vector"); if (argindex[i] && modify->compute[icompute]->size_peratom_cols == 0) error->all(FLERR,"Fix ave/spatial compute does not " "calculate a per-atom array"); if (argindex[i] && argindex[i] > modify->compute[icompute]->size_peratom_cols) error->all(FLERR, "Fix ave/spatial compute vector 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/spatial does not exist"); if (modify->fix[ifix]->peratom_flag == 0) error->all(FLERR, "Fix ave/spatial fix does not calculate per-atom values"); if (argindex[i] && modify->fix[ifix]->size_peratom_cols != 0) error->all(FLERR, "Fix ave/spatial fix does not calculate a per-atom vector"); if (argindex[i] && modify->fix[ifix]->size_peratom_cols == 0) error->all(FLERR, "Fix ave/spatial fix does not calculate a per-atom array"); if (argindex[i] && argindex[i] > modify->fix[ifix]->size_peratom_cols) error->all(FLERR,"Fix ave/spatial fix vector is accessed out-of-range"); } else if (which[i] == VARIABLE) { int ivariable = input->variable->find(ids[i]); if (ivariable < 0) error->all(FLERR,"Variable name for fix ave/spatial does not exist"); if (input->variable->atomstyle(ivariable) == 0) error->all(FLERR,"Fix ave/spatial variable is not atom-style variable"); } } // print file comment lines if (fp && me == 0) { if (title1) fprintf(fp,"%s\n",title1); else fprintf(fp,"# Spatial-averaged data for fix %s and group %s\n", id,arg[1]); if (title2) fprintf(fp,"%s\n",title2); else fprintf(fp,"# Timestep Number-of-bins\n"); if (title3) fprintf(fp,"%s\n",title3); else { if (ndim == 1) fprintf(fp,"# Bin Coord Ncount"); else if (ndim == 2) fprintf(fp,"# Bin Coord1 Coord2 Ncount"); else if (ndim == 3) fprintf(fp,"# Bin Coord1 Coord2 Coord3 Ncount"); for (int i = 0; i < nvalues; i++) fprintf(fp," %s",arg[6+3*ndim+i]); fprintf(fp,"\n"); } filepos = ftell(fp); } delete [] title1; delete [] title2; delete [] title3; // this fix produces a global array array_flag = 1; size_array_rows = BIG; size_array_cols = 1 + ndim + nvalues; extarray = 0; // setup scaling int triclinic = domain->triclinic; if (triclinic == 1 && scaleflag != REDUCED) error->all(FLERR, "Fix ave/spatial for triclinic boxes requires units reduced"); if (scaleflag == LATTICE) { xscale = domain->lattice->xlattice; yscale = domain->lattice->ylattice; zscale = domain->lattice->zlattice; } else xscale = yscale = zscale = 1.0; // apply scaling factors double scale; for (int idim = 0; idim < ndim; idim++) { if (dim[idim] == 0) scale = xscale; else if (dim[idim] == 1) scale = yscale; else if (dim[idim] == 2) scale = zscale; delta[idim] *= scale; if (originflag[idim] == COORD) origin[idim] *= scale; invdelta[idim] = 1.0/delta[idim]; } // initializations irepeat = 0; iwindow = window_limit = 0; norm = 0; maxvar = 0; varatom = NULL; maxatom = 0; bin = NULL; nbins = maxbin = 0; count_one = count_many = count_sum = count_total = NULL; coord = NULL; count_list = NULL; values_one = values_many = values_sum = values_total = NULL; values_list = NULL; // 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 nvalid = nextvalid(); modify->addstep_compute_all(nvalid); } /* ---------------------------------------------------------------------- */ FixAveSpatial::~FixAveSpatial() { delete [] which; delete [] argindex; for (int i = 0; i < nvalues; i++) delete [] ids[i]; delete [] ids; delete [] value2index; delete [] idregion; if (fp && me == 0) fclose(fp); memory->destroy(varatom); memory->destroy(bin); memory->destroy(count_one); memory->destroy(count_many); memory->destroy(count_sum); memory->destroy(count_total); memory->destroy(coord); memory->destroy(count_list); memory->destroy(values_one); memory->destroy(values_many); memory->destroy(values_sum); memory->destroy(values_total); memory->destroy(values_list); } /* ---------------------------------------------------------------------- */ int FixAveSpatial::setmask() { int mask = 0; mask |= END_OF_STEP; return mask; } /* ---------------------------------------------------------------------- */ void FixAveSpatial::init() { // set and check validity of region if (regionflag) { int iregion = domain->find_region(idregion); if (iregion == -1) error->all(FLERR,"Region ID for fix ave/spatial does not exist"); region = domain->regions[iregion]; } // # of bins cannot vary for ave = RUNNING or WINDOW if (ave == RUNNING || ave == WINDOW) { if (scaleflag != REDUCED && domain->box_change_size) error->all(FLERR, "Fix ave/spatial settings invalid with changing box size"); } // set indices and check validity of all computes,fixes,variables // check that fix frequency is acceptable 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/spatial 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/spatial does not exist"); value2index[m] = ifix; if (nevery % modify->fix[ifix]->peratom_freq) error->all(FLERR, "Fix for fix ave/spatial not computed at compatible time"); } else if (which[m] == VARIABLE) { int ivariable = input->variable->find(ids[m]); if (ivariable < 0) error->all(FLERR,"Variable name for fix ave/spatial 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); } } /* ---------------------------------------------------------------------- setup initial bins only does averaging if nvalid = current timestep ------------------------------------------------------------------------- */ void FixAveSpatial::setup(int vflag) { setup_bins(); end_of_step(); } /* ---------------------------------------------------------------------- */ void FixAveSpatial::end_of_step() { int i,j,m,n; // skip if not step which requires doing something bigint ntimestep = update->ntimestep; if (ntimestep != nvalid) return; // zero out arrays that accumulate over many samples // if box changes, first re-setup bins if (irepeat == 0) { if (domain->box_change) setup_bins(); for (m = 0; m < nbins; m++) { count_many[m] = count_sum[m] = 0.0; for (i = 0; i < nvalues; i++) values_many[m][i] = 0.0; } } // zero out arrays for one sample for (m = 0; m < nbins; m++) { count_one[m] = 0.0; for (i = 0; i < nvalues; i++) values_one[m][i] = 0.0; } // assign each atom to a bin double **x = atom->x; int *mask = atom->mask; int nlocal = atom->nlocal; if (nlocal > maxatom) { maxatom = atom->nmax; memory->destroy(bin); memory->create(bin,maxatom,"ave/spatial:bin"); } if (ndim == 1) atom2bin1d(); else if (ndim == 2) atom2bin2d(); else atom2bin3d(); // perform the computation for one sample // accumulate results of attributes,computes,fixes,variables to local copy // sum within each bin, only include atoms in fix group // compute/fix/variable may invoke computes so wrap with clear/add modify->clearstep_compute(); for (m = 0; m < nvalues; m++) { n = value2index[m]; j = argindex[m]; // V,F adds velocities,forces to values if (which[m] == V || which[m] == F) { double **attribute; if (which[m] == V) attribute = atom->v; else attribute = atom->f; if (!regionflag) { for (i = 0; i < nlocal; i++) if (mask[i] & groupbit) values_one[bin[i]][m] += attribute[i][j]; } else { for (i = 0; i < nlocal; i++) if (mask[i] & groupbit && region->match(x[i][0],x[i][1],x[i][2])) values_one[bin[i]][m] += attribute[i][j]; } // DENSITY_NUMBER adds 1 to values } else if (which[m] == DENSITY_NUMBER) { if (!regionflag) { for (i = 0; i < nlocal; i++) if (mask[i] & groupbit) values_one[bin[i]][m] += 1.0; } else { for (i = 0; i < nlocal; i++) if (mask[i] & groupbit && region->match(x[i][0],x[i][1],x[i][2])) values_one[bin[i]][m] += 1.0; } // DENSITY_MASS adds mass to values } else if (which[m] == DENSITY_MASS) { int *type = atom->type; double *mass = atom->mass; double *rmass = atom->rmass; if (!regionflag) { for (i = 0; i < nlocal; i++) if (mask[i] & groupbit) { if (rmass) values_one[bin[i]][m] += rmass[i]; else values_one[bin[i]][m] += mass[type[i]]; } } else { for (i = 0; i < nlocal; i++) if (mask[i] & groupbit && region->match(x[i][0],x[i][1],x[i][2])) { if (rmass) values_one[bin[i]][m] += rmass[i]; else values_one[bin[i]][m] += mass[type[i]]; } } // COMPUTE adds its scalar or vector component to values // 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; } double *vector = compute->vector_atom; double **array = compute->array_atom; int jm1 = j - 1; if (!regionflag) { for (i = 0; i < nlocal; i++) if (mask[i] & groupbit) { if (j == 0) values_one[bin[i]][m] += vector[i]; else values_one[bin[i]][m] += array[i][jm1]; } } else { for (i = 0; i < nlocal; i++) if (mask[i] & groupbit && region->match(x[i][0],x[i][1],x[i][2])) { if (j == 0) values_one[bin[i]][m] += vector[i]; else values_one[bin[i]][m] += array[i][jm1]; } } // FIX adds its scalar or vector component to values // access fix fields, guaranteed to be ready } else if (which[m] == FIX) { double *vector = modify->fix[n]->vector_atom; double **array = modify->fix[n]->array_atom; int jm1 = j - 1; if (!regionflag) { for (i = 0; i < nlocal; i++) if (mask[i] & groupbit) { if (j == 0) values_one[bin[i]][m] += vector[i]; else values_one[bin[i]][m] += array[i][jm1]; } } else { for (i = 0; i < nlocal; i++) if (mask[i] & groupbit && region->match(x[i][0],x[i][1],x[i][2])) { if (j == 0) values_one[bin[i]][m] += vector[i]; else values_one[bin[i]][m] += array[i][jm1]; } } // VARIABLE adds its per-atom quantities to values // evaluate atom-style variable } else if (which[m] == VARIABLE) { if (nlocal > maxvar) { maxvar = atom->nmax; memory->destroy(varatom); memory->create(varatom,maxvar,"ave/spatial:varatom"); } input->variable->compute_atom(n,igroup,varatom,1,0); if (!regionflag) { for (i = 0; i < nlocal; i++) if (mask[i] & groupbit) values_one[bin[i]][m] += varatom[i]; } else { for (i = 0; i < nlocal; i++) if (mask[i] & groupbit && region->match(x[i][0],x[i][1],x[i][2])) values_one[bin[i]][m] += varatom[i]; } } } // process a single sample // if normflag = ALL, accumulate values,count separately to many // if normflag = SAMPLE, one = value/count, accumulate one to many // exception is SAMPLE density: no normalization by atom count if (normflag == ALL) { for (m = 0; m < nbins; m++) { count_many[m] += count_one[m]; for (j = 0; j < nvalues; j++) values_many[m][j] += values_one[m][j]; } } else { MPI_Allreduce(count_one,count_many,nbins,MPI_DOUBLE,MPI_SUM,world); for (m = 0; m < nbins; m++) { if (count_many[m] > 0.0) for (j = 0; j < nvalues; j++) { if (which[j] == DENSITY_NUMBER || which[j] == DENSITY_MASS) values_many[m][j] += values_one[m][j]; else values_many[m][j] += values_one[m][j]/count_many[m]; } count_sum[m] += count_many[m]; } } // done if irepeat < nrepeat // else reset irepeat and nvalid irepeat++; if (irepeat < nrepeat) { nvalid += nevery; modify->addstep_compute(nvalid); return; } irepeat = 0; nvalid = ntimestep+nfreq - (nrepeat-1)*nevery; modify->addstep_compute(nvalid); // time average across samples // if normflag = ALL, final is total value / total count // if normflag = SAMPLE, final is sum of ave / repeat // exception is densities: normalized by repeat, not total count double repeat = nrepeat; double mv2d = force->mv2d; if (normflag == ALL) { MPI_Allreduce(count_many,count_sum,nbins,MPI_DOUBLE,MPI_SUM,world); MPI_Allreduce(&values_many[0][0],&values_sum[0][0],nbins*nvalues, MPI_DOUBLE,MPI_SUM,world); for (m = 0; m < nbins; m++) { if (count_sum[m] > 0.0) for (j = 0; j < nvalues; j++) if (which[j] == DENSITY_NUMBER) values_sum[m][j] /= repeat; else if (which[j] == DENSITY_MASS) values_sum[m][j] *= mv2d/repeat; else values_sum[m][j] /= count_sum[m]; count_sum[m] /= repeat; } } else { MPI_Allreduce(&values_many[0][0],&values_sum[0][0],nbins*nvalues, MPI_DOUBLE,MPI_SUM,world); for (m = 0; m < nbins; m++) { for (j = 0; j < nvalues; j++) values_sum[m][j] /= repeat; count_sum[m] /= repeat; } } // density is additionally normalized by bin volume for (j = 0; j < nvalues; j++) if (which[j] == DENSITY_NUMBER || which[j] == DENSITY_MASS) for (m = 0; m < nbins; m++) values_sum[m][j] /= bin_volume; // if ave = ONE, only single Nfreq timestep value is needed // if ave = RUNNING, combine with all previous Nfreq timestep values // if ave = WINDOW, comine with nwindow most recent Nfreq timestep values if (ave == ONE) { for (m = 0; m < nbins; m++) { for (i = 0; i < nvalues; i++) values_total[m][i] = values_sum[m][i]; count_total[m] = count_sum[m]; } norm = 1; } else if (ave == RUNNING) { for (m = 0; m < nbins; m++) { for (i = 0; i < nvalues; i++) values_total[m][i] += values_sum[m][i]; count_total[m] += count_sum[m]; } norm++; } else if (ave == WINDOW) { for (m = 0; m < nbins; m++) { for (i = 0; i < nvalues; i++) { values_total[m][i] += values_sum[m][i]; if (window_limit) values_total[m][i] -= values_list[iwindow][m][i]; values_list[iwindow][m][i] = values_sum[m][i]; } count_total[m] += count_sum[m]; if (window_limit) count_total[m] -= count_list[iwindow][m]; count_list[iwindow][m] = count_sum[m]; } iwindow++; if (iwindow == nwindow) { iwindow = 0; window_limit = 1; } if (window_limit) norm = nwindow; else norm = iwindow; } // output result to file if (fp && me == 0) { if (overwrite) fseek(fp,filepos,SEEK_SET); fprintf(fp,BIGINT_FORMAT " %d\n",ntimestep,nbins); if (ndim == 1) for (m = 0; m < nbins; m++) { fprintf(fp," %d %g %g",m+1,coord[m][0], count_total[m]/norm); for (i = 0; i < nvalues; i++) fprintf(fp," %g",values_total[m][i]/norm); fprintf(fp,"\n"); } else if (ndim == 2) for (m = 0; m < nbins; m++) { fprintf(fp," %d %g %g %g",m+1,coord[m][0],coord[m][1], count_total[m]/norm); for (i = 0; i < nvalues; i++) fprintf(fp," %g",values_total[m][i]/norm); fprintf(fp,"\n"); } else for (m = 0; m < nbins; m++) { fprintf(fp," %d %g %g %g %g",m+1,coord[m][0],coord[m][1],coord[m][2], count_total[m]/norm); for (i = 0; i < nvalues; i++) fprintf(fp," %g",values_total[m][i]/norm); fprintf(fp,"\n"); } fflush(fp); if (overwrite) { long fileend = ftell(fp); ftruncate(fileno(fp),fileend); } } } /* ---------------------------------------------------------------------- setup 1d, 2d, or 3d bins and their extent and coordinates called at setup() and when averaging occurs if box size changes ------------------------------------------------------------------------- */ void FixAveSpatial::setup_bins() { int i,j,k,m,n; double lo,hi,coord1,coord2; // lo = bin boundary immediately below boxlo // hi = bin boundary immediately above boxhi // allocate and initialize arrays based on new bin count double *boxlo,*boxhi,*prd; if (scaleflag == REDUCED) { boxlo = domain->boxlo_lamda; boxhi = domain->boxhi_lamda; prd = domain->prd_lamda; } else { boxlo = domain->boxlo; boxhi = domain->boxhi; prd = domain->prd; } if (domain->dimension == 3) bin_volume = domain->xprd * domain->yprd * domain->zprd; else bin_volume = domain->xprd * domain->yprd; nbins = 1; for (m = 0; m < ndim; m++) { if (originflag[m] == LOWER) origin[m] = boxlo[dim[m]]; else if (originflag[m] == UPPER) origin[m] = boxhi[dim[m]]; else if (originflag[m] == CENTER) origin[m] = 0.5 * (boxlo[dim[m]] + boxhi[dim[m]]); if (origin[m] < boxlo[dim[m]]) { n = static_cast ((boxlo[dim[m]] - origin[m]) * invdelta[m]); lo = origin[m] + n*delta[m]; } else { n = static_cast ((origin[m] - boxlo[dim[m]]) * invdelta[m]); lo = origin[m] - n*delta[m]; if (lo > boxlo[dim[m]]) lo -= delta[m]; } if (origin[m] < boxhi[dim[m]]) { n = static_cast ((boxhi[dim[m]] - origin[m]) * invdelta[m]); hi = origin[m] + n*delta[m]; if (hi < boxhi[dim[m]]) hi += delta[m]; } else { n = static_cast ((origin[m] - boxhi[dim[m]]) * invdelta[m]); hi = origin[m] - n*delta[m]; } offset[m] = lo; nlayers[m] = static_cast ((hi-lo) * invdelta[m] + 0.5); nbins *= nlayers[m]; bin_volume *= delta[m]/prd[dim[m]]; } // reallocate bin arrays if needed if (nbins > maxbin) { maxbin = nbins; memory->grow(count_one,nbins,"ave/spatial:count_one"); memory->grow(count_many,nbins,"ave/spatial:count_many"); memory->grow(count_sum,nbins,"ave/spatial:count_sum"); memory->grow(count_total,nbins,"ave/spatial:count_total"); memory->grow(coord,nbins,ndim,"ave/spatial:coord"); memory->grow(values_one,nbins,nvalues,"ave/spatial:values_one"); memory->grow(values_many,nbins,nvalues,"ave/spatial:values_many"); memory->grow(values_sum,nbins,nvalues,"ave/spatial:values_sum"); memory->grow(values_total,nbins,nvalues,"ave/spatial:values_total"); // only allocate count and values list for ave = WINDOW if (ave == WINDOW) { memory->create(count_list,nwindow,nbins,"ave/spatial:count_list"); memory->create(values_list,nwindow,nbins,nvalues, "ave/spatial:values_list"); } // reinitialize regrown count/values total since they accumulate for (m = 0; m < nbins; m++) { for (i = 0; i < nvalues; i++) values_total[m][i] = 0.0; count_total[m] = 0.0; } } // set bin coordinates if (ndim == 1) { for (i = 0; i < nlayers[0]; i++) coord[i][0] = offset[0] + (i+0.5)*delta[0]; } else if (ndim == 2) { m = 0; for (i = 0; i < nlayers[0]; i++) { coord1 = offset[0] + (i+0.5)*delta[0]; for (j = 0; j < nlayers[1]; j++) { coord[m][0] = coord1; coord[m][1] = offset[1] + (j+0.5)*delta[1]; m++; } } } else if (ndim == 3) { m = 0; for (i = 0; i < nlayers[0]; i++) { coord1 = offset[0] + (i+0.5)*delta[0]; for (j = 0; j < nlayers[1]; j++) { coord2 = offset[1] + (j+0.5)*delta[1]; for (k = 0; k < nlayers[2]; k++) { coord[m][0] = coord1; coord[m][1] = coord2; coord[m][2] = offset[2] + (k+0.5)*delta[2]; m++; } } } } } /* ---------------------------------------------------------------------- assign each atom to a 1d bin ------------------------------------------------------------------------- */ void FixAveSpatial::atom2bin1d() { int i,ibin; double *boxlo,*boxhi,*prd; double xremap; double lamda[3]; double **x = atom->x; int *mask = atom->mask; int nlocal = atom->nlocal; int idim = dim[0]; int nlayerm1 = nlayers[0] - 1; int periodicity = domain->periodicity[idim]; if (periodicity) { if (scaleflag == REDUCED) { boxlo = domain->boxlo_lamda; boxhi = domain->boxhi_lamda; prd = domain->prd_lamda; } else { boxlo = domain->boxlo; boxhi = domain->boxhi; prd = domain->prd; } } // remap each atom's relevant coord back into box via PBC if necessary // if scaleflag = REDUCED, box coords -> lamda coords if (!regionflag) { if (scaleflag == REDUCED) domain->x2lamda(nlocal); for (i = 0; i < nlocal; i++) if (mask[i] & groupbit) { xremap = x[i][idim]; if (periodicity) { if (xremap < boxlo[idim]) xremap += prd[idim]; if (xremap >= boxhi[idim]) xremap -= prd[idim]; } ibin = static_cast ((xremap - offset[0]) * invdelta[0]); ibin = MAX(ibin,0); ibin = MIN(ibin,nlayerm1); bin[i] = ibin; count_one[ibin] += 1.0; } if (scaleflag == REDUCED) domain->lamda2x(nlocal); } else { for (i = 0; i < nlocal; i++) if (mask[i] & groupbit && region->match(x[i][0],x[i][1],x[i][2])) { if (scaleflag == REDUCED) { domain->x2lamda(x[i],lamda); xremap = lamda[idim]; } else xremap = x[i][idim]; if (periodicity) { if (xremap < boxlo[idim]) xremap += prd[idim]; if (xremap >= boxhi[idim]) xremap -= prd[idim]; } ibin = static_cast ((xremap - offset[0]) * invdelta[0]); ibin = MAX(ibin,0); ibin = MIN(ibin,nlayerm1); bin[i] = ibin; count_one[ibin] += 1.0; } } } /* ---------------------------------------------------------------------- assign each atom to a 2d bin ------------------------------------------------------------------------- */ void FixAveSpatial::atom2bin2d() { int i,ibin,i1bin,i2bin; double *boxlo,*boxhi,*prd; double xremap,yremap; double lamda[3]; double **x = atom->x; int *mask = atom->mask; int nlocal = atom->nlocal; int idim = dim[0]; int jdim = dim[1]; int nlayer1m1 = nlayers[0] - 1; int nlayer2m1 = nlayers[1] - 1; int* periodicity = domain->periodicity; if (periodicity[idim] || periodicity[jdim]) { if (scaleflag == REDUCED) { boxlo = domain->boxlo_lamda; boxhi = domain->boxhi_lamda; prd = domain->prd_lamda; } else { boxlo = domain->boxlo; boxhi = domain->boxhi; prd = domain->prd; } } // remap each atom's relevant coord back into box via PBC if necessary // if scaleflag = REDUCED, box coords -> lamda coords if (!regionflag) { if (scaleflag == REDUCED) domain->x2lamda(nlocal); for (i = 0; i < nlocal; i++) if (mask[i] & groupbit) { xremap = x[i][idim]; if (periodicity[idim]) { if (xremap < boxlo[idim]) xremap += prd[idim]; if (xremap >= boxhi[idim]) xremap -= prd[idim]; } i1bin = static_cast ((xremap - offset[0]) * invdelta[0]); i1bin = MAX(i1bin,0); i1bin = MIN(i1bin,nlayer1m1); yremap = x[i][jdim]; if (periodicity[jdim]) { if (yremap < boxlo[jdim]) yremap += prd[jdim]; if (yremap >= boxhi[jdim]) yremap -= prd[jdim]; } i2bin = static_cast ((yremap - offset[1]) * invdelta[1]); i2bin = MAX(i2bin,0); i2bin = MIN(i2bin,nlayer2m1); ibin = i1bin*nlayers[1] + i2bin; bin[i] = ibin; count_one[ibin] += 1.0; } if (scaleflag == REDUCED) domain->lamda2x(nlocal); } else { for (i = 0; i < nlocal; i++) if (mask[i] & groupbit && region->match(x[i][0],x[i][1],x[i][2])) { if (scaleflag == REDUCED) { domain->x2lamda(x[i],lamda); xremap = lamda[idim]; yremap = lamda[jdim]; } else { xremap = x[i][idim]; yremap = x[i][jdim]; } if (periodicity[idim]) { if (xremap < boxlo[idim]) xremap += prd[idim]; if (xremap >= boxhi[idim]) xremap -= prd[idim]; } i1bin = static_cast ((xremap - offset[0]) * invdelta[0]); i1bin = MAX(i1bin,0); i1bin = MIN(i1bin,nlayer1m1-1); if (periodicity[jdim]) { if (yremap < boxlo[jdim]) yremap += prd[jdim]; if (yremap >= boxhi[jdim]) yremap -= prd[jdim]; } i2bin = static_cast ((yremap - offset[1]) * invdelta[1]); i2bin = MAX(i2bin,0); i2bin = MIN(i2bin,nlayer2m1-1); ibin = i1bin*nlayers[1] + i2bin; bin[i] = ibin; count_one[ibin] += 1.0; } } } /* ---------------------------------------------------------------------- assign each atom to a 3d bin ------------------------------------------------------------------------- */ void FixAveSpatial::atom2bin3d() { int i,ibin,i1bin,i2bin,i3bin; double *boxlo,*boxhi,*prd; double xremap,yremap,zremap; double lamda[3]; double **x = atom->x; int *mask = atom->mask; int nlocal = atom->nlocal; int idim = dim[0]; int jdim = dim[1]; int kdim = dim[2]; int nlayer1m1 = nlayers[0] - 1; int nlayer2m1 = nlayers[1] - 1; int nlayer3m1 = nlayers[2] - 1; int* periodicity = domain->periodicity; if (periodicity[idim] || periodicity[jdim] || periodicity[kdim]) { if (scaleflag == REDUCED) { boxlo = domain->boxlo_lamda; boxhi = domain->boxhi_lamda; prd = domain->prd_lamda; } else { boxlo = domain->boxlo; boxhi = domain->boxhi; prd = domain->prd; } } // remap each atom's relevant coord back into box via PBC if necessary // if scaleflag = REDUCED, box coords -> lamda coords if (!regionflag) { if (scaleflag == REDUCED) domain->x2lamda(nlocal); for (i = 0; i < nlocal; i++) if (mask[i] & groupbit) { xremap = x[i][idim]; if (periodicity[idim]) { if (xremap < boxlo[idim]) xremap += prd[idim]; if (xremap >= boxhi[idim]) xremap -= prd[idim]; } i1bin = static_cast ((xremap - offset[0]) * invdelta[0]); i1bin = MAX(i1bin,0); i1bin = MIN(i1bin,nlayer1m1); yremap = x[i][jdim]; if (periodicity[jdim]) { if (yremap < boxlo[jdim]) yremap += prd[jdim]; if (yremap >= boxhi[jdim]) yremap -= prd[jdim]; } i2bin = static_cast ((yremap - offset[1]) * invdelta[1]); i2bin = MAX(i2bin,0); i2bin = MIN(i2bin,nlayer2m1); zremap = x[i][kdim]; if (periodicity[kdim]) { if (zremap < boxlo[kdim]) zremap += prd[kdim]; if (zremap >= boxhi[kdim]) zremap -= prd[kdim]; } i3bin = static_cast ((zremap - offset[2]) * invdelta[2]); i3bin = MAX(i3bin,0); i3bin = MIN(i3bin,nlayer3m1); ibin = i1bin*nlayers[1]*nlayers[2] + i2bin*nlayers[2] + i3bin; bin[i] = ibin; count_one[ibin] += 1.0; } if (scaleflag == REDUCED) domain->lamda2x(nlocal); } else { for (i = 0; i < nlocal; i++) if (mask[i] & groupbit && region->match(x[i][0],x[i][1],x[i][2])) { if (scaleflag == REDUCED) { domain->x2lamda(x[i],lamda); xremap = lamda[idim]; yremap = lamda[jdim]; zremap = lamda[kdim]; } else { xremap = x[i][idim]; yremap = x[i][jdim]; zremap = x[i][kdim]; } if (periodicity[idim]) { if (xremap < boxlo[idim]) xremap += prd[idim]; if (xremap >= boxhi[idim]) xremap -= prd[idim]; } i1bin = static_cast ((xremap - offset[0]) * invdelta[0]); i1bin = MAX(i1bin,0); i1bin = MIN(i1bin,nlayer1m1); if (periodicity[jdim]) { if (yremap < boxlo[jdim]) yremap += prd[jdim]; if (yremap >= boxhi[jdim]) yremap -= prd[jdim]; } i2bin = static_cast ((yremap - offset[1]) * invdelta[1]); i2bin = MAX(i2bin,0); i2bin = MIN(i2bin,nlayer2m1); if (periodicity[kdim]) { if (zremap < boxlo[kdim]) zremap += prd[kdim]; if (zremap >= boxhi[kdim]) zremap -= prd[kdim]; } i3bin = static_cast ((zremap - offset[2]) * invdelta[2]); i3bin = MAX(i3bin,0); i3bin = MIN(i3bin,nlayer3m1); ibin = i1bin*nlayers[1]*nlayers[2] + i2bin*nlayers[2] + i3bin; bin[i] = ibin; count_one[ibin] += 1.0; } } } /* ---------------------------------------------------------------------- return I,J array value if I exceeds current bins, return 0.0 instead of generating an error column 1,2,3 = bin coords, next column = count, remaining columns = Nvalues ------------------------------------------------------------------------- */ double FixAveSpatial::compute_array(int i, int j) { if (values_total == NULL) return 0.0; if (i >= nbins) return 0.0; if (j < ndim) return coord[i][j]; j -= ndim+1; if (!norm) return 0.0; if (j < 0) return count_total[i]/norm; return values_total[i][j]/norm; } /* ---------------------------------------------------------------------- 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 FixAveSpatial::nextvalid() { bigint nvalid = (update->ntimestep/nfreq)*nfreq + nfreq; if (nvalid-nfreq == update->ntimestep && nrepeat == 1) nvalid = update->ntimestep; else nvalid -= (nrepeat-1)*nevery; if (nvalid < update->ntimestep) nvalid += nfreq; return nvalid; } /* ---------------------------------------------------------------------- memory usage of varatom and bins ------------------------------------------------------------------------- */ double FixAveSpatial::memory_usage() { double bytes = maxvar * sizeof(double); // varatom bytes += maxatom * sizeof(int); // bin bytes += 4*nbins * sizeof(double); // count one,many,sum,total bytes += ndim*nbins * sizeof(double); // coord bytes += nvalues*nbins * sizeof(double); // values one,many,sum,total bytes += nwindow*nbins * sizeof(double); // count_list bytes += nwindow*nbins*nvalues * sizeof(double); // values_list return bytes; } /* ---------------------------------------------------------------------- */ void FixAveSpatial::reset_timestep(bigint ntimestep) { if (ntimestep > nvalid) error->all(FLERR,"Fix ave/spatial missed timestep"); }