Line data Source code
1 : !--------------------------------------------------------------------------------------------------!
2 : ! CP2K: A general program to perform molecular dynamics simulations !
3 : ! Copyright 2000-2026 CP2K developers group <https://cp2k.org> !
4 : ! !
5 : ! SPDX-License-Identifier: GPL-2.0-or-later !
6 : !--------------------------------------------------------------------------------------------------!
7 :
8 : ! **************************************************************************************************
9 : !> \brief Routines for a linear scaling quickstep SCF run based on the density
10 : !> matrix, with a focus on the interface between dm_ls_scf and qs
11 : !> \par History
12 : !> 2011.04 created [Joost VandeVondele]
13 : !> \author Joost VandeVondele
14 : ! **************************************************************************************************
15 : MODULE dm_ls_scf_qs
16 : USE atomic_kind_types, ONLY: atomic_kind_type
17 : USE cp_control_types, ONLY: dft_control_type
18 : USE cp_dbcsr_api, ONLY: &
19 : dbcsr_complete_redistribute, dbcsr_copy, dbcsr_create, dbcsr_desymmetrize, &
20 : dbcsr_distribution_get, dbcsr_distribution_hold, dbcsr_distribution_new, &
21 : dbcsr_distribution_release, dbcsr_distribution_type, dbcsr_finalize, dbcsr_get_info, &
22 : dbcsr_multiply, dbcsr_p_type, dbcsr_release, dbcsr_set, dbcsr_type
23 : USE cp_dbcsr_cp2k_link, ONLY: cp_dbcsr_alloc_block_from_nbl
24 : USE cp_dbcsr_operations, ONLY: dbcsr_allocate_matrix_set
25 : USE cp_log_handling, ONLY: cp_get_default_logger,&
26 : cp_logger_get_default_unit_nr,&
27 : cp_logger_type
28 : USE cp_realspace_grid_cube, ONLY: cp_pw_to_cube
29 : USE dm_ls_scf_types, ONLY: ls_cluster_atomic,&
30 : ls_cluster_molecular,&
31 : ls_mstruct_type,&
32 : ls_scf_env_type
33 : USE input_constants, ONLY: ls_cluster_atomic,&
34 : ls_cluster_molecular
35 : USE kinds, ONLY: default_string_length,&
36 : dp
37 : USE message_passing, ONLY: mp_para_env_type
38 : USE particle_list_types, ONLY: particle_list_type
39 : USE particle_types, ONLY: particle_type
40 : USE pw_env_types, ONLY: pw_env_get,&
41 : pw_env_type
42 : USE pw_methods, ONLY: pw_zero
43 : USE pw_pool_types, ONLY: pw_pool_p_type,&
44 : pw_pool_type
45 : USE pw_types, ONLY: pw_c1d_gs_type,&
46 : pw_r3d_rs_type
47 : USE qs_atomic_block, ONLY: calculate_atomic_block_dm
48 : USE qs_collocate_density, ONLY: calculate_rho_elec
49 : USE qs_core_energies, ONLY: calculate_ptrace
50 : USE qs_density_mixing_types, ONLY: direct_mixing_nr,&
51 : gspace_mixing_nr
52 : USE qs_energy_types, ONLY: qs_energy_type
53 : USE qs_environment_types, ONLY: get_qs_env,&
54 : qs_environment_type
55 : USE qs_gspace_mixing, ONLY: gspace_mixing
56 : USE qs_harris_types, ONLY: harris_type
57 : USE qs_harris_utils, ONLY: harris_density_update
58 : USE qs_initial_guess, ONLY: calculate_mopac_dm
59 : USE qs_kind_types, ONLY: qs_kind_type
60 : USE qs_ks_methods, ONLY: qs_ks_update_qs_env
61 : USE qs_ks_types, ONLY: qs_ks_did_change,&
62 : qs_ks_env_type,&
63 : set_ks_env
64 : USE qs_mixing_utils, ONLY: charge_mixing_init,&
65 : mixing_allocate,&
66 : mixing_init
67 : USE qs_neighbor_list_types, ONLY: neighbor_list_set_p_type
68 : USE qs_rho_atom_types, ONLY: rho_atom_type
69 : USE qs_rho_methods, ONLY: qs_rho_update_rho
70 : USE qs_rho_types, ONLY: qs_rho_get,&
71 : qs_rho_type
72 : USE qs_subsys_types, ONLY: qs_subsys_get,&
73 : qs_subsys_type
74 : USE tblite_interface, ONLY: tb_get_energy
75 : USE tblite_types, ONLY: tblite_type
76 : #include "./base/base_uses.f90"
77 :
78 : IMPLICIT NONE
79 :
80 : PRIVATE
81 :
82 : CHARACTER(len=*), PARAMETER, PRIVATE :: moduleN = 'dm_ls_scf_qs'
83 :
84 : PUBLIC :: matrix_ls_create, matrix_qs_to_ls, matrix_ls_to_qs, ls_scf_init_qs, &
85 : ls_nonscf_ks, ls_nonscf_energy, ls_scf_dm_to_ks, ls_scf_qs_atomic_guess, &
86 : write_matrix_to_cube, rho_mixing_ls_init, matrix_decluster
87 :
88 : CONTAINS
89 :
90 : ! **************************************************************************************************
91 : !> \brief create a matrix for use (and as a template) in ls based on a qs template
92 : !> \param matrix_ls ...
93 : !> \param matrix_qs ...
94 : !> \param ls_mstruct ...
95 : !> \par History
96 : !> 2011.03 created [Joost VandeVondele]
97 : !> 2015.09 add support for PAO [Ole Schuett]
98 : !> \author Joost VandeVondele
99 : ! **************************************************************************************************
100 1012 : SUBROUTINE matrix_ls_create(matrix_ls, matrix_qs, ls_mstruct)
101 : TYPE(dbcsr_type) :: matrix_ls, matrix_qs
102 : TYPE(ls_mstruct_type), INTENT(IN) :: ls_mstruct
103 :
104 : CHARACTER(len=*), PARAMETER :: routineN = 'matrix_ls_create'
105 :
106 : CHARACTER(len=default_string_length) :: name
107 : INTEGER :: handle, iatom, imol, jatom, natom, nmol
108 1012 : INTEGER, ALLOCATABLE, DIMENSION(:), TARGET :: atom_to_cluster, atom_to_cluster_primus, &
109 1012 : clustered_blk_sizes, primus_of_mol
110 1012 : INTEGER, DIMENSION(:), POINTER :: clustered_col_dist, clustered_row_dist, &
111 1012 : ls_blk_sizes, ls_col_dist, ls_row_dist
112 : TYPE(dbcsr_distribution_type) :: ls_dist, ls_dist_clustered
113 :
114 1012 : CALL timeset(routineN, handle)
115 :
116 : ! Defaults -----------------------------------------------------------------------------------
117 1012 : CALL dbcsr_get_info(matrix_qs, col_blk_size=ls_blk_sizes, distribution=ls_dist)
118 1012 : CALL dbcsr_distribution_hold(ls_dist)
119 1012 : CALL dbcsr_distribution_get(ls_dist, row_dist=ls_row_dist, col_dist=ls_col_dist)
120 :
121 : ! PAO ----------------------------------------------------------------------------------------
122 1012 : IF (ls_mstruct%do_pao) THEN
123 512 : CALL dbcsr_get_info(ls_mstruct%matrix_A, col_blk_size=ls_blk_sizes)
124 : END IF
125 :
126 : ! Clustering ---------------------------------------------------------------------------------
127 1070 : SELECT CASE (ls_mstruct%cluster_type)
128 : CASE (ls_cluster_atomic)
129 : ! do nothing
130 : CASE (ls_cluster_molecular)
131 : ! create format of the clustered matrix
132 58 : CALL dbcsr_get_info(matrix_qs, nblkrows_total=natom)
133 526 : nmol = MAXVAL(ls_mstruct%atom_to_molecule)
134 174 : ALLOCATE (atom_to_cluster_primus(natom))
135 116 : ALLOCATE (atom_to_cluster(natom))
136 174 : ALLOCATE (primus_of_mol(nmol))
137 526 : DO iatom = 1, natom
138 468 : atom_to_cluster(iatom) = ls_mstruct%atom_to_molecule(iatom)
139 : ! the first atom of the molecule is the primus
140 : ! if the number of atoms per molecule is independent of system size, this is not a quadratic loop
141 : ! it assumes that all atoms of the molecule are consecutive.
142 1722 : DO jatom = iatom, 1, -1
143 1722 : IF (ls_mstruct%atom_to_molecule(jatom) == atom_to_cluster(iatom)) THEN
144 1254 : atom_to_cluster_primus(iatom) = jatom
145 : ELSE
146 : EXIT
147 : END IF
148 : END DO
149 526 : primus_of_mol(atom_to_cluster(iatom)) = atom_to_cluster_primus(iatom)
150 : END DO
151 :
152 : ! row
153 116 : ALLOCATE (clustered_row_dist(nmol))
154 188 : DO imol = 1, nmol
155 188 : clustered_row_dist(imol) = ls_row_dist(primus_of_mol(imol))
156 : END DO
157 :
158 : ! col
159 116 : ALLOCATE (clustered_col_dist(nmol))
160 188 : DO imol = 1, nmol
161 188 : clustered_col_dist(imol) = ls_col_dist(primus_of_mol(imol))
162 : END DO
163 :
164 116 : ALLOCATE (clustered_blk_sizes(nmol))
165 58 : clustered_blk_sizes = 0
166 526 : DO iatom = 1, natom
167 : clustered_blk_sizes(atom_to_cluster(iatom)) = clustered_blk_sizes(atom_to_cluster(iatom)) + &
168 526 : ls_blk_sizes(iatom)
169 : END DO
170 58 : ls_blk_sizes => clustered_blk_sizes ! redirect pointer
171 :
172 : ! create new distribution
173 : CALL dbcsr_distribution_new(ls_dist_clustered, &
174 : template=ls_dist, &
175 : row_dist=clustered_row_dist, &
176 : col_dist=clustered_col_dist, &
177 58 : reuse_arrays=.TRUE.)
178 58 : CALL dbcsr_distribution_release(ls_dist)
179 116 : ls_dist = ls_dist_clustered
180 :
181 : CASE DEFAULT
182 1012 : CPABORT("Unknown LS cluster type")
183 : END SELECT
184 :
185 : ! Create actual matrix -----------------------------------------------------------------------
186 1012 : CALL dbcsr_get_info(matrix_qs, name=name)
187 : CALL dbcsr_create(matrix_ls, &
188 : name=name, &
189 : dist=ls_dist, &
190 : matrix_type="S", &
191 : row_blk_size=ls_blk_sizes, &
192 1012 : col_blk_size=ls_blk_sizes)
193 1012 : CALL dbcsr_distribution_release(ls_dist)
194 1012 : CALL dbcsr_finalize(matrix_ls)
195 :
196 1012 : CALL timestop(handle)
197 :
198 2024 : END SUBROUTINE matrix_ls_create
199 :
200 : ! **************************************************************************************************
201 : !> \brief first link to QS, copy a QS matrix to LS matrix
202 : !> used to isolate QS style matrices from LS style
203 : !> will be useful for future features (e.g. precision, symmetry, blocking, ...)
204 : !> \param matrix_ls ...
205 : !> \param matrix_qs ...
206 : !> \param ls_mstruct ...
207 : !> \param covariant ...
208 : !> \par History
209 : !> 2010.10 created [Joost VandeVondele]
210 : !> 2015.09 add support for PAO [Ole Schuett]
211 : !> \author Joost VandeVondele
212 : ! **************************************************************************************************
213 28210 : SUBROUTINE matrix_qs_to_ls(matrix_ls, matrix_qs, ls_mstruct, covariant)
214 : TYPE(dbcsr_type) :: matrix_ls, matrix_qs
215 : TYPE(ls_mstruct_type), INTENT(IN), TARGET :: ls_mstruct
216 : LOGICAL, INTENT(IN) :: covariant
217 :
218 : CHARACTER(len=*), PARAMETER :: routineN = 'matrix_qs_to_ls'
219 :
220 : INTEGER :: handle
221 28210 : INTEGER, DIMENSION(:), POINTER :: pao_blk_sizes
222 : TYPE(dbcsr_type) :: matrix_pao, matrix_tmp
223 : TYPE(dbcsr_type), POINTER :: matrix_trafo
224 :
225 28210 : CALL timeset(routineN, handle)
226 :
227 28210 : IF (.NOT. ls_mstruct%do_pao) THEN
228 2766 : CALL matrix_cluster(matrix_ls, matrix_qs, ls_mstruct)
229 :
230 : ELSE ! using pao
231 25444 : CALL dbcsr_get_info(ls_mstruct%matrix_A, col_blk_size=pao_blk_sizes)
232 : CALL dbcsr_create(matrix_pao, &
233 : matrix_type="N", &
234 : template=matrix_qs, &
235 : row_blk_size=pao_blk_sizes, &
236 25444 : col_blk_size=pao_blk_sizes)
237 :
238 25444 : matrix_trafo => ls_mstruct%matrix_A ! contra-variant
239 25444 : IF (covariant) matrix_trafo => ls_mstruct%matrix_B ! co-variant
240 25444 : CALL dbcsr_create(matrix_tmp, template=matrix_trafo)
241 :
242 25444 : CALL dbcsr_multiply("N", "N", 1.0_dp, matrix_qs, matrix_trafo, 0.0_dp, matrix_tmp)
243 25444 : CALL dbcsr_multiply("T", "N", 1.0_dp, matrix_trafo, matrix_tmp, 0.0_dp, matrix_pao)
244 25444 : CALL dbcsr_release(matrix_tmp)
245 :
246 25444 : CALL matrix_cluster(matrix_ls, matrix_pao, ls_mstruct)
247 25444 : CALL dbcsr_release(matrix_pao)
248 : END IF
249 :
250 28210 : CALL timestop(handle)
251 :
252 28210 : END SUBROUTINE matrix_qs_to_ls
253 :
254 : ! **************************************************************************************************
255 : !> \brief Performs molecular blocking and reduction to single precision if enabled
256 : !> \param matrix_out ...
257 : !> \param matrix_in ...
258 : !> \param ls_mstruct ...
259 : !> \author Ole Schuett
260 : ! **************************************************************************************************
261 28210 : SUBROUTINE matrix_cluster(matrix_out, matrix_in, ls_mstruct)
262 : TYPE(dbcsr_type) :: matrix_out, matrix_in
263 : TYPE(ls_mstruct_type), INTENT(IN) :: ls_mstruct
264 :
265 : CHARACTER(len=*), PARAMETER :: routineN = 'matrix_cluster'
266 :
267 : INTEGER :: handle
268 : TYPE(dbcsr_type) :: matrix_in_nosym
269 :
270 28210 : CALL timeset(routineN, handle)
271 :
272 54860 : SELECT CASE (ls_mstruct%cluster_type)
273 : CASE (ls_cluster_atomic)
274 26650 : CALL dbcsr_copy(matrix_out, matrix_in)
275 :
276 : CASE (ls_cluster_molecular)
277 : ! desymmetrize the qs matrix
278 1560 : CALL dbcsr_create(matrix_in_nosym, template=matrix_in, matrix_type="N")
279 1560 : CALL dbcsr_desymmetrize(matrix_in, matrix_in_nosym)
280 :
281 : ! perform the magic complete redistribute copy
282 1560 : CALL dbcsr_complete_redistribute(matrix_in_nosym, matrix_out)
283 1560 : CALL dbcsr_release(matrix_in_nosym)
284 :
285 : CASE DEFAULT
286 28210 : CPABORT("Unknown LS cluster type")
287 : END SELECT
288 :
289 28210 : CALL timestop(handle)
290 :
291 28210 : END SUBROUTINE matrix_cluster
292 :
293 : ! **************************************************************************************************
294 : !> \brief second link to QS, copy a LS matrix to QS matrix
295 : !> used to isolate QS style matrices from LS style
296 : !> will be useful for future features (e.g. precision, symmetry, blocking, ...)
297 : !> \param matrix_qs ...
298 : !> \param matrix_ls ...
299 : !> \param ls_mstruct ...
300 : !> \param covariant ...
301 : !> \param keep_sparsity will be passed on to dbcsr_copy, by default set to .TRUE.
302 : !> \par History
303 : !> 2010.10 created [Joost VandeVondele]
304 : !> 2015.09 add support for PAO [Ole Schuett]
305 : !> \author Joost VandeVondele
306 : ! **************************************************************************************************
307 4316 : SUBROUTINE matrix_ls_to_qs(matrix_qs, matrix_ls, ls_mstruct, covariant, keep_sparsity)
308 : TYPE(dbcsr_type) :: matrix_qs, matrix_ls
309 : TYPE(ls_mstruct_type), INTENT(IN), TARGET :: ls_mstruct
310 : LOGICAL :: covariant
311 : LOGICAL, OPTIONAL :: keep_sparsity
312 :
313 : CHARACTER(len=*), PARAMETER :: routineN = 'matrix_ls_to_qs'
314 :
315 : INTEGER :: handle
316 4316 : INTEGER, DIMENSION(:), POINTER :: pao_blk_sizes
317 : LOGICAL :: my_keep_sparsity
318 : TYPE(dbcsr_type) :: matrix_declustered, matrix_tmp1, &
319 : matrix_tmp2
320 : TYPE(dbcsr_type), POINTER :: matrix_trafo
321 :
322 4316 : CALL timeset(routineN, handle)
323 :
324 4316 : my_keep_sparsity = .TRUE.
325 4316 : IF (PRESENT(keep_sparsity)) &
326 294 : my_keep_sparsity = keep_sparsity
327 :
328 4316 : IF (.NOT. ls_mstruct%do_pao) THEN
329 2648 : CALL dbcsr_create(matrix_declustered, template=matrix_qs)
330 2648 : CALL matrix_decluster(matrix_declustered, matrix_ls, ls_mstruct)
331 2648 : CALL dbcsr_copy(matrix_qs, matrix_declustered, keep_sparsity=my_keep_sparsity)
332 2648 : CALL dbcsr_release(matrix_declustered)
333 :
334 : ELSE ! using pao
335 1668 : CALL dbcsr_get_info(ls_mstruct%matrix_A, col_blk_size=pao_blk_sizes)
336 : CALL dbcsr_create(matrix_declustered, &
337 : template=matrix_qs, &
338 : row_blk_size=pao_blk_sizes, &
339 1668 : col_blk_size=pao_blk_sizes)
340 :
341 1668 : CALL matrix_decluster(matrix_declustered, matrix_ls, ls_mstruct)
342 :
343 1668 : matrix_trafo => ls_mstruct%matrix_B ! contra-variant
344 1668 : IF (covariant) matrix_trafo => ls_mstruct%matrix_A ! co-variant
345 1668 : CALL dbcsr_create(matrix_tmp1, template=matrix_trafo)
346 1668 : CALL dbcsr_create(matrix_tmp2, template=matrix_qs)
347 1668 : CALL dbcsr_multiply("N", "N", 1.0_dp, matrix_trafo, matrix_declustered, 0.0_dp, matrix_tmp1)
348 1668 : CALL dbcsr_multiply("N", "T", 1.0_dp, matrix_tmp1, matrix_trafo, 0.0_dp, matrix_tmp2)
349 1668 : CALL dbcsr_copy(matrix_qs, matrix_tmp2, keep_sparsity=my_keep_sparsity)
350 1668 : CALL dbcsr_release(matrix_declustered)
351 1668 : CALL dbcsr_release(matrix_tmp1)
352 1668 : CALL dbcsr_release(matrix_tmp2)
353 : END IF
354 :
355 4316 : CALL timestop(handle)
356 :
357 4316 : END SUBROUTINE matrix_ls_to_qs
358 :
359 : ! **************************************************************************************************
360 : !> \brief Reverses molecular blocking and reduction to single precision if enabled
361 : !> \param matrix_out ...
362 : !> \param matrix_in ...
363 : !> \param ls_mstruct ...
364 : !> \author Ole Schuett
365 : ! **************************************************************************************************
366 12296 : SUBROUTINE matrix_decluster(matrix_out, matrix_in, ls_mstruct)
367 : TYPE(dbcsr_type) :: matrix_out, matrix_in
368 : TYPE(ls_mstruct_type), INTENT(IN) :: ls_mstruct
369 :
370 : CHARACTER(len=*), PARAMETER :: routineN = 'matrix_decluster'
371 :
372 : INTEGER :: handle
373 :
374 12296 : CALL timeset(routineN, handle)
375 :
376 23706 : SELECT CASE (ls_mstruct%cluster_type)
377 : CASE (ls_cluster_atomic)
378 11410 : CALL dbcsr_copy(matrix_out, matrix_in)
379 :
380 : CASE (ls_cluster_molecular)
381 : ! perform the magic complete redistribute copy
382 886 : CALL dbcsr_complete_redistribute(matrix_in, matrix_out)
383 :
384 : CASE DEFAULT
385 12296 : CPABORT("Unknown LS cluster type")
386 : END SELECT
387 :
388 12296 : CALL timestop(handle)
389 :
390 12296 : END SUBROUTINE matrix_decluster
391 :
392 : ! **************************************************************************************************
393 : !> \brief further required initialization of QS.
394 : !> Might be factored-out since this seems common code with the other SCF.
395 : !> \param qs_env ...
396 : !> \par History
397 : !> 2010.10 created [Joost VandeVondele]
398 : !> \author Joost VandeVondele
399 : ! **************************************************************************************************
400 982 : SUBROUTINE ls_scf_init_qs(qs_env)
401 : TYPE(qs_environment_type), POINTER :: qs_env
402 :
403 : CHARACTER(len=*), PARAMETER :: routineN = 'ls_scf_init_qs'
404 :
405 : INTEGER :: handle, ispin, nspin, unit_nr
406 : TYPE(cp_logger_type), POINTER :: logger
407 982 : TYPE(dbcsr_p_type), DIMENSION(:), POINTER :: matrix_ks, matrix_s
408 : TYPE(dft_control_type), POINTER :: dft_control
409 : TYPE(neighbor_list_set_p_type), DIMENSION(:), &
410 982 : POINTER :: sab_orb
411 : TYPE(qs_ks_env_type), POINTER :: ks_env
412 :
413 982 : NULLIFY (sab_orb)
414 982 : CALL timeset(routineN, handle)
415 :
416 : ! get a useful output_unit
417 982 : logger => cp_get_default_logger()
418 982 : IF (logger%para_env%is_source()) THEN
419 491 : unit_nr = cp_logger_get_default_unit_nr(logger, local=.TRUE.)
420 : ELSE
421 : unit_nr = -1
422 : END IF
423 :
424 : ! get basic quantities from the qs_env
425 : CALL get_qs_env(qs_env, dft_control=dft_control, &
426 : matrix_s=matrix_s, &
427 : matrix_ks=matrix_ks, &
428 : ks_env=ks_env, &
429 982 : sab_orb=sab_orb)
430 :
431 982 : nspin = dft_control%nspins
432 :
433 : ! we might have to create matrix_ks
434 982 : IF (.NOT. ASSOCIATED(matrix_ks)) THEN
435 0 : CALL dbcsr_allocate_matrix_set(matrix_ks, nspin)
436 0 : DO ispin = 1, nspin
437 0 : ALLOCATE (matrix_ks(ispin)%matrix)
438 0 : CALL dbcsr_create(matrix_ks(ispin)%matrix, template=matrix_s(1)%matrix)
439 0 : CALL cp_dbcsr_alloc_block_from_nbl(matrix_ks(ispin)%matrix, sab_orb)
440 0 : CALL dbcsr_set(matrix_ks(ispin)%matrix, 0.0_dp)
441 : END DO
442 0 : CALL set_ks_env(ks_env, matrix_ks=matrix_ks)
443 : END IF
444 :
445 982 : CALL timestop(handle)
446 :
447 982 : END SUBROUTINE ls_scf_init_qs
448 :
449 : ! **************************************************************************************************
450 : !> \brief get an atomic initial guess
451 : !> \param qs_env ...
452 : !> \param ls_scf_env ...
453 : !> \param energy ...
454 : !> \param nonscf ...
455 : !> \par History
456 : !> 2012.11 created [Joost VandeVondele]
457 : !> \author Joost VandeVondele
458 : ! **************************************************************************************************
459 360 : SUBROUTINE ls_scf_qs_atomic_guess(qs_env, ls_scf_env, energy, nonscf)
460 : TYPE(qs_environment_type), POINTER :: qs_env
461 : TYPE(ls_scf_env_type) :: ls_scf_env
462 : REAL(KIND=dp) :: energy
463 : LOGICAL, INTENT(IN), OPTIONAL :: nonscf
464 :
465 : CHARACTER(len=*), PARAMETER :: routineN = 'ls_scf_qs_atomic_guess'
466 :
467 : INTEGER :: handle, nspin, unit_nr
468 : INTEGER, DIMENSION(2) :: nelectron_spin
469 : LOGICAL :: do_scf, has_unit_metric
470 360 : TYPE(atomic_kind_type), DIMENSION(:), POINTER :: atomic_kind_set
471 : TYPE(cp_logger_type), POINTER :: logger
472 360 : TYPE(dbcsr_p_type), DIMENSION(:), POINTER :: matrix_ks, matrix_s, rho_ao
473 : TYPE(dft_control_type), POINTER :: dft_control
474 : TYPE(mp_para_env_type), POINTER :: para_env
475 360 : TYPE(particle_type), DIMENSION(:), POINTER :: particle_set
476 : TYPE(qs_energy_type), POINTER :: qs_energy
477 360 : TYPE(qs_kind_type), DIMENSION(:), POINTER :: qs_kind_set
478 : TYPE(qs_ks_env_type), POINTER :: ks_env
479 : TYPE(qs_rho_type), POINTER :: rho
480 :
481 360 : CALL timeset(routineN, handle)
482 360 : NULLIFY (rho, rho_ao)
483 :
484 : ! get a useful output_unit
485 360 : logger => cp_get_default_logger()
486 360 : IF (logger%para_env%is_source()) THEN
487 180 : unit_nr = cp_logger_get_default_unit_nr(logger, local=.TRUE.)
488 : ELSE
489 180 : unit_nr = -1
490 : END IF
491 :
492 : ! get basic quantities from the qs_env
493 : CALL get_qs_env(qs_env, dft_control=dft_control, &
494 : matrix_s=matrix_s, &
495 : matrix_ks=matrix_ks, &
496 : ks_env=ks_env, &
497 : energy=qs_energy, &
498 : atomic_kind_set=atomic_kind_set, &
499 : qs_kind_set=qs_kind_set, &
500 : particle_set=particle_set, &
501 : has_unit_metric=has_unit_metric, &
502 : para_env=para_env, &
503 : nelectron_spin=nelectron_spin, &
504 360 : rho=rho)
505 :
506 360 : CALL qs_rho_get(rho, rho_ao=rho_ao)
507 :
508 360 : nspin = dft_control%nspins
509 :
510 : ! create an initial atomic guess
511 360 : IF (dft_control%qs_control%dftb .OR. dft_control%qs_control%semi_empirical .OR. &
512 : dft_control%qs_control%xtb) THEN
513 : CALL calculate_mopac_dm(rho_ao, matrix_s(1)%matrix, has_unit_metric, &
514 : dft_control, particle_set, atomic_kind_set, qs_kind_set, &
515 94 : nspin, nelectron_spin, para_env)
516 : ELSE
517 : CALL calculate_atomic_block_dm(rho_ao, matrix_s(1)%matrix, atomic_kind_set, qs_kind_set, &
518 266 : nspin, nelectron_spin, unit_nr, para_env)
519 : END IF
520 :
521 360 : do_scf = .TRUE.
522 360 : IF (PRESENT(nonscf)) do_scf = .NOT. nonscf
523 290 : IF (do_scf) THEN
524 354 : CALL qs_rho_update_rho(rho, qs_env=qs_env)
525 354 : CALL qs_ks_did_change(qs_env%ks_env, rho_changed=.TRUE.)
526 354 : CALL qs_ks_update_qs_env(qs_env, calculate_forces=.FALSE., just_energy=.FALSE.)
527 354 : CALL ls_scf_tblite_energy(qs_env, qs_energy)
528 354 : energy = qs_energy%total
529 : ELSE
530 6 : CALL ls_nonscf_ks(qs_env, ls_scf_env, energy)
531 : END IF
532 :
533 360 : CALL timestop(handle)
534 :
535 360 : END SUBROUTINE ls_scf_qs_atomic_guess
536 :
537 : ! **************************************************************************************************
538 : !> \brief use the density matrix in ls_scf_env to compute the new energy and KS matrix
539 : !> \param qs_env ...
540 : !> \param ls_scf_env ...
541 : !> \param energy_new ...
542 : !> \param iscf ...
543 : !> \par History
544 : !> 2011.04 created [Joost VandeVondele]
545 : !> 2015.02 added gspace density mixing [Patrick Seewald]
546 : !> \author Joost VandeVondele
547 : ! **************************************************************************************************
548 3422 : SUBROUTINE ls_scf_dm_to_ks(qs_env, ls_scf_env, energy_new, iscf)
549 : TYPE(qs_environment_type), POINTER :: qs_env
550 : TYPE(ls_scf_env_type) :: ls_scf_env
551 : REAL(KIND=dp) :: energy_new
552 : INTEGER, INTENT(IN) :: iscf
553 :
554 : CHARACTER(len=*), PARAMETER :: routineN = 'ls_scf_dm_to_ks'
555 :
556 : INTEGER :: handle, ispin, nspin, unit_nr
557 : TYPE(cp_logger_type), POINTER :: logger
558 3422 : TYPE(dbcsr_p_type), DIMENSION(:), POINTER :: rho_ao
559 : TYPE(mp_para_env_type), POINTER :: para_env
560 : TYPE(qs_energy_type), POINTER :: energy
561 : TYPE(qs_rho_type), POINTER :: rho
562 :
563 3422 : NULLIFY (energy, rho, rho_ao)
564 3422 : CALL timeset(routineN, handle)
565 :
566 3422 : logger => cp_get_default_logger()
567 3422 : IF (logger%para_env%is_source()) THEN
568 1711 : unit_nr = cp_logger_get_default_unit_nr(logger, local=.TRUE.)
569 : ELSE
570 : unit_nr = -1
571 : END IF
572 :
573 3422 : nspin = ls_scf_env%nspins
574 3422 : CALL get_qs_env(qs_env, para_env=para_env, energy=energy, rho=rho)
575 3422 : CALL qs_rho_get(rho, rho_ao=rho_ao)
576 :
577 : ! set the new density matrix
578 7044 : DO ispin = 1, nspin
579 : CALL matrix_ls_to_qs(rho_ao(ispin)%matrix, ls_scf_env%matrix_p(ispin), &
580 7044 : ls_scf_env%ls_mstruct, covariant=.FALSE.)
581 : END DO
582 :
583 : ! compute the corresponding KS matrix and new energy, mix density if requested
584 3422 : CALL qs_rho_update_rho(rho, qs_env=qs_env)
585 3422 : IF (ls_scf_env%do_rho_mixing) THEN
586 0 : IF (ls_scf_env%density_mixing_method == direct_mixing_nr) &
587 0 : CPABORT("Direct P mixing not implemented in linear scaling SCF. ")
588 0 : IF (ls_scf_env%density_mixing_method >= gspace_mixing_nr) THEN
589 0 : IF (iscf > MAX(ls_scf_env%mixing_store%nskip_mixing, 1)) THEN
590 : CALL gspace_mixing(qs_env, ls_scf_env%density_mixing_method, &
591 : ls_scf_env%mixing_store, rho, para_env, &
592 0 : iscf - 1)
593 0 : IF (unit_nr > 0) THEN
594 : WRITE (unit_nr, '(A57)') &
595 0 : "*********************************************************"
596 : WRITE (unit_nr, '(A13,F5.3,A20,A6,A7,I3)') &
597 0 : " Using ALPHA=", ls_scf_env%mixing_store%alpha, &
598 0 : " to mix rho: method=", ls_scf_env%mixing_store%iter_method, ", iscf=", iscf
599 : WRITE (unit_nr, '(A8,F5.3,A6,F5.3,A8)') &
600 0 : " rho_nw=", ls_scf_env%mixing_store%alpha, "*rho + ", &
601 0 : 1.0_dp - ls_scf_env%mixing_store%alpha, "*rho_old"
602 : WRITE (unit_nr, '(A57)') &
603 0 : "*********************************************************"
604 : END IF
605 : END IF
606 : END IF
607 : END IF
608 :
609 3422 : CALL qs_ks_did_change(qs_env%ks_env, rho_changed=.TRUE.)
610 : CALL qs_ks_update_qs_env(qs_env, calculate_forces=.FALSE., &
611 3422 : just_energy=.FALSE., print_active=.TRUE.)
612 3422 : CALL ls_scf_tblite_energy(qs_env, energy)
613 3422 : energy_new = energy%total
614 :
615 3422 : CALL timestop(handle)
616 :
617 3422 : END SUBROUTINE ls_scf_dm_to_ks
618 :
619 : ! **************************************************************************************************
620 : !> \brief use the external density in ls_scf_env to compute the new KS matrix
621 : !> \param qs_env ...
622 : !> \param ls_scf_env ...
623 : !> \param energy_new ...
624 : ! **************************************************************************************************
625 66 : SUBROUTINE ls_nonscf_ks(qs_env, ls_scf_env, energy_new)
626 : TYPE(qs_environment_type), POINTER :: qs_env
627 : TYPE(ls_scf_env_type) :: ls_scf_env
628 : REAL(KIND=dp) :: energy_new
629 :
630 : CHARACTER(len=*), PARAMETER :: routineN = 'ls_nonscf_ks'
631 :
632 : INTEGER :: handle, ispin, nspin
633 66 : TYPE(dbcsr_p_type), DIMENSION(:), POINTER :: rho_ao
634 : TYPE(harris_type), POINTER :: harris_env
635 : TYPE(mp_para_env_type), POINTER :: para_env
636 : TYPE(qs_energy_type), POINTER :: energy
637 : TYPE(qs_rho_type), POINTER :: rho
638 :
639 66 : NULLIFY (energy, rho, rho_ao)
640 66 : CALL timeset(routineN, handle)
641 :
642 66 : nspin = ls_scf_env%nspins
643 66 : CALL get_qs_env(qs_env, para_env=para_env, energy=energy, rho=rho)
644 66 : CALL qs_rho_get(rho, rho_ao=rho_ao)
645 :
646 : ! set the new density matrix
647 132 : DO ispin = 1, nspin
648 : CALL matrix_ls_to_qs(rho_ao(ispin)%matrix, ls_scf_env%matrix_p(ispin), &
649 132 : ls_scf_env%ls_mstruct, covariant=.FALSE.)
650 : END DO
651 :
652 66 : IF (qs_env%harris_method) THEN
653 14 : CALL get_qs_env(qs_env, harris_env=harris_env)
654 14 : CALL harris_density_update(qs_env, harris_env)
655 : END IF
656 : ! compute the corresponding KS matrix and new energy
657 66 : CALL qs_rho_update_rho(rho, qs_env=qs_env)
658 66 : IF (ls_scf_env%do_rho_mixing) THEN
659 0 : CPABORT("P mixing not implemented in linear scaling NONSCF. ")
660 : END IF
661 :
662 66 : CALL qs_ks_did_change(qs_env%ks_env, rho_changed=.TRUE.)
663 : CALL qs_ks_update_qs_env(qs_env, calculate_forces=.FALSE., &
664 66 : just_energy=.FALSE., print_active=.TRUE.)
665 66 : CALL ls_scf_tblite_energy(qs_env, energy)
666 66 : energy_new = energy%total
667 :
668 66 : CALL timestop(handle)
669 :
670 66 : END SUBROUTINE ls_nonscf_ks
671 :
672 : ! **************************************************************************************************
673 : !> \brief use the new density matrix in ls_scf_env to compute the new energy
674 : !> \param qs_env ...
675 : !> \param ls_scf_env ...
676 : ! **************************************************************************************************
677 66 : SUBROUTINE ls_nonscf_energy(qs_env, ls_scf_env)
678 : TYPE(qs_environment_type), POINTER :: qs_env
679 : TYPE(ls_scf_env_type) :: ls_scf_env
680 :
681 : CHARACTER(len=*), PARAMETER :: routineN = 'ls_nonscf_energy'
682 :
683 : INTEGER :: handle, ispin, nspin
684 66 : TYPE(dbcsr_p_type), DIMENSION(:), POINTER :: matrix_h, matrix_ks, rho_ao
685 : TYPE(mp_para_env_type), POINTER :: para_env
686 : TYPE(qs_energy_type), POINTER :: energy
687 : TYPE(qs_rho_type), POINTER :: rho
688 :
689 66 : NULLIFY (energy, rho, rho_ao)
690 66 : CALL timeset(routineN, handle)
691 66 : IF (qs_env%qmmm) THEN
692 0 : CPABORT("NYA")
693 : END IF
694 :
695 66 : nspin = ls_scf_env%nspins
696 66 : CALL get_qs_env(qs_env, para_env=para_env, energy=energy, rho=rho)
697 66 : CALL qs_rho_get(rho, rho_ao=rho_ao)
698 :
699 : ! set the new density matrix
700 132 : DO ispin = 1, nspin
701 : CALL matrix_ls_to_qs(rho_ao(ispin)%matrix, ls_scf_env%matrix_p(ispin), &
702 132 : ls_scf_env%ls_mstruct, covariant=.FALSE.)
703 : END DO
704 :
705 66 : CALL qs_ks_did_change(qs_env%ks_env, rho_changed=.TRUE.)
706 :
707 : ! band energy : Tr(PH)
708 66 : CALL get_qs_env(qs_env, matrix_ks=matrix_ks)
709 66 : CALL calculate_ptrace(matrix_ks, rho_ao, energy%band, nspin, .TRUE.)
710 : ! core energy : Tr(Ph)
711 66 : energy%total = energy%total - energy%core
712 66 : CALL get_qs_env(qs_env, matrix_h=matrix_h)
713 66 : CALL calculate_ptrace(matrix_h, rho_ao, energy%core, nspin)
714 :
715 66 : CALL timestop(handle)
716 :
717 66 : END SUBROUTINE ls_nonscf_energy
718 :
719 : ! **************************************************************************************************
720 : !> \brief update CP2K/tblite total energy after an LS_SCF KS rebuild.
721 : !> \param qs_env ...
722 : !> \param energy ...
723 : ! **************************************************************************************************
724 3842 : SUBROUTINE ls_scf_tblite_energy(qs_env, energy)
725 : TYPE(qs_environment_type), POINTER :: qs_env
726 : TYPE(qs_energy_type), POINTER :: energy
727 :
728 : TYPE(dft_control_type), POINTER :: dft_control
729 : TYPE(tblite_type), POINTER :: tb
730 :
731 3842 : NULLIFY (dft_control, tb)
732 3842 : CALL get_qs_env(qs_env, dft_control=dft_control, tb_tblite=tb)
733 :
734 3842 : IF (dft_control%qs_control%xtb .AND. dft_control%qs_control%xtb_control%do_tblite) THEN
735 112 : CPASSERT(ASSOCIATED(tb))
736 112 : CALL tb_get_energy(qs_env, tb, energy)
737 : END IF
738 :
739 3842 : END SUBROUTINE ls_scf_tblite_energy
740 :
741 : ! **************************************************************************************************
742 : !> \brief ...
743 : !> \param qs_env ...
744 : !> \param ls_scf_env ...
745 : !> \param matrix_p_ls ...
746 : !> \param unit_nr ...
747 : !> \param title ...
748 : !> \param stride ...
749 : ! **************************************************************************************************
750 6 : SUBROUTINE write_matrix_to_cube(qs_env, ls_scf_env, matrix_p_ls, unit_nr, title, stride)
751 : TYPE(qs_environment_type), POINTER :: qs_env
752 : TYPE(ls_scf_env_type) :: ls_scf_env
753 : TYPE(dbcsr_type), INTENT(IN) :: matrix_p_ls
754 : INTEGER, INTENT(IN) :: unit_nr
755 : CHARACTER(LEN=*), INTENT(IN) :: title
756 : INTEGER, DIMENSION(:), POINTER :: stride
757 :
758 : CHARACTER(len=*), PARAMETER :: routineN = 'write_matrix_to_cube'
759 :
760 : INTEGER :: handle
761 6 : TYPE(dbcsr_p_type), DIMENSION(:), POINTER :: matrix_ks
762 : TYPE(dbcsr_type), TARGET :: matrix_p_qs
763 : TYPE(particle_list_type), POINTER :: particles
764 : TYPE(pw_c1d_gs_type) :: wf_g
765 : TYPE(pw_env_type), POINTER :: pw_env
766 6 : TYPE(pw_pool_p_type), DIMENSION(:), POINTER :: pw_pools
767 : TYPE(pw_pool_type), POINTER :: auxbas_pw_pool
768 : TYPE(pw_r3d_rs_type) :: wf_r
769 : TYPE(qs_ks_env_type), POINTER :: ks_env
770 : TYPE(qs_subsys_type), POINTER :: subsys
771 :
772 6 : CALL timeset(routineN, handle)
773 :
774 6 : NULLIFY (ks_env, pw_env, auxbas_pw_pool, pw_pools, particles, subsys, matrix_ks)
775 :
776 : CALL get_qs_env(qs_env, &
777 : ks_env=ks_env, &
778 : subsys=subsys, &
779 : pw_env=pw_env, &
780 6 : matrix_ks=matrix_ks)
781 :
782 6 : CALL qs_subsys_get(subsys, particles=particles)
783 :
784 : ! convert the density matrix (ls style) to QS style
785 6 : CALL dbcsr_copy(matrix_p_qs, matrix_ks(1)%matrix)
786 6 : CALL dbcsr_set(matrix_p_qs, 0.0_dp) !zero matrix creation
787 6 : CALL matrix_ls_to_qs(matrix_p_qs, matrix_p_ls, ls_scf_env%ls_mstruct, covariant=.FALSE.)
788 :
789 : ! Print total electronic density
790 : CALL pw_env_get(pw_env=pw_env, &
791 : auxbas_pw_pool=auxbas_pw_pool, &
792 6 : pw_pools=pw_pools)
793 6 : CALL auxbas_pw_pool%create_pw(pw=wf_r)
794 6 : CALL pw_zero(wf_r)
795 6 : CALL auxbas_pw_pool%create_pw(pw=wf_g)
796 6 : CALL pw_zero(wf_g)
797 : CALL calculate_rho_elec(matrix_p=matrix_p_qs, &
798 : rho=wf_r, &
799 : rho_gspace=wf_g, &
800 6 : ks_env=ks_env)
801 :
802 : ! write this to a cube
803 : CALL cp_pw_to_cube(wf_r, unit_nr=unit_nr, title=title, &
804 6 : particles=particles, stride=stride)
805 :
806 : !free memory
807 6 : CALL auxbas_pw_pool%give_back_pw(wf_r)
808 6 : CALL auxbas_pw_pool%give_back_pw(wf_g)
809 6 : CALL dbcsr_release(matrix_p_qs)
810 :
811 6 : CALL timestop(handle)
812 :
813 6 : END SUBROUTINE write_matrix_to_cube
814 :
815 : ! **************************************************************************************************
816 : !> \brief Initialize g-space density mixing
817 : !> \param qs_env ...
818 : !> \param ls_scf_env ...
819 : ! **************************************************************************************************
820 0 : SUBROUTINE rho_mixing_ls_init(qs_env, ls_scf_env)
821 : TYPE(qs_environment_type), POINTER :: qs_env
822 : TYPE(ls_scf_env_type) :: ls_scf_env
823 :
824 : CHARACTER(len=*), PARAMETER :: routineN = 'rho_mixing_ls_init'
825 :
826 : INTEGER :: handle
827 : TYPE(dft_control_type), POINTER :: dft_control
828 : TYPE(qs_rho_type), POINTER :: rho
829 0 : TYPE(rho_atom_type), DIMENSION(:), POINTER :: rho_atom
830 :
831 0 : CALL timeset(routineN, handle)
832 :
833 0 : CALL get_qs_env(qs_env, dft_control=dft_control, rho=rho)
834 :
835 : CALL mixing_allocate(qs_env, ls_scf_env%density_mixing_method, nspins=ls_scf_env%nspins, &
836 0 : mixing_store=ls_scf_env%mixing_store)
837 0 : IF (ls_scf_env%density_mixing_method >= gspace_mixing_nr) THEN
838 0 : IF (dft_control%qs_control%gapw) THEN
839 0 : CALL get_qs_env(qs_env, rho_atom_set=rho_atom)
840 : CALL mixing_init(ls_scf_env%density_mixing_method, rho, ls_scf_env%mixing_store, &
841 0 : ls_scf_env%para_env, rho_atom=rho_atom)
842 0 : ELSEIF (dft_control%qs_control%dftb .OR. dft_control%qs_control%xtb) THEN
843 0 : CALL charge_mixing_init(ls_scf_env%mixing_store)
844 0 : ELSEIF (dft_control%qs_control%semi_empirical) THEN
845 0 : CPABORT('SE Code not possible')
846 : ELSE
847 : CALL mixing_init(ls_scf_env%density_mixing_method, rho, ls_scf_env%mixing_store, &
848 0 : ls_scf_env%para_env)
849 : END IF
850 : END IF
851 0 : CALL timestop(handle)
852 0 : END SUBROUTINE rho_mixing_ls_init
853 :
854 : END MODULE dm_ls_scf_qs
|