58 type(distributed_t) :: dist
61 integer,
parameter :: &
62 ION_COMPONENT_REAL = 1, &
70 type(ion_interaction_t),
intent(out) :: this
71 type(namespace_t),
intent(in) :: namespace
72 class(space_t),
intent(in) :: space
73 integer,
intent(in) :: natoms
87 call parse_variable(namespace,
'EwaldAlpha', 0.21_real64, this%alpha)
91 if (space%periodic_dim == 1)
then
92 call messages_write(
'For systems that are periodic in 1D, the interaction between', new_line = .
true.)
93 call messages_write(
'ions is not implemented. This affects the calculation', new_line = .
true.)
94 call messages_write(
'of total energy and forces, so both are zeroed.')
102 type(ion_interaction_t),
intent(inout) :: this
103 integer,
intent(in) :: natoms
104 type(multicomm_t),
intent(in) :: mc
120 type(ion_interaction_t),
intent(inout) :: this
136 energy_components, force_components)
137 type(ion_interaction_t),
intent(inout) :: this
138 class(space_t),
intent(in) :: space
139 type(lattice_vectors_t),
intent(in) :: latt
140 type(atom_t),
intent(in) :: atom(:)
141 integer,
intent(in) :: natoms
142 real(real64),
intent(in) :: pos(1:space%dim,1:natoms)
143 real(real64),
intent(in) :: lsize(:)
144 real(real64),
intent(out) :: energy
145 real(real64),
intent(out) :: force(:, :)
146 real(real64),
optional,
intent(out) :: energy_components(:)
147 real(real64),
optional,
intent(out) :: force_components(:, :, :)
153 if (
present(energy_components))
then
155 energy_components =
m_zero
158 if (
present(force_components))
then
167 if (space%is_periodic())
then
172 call ion_interaction_periodic(this, space, latt, atom, natoms, pos, energy, force, energy_components, force_components)
190 class(
space_t),
intent(in) :: space
191 type(
atom_t),
intent(in) :: atom(:)
192 real(real64),
intent(in) :: lsize(:)
193 real(real64) :: energy
198 assert(
size(atom) == 1)
200 assert(space%periodic_dim == 2)
202 select type(spec => atom(1)%species)
204 area = lsize(1) * lsize(2) *
m_four
205 energy =
m_pi * spec%get_density(lsize) **2 * area * spec%thickness()**3 /
m_three
227 type(
atom_t),
intent(in) :: atom(:)
228 real(real64),
intent(in) :: lsize(:)
229 real(real64) :: energy
233 logical :: lattice_is_orthogonal
239 lattice_is_orthogonal = .not. latt%nonorthogonal
241 do iatom = dist%start, dist%end
242 spec => atom(iatom)%species
253 assert(lattice_is_orthogonal)
254 energy = energy +
m_pi * zi**2 / (
m_four * lsize(1)*lsize(2)) * spec%thickness() /
m_three
269 class(
space_t),
intent(in) :: space
270 type(
atom_t),
intent(in) :: atom(:)
271 real(real64),
intent(in) :: pos(:,:)
272 real(real64),
intent(in) :: lsize(:)
273 real(real64),
intent(out) :: energy
274 real(real64),
intent(out) :: force(:, :)
276 class(
species_t),
pointer :: species_i, species_j
277 real(real64) :: r(space%dim),
f(space%dim)
278 real(real64) :: r_mag
280 real(real64) :: zi, zj
281 integer :: iatom, jatom, natoms
287 force(1:space%dim, 1:natoms) =
m_zero
289 do iatom = dist%start, dist%end
290 species_i => atom(iatom)%species
291 zi = species_i%get_zval()
293 do jatom = iatom + 1, natoms
294 species_j => atom(jatom)%species
295 zj = species_j%get_zval()
297 r = pos(:, iatom) - pos(:, jatom)
299 u_e = zi * zj / r_mag
301 energy = energy + u_e
302 f(1:space%dim) = (u_e / r_mag**2) * r(1:space%dim)
303 force(1:space%dim, iatom) = force(1:space%dim, iatom) +
f(1:space%dim)
304 force(1:space%dim, jatom) = force(1:space%dim, jatom) -
f(1:space%dim)
311 nullify(species_i, species_j)
319 energy_components, force_components)
321 class(
space_t),
intent(in) :: space
323 type(
atom_t),
intent(in) :: atom(:)
324 integer,
intent(in) :: natoms
325 real(real64),
intent(in) :: pos(1:space%dim,1:natoms)
326 real(real64),
intent(out) :: energy
327 real(real64),
intent(out) :: force(:, :)
328 real(real64),
optional,
intent(out) :: energy_components(:)
329 real(real64),
optional,
intent(out) :: force_components(:, :, :)
331 real(real64) :: ereal, efourier, epseudo, eself
332 real(real64) :: charge
337 force(1:space%dim, 1:natoms) =
m_zero
339 call ewald_short(this%dist, space, latt, atom, pos, this%alpha, ereal, force)
340 if (
present(force_components))
then
341 force_components(1:space%dim, 1:natoms, ion_component_real) = force(1:space%dim, 1:natoms)
347 select case (space%periodic_dim)
356 if (
present(force_components))
then
357 force_components(1:space%dim, 1:natoms, ion_component_real) =
m_zero
364 call ewald_long_2d(this, space, latt, atom, natoms, pos, efourier, force)
366 call ewald_long_3d(this, space, latt, atom, natoms, pos, efourier, force, charge)
372 if (
present(energy_components))
then
373 energy_components(ion_component_real) = ereal
378 if (
present(force_components))
then
381 force(1:space%dim, 1:natoms) - force_components(1:space%dim, 1:natoms, ion_component_real)
384 energy = ereal + efourier + eself + epseudo
411 subroutine ewald_short(dist, space, latt, atom, pos, alpha, ereal, force)
415 type(
atom_t),
intent(in) :: atom(:)
416 real(real64),
intent(in) :: pos(:, :)
418 real(real64),
intent(in) :: alpha
419 real(real64),
intent(out) :: ereal
420 real(real64),
intent(inout) :: force(:, :)
422 integer :: iatom, jatom, icopy, natoms
423 real(real64) :: rnorm, xi(space%dim)
424 real(real64) :: force_real(space%dim)
425 real(real64) :: zi, zj
426 real(real64) :: erfc_rnorm
429 real(real64) :: charge, coeff
435 rcut = 6.0_real64 / alpha
440 do iatom = dist%start, dist%end
441 if (.not. atom(iatom)%species%represents_real_atom()) cycle
442 zi = atom(iatom)%species%get_zval()
443 charge = charge + zi**2
448 do icopy = 1, latt_iter%n_cells
449 rnorm = norm2(latt_iter%get(icopy))
451 if (rnorm > rcut) cycle
452 ereal = ereal +
m_half * charge * erfc(alpha * rnorm) /rnorm
460 do iatom = dist%start, dist%end
461 if (.not. atom(iatom)%species%represents_real_atom()) cycle
462 zi = atom(iatom)%species%get_zval()
465 do jatom = iatom + 1, natoms
466 zj = atom(jatom)%species%get_zval()
471 do icopy = 1, latt_iter%n_cells
472 xi = pos(:, iatom) + latt_iter%get(icopy)
473 rnorm = norm2(xi - pos(:, jatom))
474 if (rnorm > rcut) cycle
476 erfc_rnorm = erfc(alpha * rnorm) / rnorm
479 ereal = ereal + charge * erfc_rnorm
481 force_real(:) = charge * (xi - pos(:, jatom)) * &
482 (erfc_rnorm + coeff *
exp(-(alpha*rnorm)**2)) / rnorm**2
485 force(1:space%dim, jatom) = force(1:space%dim, jatom) - force_real
488 force(1:space%dim, iatom) = force(1:space%dim, iatom) + force_real
508 type(
atom_t),
intent(in) :: atom(:)
509 real(real64),
intent(in) :: alpha
510 real(real64),
intent(out) :: eself
511 real(real64),
intent(out) :: charge
521 do iatom = dist%start, dist%end
522 zi = atom(iatom)%species%get_zval()
524 eself = eself - alpha /
sqrt(
m_pi) * zi**2
534 subroutine ewald_long_3d(this, space, latt, atom, natoms, pos, efourier, force, charge)
536 class(
space_t),
intent(in) :: space
538 type(
atom_t),
intent(in) :: atom(:)
539 integer,
intent(in) :: natoms
540 real(real64),
intent(in) :: pos(:,:)
541 real(real64),
intent(inout) :: efourier
542 real(real64),
intent(inout) :: force(:, :)
543 real(real64),
intent(in) :: charge
545 real(real64) :: rcut, gmax_squared
547 integer :: ix, iy, iz, isph
548 real(real64) :: gvec(3), gred(3), gg2, gx
549 real(real64) :: factor
550 complex(real64) :: sumatoms, tmp(3), aa
552 complex(real64),
allocatable :: phase(:)
556 assert(space%dim == 3)
557 assert(space%periodic_dim == 3)
560 safe_allocate(phase(1:natoms))
563 rcut =
sqrt(minval(sum(latt%klattice**2, dim=1)))
566 isph = ceiling(9.5_real64*this%alpha/rcut)
569 efourier = -
m_pi*charge**2/(
m_two*this%alpha**2*latt%rcell_volume)
572 gmax_squared = isph**2 * minval(sum(latt%klattice**2, dim=1))
582 if (ix == 0 .and. iy < 0) cycle
583 if (ix == 0 .and. iy == 0 .and. iz <= 0) cycle
587 gg2 = dot_product(gvec, gvec)
589 if (gg2 > gmax_squared*1.001_real64) cycle
591 gx = -0.25_real64*gg2/this%alpha**2
593 if (gx < -36.0_real64) cycle
598 if (factor < epsilon(factor)) cycle
603 gx = sum(gvec*pos(:,iatom))
604 aa = atom(iatom)%species%get_zval()*cmplx(
cos(gx),
sin(gx), real64)
606 sumatoms = sumatoms + aa
609 efourier = efourier + factor * real(sumatoms*conjg(sumatoms), real64)
612 tmp =
m_zi*gvec*phase(iatom)
613 force(1:space%dim, iatom) = force(1:space%dim, iatom) - factor*real(conjg(tmp)*sumatoms + tmp*conjg(sumatoms), real64)
621 safe_deallocate_a(phase)
630 subroutine ewald_long_2d(this, space, latt, atom, natoms, pos, efourier, force)
632 class(
space_t),
intent(in) :: space
634 type(
atom_t),
intent(in) :: atom(:)
635 integer,
intent(in) :: natoms
636 real(real64),
intent(in) :: pos(1:space%dim,1:natoms)
637 real(real64),
intent(inout) :: efourier
638 real(real64),
intent(inout) :: force(:, :)
640 real(real64) :: rcut, gmax_squared
641 integer :: iatom, jatom
642 integer :: ix, iy, ix_max, iy_max
643 real(real64) :: gvec(space%dim), gg2, gx, gg_abs
644 real(real64) :: factor,factor1,factor2, coeff
645 real(real64) :: dz_max, dz_ij, erfc1, erfc2, tmp_erf
646 real(real64),
allocatable :: force_tmp(:,:)
647 real(real64),
parameter :: tol = 1e-10_real64
651 assert(space%periodic_dim == 2)
652 assert(space%dim == 2 .or. space%dim == 3)
657 if (space%dim == 3)
then
660 do jatom = iatom + 1, natoms
661 dz_max = max(dz_max, abs(pos(3, iatom) - pos(3, jatom)))
670 rcut =
m_two*this%alpha*4.6_real64 +
m_two*this%alpha**2*dz_max
671 if (dz_max > tol)
then
674 erfc1 = erfc(this%alpha*dz_max +
m_half*rcut/this%alpha)
675 if (erfc1 *
exp(rcut*dz_max) < 1.e-10_real64)
exit
676 rcut = rcut * 1.414_real64
680 ix_max = ceiling(rcut/norm2(latt%klattice(:, 1)))
681 iy_max = ceiling(rcut/norm2(latt%klattice(:, 2)))
683 safe_allocate(force_tmp(1:space%dim, 1:natoms))
688 factor =
m_pi/latt%rcell_volume
691 do iatom = this%dist%start, this%dist%end
694 if (space%dim == 3)
then
695 dz_ij = pos(3, iatom) - pos(3, jatom)
700 tmp_erf = erf(this%alpha*dz_ij)
701 factor1 = dz_ij*tmp_erf
702 factor2 =
exp(-(this%alpha*dz_ij)**2)/(this%alpha*
sqrt(
m_pi))
704 efourier = efourier - factor &
705 * atom(iatom)%species%get_zval()*atom(jatom)%species%get_zval() * (factor1 + factor2)
708 if (iatom == jatom)cycle
711 if (space%dim == 3)
then
712 force_tmp(3, iatom) = force_tmp(3, iatom) - (-
m_two*factor) &
713 * atom(iatom)%species%get_zval()*atom(jatom)%species%get_zval() * tmp_erf
720 gmax_squared = sum(ix_max*latt%klattice(:, 1)**2)
721 gmax_squared = min(gmax_squared, sum(iy_max*latt%klattice(:, 2)**2))
725 do ix = -ix_max, ix_max
726 do iy = -iy_max, iy_max
728 gvec = ix*latt%klattice(:, 1) + iy*latt%klattice(:, 2)
732 if (gg2 <
m_epsilon .or. gg2 > gmax_squared*1.001_real64) cycle
734 factor =
m_half*
m_pi/(latt%rcell_volume*gg_abs)
736 do iatom = this%dist%start, this%dist%end
737 do jatom = iatom, natoms
739 gx = sum(gvec(1:2) * (pos(1:2, iatom) - pos(1:2, jatom)))
740 gx = gvec(1)*(pos(1, iatom) - pos(1, jatom)) + gvec(2)*(pos(2, iatom) - pos(2, jatom))
741 if (space%dim == 3)
then
742 dz_ij = pos(3, iatom) - pos(3, jatom)
747 erfc1 = erfc(this%alpha*dz_ij +
m_half*gg_abs/this%alpha)
749 factor1 =
exp(gg_abs*dz_ij)*erfc1
753 erfc2 = erfc(-this%alpha*dz_ij +
m_half*gg_abs/this%alpha)
755 factor2 =
exp(-gg_abs*dz_ij)*erfc2
760 if (iatom == jatom)
then
766 efourier = efourier &
768 * atom(iatom)%species%get_zval()*atom(jatom)%species%get_zval() &
769 *
cos(gx)* ( factor1 + factor2)
772 if (iatom == jatom) cycle
774 force_tmp(1:2, iatom) = force_tmp(1:2, iatom) &
775 +
m_two * factor * gvec(1:2) &
776 * atom(iatom)%species%get_zval()*atom(jatom)%species%get_zval() &
777 *
sin(gx)*(factor1 + factor2)
779 force_tmp(1:2, jatom) = force_tmp(1:2, jatom) &
780 -
m_two * factor * gvec(1:2) &
781 * atom(iatom)%species%get_zval()*atom(jatom)%species%get_zval() &
782 *
sin(gx)*(factor1 + factor2)
784 factor1 = gg_abs*erfc1 &
787 factor1 = factor1*
exp(gg_abs*dz_ij)
792 factor2 = gg_abs*erfc2 &
795 factor2 = factor2*
exp(-gg_abs*dz_ij)
800 if (space%dim == 3)
then
801 force_tmp(3, iatom) = force_tmp(3, iatom) &
803 * atom(iatom)%species%get_zval()*atom(jatom)%species%get_zval() &
804 *
cos(gx)* ( factor1 - factor2)
805 force_tmp(3, jatom) = force_tmp(3, jatom) &
807 * atom(iatom)%species%get_zval()*atom(jatom)%species%get_zval() &
808 *
cos(gx)* ( factor1 - factor2)
821 force = force + force_tmp
823 safe_deallocate_a(force_tmp)
838 type(
atom_t),
intent(in) :: atom(:)
839 real(real64),
intent(in) :: charge
840 real(real64),
intent(out) :: epseudo
848 do iatom = dist%start, dist%end
849 select type(spec => atom(iatom)%species)
852 epseudo = epseudo +
m_pi *zi * &
853 (spec%ps%sigma_erf *
sqrt(
m_two))**2 / latt%rcell_volume * charge
865 class(
space_t),
intent(in) :: space
867 type(
atom_t),
intent(in) :: atom(:)
868 integer,
intent(in) :: natoms
869 real(real64),
intent(in) :: pos(1:space%dim,1:natoms)
870 real(real64),
intent(out) :: stress_ii(space%dim, space%dim)
872 real(real64) :: stress_short(1:space%dim, 1:space%dim), stress_Ewald(1:space%dim, 1:space%dim)
879 assert(space%is_periodic())
885 select case(space%periodic_dim)
887 call ewald_3d_stress(this, space, latt, atom, natoms, pos, stress_ewald)
889 call ewald_2d_stress(this, space, latt, atom, natoms, pos, stress_ewald)
894 stress_ii = stress_short + stress_ewald
920 class(
space_t),
intent(in) :: space
922 type(
atom_t),
intent(in) :: atom(:)
923 integer,
intent(in) :: natoms
924 real(real64),
intent(in) :: pos(1:space%dim,1:natoms)
925 real(real64),
intent(out) :: stress_short(1:space%dim, 1:space%dim)
927 real(real64) :: xi(space%dim)
928 real(real64) :: r_ij, zi, zj, Hp, factor
929 integer :: iatom, jatom, icopy, idir, jdir
930 real(real64) :: alpha, rcut
937 assert(space%is_periodic())
942 rcut = 6.0_real64/alpha
948 do iatom = this%dist%start, this%dist%end
949 select type(spec => atom(iatom)%species)
953 zi = atom(iatom)%species%get_zval()
955 do icopy = 1, latt_iter%n_cells
956 xi = pos(:, iatom) + latt_iter%get(icopy)
959 zj = atom(jatom)%species%get_zval()
960 r_ij = norm2(xi - pos(:, jatom))
964 hp = -
m_two/
sqrt(
m_pi)*
exp(-(alpha*r_ij)**2) - erfc(alpha*r_ij)/(alpha*r_ij)
965 factor =
m_half*zj*zi*alpha*hp
966 do idir = 1, space%periodic_dim
967 do jdir = 1, space%periodic_dim
968 stress_short(idir, jdir) = stress_short(idir, jdir) &
969 - factor*(xi(idir) - pos(idir, jatom))*(xi(jdir) - pos(jdir, jatom))/(r_ij**2)
977 if (this%dist%parallel)
then
981 stress_short = stress_short/latt%rcell_volume
1005 subroutine ewald_3d_stress(this, space, latt, atom, natoms, pos, stress_Ewald)
1007 class(
space_t),
intent(in) :: space
1009 type(
atom_t),
intent(in) :: atom(:)
1010 integer,
intent(in) :: natoms
1011 real(real64),
intent(in) :: pos(1:space%dim,1:natoms)
1012 real(real64),
intent(out) :: stress_ewald(3, 3)
1014 real(real64) :: zi, rcut, gmax_squared
1016 integer :: ix, iy, iz, isph, idim, idir, jdir
1017 real(real64) :: gred(3), gvec(3), gg2, gx
1018 real(real64) :: factor, charge, charge_sq, off_diagonal_weight
1019 complex(real64) :: sumatoms, aa
1025 assert(space%dim == 3)
1026 assert(space%periodic_dim == 3)
1035 do iatom = 1, natoms
1036 zi = atom(iatom)%species%get_zval()
1037 charge = charge + zi
1038 charge_sq = charge_sq + zi**2
1043 do idim = 1, space%periodic_dim
1044 rcut = min(rcut, sum(latt%klattice(1:space%periodic_dim, idim)**2))
1049 isph = ceiling(9.5_real64*this%alpha/rcut)
1052 gmax_squared = isph**2 * minval(sum(latt%klattice**2, dim=1))
1062 if (ix == 0 .and. iy < 0) cycle
1063 if (ix == 0 .and. iy == 0 .and. iz <= 0) cycle
1070 if (gg2 > gmax_squared*1.001_real64) cycle
1072 gx = -0.25_real64*gg2/this%alpha**2
1074 if (gx < -36.0_real64) cycle
1079 if (factor < epsilon(factor)) cycle
1083 do iatom = 1, natoms
1084 gx = sum(gvec*pos(:, iatom))
1085 aa = atom(iatom)%species%get_zval()*cmplx(
cos(gx),
sin(gx), real64)
1086 sumatoms = sumatoms + aa
1089 factor = factor*abs(sumatoms)**2
1090 off_diagonal_weight = -
m_two*factor/gg2*(0.25_real64*gg2/this%alpha**2+
m_one)
1094 stress_ewald(idir, jdir) = stress_ewald(idir, jdir) &
1095 + gvec(idir) * gvec(jdir) * off_diagonal_weight
1097 stress_ewald(idir, idir) = stress_ewald(idir, idir) + factor
1106 factor =
m_half*
m_pi*charge**2/(latt%rcell_volume*this%alpha**2)
1108 stress_ewald(idir,idir) = stress_ewald(idir,idir) - factor
1111 stress_ewald = stress_ewald / latt%rcell_volume
1135 subroutine ewald_2d_stress(this, space, latt, atom, natoms, pos, stress_Ewald)
1137 type(
space_t),
intent(in) :: space
1139 type(
atom_t),
intent(in) :: atom(:)
1140 integer,
intent(in) :: natoms
1141 real(real64),
intent(in) :: pos(1:space%dim,1:natoms)
1142 real(real64),
intent(out) :: stress_Ewald(3, 3)
1144 real(real64) :: rcut, efourier
1145 integer :: iatom, jatom, idir, jdir
1146 integer :: ix, iy, ix_max, iy_max
1147 real(real64) :: gvec(3), gred(3), gg2, cos_gx, gg_abs, gmax_squared
1148 real(real64) :: factor,factor1,factor2, coeff, e_ewald
1149 real(real64) :: dz_max, z_ij, erfc1, erfc2, diff(3)
1150 real(real64),
parameter :: tol = 1e-10_real64
1154 assert(space%periodic_dim == 2)
1155 assert(space%dim == 3)
1161 do iatom = 1, natoms
1162 do jatom = iatom + 1, natoms
1163 dz_max = max(dz_max, abs(pos(3, iatom) - pos(3, jatom)))
1169 rcut =
m_two*this%alpha*4.6_real64 +
m_two*this%alpha**2*dz_max
1170 if (dz_max > tol)
then
1173 erfc1 = erfc(this%alpha*dz_max +
m_half*rcut/this%alpha)
1174 if (erfc1 *
exp(rcut*dz_max) < tol)
exit
1175 rcut = rcut * 1.414_real64
1181 factor =
m_pi/latt%rcell_volume
1183 do iatom = 1, natoms
1184 do jatom = 1, natoms
1185 z_ij = pos(3, iatom) - pos(3, jatom)
1187 factor1 = z_ij * erf(this%alpha*z_ij)
1188 factor2 =
exp(-(this%alpha*z_ij)**2)/(this%alpha*
sqrt(
m_pi))
1190 efourier = efourier - factor &
1191 * atom(iatom)%species%get_zval()*atom(jatom)%species%get_zval() * (factor1 + factor2)
1197 stress_ewald(idir, idir) = efourier
1201 ix_max = ceiling(rcut/norm2(latt%klattice(:, 1)))
1202 iy_max = ceiling(rcut/norm2(latt%klattice(:, 2)))
1203 gmax_squared = sum(ix_max*latt%klattice(:, 1)**2)
1204 gmax_squared = min(gmax_squared, sum(iy_max*latt%klattice(:, 2)**2))
1209 do ix = -ix_max, ix_max
1210 do iy = -iy_max, iy_max
1214 gg2 = dot_product(gvec,gvec)
1217 if (gg2 <
m_epsilon .or. gg2 > gmax_squared*1.001_real64) cycle
1220 factor =
m_fourth*
m_pi/(latt%rcell_volume*this%alpha*gg2)
1222 do iatom = 1, natoms
1223 do jatom = iatom, natoms
1224 diff = pos(:, iatom) - pos(:, jatom)
1225 cos_gx =
cos(sum(gvec(1:2) * diff(1:2)))
1231 if (iatom == jatom)
then
1239 stress_ewald(idir, jdir) = stress_ewald(idir, jdir) &
1240 - factor*gvec(idir)*gvec(jdir) * cos_gx * (factor1 + factor2) * coeff&
1241 * atom(iatom)%species%get_zval()*atom(jatom)%species%get_zval()
1246 factor1 =
exp(-gg_abs*z_ij)*erfc1
1251 factor2 =
exp(gg_abs*z_ij)*erfc2
1256 e_ewald =
m_half *
m_pi/latt%rcell_volume * coeff &
1257 * atom(iatom)%species%get_zval() * atom(jatom)%species%get_zval() &
1258 * cos_gx / gg_abs * (factor1 + factor2)
1261 stress_ewald(idir, idir) = stress_ewald(idir, idir) + e_ewald
1271 stress_ewald = stress_ewald / latt%rcell_volume
1278 real(real64) function screening_function_2d(alpha, z_ij, gg_abs, erfc_arg) result(factor)
1279 real(real64),
intent(in) :: alpha
1280 real(real64),
intent(in) :: z_ij
1281 real(real64),
intent(in) :: gg_abs
1282 real(real64),
intent(out) :: erfc_arg
1286 arg = -alpha*z_ij +
m_half*gg_abs/alpha
1287 erfc_arg = erfc(arg)
1289 factor = factor*
exp(-gg_abs*z_ij)
1297 class(space_t),
intent(in) :: space
1298 type(lattice_vectors_t),
intent(in) :: latt
1299 type(atom_t),
intent(in) :: atom(:)
1300 integer,
intent(in) :: natoms
1301 real(real64),
intent(in) :: pos(1:space%dim,1:natoms)
1302 real(real64),
intent(in) :: lsize(:)
1303 type(namespace_t),
intent(in) :: namespace
1304 type(multicomm_t),
intent(in) :: mc
1307 real(real64) :: energy
1308 real(real64),
allocatable :: force(:, :), force_components(:, :, :)
1309 real(real64) :: energy_components(1:ION_NUM_COMPONENTS)
1310 integer :: iatom, idir
1317 safe_allocate(force(1:space%dim, 1:natoms))
1318 safe_allocate(force_components(1:space%dim, 1:natoms, 1:ion_num_components))
1321 energy_components = energy_components, force_components = force_components)
1323 call messages_write(
'Ionic energy =')
1324 call messages_write(energy, fmt =
'(f20.10)')
1325 call messages_info(namespace=namespace)
1327 call messages_write(
'Real space energy =')
1329 call messages_info(namespace=namespace)
1331 call messages_write(
'Self energy =')
1333 call messages_info(namespace=namespace)
1335 call messages_write(
'Fourier energy =')
1337 call messages_info(namespace=namespace)
1339 call messages_info(namespace=namespace)
1341 do iatom = 1, natoms
1342 call messages_write(
'Ionic force atom')
1343 call messages_write(iatom)
1344 call messages_write(
' =')
1345 do idir = 1, space%dim
1346 call messages_write(force(idir, iatom), fmt =
'(f20.10)')
1348 call messages_info(namespace=namespace)
1350 call messages_write(
'Real space force atom')
1351 call messages_write(iatom)
1352 call messages_write(
' =')
1353 do idir = 1, space%dim
1354 call messages_write(force_components(idir, iatom,
ion_component_real), fmt =
'(f20.10)')
1356 call messages_info(namespace=namespace)
1358 call messages_write(
'Fourier space force atom')
1359 call messages_write(iatom)
1360 call messages_write(
' =')
1361 do idir = 1, space%dim
1364 call messages_info(namespace=namespace)
1366 call messages_info(namespace=namespace)
1369 safe_deallocate_a(force)
1370 safe_deallocate_a(force_components)
double exp(double __x) __attribute__((__nothrow__
double sin(double __x) __attribute__((__nothrow__
double sqrt(double __x) __attribute__((__nothrow__
double cos(double __x) __attribute__((__nothrow__
pure logical function, public all_species_are_jellium_slab(atom)
Check if all species are jellium slab.
pure logical function, public any_species_is_jellium_sphere(atom)
Check if any species is a jellium sphere.
type(debug_t), save, public debug
subroutine, public distributed_end(this)
subroutine, public distributed_nullify(this, total)
subroutine, public distributed_init(this, total, comm, tag, scalapack_compat)
Distribute N instances across M processes of communicator comm
real(real64), parameter, public m_two
real(real64), parameter, public m_max_exp_arg
real(real64), parameter, public m_zero
real(real64), parameter, public m_four
real(real64), parameter, public m_pi
some mathematical constants
real(real64), parameter, public m_fourth
complex(real64), parameter, public m_z0
complex(real64), parameter, public m_zi
real(real64), parameter, public r_min_atom_dist
Minimal distance between two distinguishable atoms.
real(real64), parameter, public m_epsilon
real(real64), parameter, public m_half
real(real64), parameter, public m_one
real(real64), parameter, public m_three
real(real64), parameter, public m_five
subroutine, public ion_interaction_stress(this, space, latt, atom, natoms, pos, stress_ii)
Computes the contribution to the stress tensor the ion-ion energy.
subroutine, public ion_interaction_init_parallelization(this, natoms, mc)
integer, parameter ion_component_self
real(real64) function jellium_slab_energy_periodic(space, atom, lsize)
Electrostatic energy of a periodic jellium slab.
subroutine, public ion_interaction_test(space, latt, atom, natoms, pos, lsize, namespace, mc)
subroutine ewald_long_2d(this, space, latt, atom, natoms, pos, efourier, force)
Computes the long-range part of the 2D Ewald summation.
real(real64) function screening_function_2d(alpha, z_ij, gg_abs, erfc_arg)
Auxiliary function for the Ewald 2D stress.
subroutine ion_interaction_stress_short(this, space, latt, atom, natoms, pos, stress_short)
Computes the short-range contribution to the stress tensor the ion-ion energy.
subroutine ion_interaction_periodic(this, space, latt, atom, natoms, pos, energy, force, energy_components, force_components)
Total Ewald electrostatic energy and forces, for 1D, 2D and 3D systems.
real(real64) function jellium_self_energy_finite(dist, latt, atom, lsize)
Electrostatic self-interaction for jellium instances, with orthogonal cells.
subroutine, public ion_interaction_init(this, namespace, space, natoms)
subroutine ewald_short(dist, space, latt, atom, pos, alpha, ereal, force)
Short range component of the Ewald electrostatic energy and force.
subroutine pseudopotential_correction_3d(dist, latt, atom, charge, epseudo)
G=0 component of Ewald energy arising from the pseudopotentials, for 3D systems.
subroutine ewald_long_3d(this, space, latt, atom, natoms, pos, efourier, force, charge)
Computes the long-range part of the 3D Ewald summation.
integer, parameter ion_component_real
integer, parameter ion_num_components
subroutine ewald_3d_stress(this, space, latt, atom, natoms, pos, stress_Ewald)
Computes the contribution to the stress tensor from the 3D Ewald sum.
integer, parameter ion_component_fourier
subroutine ion_interaction_finite(dist, space, atom, pos, lsize, energy, force)
Electrostatic Ewald energy and forces for finite systems.
subroutine, public ion_interaction_end(this)
subroutine, public ion_interaction_calculate(this, space, latt, atom, natoms, pos, lsize, energy, force, energy_components, force_components)
Top level routine for computing electrostatic energies and forces between ions.
subroutine ewald_2d_stress(this, space, latt, atom, natoms, pos, stress_Ewald)
Computes the contribution to the stress tensor from the 2D Ewald sum.
subroutine ewald_self_interaction(dist, atom, alpha, eself, charge)
@ brief Ewald self-interaction energy
subroutine, public kpoints_to_absolute(latt, kin, kout)
subroutine, public messages_not_implemented(feature, namespace)
subroutine, public messages_warning(no_lines, all_nodes, namespace)
This module handles the communicators for the various parallelization strategies.
subroutine, public profiling_out(label)
Increment out counter and sum up difference between entry and exit time.
subroutine, public profiling_in(label, exclude)
Increment in counter and save entry time.
static double f(double w, void *p)
Distribution of N instances over mpi_grpsize processes, for the local rank mpi_grprank....
The following class implements a lattice iterator. It allows one to loop over all cells that are with...
An abstract class for species. Derived classes include jellium, all electron, and pseudopotential spe...