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 double cellSizeFactor,
const size_t sortingThreshold)
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; }
157 void reserve(
size_t numParticles,
size_t numParticlesHaloEstimate)
override {
170 this->
_cells[CellTypes::HALO].addParticle(haloParticle);
183 std::tuple<const Particle_T *, size_t, size_t>
getParticle(
size_t cellIndex,
size_t particleIndex,
184 IteratorBehavior iteratorBehavior,
185 const std::array<double, 3> &boxMin,
186 const std::array<double, 3> &boxMax)
const override {
187 return getParticleImpl<true>(cellIndex, particleIndex, iteratorBehavior, boxMin, boxMax);
189 std::tuple<const Particle_T *, size_t, size_t>
getParticle(
size_t cellIndex,
size_t particleIndex,
190 IteratorBehavior iteratorBehavior)
const override {
192 constexpr std::array<double, 3> boxMin{std::numeric_limits<double>::lowest(), std::numeric_limits<double>::lowest(),
193 std::numeric_limits<double>::lowest()};
195 constexpr std::array<double, 3> boxMax{std::numeric_limits<double>::max(), std::numeric_limits<double>::max(),
196 std::numeric_limits<double>::max()};
197 return getParticleImpl<false>(cellIndex, particleIndex, iteratorBehavior, boxMin, boxMax);
217 template <
bool regionIter>
218 std::tuple<const ParticleType *, size_t, size_t>
getParticleImpl(
size_t cellIndex,
size_t particleIndex,
219 IteratorBehavior iteratorBehavior,
220 const std::array<double, 3> &boxMin,
221 const std::array<double, 3> &boxMax)
const {
222 using namespace autopas::utils::ArrayMath::literals;
226 return {
nullptr, 0, 0};
230 if (cellIndex == 0 and not(iteratorBehavior & IteratorBehavior::owned)) {
235 if (cellIndex < 10 and cellIndex > HALO) {
236 return {
nullptr, 0, 0};
239 std::array<double, 3> boxMinWithSafetyMargin = boxMin;
240 std::array<double, 3> boxMaxWithSafetyMargin = boxMax;
241 if constexpr (regionIter) {
247 std::vector<size_t> currentCellIndex{};
252 if (particleIndex >= currentCellPtr->
size() or
253 not containerIteratorUtils::particleFulfillsIteratorRequirements<regionIter>(
254 (*currentCellPtr)[particleIndex], iteratorBehavior, boxMin, boxMax)) {
256 std::tie(currentCellPtr, particleIndex) =
257 advanceIteratorIndices<regionIter>(currentCellIndex, currentCellPtr, particleIndex, iteratorBehavior, boxMin,
258 boxMax, boxMinWithSafetyMargin, boxMaxWithSafetyMargin);
262 if (currentCellPtr ==
nullptr) {
263 return {
nullptr, 0, 0};
266 const Particle_T *retPtr = &((*currentCellPtr)[particleIndex]);
269 if (currentCellIndex.empty()) {
274 for (
int i =
static_cast<int>(currentCellIndex.size()) - 1; i >= 0; --i) {
276 cellIndex += currentCellIndex[i];
280 return {retPtr, cellIndex, particleIndex};
294 std::vector<size_t> currentCellIndex;
296 currentCellIndex.reserve(10);
297 currentCellIndex.push_back(cellIndex % 10);
302 currentCellIndex.push_back(cellIndex % 10);
304 currentCell = currentCell->
getChild(currentCellIndex.back());
313 if (particle.isOwned()) {
314 return this->
_cells[CellTypes::OWNED].deleteParticle(particle);
315 }
else if (particle.isHalo()) {
316 return this->
_cells[CellTypes::HALO].deleteParticle(particle);
325 auto &particleVec = cell->_particles;
326 auto &particle = particleVec[particleIndex];
328 particle = particleVec.back();
329 particleVec.pop_back();
330 return particleIndex < particleVec.size();
337 IteratorBehavior behavior,
346 IteratorBehavior behavior,
348 nullptr)
const override {
356 const std::array<double, 3> &lowerCorner,
const std::array<double, 3> &higherCorner, IteratorBehavior behavior,
365 const std::array<double, 3> &lowerCorner,
const std::array<double, 3> &higherCorner, IteratorBehavior behavior,
367 nullptr)
const override {
375 using namespace autopas::utils::ArrayMath::literals;
378 const std::array<unsigned long, 3> dims = {1, 1, 1};
387 [[nodiscard]]
size_t size()
const override {
388 return this->
_cells[CellTypes::OWNED].size() + this->
_cells[CellTypes::HALO].size();
395 return this->
_cells[CellTypes::OWNED].getNumberOfParticles(behavior) +
396 this->
_cells[CellTypes::HALO].getNumberOfParticles(behavior);
403 throw std::runtime_error(
"[Octree.h]: This cell container (octree) contains only two cells");
405 return i == CellTypes::HALO;
410 throw std::runtime_error(
"[Octree.h]: This cell container (octree) contains only two cells");
412 return i == CellTypes::OWNED;
421 template <
typename Lambda>
422 void forEach(Lambda forEachLambda, IteratorBehavior behavior = IteratorBehavior::ownedOrHalo) {
423 if (behavior & IteratorBehavior::owned) this->
_cells[OWNED].forEach(forEachLambda);
424 if (behavior & IteratorBehavior::halo) this->
_cells[HALO].forEach(forEachLambda);
425 if (not(behavior & IteratorBehavior::ownedOrHalo))
437 template <
typename Lambda,
typename A>
438 void reduce(Lambda reduceLambda, A &result, IteratorBehavior behavior = IteratorBehavior::ownedOrHalo) {
439 if (behavior & IteratorBehavior::owned) this->
_cells[OWNED].reduce(reduceLambda, result);
440 if (behavior & IteratorBehavior::halo) this->
_cells[HALO].reduce(reduceLambda, result);
441 if (not(behavior & IteratorBehavior::ownedOrHalo))
448 template <
typename Lambda>
450 const std::array<double, 3> &higherCorner, IteratorBehavior behavior) {
451 if (behavior & IteratorBehavior::owned)
452 this->
_cells[OWNED].forEachInRegion(forEachLambda, lowerCorner, higherCorner);
453 if (behavior & IteratorBehavior::halo) this->
_cells[HALO].forEachInRegion(forEachLambda, lowerCorner, higherCorner);
454 if (not(behavior & IteratorBehavior::ownedOrHalo))
461 template <
typename Lambda,
typename A>
462 void reduceInRegion(Lambda reduceLambda, A &result,
const std::array<double, 3> &lowerCorner,
463 const std::array<double, 3> &higherCorner, IteratorBehavior behavior) {
464 if (behavior & IteratorBehavior::owned)
465 this->
_cells[OWNED].reduceInRegion(reduceLambda, result, lowerCorner, higherCorner);
466 if (behavior & IteratorBehavior::halo)
467 this->
_cells[HALO].reduceInRegion(reduceLambda, result, lowerCorner, higherCorner);
468 if (not(behavior & IteratorBehavior::ownedOrHalo))
488 template <
bool regionIter>
489 std::tuple<OctreeLeafNode<Particle_T> *,
size_t> advanceIteratorIndices(
491 size_t particleIndex, IteratorBehavior iteratorBehavior,
const std::array<double, 3> &boxMin,
492 const std::array<double, 3> &boxMax,
const std::array<double, 3> &boxMinWithSafetyMargin,
493 const std::array<double, 3> &boxMaxWithSafetyMargin)
const {
495 const size_t minLevel = 0;
505 bool isRelevant = cellPtr->
size() > 0;
506 if constexpr (regionIter) {
507 isRelevant =
utils::boxesOverlap(cellPtr->getBoxMin(), cellPtr->getBoxMax(), boxMinWithSafetyMargin,
508 boxMaxWithSafetyMargin);
520 bool forceJumpToNextCell =
false;
522 while (particleIndex >= currentCellInterfacePtr->
size() or forceJumpToNextCell) {
525 while (currentCellIndex.back() == 7) {
526 currentCellInterfacePtr = currentCellInterfacePtr->
getParent();
527 currentCellIndex.pop_back();
529 if (currentCellIndex.size() < minLevel or currentCellIndex.empty()) {
530 currentCellIndex.clear();
531 return {
nullptr, std::numeric_limits<
decltype(particleIndex)>::max()};
536 ++currentCellIndex.back();
538 if (currentCellIndex.size() == 1) {
540 if (currentCellIndex.back() > HALO) {
541 return {
nullptr, std::numeric_limits<
decltype(particleIndex)>::max()};
544 if ((iteratorBehavior & IteratorBehavior::halo)
546 currentCellInterfacePtr = this->
_cells[HALO].getRaw();
549 currentCellIndex.clear();
550 return {
nullptr, std::numeric_limits<
decltype(particleIndex)>::max()};
553 currentCellInterfacePtr = currentCellInterfacePtr->
getParent()->getChild(currentCellIndex.back());
556 if (not cellIsRelevant(currentCellInterfacePtr)) {
557 forceJumpToNextCell =
true;
561 forceJumpToNextCell =
false;
562 while (currentCellInterfacePtr->
hasChildren() and not forceJumpToNextCell) {
564 size_t firstRelevantChild = 0;
566 const auto *childToCheck = currentCellInterfacePtr->
getChild(firstRelevantChild);
567 while (not cellIsRelevant(childToCheck)) {
568 ++firstRelevantChild;
569 childToCheck = currentCellInterfacePtr->
getChild(firstRelevantChild);
570 if (firstRelevantChild > 7) {
574 forceJumpToNextCell =
true;
578 currentCellIndex.push_back(firstRelevantChild);
579 currentCellInterfacePtr = currentCellInterfacePtr->
getChild(firstRelevantChild);
587 if (currentLeafCellPtr ==
nullptr) {
591 }
while (not containerIteratorUtils::particleFulfillsIteratorRequirements<regionIter>(
592 (*currentLeafCellPtr)[particleIndex], iteratorBehavior, boxMin, boxMax));
593 return {currentLeafCellPtr, particleIndex};
The CellBasedParticleContainer class stores particles in some object and provides methods to iterate ...
Definition: CellBasedParticleContainer.h:25
const std::array< double, 3 > & getBoxMax() const final
Get the upper corner of the container without halo.
Definition: CellBasedParticleContainer.h:71
double getVerletSkin() const final
Returns the verlet Skin length.
Definition: CellBasedParticleContainer.h:96
void deleteAllParticles() override
Deletes all particles from the container.
Definition: CellBasedParticleContainer.h:101
const std::array< double, 3 > & getBoxMin() const final
Get the lower corner of the container without halo.
Definition: CellBasedParticleContainer.h:76
double getInteractionLength() const final
Return the interaction length (cutoff+skin) of the container.
Definition: CellBasedParticleContainer.h:91
std::vector< ParticleCellType > _cells
Vector of particle cells.
Definition: CellBasedParticleContainer.h:159
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:449
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:292
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:408
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:157
void rebuildNeighborLists(TraversalInterface *traversal) override
Rebuilds the neighbor lists for the next traversals.
Definition: Octree.h:181
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:401
void reduce(Lambda reduceLambda, A &result, IteratorBehavior behavior=IteratorBehavior::ownedOrHalo)
Reduce properties of particles as defined by a lambda function.
Definition: Octree.h:438
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:169
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:336
size_t getNumberOfParticles(IteratorBehavior behavior) const override
Get the number of particles with respect to the specified IteratorBehavior.
Definition: Octree.h:394
bool updateHaloParticle(const ParticleType &haloParticle) override
Update a halo particle of the container with the given haloParticle.
Definition: Octree.h:176
typename ParticleCellType::ParticleType ParticleType
The particle type used in this container.
Definition: Octree.h:51
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:422
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:183
size_t size() const override
Get the total number of particles saved in the container (owned + halo + dummy).
Definition: Octree.h:387
void deleteHaloParticles() override
Deletes all halo particles.
Definition: Octree.h:399
void addParticleImpl(const ParticleType &p) override
Adds a particle to the container.
Definition: Octree.h:164
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:374
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:218
Octree(const std::array< double, 3 > &boxMin, const std::array< double, 3 > &boxMax, const double cutoff, const double skin, const double cellSizeFactor, const size_t sortingThreshold)
Construct a new octree with two sub-octrees: One for the owned particles and one for the halo particl...
Definition: Octree.h:74
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:364
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:312
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:189
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:462
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:345
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:323
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:355
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
@ 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