/* * Project: MoleCuilder * Description: creates and alters molecular systems * Copyright (C) 2010 University of Bonn. All rights reserved. * Please see the LICENSE file or "Copyright notice" in builder.cpp for details. */ /** \file linkedcell.cpp * * Function implementations for the class LinkedCell. * */ // include config.h #ifdef HAVE_CONFIG_H #include #endif #include "Helpers/MemDebug.hpp" #include "atom.hpp" #include "Helpers/helpers.hpp" #include "linkedcell.hpp" #include "Helpers/Verbose.hpp" #include "Helpers/Log.hpp" #include "molecule.hpp" #include "PointCloud.hpp" #include "tesselation.hpp" #include "LinearAlgebra/Vector.hpp" // ========================================================= class LinkedCell =========================================== /** Constructor for class LinkedCell::LinkedNodes. */ LinkedCell::LinkedNodes::LinkedNodes() {} /** Destructor for class LinkedCell::LinkedNodes. */ LinkedCell::LinkedNodes::~LinkedNodes() {} TesselPoint * LinkedCell::LinkedNodes::getValue (const_iterator &rhs) const { return *rhs; } TesselPoint * LinkedCell::LinkedNodes::getValue (iterator &rhs) const { return *rhs; } /** Constructor for class LinkedCell. */ LinkedCell::LinkedCell() : LC(NULL), RADIUS(0.), index(-1) { for(int i=0;iat(i); min[i] = Walker->at(i); } set.GoToFirst(); while (!set.IsEnd()) { Walker = set.GetPoint(); for (int i=0;iat(i)) max[i] = Walker->at(i); if (min[i] > Walker->at(i)) min[i] = Walker->at(i); } set.GoToNext(); } DoLog(2) && (Log() << Verbose(2) << "Bounding box is " << min << " and " << max << "." << endl); // 2. find then number of cells per axis for (int i=0;i(floor((max[i] - min[i])/RADIUS)+1); } DoLog(2) && (Log() << Verbose(2) << "Number of cells per axis are " << N[0] << ", " << N[1] << " and " << N[2] << "." << endl); // 3. allocate the lists DoLog(2) && (Log() << Verbose(2) << "Allocating cells ... "); if (LC != NULL) { DoeLog(1) && (eLog()<< Verbose(1) << "Linked Cell list is already allocated, I do nothing." << endl); return; } ASSERT(N[0]*N[1]*N[2] < MAX_LINKEDCELLNODES, "Number linked of linked cell nodes exceded hard-coded limit, use greater edge length!"); LC = new LinkedNodes[N[0]*N[1]*N[2]]; for (index=0;index(floor((Walker->at(i) - min[i])/RADIUS)); } index = n[0] * N[1] * N[2] + n[1] * N[2] + n[2]; LC[index].push_back(Walker); //Log() << Verbose(2) << *Walker << " goes into cell " << n[0] << ", " << n[1] << ", " << n[2] << " with No. " << index << "." << endl; set.GoToNext(); } DoLog(0) && (Log() << Verbose(0) << "done." << endl); DoLog(1) && (Log() << Verbose(1) << "End of LinkedCell" << endl); }; /** Destructor for class LinkedCell. */ LinkedCell::~LinkedCell() { if (LC != NULL) for (index=0;index=0) && (n[i] < N[i])); // if (!status) // DoeLog(1) && (eLog()<< Verbose(1) << "indices are out of bounds!" << endl); return status; }; /** Checks whether LinkedCell::n[] plus relative offset is each within [0,N[]]. * Note that for this check we don't admonish if out of bounds. * \param relative[NDIM] relative offset to current cell * \return if all in intervals - true, else -false */ bool LinkedCell::CheckBounds(const int relative[NDIM]) const { bool status = true; for(int i=0;i=0) && (n[i]+relative[i] < N[i])); return status; }; /** Returns a pointer to the current cell. * \return LinkedAtoms pointer to current cell, NULL if LinkedCell::n[] are out of bounds. */ const LinkedCell::LinkedNodes* LinkedCell::GetCurrentCell() const { if (CheckBounds()) { index = n[0] * N[1] * N[2] + n[1] * N[2] + n[2]; return (&(LC[index])); } else { return NULL; } }; /** Returns a pointer to the current cell. * \param relative[NDIM] offset for each axis with respect to the current cell LinkedCell::n[NDIM] * \return LinkedAtoms pointer to current cell, NULL if LinkedCell::n[]+relative[] are out of bounds. */ const LinkedCell::LinkedNodes* LinkedCell::GetRelativeToCurrentCell(const int relative[NDIM]) const { if (CheckBounds(relative)) { index = (n[0]+relative[0]) * N[1] * N[2] + (n[1]+relative[1]) * N[2] + (n[2]+relative[2]); return (&(LC[index])); } else { return NULL; } }; /** Set the index to the cell containing a given Vector *x. * \param *x Vector with coordinates * \return Vector is inside bounding box - true, else - false */ bool LinkedCell::SetIndexToVector(const Vector & x) const { for (int i=0;i(floor((Walker->at(i) - min[i])/RADIUS)); } index = n[0] * N[1] * N[2] + n[1] * N[2] + n[2]; if (CheckBounds()) { for (LinkedNodes::iterator Runner = LC[index].begin(); Runner != LC[index].end(); Runner++) status = status || ((*Runner) == Walker); return status; } else { DoeLog(1) && (eLog()<< Verbose(1) << "Node at " << *Walker << " is out of bounds." << endl); return false; } }; /** Calculates the interval bounds of the linked cell grid. * \param lower lower bounds * \param upper upper bounds * \param step how deep to check the neighbouring cells (i.e. number of layers to check) */ void LinkedCell::GetNeighbourBounds(int lower[NDIM], int upper[NDIM], int step) const { for (int i=0;i= N[i]) lower[i] = N[i]-1; upper[i] = n[i]+step; if (upper[i] >= N[i]) upper[i] = N[i]-1; if (upper[i] < 0) upper[i] = 0; //Log() << Verbose(0) << "axis " << i << " has bounds [" << lower[i] << "," << upper[i] << "]" << endl; } }; /** Returns a list with all neighbours from the current LinkedCell::index. * \param distance (if no distance, then adjacent cells are taken) * \return list of tesselpoints */ LinkedCell::LinkedNodes* LinkedCell::GetallNeighbours(const double distance) const { int Nlower[NDIM], Nupper[NDIM]; TesselPoint *Walker = NULL; LinkedNodes *TesselList = new LinkedNodes; // then go through the current and all neighbouring cells and check the contained points for possible candidates const int step = (distance == 0) ? 1 : (int)floor(distance/RADIUS + 1.); GetNeighbourBounds(Nlower, Nupper, step); //Log() << Verbose(0) << endl; for (n[0] = Nlower[0]; n[0] <= Nupper[0]; n[0]++) for (n[1] = Nlower[1]; n[1] <= Nupper[1]; n[1]++) for (n[2] = Nlower[2]; n[2] <= Nupper[2]; n[2]++) { const LinkedNodes *List = GetCurrentCell(); //Log() << Verbose(1) << "Current cell is " << n[0] << ", " << n[1] << ", " << n[2] << " with No. " << index << "." << endl; if (List != NULL) { for (LinkedNodes::const_iterator Runner = List->begin(); Runner != List->end(); Runner++) { Walker = *Runner; TesselList->push_back(Walker); } } } return TesselList; }; /** Set the index to the cell containing a given Vector *x, which is not inside the LinkedCell's domain * Note that as we have to check distance from every corner of the closest cell, this function is faw more * expensive and if Vector is known to be inside LinkedCell's domain, then SetIndexToVector() should be used. * \param *x Vector with coordinates * \return minimum squared distance of cell to given vector (if inside of domain, distance is 0) */ double LinkedCell::SetClosestIndexToOutsideVector(const Vector * const x) const { for (int i=0;iat(i) - min[i])/RADIUS); if (n[i] < 0) n[i] = 0; if (n[i] >= N[i]) n[i] = N[i]-1; } // calculate distance of cell to vector double distanceSquared = 0.; bool outside = true; // flag whether x is found in- or outside of LinkedCell's domain/closest cell Vector corner; // current corner of closest cell Vector tester; // Vector pointing from corner to center of closest cell Vector Distance; // Vector from corner of closest cell to x Vector center; // center of the closest cell for (int i=0;i 2.*radius) { DoeLog(1) && (eLog()<< Verbose(1) << "Vector " << *center << " is too far away from any atom in LinkedCell's bounding box." << endl); return TesselList; } else DoLog(1) && (Log() << Verbose(1) << "Distance of closest cell to center of sphere with radius " << radius << " is " << dist << "." << endl); // gather all neighbours first, then look who fulfills distance criteria NeighbourList = GetallNeighbours(2.*radius-dist); //Log() << Verbose(1) << "I found " << NeighbourList->size() << " neighbours to check." << endl; if (NeighbourList != NULL) { for (LinkedNodes::const_iterator Runner = NeighbourList->begin(); Runner != NeighbourList->end(); Runner++) { Walker = *Runner; //Log() << Verbose(1) << "Current neighbour is at " << *Walker->node << "." << endl; if ((Walker->DistanceSquared(*center) - radiusSquared) < MYEPSILON) { TesselList->push_back(Walker); } } delete(NeighbourList); } else DoeLog(2) && (eLog()<< Verbose(2) << "Around vector " << *center << " there are no atoms." << endl); return TesselList; };