Line data Source code
1 : !--------------------------------------------------------------------------------------------------!
2 : ! CP2K: A general program to perform molecular dynamics simulations !
3 : ! Copyright 2000-2025 CP2K developers group <https://cp2k.org> !
4 : ! !
5 : ! SPDX-License-Identifier: GPL-2.0-or-later !
6 : !--------------------------------------------------------------------------------------------------!
7 :
8 : ! **************************************************************************************************
9 : !> \brief Calls routines to get RI integrals and calculate total energies
10 : !> \par History
11 : !> 10.2011 created [Joost VandeVondele and Mauro Del Ben]
12 : !> 07.2019 split from mp2_gpw.F [Frederick Stein]
13 : ! **************************************************************************************************
14 : MODULE mp2_gpw
15 : USE atomic_kind_types, ONLY: atomic_kind_type
16 : USE basis_set_types, ONLY: get_gto_basis_set,&
17 : gto_basis_set_p_type,&
18 : gto_basis_set_type
19 : USE cell_types, ONLY: cell_type,&
20 : get_cell
21 : USE cp_blacs_env, ONLY: BLACS_GRID_SQUARE,&
22 : cp_blacs_env_create,&
23 : cp_blacs_env_release,&
24 : cp_blacs_env_type
25 : USE cp_control_types, ONLY: dft_control_type
26 : USE cp_dbcsr_api, ONLY: &
27 : dbcsr_clear_mempools, dbcsr_copy, dbcsr_create, dbcsr_distribution_release, &
28 : dbcsr_distribution_type, dbcsr_filter, dbcsr_init_p, dbcsr_iterator_blocks_left, &
29 : dbcsr_iterator_next_block, dbcsr_iterator_start, dbcsr_iterator_stop, dbcsr_iterator_type, &
30 : dbcsr_p_type, dbcsr_release, dbcsr_type, dbcsr_type_no_symmetry, dbcsr_type_symmetric
31 : USE cp_dbcsr_contrib, ONLY: dbcsr_reserve_all_blocks
32 : USE cp_dbcsr_cp2k_link, ONLY: cp_dbcsr_alloc_block_from_nbl
33 : USE cp_dbcsr_operations, ONLY: cp_dbcsr_dist2d_to_dist,&
34 : cp_dbcsr_m_by_n_from_row_template
35 : USE cp_fm_types, ONLY: cp_fm_get_info,&
36 : cp_fm_release,&
37 : cp_fm_type
38 : USE cp_log_handling, ONLY: &
39 : cp_add_default_logger, cp_get_default_logger, cp_logger_create, &
40 : cp_logger_get_default_unit_nr, cp_logger_release, cp_logger_set, cp_logger_type, &
41 : cp_rm_default_logger, cp_to_string
42 : USE dbt_api, ONLY: dbt_type
43 : USE distribution_1d_types, ONLY: distribution_1d_release,&
44 : distribution_1d_type
45 : USE distribution_2d_types, ONLY: distribution_2d_release,&
46 : distribution_2d_type
47 : USE distribution_methods, ONLY: distribute_molecules_1d,&
48 : distribute_molecules_2d
49 : USE group_dist_types, ONLY: create_group_dist,&
50 : get_group_dist,&
51 : group_dist_d1_type,&
52 : release_group_dist
53 : USE hfx_types, ONLY: block_ind_type,&
54 : hfx_compression_type
55 : USE input_constants, ONLY: &
56 : do_eri_gpw, do_eri_os, do_potential_coulomb, do_potential_id, do_potential_truncated, &
57 : eri_default, mp2_method_gpw, ri_default, ri_mp2_method_gpw, rpa_exchange_none
58 : USE input_section_types, ONLY: section_vals_val_get
59 : USE kinds, ONLY: dp
60 : USE kpoint_types, ONLY: kpoint_type
61 : USE libint_wrapper, ONLY: cp_libint_static_cleanup,&
62 : cp_libint_static_init
63 : USE machine, ONLY: default_output_unit,&
64 : m_flush
65 : USE message_passing, ONLY: mp_para_env_release,&
66 : mp_para_env_type
67 : USE molecule_kind_types, ONLY: molecule_kind_type
68 : USE molecule_types, ONLY: molecule_type
69 : USE mp2_cphf, ONLY: solve_z_vector_eq
70 : USE mp2_gpw_method, ONLY: mp2_gpw_compute
71 : USE mp2_integrals, ONLY: mp2_ri_gpw_compute_in
72 : USE mp2_ri_gpw, ONLY: mp2_ri_gpw_compute_en
73 : USE mp2_ri_grad, ONLY: calc_ri_mp2_nonsep
74 : USE mp2_types, ONLY: mp2_type,&
75 : three_dim_real_array
76 : USE particle_methods, ONLY: get_particle_set
77 : USE particle_types, ONLY: particle_type
78 : USE qs_environment_types, ONLY: get_qs_env,&
79 : qs_environment_type
80 : USE qs_integral_utils, ONLY: basis_set_list_setup
81 : USE qs_interactions, ONLY: init_interaction_radii
82 : USE qs_kind_types, ONLY: get_qs_kind,&
83 : qs_kind_type
84 : USE qs_ks_types, ONLY: qs_ks_env_type
85 : USE qs_mo_types, ONLY: get_mo_set,&
86 : mo_set_type
87 : USE qs_neighbor_list_types, ONLY: neighbor_list_set_p_type,&
88 : release_neighbor_list_sets
89 : USE qs_neighbor_lists, ONLY: atom2d_build,&
90 : atom2d_cleanup,&
91 : build_neighbor_lists,&
92 : local_atoms_type,&
93 : pair_radius_setup
94 : USE rpa_main, ONLY: rpa_ri_compute_en
95 : USE rpa_rse, ONLY: rse_energy
96 : #include "./base/base_uses.f90"
97 :
98 : IMPLICIT NONE
99 :
100 : PRIVATE
101 :
102 : CHARACTER(len=*), PARAMETER, PRIVATE :: moduleN = 'mp2_gpw'
103 :
104 : PUBLIC :: mp2_gpw_main, create_mat_munu, grep_rows_in_subgroups, build_dbcsr_from_rows
105 :
106 : CONTAINS
107 :
108 : ! **************************************************************************************************
109 : !> \brief with a big bang to mp2
110 : !> \param qs_env ...
111 : !> \param mp2_env ...
112 : !> \param Emp2 ...
113 : !> \param Emp2_Cou ...
114 : !> \param Emp2_EX ...
115 : !> \param Emp2_S ...
116 : !> \param Emp2_T ...
117 : !> \param mos_mp2 ...
118 : !> \param para_env ...
119 : !> \param unit_nr ...
120 : !> \param calc_forces ...
121 : !> \param calc_ex ...
122 : !> \param do_ri_mp2 ...
123 : !> \param do_ri_rpa ...
124 : !> \param do_ri_sos_laplace_mp2 ...
125 : !> \author Mauro Del Ben and Joost VandeVondele
126 : ! **************************************************************************************************
127 668 : SUBROUTINE mp2_gpw_main(qs_env, mp2_env, Emp2, Emp2_Cou, Emp2_EX, Emp2_S, Emp2_T, &
128 668 : mos_mp2, para_env, unit_nr, calc_forces, calc_ex, do_ri_mp2, do_ri_rpa, &
129 : do_ri_sos_laplace_mp2)
130 : TYPE(qs_environment_type), POINTER :: qs_env
131 : TYPE(mp2_type) :: mp2_env
132 : REAL(KIND=dp), INTENT(OUT) :: Emp2, Emp2_Cou, Emp2_EX, Emp2_S, Emp2_T
133 : TYPE(mo_set_type), DIMENSION(:), INTENT(IN) :: mos_mp2
134 : TYPE(mp_para_env_type), POINTER :: para_env
135 : INTEGER, INTENT(IN) :: unit_nr
136 : LOGICAL, INTENT(IN) :: calc_forces, calc_ex
137 : LOGICAL, INTENT(IN), OPTIONAL :: do_ri_mp2, do_ri_rpa, &
138 : do_ri_sos_laplace_mp2
139 :
140 : CHARACTER(LEN=*), PARAMETER :: routineN = 'mp2_gpw_main'
141 :
142 : INTEGER :: blacs_grid_layout, bse_lev_virt, color_sub, dimen_RI, dimen_RI_red, eri_method, &
143 : handle, ispin, local_unit_nr, my_group_L_end, my_group_L_size, my_group_L_start, nmo, &
144 : nspins, potential_type, ri_metric_type
145 668 : INTEGER, ALLOCATABLE, DIMENSION(:) :: ends_array_mc, ends_array_mc_block, gw_corr_lev_occ, &
146 668 : gw_corr_lev_virt, homo, starts_array_mc, starts_array_mc_block
147 : INTEGER, DIMENSION(3) :: periodic
148 : LOGICAL :: blacs_repeatable, do_bse, do_im_time, do_kpoints_cubic_RPA, my_do_gw, &
149 : my_do_ri_mp2, my_do_ri_rpa, my_do_ri_sos_laplace_mp2
150 : REAL(KIND=dp) :: Emp2_AB, Emp2_BB, Emp2_Cou_BB, &
151 : Emp2_EX_BB, eps_gvg_rspace_old, &
152 : eps_pgf_orb_old, eps_rho_rspace_old
153 668 : REAL(KIND=dp), ALLOCATABLE, DIMENSION(:, :) :: Eigenval
154 668 : REAL(KIND=dp), ALLOCATABLE, DIMENSION(:, :, :) :: BIb_C_bse_ab, BIb_C_bse_ij
155 668 : REAL(KIND=dp), DIMENSION(:), POINTER :: mo_eigenvalues
156 668 : TYPE(atomic_kind_type), DIMENSION(:), POINTER :: atomic_kind_set
157 : TYPE(block_ind_type), ALLOCATABLE, &
158 668 : DIMENSION(:, :, :) :: t_3c_O_ind
159 : TYPE(cell_type), POINTER :: cell
160 : TYPE(cp_blacs_env_type), POINTER :: blacs_env_sub, blacs_env_sub_mat_munu
161 : TYPE(cp_fm_type) :: fm_matrix_PQ
162 668 : TYPE(cp_fm_type), ALLOCATABLE, DIMENSION(:) :: mo_coeff
163 668 : TYPE(cp_fm_type), ALLOCATABLE, DIMENSION(:, :) :: fm_matrix_L_kpoints, fm_matrix_Minv, &
164 668 : fm_matrix_Minv_L_kpoints, &
165 668 : fm_matrix_Minv_Vtrunc_Minv
166 : TYPE(cp_fm_type), POINTER :: mo_coeff_ptr
167 : TYPE(cp_logger_type), POINTER :: logger, logger_sub
168 : TYPE(dbcsr_p_type) :: mat_munu, mat_P_global
169 668 : TYPE(dbcsr_p_type), ALLOCATABLE, DIMENSION(:) :: mo_coeff_all, mo_coeff_gw, mo_coeff_o, &
170 668 : mo_coeff_o_bse, mo_coeff_v, &
171 668 : mo_coeff_v_bse
172 668 : TYPE(dbcsr_p_type), DIMENSION(:), POINTER :: matrix_s
173 668 : TYPE(dbcsr_p_type), DIMENSION(:, :), POINTER :: matrix_s_kp
174 4676 : TYPE(dbt_type) :: t_3c_M
175 668 : TYPE(dbt_type), ALLOCATABLE, DIMENSION(:, :) :: t_3c_O
176 : TYPE(dft_control_type), POINTER :: dft_control
177 668 : TYPE(group_dist_d1_type) :: gd_array, gd_B_all, gd_B_occ_bse, &
178 668 : gd_B_virt_bse
179 : TYPE(group_dist_d1_type), ALLOCATABLE, &
180 668 : DIMENSION(:) :: gd_B_virtual
181 : TYPE(hfx_compression_type), ALLOCATABLE, &
182 668 : DIMENSION(:, :, :) :: t_3c_O_compressed
183 : TYPE(kpoint_type), POINTER :: kpoints, kpoints_from_DFT
184 668 : TYPE(mo_set_type), DIMENSION(:), POINTER :: mos
185 : TYPE(mp_para_env_type), POINTER :: para_env_sub
186 : TYPE(neighbor_list_set_p_type), DIMENSION(:), &
187 668 : POINTER :: sab_orb_sub
188 668 : TYPE(particle_type), DIMENSION(:), POINTER :: particle_set
189 668 : TYPE(qs_kind_type), DIMENSION(:), POINTER :: qs_kind_set
190 : TYPE(qs_ks_env_type), POINTER :: ks_env
191 : TYPE(three_dim_real_array), ALLOCATABLE, &
192 668 : DIMENSION(:) :: BIb_C, BIb_C_gw
193 :
194 668 : CALL timeset(routineN, handle)
195 :
196 : ! check if we want to do ri-mp2
197 668 : my_do_ri_mp2 = .FALSE.
198 668 : IF (PRESENT(do_ri_mp2)) my_do_ri_mp2 = do_ri_mp2
199 :
200 : ! check if we want to do ri-rpa
201 668 : my_do_ri_rpa = .FALSE.
202 668 : IF (PRESENT(do_ri_rpa)) my_do_ri_rpa = do_ri_rpa
203 :
204 : ! check if we want to do ri-sos-laplace-mp2
205 668 : my_do_ri_sos_laplace_mp2 = .FALSE.
206 668 : IF (PRESENT(do_ri_sos_laplace_mp2)) my_do_ri_sos_laplace_mp2 = do_ri_sos_laplace_mp2
207 :
208 : ! GW and SOS-MP2 cannot be used together
209 668 : IF (my_do_ri_sos_laplace_mp2) THEN
210 58 : CPASSERT(.NOT. mp2_env%ri_rpa%do_ri_g0w0)
211 : END IF
212 :
213 : ! check if we want to do imaginary time
214 668 : do_im_time = mp2_env%do_im_time
215 668 : do_bse = qs_env%mp2_env%bse%do_bse
216 668 : do_kpoints_cubic_RPA = qs_env%mp2_env%ri_rpa_im_time%do_im_time_kpoints
217 :
218 668 : IF (do_kpoints_cubic_RPA .AND. mp2_env%ri_rpa%do_ri_g0w0) THEN
219 0 : CPABORT("Full RPA k-points (DO_KPOINTS in LOW_SCALING section) not implemented with GW")
220 : END IF
221 :
222 : ! Get the number of spins
223 668 : nspins = SIZE(mos_mp2)
224 :
225 : ! ... setup needed to be able to qs_integrate in a subgroup.
226 668 : IF (do_kpoints_cubic_RPA) THEN
227 4 : CALL get_qs_env(qs_env=qs_env, dft_control=dft_control, kpoints=kpoints_from_DFT)
228 4 : mos(1:nspins) => kpoints_from_DFT%kp_env(1)%kpoint_env%mos(1:nspins, 1)
229 : ELSE
230 664 : CALL get_qs_env(qs_env=qs_env, dft_control=dft_control, mos=mos)
231 : END IF
232 668 : CALL get_mo_set(mo_set=mos_mp2(1), nmo=nmo)
233 6160 : ALLOCATE (homo(nspins), Eigenval(nmo, nspins), mo_coeff(nspins))
234 1484 : DO ispin = 1, nspins
235 : CALL get_mo_set(mo_set=mos_mp2(ispin), &
236 : eigenvalues=mo_eigenvalues, homo=homo(ispin), &
237 816 : mo_coeff=mo_coeff_ptr)
238 816 : mo_coeff(ispin) = mo_coeff_ptr
239 17770 : Eigenval(:, ispin) = mo_eigenvalues(1:nmo)
240 : END DO
241 :
242 : ! a para_env
243 668 : color_sub = para_env%mepos/mp2_env%mp2_num_proc
244 668 : ALLOCATE (para_env_sub)
245 668 : CALL para_env_sub%from_split(para_env, color_sub)
246 :
247 : ! each of the sub groups might need to generate output
248 668 : logger => cp_get_default_logger()
249 668 : IF (para_env%is_source()) THEN
250 334 : local_unit_nr = cp_logger_get_default_unit_nr(logger, local=.FALSE.)
251 : ELSE
252 334 : local_unit_nr = default_output_unit
253 : END IF
254 :
255 : ! get stuff
256 : CALL get_qs_env(qs_env, &
257 : ks_env=ks_env, &
258 : qs_kind_set=qs_kind_set, &
259 : cell=cell, &
260 : particle_set=particle_set, &
261 : atomic_kind_set=atomic_kind_set, &
262 : dft_control=dft_control, &
263 668 : matrix_s_kp=matrix_s_kp)
264 :
265 668 : CALL get_cell(cell=cell, periodic=periodic)
266 :
267 668 : IF (do_im_time) THEN
268 134 : IF (mp2_env%ri_metric%potential_type == ri_default) THEN
269 420 : IF (SUM(periodic) == 1 .OR. SUM(periodic) == 3) THEN
270 6 : mp2_env%ri_metric%potential_type = do_potential_id
271 : ELSE
272 54 : mp2_env%ri_metric%potential_type = do_potential_truncated
273 : END IF
274 : END IF
275 :
276 : ! statically initialize libint
277 134 : CALL cp_libint_static_init()
278 :
279 : END IF
280 :
281 668 : IF (mp2_env%ri_metric%potential_type == ri_default) THEN
282 318 : mp2_env%ri_metric%potential_type = do_potential_coulomb
283 : END IF
284 :
285 668 : IF (mp2_env%eri_method == eri_default) THEN
286 1136 : IF (SUM(periodic) > 0) mp2_env%eri_method = do_eri_gpw
287 1136 : IF (SUM(periodic) == 0) mp2_env%eri_method = do_eri_os
288 1136 : IF (SUM(mp2_env%ri_rpa_im_time%kp_grid) > 0) mp2_env%eri_method = do_eri_os
289 284 : IF (mp2_env%method == mp2_method_gpw) mp2_env%eri_method = do_eri_gpw
290 284 : IF (mp2_env%method == ri_mp2_method_gpw) mp2_env%eri_method = do_eri_gpw
291 284 : IF (mp2_env%ri_rpa_im_time%do_im_time_kpoints) mp2_env%eri_method = do_eri_os
292 284 : IF (calc_forces .AND. mp2_env%eri_method == do_eri_os) mp2_env%eri_method = do_eri_gpw
293 : END IF
294 668 : eri_method = mp2_env%eri_method
295 :
296 668 : IF (unit_nr > 0 .AND. mp2_env%eri_method == do_eri_gpw) THEN
297 : WRITE (UNIT=unit_nr, FMT="(T3,A,T71,F10.1)") &
298 188 : "GPW_INFO| Density cutoff [a.u.]:", mp2_env%mp2_gpw%cutoff*0.5_dp
299 : WRITE (UNIT=unit_nr, FMT="(T3,A,T71,F10.1)") &
300 188 : "GPW_INFO| Relative density cutoff [a.u.]:", mp2_env%mp2_gpw%relative_cutoff*0.5_dp
301 188 : CALL m_flush(unit_nr)
302 : END IF
303 :
304 : ! MG: Disable logger layer for BSE, misses some key information to print cube files properly
305 668 : IF (.NOT. (mp2_env%ri_g0w0%print_local_bandgap .OR. mp2_env%bse%do_nto_analysis)) THEN
306 : ! a logger
307 658 : NULLIFY (logger_sub)
308 : CALL cp_logger_create(logger_sub, para_env=para_env_sub, &
309 : default_global_unit_nr=local_unit_nr, &
310 658 : close_global_unit_on_dealloc=.FALSE.)
311 658 : CALL cp_logger_set(logger_sub, local_filename="MP2_localLog")
312 : ! set to a custom print level (we could also have a different print level for para_env%source)
313 658 : logger_sub%iter_info%print_level = mp2_env%mp2_gpw%print_level
314 658 : CALL cp_add_default_logger(logger_sub)
315 : END IF
316 :
317 : ! a blacs_env (ignore the globenv stored defaults for now)
318 668 : blacs_grid_layout = BLACS_GRID_SQUARE
319 668 : blacs_repeatable = .TRUE.
320 668 : NULLIFY (blacs_env_sub)
321 : CALL cp_blacs_env_create(blacs_env_sub, para_env_sub, &
322 : blacs_grid_layout, &
323 668 : blacs_repeatable)
324 :
325 668 : blacs_env_sub_mat_munu => blacs_env_sub
326 :
327 668 : matrix_s(1:1) => matrix_s_kp(1:1, 1)
328 :
329 668 : CALL get_eps_old(dft_control, eps_pgf_orb_old, eps_rho_rspace_old, eps_gvg_rspace_old)
330 :
331 : CALL create_mat_munu(mat_munu, qs_env, mp2_env%mp2_gpw%eps_grid, &
332 : blacs_env_sub_mat_munu, do_alloc_blocks_from_nbl=.NOT. do_im_time, sab_orb_sub=sab_orb_sub, &
333 : do_kpoints=mp2_env%ri_rpa_im_time%do_im_time_kpoints, &
334 668 : dbcsr_sym_type=dbcsr_type_symmetric)
335 :
336 : ! which RI metric we want to have
337 668 : ri_metric_type = mp2_env%ri_metric%potential_type
338 :
339 : ! which interaction potential
340 668 : potential_type = mp2_env%potential_parameter%potential_type
341 :
342 : ! check if we want to do ri-g0w0 on top of ri-rpa
343 668 : my_do_gw = mp2_env%ri_rpa%do_ri_g0w0
344 2004 : ALLOCATE (gw_corr_lev_occ(nspins), gw_corr_lev_virt(nspins))
345 668 : gw_corr_lev_occ(1) = mp2_env%ri_g0w0%corr_mos_occ
346 668 : gw_corr_lev_virt(1) = mp2_env%ri_g0w0%corr_mos_virt
347 668 : IF (nspins == 2) THEN
348 148 : gw_corr_lev_occ(2) = mp2_env%ri_g0w0%corr_mos_occ_beta
349 148 : gw_corr_lev_virt(2) = mp2_env%ri_g0w0%corr_mos_virt_beta
350 : END IF
351 :
352 668 : IF (do_bse) THEN
353 30 : IF (nspins > 1) THEN
354 0 : CPABORT("BSE not implemented for open shell calculations")
355 : END IF
356 : !Keep default behavior for occupied
357 : ! We do not implement an explicit bse_lev_occ here, because the small number of occupied levels
358 : ! does not critically influence the memory
359 30 : bse_lev_virt = gw_corr_lev_virt(1)
360 : END IF
361 :
362 : ! After the components are inside of the routines, we can move this line insight the branch
363 7272 : ALLOCATE (mo_coeff_o(nspins), mo_coeff_v(nspins), mo_coeff_all(nspins), mo_coeff_gw(nspins))
364 :
365 : ! Always allocate for usage in call of replicate_mat_to_subgroup
366 2004 : ALLOCATE (mo_coeff_o_bse(1), mo_coeff_v_bse(1))
367 :
368 : ! for imag. time, we do not need this
369 668 : IF (.NOT. do_im_time) THEN
370 :
371 : ! new routine: replicate a full matrix from one para_env to a smaller one
372 : ! keeping the memory usage as small as possible in this case the
373 : ! output the two part of the C matrix (virtual, occupied)
374 1188 : DO ispin = 1, nspins
375 :
376 : CALL replicate_mat_to_subgroup(para_env, para_env_sub, mo_coeff(ispin), homo(ispin), mat_munu%matrix, &
377 : mo_coeff_o(ispin)%matrix, mo_coeff_v(ispin)%matrix, &
378 : mo_coeff_all(ispin)%matrix, mo_coeff_gw(ispin)%matrix, &
379 : my_do_gw, gw_corr_lev_occ(ispin), gw_corr_lev_virt(ispin), do_bse, &
380 : bse_lev_virt, mo_coeff_o_bse(1)%matrix, mo_coeff_v_bse(1)%matrix, &
381 1188 : mp2_env%mp2_gpw%eps_filter)
382 :
383 : END DO
384 :
385 : END IF
386 :
387 : ! now we're kind of ready to go....
388 668 : Emp2_S = 0.0_dp
389 668 : Emp2_T = 0.0_dp
390 668 : IF (my_do_ri_mp2 .OR. my_do_ri_rpa .OR. my_do_ri_sos_laplace_mp2) THEN
391 : ! RI-GPW integrals (same stuff for both RPA and MP2)
392 654 : IF (nspins == 2) THEN
393 : ! open shell case (RI) here the (ia|K) integrals are computed for both the alpha and beta components
394 : CALL mp2_ri_gpw_compute_in( &
395 : BIb_C, BIb_C_gw, BIb_C_bse_ij, BIb_C_bse_ab, gd_array, gd_B_virtual, dimen_RI, dimen_RI_red, qs_env, &
396 : para_env, para_env_sub, color_sub, cell, particle_set, &
397 : atomic_kind_set, qs_kind_set, fm_matrix_PQ, fm_matrix_L_kpoints, fm_matrix_Minv_L_kpoints, &
398 : fm_matrix_Minv, fm_matrix_Minv_Vtrunc_Minv, nmo, homo, mat_munu, sab_orb_sub, &
399 : mo_coeff_o, mo_coeff_v, mo_coeff_all, mo_coeff_gw, mo_coeff_o_bse, mo_coeff_v_bse, &
400 : mp2_env%mp2_gpw%eps_filter, unit_nr, &
401 : mp2_env%mp2_memory, mp2_env%calc_PQ_cond_num, calc_forces, blacs_env_sub, my_do_gw .AND. .NOT. do_im_time, &
402 : do_bse, gd_B_all, starts_array_mc, ends_array_mc, starts_array_mc_block, ends_array_mc_block, &
403 : gw_corr_lev_occ(1), gw_corr_lev_virt(1), &
404 : bse_lev_virt, &
405 : do_im_time, do_kpoints_cubic_RPA, kpoints, &
406 : t_3c_M, t_3c_O, t_3c_O_compressed, t_3c_O_ind, &
407 : mp2_env%ri_metric, &
408 288 : gd_B_occ_bse, gd_B_virt_bse)
409 : ELSE
410 : ! closed shell case (RI)
411 : CALL mp2_ri_gpw_compute_in(BIb_C, BIb_C_gw, BIb_C_bse_ij, BIb_C_bse_ab, gd_array, gd_B_virtual, &
412 : dimen_RI, dimen_RI_red, qs_env, para_env, para_env_sub, &
413 : color_sub, cell, particle_set, &
414 : atomic_kind_set, qs_kind_set, fm_matrix_PQ, &
415 : fm_matrix_L_kpoints, fm_matrix_Minv_L_kpoints, &
416 : fm_matrix_Minv, fm_matrix_Minv_Vtrunc_Minv, nmo, homo, &
417 : mat_munu, sab_orb_sub, &
418 : mo_coeff_o, mo_coeff_v, mo_coeff_all, mo_coeff_gw, mo_coeff_o_bse, mo_coeff_v_bse, &
419 : mp2_env%mp2_gpw%eps_filter, unit_nr, &
420 : mp2_env%mp2_memory, mp2_env%calc_PQ_cond_num, calc_forces, &
421 : blacs_env_sub, my_do_gw .AND. .NOT. do_im_time, do_bse, gd_B_all, &
422 : starts_array_mc, ends_array_mc, &
423 : starts_array_mc_block, ends_array_mc_block, &
424 : gw_corr_lev_occ(1), gw_corr_lev_virt(1), &
425 : bse_lev_virt, &
426 : do_im_time, do_kpoints_cubic_RPA, kpoints, &
427 : t_3c_M, t_3c_O, t_3c_O_compressed, t_3c_O_ind, &
428 962 : mp2_env%ri_metric, gd_B_occ_bse, gd_B_virt_bse)
429 : END IF
430 :
431 : ELSE
432 : ! Canonical MP2-GPW
433 14 : IF (nspins == 2) THEN
434 : ! alpha-alpha and alpha-beta components
435 2 : IF (unit_nr > 0) WRITE (unit_nr, *)
436 2 : IF (unit_nr > 0) WRITE (unit_nr, '(T3,A)') 'Alpha (ia|'
437 : CALL mp2_gpw_compute( &
438 : Emp2, Emp2_Cou, Emp2_EX, qs_env, para_env, para_env_sub, color_sub, &
439 : cell, particle_set, &
440 : atomic_kind_set, qs_kind_set, Eigenval, nmo, homo, mat_munu, &
441 : sab_orb_sub, mo_coeff_o, mo_coeff_v, mp2_env%mp2_gpw%eps_filter, unit_nr, &
442 2 : mp2_env%mp2_memory, calc_ex, blacs_env_sub, Emp2_AB)
443 :
444 : ! beta-beta component
445 2 : IF (unit_nr > 0) WRITE (unit_nr, *)
446 2 : IF (unit_nr > 0) WRITE (unit_nr, '(T3,A)') 'Beta (ia|'
447 : CALL mp2_gpw_compute( &
448 : Emp2_BB, Emp2_Cou_BB, Emp2_EX_BB, qs_env, para_env, para_env_sub, color_sub, cell, particle_set, &
449 : atomic_kind_set, qs_kind_set, Eigenval(:, 2:2), nmo, homo(2:2), mat_munu, &
450 : sab_orb_sub, mo_coeff_o(2:2), mo_coeff_v(2:2), mp2_env%mp2_gpw%eps_filter, unit_nr, &
451 2 : mp2_env%mp2_memory, calc_ex, blacs_env_sub)
452 :
453 : ! make order on the MP2 energy contributions
454 2 : Emp2_Cou = Emp2_Cou*0.25_dp
455 2 : Emp2_EX = Emp2_EX*0.5_dp
456 :
457 2 : Emp2_Cou_BB = Emp2_Cou_BB*0.25_dp
458 2 : Emp2_EX_BB = Emp2_EX_BB*0.5_dp
459 :
460 2 : Emp2_S = Emp2_AB
461 2 : Emp2_T = Emp2_Cou + Emp2_Cou_BB + Emp2_EX + Emp2_EX_BB
462 :
463 2 : Emp2_Cou = Emp2_Cou + Emp2_Cou_BB + Emp2_AB
464 2 : Emp2_EX = Emp2_EX + Emp2_EX_BB
465 2 : Emp2 = Emp2_EX + Emp2_Cou
466 :
467 : ELSE
468 : ! closed shell case
469 : CALL mp2_gpw_compute( &
470 : Emp2, Emp2_Cou, Emp2_EX, qs_env, para_env, para_env_sub, color_sub, cell, particle_set, &
471 : atomic_kind_set, qs_kind_set, Eigenval(:, 1:1), nmo, homo(1:1), mat_munu, &
472 : sab_orb_sub, mo_coeff_o(1:1), mo_coeff_v(1:1), mp2_env%mp2_gpw%eps_filter, unit_nr, &
473 12 : mp2_env%mp2_memory, calc_ex, blacs_env_sub)
474 : END IF
475 : END IF
476 :
477 : ! Free possibly large buffers allocated by dbcsr on the GPU,
478 : ! large hybrid dgemm/pdgemm's coming later will need the space.
479 668 : CALL dbcsr_clear_mempools()
480 :
481 668 : IF (calc_forces .AND. .NOT. do_im_time) THEN
482 : ! make a copy of mo_coeff_o and mo_coeff_v
483 1532 : ALLOCATE (mp2_env%ri_grad%mo_coeff_o(nspins), mp2_env%ri_grad%mo_coeff_v(nspins))
484 630 : DO ispin = 1, nspins
485 358 : NULLIFY (mp2_env%ri_grad%mo_coeff_o(ispin)%matrix)
486 358 : CALL dbcsr_init_p(mp2_env%ri_grad%mo_coeff_o(ispin)%matrix)
487 : CALL dbcsr_copy(mp2_env%ri_grad%mo_coeff_o(ispin)%matrix, mo_coeff_o(ispin)%matrix, &
488 358 : name="mo_coeff_o"//cp_to_string(ispin))
489 358 : NULLIFY (mp2_env%ri_grad%mo_coeff_v(ispin)%matrix)
490 358 : CALL dbcsr_init_p(mp2_env%ri_grad%mo_coeff_v(ispin)%matrix)
491 : CALL dbcsr_copy(mp2_env%ri_grad%mo_coeff_v(ispin)%matrix, mo_coeff_v(ispin)%matrix, &
492 630 : name="mo_coeff_v"//cp_to_string(ispin))
493 : END DO
494 272 : CALL get_group_dist(gd_array, color_sub, my_group_L_start, my_group_L_end, my_group_L_size)
495 : END IF
496 : ! Copy mo coeffs for RPA exchange correction
497 668 : IF (mp2_env%ri_rpa%exchange_correction /= rpa_exchange_none) THEN
498 76 : ALLOCATE (mp2_env%ri_rpa%mo_coeff_o(nspins), mp2_env%ri_rpa%mo_coeff_v(nspins))
499 26 : DO ispin = 1, nspins
500 14 : CALL dbcsr_copy(mp2_env%ri_rpa%mo_coeff_o(ispin), mo_coeff_o(ispin)%matrix, name="mo_coeff_o")
501 26 : CALL dbcsr_copy(mp2_env%ri_rpa%mo_coeff_v(ispin), mo_coeff_v(ispin)%matrix, name="mo_coeff_v")
502 : END DO
503 : END IF
504 :
505 668 : IF (.NOT. do_im_time) THEN
506 :
507 1188 : DO ispin = 1, nspins
508 654 : CALL dbcsr_release(mo_coeff_o(ispin)%matrix)
509 654 : DEALLOCATE (mo_coeff_o(ispin)%matrix)
510 654 : CALL dbcsr_release(mo_coeff_v(ispin)%matrix)
511 654 : DEALLOCATE (mo_coeff_v(ispin)%matrix)
512 1188 : IF (my_do_gw) THEN
513 62 : CALL dbcsr_release(mo_coeff_all(ispin)%matrix)
514 62 : DEALLOCATE (mo_coeff_all(ispin)%matrix)
515 : END IF
516 : END DO
517 534 : DEALLOCATE (mo_coeff_o, mo_coeff_v)
518 534 : IF (my_do_gw) DEALLOCATE (mo_coeff_all)
519 :
520 : END IF
521 668 : IF (do_bse) THEN
522 30 : CALL dbcsr_release(mo_coeff_o_bse(1)%matrix)
523 30 : CALL dbcsr_release(mo_coeff_v_bse(1)%matrix)
524 30 : DEALLOCATE (mo_coeff_o_bse(1)%matrix)
525 30 : DEALLOCATE (mo_coeff_v_bse(1)%matrix)
526 : END IF
527 668 : DEALLOCATE (mo_coeff_o_bse, mo_coeff_v_bse)
528 :
529 : ! Release some memory for RPA exchange correction
530 668 : IF (calc_forces .AND. do_im_time .OR. &
531 : (.NOT. calc_forces .AND. mp2_env%ri_rpa%exchange_correction == rpa_exchange_none)) THEN
532 :
533 384 : CALL dbcsr_release(mat_munu%matrix)
534 384 : DEALLOCATE (mat_munu%matrix)
535 :
536 384 : CALL release_neighbor_list_sets(sab_orb_sub)
537 :
538 : END IF
539 :
540 : ! decide if to do RI-RPA or RI-MP2
541 668 : IF (my_do_ri_rpa .OR. my_do_ri_sos_laplace_mp2) THEN
542 :
543 300 : IF (do_im_time) CALL create_matrix_P(mat_P_global, qs_env, mp2_env, para_env)
544 :
545 730 : IF (.NOT. ALLOCATED(BIb_C)) ALLOCATE (BIb_C(nspins))
546 1080 : IF (.NOT. ALLOCATED(BIb_C_gw)) ALLOCATE (BIb_C_gw(nspins))
547 730 : IF (.NOT. ALLOCATED(gd_B_virtual)) ALLOCATE (gd_B_virtual(nspins))
548 :
549 : ! RI-RPA
550 : CALL rpa_ri_compute_en(qs_env, Emp2, mp2_env, BIb_C, BIb_C_gw, BIb_C_bse_ij, BIb_C_bse_ab, &
551 : para_env, para_env_sub, color_sub, &
552 : gd_array, gd_B_virtual, gd_B_all, gd_B_occ_bse, gd_B_virt_bse, &
553 : mo_coeff, fm_matrix_PQ, fm_matrix_L_kpoints, fm_matrix_Minv_L_kpoints, &
554 : fm_matrix_Minv, fm_matrix_Minv_Vtrunc_Minv, kpoints, &
555 : Eigenval, nmo, homo, dimen_RI, dimen_RI_red, gw_corr_lev_occ, gw_corr_lev_virt, &
556 : bse_lev_virt, &
557 : unit_nr, my_do_ri_sos_laplace_mp2, my_do_gw, do_im_time, do_bse, matrix_s, &
558 : mat_munu, mat_P_global, t_3c_M, t_3c_O, t_3c_O_compressed, t_3c_O_ind, &
559 : starts_array_mc, ends_array_mc, &
560 300 : starts_array_mc_block, ends_array_mc_block, calc_forces)
561 :
562 300 : IF (mp2_env%ri_rpa%do_rse) &
563 6 : CALL rse_energy(qs_env, mp2_env, para_env, dft_control, mo_coeff, homo, Eigenval)
564 :
565 300 : IF (do_im_time) THEN
566 134 : IF (ASSOCIATED(mat_P_global%matrix)) THEN
567 134 : CALL dbcsr_release(mat_P_global%matrix)
568 134 : DEALLOCATE (mat_P_global%matrix)
569 : END IF
570 :
571 134 : CALL cp_libint_static_cleanup()
572 134 : IF (calc_forces) CALL cp_fm_release(fm_matrix_PQ)
573 : END IF
574 :
575 : ! Release some memory for RPA exchange correction
576 300 : IF (mp2_env%ri_rpa%exchange_correction /= rpa_exchange_none) THEN
577 :
578 12 : CALL dbcsr_release(mat_munu%matrix)
579 12 : DEALLOCATE (mat_munu%matrix)
580 :
581 12 : CALL release_neighbor_list_sets(sab_orb_sub)
582 :
583 : END IF
584 :
585 : ELSE
586 368 : IF (my_do_ri_mp2) THEN
587 354 : Emp2 = 0.0_dp
588 354 : Emp2_Cou = 0.0_dp
589 354 : Emp2_EX = 0.0_dp
590 :
591 : ! RI-MP2-GPW compute energy
592 : CALL mp2_ri_gpw_compute_en( &
593 : Emp2_Cou, Emp2_EX, Emp2_S, Emp2_T, BIb_C, mp2_env, para_env, para_env_sub, color_sub, &
594 : gd_array, gd_B_virtual, &
595 354 : Eigenval, nmo, homo, dimen_RI_red, unit_nr, calc_forces, calc_ex)
596 :
597 : END IF
598 : END IF
599 :
600 : ! if we need forces time to calculate the MP2 non-separable contribution
601 : ! and start computing the Lagrangian
602 668 : IF (calc_forces .AND. .NOT. do_im_time) THEN
603 :
604 : CALL calc_ri_mp2_nonsep(qs_env, mp2_env, para_env, para_env_sub, cell, &
605 : particle_set, atomic_kind_set, qs_kind_set, &
606 : mo_coeff, dimen_RI, Eigenval, &
607 : my_group_L_start, my_group_L_end, my_group_L_size, &
608 272 : sab_orb_sub, mat_munu, blacs_env_sub)
609 :
610 630 : DO ispin = 1, nspins
611 358 : CALL dbcsr_release(mp2_env%ri_grad%mo_coeff_o(ispin)%matrix)
612 358 : DEALLOCATE (mp2_env%ri_grad%mo_coeff_o(ispin)%matrix)
613 :
614 358 : CALL dbcsr_release(mp2_env%ri_grad%mo_coeff_v(ispin)%matrix)
615 630 : DEALLOCATE (mp2_env%ri_grad%mo_coeff_v(ispin)%matrix)
616 : END DO
617 272 : DEALLOCATE (mp2_env%ri_grad%mo_coeff_o, mp2_env%ri_grad%mo_coeff_v)
618 :
619 272 : CALL dbcsr_release(mat_munu%matrix)
620 272 : DEALLOCATE (mat_munu%matrix)
621 :
622 272 : CALL release_neighbor_list_sets(sab_orb_sub)
623 :
624 : END IF
625 :
626 : !XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXx
627 : ! moved from above
628 668 : IF (my_do_gw .AND. .NOT. do_im_time) THEN
629 120 : DO ispin = 1, nspins
630 62 : CALL dbcsr_release(mo_coeff_gw(ispin)%matrix)
631 120 : DEALLOCATE (mo_coeff_gw(ispin)%matrix)
632 : END DO
633 58 : DEALLOCATE (mo_coeff_gw)
634 : END IF
635 :
636 : ! re-init the radii to be able to generate pair lists with MP2-appropriate screening
637 668 : dft_control%qs_control%eps_pgf_orb = eps_pgf_orb_old
638 668 : dft_control%qs_control%eps_rho_rspace = eps_rho_rspace_old
639 668 : dft_control%qs_control%eps_gvg_rspace = eps_gvg_rspace_old
640 668 : CALL init_interaction_radii(dft_control%qs_control, qs_kind_set)
641 :
642 668 : CALL cp_blacs_env_release(blacs_env_sub)
643 :
644 668 : IF (.NOT. (mp2_env%ri_g0w0%print_local_bandgap .OR. mp2_env%bse%do_nto_analysis)) THEN
645 658 : CALL cp_rm_default_logger()
646 658 : CALL cp_logger_release(logger_sub)
647 : END IF
648 :
649 668 : CALL mp_para_env_release(para_env_sub)
650 :
651 : ! finally solve the z-vector equation if forces are required
652 668 : IF (calc_forces .AND. .NOT. do_im_time) THEN
653 : CALL solve_z_vector_eq(qs_env, mp2_env, para_env, dft_control, &
654 272 : mo_coeff, homo, Eigenval, unit_nr)
655 : END IF
656 :
657 668 : DEALLOCATE (Eigenval, mo_coeff)
658 :
659 668 : CALL timestop(handle)
660 :
661 5298 : END SUBROUTINE mp2_gpw_main
662 :
663 : ! **************************************************************************************************
664 : !> \brief ...
665 : !> \param para_env ...
666 : !> \param para_env_sub ...
667 : !> \param mo_coeff ...
668 : !> \param homo ...
669 : !> \param mat_munu ...
670 : !> \param mo_coeff_o ...
671 : !> \param mo_coeff_v ...
672 : !> \param mo_coeff_all ...
673 : !> \param mo_coeff_gw ...
674 : !> \param my_do_gw ...
675 : !> \param gw_corr_lev_occ ...
676 : !> \param gw_corr_lev_virt ...
677 : !> \param my_do_bse ...
678 : !> \param bse_lev_virt ...
679 : !> \param mo_coeff_o_bse ...
680 : !> \param mo_coeff_v_bse ...
681 : !> \param eps_filter ...
682 : ! **************************************************************************************************
683 654 : SUBROUTINE replicate_mat_to_subgroup(para_env, para_env_sub, mo_coeff, homo, mat_munu, &
684 : mo_coeff_o, mo_coeff_v, mo_coeff_all, mo_coeff_gw, my_do_gw, &
685 : gw_corr_lev_occ, gw_corr_lev_virt, my_do_bse, &
686 : bse_lev_virt, mo_coeff_o_bse, mo_coeff_v_bse, eps_filter)
687 : TYPE(mp_para_env_type), INTENT(IN) :: para_env, para_env_sub
688 : TYPE(cp_fm_type), INTENT(IN) :: mo_coeff
689 : INTEGER, INTENT(IN) :: homo
690 : TYPE(dbcsr_type), INTENT(INOUT) :: mat_munu
691 : TYPE(dbcsr_type), POINTER :: mo_coeff_o, mo_coeff_v, mo_coeff_all, &
692 : mo_coeff_gw
693 : LOGICAL, INTENT(IN) :: my_do_gw
694 : INTEGER, INTENT(IN) :: gw_corr_lev_occ, gw_corr_lev_virt
695 : LOGICAL, INTENT(IN) :: my_do_bse
696 : INTEGER, INTENT(IN) :: bse_lev_virt
697 : TYPE(dbcsr_type), POINTER :: mo_coeff_o_bse, mo_coeff_v_bse
698 : REAL(KIND=dp), INTENT(IN) :: eps_filter
699 :
700 : CHARACTER(LEN=*), PARAMETER :: routineN = 'replicate_mat_to_subgroup'
701 :
702 : INTEGER :: handle
703 654 : REAL(KIND=dp), ALLOCATABLE, DIMENSION(:, :) :: C
704 654 : TYPE(group_dist_d1_type) :: gd_array
705 :
706 654 : CALL timeset(routineN, handle)
707 :
708 : CALL grep_rows_in_subgroups(para_env, para_env_sub, mo_coeff, gd_array, C)
709 :
710 : ! create and fill mo_coeff_o, mo_coeff_v and mo_coeff_all
711 654 : ALLOCATE (mo_coeff_o)
712 : CALL build_dbcsr_from_rows(para_env_sub, mo_coeff_o, C(:, 1:homo), &
713 654 : mat_munu, gd_array, eps_filter)
714 :
715 654 : ALLOCATE (mo_coeff_v)
716 : CALL build_dbcsr_from_rows(para_env_sub, mo_coeff_v, C(:, homo + 1:), &
717 654 : mat_munu, gd_array, eps_filter)
718 :
719 654 : IF (my_do_gw) THEN
720 62 : ALLOCATE (mo_coeff_gw)
721 : CALL build_dbcsr_from_rows(para_env_sub, mo_coeff_gw, C(:, homo - gw_corr_lev_occ + 1:homo + gw_corr_lev_virt), &
722 62 : mat_munu, gd_array, eps_filter)
723 :
724 : ! all levels
725 62 : ALLOCATE (mo_coeff_all)
726 : CALL build_dbcsr_from_rows(para_env_sub, mo_coeff_all, C, &
727 62 : mat_munu, gd_array, eps_filter)
728 :
729 : END IF
730 :
731 654 : IF (my_do_bse) THEN
732 :
733 30 : ALLOCATE (mo_coeff_o_bse)
734 : CALL build_dbcsr_from_rows(para_env_sub, mo_coeff_o_bse, C(:, 1:homo), &
735 30 : mat_munu, gd_array, eps_filter)
736 :
737 30 : ALLOCATE (mo_coeff_v_bse)
738 : CALL build_dbcsr_from_rows(para_env_sub, mo_coeff_v_bse, C(:, homo + 1:homo + bse_lev_virt), &
739 30 : mat_munu, gd_array, eps_filter)
740 :
741 : END IF
742 654 : DEALLOCATE (C)
743 654 : CALL release_group_dist(gd_array)
744 :
745 654 : CALL timestop(handle)
746 :
747 654 : END SUBROUTINE replicate_mat_to_subgroup
748 :
749 : ! **************************************************************************************************
750 : !> \brief ...
751 : !> \param para_env ...
752 : !> \param para_env_sub ...
753 : !> \param mo_coeff ...
754 : !> \param gd_array ...
755 : !> \param C ...
756 : ! **************************************************************************************************
757 732 : SUBROUTINE grep_rows_in_subgroups(para_env, para_env_sub, mo_coeff, gd_array, C)
758 : TYPE(mp_para_env_type), INTENT(IN) :: para_env, para_env_sub
759 : TYPE(cp_fm_type), INTENT(IN) :: mo_coeff
760 : TYPE(group_dist_d1_type), INTENT(OUT) :: gd_array
761 : REAL(KIND=dp), ALLOCATABLE, DIMENSION(:, :), &
762 : INTENT(OUT) :: C
763 :
764 : CHARACTER(LEN=*), PARAMETER :: routineN = 'grep_rows_in_subgroups'
765 :
766 : INTEGER :: handle, i_global, iiB, j_global, jjB, max_row_col_local, my_mu_end, my_mu_size, &
767 : my_mu_start, ncol_global, ncol_local, ncol_rec, nrow_global, nrow_local, nrow_rec, &
768 : proc_receive_static, proc_send_static, proc_shift
769 732 : INTEGER, ALLOCATABLE, DIMENSION(:, :) :: local_col_row_info, rec_col_row_info
770 732 : INTEGER, DIMENSION(:), POINTER :: col_indices, col_indices_rec, &
771 732 : row_indices, row_indices_rec
772 732 : REAL(KIND=dp), ALLOCATABLE, DIMENSION(:, :) :: local_C, rec_C
773 : REAL(KIND=dp), CONTIGUOUS, DIMENSION(:, :), &
774 732 : POINTER :: local_C_internal
775 :
776 732 : CALL timeset(routineN, handle)
777 :
778 : CALL cp_fm_get_info(matrix=mo_coeff, &
779 : ncol_global=ncol_global, &
780 : nrow_global=nrow_global, &
781 : nrow_local=nrow_local, &
782 : ncol_local=ncol_local, &
783 : row_indices=row_indices, &
784 : col_indices=col_indices, &
785 732 : local_data=local_C_internal)
786 :
787 732 : CALL create_group_dist(gd_array, para_env_sub%num_pe, nrow_global)
788 732 : CALL get_group_dist(gd_array, para_env_sub%mepos, my_mu_start, my_mu_end, my_mu_size)
789 :
790 : ! local storage for the C matrix
791 2928 : ALLOCATE (C(my_mu_size, ncol_global))
792 280721 : C = 0.0_dp
793 :
794 2928 : ALLOCATE (local_C(nrow_local, ncol_local))
795 151681 : local_C(:, :) = local_C_internal(1:nrow_local, 1:ncol_local)
796 732 : NULLIFY (local_C_internal)
797 :
798 732 : max_row_col_local = MAX(nrow_local, ncol_local)
799 732 : CALL para_env%max(max_row_col_local)
800 :
801 2928 : ALLOCATE (local_col_row_info(0:max_row_col_local, 2))
802 28892 : local_col_row_info = 0
803 : ! 0,1 nrows
804 732 : local_col_row_info(0, 1) = nrow_local
805 7157 : local_col_row_info(1:nrow_local, 1) = row_indices(1:nrow_local)
806 : ! 0,2 ncols
807 732 : local_col_row_info(0, 2) = ncol_local
808 13282 : local_col_row_info(1:ncol_local, 2) = col_indices(1:ncol_local)
809 :
810 1464 : ALLOCATE (rec_col_row_info(0:max_row_col_local, 2))
811 :
812 : ! accumulate data on C buffer starting from myself
813 7157 : DO iiB = 1, nrow_local
814 6425 : i_global = row_indices(iiB)
815 7157 : IF (i_global >= my_mu_start .AND. i_global <= my_mu_end) THEN
816 143933 : DO jjB = 1, ncol_local
817 137611 : j_global = col_indices(jjB)
818 143933 : C(i_global - my_mu_start + 1, j_global) = local_C(iiB, jjB)
819 : END DO
820 : END IF
821 : END DO
822 :
823 : ! start ring communication for collecting the data from the other
824 732 : proc_send_static = MODULO(para_env%mepos + 1, para_env%num_pe)
825 732 : proc_receive_static = MODULO(para_env%mepos - 1, para_env%num_pe)
826 1464 : DO proc_shift = 1, para_env%num_pe - 1
827 : ! first exchange information on the local data
828 28892 : rec_col_row_info = 0
829 732 : CALL para_env%sendrecv(local_col_row_info, proc_send_static, rec_col_row_info, proc_receive_static)
830 732 : nrow_rec = rec_col_row_info(0, 1)
831 732 : ncol_rec = rec_col_row_info(0, 2)
832 :
833 2196 : ALLOCATE (row_indices_rec(nrow_rec))
834 7157 : row_indices_rec = rec_col_row_info(1:nrow_rec, 1)
835 :
836 2196 : ALLOCATE (col_indices_rec(ncol_rec))
837 13282 : col_indices_rec = rec_col_row_info(1:ncol_rec, 2)
838 :
839 2928 : ALLOCATE (rec_C(nrow_rec, ncol_rec))
840 151681 : rec_C = 0.0_dp
841 :
842 : ! then send and receive the real data
843 732 : CALL para_env%sendrecv(local_C, proc_send_static, rec_C, proc_receive_static)
844 :
845 : ! accumulate the received data on C buffer
846 7157 : DO iiB = 1, nrow_rec
847 6425 : i_global = row_indices_rec(iiB)
848 7157 : IF (i_global >= my_mu_start .AND. i_global <= my_mu_end) THEN
849 135752 : DO jjB = 1, ncol_rec
850 129828 : j_global = col_indices_rec(jjB)
851 135752 : C(i_global - my_mu_start + 1, j_global) = rec_C(iiB, jjB)
852 : END DO
853 : END IF
854 : END DO
855 :
856 28892 : local_col_row_info(:, :) = rec_col_row_info
857 732 : DEALLOCATE (local_C)
858 2196 : ALLOCATE (local_C(nrow_rec, ncol_rec))
859 151681 : local_C(:, :) = rec_C
860 :
861 732 : DEALLOCATE (col_indices_rec)
862 732 : DEALLOCATE (row_indices_rec)
863 1464 : DEALLOCATE (rec_C)
864 : END DO
865 :
866 732 : DEALLOCATE (local_C)
867 732 : DEALLOCATE (local_col_row_info)
868 732 : DEALLOCATE (rec_col_row_info)
869 :
870 732 : CALL timestop(handle)
871 :
872 2928 : END SUBROUTINE grep_rows_in_subgroups
873 :
874 : ! **************************************************************************************************
875 : !> \brief Encapsulate the building of dbcsr_matrices mo_coeff_(v,o,all)
876 : !> \param para_env_sub ...
877 : !> \param mo_coeff_to_build ...
878 : !> \param Cread ...
879 : !> \param mat_munu ...
880 : !> \param gd_array ...
881 : !> \param eps_filter ...
882 : !> \author Jan Wilhelm, Code by Mauro Del Ben
883 : ! **************************************************************************************************
884 1570 : SUBROUTINE build_dbcsr_from_rows(para_env_sub, mo_coeff_to_build, Cread, &
885 : mat_munu, gd_array, eps_filter)
886 : TYPE(mp_para_env_type), INTENT(IN) :: para_env_sub
887 : TYPE(dbcsr_type) :: mo_coeff_to_build
888 : REAL(KIND=dp), DIMENSION(:, :), INTENT(IN) :: Cread
889 : TYPE(dbcsr_type), INTENT(INOUT) :: mat_munu
890 : TYPE(group_dist_d1_type), INTENT(IN) :: gd_array
891 : REAL(KIND=dp), INTENT(IN) :: eps_filter
892 :
893 : CHARACTER(LEN=*), PARAMETER :: routineN = 'build_dbcsr_from_rows'
894 :
895 : INTEGER :: col, col_offset, col_size, handle, i, i_global, j, j_global, my_mu_end, &
896 : my_mu_start, ncol_global, proc_receive, proc_send, proc_shift, rec_mu_end, rec_mu_size, &
897 : rec_mu_start, row, row_offset, row_size
898 1570 : REAL(KIND=dp), ALLOCATABLE, DIMENSION(:, :) :: rec_C
899 1570 : REAL(KIND=dp), DIMENSION(:, :), POINTER :: data_block
900 : TYPE(dbcsr_iterator_type) :: iter
901 :
902 1570 : CALL timeset(routineN, handle)
903 :
904 1570 : ncol_global = SIZE(Cread, 2)
905 :
906 1570 : CALL get_group_dist(gd_array, para_env_sub%mepos, my_mu_start, my_mu_end)
907 :
908 : CALL cp_dbcsr_m_by_n_from_row_template(mo_coeff_to_build, template=mat_munu, n=ncol_global, &
909 1570 : sym=dbcsr_type_no_symmetry)
910 1570 : CALL dbcsr_reserve_all_blocks(mo_coeff_to_build)
911 :
912 : ! accumulate data on mo_coeff_to_build starting from myself
913 1570 : CALL dbcsr_iterator_start(iter, mo_coeff_to_build)
914 5935 : DO WHILE (dbcsr_iterator_blocks_left(iter))
915 : CALL dbcsr_iterator_next_block(iter, row, col, data_block, &
916 : row_size=row_size, col_size=col_size, &
917 4365 : row_offset=row_offset, col_offset=col_offset)
918 34577 : DO i = 1, row_size
919 28642 : i_global = row_offset + i - 1
920 33007 : IF (i_global >= my_mu_start .AND. i_global <= my_mu_end) THEN
921 357858 : DO j = 1, col_size
922 329508 : j_global = col_offset + j - 1
923 357858 : data_block(i, j) = Cread(i_global - my_mu_start + 1, col_offset + j - 1)
924 : END DO
925 : END IF
926 : END DO
927 : END DO
928 1570 : CALL dbcsr_iterator_stop(iter)
929 :
930 : ! start ring communication in the subgroup for collecting the data from the other
931 : ! proc (occupied)
932 1704 : DO proc_shift = 1, para_env_sub%num_pe - 1
933 134 : proc_send = MODULO(para_env_sub%mepos + proc_shift, para_env_sub%num_pe)
934 134 : proc_receive = MODULO(para_env_sub%mepos - proc_shift, para_env_sub%num_pe)
935 :
936 134 : CALL get_group_dist(gd_array, proc_receive, rec_mu_start, rec_mu_end, rec_mu_size)
937 :
938 536 : ALLOCATE (rec_C(rec_mu_size, ncol_global))
939 10427 : rec_C = 0.0_dp
940 :
941 : ! then send and receive the real data
942 10427 : CALL para_env_sub%sendrecv(Cread, proc_send, rec_C, proc_receive)
943 :
944 : ! accumulate data on mo_coeff_to_build the data received from proc_rec
945 134 : CALL dbcsr_iterator_start(iter, mo_coeff_to_build)
946 303 : DO WHILE (dbcsr_iterator_blocks_left(iter))
947 : CALL dbcsr_iterator_next_block(iter, row, col, data_block, &
948 : row_size=row_size, col_size=col_size, &
949 169 : row_offset=row_offset, col_offset=col_offset)
950 1245 : DO i = 1, row_size
951 942 : i_global = row_offset + i - 1
952 1111 : IF (i_global >= rec_mu_start .AND. i_global <= rec_mu_end) THEN
953 2507 : DO j = 1, col_size
954 2215 : j_global = col_offset + j - 1
955 2507 : data_block(i, j) = rec_C(i_global - rec_mu_start + 1, col_offset + j - 1)
956 : END DO
957 : END IF
958 : END DO
959 : END DO
960 134 : CALL dbcsr_iterator_stop(iter)
961 :
962 1972 : DEALLOCATE (rec_C)
963 :
964 : END DO
965 1570 : CALL dbcsr_filter(mo_coeff_to_build, eps_filter)
966 :
967 1570 : CALL timestop(handle)
968 :
969 3140 : END SUBROUTINE build_dbcsr_from_rows
970 :
971 : ! **************************************************************************************************
972 : !> \brief Encapsulate the building of dbcsr_matrix mat_munu
973 : !> \param mat_munu ...
974 : !> \param qs_env ...
975 : !> \param eps_grid ...
976 : !> \param blacs_env_sub ...
977 : !> \param do_ri_aux_basis ...
978 : !> \param do_mixed_basis ...
979 : !> \param group_size_prim ...
980 : !> \param do_alloc_blocks_from_nbl ...
981 : !> \param do_kpoints ...
982 : !> \param sab_orb_sub ...
983 : !> \param dbcsr_sym_type ...
984 : !> \author Jan Wilhelm, code by Mauro Del Ben
985 : ! **************************************************************************************************
986 952 : SUBROUTINE create_mat_munu(mat_munu, qs_env, eps_grid, blacs_env_sub, &
987 : do_ri_aux_basis, do_mixed_basis, group_size_prim, &
988 : do_alloc_blocks_from_nbl, do_kpoints, sab_orb_sub, dbcsr_sym_type)
989 :
990 : TYPE(dbcsr_p_type), INTENT(OUT) :: mat_munu
991 : TYPE(qs_environment_type), POINTER :: qs_env
992 : REAL(KIND=dp) :: eps_grid
993 : TYPE(cp_blacs_env_type), POINTER :: blacs_env_sub
994 : LOGICAL, INTENT(IN), OPTIONAL :: do_ri_aux_basis, do_mixed_basis
995 : INTEGER, INTENT(IN), OPTIONAL :: group_size_prim
996 : LOGICAL, INTENT(IN), OPTIONAL :: do_alloc_blocks_from_nbl, do_kpoints
997 : TYPE(neighbor_list_set_p_type), DIMENSION(:), &
998 : OPTIONAL, POINTER :: sab_orb_sub
999 : CHARACTER, OPTIONAL :: dbcsr_sym_type
1000 :
1001 : CHARACTER(LEN=*), PARAMETER :: routineN = 'create_mat_munu'
1002 :
1003 : CHARACTER :: my_dbcsr_sym_type
1004 : INTEGER :: handle, ikind, natom, nkind
1005 952 : INTEGER, DIMENSION(:), POINTER :: col_blk_sizes, row_blk_sizes
1006 : LOGICAL :: my_do_alloc_blocks_from_nbl, &
1007 : my_do_kpoints, my_do_mixed_basis, &
1008 : my_do_ri_aux_basis
1009 952 : LOGICAL, ALLOCATABLE, DIMENSION(:) :: orb_present
1010 952 : REAL(dp), ALLOCATABLE, DIMENSION(:) :: orb_radius
1011 952 : REAL(dp), ALLOCATABLE, DIMENSION(:, :) :: pair_radius
1012 : REAL(KIND=dp) :: subcells
1013 952 : TYPE(atomic_kind_type), DIMENSION(:), POINTER :: atomic_kind_set
1014 : TYPE(cell_type), POINTER :: cell
1015 : TYPE(dbcsr_distribution_type), POINTER :: dbcsr_dist_sub
1016 : TYPE(dft_control_type), POINTER :: dft_control
1017 : TYPE(distribution_1d_type), POINTER :: local_molecules_sub, local_particles_sub
1018 : TYPE(distribution_2d_type), POINTER :: distribution_2d_sub
1019 952 : TYPE(gto_basis_set_p_type), DIMENSION(:), POINTER :: basis_set_ri_aux
1020 : TYPE(gto_basis_set_type), POINTER :: orb_basis_set
1021 952 : TYPE(local_atoms_type), ALLOCATABLE, DIMENSION(:) :: atom2d
1022 952 : TYPE(molecule_kind_type), DIMENSION(:), POINTER :: molecule_kind_set
1023 952 : TYPE(molecule_type), DIMENSION(:), POINTER :: molecule_set
1024 : TYPE(neighbor_list_set_p_type), DIMENSION(:), &
1025 952 : POINTER :: my_sab_orb_sub
1026 952 : TYPE(particle_type), DIMENSION(:), POINTER :: particle_set
1027 952 : TYPE(qs_kind_type), DIMENSION(:), POINTER :: qs_kind_set
1028 :
1029 952 : CALL timeset(routineN, handle)
1030 :
1031 952 : NULLIFY (basis_set_ri_aux)
1032 :
1033 952 : my_do_ri_aux_basis = .FALSE.
1034 952 : IF (PRESENT(do_ri_aux_basis)) THEN
1035 218 : my_do_ri_aux_basis = do_ri_aux_basis
1036 : END IF
1037 :
1038 952 : my_do_mixed_basis = .FALSE.
1039 952 : IF (PRESENT(do_mixed_basis)) THEN
1040 0 : my_do_mixed_basis = do_mixed_basis
1041 : END IF
1042 :
1043 952 : my_do_alloc_blocks_from_nbl = .FALSE.
1044 952 : IF (PRESENT(do_alloc_blocks_from_nbl)) THEN
1045 734 : my_do_alloc_blocks_from_nbl = do_alloc_blocks_from_nbl
1046 : END IF
1047 :
1048 952 : my_do_kpoints = .FALSE.
1049 952 : IF (PRESENT(do_kpoints)) THEN
1050 802 : my_do_kpoints = do_kpoints
1051 : END IF
1052 :
1053 952 : my_dbcsr_sym_type = dbcsr_type_no_symmetry
1054 952 : IF (PRESENT(dbcsr_sym_type)) THEN
1055 734 : my_dbcsr_sym_type = dbcsr_sym_type
1056 : END IF
1057 :
1058 : CALL get_qs_env(qs_env, &
1059 : qs_kind_set=qs_kind_set, &
1060 : cell=cell, &
1061 : particle_set=particle_set, &
1062 : atomic_kind_set=atomic_kind_set, &
1063 : molecule_set=molecule_set, &
1064 : molecule_kind_set=molecule_kind_set, &
1065 952 : dft_control=dft_control)
1066 :
1067 952 : IF (my_do_kpoints) THEN
1068 : ! please choose EPS_PGF_ORB in QS section smaller than EPS_GRID in WFC_GPW section
1069 8 : IF (eps_grid < dft_control%qs_control%eps_pgf_orb) THEN
1070 0 : eps_grid = dft_control%qs_control%eps_pgf_orb
1071 0 : CPWARN("WFC_GPW%EPS_GRID has been set to QS%EPS_PGF_ORB")
1072 : END IF
1073 : END IF
1074 :
1075 : ! hack hack hack XXXXXXXXXXXXXXX ... to be fixed
1076 952 : dft_control%qs_control%eps_pgf_orb = eps_grid
1077 952 : dft_control%qs_control%eps_rho_rspace = eps_grid
1078 952 : dft_control%qs_control%eps_gvg_rspace = eps_grid
1079 952 : CALL init_interaction_radii(dft_control%qs_control, qs_kind_set)
1080 :
1081 : ! get a distribution_1d
1082 952 : NULLIFY (local_particles_sub, local_molecules_sub)
1083 : CALL distribute_molecules_1d(atomic_kind_set=atomic_kind_set, &
1084 : particle_set=particle_set, &
1085 : local_particles=local_particles_sub, &
1086 : molecule_kind_set=molecule_kind_set, &
1087 : molecule_set=molecule_set, &
1088 : local_molecules=local_molecules_sub, &
1089 952 : force_env_section=qs_env%input)
1090 :
1091 : ! get a distribution_2d
1092 952 : NULLIFY (distribution_2d_sub)
1093 : CALL distribute_molecules_2d(cell=cell, &
1094 : atomic_kind_set=atomic_kind_set, &
1095 : qs_kind_set=qs_kind_set, &
1096 : particle_set=particle_set, &
1097 : molecule_kind_set=molecule_kind_set, &
1098 : molecule_set=molecule_set, &
1099 : distribution_2d=distribution_2d_sub, &
1100 : blacs_env=blacs_env_sub, &
1101 952 : force_env_section=qs_env%input)
1102 :
1103 : ! Build the sub orbital-orbital overlap neighbor lists
1104 952 : CALL section_vals_val_get(qs_env%input, "DFT%SUBCELLS", r_val=subcells)
1105 952 : nkind = SIZE(atomic_kind_set)
1106 4556 : ALLOCATE (atom2d(nkind))
1107 :
1108 : CALL atom2d_build(atom2d, local_particles_sub, distribution_2d_sub, atomic_kind_set, &
1109 952 : molecule_set, molecule_only=.FALSE., particle_set=particle_set)
1110 :
1111 2856 : ALLOCATE (orb_present(nkind))
1112 2856 : ALLOCATE (orb_radius(nkind))
1113 3808 : ALLOCATE (pair_radius(nkind, nkind))
1114 :
1115 2652 : DO ikind = 1, nkind
1116 1700 : CALL get_qs_kind(qs_kind_set(ikind), basis_set=orb_basis_set)
1117 2652 : IF (ASSOCIATED(orb_basis_set)) THEN
1118 1700 : orb_present(ikind) = .TRUE.
1119 1700 : CALL get_gto_basis_set(gto_basis_set=orb_basis_set, kind_radius=orb_radius(ikind))
1120 : ELSE
1121 0 : orb_present(ikind) = .FALSE.
1122 0 : orb_radius(ikind) = 0.0_dp
1123 : END IF
1124 : END DO
1125 :
1126 952 : CALL pair_radius_setup(orb_present, orb_present, orb_radius, orb_radius, pair_radius)
1127 :
1128 952 : IF (PRESENT(sab_orb_sub)) THEN
1129 734 : NULLIFY (sab_orb_sub)
1130 : ! for cubic RPA/GW with kpoints, we need all neighbors and not only the symmetric ones
1131 734 : IF (my_do_kpoints) THEN
1132 : CALL build_neighbor_lists(sab_orb_sub, particle_set, atom2d, cell, pair_radius, &
1133 : mic=.FALSE., subcells=subcells, molecular=.FALSE., nlname="sab_orb_sub", &
1134 4 : symmetric=.FALSE.)
1135 : ELSE
1136 : CALL build_neighbor_lists(sab_orb_sub, particle_set, atom2d, cell, pair_radius, &
1137 730 : mic=.FALSE., subcells=subcells, molecular=.FALSE., nlname="sab_orb_sub")
1138 : END IF
1139 : ELSE
1140 218 : NULLIFY (my_sab_orb_sub)
1141 : ! for cubic RPA/GW with kpoints, we need all neighbors and not only the symmetric ones
1142 218 : IF (my_do_kpoints) THEN
1143 : CALL build_neighbor_lists(my_sab_orb_sub, particle_set, atom2d, cell, pair_radius, &
1144 : mic=.FALSE., subcells=subcells, molecular=.FALSE., nlname="sab_orb_sub", &
1145 4 : symmetric=.FALSE.)
1146 : ELSE
1147 : CALL build_neighbor_lists(my_sab_orb_sub, particle_set, atom2d, cell, pair_radius, &
1148 214 : mic=.FALSE., subcells=subcells, molecular=.FALSE., nlname="sab_orb_sub")
1149 : END IF
1150 : END IF
1151 952 : CALL atom2d_cleanup(atom2d)
1152 952 : DEALLOCATE (atom2d)
1153 952 : DEALLOCATE (orb_present, orb_radius, pair_radius)
1154 :
1155 : ! a dbcsr_dist
1156 952 : ALLOCATE (dbcsr_dist_sub)
1157 952 : CALL cp_dbcsr_dist2d_to_dist(distribution_2d_sub, dbcsr_dist_sub)
1158 :
1159 : ! build a dbcsr matrix the hard way
1160 952 : natom = SIZE(particle_set)
1161 2856 : ALLOCATE (row_blk_sizes(natom))
1162 952 : IF (my_do_ri_aux_basis) THEN
1163 :
1164 734 : ALLOCATE (basis_set_ri_aux(nkind))
1165 190 : CALL basis_set_list_setup(basis_set_ri_aux, "RI_AUX", qs_kind_set)
1166 190 : CALL get_particle_set(particle_set, qs_kind_set, nsgf=row_blk_sizes, basis=basis_set_ri_aux)
1167 190 : DEALLOCATE (basis_set_ri_aux)
1168 :
1169 762 : ELSE IF (my_do_mixed_basis) THEN
1170 :
1171 0 : ALLOCATE (basis_set_ri_aux(nkind))
1172 0 : CALL basis_set_list_setup(basis_set_ri_aux, "RI_AUX", qs_kind_set)
1173 0 : CALL get_particle_set(particle_set, qs_kind_set, nsgf=row_blk_sizes, basis=basis_set_ri_aux)
1174 0 : DEALLOCATE (basis_set_ri_aux)
1175 :
1176 0 : ALLOCATE (col_blk_sizes(natom))
1177 :
1178 0 : CALL get_particle_set(particle_set, qs_kind_set, nsgf=col_blk_sizes)
1179 0 : col_blk_sizes = col_blk_sizes*group_size_prim
1180 :
1181 : ELSE
1182 762 : CALL get_particle_set(particle_set, qs_kind_set, nsgf=row_blk_sizes)
1183 : END IF
1184 :
1185 : NULLIFY (mat_munu%matrix)
1186 952 : ALLOCATE (mat_munu%matrix)
1187 :
1188 952 : IF (my_do_ri_aux_basis) THEN
1189 :
1190 : CALL dbcsr_create(matrix=mat_munu%matrix, &
1191 : name="(ai|munu)", &
1192 : dist=dbcsr_dist_sub, matrix_type=my_dbcsr_sym_type, &
1193 190 : row_blk_size=row_blk_sizes, col_blk_size=row_blk_sizes)
1194 :
1195 762 : ELSE IF (my_do_mixed_basis) THEN
1196 :
1197 : CALL dbcsr_create(matrix=mat_munu%matrix, &
1198 : name="(ai|munu)", &
1199 : dist=dbcsr_dist_sub, matrix_type=my_dbcsr_sym_type, &
1200 0 : row_blk_size=row_blk_sizes, col_blk_size=col_blk_sizes)
1201 :
1202 : ELSE
1203 :
1204 : CALL dbcsr_create(matrix=mat_munu%matrix, &
1205 : name="(ai|munu)", &
1206 : dist=dbcsr_dist_sub, matrix_type=my_dbcsr_sym_type, &
1207 762 : row_blk_size=row_blk_sizes, col_blk_size=row_blk_sizes)
1208 :
1209 762 : IF (my_do_alloc_blocks_from_nbl) THEN
1210 :
1211 600 : IF (PRESENT(sab_orb_sub)) THEN
1212 600 : CALL cp_dbcsr_alloc_block_from_nbl(mat_munu%matrix, sab_orb_sub)
1213 : ELSE
1214 0 : CALL cp_dbcsr_alloc_block_from_nbl(mat_munu%matrix, my_sab_orb_sub)
1215 : END IF
1216 :
1217 : END IF
1218 :
1219 : END IF
1220 :
1221 952 : DEALLOCATE (row_blk_sizes)
1222 :
1223 952 : IF (my_do_mixed_basis) THEN
1224 0 : DEALLOCATE (col_blk_sizes)
1225 : END IF
1226 :
1227 952 : CALL dbcsr_distribution_release(dbcsr_dist_sub)
1228 952 : DEALLOCATE (dbcsr_dist_sub)
1229 :
1230 952 : CALL distribution_2d_release(distribution_2d_sub)
1231 :
1232 952 : CALL distribution_1d_release(local_particles_sub)
1233 952 : CALL distribution_1d_release(local_molecules_sub)
1234 :
1235 952 : IF (.NOT. PRESENT(sab_orb_sub)) THEN
1236 218 : CALL release_neighbor_list_sets(my_sab_orb_sub)
1237 : END IF
1238 :
1239 952 : CALL timestop(handle)
1240 :
1241 3808 : END SUBROUTINE create_mat_munu
1242 :
1243 : ! **************************************************************************************************
1244 : !> \brief ...
1245 : !> \param mat_P_global ...
1246 : !> \param qs_env ...
1247 : !> \param mp2_env ...
1248 : !> \param para_env ...
1249 : ! **************************************************************************************************
1250 134 : SUBROUTINE create_matrix_P(mat_P_global, qs_env, mp2_env, para_env)
1251 :
1252 : TYPE(dbcsr_p_type), INTENT(OUT) :: mat_P_global
1253 : TYPE(qs_environment_type), POINTER :: qs_env
1254 : TYPE(mp2_type) :: mp2_env
1255 : TYPE(mp_para_env_type), POINTER :: para_env
1256 :
1257 : CHARACTER(LEN=*), PARAMETER :: routineN = 'create_matrix_P'
1258 :
1259 : INTEGER :: blacs_grid_layout, handle
1260 : LOGICAL :: blacs_repeatable
1261 : TYPE(cp_blacs_env_type), POINTER :: blacs_env_global
1262 :
1263 134 : CALL timeset(routineN, handle)
1264 :
1265 134 : blacs_grid_layout = BLACS_GRID_SQUARE
1266 134 : blacs_repeatable = .TRUE.
1267 134 : NULLIFY (blacs_env_global)
1268 : CALL cp_blacs_env_create(blacs_env_global, para_env, &
1269 : blacs_grid_layout, &
1270 134 : blacs_repeatable)
1271 :
1272 : CALL create_mat_munu(mat_P_global, qs_env, mp2_env%mp2_gpw%eps_grid, &
1273 : blacs_env_global, do_ri_aux_basis=.TRUE., &
1274 134 : do_kpoints=mp2_env%ri_rpa_im_time%do_im_time_kpoints)
1275 :
1276 134 : CALL dbcsr_reserve_all_blocks(mat_P_global%matrix)
1277 134 : CALL cp_blacs_env_release(blacs_env_global)
1278 :
1279 134 : CALL timestop(handle)
1280 :
1281 134 : END SUBROUTINE create_matrix_P
1282 :
1283 : ! **************************************************************************************************
1284 : !> \brief ...
1285 : !> \param dft_control ...
1286 : !> \param eps_pgf_orb_old ...
1287 : !> \param eps_rho_rspace_old ...
1288 : !> \param eps_gvg_rspace_old ...
1289 : ! **************************************************************************************************
1290 668 : PURE SUBROUTINE get_eps_old(dft_control, eps_pgf_orb_old, eps_rho_rspace_old, eps_gvg_rspace_old)
1291 :
1292 : TYPE(dft_control_type), INTENT(INOUT) :: dft_control
1293 : REAL(kind=dp), INTENT(OUT) :: eps_pgf_orb_old, eps_rho_rspace_old, &
1294 : eps_gvg_rspace_old
1295 :
1296 : ! re-init the radii to be able to generate pair lists with MP2-appropriate screening
1297 668 : eps_pgf_orb_old = dft_control%qs_control%eps_pgf_orb
1298 668 : eps_rho_rspace_old = dft_control%qs_control%eps_rho_rspace
1299 668 : eps_gvg_rspace_old = dft_control%qs_control%eps_gvg_rspace
1300 :
1301 668 : END SUBROUTINE get_eps_old
1302 :
1303 : END MODULE mp2_gpw
|