2020-08-18 14:27:28 +02:00
|
|
|
/*
|
2022-09-05 10:39:42 +02:00
|
|
|
* Copyright (C) 2022 NHR@FAU, University Erlangen-Nuremberg.
|
|
|
|
* All rights reserved. This file is part of MD-Bench.
|
|
|
|
* Use of this source code is governed by a LGPL-3.0
|
|
|
|
* license that can be found in the LICENSE file.
|
2020-08-18 14:27:28 +02:00
|
|
|
*/
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
2022-01-31 17:49:22 +01:00
|
|
|
#include <math.h>
|
2020-08-18 14:27:28 +02:00
|
|
|
|
|
|
|
#include <pbc.h>
|
|
|
|
#include <atom.h>
|
|
|
|
#include <allocate.h>
|
2022-01-25 21:00:11 +01:00
|
|
|
#include <neighbor.h>
|
2022-03-09 17:23:49 +01:00
|
|
|
#include <util.h>
|
2020-08-18 14:27:28 +02:00
|
|
|
|
|
|
|
#define DELTA 20000
|
|
|
|
|
|
|
|
static int NmaxGhost;
|
|
|
|
|
2021-11-03 00:57:24 +01:00
|
|
|
static void growPbc(Atom*);
|
2020-08-18 14:27:28 +02:00
|
|
|
|
2020-08-19 09:00:35 +02:00
|
|
|
/* exported subroutines */
|
2022-01-25 00:43:10 +01:00
|
|
|
void initPbc(Atom* atom) {
|
2020-08-18 14:27:28 +02:00
|
|
|
NmaxGhost = 0;
|
2021-11-03 00:57:24 +01:00
|
|
|
atom->border_map = NULL;
|
2022-01-25 12:19:28 +01:00
|
|
|
atom->PBCx = NULL; atom->PBCy = NULL; atom->PBCz = NULL;
|
2020-08-18 14:27:28 +02:00
|
|
|
}
|
|
|
|
|
2020-08-19 09:00:35 +02:00
|
|
|
/* update coordinates of ghost atoms */
|
|
|
|
/* uses mapping created in setupPbc */
|
2022-02-02 00:49:55 +01:00
|
|
|
void updatePbc(Atom *atom, Parameter *param, int firstUpdate) {
|
2022-03-10 01:31:50 +01:00
|
|
|
DEBUG_MESSAGE("updatePbc start\n");
|
2022-03-10 22:33:41 +01:00
|
|
|
int jfac = MAX(1, CLUSTER_N / CLUSTER_M);
|
|
|
|
int ncj = atom->Nclusters_local / jfac;
|
2020-11-05 12:41:44 +01:00
|
|
|
MD_FLOAT xprd = param->xprd;
|
|
|
|
MD_FLOAT yprd = param->yprd;
|
|
|
|
MD_FLOAT zprd = param->zprd;
|
2020-08-18 14:27:28 +02:00
|
|
|
|
2022-01-31 17:49:22 +01:00
|
|
|
for(int cg = 0; cg < atom->Nclusters_ghost; cg++) {
|
2022-03-10 22:33:41 +01:00
|
|
|
const int cj = ncj + cg;
|
|
|
|
int cj_vec_base = CJ_VECTOR_BASE_INDEX(cj);
|
|
|
|
int bmap_vec_base = CJ_VECTOR_BASE_INDEX(atom->border_map[cg]);
|
|
|
|
MD_FLOAT *cj_x = &atom->cl_x[cj_vec_base];
|
2022-03-09 02:25:39 +01:00
|
|
|
MD_FLOAT *bmap_x = &atom->cl_x[bmap_vec_base];
|
2022-01-31 17:49:22 +01:00
|
|
|
MD_FLOAT bbminx = INFINITY, bbmaxx = -INFINITY;
|
|
|
|
MD_FLOAT bbminy = INFINITY, bbmaxy = -INFINITY;
|
|
|
|
MD_FLOAT bbminz = INFINITY, bbmaxz = -INFINITY;
|
2022-01-25 00:43:10 +01:00
|
|
|
|
2022-03-10 22:33:41 +01:00
|
|
|
for(int cjj = 0; cjj < atom->jclusters[cj].natoms; cjj++) {
|
|
|
|
MD_FLOAT xtmp = bmap_x[CL_X_OFFSET + cjj] + atom->PBCx[cg] * xprd;
|
|
|
|
MD_FLOAT ytmp = bmap_x[CL_Y_OFFSET + cjj] + atom->PBCy[cg] * yprd;
|
|
|
|
MD_FLOAT ztmp = bmap_x[CL_Z_OFFSET + cjj] + atom->PBCz[cg] * zprd;
|
2022-01-31 17:49:22 +01:00
|
|
|
|
2022-03-10 22:33:41 +01:00
|
|
|
cj_x[CL_X_OFFSET + cjj] = xtmp;
|
|
|
|
cj_x[CL_Y_OFFSET + cjj] = ytmp;
|
|
|
|
cj_x[CL_Z_OFFSET + cjj] = ztmp;
|
2022-01-31 17:49:22 +01:00
|
|
|
|
2022-02-02 00:49:55 +01:00
|
|
|
if(firstUpdate) {
|
2022-01-31 17:49:22 +01:00
|
|
|
// TODO: To create the bounding boxes faster, we can use SIMD operations
|
|
|
|
if(bbminx > xtmp) { bbminx = xtmp; }
|
|
|
|
if(bbmaxx < xtmp) { bbmaxx = xtmp; }
|
|
|
|
if(bbminy > ytmp) { bbminy = ytmp; }
|
|
|
|
if(bbmaxy < ytmp) { bbmaxy = ytmp; }
|
|
|
|
if(bbminz > ztmp) { bbminz = ztmp; }
|
|
|
|
if(bbmaxz < ztmp) { bbmaxz = ztmp; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-02 00:49:55 +01:00
|
|
|
if(firstUpdate) {
|
2022-03-10 22:33:41 +01:00
|
|
|
for(int cjj = atom->jclusters[cj].natoms; cjj < CLUSTER_N; cjj++) {
|
|
|
|
cj_x[CL_X_OFFSET + cjj] = INFINITY;
|
|
|
|
cj_x[CL_Y_OFFSET + cjj] = INFINITY;
|
|
|
|
cj_x[CL_Z_OFFSET + cjj] = INFINITY;
|
2022-02-02 00:49:55 +01:00
|
|
|
}
|
|
|
|
|
2022-03-10 22:33:41 +01:00
|
|
|
atom->jclusters[cj].bbminx = bbminx;
|
|
|
|
atom->jclusters[cj].bbmaxx = bbmaxx;
|
|
|
|
atom->jclusters[cj].bbminy = bbminy;
|
|
|
|
atom->jclusters[cj].bbmaxy = bbmaxy;
|
|
|
|
atom->jclusters[cj].bbminz = bbminz;
|
|
|
|
atom->jclusters[cj].bbmaxz = bbmaxz;
|
2022-01-25 00:43:10 +01:00
|
|
|
}
|
2020-08-18 14:27:28 +02:00
|
|
|
}
|
2022-03-10 01:31:50 +01:00
|
|
|
|
|
|
|
DEBUG_MESSAGE("updatePbc end\n");
|
2020-08-18 14:27:28 +02:00
|
|
|
}
|
|
|
|
|
2020-08-19 09:00:35 +02:00
|
|
|
/* relocate atoms that have left domain according
|
|
|
|
* to periodic boundary conditions */
|
2022-01-25 00:43:10 +01:00
|
|
|
void updateAtomsPbc(Atom *atom, Parameter *param) {
|
2020-11-05 12:41:44 +01:00
|
|
|
MD_FLOAT xprd = param->xprd;
|
|
|
|
MD_FLOAT yprd = param->yprd;
|
|
|
|
MD_FLOAT zprd = param->zprd;
|
2020-08-18 14:27:28 +02:00
|
|
|
|
|
|
|
for(int i = 0; i < atom->Nlocal; i++) {
|
2021-03-23 10:03:55 +01:00
|
|
|
if(atom_x(i) < 0.0) {
|
|
|
|
atom_x(i) += xprd;
|
|
|
|
} else if(atom_x(i) >= xprd) {
|
|
|
|
atom_x(i) -= xprd;
|
2022-01-25 12:19:28 +01:00
|
|
|
}
|
2020-08-18 14:27:28 +02:00
|
|
|
|
2021-03-23 10:03:55 +01:00
|
|
|
if(atom_y(i) < 0.0) {
|
|
|
|
atom_y(i) += yprd;
|
|
|
|
} else if(atom_y(i) >= yprd) {
|
|
|
|
atom_y(i) -= yprd;
|
2020-08-18 14:27:28 +02:00
|
|
|
}
|
|
|
|
|
2021-03-23 10:03:55 +01:00
|
|
|
if(atom_z(i) < 0.0) {
|
|
|
|
atom_z(i) += zprd;
|
|
|
|
} else if(atom_z(i) >= zprd) {
|
|
|
|
atom_z(i) -= zprd;
|
2020-08-18 14:27:28 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-19 09:00:35 +02:00
|
|
|
/* setup periodic boundary conditions by
|
|
|
|
* defining ghost atoms around domain
|
|
|
|
* only creates mapping and coordinate corrections
|
|
|
|
* that are then enforced in updatePbc */
|
2022-01-31 23:46:20 +01:00
|
|
|
#define ADDGHOST(dx,dy,dz); \
|
|
|
|
Nghost++; \
|
2022-03-10 22:33:41 +01:00
|
|
|
const int cg = ncj + Nghost; \
|
|
|
|
const int cj_natoms = atom->jclusters[cj].natoms; \
|
|
|
|
atom->border_map[Nghost] = cj; \
|
2022-01-31 23:46:20 +01:00
|
|
|
atom->PBCx[Nghost] = dx; \
|
|
|
|
atom->PBCy[Nghost] = dy; \
|
|
|
|
atom->PBCz[Nghost] = dz; \
|
2022-03-10 22:33:41 +01:00
|
|
|
atom->jclusters[cg].natoms = cj_natoms; \
|
|
|
|
Nghost_atoms += cj_natoms; \
|
|
|
|
int cj_sca_base = CJ_SCALAR_BASE_INDEX(cj); \
|
|
|
|
int cg_sca_base = CJ_SCALAR_BASE_INDEX(cg); \
|
|
|
|
for(int cjj = 0; cjj < cj_natoms; cjj++) { \
|
|
|
|
atom->cl_type[cg_sca_base + cjj] = atom->cl_type[cj_sca_base + cjj]; \
|
2022-01-25 00:43:10 +01:00
|
|
|
}
|
|
|
|
|
2022-01-25 12:19:28 +01:00
|
|
|
/* internal subroutines */
|
|
|
|
void growPbc(Atom* atom) {
|
|
|
|
int nold = NmaxGhost;
|
|
|
|
NmaxGhost += DELTA;
|
|
|
|
|
|
|
|
atom->border_map = (int*) reallocate(atom->border_map, ALIGNMENT, NmaxGhost * sizeof(int), nold * sizeof(int));
|
|
|
|
atom->PBCx = (int*) reallocate(atom->PBCx, ALIGNMENT, NmaxGhost * sizeof(int), nold * sizeof(int));
|
|
|
|
atom->PBCy = (int*) reallocate(atom->PBCy, ALIGNMENT, NmaxGhost * sizeof(int), nold * sizeof(int));
|
|
|
|
atom->PBCz = (int*) reallocate(atom->PBCz, ALIGNMENT, NmaxGhost * sizeof(int), nold * sizeof(int));
|
|
|
|
}
|
|
|
|
|
2022-01-25 00:43:10 +01:00
|
|
|
void setupPbc(Atom *atom, Parameter *param) {
|
2022-03-10 01:31:50 +01:00
|
|
|
DEBUG_MESSAGE("setupPbc start\n");
|
2020-11-05 12:41:44 +01:00
|
|
|
MD_FLOAT xprd = param->xprd;
|
|
|
|
MD_FLOAT yprd = param->yprd;
|
|
|
|
MD_FLOAT zprd = param->zprd;
|
|
|
|
MD_FLOAT Cutneigh = param->cutneigh;
|
2022-03-10 22:33:41 +01:00
|
|
|
int jfac = MAX(1, CLUSTER_N / CLUSTER_M);
|
|
|
|
int ncj = atom->Nclusters_local / jfac;
|
2020-08-18 14:27:28 +02:00
|
|
|
int Nghost = -1;
|
2022-01-31 23:46:20 +01:00
|
|
|
int Nghost_atoms = 0;
|
2020-08-18 14:27:28 +02:00
|
|
|
|
2022-03-10 22:33:41 +01:00
|
|
|
for(int cj = 0; cj < ncj; cj++) {
|
|
|
|
if(atom->jclusters[cj].natoms > 0) {
|
|
|
|
if(atom->Nclusters_local + (Nghost + 7) * jfac >= atom->Nclusters_max) {
|
|
|
|
growClusters(atom);
|
|
|
|
}
|
2022-01-25 00:43:10 +01:00
|
|
|
|
2022-03-10 22:33:41 +01:00
|
|
|
if((Nghost + 7) * jfac >= NmaxGhost) {
|
|
|
|
growPbc(atom);
|
|
|
|
}
|
2020-08-18 14:27:28 +02:00
|
|
|
|
2022-03-10 22:33:41 +01:00
|
|
|
MD_FLOAT bbminx = atom->jclusters[cj].bbminx;
|
|
|
|
MD_FLOAT bbmaxx = atom->jclusters[cj].bbmaxx;
|
|
|
|
MD_FLOAT bbminy = atom->jclusters[cj].bbminy;
|
|
|
|
MD_FLOAT bbmaxy = atom->jclusters[cj].bbmaxy;
|
|
|
|
MD_FLOAT bbminz = atom->jclusters[cj].bbminz;
|
|
|
|
MD_FLOAT bbmaxz = atom->jclusters[cj].bbmaxz;
|
|
|
|
|
|
|
|
/* Setup ghost atoms */
|
|
|
|
/* 6 planes */
|
|
|
|
if (bbminx < Cutneigh) { ADDGHOST(+1,0,0); }
|
|
|
|
if (bbmaxx >= (xprd-Cutneigh)) { ADDGHOST(-1,0,0); }
|
|
|
|
if (bbminy < Cutneigh) { ADDGHOST(0,+1,0); }
|
|
|
|
if (bbmaxy >= (yprd-Cutneigh)) { ADDGHOST(0,-1,0); }
|
|
|
|
if (bbminz < Cutneigh) { ADDGHOST(0,0,+1); }
|
|
|
|
if (bbmaxz >= (zprd-Cutneigh)) { ADDGHOST(0,0,-1); }
|
|
|
|
/* 8 corners */
|
|
|
|
if (bbminx < Cutneigh && bbminy < Cutneigh && bbminz < Cutneigh) { ADDGHOST(+1,+1,+1); }
|
|
|
|
if (bbminx < Cutneigh && bbmaxy >= (yprd-Cutneigh) && bbminz < Cutneigh) { ADDGHOST(+1,-1,+1); }
|
|
|
|
if (bbminx < Cutneigh && bbminy < Cutneigh && bbmaxz >= (zprd-Cutneigh)) { ADDGHOST(+1,+1,-1); }
|
|
|
|
if (bbminx < Cutneigh && bbmaxy >= (yprd-Cutneigh) && bbmaxz >= (zprd-Cutneigh)) { ADDGHOST(+1,-1,-1); }
|
|
|
|
if (bbmaxx >= (xprd-Cutneigh) && bbminy < Cutneigh && bbminz < Cutneigh) { ADDGHOST(-1,+1,+1); }
|
|
|
|
if (bbmaxx >= (xprd-Cutneigh) && bbmaxy >= (yprd-Cutneigh) && bbminz < Cutneigh) { ADDGHOST(-1,-1,+1); }
|
|
|
|
if (bbmaxx >= (xprd-Cutneigh) && bbminy < Cutneigh && bbmaxz >= (zprd-Cutneigh)) { ADDGHOST(-1,+1,-1); }
|
|
|
|
if (bbmaxx >= (xprd-Cutneigh) && bbmaxy >= (yprd-Cutneigh) && bbmaxz >= (zprd-Cutneigh)) { ADDGHOST(-1,-1,-1); }
|
|
|
|
/* 12 edges */
|
|
|
|
if (bbminx < Cutneigh && bbminz < Cutneigh) { ADDGHOST(+1,0,+1); }
|
|
|
|
if (bbminx < Cutneigh && bbmaxz >= (zprd-Cutneigh)) { ADDGHOST(+1,0,-1); }
|
|
|
|
if (bbmaxx >= (xprd-Cutneigh) && bbminz < Cutneigh) { ADDGHOST(-1,0,+1); }
|
|
|
|
if (bbmaxx >= (xprd-Cutneigh) && bbmaxz >= (zprd-Cutneigh)) { ADDGHOST(-1,0,-1); }
|
|
|
|
if (bbminy < Cutneigh && bbminz < Cutneigh) { ADDGHOST(0,+1,+1); }
|
|
|
|
if (bbminy < Cutneigh && bbmaxz >= (zprd-Cutneigh)) { ADDGHOST(0,+1,-1); }
|
|
|
|
if (bbmaxy >= (yprd-Cutneigh) && bbminz < Cutneigh) { ADDGHOST(0,-1,+1); }
|
|
|
|
if (bbmaxy >= (yprd-Cutneigh) && bbmaxz >= (zprd-Cutneigh)) { ADDGHOST(0,-1,-1); }
|
|
|
|
if (bbminy < Cutneigh && bbminx < Cutneigh) { ADDGHOST(+1,+1,0); }
|
|
|
|
if (bbminy < Cutneigh && bbmaxx >= (xprd-Cutneigh)) { ADDGHOST(-1,+1,0); }
|
|
|
|
if (bbmaxy >= (yprd-Cutneigh) && bbminx < Cutneigh) { ADDGHOST(+1,-1,0); }
|
|
|
|
if (bbmaxy >= (yprd-Cutneigh) && bbmaxx >= (xprd-Cutneigh)) { ADDGHOST(-1,-1,0); }
|
|
|
|
}
|
2020-08-18 14:27:28 +02:00
|
|
|
}
|
2022-01-25 00:43:10 +01:00
|
|
|
|
2022-03-10 22:33:41 +01:00
|
|
|
if(ncj + (Nghost + 1) * jfac >= atom->Nclusters_max) {
|
2022-02-02 18:00:44 +01:00
|
|
|
growClusters(atom);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add dummy cluster at the end
|
2022-03-10 22:33:41 +01:00
|
|
|
int cj_vec_base = CJ_VECTOR_BASE_INDEX(ncj + Nghost + 1);
|
|
|
|
MD_FLOAT *cj_x = &atom->cl_x[cj_vec_base];
|
|
|
|
for(int cjj = 0; cjj < CLUSTER_N; cjj++) {
|
|
|
|
cj_x[CL_X_OFFSET + cjj] = INFINITY;
|
|
|
|
cj_x[CL_Y_OFFSET + cjj] = INFINITY;
|
|
|
|
cj_x[CL_Z_OFFSET + cjj] = INFINITY;
|
2022-02-02 18:00:44 +01:00
|
|
|
}
|
|
|
|
|
2020-08-18 14:27:28 +02:00
|
|
|
// increase by one to make it the ghost atom count
|
2022-03-10 22:33:41 +01:00
|
|
|
atom->dummy_cj = ncj + Nghost + 1;
|
2022-01-31 23:46:20 +01:00
|
|
|
atom->Nghost = Nghost_atoms;
|
2022-01-25 00:43:10 +01:00
|
|
|
atom->Nclusters_ghost = Nghost + 1;
|
|
|
|
atom->Nclusters = atom->Nclusters_local + Nghost + 1;
|
2022-01-25 21:00:11 +01:00
|
|
|
|
2022-01-27 03:07:31 +01:00
|
|
|
// Update created ghost clusters positions
|
2022-01-31 17:49:22 +01:00
|
|
|
updatePbc(atom, param, 1);
|
2022-03-10 01:31:50 +01:00
|
|
|
DEBUG_MESSAGE("setupPbc end\n");
|
2020-08-18 14:27:28 +02:00
|
|
|
}
|