ESPResSo
Extensible Simulation Package for Research on Soft Matter Systems
Loading...
Searching...
No Matches
elc.hpp
Go to the documentation of this file.
1/*
2 * Copyright (C) 2010-2026 The ESPResSo project
3 * Copyright (C) 2002,2003,2004,2005,2006,2007,2008,2009,2010
4 * Max-Planck-Institute for Polymer Research, Theory Group
5 *
6 * This file is part of ESPResSo.
7 *
8 * ESPResSo is free software: you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation, either version 3 of the License, or
11 * (at your option) any later version.
12 *
13 * ESPResSo is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License
19 * along with this program. If not, see <http://www.gnu.org/licenses/>.
20 */
21
22/** @file
23 * @brief ELC algorithm for long-range Coulomb interactions.
24 *
25 * Implementation of the ELC method for the calculation of the electrostatic
26 * interaction in two dimensional periodic systems. For details on the method
27 * see MMM in general. The ELC method works together with any three-dimensional
28 * method, for example @ref p3m.hpp "P3M", with metallic boundary conditions.
29 */
30
31#pragma once
32
33#include <config/config.hpp>
34
35#ifdef ESPRESSO_P3M
36
37#include "actor/traits.hpp"
38
40
41#include "BoxGeometry.hpp"
42#include "Particle.hpp"
43#include "ParticleRange.hpp"
44
45#include <utils/Vector.hpp>
46#include <utils/math/sqr.hpp>
47
48#include <cstddef>
49#include <memory>
50#include <optional>
51#include <string>
52#include <type_traits>
53#include <utility>
54#include <variant>
55
57
58namespace traits {
59template <>
61} // namespace traits
62
63/** @brief Parameters for the ELC method */
64struct elc_data {
65 elc_data(double maxPWerror, double gap_size, double far_cut, bool neutralize,
66 double delta_top, double delta_bot, bool const_pot, double pot_diff);
67
68 /**
69 * @brief Maximal allowed pairwise error for the potential and force.
70 * Note that this counts for the plain 1/r contribution
71 * alone, without the prefactor and the charge prefactor.
72 */
73 double maxPWerror;
74 /** Size of the empty gap. Note that ELC relies on the user to make sure
75 * that this condition is fulfilled.
76 */
77 double gap_size;
78 /** Up to where particles can be found. */
79 double box_h;
80 /**
81 * @brief Cutoff of the exponential sum.
82 * Since in all other MMM methods this is the far formula,
83 * it is given the same name here.
84 */
85 double far_cut;
86 /** Squared value of #far_cut. */
87 double far_cut2;
88 /** Flag whether #far_cut was set by the user, or calculated by ESPResSo.
89 * In the latter case, the cutoff will be adapted if important parameters,
90 * such as the box dimensions, change.
91 */
93
94 /// @brief Flag whether there is any dielectric contrast in the system.
96 /// @brief Flag whether a constant potential difference is applied.
98 /**
99 * @brief Flag whether the box is neutralized by a homogeneous background.
100 * If true, use a homogeneous neutralizing background for non-neutral
101 * systems. Unlike the 3D case, this background adds an additional
102 * force pointing towards the system center, and the gap size
103 * enters into the value of the forces, so be careful with this.
104 */
106
107 /// dielectric contrast in the upper part of the simulation cell.
109 /// dielectric contrast in the lower part of the simulation cell.
111 /// @brief Constant potential difference.
112 double pot_diff;
113
114 /** Layer around the dielectric contrast in which we trick around. */
116 /** The space that is finally left. */
117 double space_box;
118
119 /// pairwise contributions from lower and upper layers
121 std::size_t p1, std::size_t p2,
122 auto &aosoa, double q1q2,
123 auto &&kernel) const {
124 if (aosoa.position(p1, 2) < space_layer) {
125 auto const q_eff = delta_mid_bot * q1q2;
126 auto pos2 = aosoa.get_vector_at(aosoa.position, p2);
127 auto pos1 = aosoa.get_vector_at(aosoa.position, p1);
128 pos1[2] *= -1.;
129 auto const d = box_geo.get_mi_vector(pos2, pos1);
130 kernel(q_eff, d);
131 }
132 if (aosoa.position(p1, 2) > (box_h - space_layer)) {
133 auto const q_eff = delta_mid_top * q1q2;
134 auto const z = 2. * box_h - aosoa.position(p1, 2);
135 auto pos2 = aosoa.get_vector_at(aosoa.position, p2);
136 auto pos1 = aosoa.get_vector_at(aosoa.position, p1);
137 pos1[2] = 2. * box_h - pos1[2];
138 auto const d = box_geo.get_mi_vector(pos2, pos1);
139 kernel(q_eff, d);
140 }
141 }
142
143 /// pairwise contributions from lower and upper layers
145 Utils::Vector3d const &pos1,
146 Utils::Vector3d const &pos2, double q1q2,
147 auto &&kernel) const {
148 if (pos1[2] < space_layer) {
149 auto const q_eff = delta_mid_bot * q1q2;
150 auto const d = box_geo.get_mi_vector(pos2, {pos1[0], pos1[1], -pos1[2]});
151 kernel(q_eff, d);
152 }
153 if (pos1[2] > (box_h - space_layer)) {
154 auto const q_eff = delta_mid_top * q1q2;
155 auto const z = 2. * box_h - pos1[2];
156 auto const d = box_geo.get_mi_vector(pos2, {pos1[0], pos1[1], z});
157 kernel(q_eff, d);
158 }
159 }
160
161 /// self energies of top and bottom layers with their virtual images
163 BoxGeometry const &box_geo,
164 ParticleRange const &particles) const {
165 auto energy = 0.;
166 for (auto const &p : particles) {
168 box_geo, p.pos(), p.pos(), Utils::sqr(p.q()),
169 [&](double q1q2, Utils::Vector3d const &d) {
170 energy += p3m.pair_energy(q1q2, d.norm());
171 });
172 }
173 return energy;
174 }
175
176 /// forces of particles in border layers with themselves
178 BoxGeometry const &box_geo,
179 ParticleRange const &particles) const {
180 for (auto &p : particles) {
182 box_geo, p.pos(), p.pos(), Utils::sqr(p.q()),
183 [&](double q1q2, Utils::Vector3d const &d) {
184 p.force() += p3m.pair_force(q1q2, d, d.norm());
185 });
186 }
187 }
188};
189
191 : public Coulomb::Actor<ElectrostaticLayerCorrection> {
192 using BaseSolver = std::variant<std::shared_ptr<CoulombP3M>>;
193
196
197 /** Electrostatics solver that is adapted. */
199
200 ElectrostaticLayerCorrection(elc_data &&parameters, BaseSolver &&solver);
201
203 visit_base_solver(
204 [this](auto &solver) { solver->bind_system(m_system.lock()); });
205 sanity_checks_periodicity();
206 sanity_checks_cell_structure();
208 sanity_checks_dielectric_contrasts();
209 /* Most ELC parameters do not depend on the P3M parameters,
210 * but the P3M parameters depend on the ELC parameters during tuning,
211 * therefore ELC needs to be tuned before P3M. */
212 recalc_box_h();
213 recalc_far_cut();
214 visit_base_solver([](auto &solver) { solver->on_activation(); });
215 /* With dielectric contrasts, the ELC space layer depends
216 * on the P3M real-space cutoff, and the P3M FFT parameters
217 * depend on the ELC space layer */
219 recalc_space_layer();
220 visit_base_solver([](auto &actor) { actor->init(); });
221 }
222 }
223 /** @brief Recalculate all box-length-dependent parameters. */
225 visit_base_solver([](auto &actor) { actor->on_boxl_change(); });
226 recalc_box_h();
227 recalc_far_cut();
228 recalc_space_layer();
229 }
230 void on_node_grid_change() const {
231 visit_base_solver([](auto &solver) { solver->on_node_grid_change(); });
232 }
234 sanity_checks_periodicity();
235 visit_base_solver([](auto &solver) { solver->on_periodicity_change(); });
236 }
238 sanity_checks_cell_structure();
239 visit_base_solver([](auto &solver) { solver->on_cell_structure_change(); });
240 recalc_box_h();
241 recalc_far_cut();
243 recalc_space_layer();
244 visit_base_solver([](auto &actor) { actor->init(); });
245 }
246 }
247 /** @brief Recalculate all derived parameters. */
248 void init() {
249 visit_base_solver([](auto &actor) { actor->init(); });
250 recalc_box_h();
251 recalc_far_cut();
252 recalc_space_layer();
253 }
254
255 void sanity_checks() const {
256 sanity_checks_periodicity();
257 sanity_checks_cell_structure();
259 sanity_checks_dielectric_contrasts();
260 visit_base_solver([](auto &actor) { actor->sanity_checks(); });
261 }
262
263 /**
264 * @brief Veto real-space cutoff values that are incompatible with ELC.
265 * When ELC is used with dielectric contrasts, the short-range cutoff needs
266 * to be smaller than the gap size to allow placement of the image charges.
267 */
268 std::optional<std::string> veto_r_cut(double r_cut) const {
269 if (elc.dielectric_contrast_on and r_cut >= elc.gap_size) {
270 return {std::string("conflict with ELC w/ dielectric contrasts")};
271 }
272 return {};
273 }
274
276 double dist) const {
277 return std::visit(
278 [&](auto &solver) { return solver->pair_force(q1q2, d, dist); },
280 }
281
282 /** @brief Calculate short-range pair energy correction. */
283 double pair_energy_correction(std::size_t p1, std::size_t p2, auto &aosoa,
284 double q1q2) const {
285 double energy = 0.;
287 energy = std::visit(
288 [this, &aosoa, p1, p2, q1q2](auto &p3m_ptr) {
289 auto const &p3m = *p3m_ptr;
290 auto energy = 0.;
292 *m_box_geo, p1, p2, aosoa, q1q2,
293 [&](double q_eff, Utils::Vector3d const &d) {
294 energy += p3m.pair_energy(q_eff, d.norm());
295 });
297 *m_box_geo, p2, p1, aosoa, q1q2,
298 [&](double q_eff, Utils::Vector3d const &d) {
299 energy += p3m.pair_energy(q_eff, d.norm());
300 });
301 return energy / 2.;
302 },
304 }
305 return energy;
306 }
307
308 /** @brief Calculate short-range pair energy correction. */
310 Utils::Vector3d const &pos2,
311 double q1q2) const {
312 double energy = 0.;
314 energy = std::visit(
315 [this, &pos1, &pos2, q1q2](auto &p3m_ptr) {
316 auto const &p3m = *p3m_ptr;
317 auto energy = 0.;
319 *m_box_geo, pos1, pos2, q1q2,
320 [&](double q_eff, Utils::Vector3d const &d) {
321 energy += p3m.pair_energy(q_eff, d.norm());
322 });
324 *m_box_geo, pos2, pos1, q1q2,
325 [&](double q_eff, Utils::Vector3d const &d) {
326 energy += p3m.pair_energy(q_eff, d.norm());
327 });
328 return energy / 2.;
329 },
331 }
332 return energy;
333 }
334
335 /** @brief Add short-range pair force corrections. */
337 Utils::Vector3d const &pos2,
338 Utils::Vector3d &p1f_asym,
339 Utils::Vector3d &p2f_asym,
340 double q1q2) const {
342 std::visit(
343 [this, &pos1, &pos2, &p1f_asym, &p2f_asym, q1q2](auto &p3m_ptr) {
344 auto const &p3m = *p3m_ptr;
346 *m_box_geo, pos1, pos2, q1q2,
347 [&](double q_eff, Utils::Vector3d const &d) {
348 p1f_asym += p3m.pair_force(q_eff, d, d.norm());
349 });
351 *m_box_geo, pos2, pos1, q1q2,
352 [&](double q_eff, Utils::Vector3d const &d) {
353 p2f_asym += p3m.pair_force(q_eff, d, d.norm());
354 });
355 },
357 }
358 }
359
360 /** @brief Calculate long-range electrostatic energy with corrections. */
361 double long_range_energy() const;
362 /** @brief Accumulate long-range electrostatic forces with corrections. */
363 void add_long_range_forces() const;
364
365private:
366 /** Check if a charged particle is in the gap region. */
367 void check_gap(Particle const &p) const;
368 double tune_far_cut() const;
369 void adapt_solver();
370 /** pairwise contributions from the lowest and top layers to the energy */
371 double dipole_energy() const;
372 void add_dipole_force() const;
373 double z_energy() const;
374 void add_z_force() const;
375
376 void recalc_box_h();
377 void recalc_far_cut() {
378 if (elc.far_calculated) {
379 elc.far_cut = tune_far_cut();
380 }
382 }
383 void recalc_space_layer();
384
385 void sanity_checks_cell_structure() const {}
386 void sanity_checks_periodicity() const;
387 void sanity_checks_dielectric_contrasts() const;
388
389 /// the force calculation
390 void add_force() const;
391 /// the energy calculation
392 double calc_energy() const;
393
394 void visit_base_solver(auto &&visitor) const {
395 std::visit(visitor, base_solver);
396 }
397};
398
399#endif // ESPRESSO_P3M
Vector implementation and trait types for boost qvm interoperability.
ESPRESSO_ATTR_ALWAYS_INLINE Utils::Vector3< T > get_mi_vector(Utils::Vector3< T > const &a, Utils::Vector3< T > const &b) const
Get the minimum-image vector between two coordinates.
A range of particles.
std::weak_ptr< System > m_system
T norm() const
Definition Vector.hpp:159
DEVICE_QUALIFIER constexpr T sqr(T x)
Calculates the SQuaRe of x.
Definition sqr.hpp:28
P3M algorithm for long-range Coulomb interaction.
P3M solver.
Definition p3m.hpp:55
void on_boxl_change()
Recalculate all box-length-dependent parameters.
Definition elc.hpp:224
void on_periodicity_change() const
Definition elc.hpp:233
void sanity_checks() const
Definition elc.hpp:255
void add_long_range_forces() const
Accumulate long-range electrostatic forces with corrections.
Definition elc.cpp:1248
std::optional< std::string > veto_r_cut(double r_cut) const
Veto real-space cutoff values that are incompatible with ELC.
Definition elc.hpp:268
std::variant< std::shared_ptr< CoulombP3M > > BaseSolver
Definition elc.hpp:192
BaseSolver base_solver
Electrostatics solver that is adapted.
Definition elc.hpp:198
double pair_energy_correction(Utils::Vector3d const &pos1, Utils::Vector3d const &pos2, double q1q2) const
Calculate short-range pair energy correction.
Definition elc.hpp:309
Utils::Vector3d pair_force(double q1q2, Utils::Vector3d const &d, double dist) const
Definition elc.hpp:275
void add_pair_force_corrections(Utils::Vector3d const &pos1, Utils::Vector3d const &pos2, Utils::Vector3d &p1f_asym, Utils::Vector3d &p2f_asym, double q1q2) const
Add short-range pair force corrections.
Definition elc.hpp:336
void init()
Recalculate all derived parameters.
Definition elc.hpp:248
double long_range_energy() const
Calculate long-range electrostatic energy with corrections.
Definition elc.cpp:1205
void on_node_grid_change() const
Definition elc.hpp:230
BoxGeometry * m_box_geo
Definition elc.hpp:195
double pair_energy_correction(std::size_t p1, std::size_t p2, auto &aosoa, double q1q2) const
Calculate short-range pair energy correction.
Definition elc.hpp:283
Struct holding all information for one particle.
Definition Particle.hpp:435
Parameters for the ELC method.
Definition elc.hpp:64
double dielectric_layers_self_energy(CoulombP3M const &p3m, BoxGeometry const &box_geo, ParticleRange const &particles) const
self energies of top and bottom layers with their virtual images
Definition elc.hpp:162
double maxPWerror
Maximal allowed pairwise error for the potential and force.
Definition elc.hpp:73
double pot_diff
Constant potential difference.
Definition elc.hpp:112
double box_h
Up to where particles can be found.
Definition elc.hpp:79
void dielectric_layers_contribution(BoxGeometry const &box_geo, std::size_t p1, std::size_t p2, auto &aosoa, double q1q2, auto &&kernel) const
pairwise contributions from lower and upper layers
Definition elc.hpp:120
bool dielectric_contrast_on
Flag whether there is any dielectric contrast in the system.
Definition elc.hpp:95
double space_box
The space that is finally left.
Definition elc.hpp:117
void dielectric_layers_contribution(BoxGeometry const &box_geo, Utils::Vector3d const &pos1, Utils::Vector3d const &pos2, double q1q2, auto &&kernel) const
pairwise contributions from lower and upper layers
Definition elc.hpp:144
bool neutralize
Flag whether the box is neutralized by a homogeneous background.
Definition elc.hpp:105
double far_cut
Cutoff of the exponential sum.
Definition elc.hpp:85
double space_layer
Layer around the dielectric contrast in which we trick around.
Definition elc.hpp:115
bool far_calculated
Flag whether far_cut was set by the user, or calculated by ESPResSo.
Definition elc.hpp:92
double gap_size
Size of the empty gap.
Definition elc.hpp:77
double delta_mid_bot
dielectric contrast in the lower part of the simulation cell.
Definition elc.hpp:110
void dielectric_layers_self_forces(CoulombP3M const &p3m, BoxGeometry const &box_geo, ParticleRange const &particles) const
forces of particles in border layers with themselves
Definition elc.hpp:177
bool const_pot
Flag whether a constant potential difference is applied.
Definition elc.hpp:97
double far_cut2
Squared value of far_cut.
Definition elc.hpp:87
double delta_mid_top
dielectric contrast in the upper part of the simulation cell.
Definition elc.hpp:108
Whether an actor is a layer correction method.
Definition traits.hpp:26