Octopus
poisson_cg.F90
Go to the documentation of this file.
1!! Cropyright (C) 2004-2006 M. Marques, A. Castro, A. Rubio, G. Bertsch
2!!
3!! This program is free software; you can redistribute it and/or modify
4!! it under the terms of the GNU General Public License as published by
5!! the Free Software Foundation; either version 2, or (at your option)
6!! any later version.
7!!
8!! This program is distributed in the hope that it will be useful,
9!! but WITHOUT ANY WARRANTY; without even the implied warranty of
10!! MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11!! GNU General Public License for more details.
12!!
13!! You should have received a copy of the GNU General Public License
14!! along with this program; if not, write to the Free Software
15!! Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
16!! 02110-1301, USA.
17!!
18
19#include "global.h"
20
22 use debug_oct_m
24 use global_oct_m
25 use iso_c_binding, only: c_loc
26 use, intrinsic :: iso_fortran_env
29 use mesh_oct_m
36
37 implicit none
38
39 private
40 public :: &
45
46 real(real64), public :: threshold
47 integer, public :: maxiter
48
49contains
50
51
52 ! ---------------------------------------------------------
53 subroutine poisson_cg_init(thr, itr)
54 integer, intent(in) :: itr
55 real(real64), intent(in) :: thr
56
57 push_sub(poisson_cg_init)
58 threshold = thr
59 maxiter = itr
60 pop_sub(poisson_cg_init)
61 end subroutine poisson_cg_init
62
63
64 ! ---------------------------------------------------------
65 subroutine poisson_cg_end
66
67 end subroutine poisson_cg_end
68
69
70 ! ---------------------------------------------------------
71 subroutine poisson_cg1(namespace, der, corrector, pot, rho)
72 type(namespace_t), intent(in) :: namespace
73 type(derivatives_t), target, intent(in) :: der
74 type(poisson_corr_t), intent(in) :: corrector
75 real(real64), contiguous, intent(inout) :: pot(:)
76 real(real64), intent(in) :: rho(:)
77
78 integer :: iter
79 real(real64) :: res
80 real(real64), allocatable :: wk(:), lwk(:), rhs(:), sol(:)
81
82 push_sub(poisson_cg1)
83
84 safe_allocate( wk(1:der%mesh%np_part))
85 safe_allocate(lwk(1:der%mesh%np_part))
86 safe_allocate(rhs(1:der%mesh%np_part))
87 safe_allocate(sol(1:der%mesh%np_part))
88
89 ! build initial guess for the potential
90 wk(1:der%mesh%np) = pot(1:der%mesh%np)
91 if (der%periodic_dim > 0) then
92 call dderivatives_lapl(der, wk, lwk, ghost_update = .true.)
93 else
94 call poisson_boundary_conditions(corrector, der%mesh, rho, wk)
95 call dderivatives_lapl(der, wk, lwk, ghost_update = .true., set_bc = .false.)
96 end if
97
98 ! Solving -\Delta = rhs, as CG require SDP operator
99 rhs(1:der%mesh%np) = (m_four*m_pi*rho(1:der%mesh%np) + lwk(1:der%mesh%np))
100 if (der%periodic_dim > 0) call dmf_remove_average(der%mesh, rhs)
101 safe_deallocate_a(wk)
102 safe_deallocate_a(lwk) ! they are no longer needed
103
104 der_pointer => der
105 mesh_pointer => der%mesh
106 mesh_aux => der%mesh
107 call lalg_copy(der%mesh%np, rhs, sol)
108 iter = maxiter
109 if (der%periodic_dim > 0) then
110 call dconjugate_gradients(der%mesh%np, sol, rhs, dlaplacian_op, internal_dotp, iter, res, &
111 threshold, userdata=[c_loc(der)], nullspace=dmf_remove_average_aux)
112 else
113 call dconjugate_gradients(der%mesh%np, sol, rhs, dlaplacian_op, internal_dotp, iter, res, threshold, userdata=[c_loc(der)])
114 end if
115 if (res >= threshold) then
116 message(1) = 'Conjugate-gradients Poisson solver did not converge.'
117 write(message(2), '(a,i8)') ' Iter = ',iter
118 write(message(3), '(a,e14.6)') ' Res = ', res
119 call messages_warning(3, namespace=namespace)
120 end if
121 nullify(der_pointer, mesh_pointer)
122 pot(1:der%mesh%np) = pot(1:der%mesh%np) + sol(1:der%mesh%np)
123
124 safe_deallocate_a(rhs)
125 safe_deallocate_a(sol)
126 pop_sub(poisson_cg1)
127 end subroutine poisson_cg1
128
129
130 ! ---------------------------------------------------------
131 subroutine poisson_cg2(namespace, der, pot, rho)
132 type(namespace_t), intent(in) :: namespace
133 type(derivatives_t), target, intent(in) :: der
134 real(real64), contiguous, intent(inout) :: pot(:)
135 real(real64), intent(in) :: rho(:)
136
137 integer :: iter, ip
138 real(real64), allocatable :: potc(:), rhs(:)
139 real(real64) :: res
140
141 push_sub(poisson_cg2)
143 iter = maxiter
144 der_pointer => der
145 mesh_pointer => der%mesh
146
147 safe_allocate(rhs(1:der%mesh%np))
148 safe_allocate(potc(1:der%mesh%np_part))
149
150 ! Solving -\Delta = rhs, as CG require SDP operator
151 do ip = 1, der%mesh%np
152 rhs(ip) = 4.0_real64*m_pi*rho(ip)
153 end do
154 call lalg_copy(der%mesh%np, pot, potc)
155
156 call dconjugate_gradients(der%mesh%np, potc, rhs, dlaplacian_op, internal_dotp, iter, res, threshold, userdata=[c_loc(der)])
157
158 if (res >= threshold) then
159 message(1) = 'Conjugate-gradients Poisson solver did not converge.'
160 write(message(2), '(a,i8)') ' Iter = ', iter
161 write(message(3), '(a,e14.6)') ' Res = ', res
162 call messages_warning(3, namespace=namespace)
163 end if
164
165 call lalg_copy(der%mesh%np, potc, pot)
167 nullify(der_pointer, mesh_pointer)
168
169 safe_deallocate_a(rhs)
170 safe_deallocate_a(potc)
171
172 pop_sub(poisson_cg2)
173 end subroutine poisson_cg2
174
175end module poisson_cg_oct_m
176
177!! Local Variables:
178!! mode: f90
179!! coding: utf-8
180!! End:
Copies a vector x, to a vector y.
Definition: lalg_basic.F90:188
This module calculates the derivatives (gradients, Laplacians, etc.) of a function.
subroutine, public dderivatives_lapl(der, ff, op_ff, ghost_update, set_bc, factor)
apply the Laplacian to a mesh function
real(real64), parameter, public m_four
Definition: global.F90:204
real(real64), parameter, public m_pi
some mathematical constants
Definition: global.F90:198
Computes and , suitable as an operator callback for iterative solvers (CG, QMR, etc....
subroutine, public dlaplacian_op(x, hx, userdata)
Computes the negative Laplacian operator action: .
This module defines various routines, operating on mesh functions.
class(mesh_t), pointer, public mesh_aux
Globally-scoped pointer to the mesh instance.
subroutine, public dmf_remove_average(mesh, ff, reduce)
Remove the average value of a mesh function.
subroutine, public dmf_remove_average_aux(f)
Interface for nullspace removal in linear solvers.
This module defines the meshes, which are used in Octopus.
Definition: mesh.F90:120
subroutine, public messages_warning(no_lines, all_nodes, namespace)
Definition: messages.F90:525
character(len=256), dimension(max_lines), public message
to be output by fatal, warning
Definition: messages.F90:162
subroutine, public poisson_cg2(namespace, der, pot, rho)
Definition: poisson_cg.F90:227
subroutine, public poisson_cg1(namespace, der, corrector, pot, rho)
Definition: poisson_cg.F90:167
subroutine, public poisson_cg_init(thr, itr)
Definition: poisson_cg.F90:149
subroutine, public poisson_cg_end
Definition: poisson_cg.F90:161
real(real64) function, public internal_dotp(xx, yy)
subroutine, public poisson_boundary_conditions(this, mesh, rho, pot)
type(mesh_t), pointer, public mesh_pointer
type(derivatives_t), pointer, public der_pointer
This module is intended to contain "only mathematical" functions and procedures.
Definition: solvers.F90:117
int true(void)