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-2022 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 <memory>
49#include <optional>
50#include <stdexcept>
51#include <string>
52#include <utility>
53#include <variant>
54
56
57namespace traits {
58template <>
60} // namespace traits
61
62/** @brief Parameters for the ELC method */
63struct elc_data {
64 elc_data(double maxPWerror, double gap_size, double far_cut, bool neutralize,
65 double delta_top, double delta_bot, bool const_pot, double pot_diff);
66
67 /**
68 * @brief Maximal allowed pairwise error for the potential and force.
69 * Note that this counts for the plain 1/r contribution
70 * alone, without the prefactor and the charge prefactor.
71 */
72 double maxPWerror;
73 /** Size of the empty gap. Note that ELC relies on the user to make sure
74 * that this condition is fulfilled.
75 */
76 double gap_size;
77 /** Up to where particles can be found. */
78 double box_h;
79 /**
80 * @brief Cutoff of the exponential sum.
81 * Since in all other MMM methods this is the far formula,
82 * it is given the same name here.
83 */
84 double far_cut;
85 /** Squared value of #far_cut. */
86 double far_cut2;
87 /** Flag whether #far_cut was set by the user, or calculated by ESPResSo.
88 * In the latter case, the cutoff will be adapted if important parameters,
89 * such as the box dimensions, change.
90 */
92
93 /// @brief Flag whether there is any dielectric contrast in the system.
95 /// @brief Flag whether a constant potential difference is applied.
97 /**
98 * @brief Flag whether the box is neutralized by a homogeneous background.
99 * If true, use a homogeneous neutralizing background for non-neutral
100 * systems. Unlike the 3D case, this background adds an additional
101 * force pointing towards the system center, and the gap size
102 * enters into the value of the forces, so be careful with this.
103 */
105
106 /// dielectric contrast in the upper part of the simulation cell.
108 /// dielectric contrast in the lower part of the simulation cell.
110 /// @brief Constant potential difference.
111 double pot_diff;
112
113 /** Layer around the dielectric contrast in which we trick around. */
115 /** The space that is finally left. */
116 double space_box;
117
118#ifdef ESPRESSO_SHARED_MEMORY_PARALLELISM
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#endif // ESPRESSO_SHARED_MEMORY_PARALLELISM
143
144 /// pairwise contributions from lower and upper layers
146 Utils::Vector3d const &pos1,
147 Utils::Vector3d const &pos2, double q1q2,
148 auto &&kernel) const {
149 if (pos1[2] < space_layer) {
150 auto const q_eff = delta_mid_bot * q1q2;
151 auto const d = box_geo.get_mi_vector(pos2, {pos1[0], pos1[1], -pos1[2]});
152 kernel(q_eff, d);
153 }
154 if (pos1[2] > (box_h - space_layer)) {
155 auto const q_eff = delta_mid_top * q1q2;
156 auto const z = 2. * box_h - pos1[2];
157 auto const d = box_geo.get_mi_vector(pos2, {pos1[0], pos1[1], z});
158 kernel(q_eff, d);
159 }
160 }
161
162 /// self energies of top and bottom layers with their virtual images
164 BoxGeometry const &box_geo,
165 ParticleRange const &particles) const {
166 auto energy = 0.;
167 for (auto const &p : particles) {
169 box_geo, p.pos(), p.pos(), Utils::sqr(p.q()),
170 [&](double q1q2, Utils::Vector3d const &d) {
171 energy += p3m.pair_energy(q1q2, d.norm());
172 });
173 }
174 return energy;
175 }
176
177 /// forces of particles in border layers with themselves
179 BoxGeometry const &box_geo,
180 ParticleRange const &particles) const {
181 for (auto &p : particles) {
183 box_geo, p.pos(), p.pos(), Utils::sqr(p.q()),
184 [&](double q1q2, Utils::Vector3d const &d) {
185 p.force() += p3m.pair_force(q1q2, d, d.norm());
186 });
187 }
188 }
189};
190
192 : public Coulomb::Actor<ElectrostaticLayerCorrection> {
193 using BaseSolver = std::variant<std::shared_ptr<CoulombP3M>>;
194
197
198 /** Electrostatics solver that is adapted. */
200
201 ElectrostaticLayerCorrection(elc_data &&parameters, BaseSolver &&solver);
202
204 visit_base_solver(
205 [this](auto &solver) { solver->bind_system(m_system.lock()); });
206 sanity_checks_periodicity();
207 sanity_checks_cell_structure();
209 sanity_checks_dielectric_contrasts();
210 /* Most ELC parameters do not depend on the P3M parameters,
211 * but the P3M parameters depend on the ELC parameters during tuning,
212 * therefore ELC needs to be tuned before P3M. */
213 recalc_box_h();
214 recalc_far_cut();
215 visit_base_solver([](auto &solver) { solver->on_activation(); });
216 /* With dielectric contrasts, the ELC space layer depends
217 * on the P3M real-space cutoff, and the P3M FFT parameters
218 * depend on the ELC space layer */
220 recalc_space_layer();
221 visit_base_solver([](auto &actor) { actor->init(); });
222 }
223 }
224 /** @brief Recalculate all box-length-dependent parameters. */
226 visit_base_solver([](auto &actor) { actor->on_boxl_change(); });
227 recalc_box_h();
228 recalc_far_cut();
229 recalc_space_layer();
230 }
231 void on_node_grid_change() const {
232 visit_base_solver([](auto &solver) { solver->on_node_grid_change(); });
233 }
235 sanity_checks_periodicity();
236 visit_base_solver([](auto &solver) { solver->on_periodicity_change(); });
237 }
239 sanity_checks_cell_structure();
240 visit_base_solver([](auto &solver) { solver->on_cell_structure_change(); });
241 recalc_box_h();
242 recalc_far_cut();
244 recalc_space_layer();
245 visit_base_solver([](auto &actor) { actor->init(); });
246 }
247 }
248 /** @brief Recalculate all derived parameters. */
249 void init() {
250 visit_base_solver([](auto &actor) { actor->init(); });
251 recalc_box_h();
252 recalc_far_cut();
253 recalc_space_layer();
254 }
255
256 void sanity_checks() const {
257 sanity_checks_periodicity();
258 sanity_checks_cell_structure();
260 sanity_checks_dielectric_contrasts();
261 visit_base_solver([](auto &actor) { actor->sanity_checks(); });
262 }
263
264 /**
265 * @brief Veto real-space cutoff values that are incompatible with ELC.
266 * When ELC is used with dielectric contrasts, the short-range cutoff needs
267 * to be smaller than the gap size to allow placement of the image charges.
268 */
269 std::optional<std::string> veto_r_cut(double r_cut) const {
270 if (elc.dielectric_contrast_on and r_cut >= elc.gap_size) {
271 return {std::string("conflict with ELC w/ dielectric contrasts")};
272 }
273 return {};
274 }
275
276#ifdef ESPRESSO_SHARED_MEMORY_PARALLELISM
277 /** @brief Calculate short-range pair energy correction. */
278 double pair_energy_correction(std::size_t p1, std::size_t p2, auto &aosoa,
279 double q1q2) const {
280 double energy = 0.;
282 energy = std::visit(
283 [this, &aosoa, p1, p2, q1q2](auto &p3m_ptr) {
284 auto const &p3m = *p3m_ptr;
285 auto energy = 0.;
287 *m_box_geo, p1, p2, aosoa, q1q2,
288 [&](double q_eff, Utils::Vector3d const &d) {
289 energy += p3m.pair_energy(q_eff, d.norm());
290 });
292 *m_box_geo, p2, p1, aosoa, q1q2,
293 [&](double q_eff, Utils::Vector3d const &d) {
294 energy += p3m.pair_energy(q_eff, d.norm());
295 });
296 return energy / 2.;
297 },
299 }
300 return energy;
301 }
302#endif // ESPRESSO_SHARED_MEMORY_PARALLELISM
303
304 /** @brief Calculate short-range pair energy correction. */
306 Utils::Vector3d const &pos2,
307 double q1q2) const {
308 double energy = 0.;
310 energy = std::visit(
311 [this, &pos1, &pos2, q1q2](auto &p3m_ptr) {
312 auto const &p3m = *p3m_ptr;
313 auto energy = 0.;
315 *m_box_geo, pos1, pos2, q1q2,
316 [&](double q_eff, Utils::Vector3d const &d) {
317 energy += p3m.pair_energy(q_eff, d.norm());
318 });
320 *m_box_geo, pos2, pos1, q1q2,
321 [&](double q_eff, Utils::Vector3d const &d) {
322 energy += p3m.pair_energy(q_eff, d.norm());
323 });
324 return energy / 2.;
325 },
327 }
328 return energy;
329 }
330
331 /** @brief Add short-range pair force corrections. */
333 Utils::Vector3d const &pos2,
334 Utils::Vector3d &p1f_asym,
335 Utils::Vector3d &p2f_asym,
336 double q1q2) const {
338 std::visit(
339 [this, &pos1, &pos2, &p1f_asym, &p2f_asym, q1q2](auto &p3m_ptr) {
340 auto const &p3m = *p3m_ptr;
342 *m_box_geo, pos1, pos2, q1q2,
343 [&](double q_eff, Utils::Vector3d const &d) {
344 p1f_asym += p3m.pair_force(q_eff, d, d.norm());
345 });
347 *m_box_geo, pos2, pos1, q1q2,
348 [&](double q_eff, Utils::Vector3d const &d) {
349 p2f_asym += p3m.pair_force(q_eff, d, d.norm());
350 });
351 },
353 }
354 }
355
356 double long_range_energy(ParticleRange const &particles) const;
357 void add_long_range_forces(ParticleRange const &particles) const;
358
359private:
360 /** Check if a charged particle is in the gap region. */
361 void check_gap(Particle const &p) const;
362 double tune_far_cut() const;
363 void adapt_solver();
364 /** pairwise contributions from the lowest and top layers to the energy */
365 double dipole_energy(ParticleRange const &particles) const;
366 void add_dipole_force(ParticleRange const &particles) const;
367 double z_energy(ParticleRange const &particles) const;
368 void add_z_force(ParticleRange const &particles) const;
369
370 void recalc_box_h();
371 void recalc_far_cut() {
372 if (elc.far_calculated) {
373 elc.far_cut = tune_far_cut();
374 }
376 }
377 void recalc_space_layer();
378
379 void sanity_checks_cell_structure() const {}
380 void sanity_checks_periodicity() const;
381 void sanity_checks_dielectric_contrasts() const;
382
383 /// the force calculation
384 void add_force(ParticleRange const &particles) const;
385 /// the energy calculation
386 double calc_energy(ParticleRange const &particles) const;
387
388 void visit_base_solver(auto &&visitor) const {
389 std::visit(visitor, base_solver);
390 }
391};
392
393#endif // ESPRESSO_P3M
Vector implementation and trait types for boost qvm interoperability.
ESPRESSO_ATTR_ALWAYS_INLINE Utils::Vector< T, 3 > get_mi_vector(const Utils::Vector< T, 3 > &a, const Utils::Vector< T, 3 > &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:160
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:57
void on_boxl_change()
Recalculate all box-length-dependent parameters.
Definition elc.hpp:225
void on_periodicity_change() const
Definition elc.hpp:234
void sanity_checks() const
Definition elc.hpp:256
std::optional< std::string > veto_r_cut(double r_cut) const
Veto real-space cutoff values that are incompatible with ELC.
Definition elc.hpp:269
std::variant< std::shared_ptr< CoulombP3M > > BaseSolver
Definition elc.hpp:193
BaseSolver base_solver
Electrostatics solver that is adapted.
Definition elc.hpp:199
double pair_energy_correction(Utils::Vector3d const &pos1, Utils::Vector3d const &pos2, double q1q2) const
Calculate short-range pair energy correction.
Definition elc.hpp:305
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:332
void init()
Recalculate all derived parameters.
Definition elc.hpp:249
void on_node_grid_change() const
Definition elc.hpp:231
BoxGeometry * m_box_geo
Definition elc.hpp:196
void add_long_range_forces(ParticleRange const &particles) const
Definition elc.cpp:1245
double long_range_energy(ParticleRange const &particles) const
Definition elc.cpp:1203
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:278
Struct holding all information for one particle.
Definition Particle.hpp:450
Parameters for the ELC method.
Definition elc.hpp:63
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:163
double maxPWerror
Maximal allowed pairwise error for the potential and force.
Definition elc.hpp:72
double pot_diff
Constant potential difference.
Definition elc.hpp:111
double box_h
Up to where particles can be found.
Definition elc.hpp:78
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:94
double space_box
The space that is finally left.
Definition elc.hpp:116
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:145
bool neutralize
Flag whether the box is neutralized by a homogeneous background.
Definition elc.hpp:104
double far_cut
Cutoff of the exponential sum.
Definition elc.hpp:84
double space_layer
Layer around the dielectric contrast in which we trick around.
Definition elc.hpp:114
bool far_calculated
Flag whether far_cut was set by the user, or calculated by ESPResSo.
Definition elc.hpp:91
double gap_size
Size of the empty gap.
Definition elc.hpp:76
double delta_mid_bot
dielectric contrast in the lower part of the simulation cell.
Definition elc.hpp:109
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:178
bool const_pot
Flag whether a constant potential difference is applied.
Definition elc.hpp:96
double far_cut2
Squared value of far_cut.
Definition elc.hpp:86
double delta_mid_top
dielectric contrast in the upper part of the simulation cell.
Definition elc.hpp:107
Whether an actor is a layer correction method.
Definition traits.hpp:26