39template <
class Particle_T>
74 Octree(
const std::array<double, 3> &boxMin,
const std::array<double, 3> &boxMax,
const double cutoff,
75 const double skin,
const unsigned int rebuildFrequency,
const double cellSizeFactor)
77 using namespace autopas::utils::ArrayMath::literals;
80 int unsigned treeSplitThreshold = 16;
89 auto haloBoxMin = boxMin - interactionLength;
90 auto haloBoxMax = boxMax + interactionLength;
100 [[nodiscard]] std::vector<ParticleType>
updateContainer(
bool keepNeighborListValid)
override {
102 std::vector<Particle_T> invalidParticles{};
104 if (keepNeighborListValid) {
114 std::vector<Particle_T *> particleRefs;
115 this->
_cells[CellTypes::OWNED].collectAllParticles(particleRefs);
116 std::vector<Particle_T> particles{};
117 particles.reserve(particleRefs.size());
119 for (
auto *p : particleRefs) {
123 }
else if (
utils::inBox(p->getR(), this->getBoxMin(), this->getBoxMax())) {
124 particles.push_back(*p);
126 invalidParticles.push_back(*p);
134 for (
auto &particle : particles) {
139 return invalidParticles;
144 traversalInterface->setCells(&this->
_cells);
155 [[nodiscard]] ContainerOption
getContainerType()
const override {
return ContainerOption::octree; }
165 void reserve(
size_t numParticles,
size_t numParticlesHaloEstimate)
override {
178 this->
_cells[CellTypes::HALO].addParticle(haloParticle);
191 std::tuple<const Particle_T *, size_t, size_t>
getParticle(
size_t cellIndex,
size_t particleIndex,
192 IteratorBehavior iteratorBehavior,
193 const std::array<double, 3> &boxMin,
194 const std::array<double, 3> &boxMax)
const override {
195 return getParticleImpl<true>(cellIndex, particleIndex, iteratorBehavior, boxMin, boxMax);
197 std::tuple<const Particle_T *, size_t, size_t>
getParticle(
size_t cellIndex,
size_t particleIndex,
198 IteratorBehavior iteratorBehavior)
const override {
200 constexpr std::array<double, 3> boxMin{std::numeric_limits<double>::lowest(), std::numeric_limits<double>::lowest(),
201 std::numeric_limits<double>::lowest()};
203 constexpr std::array<double, 3> boxMax{std::numeric_limits<double>::max(), std::numeric_limits<double>::max(),
204 std::numeric_limits<double>::max()};
205 return getParticleImpl<false>(cellIndex, particleIndex, iteratorBehavior, boxMin, boxMax);
225 template <
bool regionIter>
226 std::tuple<const ParticleType *, size_t, size_t>
getParticleImpl(
size_t cellIndex,
size_t particleIndex,
227 IteratorBehavior iteratorBehavior,
228 const std::array<double, 3> &boxMin,
229 const std::array<double, 3> &boxMax)
const {
230 using namespace autopas::utils::ArrayMath::literals;
234 return {
nullptr, 0, 0};
238 if (cellIndex == 0 and not(iteratorBehavior & IteratorBehavior::owned)) {
243 if (cellIndex < 10 and cellIndex > HALO) {
244 return {
nullptr, 0, 0};
247 std::array<double, 3> boxMinWithSafetyMargin = boxMin;
248 std::array<double, 3> boxMaxWithSafetyMargin = boxMax;
249 if constexpr (regionIter) {
255 std::vector<size_t> currentCellIndex{};
260 if (particleIndex >= currentCellPtr->
size() or
261 not containerIteratorUtils::particleFulfillsIteratorRequirements<regionIter>(
262 (*currentCellPtr)[particleIndex], iteratorBehavior, boxMin, boxMax)) {
264 std::tie(currentCellPtr, particleIndex) =
265 advanceIteratorIndices<regionIter>(currentCellIndex, currentCellPtr, particleIndex, iteratorBehavior, boxMin,
266 boxMax, boxMinWithSafetyMargin, boxMaxWithSafetyMargin);
270 if (currentCellPtr ==
nullptr) {
271 return {
nullptr, 0, 0};
274 const Particle_T *retPtr = &((*currentCellPtr)[particleIndex]);
277 if (currentCellIndex.empty()) {
282 for (
int i =
static_cast<int>(currentCellIndex.size()) - 1; i >= 0; --i) {
284 cellIndex += currentCellIndex[i];
288 return {retPtr, cellIndex, particleIndex};
302 std::vector<size_t> currentCellIndex;
304 currentCellIndex.reserve(10);
305 currentCellIndex.push_back(cellIndex % 10);
310 currentCellIndex.push_back(cellIndex % 10);
312 currentCell = currentCell->
getChild(currentCellIndex.back());
321 if (particle.isOwned()) {
322 return this->
_cells[CellTypes::OWNED].deleteParticle(particle);
323 }
else if (particle.isHalo()) {
324 return this->
_cells[CellTypes::HALO].deleteParticle(particle);
333 auto &particleVec = cell->_particles;
334 auto &particle = particleVec[particleIndex];
336 particle = particleVec.back();
337 particleVec.pop_back();
338 return particleIndex < particleVec.size();
345 IteratorBehavior behavior,
354 IteratorBehavior behavior,
356 nullptr)
const override {
364 const std::array<double, 3> &lowerCorner,
const std::array<double, 3> &higherCorner, IteratorBehavior behavior,
373 const std::array<double, 3> &lowerCorner,
const std::array<double, 3> &higherCorner, IteratorBehavior behavior,
375 nullptr)
const override {
383 using namespace autopas::utils::ArrayMath::literals;
386 const std::array<unsigned long, 3> dims = {1, 1, 1};
395 [[nodiscard]]
size_t size()
const override {
396 return this->
_cells[CellTypes::OWNED].size() + this->
_cells[CellTypes::HALO].size();
403 return this->
_cells[CellTypes::OWNED].getNumberOfParticles(behavior) +
404 this->
_cells[CellTypes::HALO].getNumberOfParticles(behavior);
411 throw std::runtime_error(
"[Octree.h]: This cell container (octree) contains only two cells");
413 return i == CellTypes::HALO;
418 throw std::runtime_error(
"[Octree.h]: This cell container (octree) contains only two cells");
420 return i == CellTypes::OWNED;
429 template <
typename Lambda>
430 void forEach(Lambda forEachLambda, IteratorBehavior behavior = IteratorBehavior::ownedOrHalo) {
431 if (behavior & IteratorBehavior::owned) this->
_cells[OWNED].forEach(forEachLambda);
432 if (behavior & IteratorBehavior::halo) this->
_cells[HALO].forEach(forEachLambda);
433 if (not(behavior & IteratorBehavior::ownedOrHalo))
445 template <
typename Lambda,
typename A>
446 void reduce(Lambda reduceLambda, A &result, IteratorBehavior behavior = IteratorBehavior::ownedOrHalo) {
447 if (behavior & IteratorBehavior::owned) this->
_cells[OWNED].reduce(reduceLambda, result);
448 if (behavior & IteratorBehavior::halo) this->
_cells[HALO].reduce(reduceLambda, result);
449 if (not(behavior & IteratorBehavior::ownedOrHalo))
456 template <
typename Lambda>
458 const std::array<double, 3> &higherCorner, IteratorBehavior behavior) {
459 if (behavior & IteratorBehavior::owned)
460 this->
_cells[OWNED].forEachInRegion(forEachLambda, lowerCorner, higherCorner);
461 if (behavior & IteratorBehavior::halo) this->
_cells[HALO].forEachInRegion(forEachLambda, lowerCorner, higherCorner);
462 if (not(behavior & IteratorBehavior::ownedOrHalo))
469 template <
typename Lambda,
typename A>
470 void reduceInRegion(Lambda reduceLambda, A &result,
const std::array<double, 3> &lowerCorner,
471 const std::array<double, 3> &higherCorner, IteratorBehavior behavior) {
472 if (behavior & IteratorBehavior::owned)
473 this->
_cells[OWNED].reduceInRegion(reduceLambda, result, lowerCorner, higherCorner);
474 if (behavior & IteratorBehavior::halo)
475 this->
_cells[HALO].reduceInRegion(reduceLambda, result, lowerCorner, higherCorner);
476 if (not(behavior & IteratorBehavior::ownedOrHalo))
496 template <
bool regionIter>
497 std::tuple<OctreeLeafNode<Particle_T> *,
size_t> advanceIteratorIndices(
499 size_t particleIndex, IteratorBehavior iteratorBehavior,
const std::array<double, 3> &boxMin,
500 const std::array<double, 3> &boxMax,
const std::array<double, 3> &boxMinWithSafetyMargin,
501 const std::array<double, 3> &boxMaxWithSafetyMargin)
const {
503 const size_t minLevel = 0;
513 bool isRelevant = cellPtr->
size() > 0;
514 if constexpr (regionIter) {
515 isRelevant =
utils::boxesOverlap(cellPtr->getBoxMin(), cellPtr->getBoxMax(), boxMinWithSafetyMargin,
516 boxMaxWithSafetyMargin);
528 bool forceJumpToNextCell =
false;
530 while (particleIndex >= currentCellInterfacePtr->
size() or forceJumpToNextCell) {
533 while (currentCellIndex.back() == 7) {
534 currentCellInterfacePtr = currentCellInterfacePtr->
getParent();
535 currentCellIndex.pop_back();
537 if (currentCellIndex.size() < minLevel or currentCellIndex.empty()) {
538 currentCellIndex.clear();
539 return {
nullptr, std::numeric_limits<
decltype(particleIndex)>::max()};
544 ++currentCellIndex.back();
546 if (currentCellIndex.size() == 1) {
548 if (currentCellIndex.back() > HALO) {
549 return {
nullptr, std::numeric_limits<
decltype(particleIndex)>::max()};
552 if ((iteratorBehavior & IteratorBehavior::halo)
554 currentCellInterfacePtr = this->
_cells[HALO].getRaw();
557 currentCellIndex.clear();
558 return {
nullptr, std::numeric_limits<
decltype(particleIndex)>::max()};
561 currentCellInterfacePtr = currentCellInterfacePtr->
getParent()->getChild(currentCellIndex.back());
564 if (not cellIsRelevant(currentCellInterfacePtr)) {
565 forceJumpToNextCell =
true;
569 forceJumpToNextCell =
false;
570 while (currentCellInterfacePtr->
hasChildren() and not forceJumpToNextCell) {
572 size_t firstRelevantChild = 0;
574 const auto *childToCheck = currentCellInterfacePtr->
getChild(firstRelevantChild);
575 while (not cellIsRelevant(childToCheck)) {
576 ++firstRelevantChild;
577 childToCheck = currentCellInterfacePtr->
getChild(firstRelevantChild);
578 if (firstRelevantChild > 7) {
582 forceJumpToNextCell =
true;
586 currentCellIndex.push_back(firstRelevantChild);
587 currentCellInterfacePtr = currentCellInterfacePtr->
getChild(firstRelevantChild);
595 if (currentLeafCellPtr ==
nullptr) {
599 }
while (not containerIteratorUtils::particleFulfillsIteratorRequirements<regionIter>(
600 (*currentLeafCellPtr)[particleIndex], iteratorBehavior, boxMin, boxMax));
601 return {currentLeafCellPtr, particleIndex};
The CellBasedParticleContainer class stores particles in some object and provides methods to iterate ...
Definition: CellBasedParticleContainer.h:25
double getVerletSkin() const final
Returns the verlet Skin length.
Definition: CellBasedParticleContainer.h:94
double getInteractionLength() const final
Return the interaction length (cutoff+skin) of the container.
Definition: CellBasedParticleContainer.h:89
std::vector< OctreeNodeWrapper< Particle_T > > _cells
Vector of particle cells.
Definition: CellBasedParticleContainer.h:157
const std::array< double, 3 > & getBoxMin() const final
Get the lower corner of the container without halo.
Definition: CellBasedParticleContainer.h:74
const std::array< double, 3 > & getBoxMax() const final
Get the upper corner of the container without halo.
Definition: CellBasedParticleContainer.h:69
void deleteAllParticles() override
Deletes all particles from the container.
Definition: CellBasedParticleContainer.h:99
Public iterator class that iterates over a particle container and additional vectors (which are typic...
Definition: ContainerIterator.h:93
std::conditional_t< modifiable, std::vector< std::vector< Particle_T > * >, std::vector< std::vector< Particle_T > const * > > ParticleVecType
Type of the additional vector collection.
Definition: ContainerIterator.h:106
This interface exists to provide a row interface for octree to add its cells.
Definition: OTTraversalInterface.h:22
An octree leaf node.
Definition: OctreeLeafNode.h:27
size_t size() const override
Get the total number of particles saved in the container (owned + halo + dummy).
Definition: OctreeLeafNode.h:147
Log an octree to a .vtk file.
Definition: OctreeLogger.h:25
The base class that provides the necessary function definitions that can be applied to an octree.
Definition: OctreeNodeInterface.h:32
virtual OctreeNodeInterface< Particle_T > * getChild(int index)=0
Get a child by its index from the node.
OctreeNodeInterface< Particle_T > * getParent() const
Get the parent node of this node.
Definition: OctreeNodeInterface.h:376
virtual size_t size() const =0
Get the total number of particles saved in the container (owned + halo + dummy).
virtual bool hasChildren()=0
Check if the node is a leaf or an inner node.
This class wraps the functionality provided by the octree leaves and inner nodes in a structure that ...
Definition: OctreeNodeWrapper.h:37
typename ParticleCell::ParticleType ParticleType
The contained particle type.
Definition: OctreeNodeWrapper.h:46
The octree is a CellBasedParticleContainer that consists internally of two octrees.
Definition: Octree.h:41
static constexpr size_t invalidCellIndex
A cell index that is definitely always invalid.
Definition: Octree.h:63
void forEachInRegion(Lambda forEachLambda, const std::array< double, 3 > &lowerCorner, const std::array< double, 3 > &higherCorner, IteratorBehavior behavior)
Execute code on all particles in this container in a certain region as defined by a lambda function.
Definition: Octree.h:457
CellType getParticleCellTypeEnum() const override
Get the ParticleCell type as an Enum.
Definition: Octree.h:163
std::tuple< std::vector< size_t >, OctreeLeafNode< Particle_T > * > getLeafCellByIndex(size_t cellIndex) const
Helper function to retrieve the pointer to a leaf cell as well as properly parse the cell index.
Definition: Octree.h:300
bool cellCanContainOwnedParticles(std::size_t i) const override
Checks if the cell with the one-dimensional index index1d can contain owned particles.
Definition: Octree.h:416
Octree(const std::array< double, 3 > &boxMin, const std::array< double, 3 > &boxMax, const double cutoff, const double skin, const unsigned int rebuildFrequency, const double cellSizeFactor)
Construct a new octree with two sub-octrees: One for the owned particles and one for the halo particl...
Definition: Octree.h:74
void reserve(size_t numParticles, size_t numParticlesHaloEstimate) override
Reserve memory for a given number of particles in the container and logic layers.
Definition: Octree.h:165
void rebuildNeighborLists(TraversalInterface *traversal) override
Rebuilds the neighbor lists for the next traversals.
Definition: Octree.h:189
bool cellCanContainHaloParticles(std::size_t i) const override
Checks if the cell with the one-dimensional index index1d can contain halo particles.
Definition: Octree.h:409
void reduce(Lambda reduceLambda, A &result, IteratorBehavior behavior=IteratorBehavior::ownedOrHalo)
Reduce properties of particles as defined by a lambda function.
Definition: Octree.h:446
CellTypes
This particle container contains two cells.
Definition: Octree.h:58
void computeInteractions(TraversalInterface *traversal) override
Iterates over all particle multiples (e.g.
Definition: Octree.h:142
void addHaloParticleImpl(const ParticleType &haloParticle) override
Adds a particle to the container that lies in the halo region of the container.
Definition: Octree.h:177
ContainerIterator< Particle_T, true, false > begin(IteratorBehavior behavior, typename ContainerIterator< Particle_T, true, false >::ParticleVecType *additionalVectors=nullptr) override
Iterate over all particles using for(auto iter = container.begin(); iter.isValid(); ++iter) .
Definition: Octree.h:344
size_t getNumberOfParticles(IteratorBehavior behavior) const override
Get the number of particles with respect to the specified IteratorBehavior.
Definition: Octree.h:402
bool updateHaloParticle(const ParticleType &haloParticle) override
Update a halo particle of the container with the given haloParticle.
Definition: Octree.h:184
void forEach(Lambda forEachLambda, IteratorBehavior behavior=IteratorBehavior::ownedOrHalo)
Execute code on all particles in this container as defined by a lambda function.
Definition: Octree.h:430
std::tuple< const Particle_T *, size_t, size_t > getParticle(size_t cellIndex, size_t particleIndex, IteratorBehavior iteratorBehavior, const std::array< double, 3 > &boxMin, const std::array< double, 3 > &boxMax) const override
Fetch the pointer to a particle, identified via a cell and particle index.
Definition: Octree.h:191
size_t size() const override
Get the total number of particles saved in the container (owned + halo + dummy).
Definition: Octree.h:395
void deleteHaloParticles() override
Deletes all halo particles.
Definition: Octree.h:407
void addParticleImpl(const ParticleType &p) override
Adds a particle to the container.
Definition: Octree.h:172
std::vector< ParticleType > updateContainer(bool keepNeighborListValid) override
Updates the container.
Definition: Octree.h:100
TraversalSelectorInfo getTraversalSelectorInfo() const override
Generates a traversal selector info for this container.
Definition: Octree.h:382
std::tuple< const ParticleType *, size_t, size_t > getParticleImpl(size_t cellIndex, size_t particleIndex, IteratorBehavior iteratorBehavior, const std::array< double, 3 > &boxMin, const std::array< double, 3 > &boxMax) const
Container specific implementation for getParticle.
Definition: Octree.h:226
ContainerIterator< Particle_T, false, true > getRegionIterator(const std::array< double, 3 > &lowerCorner, const std::array< double, 3 > &higherCorner, IteratorBehavior behavior, typename ContainerIterator< Particle_T, false, true >::ParticleVecType *additionalVectors=nullptr) const override
Iterate over all particles in a specified region for(auto iter = container.getRegionIterator(lowCorne...
Definition: Octree.h:372
bool deleteParticle(Particle_T &particle) override
Deletes the given particle as long as this does not compromise the validity of the container.
Definition: Octree.h:320
std::tuple< const Particle_T *, size_t, size_t > getParticle(size_t cellIndex, size_t particleIndex, IteratorBehavior iteratorBehavior) const override
Fetch the pointer to a particle, identified via a cell and particle index.
Definition: Octree.h:197
typename ParticleCell::ParticleType ParticleType
The particle type used in this container.
Definition: Octree.h:51
void reduceInRegion(Lambda reduceLambda, A &result, const std::array< double, 3 > &lowerCorner, const std::array< double, 3 > &higherCorner, IteratorBehavior behavior)
Execute code on all particles in this container in a certain region as defined by a lambda function.
Definition: Octree.h:470
ContainerIterator< Particle_T, false, false > begin(IteratorBehavior behavior, typename ContainerIterator< Particle_T, false, false >::ParticleVecType *additionalVectors=nullptr) const override
Iterate over all particles using for(auto iter = container.begin(); iter.isValid(); ++iter) .
Definition: Octree.h:353
ContainerOption getContainerType() const override
Get the ContainerType.
Definition: Octree.h:155
bool deleteParticle(size_t cellIndex, size_t particleIndex) override
Deletes the particle at the given index positions as long as this does not compromise the validity of...
Definition: Octree.h:331
ContainerIterator< Particle_T, true, true > getRegionIterator(const std::array< double, 3 > &lowerCorner, const std::array< double, 3 > &higherCorner, IteratorBehavior behavior, typename ContainerIterator< Particle_T, true, true >::ParticleVecType *additionalVectors=nullptr) override
Iterate over all particles in a specified region for(auto iter = container.getRegionIterator(lowCorne...
Definition: Octree.h:363
This interface serves as a common parent class for all traversals.
Definition: TraversalInterface.h:18
virtual void endTraversal()=0
Finalizes the traversal.
virtual void traverseParticles()=0
Traverse the particles by pairs, triplets etc.
virtual void initTraversal()=0
Initializes the traversal.
Info for traversals of a specific container.
Definition: TraversalSelectorInfo.h:14
Interface class to handle cell borders and cell types of cells.
Definition: CellBorderAndFlagManager.h:17
static void exception(const Exception e)
Handle an exception derived by std::exception.
Definition: ExceptionHandler.h:63
std::vector< typename ContainerType::ParticleType > collectParticlesAndMarkNonOwnedAsDummy(ContainerType &container)
Collects leaving particles and marks halo particles as dummy.
Definition: LeavingParticleCollector.h:85
static bool checkParticleInCellAndUpdateByIDAndPosition(CellType &cell, const typename CellType::ParticleType &particle, double absError)
Same as checkParticleInCellAndUpdateByID(CellType, ParticleType), but additionally checks whether the...
Definition: ParticleCellHelpers.h:39
bool boxesOverlap(const std::array< T, 3 > &boxALow, const std::array< T, 3 > &boxAHigh, const std::array< T, 3 > &boxBLow, const std::array< T, 3 > &boxBHigh)
Checks if two boxes have overlap.
Definition: inBox.h:67
bool inBox(const std::array< T, 3 > &position, const std::array< T, 3 > &low, const std::array< T, 3 > &high)
Checks if position is inside of a box defined by low and high.
Definition: inBox.h:26
This is the main namespace of AutoPas.
Definition: AutoPasDecl.h:32
CellType
The ParticleCell Type as an Enum.
Definition: ParticleCell.h:19
@ FullParticleCell
FullParticleCell : Default cell type for almost everything.
@ halo
Halo state, a particle with this state is an actual particle, but not owned by the current AutoPas ob...
@ owned
Owned state, a particle with this state is an actual particle and owned by the current AutoPas object...
int autopas_get_thread_num()
Dummy for omp_set_lock() when no OpenMP is available.
Definition: WrapOpenMP.h:132