2020-08-11 16:34:22 +02:00
|
|
|
/*
|
|
|
|
* =======================================================================================
|
|
|
|
*
|
|
|
|
* Author: Jan Eitzinger (je), jan.eitzinger@fau.de
|
2020-08-18 14:27:28 +02:00
|
|
|
* Copyright (c) 2020 RRZE, University Erlangen-Nuremberg
|
2020-08-11 16:34:22 +02:00
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
|
|
* in the Software without restriction, including without limitation the rights
|
|
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in all
|
|
|
|
* copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
|
|
* SOFTWARE.
|
|
|
|
*
|
|
|
|
* =======================================================================================
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <limits.h>
|
|
|
|
#include <math.h>
|
|
|
|
#include <float.h>
|
|
|
|
|
2020-08-19 09:00:35 +02:00
|
|
|
#include <timing.h>
|
2020-08-18 14:27:28 +02:00
|
|
|
#include <allocate.h>
|
|
|
|
#include <neighbor.h>
|
|
|
|
#include <parameter.h>
|
|
|
|
#include <atom.h>
|
|
|
|
#include <thermo.h>
|
|
|
|
#include <pbc.h>
|
|
|
|
|
2020-08-11 16:34:22 +02:00
|
|
|
#define HLINE "----------------------------------------------------------------------------\n"
|
|
|
|
|
2020-08-19 09:00:35 +02:00
|
|
|
typedef enum {
|
|
|
|
TOTAL = 0,
|
|
|
|
NEIGH,
|
|
|
|
FORCE,
|
|
|
|
NUMTIMER
|
|
|
|
} timertype;
|
2020-08-11 16:34:22 +02:00
|
|
|
|
2020-08-14 08:32:36 +02:00
|
|
|
|
2020-08-19 09:00:35 +02:00
|
|
|
void init(Parameter *param)
|
|
|
|
{
|
2020-08-11 16:34:22 +02:00
|
|
|
param->epsilon = 1.0;
|
|
|
|
param->sigma6 = 1.0;
|
|
|
|
param->rho = 0.8442;
|
|
|
|
param->ntimes = 200;
|
|
|
|
param->dt = 0.005;
|
|
|
|
param->nx = 32;
|
|
|
|
param->ny = 32;
|
|
|
|
param->nz = 32;
|
|
|
|
param->cutforce = 2.5;
|
2020-08-18 14:27:28 +02:00
|
|
|
param->cutneigh = param->cutforce + 0.30;
|
2020-08-11 16:34:22 +02:00
|
|
|
param->temp = 1.44;
|
|
|
|
param->nstat = 100;
|
|
|
|
param->mass = 1.0;
|
|
|
|
param->dtforce = 0.5 * param->dt;
|
2020-08-18 14:27:28 +02:00
|
|
|
param->every = 20;
|
2020-08-11 16:34:22 +02:00
|
|
|
double lattice = pow((4.0 / param->rho), (1.0 / 3.0));
|
2020-08-18 14:27:28 +02:00
|
|
|
param->xprd = param->nx * lattice;
|
|
|
|
param->yprd = param->ny * lattice;
|
|
|
|
param->zprd = param->nz * lattice;
|
2020-08-14 08:32:36 +02:00
|
|
|
}
|
|
|
|
|
2020-08-19 09:22:43 +02:00
|
|
|
double setup(
|
|
|
|
Parameter *param,
|
|
|
|
Atom *atom,
|
|
|
|
Neighbor *neighbor)
|
|
|
|
{
|
|
|
|
double S, E;
|
|
|
|
|
|
|
|
S = getTimeStamp();
|
2020-08-19 09:00:35 +02:00
|
|
|
initAtom(atom);
|
|
|
|
initNeighbor(neighbor, param);
|
|
|
|
initPbc();
|
|
|
|
setupNeighbor();
|
|
|
|
createAtom(atom, param);
|
|
|
|
setupThermo(param, atom->Natoms);
|
|
|
|
adjustThermo(param, atom);
|
|
|
|
setupPbc(atom, param);
|
|
|
|
updatePbc(atom, param);
|
|
|
|
buildNeighbor(atom, neighbor);
|
2020-08-19 09:22:43 +02:00
|
|
|
E = getTimeStamp();
|
|
|
|
|
|
|
|
return E-S;
|
|
|
|
}
|
|
|
|
|
|
|
|
double reneighbour(
|
|
|
|
Parameter *param,
|
|
|
|
Atom *atom,
|
|
|
|
Neighbor *neighbor)
|
|
|
|
{
|
|
|
|
double S, E;
|
|
|
|
|
|
|
|
S = getTimeStamp();
|
|
|
|
updateAtomsPbc(atom, param);
|
|
|
|
setupPbc(atom, param);
|
|
|
|
updatePbc(atom, param);
|
|
|
|
/* sortAtom(); */
|
|
|
|
buildNeighbor(atom, neighbor);
|
|
|
|
E = getTimeStamp();
|
|
|
|
|
|
|
|
return E-S;
|
2020-08-19 09:00:35 +02:00
|
|
|
}
|
|
|
|
|
2020-08-18 14:27:28 +02:00
|
|
|
void initialIntegrate(Parameter *param, Atom *atom)
|
2020-08-14 08:32:36 +02:00
|
|
|
{
|
2020-08-18 14:27:28 +02:00
|
|
|
double* x = atom->x; double* y = atom->y; double* z = atom->z;
|
|
|
|
double* fx = atom->fx; double* fy = atom->fy; double* fz = atom->fz;
|
|
|
|
double* vx = atom->vx; double* vy = atom->vy; double* vz = atom->vz;
|
2020-08-11 16:34:22 +02:00
|
|
|
|
2020-08-18 14:27:28 +02:00
|
|
|
for(int i = 0; i < atom->Nlocal; i++) {
|
2020-08-11 16:34:22 +02:00
|
|
|
vx[i] += param->dtforce * fx[i];
|
|
|
|
vy[i] += param->dtforce * fy[i];
|
|
|
|
vz[i] += param->dtforce * fz[i];
|
|
|
|
x[i] += param->dt * vx[i];
|
|
|
|
y[i] += param->dt * vy[i];
|
|
|
|
z[i] += param->dt * vz[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-18 14:27:28 +02:00
|
|
|
void finalIntegrate(Parameter *param, Atom *atom)
|
2020-08-11 16:34:22 +02:00
|
|
|
{
|
2020-08-18 14:27:28 +02:00
|
|
|
double* fx = atom->fx; double* fy = atom->fy; double* fz = atom->fz;
|
|
|
|
double* vx = atom->vx; double* vy = atom->vy; double* vz = atom->vz;
|
|
|
|
|
|
|
|
for(int i = 0; i < atom->Nlocal; i++) {
|
2020-08-11 16:34:22 +02:00
|
|
|
vx[i] += param->dtforce * fx[i];
|
|
|
|
vy[i] += param->dtforce * fy[i];
|
|
|
|
vz[i] += param->dtforce * fz[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-19 09:22:43 +02:00
|
|
|
double computeForce(Parameter *param, Atom *atom, Neighbor *neighbor)
|
2020-08-11 16:34:22 +02:00
|
|
|
{
|
2020-08-18 14:27:28 +02:00
|
|
|
int Nlocal = atom->Natoms;
|
2020-08-11 16:34:22 +02:00
|
|
|
int* neighs;
|
|
|
|
double cutforcesq = param->cutforce * param->cutforce;
|
|
|
|
double sigma6 = param->sigma6;
|
|
|
|
double epsilon = param->epsilon;
|
2020-08-18 14:27:28 +02:00
|
|
|
double* x = atom->x; double* y = atom->y; double* z = atom->z;
|
|
|
|
double* fx = atom->fx; double* fy = atom->fy; double* fz = atom->fz;
|
2020-08-19 09:22:43 +02:00
|
|
|
double S, E;
|
2020-08-11 16:34:22 +02:00
|
|
|
|
2020-08-19 09:22:43 +02:00
|
|
|
S = getTimeStamp();
|
2020-08-11 16:34:22 +02:00
|
|
|
for(int i = 0; i < Nlocal; i++) {
|
|
|
|
fx[i] = 0.0;
|
|
|
|
fy[i] = 0.0;
|
|
|
|
fz[i] = 0.0;
|
|
|
|
}
|
|
|
|
|
|
|
|
for(int i = 0; i < Nlocal; i++) {
|
|
|
|
neighs = &neighbor->neighbors[i * neighbor->maxneighs];
|
|
|
|
int numneighs = neighbor->numneigh[i];
|
|
|
|
double xtmp = x[i];
|
|
|
|
double ytmp = y[i];
|
|
|
|
double ztmp = z[i];
|
|
|
|
|
|
|
|
double fix = 0;
|
|
|
|
double fiy = 0;
|
|
|
|
double fiz = 0;
|
|
|
|
|
|
|
|
for(int k = 0; k < numneighs; k++) {
|
|
|
|
int j = neighs[k];
|
|
|
|
double delx = xtmp - x[j];
|
|
|
|
double dely = ytmp - y[j];
|
|
|
|
double delz = ztmp - z[j];
|
|
|
|
double rsq = delx * delx + dely * dely + delz * delz;
|
|
|
|
|
|
|
|
if(rsq < cutforcesq) {
|
|
|
|
double sr2 = 1.0 / rsq;
|
|
|
|
double sr6 = sr2 * sr2 * sr2 * sigma6;
|
|
|
|
double force = 48.0 * sr6 * (sr6 - 0.5) * sr2 * epsilon;
|
|
|
|
fix += delx * force;
|
|
|
|
fiy += dely * force;
|
|
|
|
fiz += delz * force;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fx[i] += fix;
|
|
|
|
fy[i] += fiy;
|
|
|
|
fz[i] += fiz;
|
|
|
|
}
|
2020-08-19 09:22:43 +02:00
|
|
|
E = getTimeStamp();
|
|
|
|
|
|
|
|
return E-S;
|
2020-08-11 16:34:22 +02:00
|
|
|
}
|
|
|
|
|
2020-08-18 14:27:28 +02:00
|
|
|
|
|
|
|
void printAtomState(Atom *atom)
|
|
|
|
{
|
|
|
|
printf("Atom counts: Natoms=%d Nlocal=%d Nghost=%d Nmax=%d\n",
|
|
|
|
atom->Natoms, atom->Nlocal, atom->Nghost, atom->Nmax);
|
|
|
|
|
|
|
|
int nall = atom->Nlocal + atom->Nghost;
|
|
|
|
|
|
|
|
for (int i=0; i<nall; i++) {
|
|
|
|
printf("%d %f %f %f\n", i, atom->x[i], atom->y[i], atom->z[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-11 16:34:22 +02:00
|
|
|
int main (int argc, char** argv)
|
|
|
|
{
|
2020-08-19 09:00:35 +02:00
|
|
|
double timer[NUMTIMER];
|
2020-08-18 14:27:28 +02:00
|
|
|
Atom atom;
|
2020-08-11 16:34:22 +02:00
|
|
|
Neighbor neighbor;
|
|
|
|
Parameter param;
|
|
|
|
|
2020-08-19 09:00:35 +02:00
|
|
|
init(¶m);
|
|
|
|
|
|
|
|
for(int i = 0; i < argc; i++)
|
|
|
|
{
|
|
|
|
if((strcmp(argv[i], "-n") == 0) || (strcmp(argv[i], "--nsteps") == 0))
|
|
|
|
{
|
|
|
|
param.ntimes = atoi(argv[++i]);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if((strcmp(argv[i], "-nx") == 0))
|
|
|
|
{
|
|
|
|
param.nx = atoi(argv[++i]);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if((strcmp(argv[i], "-ny") == 0))
|
|
|
|
{
|
|
|
|
param.ny = atoi(argv[++i]);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if((strcmp(argv[i], "-nz") == 0))
|
|
|
|
{
|
|
|
|
param.nz = atoi(argv[++i]);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if((strcmp(argv[i], "-h") == 0) || (strcmp(argv[i], "--help") == 0))
|
|
|
|
{
|
|
|
|
printf("MD Bench: A minimalistic re-implementation of miniMD\n");
|
|
|
|
printf(HLINE);
|
2020-08-19 09:22:43 +02:00
|
|
|
printf("-n / --nsteps <int>: set number of timesteps for simulation\n");
|
|
|
|
printf("-nx/-ny/-nz <int>: set linear dimension of systembox in x/y/z direction\n");
|
2020-08-19 09:00:35 +02:00
|
|
|
printf(HLINE);
|
|
|
|
exit(EXIT_SUCCESS);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
setup(¶m, &atom, &neighbor);
|
2020-08-18 14:27:28 +02:00
|
|
|
computeThermo(0, ¶m, &atom);
|
2020-08-19 09:00:35 +02:00
|
|
|
computeForce(¶m, &atom, &neighbor);
|
2020-08-11 16:34:22 +02:00
|
|
|
|
2020-08-19 09:22:43 +02:00
|
|
|
timer[FORCE] = 0.0;
|
|
|
|
timer[NEIGH] = 0.0;
|
|
|
|
timer[TOTAL] = getTimeStamp();
|
|
|
|
|
2020-08-11 16:34:22 +02:00
|
|
|
for(int n = 0; n < param.ntimes; n++) {
|
|
|
|
|
2020-08-18 14:27:28 +02:00
|
|
|
initialIntegrate(¶m, &atom);
|
2020-08-11 16:34:22 +02:00
|
|
|
|
2020-08-18 14:27:28 +02:00
|
|
|
if((n + 1) % param.every) {
|
|
|
|
updatePbc(&atom, ¶m);
|
2020-08-17 14:01:46 +02:00
|
|
|
} else {
|
2020-08-19 09:22:43 +02:00
|
|
|
timer[NEIGH] += reneighbour(¶m, &atom, &neighbor);
|
2020-08-11 16:34:22 +02:00
|
|
|
}
|
|
|
|
|
2020-08-19 09:22:43 +02:00
|
|
|
timer[FORCE] += computeForce(¶m, &atom, &neighbor);
|
2020-08-18 14:27:28 +02:00
|
|
|
finalIntegrate(¶m, &atom);
|
2020-08-11 16:34:22 +02:00
|
|
|
|
2020-08-17 14:01:46 +02:00
|
|
|
if(!((n + 1) % param.nstat) && (n+1) < param.ntimes) {
|
2020-08-18 14:27:28 +02:00
|
|
|
computeThermo(n + 1, ¶m, &atom);
|
2020-08-11 16:34:22 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-19 09:22:43 +02:00
|
|
|
timer[TOTAL] = getTimeStamp() - timer[TOTAL];
|
2020-08-18 14:27:28 +02:00
|
|
|
computeThermo(-1, ¶m, &atom);
|
2020-08-12 15:38:08 +02:00
|
|
|
|
2020-08-19 09:22:43 +02:00
|
|
|
printf(HLINE);
|
|
|
|
printf("System: %d atoms, Steps: %d\n", atom.Natoms, param.ntimes);
|
|
|
|
printf("TOTAL %.2fs FORCE %.2fs NEIGH %.2fs REST %.2fs\n",
|
|
|
|
timer[TOTAL], timer[FORCE], timer[NEIGH], timer[TOTAL]-timer[FORCE]-timer[NEIGH]);
|
|
|
|
printf(HLINE);
|
|
|
|
printf("Performance: %.2f million atom updates per second\n", 1e-6 * (double) (atom.Natoms * param.ntimes) / timer[TOTAL]);
|
|
|
|
|
2020-08-11 16:34:22 +02:00
|
|
|
return EXIT_SUCCESS;
|
|
|
|
}
|