MADNESS 0.10.1
displacements.h
Go to the documentation of this file.
1/*
2 This file is part of MADNESS.
3
4 Copyright (C) 2007,2010 Oak Ridge National Laboratory
5
6 This program is free software; you can redistribute it and/or modify
7 it under the terms of the GNU General Public License as published by
8 the Free Software Foundation; either version 2 of the License, or
9 (at your option) any later version.
10
11 This program is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
15
16 You should have received a copy of the GNU General Public License
17 along with this program; if not, write to the Free Software
18 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
19
20 For more information please contact:
21
22 Robert J. Harrison
23 Oak Ridge National Laboratory
24 One Bethel Valley Road
25 P.O. Box 2008, MS-6367
26
27 email: harrisonrj@ornl.gov
28 tel: 865-241-3937
29 fax: 865-572-0680
30
31 $Id$
32*/
33#ifndef MADNESS_MRA_DISPLACEMENTS_H__INCLUDED
34#define MADNESS_MRA_DISPLACEMENTS_H__INCLUDED
35
36#include <madness/mra/indexit.h>
38
39#include <algorithm>
40#include <array>
41#include <functional>
42#include <iterator>
43#include <optional>
44#include <tuple>
45#include <utility>
46#include <vector>
47
48namespace madness {
49 /// Holds displacements for applying operators to avoid replicating for all operators
50 template <std::size_t NDIM>
52
53 static std::vector< Key<NDIM> > disp; ///< standard displacements to be used with standard kernels (range-unrestricted, no lattice sum)
54 static array_of_bools<NDIM> periodic_axes; ///< along which axes lattice summation is performed?
55 static std::vector< Key<NDIM> > disp_periodic[64]; ///< displacements to be used with lattice-summed kernels
56
57 public:
58 static int bmax_default() {
59 int bmax;
60 if (NDIM == 1) bmax = 7;
61 else if (NDIM == 2) bmax = 5;
62 else if (NDIM == 3) bmax = 3;
63 else if (NDIM == 4) bmax = 3;
64 else if (NDIM == 5) bmax = 3;
65 else if (NDIM == 6) bmax = 3;
66 else bmax = 2;
67 return bmax;
68 }
69
70 private:
71 static bool cmp_keys(const Key<NDIM>& a, const Key<NDIM>& b) {
72 return a.distsq() < b.distsq();
73 }
74
75 static bool cmp_keys_periodic(const Key<NDIM>& a, const Key<NDIM>& b) {
76 return a.distsq_bc(periodic_axes) < b.distsq_bc(periodic_axes);
77 }
78
79 static void make_disp(int bmax) {
80 // Note newer loop structure in make_disp_periodic_sum
82
83 int num = 1;
84 for (std::size_t i=0; i<NDIM; ++i) num *= (2*bmax + 1);
85 disp.resize(num,Key<NDIM>(0));
86
87 num = 0;
88 if (NDIM == 1) {
89 for (d[0]=-bmax; d[0]<=bmax; ++d[0])
90 disp[num++] = Key<NDIM>(0,d);
91 }
92 else if (NDIM == 2) {
93 for (d[0]=-bmax; d[0]<=bmax; ++d[0])
94 for (d[1]=-bmax; d[1]<=bmax; ++d[1])
95 disp[num++] = Key<NDIM>(0,d);
96 }
97 else if (NDIM == 3) {
98 for (d[0]=-bmax; d[0]<=bmax; ++d[0])
99 for (d[1]=-bmax; d[1]<=bmax; ++d[1])
100 for (d[2]=-bmax; d[2]<=bmax; ++d[2])
101 disp[num++] = Key<NDIM>(0,d);
102 }
103 else if (NDIM == 4) {
104 for (d[0]=-bmax; d[0]<=bmax; ++d[0])
105 for (d[1]=-bmax; d[1]<=bmax; ++d[1])
106 for (d[2]=-bmax; d[2]<=bmax; ++d[2])
107 for (d[3]=-bmax; d[3]<=bmax; ++d[3])
108 disp[num++] = Key<NDIM>(0,d);
109 }
110 else if (NDIM == 5) {
111 for (d[0]=-bmax; d[0]<=bmax; ++d[0])
112 for (d[1]=-bmax; d[1]<=bmax; ++d[1])
113 for (d[2]=-bmax; d[2]<=bmax; ++d[2])
114 for (d[3]=-bmax; d[3]<=bmax; ++d[3])
115 for (d[4]=-bmax; d[4]<=bmax; ++d[4])
116
117 disp[num++] = Key<NDIM>(0,d);
118 }
119 else if (NDIM == 6) {
120 for (d[0]=-bmax; d[0]<=bmax; ++d[0])
121 for (d[1]=-bmax; d[1]<=bmax; ++d[1])
122 for (d[2]=-bmax; d[2]<=bmax; ++d[2])
123 for (d[3]=-bmax; d[3]<=bmax; ++d[3])
124 for (d[4]=-bmax; d[4]<=bmax; ++d[4])
125 for (d[5]=-bmax; d[5]<=bmax; ++d[5])
126 disp[num++] = Key<NDIM>(0,d);
127 }
128 else {
129 MADNESS_EXCEPTION("make_disp: hard dimension loop",NDIM);
130 }
131
132 std::sort(disp.begin(), disp.end(), cmp_keys);
133 }
134
135 static void make_disp_periodic(int bmax, Level n) {
136 MADNESS_ASSERT(periodic_axes.any()); // else use make_disp
137 Translation twon = Translation(1)<<n;
138
139 if (bmax > (twon-1)) bmax=twon-1;
140
141 // Make permissible 1D translations, periodic and nonperiodic (for mixed BC)
142 Translation bp[4*bmax+1];
143 Translation bnp[2*bmax+1];
144 int ip=0;
145 int inp=0;
146 for (Translation lx=-bmax; lx<=bmax; ++lx) {
147 bp[ip++] = lx;
148 if ((lx < 0) && (lx+twon > bmax)) bp[ip++] = lx + twon;
149 if ((lx > 0) && (lx-twon <-bmax)) bp[ip++] = lx - twon;
150 bnp[inp++] = lx;
151 }
152 MADNESS_ASSERT(ip <= 4*bmax+1);
153 MADNESS_ASSERT(inp <= 2*bmax+1);
154 const int nbp = ip;
155 const int nbnp = inp;
156
157 MADNESS_PRAGMA_CLANG(diagnostic push)
158 MADNESS_PRAGMA_CLANG(diagnostic ignored "-Wundefined-var-template")
159
160 disp_periodic[n] = std::vector< Key<NDIM> >();
162 for(int i=0; i!=NDIM; ++i) {
163 lim[i] = periodic_axes[i] ? nbp : nbnp;
164 }
165 for (IndexIterator index(lim); index; ++index) {
167 for (std::size_t i=0; i<NDIM; ++i) {
168 d[i] = periodic_axes[i] ? bp[index[i]] : bnp[index[i]];
169 }
170 disp_periodic[n].push_back(Key<NDIM>(n,d));
171 }
172
173 std::sort(disp_periodic[n].begin(), disp_periodic[n].end(), cmp_keys_periodic);
174// print("KEYS AT LEVEL", n);
175// print(disp_periodic[n]);
176
177 MADNESS_PRAGMA_CLANG(diagnostic pop)
178
179 }
180
181
182 public:
183 /// first time this is called displacements are generated.
184 /// if boundary conditions are not periodic, the periodic displacements
185 /// are generated for all axes. This allows to support application of
186 /// operators with boundary conditions periodic along any axis (including all).
187 /// If need to use periodic boundary conditions
188 /// for some axes only, make sure to set the boundary conditions appropriately
189 /// before the first call to this
191 MADNESS_PRAGMA_CLANG(diagnostic push)
192 MADNESS_PRAGMA_CLANG(diagnostic ignored "-Wundefined-var-template")
193
194 if (disp.empty()) {
196 }
197
198 if constexpr (NDIM <= 3) {
199 if (disp_periodic[0].empty()) { // if not initialized yet
200 if (FunctionDefaults<NDIM>::get_bc().is_periodic().any())
202 FunctionDefaults<NDIM>::get_bc().is_periodic());
203 else
205 }
206 }
207
208 MADNESS_PRAGMA_CLANG(diagnostic pop)
209 }
210
211 const std::vector< Key<NDIM> >& get_disp(Level n,
212 const array_of_bools<NDIM>& kernel_lattice_sum_axes) {
213 MADNESS_PRAGMA_CLANG(diagnostic push)
214 MADNESS_PRAGMA_CLANG(diagnostic ignored "-Wundefined-var-template")
215
216 if (kernel_lattice_sum_axes.any()) {
217 MADNESS_ASSERT(NDIM <= 3);
218 MADNESS_ASSERT(n < std::extent_v<decltype(disp_periodic)>);
219 if ((kernel_lattice_sum_axes && periodic_axes) != kernel_lattice_sum_axes) {
220 std::string msg =
221 "Displacements<" + std::to_string(NDIM) +
222 ">::get_disp(level, kernel_lattice_sum_axes): kernel_lattice_sum_axes is set for some axes that were not periodic in the FunctionDefault's boundary conditions active at the time when Displacements were initialized; invoke Displacements<NDIM>::reset_periodic_axes(kernel_lattice_sum_axes) to rebuild the periodic displacements";
223 MADNESS_EXCEPTION(msg.c_str(), 1);
224 }
225 return disp_periodic[n];
226 }
227 else {
228 return disp;
229 }
230
231 MADNESS_PRAGMA_CLANG(diagnostic pop)
232 }
233
234 /// return the standard displacements appropriate for operators w/o lattice summation
235 const std::vector< Key<NDIM> >& get_disp() {
236 MADNESS_PRAGMA_CLANG(diagnostic push)
237 MADNESS_PRAGMA_CLANG(diagnostic ignored "-Wundefined-var-template")
238
239 return disp;
240
241 MADNESS_PRAGMA_CLANG(diagnostic pop)
242 }
243
244 /// rebuilds periodic displacements so that they are optimal for the given set of periodic axes
245
246 /// this must be done while no references to prior periodic displacements are outstanding (i.e. no operator application
247 /// tasks in flight)
248 /// \param new_periodic_axes the new periodic axes
249 static void reset_periodic_axes(const array_of_bools<NDIM>& new_periodic_axes) {
250 MADNESS_PRAGMA_CLANG(diagnostic push)
251 MADNESS_PRAGMA_CLANG(diagnostic ignored "-Wundefined-var-template")
252
253 MADNESS_ASSERT(new_periodic_axes.any()); // else why call this?
254 if (new_periodic_axes != periodic_axes) {
255
256 periodic_axes = new_periodic_axes;
257 Level nmax = 8 * sizeof(Translation) - 2;
258 for (Level n = 0; n < nmax; ++n)
260 }
261 MADNESS_PRAGMA_CLANG(diagnostic pop)
262 }
263 };
264
265 template <std::size_t N, std::size_t M>
266 constexpr std::enable_if_t<N>=M, std::array<std::size_t, N-M>> iota_array(std::array<std::size_t, M> values_to_skip_sorted) {
267 std::array<std::size_t, N - M> result;
268 if constexpr (N != M) {
269 std::size_t nadded = 0;
270 auto value_to_skip_it = values_to_skip_sorted.begin();
271 assert(*value_to_skip_it < N);
272 auto value_to_skip = *value_to_skip_it;
273 for (std::size_t i = 0; i < N; ++i) {
274 if (i < value_to_skip) {
275 result[nadded++] = i;
276 } else if (value_to_skip_it != values_to_skip_sorted.end()) {
277 ++value_to_skip_it;
278 if (value_to_skip_it != values_to_skip_sorted.end()) {
279 value_to_skip = *value_to_skip_it;
280 } else
281 value_to_skip = N;
282 }
283 }
284 }
285 return result;
286 }
287
288 /**
289 * Generates points at the finite-thickness surface of an N-dimensional box [C1-L1,C1+L1]x...x[CN-LN,CN+LN] centered at point {C1,...CN} in Z^N.
290 * For finite thickness T={T1,...,TN} point {x1,...,xN} is at the surface face perpendicular to axis i xi>=Ci-Li-Ti and xi<=Ci-Li+Ti OR xi>=Ci+Li-Ti and xi<=Ci+Li+Ti.
291 * For dimensions with unlimited size the point coordinates are limited to [0,2^n], with n being the level of the box.
292 */
293 template<std::size_t NDIM>
295 public:
299 /// this callable filters out points and/or displacements; note that the displacement is optional (this use case supports filtering based on point pattern onlu) and non-const to make it possible for the filter function to update the displacement (e.g. to map it back to the simulation cell)
300 using Filter = std::function<bool(Level, const PointPattern&, std::optional<Displacement>&)>;
301
302 private:
303 using BoxRadius = std::array<std::optional<Translation>, NDIM>; // null radius = unlimited size
304 using SurfaceThickness = std::array<std::optional<Translation>, NDIM>; // null thickness for dimensions with null radius
305 using Box = std::array<std::pair<Translation, Translation>, NDIM>;
306 using Hollowness = std::array<bool, NDIM>; // this can be uninitialized, unlike array_of_bools
308
309 Point center_; ///< Center point of the box
310 BoxRadius box_radius_; ///< halved size of the box in each dimension
312 surface_thickness_; ///< surface thickness in each dimension
313 Box box_; ///< box bounds in each dimension
314 Hollowness hollowness_; ///< does box contain non-surface points along each dimension?
315 Periodicity is_periodic_; ///< which dimensions are periodic?
316 Filter filter_; ///< optional filter function
317 Displacement probing_displacement_; ///< displacement to a nearby point on the surface; it may not be able to pass the filter, but is sufficiently representative of the surface displacements to allow screening with isotropic kernels
318
319 /**
320 * @brief Iterator class for lazy generation of surface points
321 *
322 * This iterator generates surface points on-demand by tracking the current fixed
323 * dimension and positions in each dimension. It implements the InputIterator concept.
324 */
325 class Iterator {
326 public:
327 enum Type {Begin, End};
328 private:
329 const BoxSurfaceDisplacementRange* parent; ///< Pointer to parent box
330 Point point; ///< Current point
331 mutable std::optional<Displacement> disp; ///< Memoized displacement from parent->center_ to point, computed by displacement(), reset by advance()
332 size_t fixed_dim; ///< Current fixed dimension (i.e. faces perpendicular to this axis are being iterated over)
333 Box box; ///< updated box bounds in each dimension, used to avoid duplicate displacements by excluding the surface displacements for each processed fixed dim
334 bool done; ///< Flag indicating iteration completion
335
336 // return true if we have another surface layer
338 Vector<Translation, NDIM> l = point.translation();
339 if (l[fixed_dim] !=
340 parent->box_[fixed_dim].second +
341 parent->surface_thickness_[fixed_dim].value_or(0)) {
342 // if box is hollow along this dim (has 2 surface layers) and exhausted all layers on the "negative" side of the fixed dimension, move to the first layer on the "positive" side
344 l[fixed_dim] ==
345 parent->box_[fixed_dim].first +
346 parent->surface_thickness_[fixed_dim].value_or(0)) {
347 l[fixed_dim] =
348 parent->box_[fixed_dim].second -
349 parent->surface_thickness_[fixed_dim].value_or(0);
350 } else
351 ++l[fixed_dim];
352 point = Point(point.level(), l);
353 return true;
354 } else
355 return false;
356 };
357
358 /**
359 * @brief Advances the iterator to the next surface point
360 *
361 * This function implements the logic for traversing the box surface by:
362 * 1. Incrementing displacement in non-fixed dimensions
363 * 2. Switching sides in the fixed dimension when needed
364 * 3. Moving to the next fixed dimension when current one is exhausted
365 */
366 void advance() {
367 disp.reset();
368
369 auto increment_along_dim = [this](size_t dim) {
371 Vector<Translation, NDIM> unit_displacement(0); unit_displacement[dim] = 1;
372 point = point.neighbor(unit_displacement);
373 };
374
375 for (int64_t i = NDIM - 1; i >= 0; --i) {
376 if (i == fixed_dim) continue;
377
378 if (point[i] < box[i].second) {
379 increment_along_dim(i);
380 return;
381 }
383 }
384
385 // move to the next surface layer normal to the fixed dimension
386 while (bool have_another_surface_layer = next_surface_layer()) {
387 const auto filtered_out = [&,this]() {
388 bool result = false;
389 const auto& filter = this->parent->filter_;
390 if (filter) {
391 PointPattern point_pattern;
392 point_pattern[fixed_dim] = point[fixed_dim];
393 std::optional<Displacement> nulldisp;
394 result = !filter(point.level(), point_pattern, nulldisp);
395 }
396 return result;
397 };
398
399 if (!filtered_out())
400 return;
401 }
402
403 // ready to switch to next fixed dimension with finite radius
404 // but first update box bounds to exclude the surface displacements for the current fixed dimension
405 // WARNING if box along this dimension is not hollow we are done!
407 box[fixed_dim] = {
408 parent->box_[fixed_dim].first +
409 parent->surface_thickness_[fixed_dim].value_or(0) + 1,
410 parent->box_[fixed_dim].second -
411 parent->surface_thickness_[fixed_dim].value_or(0) - 1};
412 }
413 else {
414 done = true;
415 return;
416 }
417 // onto next dimension
418 ++fixed_dim;
419 while (!parent->box_radius_[fixed_dim] && fixed_dim < NDIM) {
420 ++fixed_dim;
421 }
422
423
424 if (fixed_dim >= NDIM) {
425 done = true;
426 return;
427 }
428
429 // reset upon moving to the next fixed dimension
430 for (size_t i = 0; i < NDIM; ++i) {
432 }
433 }
434
436 if (parent->filter_) {
437 const auto filtered_out = [&]() -> bool {
438 this->displacement(); // ensure disp is up to date
439 return !parent->filter_(point.level(), point.translation(), disp);
440 };
441
442 // if displacement has value, filter has already been applied to it, just advance it
443 if (!done && disp) this->advance();
444
445 while (!done && filtered_out()) {
446 this->advance();
447 }
448 }
449 else
450 this->advance();
451 }
452
453 void reset_along_dim(size_t dim) {
454 const auto is_fixed_dim = dim == fixed_dim;
455 Vector<Translation, NDIM> l = point.translation();
456 // for fixed dimension start with first surface layer, else use box lower bound (N.B. it's updated as fixed dimensions change)
457 Translation l_dim_min =
458 is_fixed_dim
459 ? parent->box_[dim].first -
460 parent->surface_thickness_[dim].value_or(0)
461 : box[dim].first;
462 // if dimension is periodic, only include *unique* displacements (modulo period)
463 if (parent->is_periodic_[dim]) {
464 const auto period = 1 << parent->center_.level();
465 const Translation l_dim_max =
466 is_fixed_dim ? parent->box_[dim].second +
467 parent->surface_thickness_[dim].value_or(0) :
468 box[dim].second;
469 l_dim_min = std::max(l_dim_min, l_dim_max - period + 1);
470 // fixed dim only: l_dim_min may not correspond to a surface layer
471 // this can only happen if l_dim_min is in the gap between the surface layers
472 if (is_fixed_dim && parent->hollowness_[dim]) {
473 if (l_dim_min > parent->box_[dim].first +
474 parent->surface_thickness_[dim].value_or(0)) {
475 l_dim_min = std::max(
476 l_dim_min, parent->box_[dim].second -
477 parent->surface_thickness_[dim].value_or(0));
478 }
479 }
480 }
481 l[dim] = l_dim_min;
482
483 point = Point(point.level(), l);
484
485 // if the entire surface layer is filtered out, pick the next one
486 if (dim == fixed_dim) {
487
488 const auto filtered_out = [&,this]() {
489 bool result = false;
490 const auto& filter = this->parent->filter_;
491 if (filter) {
492 PointPattern point_pattern;
493 point_pattern[fixed_dim] = point[fixed_dim];
494 std::optional<Displacement> nulldisp;
495 result = !filter(point.level(), point_pattern, nulldisp);
496 }
497 return result;
498 };
499
500 if (filtered_out()) {
501 bool have_another_surface_layer;
502 while ((have_another_surface_layer = next_surface_layer())) {
503 if (!filtered_out())
504 break;
505 }
506 MADNESS_ASSERT(have_another_surface_layer);
507 }
508
509 }
510 };
511
512 /**
513 * @return displacement from the center to the current point
514 */
515 const std::optional<Displacement>& displacement() const {
516 if (!disp) {
518 }
519 return disp;
520 }
521
522 public:
523 // Iterator type definitions for STL compatibility
524 using iterator_category = std::input_iterator_tag;
526 using difference_type = std::ptrdiff_t;
527 using pointer = const Point*;
528 using reference = const Point&;
529
530 /**
531 * @brief Constructs an iterator
532 *
533 * @param p Pointer to the parent BoxSurfaceDisplacementRange
534 * @param type the type of iterator (Begin or End)
535 */
537 : parent(p), point(parent->center_.level()), fixed_dim(type == End ? NDIM : 0), done(type == End) {
538 if (type != End) {
539 // skip to first dimensions with limited range
540 while (!parent->box_radius_[fixed_dim] && fixed_dim < NDIM) {
541 ++fixed_dim;
542 }
543
544 for (size_t d = 0; d != NDIM; ++d) {
545 // min/max displacements along this axis ... N.B. take into account surface thickness!
546 box[d] = parent->box_radius_[d] ? std::pair{parent->box_[d].first -
547 parent->surface_thickness_[d].value_or(0),
548 parent->box_[d].second +
549 parent->surface_thickness_[d].value_or(0)} : parent->box_[d];
551 }
553 }
554 }
555
556 /**
557 * @brief Dereferences the iterator
558 * @return A const reference to the current displacement
559 */
560 reference operator*() const { return *displacement(); }
561
562 /**
563 * @brief Arrow operator for member access
564 * @return A const pointer to the current displacement
565 */
566 pointer operator->() const { return &(*(*this)); }
567
568 /**
569 * @brief Pre-increment operator
570 * @return Reference to this iterator after advancement
571 */
574 return *this;
575 }
576
577 /**
578 * @brief Post-increment operator
579 * @return Copy of the iterator before advancement
580 */
582 Iterator tmp = *this;
583 ++(*this);
584 return tmp;
585 }
586
587 /**
588 * @brief Equality comparison operator
589 * @param a First iterator
590 * @param b Second iterator
591 * @return true if iterators are equivalent
592 */
593 friend bool operator==(const Iterator& a, const Iterator& b) {
594 if (a.done && b.done) return true;
595 if (a.done || b.done) return false;
596 return a.fixed_dim == b.fixed_dim &&
597 a.point == b.point;
598 }
599
600 /**
601 * @brief Inequality comparison operator
602 * @param a First iterator
603 * @param b Second iterator
604 * @return true if iterators are not equivalent
605 */
606 friend bool operator!=(const Iterator& a, const Iterator& b) {
607 return !(a == b);
608 }
609 };
610
611 friend class Iterator;
612
613 public:
614 /**
615 * @brief Constructs a box with different sizes for each dimension
616 *
617 * @param center Center of the box
618 * @param box_radius Box radius in each dimension
619 * @param surface_thickness Surface thickness in each dimension
620 * @param is_periodic whether each dimension is periodic; along periodic range-restricted dimensions only one side of the box is iterated over.
621 * @param filter Optional filter function (if returns false, displacement is dropped; default: no filter); it may update the displacement to make it valid as needed (e.g. map displacement to the simulation cell)
622 * @pre `box_radius[d]>0 && surface_thickness[d]<=box_radius[d]`
623 *
624 */
626 const std::array<std::optional<std::int64_t>, NDIM>& box_radius,
627 const std::array<std::optional<std::int64_t>, NDIM>& surface_thickness,
629 Filter filter = {})
632 // initialize box bounds
633 bool has_finite_dimensions = false;
634 const auto n = center_.level();
635 Vector<Translation, NDIM> probing_displacement_vec(0);
636 for (int d=0; d!= NDIM; ++d) {
637 if (box_radius_[d]) {
638 auto r = *box_radius_[d]; // in units of 2^{n-1}
639 r = (n == 0) ? (r+1)/2 : (r * Translation(1) << (n-1));
640 MADNESS_ASSERT(r > 0);
641 box_[d] = {center_[d] - r, center_[d] + r};
642 if (!has_finite_dimensions) // first finite dimension? probing displacement will be nonzero along it, zero along all others
643 probing_displacement_vec[d] = r;
644 has_finite_dimensions = true;
645 } else {
646 box_[d] = {0, (1 << center_.level()) - 1};
647 }
648 }
649 MADNESS_ASSERT(has_finite_dimensions);
650 probing_displacement_ = Displacement(n, probing_displacement_vec);
651 for (int d=0; d!= NDIM; ++d) {
652 // surface thickness should be only given for finite-radius dimensions
653 MADNESS_ASSERT(!(box_radius_[d].has_value() ^ surface_thickness_[d].has_value()));
654 MADNESS_ASSERT(surface_thickness_[d].value_or(0) >= 0);
655 hollowness_[d] = surface_thickness_[d] ? (box_[d].first + surface_thickness_[d].value() < box_[d].second - surface_thickness_[d].value()) : false;
656 }
657 }
658
659 /**
660 * @brief Returns an iterator to the beginning of the surface points
661 * @return Iterator pointing to the first surface point
662 */
663 auto begin() const { return Iterator(this, Iterator::Begin); }
664
665 /**
666 * @brief Returns an iterator to the end of the surface points
667 * @return Iterator indicating the end of iteration
668 */
669 auto end() const { return Iterator(this, Iterator::End); }
670
671 // /**
672 // * @brief Returns a view over the surface points
673 // *
674 // * This operator allows the class to be used with C++20 ranges.
675 // *
676 // * @return A view over the surface points
677 // */
678 // auto operator()() const {
679 // return std::ranges::subrange(begin(), end());
680 // }
681
682 /* @return the center of the box
683 */
684 const Key<NDIM>& center() const { return center_; }
685
686 /**
687 * @return the radius of the box in each dimension
688 */
689 const std::array<std::optional<int64_t>, NDIM>& box_radius() const { return box_radius_; }
690
691 /**
692 * @return the surface thickness in each dimension
693 */
694 const std::array<std::optional<int64_t>, NDIM>& surface_thickness() const { return surface_thickness_; }
695
696 /**
697 * @return flags indicating whether each dimension is periodic
698 */
700
701 /**
702 * @return 'probing" displacement to a nearby point *on* the surface; it may not necessarily be in the range of iteration (e.g., it may not be able to pass the filter) but is representative of the surface displacements for the purposes of screening
703 */
706 }
707 }; // BoxSurfaceDisplacementRange
708
709
710 /// This is used to filter out box surface displacements that
711 /// - take us outside of the target domain, or
712 /// - were already utilized as part of the the standard displacements list.
713 /// For dealing with the lattice-summed operators the filter
714 /// can adjusts the displacement to make sure that we end up in
715 /// the simulation cell.
716 template <size_t NDIM>
718 public:
723 using DistanceSquaredFunc = std::function<Translation(const Displacement&)>;
724
725 /// \param domain_is_infinite whether the domain along each axis is finite (simulation cell) or infinite (the entire axis); if true for a given axis then any destination coordinate is valid, else only values in [0,2^n) are valid
726 /// \param domain_is_periodic if true for a given axis, displacement to x and x+2^n are equivalent, hence will be canonicalized to end up in the simulation cell. Periodic axes imply infinite domain.
727 /// \param range the kernel range for each axis
728 /// \param default_distance_squared function that converts a displacement to its effective distance squared (effective may be different from the real distance squared due to periodicity)
729 /// \param max_distsq_reached max effective distance squared reached by standard displacements
731 const array_of_bools<NDIM>& domain_is_infinite,
732 const array_of_bools<NDIM>& domain_is_periodic,
733 const std::array<KernelRange, NDIM>& range,
734 DistanceSquaredFunc default_distance_squared,
735 Translation max_distsq_reached
736 ) :
737 domain_is_infinite_(domain_is_infinite),
738 domain_is_periodic_(domain_is_periodic),
739 range_(range),
740 default_distance_squared_(default_distance_squared),
741 max_distsq_reached_(max_distsq_reached)
742 {}
743
744 /// Apply filter to a displacement ending up at a point or a group of points (point pattern)
745
746 /// @param level the tree level
747 /// @param dest the target point (when all elements are nonnull) or point pattern (when only some are).
748 /// The latter is useful to skip the entire surface layer. The point coordinates are
749 /// only used to determine whether we end up in or out of the domain.
750 /// @param displacement the optional displacement; if given then will check if it's among
751 /// the standard displacement and whether it was used as part of the standard displacement
752 /// set; if it has not been used and the operator is lattice summed, the displacement
753 /// will be adjusted to end up in the simulation cell.
754 /// @return true if the displacement is to be used
756 const Level level,
757 const PointPattern& dest,
758 std::optional<Displacement>& displacement
759 ) const {
760 // preliminaries
761 const auto twon = (static_cast<Translation>(1) << level); // number of boxes along an axis
762 // map_to_range_twon(x) returns for x >= 0 ? x % 2^level : map_to_range_twon(x+2^level)
763 // idiv is generally slow, so instead use bit logic that relies on 2's complement representation of integers
764 const auto map_to_range_twon = [&, mask = ((~(static_cast<std::uint64_t>(0)) << (64-level)) >> (64-level))](std::int64_t x) -> std::int64_t {
765 const std::int64_t x_mapped = x & mask;
766 MADNESS_ASSERT(x_mapped >=0 && x_mapped < twon && (std::abs(x_mapped-x)%twon==0));
767 return x_mapped;
768 };
769
770 const auto out_of_domain = [&](const Translation& t) -> bool {
771 return t < 0 || t >= twon;
772 };
773
774 // check that dest is in the domain
775 const bool dest_is_in_domain = [&]() {
776 for(auto d=0; d!=NDIM; ++d) {
777 // - if domain is periodic, all displacements will be mapped back to the simulation cell by for_each/neighbor
778 // - if kernel is lattice summed mapping back to the simulation cell for standard displacements
779 // is done during their construction, and for boundary displacements manually (see IMPORTANT below in this function)
780 // N.B. due to this mapping back into the cell only half of the surface displacements was generated by BoxSurfaceDisplacementRange
781 if (domain_is_infinite_[d] || domain_is_periodic_[d]) continue;
782 if (dest[d].has_value() && out_of_domain(*dest[d])) return false;
783 }
784 return true;
785 }();
786
787 if (dest_is_in_domain) {
788 if (displacement.has_value()) {
789
790 // N.B. avoid duplicates of standard displacements previously included:
791 // A displacement has been possibly considered if along EVERY axis the "effective" displacement size
792 // fits within the box explored by the standard displacement.
793 // If so, skip if <= max magnitude of standard displacements encountered
794 // Otherwise this is a new non-standard displacement, consider it
795 bool among_standard_displacements = true;
796 for(auto d=0; d!=NDIM; ++d) {
797 const auto disp_d = (*displacement)[d];
798 auto bmax_standard = Displacements<NDIM>::bmax_default();
799
800 // the effective displacement length depends on whether lattice summation is performed along it
801 // compare Displacements::make_disp vs Displacements::make_disp_periodic
802 auto disp_d_eff_abs = std::abs(disp_d);
803 if (domain_is_periodic_[d]) {
804 MADNESS_ASSERT(range_[d].N() <= 2); // displacements that exceed 1 whole cell need bit more complex logic
805
806 // for "periodic" displacements the effective disp_d is the shortest of {..., disp_d-twon, disp_d, disp_d+twon, ...} ... see make_disp_periodic
807 const std::int64_t disp_d_eff = map_to_range_twon(disp_d);
808 disp_d_eff_abs = std::min(disp_d_eff,std::abs(disp_d_eff-twon));
809
810 // IMPORTANT for lattice-summed axes, if the destination is out of the simulation cell map the displacement back to the cell
811 // same logic as for disp_d: dest[d] -> dest[d] % twon
812 if (dest[d].has_value()) {
813 const Translation dest_d = dest[d].value();
814 const auto dest_d_in_cell = map_to_range_twon(dest_d);
815 MADNESS_ASSERT(!out_of_domain(
816 dest_d_in_cell));
817 // adjust displacement[d] so that it produces dest_d_cell, not dest_d
818 auto t = (*displacement).translation();
819 t[d] += (dest_d_in_cell - dest_d);
820 displacement.emplace(displacement->level(), t);
821 }
822
823 // N.B. bmax in make_disp_periodic is wrapped around, adjust
824 if (Displacements<NDIM>::bmax_default() >= twon) bmax_standard = twon-1;
825 }
826
827 if (disp_d_eff_abs > bmax_standard) {
828 among_standard_displacements = false;
829 break;
830 }
831 }
832 if (among_standard_displacements) {
833 const auto distsq = default_distance_squared_(*displacement);
834 if (distsq > max_distsq_reached_) { // among standard displacements => keep if longer than the longest standard displacement considered
835 return true;
836 } {
837 return false;
838 }
839 }
840 else // not among standard displacements => keep it
841 return true;
842 }
843 else // skip the displacement-based filter if not given
844 return true;
845 }
846 else
847 return false;
848 }
849
850 private:
853 std::array<KernelRange, NDIM> range_;
856 };
857
858} // namespace madness
859#endif // MADNESS_MRA_DISPLACEMENTS_H__INCLUDED
Definition displacements.h:717
bool operator()(const Level level, const PointPattern &dest, std::optional< Displacement > &displacement) const
Apply filter to a displacement ending up at a point or a group of points (point pattern)
Definition displacements.h:755
array_of_bools< NDIM > domain_is_periodic_
Definition displacements.h:852
array_of_bools< NDIM > domain_is_infinite_
Definition displacements.h:851
std::array< KernelRange, NDIM > range_
Definition displacements.h:853
std::function< Translation(const Displacement &)> DistanceSquaredFunc
Definition displacements.h:723
Translation max_distsq_reached_
Definition displacements.h:855
Key< NDIM > Point
Definition displacements.h:719
DistanceSquaredFunc default_distance_squared_
Definition displacements.h:854
Vector< std::optional< Translation >, NDIM > PointPattern
Definition displacements.h:720
Key< NDIM > Displacement
Definition displacements.h:721
BoxSurfaceDisplacementFilter(const array_of_bools< NDIM > &domain_is_infinite, const array_of_bools< NDIM > &domain_is_periodic, const std::array< KernelRange, NDIM > &range, DistanceSquaredFunc default_distance_squared, Translation max_distsq_reached)
Definition displacements.h:730
Iterator class for lazy generation of surface points.
Definition displacements.h:325
std::optional< Displacement > disp
Memoized displacement from parent->center_ to point, computed by displacement(), reset by advance()
Definition displacements.h:331
Iterator(const BoxSurfaceDisplacementRange *p, Type type)
Constructs an iterator.
Definition displacements.h:536
@ End
Definition displacements.h:327
@ Begin
Definition displacements.h:327
size_t fixed_dim
Current fixed dimension (i.e. faces perpendicular to this axis are being iterated over)
Definition displacements.h:332
const std::optional< Displacement > & displacement() const
Definition displacements.h:515
friend bool operator!=(const Iterator &a, const Iterator &b)
Inequality comparison operator.
Definition displacements.h:606
bool done
Flag indicating iteration completion.
Definition displacements.h:334
const Point * pointer
Definition displacements.h:527
pointer operator->() const
Arrow operator for member access.
Definition displacements.h:566
Iterator operator++(int)
Post-increment operator.
Definition displacements.h:581
std::ptrdiff_t difference_type
Definition displacements.h:526
std::input_iterator_tag iterator_category
Definition displacements.h:524
Box box
updated box bounds in each dimension, used to avoid duplicate displacements by excluding the surface ...
Definition displacements.h:333
const BoxSurfaceDisplacementRange * parent
Pointer to parent box.
Definition displacements.h:329
bool next_surface_layer()
Definition displacements.h:337
reference operator*() const
Dereferences the iterator.
Definition displacements.h:560
Point value_type
Definition displacements.h:525
void reset_along_dim(size_t dim)
Definition displacements.h:453
void advance_till_valid()
Definition displacements.h:435
Point point
Current point.
Definition displacements.h:330
friend bool operator==(const Iterator &a, const Iterator &b)
Equality comparison operator.
Definition displacements.h:593
Iterator & operator++()
Pre-increment operator.
Definition displacements.h:572
const Point & reference
Definition displacements.h:528
void advance()
Advances the iterator to the next surface point.
Definition displacements.h:366
Definition displacements.h:294
Displacement probing_displacement_
displacement to a nearby point on the surface; it may not be able to pass the filter,...
Definition displacements.h:317
Key< NDIM > Point
Definition displacements.h:296
const Key< NDIM > & center() const
Definition displacements.h:684
std::array< std::optional< Translation >, NDIM > SurfaceThickness
Definition displacements.h:304
const std::array< std::optional< int64_t >, NDIM > & box_radius() const
Definition displacements.h:689
SurfaceThickness surface_thickness_
surface thickness in each dimension
Definition displacements.h:312
std::array< std::optional< Translation >, NDIM > BoxRadius
Definition displacements.h:303
const std::array< std::optional< int64_t >, NDIM > & surface_thickness() const
Definition displacements.h:694
BoxSurfaceDisplacementRange(const Key< NDIM > &center, const std::array< std::optional< std::int64_t >, NDIM > &box_radius, const std::array< std::optional< std::int64_t >, NDIM > &surface_thickness, const array_of_bools< NDIM > &is_periodic, Filter filter={})
Constructs a box with different sizes for each dimension.
Definition displacements.h:625
Hollowness hollowness_
does box contain non-surface points along each dimension?
Definition displacements.h:314
auto begin() const
Returns an iterator to the beginning of the surface points.
Definition displacements.h:663
Box box_
box bounds in each dimension
Definition displacements.h:313
Filter filter_
optional filter function
Definition displacements.h:316
const Displacement & probing_displacement() const
Definition displacements.h:704
Point center_
Center point of the box.
Definition displacements.h:309
Periodicity is_periodic_
which dimensions are periodic?
Definition displacements.h:315
std::function< bool(Level, const PointPattern &, std::optional< Displacement > &)> Filter
this callable filters out points and/or displacements; note that the displacement is optional (this u...
Definition displacements.h:300
Key< NDIM > Displacement
Definition displacements.h:298
std::array< bool, NDIM > Hollowness
Definition displacements.h:306
const array_of_bools< NDIM > & is_periodic() const
Definition displacements.h:699
auto end() const
Returns an iterator to the end of the surface points.
Definition displacements.h:669
Vector< std::optional< Translation >, NDIM > PointPattern
Definition displacements.h:297
std::array< std::pair< Translation, Translation >, NDIM > Box
Definition displacements.h:305
BoxRadius box_radius_
halved size of the box in each dimension
Definition displacements.h:310
Holds displacements for applying operators to avoid replicating for all operators.
Definition displacements.h:51
static std::vector< Key< NDIM > > disp
standard displacements to be used with standard kernels (range-unrestricted, no lattice sum)
Definition displacements.h:53
static bool cmp_keys(const Key< NDIM > &a, const Key< NDIM > &b)
Definition displacements.h:71
const std::vector< Key< NDIM > > & get_disp()
return the standard displacements appropriate for operators w/o lattice summation
Definition displacements.h:235
const std::vector< Key< NDIM > > & get_disp(Level n, const array_of_bools< NDIM > &kernel_lattice_sum_axes)
Definition displacements.h:211
static array_of_bools< NDIM > periodic_axes
along which axes lattice summation is performed?
Definition displacements.h:54
static void reset_periodic_axes(const array_of_bools< NDIM > &new_periodic_axes)
rebuilds periodic displacements so that they are optimal for the given set of periodic axes
Definition displacements.h:249
static void make_disp(int bmax)
Definition displacements.h:79
static std::vector< Key< NDIM > > disp_periodic[64]
displacements to be used with lattice-summed kernels
Definition displacements.h:55
static void make_disp_periodic(int bmax, Level n)
Definition displacements.h:135
static int bmax_default()
Definition displacements.h:58
static bool cmp_keys_periodic(const Key< NDIM > &a, const Key< NDIM > &b)
Definition displacements.h:75
Displacements()
Definition displacements.h:190
FunctionDefaults holds default paramaters as static class members.
Definition funcdefaults.h:100
Definition indexit.h:55
Key is the index for a node of the 2^NDIM-tree.
Definition key.h:69
A simple, fixed dimension vector.
Definition vector.h:64
syntactic sugar for std::array<bool, N>
Definition array_of_bools.h:19
bool any() const
Definition array_of_bools.h:38
char * p(char *buf, const char *name, int k, int initial_level, double thresh, int order)
Definition derivatives.cc:72
real_function_3d mask
Definition dirac-hatom.cc:27
Provides FunctionDefaults and utilities for coordinate transformation.
Provides IndexIterator.
#define MADNESS_PRAGMA_CLANG(x)
Definition madness_config.h:200
#define MADNESS_EXCEPTION(msg, value)
Macro for throwing a MADNESS exception.
Definition madness_exception.h:119
#define MADNESS_ASSERT(condition)
Assert a condition that should be free of side-effects since in release builds this might be a no-op.
Definition madness_exception.h:134
Namespace for all elements and tools of MADNESS.
Definition DFParameters.h:10
int64_t Translation
Definition key.h:57
Key< NDIM > displacement(const Key< NDIM > &source, const Key< NDIM > &target)
given a source and a target, return the displacement in translation
Definition key.h:451
int Level
Definition key.h:58
static double pop(std::vector< double > &v)
Definition SCF.cc:113
constexpr std::array< std::size_t, N-M > iota_array(std::array< std::size_t, M > values_to_skip_sorted)
Definition displacements.h:266
std::string type(const PairType &n)
Definition PNOParameters.h:18
Definition mraimpl.h:50
static long abs(long a)
Definition tensor.h:218
static const double b
Definition nonlinschro.cc:119
static const double d
Definition nonlinschro.cc:121
static const double a
Definition nonlinschro.cc:118
#define N
Definition testconv.cc:37
constexpr std::size_t NDIM
Definition testgconv.cc:54