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 Definition and initialisation of the mo data type.
10 : !> \par History
11 : !> - adapted to the new QS environment data structure (02.04.2002,MK)
12 : !> - set_mo_occupation added (17.04.02,MK)
13 : !> - correct_mo_eigenvalues added (18.04.02,MK)
14 : !> - calculate_density_matrix moved from qs_scf to here (22.04.02,MK)
15 : !> - mo_set_p_type added (23.04.02,MK)
16 : !> - PRIVATE attribute set for TYPE mo_set_type (23.04.02,MK)
17 : !> - started conversion to LSD (1.2003, Joost VandeVondele)
18 : !> - Split of from qs_mo_types (07.2014, JGH)
19 : !> \author Matthias Krack (09.05.2001,MK)
20 : ! **************************************************************************************************
21 : MODULE qs_mo_io
22 :
23 : USE atomic_kind_types, ONLY: get_atomic_kind
24 : USE basis_set_types, ONLY: get_gto_basis_set,&
25 : gto_basis_set_p_type,&
26 : gto_basis_set_type
27 : USE cp_dbcsr_api, ONLY: dbcsr_binary_write,&
28 : dbcsr_create,&
29 : dbcsr_p_type,&
30 : dbcsr_release,&
31 : dbcsr_type
32 : USE cp_dbcsr_contrib, ONLY: dbcsr_checksum
33 : USE cp_dbcsr_operations, ONLY: copy_dbcsr_to_fm,&
34 : copy_fm_to_dbcsr,&
35 : dbcsr_deallocate_matrix_set
36 : USE cp_dbcsr_output, ONLY: cp_dbcsr_write_sparse_matrix
37 : USE cp_files, ONLY: close_file,&
38 : open_file
39 : USE cp_fm_types, ONLY: cp_fm_get_info,&
40 : cp_fm_get_submatrix,&
41 : cp_fm_set_all,&
42 : cp_fm_set_submatrix,&
43 : cp_fm_to_fm,&
44 : cp_fm_type,&
45 : cp_fm_write_unformatted
46 : USE cp_log_handling, ONLY: cp_get_default_logger,&
47 : cp_logger_get_default_unit_nr,&
48 : cp_logger_type,&
49 : cp_to_string
50 : USE cp_output_handling, ONLY: cp_p_file,&
51 : cp_print_key_finished_output,&
52 : cp_print_key_generate_filename,&
53 : cp_print_key_should_output,&
54 : cp_print_key_unit_nr
55 : USE input_section_types, ONLY: section_vals_get_subs_vals,&
56 : section_vals_type,&
57 : section_vals_val_get
58 : USE kahan_sum, ONLY: accurate_sum
59 : USE kinds, ONLY: default_path_length,&
60 : default_string_length,&
61 : dp
62 : USE message_passing, ONLY: mp_para_env_type
63 : USE orbital_pointers, ONLY: indco,&
64 : nco,&
65 : nso
66 : USE orbital_symbols, ONLY: cgf_symbol,&
67 : sgf_symbol
68 : USE orbital_transformation_matrices, ONLY: orbtramat
69 : USE particle_types, ONLY: particle_type
70 : USE physcon, ONLY: evolt
71 : USE qs_density_matrices, ONLY: calculate_density_matrix
72 : USE qs_dftb_types, ONLY: qs_dftb_atom_type
73 : USE qs_dftb_utils, ONLY: get_dftb_atom_param
74 : USE qs_environment_types, ONLY: get_qs_env,&
75 : qs_environment_type
76 : USE qs_kind_types, ONLY: get_qs_kind,&
77 : get_qs_kind_set,&
78 : qs_kind_type
79 : USE qs_ks_types, ONLY: qs_ks_env_type
80 : USE qs_mo_methods, ONLY: calculate_subspace_eigenvalues
81 : USE qs_mo_occupation, ONLY: set_mo_occupation
82 : USE qs_mo_types, ONLY: get_mo_set,&
83 : mo_set_type
84 : USE qs_neighbor_list_types, ONLY: neighbor_list_set_p_type,&
85 : release_neighbor_list_sets
86 : USE qs_neighbor_lists, ONLY: setup_neighbor_list
87 : USE qs_overlap, ONLY: build_overlap_matrix_simple
88 : #include "./base/base_uses.f90"
89 :
90 : IMPLICIT NONE
91 :
92 : PRIVATE
93 :
94 : CHARACTER(len=*), PARAMETER, PRIVATE :: moduleN = 'qs_mo_io'
95 :
96 : PUBLIC :: wfn_restart_file_name, &
97 : write_rt_mos_to_restart, &
98 : read_rt_mos_from_restart, &
99 : write_dm_binary_restart, &
100 : write_mo_set_to_output_unit, &
101 : write_mo_set_to_restart, &
102 : read_mo_set_from_restart, &
103 : read_mos_restart_low, &
104 : write_mo_set_low
105 :
106 : CONTAINS
107 :
108 : ! **************************************************************************************************
109 : !> \brief ...
110 : !> \param mo_array ...
111 : !> \param particle_set ...
112 : !> \param dft_section ...
113 : !> \param qs_kind_set ...
114 : !> \param matrix_ks ...
115 : ! **************************************************************************************************
116 187803 : SUBROUTINE write_mo_set_to_restart(mo_array, particle_set, dft_section, qs_kind_set, matrix_ks)
117 :
118 : TYPE(mo_set_type), DIMENSION(:), INTENT(IN) :: mo_array
119 : TYPE(particle_type), DIMENSION(:), POINTER :: particle_set
120 : TYPE(section_vals_type), POINTER :: dft_section
121 : TYPE(qs_kind_type), DIMENSION(:), POINTER :: qs_kind_set
122 : TYPE(dbcsr_p_type), DIMENSION(:), OPTIONAL, &
123 : POINTER :: matrix_ks
124 :
125 : CHARACTER(LEN=*), PARAMETER :: routineN = 'write_mo_set_to_restart'
126 : CHARACTER(LEN=30), DIMENSION(2), PARAMETER :: &
127 : keys = ["SCF%PRINT%RESTART_HISTORY", "SCF%PRINT%RESTART "]
128 :
129 : INTEGER :: handle, ikey, ires, ispin
130 : TYPE(cp_logger_type), POINTER :: logger
131 :
132 187803 : CALL timeset(routineN, handle)
133 :
134 187803 : logger => cp_get_default_logger()
135 :
136 : IF (BTEST(cp_print_key_should_output(logger%iter_info, &
137 187803 : dft_section, keys(1)), cp_p_file) .OR. &
138 : BTEST(cp_print_key_should_output(logger%iter_info, &
139 : dft_section, keys(2)), cp_p_file)) THEN
140 :
141 18977 : IF (mo_array(1)%use_mo_coeff_b) THEN
142 : ! we are using the dbcsr mo_coeff
143 : ! we copy it to the fm for anycase
144 13206 : DO ispin = 1, SIZE(mo_array)
145 7131 : IF (.NOT. ASSOCIATED(mo_array(ispin)%mo_coeff_b)) THEN
146 0 : CPASSERT(.FALSE.)
147 : END IF
148 : CALL copy_dbcsr_to_fm(mo_array(ispin)%mo_coeff_b, &
149 13206 : mo_array(ispin)%mo_coeff) !fm->dbcsr
150 : END DO
151 : END IF
152 :
153 56931 : DO ikey = 1, SIZE(keys)
154 37954 : IF (BTEST(cp_print_key_should_output(logger%iter_info, &
155 187803 : dft_section, keys(ikey)), cp_p_file)) THEN
156 : ires = cp_print_key_unit_nr(logger, dft_section, keys(ikey), &
157 : extension=".wfn", file_status="REPLACE", file_action="WRITE", &
158 18989 : do_backup=.TRUE., file_form="UNFORMATTED")
159 18989 : IF (PRESENT(matrix_ks)) THEN
160 : CALL write_mo_set_low(mo_array, particle_set=particle_set, qs_kind_set=qs_kind_set, &
161 6059 : ires=ires, matrix_ks=matrix_ks)
162 : ELSE
163 : CALL write_mo_set_low(mo_array, particle_set=particle_set, qs_kind_set=qs_kind_set, &
164 12930 : ires=ires)
165 : END IF
166 18989 : CALL cp_print_key_finished_output(ires, logger, dft_section, TRIM(keys(ikey)))
167 : END IF
168 : END DO
169 : END IF
170 :
171 187803 : CALL timestop(handle)
172 :
173 187803 : END SUBROUTINE write_mo_set_to_restart
174 :
175 : ! **************************************************************************************************
176 : !> \brief calculates density matrix from mo set and writes the density matrix
177 : !> into a binary restart file
178 : !> \param mo_array mos
179 : !> \param dft_section dft input section
180 : !> \param tmpl_matrix template dbcsr matrix
181 : !> \author Mohammad Hossein Bani-Hashemian
182 : ! **************************************************************************************************
183 10879 : SUBROUTINE write_dm_binary_restart(mo_array, dft_section, tmpl_matrix)
184 :
185 : TYPE(mo_set_type), DIMENSION(:), INTENT(IN) :: mo_array
186 : TYPE(section_vals_type), POINTER :: dft_section
187 : TYPE(dbcsr_p_type), DIMENSION(:), POINTER :: tmpl_matrix
188 :
189 : CHARACTER(LEN=*), PARAMETER :: routineN = 'write_dm_binary_restart'
190 :
191 : CHARACTER(LEN=default_path_length) :: file_name, project_name
192 : INTEGER :: handle, ispin, unit_nr
193 : LOGICAL :: do_dm_restart
194 : REAL(KIND=dp) :: cs_pos
195 : TYPE(cp_logger_type), POINTER :: logger
196 : TYPE(dbcsr_type), POINTER :: matrix_p_tmp
197 :
198 10879 : CALL timeset(routineN, handle)
199 10879 : logger => cp_get_default_logger()
200 10879 : IF (logger%para_env%is_source()) THEN
201 5568 : unit_nr = cp_logger_get_default_unit_nr(logger, local=.TRUE.)
202 : ELSE
203 : unit_nr = -1
204 : END IF
205 :
206 10879 : project_name = logger%iter_info%project_name
207 10879 : CALL section_vals_val_get(dft_section, "SCF%PRINT%DM_RESTART_WRITE", l_val=do_dm_restart)
208 10879 : NULLIFY (matrix_p_tmp)
209 :
210 10879 : IF (do_dm_restart) THEN
211 0 : ALLOCATE (matrix_p_tmp)
212 0 : DO ispin = 1, SIZE(mo_array)
213 0 : CALL dbcsr_create(matrix_p_tmp, template=tmpl_matrix(ispin)%matrix, name="DM RESTART")
214 :
215 0 : IF (.NOT. ASSOCIATED(mo_array(ispin)%mo_coeff_b)) CPABORT("mo_coeff_b NOT ASSOCIATED")
216 :
217 0 : CALL copy_fm_to_dbcsr(mo_array(ispin)%mo_coeff, mo_array(ispin)%mo_coeff_b)
218 : CALL calculate_density_matrix(mo_array(ispin), matrix_p_tmp, &
219 0 : use_dbcsr=.TRUE., retain_sparsity=.FALSE.)
220 :
221 0 : WRITE (file_name, '(A,I0,A)') TRIM(project_name)//"_SCF_DM_SPIN_", ispin, "_RESTART.dm"
222 0 : cs_pos = dbcsr_checksum(matrix_p_tmp, pos=.TRUE.)
223 0 : IF (unit_nr > 0) THEN
224 0 : WRITE (unit_nr, '(T2,A,E20.8)') "Writing restart DM "//TRIM(file_name)//" with checksum: ", cs_pos
225 : END IF
226 0 : CALL dbcsr_binary_write(matrix_p_tmp, file_name)
227 :
228 0 : CALL dbcsr_release(matrix_p_tmp)
229 : END DO
230 0 : DEALLOCATE (matrix_p_tmp)
231 : END IF
232 :
233 10879 : CALL timestop(handle)
234 :
235 10879 : END SUBROUTINE write_dm_binary_restart
236 :
237 : ! **************************************************************************************************
238 : !> \brief ...
239 : !> \param mo_array ...
240 : !> \param rt_mos ...
241 : !> \param particle_set ...
242 : !> \param dft_section ...
243 : !> \param qs_kind_set ...
244 : ! **************************************************************************************************
245 444 : SUBROUTINE write_rt_mos_to_restart(mo_array, rt_mos, particle_set, dft_section, qs_kind_set)
246 :
247 : TYPE(mo_set_type), DIMENSION(:), INTENT(IN) :: mo_array
248 : TYPE(cp_fm_type), DIMENSION(:), INTENT(IN) :: rt_mos
249 : TYPE(particle_type), DIMENSION(:), POINTER :: particle_set
250 : TYPE(section_vals_type), POINTER :: dft_section
251 : TYPE(qs_kind_type), DIMENSION(:), POINTER :: qs_kind_set
252 :
253 : CHARACTER(LEN=*), PARAMETER :: routineN = 'write_rt_mos_to_restart'
254 : CHARACTER(LEN=43), DIMENSION(2), PARAMETER :: keys = [ &
255 : "REAL_TIME_PROPAGATION%PRINT%RESTART_HISTORY", &
256 : "REAL_TIME_PROPAGATION%PRINT%RESTART "]
257 :
258 : INTEGER :: handle, ikey, ires
259 : TYPE(cp_logger_type), POINTER :: logger
260 :
261 444 : CALL timeset(routineN, handle)
262 444 : logger => cp_get_default_logger()
263 :
264 : IF (BTEST(cp_print_key_should_output(logger%iter_info, &
265 444 : dft_section, keys(1)), cp_p_file) .OR. &
266 : BTEST(cp_print_key_should_output(logger%iter_info, &
267 : dft_section, keys(2)), cp_p_file)) THEN
268 :
269 342 : DO ikey = 1, SIZE(keys)
270 :
271 228 : IF (BTEST(cp_print_key_should_output(logger%iter_info, &
272 444 : dft_section, keys(ikey)), cp_p_file)) THEN
273 : ires = cp_print_key_unit_nr(logger, dft_section, keys(ikey), &
274 : extension=".rtpwfn", file_status="REPLACE", file_action="WRITE", &
275 114 : do_backup=.TRUE., file_form="UNFORMATTED")
276 : CALL write_mo_set_low(mo_array, qs_kind_set=qs_kind_set, particle_set=particle_set, &
277 114 : ires=ires, rt_mos=rt_mos)
278 114 : CALL cp_print_key_finished_output(ires, logger, dft_section, TRIM(keys(ikey)))
279 : END IF
280 : END DO
281 : END IF
282 :
283 444 : CALL timestop(handle)
284 :
285 444 : END SUBROUTINE write_rt_mos_to_restart
286 :
287 : ! **************************************************************************************************
288 : !> \brief ...
289 : !> \param mo_array ...
290 : !> \param qs_kind_set ...
291 : !> \param particle_set ...
292 : !> \param ires ...
293 : !> \param rt_mos ...
294 : !> \param matrix_ks ...
295 : ! **************************************************************************************************
296 19111 : SUBROUTINE write_mo_set_low(mo_array, qs_kind_set, particle_set, ires, rt_mos, matrix_ks)
297 :
298 : TYPE(mo_set_type), DIMENSION(:), INTENT(IN) :: mo_array
299 : TYPE(qs_kind_type), DIMENSION(:), POINTER :: qs_kind_set
300 : TYPE(particle_type), DIMENSION(:), POINTER :: particle_set
301 : INTEGER :: ires
302 : TYPE(cp_fm_type), DIMENSION(:), INTENT(IN), &
303 : OPTIONAL :: rt_mos
304 : TYPE(dbcsr_p_type), DIMENSION(:), OPTIONAL, &
305 : POINTER :: matrix_ks
306 :
307 : CHARACTER(LEN=*), PARAMETER :: routineN = 'write_mo_set_low'
308 :
309 : INTEGER :: handle, iatom, ikind, imat, iset, &
310 : ishell, ispin, lmax, lshell, &
311 : max_block, nao, natom, nmo, nset, &
312 : nset_max, nshell_max, nspin
313 19111 : INTEGER, DIMENSION(:), POINTER :: nset_info, nshell
314 19111 : INTEGER, DIMENSION(:, :), POINTER :: l, nshell_info
315 19111 : INTEGER, DIMENSION(:, :, :), POINTER :: nso_info
316 19111 : REAL(KIND=dp), DIMENSION(:), POINTER :: mo_eigenvalues, mo_occupation_numbers
317 : TYPE(cp_fm_type), POINTER :: mo_coeff
318 : TYPE(gto_basis_set_type), POINTER :: orb_basis_set
319 : TYPE(qs_dftb_atom_type), POINTER :: dftb_parameter
320 :
321 19111 : CALL timeset(routineN, handle)
322 :
323 19111 : NULLIFY (mo_coeff)
324 : NULLIFY (mo_eigenvalues)
325 19111 : NULLIFY (mo_occupation_numbers)
326 :
327 19111 : nspin = SIZE(mo_array)
328 19111 : nao = mo_array(1)%nao
329 :
330 19111 : IF (ires > 0) THEN
331 : ! Create some info about the basis set first
332 9731 : natom = SIZE(particle_set, 1)
333 9731 : nset_max = 0
334 9731 : nshell_max = 0
335 :
336 61659 : DO iatom = 1, natom
337 51928 : NULLIFY (orb_basis_set, dftb_parameter)
338 51928 : CALL get_atomic_kind(particle_set(iatom)%atomic_kind, kind_number=ikind)
339 : CALL get_qs_kind(qs_kind_set(ikind), &
340 : basis_set=orb_basis_set, &
341 51928 : dftb_parameter=dftb_parameter)
342 113587 : IF (ASSOCIATED(orb_basis_set)) THEN
343 : CALL get_gto_basis_set(gto_basis_set=orb_basis_set, &
344 : nset=nset, &
345 : nshell=nshell, &
346 44370 : l=l)
347 44370 : nset_max = MAX(nset_max, nset)
348 128325 : DO iset = 1, nset
349 128325 : nshell_max = MAX(nshell_max, nshell(iset))
350 : END DO
351 7558 : ELSEIF (ASSOCIATED(dftb_parameter)) THEN
352 7557 : CALL get_dftb_atom_param(dftb_parameter, lmax=lmax)
353 7557 : nset_max = MAX(nset_max, 1)
354 7557 : nshell_max = MAX(nshell_max, lmax + 1)
355 : ELSE
356 : ! We assume here an atom without a basis set
357 : ! CPABORT("Unknown basis type. ")
358 : END IF
359 : END DO
360 :
361 48655 : ALLOCATE (nso_info(nshell_max, nset_max, natom))
362 361075 : nso_info(:, :, :) = 0
363 :
364 38924 : ALLOCATE (nshell_info(nset_max, natom))
365 163534 : nshell_info(:, :) = 0
366 :
367 29193 : ALLOCATE (nset_info(natom))
368 61659 : nset_info(:) = 0
369 :
370 61659 : DO iatom = 1, natom
371 51928 : NULLIFY (orb_basis_set, dftb_parameter)
372 51928 : CALL get_atomic_kind(particle_set(iatom)%atomic_kind, kind_number=ikind)
373 : CALL get_qs_kind(qs_kind_set(ikind), &
374 51928 : basis_set=orb_basis_set, dftb_parameter=dftb_parameter)
375 113587 : IF (ASSOCIATED(orb_basis_set)) THEN
376 : CALL get_gto_basis_set(gto_basis_set=orb_basis_set, &
377 : nset=nset, &
378 : nshell=nshell, &
379 44370 : l=l)
380 44370 : nset_info(iatom) = nset
381 128325 : DO iset = 1, nset
382 83955 : nshell_info(iset, iatom) = nshell(iset)
383 245069 : DO ishell = 1, nshell(iset)
384 116744 : lshell = l(ishell, iset)
385 200699 : nso_info(ishell, iset, iatom) = nso(lshell)
386 : END DO
387 : END DO
388 7558 : ELSEIF (ASSOCIATED(dftb_parameter)) THEN
389 7557 : CALL get_dftb_atom_param(dftb_parameter, lmax=lmax)
390 7557 : nset_info(iatom) = 1
391 7557 : nshell_info(1, iatom) = lmax + 1
392 17706 : DO ishell = 1, lmax + 1
393 10149 : lshell = ishell - 1
394 17706 : nso_info(ishell, 1, iatom) = nso(lshell)
395 : END DO
396 : ELSE
397 : ! We assume here an atom without a basis set
398 : ! CPABORT("Unknown basis type. ")
399 : END IF
400 : END DO
401 :
402 9731 : WRITE (ires) natom, nspin, nao, nset_max, nshell_max
403 61659 : WRITE (ires) nset_info
404 163534 : WRITE (ires) nshell_info
405 361075 : WRITE (ires) nso_info
406 :
407 9731 : DEALLOCATE (nset_info)
408 :
409 9731 : DEALLOCATE (nshell_info)
410 :
411 9731 : DEALLOCATE (nso_info)
412 : END IF
413 :
414 : ! Use the ScaLAPACK block size as a default for buffering columns
415 19111 : CALL cp_fm_get_info(mo_array(1)%mo_coeff, ncol_block=max_block)
416 41230 : DO ispin = 1, nspin
417 22119 : mo_coeff => mo_array(ispin)%mo_coeff
418 22119 : nmo = mo_array(ispin)%nmo
419 22119 : IF (nmo > 0) THEN
420 21865 : mo_eigenvalues => mo_array(ispin)%eigenvalues
421 21865 : mo_occupation_numbers => mo_array(ispin)%occupation_numbers
422 21865 : IF (PRESENT(matrix_ks)) THEN
423 : ! With OT: use the Kohn-Sham matrix for the update of the MO eigenvalues
424 : CALL calculate_subspace_eigenvalues(orbitals=mo_coeff, &
425 : ks_matrix=matrix_ks(ispin)%matrix, &
426 7071 : evals_arg=mo_eigenvalues)
427 : END IF
428 21865 : IF (ires > 0) THEN
429 11117 : WRITE (ires) nmo, &
430 11117 : mo_array(ispin)%homo, &
431 11117 : mo_array(ispin)%lfomo, &
432 22234 : mo_array(ispin)%nelectron
433 248512 : WRITE (ires) mo_eigenvalues(1:nmo), mo_occupation_numbers(1:nmo)
434 : END IF
435 : END IF
436 41230 : IF (PRESENT(rt_mos)) THEN
437 438 : DO imat = 2*ispin - 1, 2*ispin
438 438 : CALL cp_fm_write_unformatted(rt_mos(imat), ires)
439 : END DO
440 : ELSE
441 21973 : CALL cp_fm_write_unformatted(mo_coeff, ires)
442 : END IF
443 : END DO
444 :
445 19111 : CALL timestop(handle)
446 :
447 19111 : END SUBROUTINE write_mo_set_low
448 :
449 : ! **************************************************************************************************
450 : !> \brief ...
451 : !> \param filename ...
452 : !> \param exist ...
453 : !> \param section ...
454 : !> \param logger ...
455 : !> \param kp ...
456 : !> \param xas ...
457 : !> \param rtp ...
458 : ! **************************************************************************************************
459 1250 : SUBROUTINE wfn_restart_file_name(filename, exist, section, logger, kp, xas, rtp)
460 : CHARACTER(LEN=default_path_length), INTENT(OUT) :: filename
461 : LOGICAL, INTENT(OUT) :: exist
462 : TYPE(section_vals_type), POINTER :: section
463 : TYPE(cp_logger_type), POINTER :: logger
464 : LOGICAL, INTENT(IN), OPTIONAL :: kp, xas, rtp
465 :
466 : INTEGER :: n_rep_val
467 : LOGICAL :: my_kp, my_rtp, my_xas
468 : TYPE(section_vals_type), POINTER :: print_key
469 :
470 625 : my_kp = .FALSE.
471 625 : my_xas = .FALSE.
472 625 : my_rtp = .FALSE.
473 625 : IF (PRESENT(kp)) my_kp = kp
474 625 : IF (PRESENT(xas)) my_xas = xas
475 625 : IF (PRESENT(rtp)) my_rtp = rtp
476 :
477 625 : exist = .FALSE.
478 625 : CALL section_vals_val_get(section, "WFN_RESTART_FILE_NAME", n_rep_val=n_rep_val)
479 625 : IF (n_rep_val > 0) THEN
480 474 : CALL section_vals_val_get(section, "WFN_RESTART_FILE_NAME", c_val=filename)
481 : ELSE
482 151 : IF (my_xas) THEN
483 : ! try to read from the filename that is generated automatically from the printkey
484 4 : print_key => section_vals_get_subs_vals(section, "PRINT%RESTART")
485 : filename = cp_print_key_generate_filename(logger, print_key, &
486 4 : extension="", my_local=.FALSE.)
487 147 : ELSE IF (my_rtp) THEN
488 : ! try to read from the filename that is generated automatically from the printkey
489 3 : print_key => section_vals_get_subs_vals(section, "REAL_TIME_PROPAGATION%PRINT%RESTART")
490 : filename = cp_print_key_generate_filename(logger, print_key, &
491 3 : extension=".rtpwfn", my_local=.FALSE.)
492 144 : ELSE IF (my_kp) THEN
493 : ! try to read from the filename that is generated automatically from the printkey
494 5 : print_key => section_vals_get_subs_vals(section, "SCF%PRINT%RESTART")
495 : filename = cp_print_key_generate_filename(logger, print_key, &
496 5 : extension=".kp", my_local=.FALSE.)
497 : ELSE
498 : ! try to read from the filename that is generated automatically from the printkey
499 139 : print_key => section_vals_get_subs_vals(section, "SCF%PRINT%RESTART")
500 : filename = cp_print_key_generate_filename(logger, print_key, &
501 139 : extension=".wfn", my_local=.FALSE.)
502 : END IF
503 : END IF
504 625 : IF (.NOT. my_xas) THEN
505 619 : INQUIRE (FILE=filename, exist=exist)
506 : END IF
507 :
508 625 : END SUBROUTINE wfn_restart_file_name
509 :
510 : ! **************************************************************************************************
511 : !> \brief ...
512 : !> \param mo_array ...
513 : !> \param qs_kind_set ...
514 : !> \param particle_set ...
515 : !> \param para_env ...
516 : !> \param id_nr ...
517 : !> \param multiplicity ...
518 : !> \param dft_section ...
519 : !> \param natom_mismatch ...
520 : !> \param cdft ...
521 : !> \param out_unit ...
522 : ! **************************************************************************************************
523 527 : SUBROUTINE read_mo_set_from_restart(mo_array, qs_kind_set, particle_set, &
524 : para_env, id_nr, multiplicity, dft_section, natom_mismatch, &
525 : cdft, out_unit)
526 :
527 : TYPE(mo_set_type), DIMENSION(:), INTENT(INOUT) :: mo_array
528 : TYPE(qs_kind_type), DIMENSION(:), POINTER :: qs_kind_set
529 : TYPE(particle_type), DIMENSION(:), POINTER :: particle_set
530 : TYPE(mp_para_env_type), POINTER :: para_env
531 : INTEGER, INTENT(IN) :: id_nr, multiplicity
532 : TYPE(section_vals_type), POINTER :: dft_section
533 : LOGICAL, INTENT(OUT), OPTIONAL :: natom_mismatch
534 : LOGICAL, INTENT(IN), OPTIONAL :: cdft
535 : INTEGER, INTENT(IN), OPTIONAL :: out_unit
536 :
537 : CHARACTER(LEN=*), PARAMETER :: routineN = 'read_mo_set_from_restart'
538 :
539 : CHARACTER(LEN=default_path_length) :: file_name
540 : INTEGER :: handle, ispin, my_out_unit, natom, &
541 : nspin, restart_unit
542 : LOGICAL :: exist, my_cdft
543 : TYPE(cp_logger_type), POINTER :: logger
544 :
545 527 : CALL timeset(routineN, handle)
546 527 : logger => cp_get_default_logger()
547 527 : my_cdft = .FALSE.
548 527 : IF (PRESENT(cdft)) my_cdft = cdft
549 527 : my_out_unit = -1
550 527 : IF (PRESENT(out_unit)) my_out_unit = out_unit
551 :
552 527 : nspin = SIZE(mo_array)
553 527 : restart_unit = -1
554 :
555 527 : IF (para_env%is_source()) THEN
556 :
557 282 : natom = SIZE(particle_set, 1)
558 282 : CALL wfn_restart_file_name(file_name, exist, dft_section, logger)
559 282 : IF (id_nr /= 0) THEN
560 : ! Is it one of the backup files?
561 1 : file_name = TRIM(file_name)//".bak-"//ADJUSTL(cp_to_string(id_nr))
562 : END IF
563 :
564 : CALL open_file(file_name=file_name, &
565 : file_action="READ", &
566 : file_form="UNFORMATTED", &
567 : file_status="OLD", &
568 282 : unit_number=restart_unit)
569 :
570 : END IF
571 :
572 : CALL read_mos_restart_low(mo_array, para_env=para_env, qs_kind_set=qs_kind_set, &
573 : particle_set=particle_set, natom=natom, &
574 527 : rst_unit=restart_unit, multiplicity=multiplicity, natom_mismatch=natom_mismatch)
575 :
576 527 : IF (PRESENT(natom_mismatch)) THEN
577 : ! read_mos_restart_low only the io_node returns natom_mismatch, must broadcast it
578 497 : CALL para_env%bcast(natom_mismatch)
579 497 : IF (natom_mismatch) THEN
580 0 : IF (para_env%is_source()) CALL close_file(unit_number=restart_unit)
581 0 : CALL timestop(handle)
582 0 : RETURN
583 : END IF
584 : END IF
585 :
586 : ! Close restart file
587 527 : IF (para_env%is_source()) THEN
588 282 : IF (my_out_unit > 0) THEN
589 : WRITE (UNIT=my_out_unit, FMT="(T2,A)") &
590 6 : "WFN_RESTART| Restart file "//TRIM(file_name)//" read"
591 : END IF
592 282 : CALL close_file(unit_number=restart_unit)
593 : END IF
594 :
595 : ! CDFT has no real dft_section and does not need to print
596 527 : IF (.NOT. my_cdft) THEN
597 1408 : DO ispin = 1, nspin
598 : CALL write_mo_set_to_output_unit(mo_array(ispin), qs_kind_set, particle_set, &
599 1408 : dft_section, 4, 0, final_mos=.FALSE.)
600 : END DO
601 : END IF
602 :
603 527 : CALL timestop(handle)
604 :
605 : END SUBROUTINE read_mo_set_from_restart
606 :
607 : ! **************************************************************************************************
608 : !> \brief ...
609 : !> \param mo_array ...
610 : !> \param rt_mos ...
611 : !> \param qs_kind_set ...
612 : !> \param particle_set ...
613 : !> \param para_env ...
614 : !> \param id_nr ...
615 : !> \param multiplicity ...
616 : !> \param dft_section ...
617 : ! **************************************************************************************************
618 8 : SUBROUTINE read_rt_mos_from_restart(mo_array, rt_mos, qs_kind_set, particle_set, para_env, id_nr, multiplicity, dft_section)
619 :
620 : TYPE(mo_set_type), DIMENSION(:), INTENT(INOUT) :: mo_array
621 : TYPE(cp_fm_type), DIMENSION(:), POINTER :: rt_mos
622 : TYPE(qs_kind_type), DIMENSION(:), POINTER :: qs_kind_set
623 : TYPE(particle_type), DIMENSION(:), POINTER :: particle_set
624 : TYPE(mp_para_env_type), POINTER :: para_env
625 : INTEGER, INTENT(IN) :: id_nr, multiplicity
626 : TYPE(section_vals_type), POINTER :: dft_section
627 :
628 : CHARACTER(LEN=*), PARAMETER :: routineN = 'read_rt_mos_from_restart'
629 :
630 : CHARACTER(LEN=default_path_length) :: file_name
631 : INTEGER :: handle, ispin, natom, nspin, &
632 : restart_unit, unit_nr
633 : LOGICAL :: exist
634 : TYPE(cp_logger_type), POINTER :: logger
635 :
636 8 : CALL timeset(routineN, handle)
637 8 : logger => cp_get_default_logger()
638 :
639 8 : nspin = SIZE(mo_array)
640 8 : restart_unit = -1
641 :
642 8 : IF (para_env%is_source()) THEN
643 :
644 4 : natom = SIZE(particle_set, 1)
645 4 : CALL wfn_restart_file_name(file_name, exist, dft_section, logger, rtp=.TRUE.)
646 4 : IF (id_nr /= 0) THEN
647 : ! Is it one of the backup files?
648 0 : file_name = TRIM(file_name)//".bak-"//ADJUSTL(cp_to_string(id_nr))
649 : END IF
650 :
651 4 : unit_nr = cp_logger_get_default_unit_nr(logger, local=.TRUE.)
652 4 : IF (unit_nr > 0) THEN
653 4 : WRITE (unit_nr, '(T2,A)') "Read RTP restart from the file: "//TRIM(file_name)
654 : END IF
655 :
656 : CALL open_file(file_name=file_name, &
657 : file_action="READ", &
658 : file_form="UNFORMATTED", &
659 : file_status="OLD", &
660 4 : unit_number=restart_unit)
661 :
662 : END IF
663 :
664 : CALL read_mos_restart_low(mo_array, rt_mos=rt_mos, para_env=para_env, &
665 : particle_set=particle_set, qs_kind_set=qs_kind_set, natom=natom, &
666 8 : rst_unit=restart_unit, multiplicity=multiplicity)
667 :
668 : ! Close restart file
669 8 : IF (para_env%is_source()) CALL close_file(unit_number=restart_unit)
670 :
671 16 : DO ispin = 1, nspin
672 : CALL write_mo_set_to_output_unit(mo_array(ispin), qs_kind_set, particle_set, &
673 16 : dft_section, 4, 0, final_mos=.FALSE.)
674 : END DO
675 :
676 8 : CALL timestop(handle)
677 :
678 8 : END SUBROUTINE read_rt_mos_from_restart
679 :
680 : ! **************************************************************************************************
681 : !> \brief Reading the mos from apreviously defined restart file
682 : !> \param mos ...
683 : !> \param para_env ...
684 : !> \param qs_kind_set ...
685 : !> \param particle_set ...
686 : !> \param natom ...
687 : !> \param rst_unit ...
688 : !> \param multiplicity ...
689 : !> \param rt_mos ...
690 : !> \param natom_mismatch ...
691 : !> \par History
692 : !> 12.2007 created [MI]
693 : !> \author MI
694 : ! **************************************************************************************************
695 585 : SUBROUTINE read_mos_restart_low(mos, para_env, qs_kind_set, particle_set, natom, rst_unit, &
696 : multiplicity, rt_mos, natom_mismatch)
697 :
698 : TYPE(mo_set_type), DIMENSION(:), INTENT(INOUT) :: mos
699 : TYPE(mp_para_env_type), POINTER :: para_env
700 : TYPE(qs_kind_type), DIMENSION(:), POINTER :: qs_kind_set
701 : TYPE(particle_type), DIMENSION(:), POINTER :: particle_set
702 : INTEGER, INTENT(IN) :: natom, rst_unit
703 : INTEGER, INTENT(in), OPTIONAL :: multiplicity
704 : TYPE(cp_fm_type), DIMENSION(:), OPTIONAL, POINTER :: rt_mos
705 : LOGICAL, INTENT(OUT), OPTIONAL :: natom_mismatch
706 :
707 : INTEGER :: homo, homo_read, i, iatom, ikind, imat, irow, iset, iset_read, ishell, &
708 : ishell_read, iso, ispin, lfomo_read, lmax, lshell, my_mult, nao, nao_read, natom_read, &
709 : nelectron, nelectron_read, nmo, nmo_read, nnshell, nset, nset_max, nshell_max, nspin, &
710 : nspin_read, offset_read
711 585 : INTEGER, DIMENSION(:), POINTER :: nset_info, nshell
712 585 : INTEGER, DIMENSION(:, :), POINTER :: l, nshell_info
713 585 : INTEGER, DIMENSION(:, :, :), POINTER :: nso_info, offset_info
714 : LOGICAL :: minbas, natom_match, use_this
715 585 : REAL(KIND=dp), ALLOCATABLE, DIMENSION(:) :: eig_read, occ_read
716 585 : REAL(KIND=dp), DIMENSION(:, :), POINTER :: vecbuffer, vecbuffer_read
717 : TYPE(cp_logger_type), POINTER :: logger
718 : TYPE(gto_basis_set_type), POINTER :: orb_basis_set
719 : TYPE(qs_dftb_atom_type), POINTER :: dftb_parameter
720 :
721 1170 : logger => cp_get_default_logger()
722 :
723 585 : nspin = SIZE(mos)
724 585 : nao = mos(1)%nao
725 585 : my_mult = 0
726 585 : IF (PRESENT(multiplicity)) my_mult = multiplicity
727 :
728 585 : IF (para_env%is_source()) THEN
729 311 : READ (rst_unit) natom_read, nspin_read, nao_read, nset_max, nshell_max
730 311 : IF (PRESENT(rt_mos)) THEN
731 4 : IF (nspin_read /= nspin) THEN
732 0 : CPABORT("To change nspin is not possible. ")
733 : END IF
734 : ELSE
735 : ! we should allow for restarting with different spin settings
736 307 : IF (nspin_read /= nspin) THEN
737 : WRITE (cp_logger_get_default_unit_nr(logger), *) &
738 0 : "READ RESTART : WARNING : nspin is not equal "
739 : END IF
740 : ! this case needs fixing of homo/lfomo/nelec/occupations ...
741 307 : IF (nspin_read > nspin) THEN
742 0 : CPABORT("Reducing nspin is not possible. ")
743 : END IF
744 : END IF
745 :
746 311 : natom_match = (natom_read == natom)
747 :
748 311 : IF (natom_match) THEN ! actually do the read read
749 :
750 : ! Let's make it possible to change the basis set
751 1555 : ALLOCATE (nso_info(nshell_max, nset_max, natom_read))
752 1244 : ALLOCATE (nshell_info(nset_max, natom_read))
753 933 : ALLOCATE (nset_info(natom_read))
754 1244 : ALLOCATE (offset_info(nshell_max, nset_max, natom_read))
755 :
756 311 : IF (nao_read /= nao) THEN
757 : WRITE (cp_logger_get_default_unit_nr(logger), *) &
758 1 : " READ RESTART : WARNING : DIFFERENT # AOs ", nao, nao_read
759 1 : IF (PRESENT(rt_mos)) &
760 0 : CPABORT("To change basis is not possible. ")
761 : END IF
762 :
763 1140 : READ (rst_unit) nset_info
764 2829 : READ (rst_unit) nshell_info
765 6925 : READ (rst_unit) nso_info
766 :
767 311 : i = 1
768 1140 : DO iatom = 1, natom
769 2663 : DO iset = 1, nset_info(iatom)
770 4879 : DO ishell = 1, nshell_info(iset, iatom)
771 2527 : offset_info(ishell, iset, iatom) = i
772 4050 : i = i + nso_info(ishell, iset, iatom)
773 : END DO
774 : END DO
775 : END DO
776 :
777 933 : ALLOCATE (vecbuffer_read(1, nao_read))
778 :
779 : END IF ! natom_match
780 : END IF ! ionode
781 :
782 : ! make natom_match and natom_mismatch uniform across all nodes
783 585 : CALL para_env%bcast(natom_match)
784 585 : IF (PRESENT(natom_mismatch)) natom_mismatch = .NOT. natom_match
785 : ! handle natom_match false
786 585 : IF (.NOT. natom_match) THEN
787 0 : IF (PRESENT(natom_mismatch)) THEN
788 : WRITE (cp_logger_get_default_unit_nr(logger), *) &
789 0 : " READ RESTART : WARNING : DIFFERENT natom, returning ", natom, natom_read
790 : RETURN
791 : ELSE
792 0 : CPABORT("Incorrect number of atoms in restart file. ")
793 : END IF
794 : END IF
795 :
796 585 : CALL para_env%bcast(nspin_read)
797 :
798 1755 : ALLOCATE (vecbuffer(1, nao))
799 :
800 1596 : DO ispin = 1, nspin
801 :
802 1011 : nmo = mos(ispin)%nmo
803 1011 : homo = mos(ispin)%homo
804 5286 : mos(ispin)%eigenvalues(:) = 0.0_dp
805 5286 : mos(ispin)%occupation_numbers(:) = 0.0_dp
806 1011 : CALL cp_fm_set_all(mos(ispin)%mo_coeff, 0.0_dp)
807 :
808 1011 : IF (para_env%is_source() .AND. (nmo > 0)) THEN
809 533 : READ (rst_unit) nmo_read, homo_read, lfomo_read, nelectron_read
810 2132 : ALLOCATE (eig_read(nmo_read), occ_read(nmo_read))
811 533 : eig_read = 0.0_dp
812 533 : occ_read = 0.0_dp
813 :
814 533 : nmo = MIN(nmo, nmo_read)
815 : IF (nmo_read < nmo) &
816 : CALL cp_warn(__LOCATION__, &
817 : "The number of MOs on the restart unit is smaller than the number of "// &
818 : "the allocated MOs. The MO set will be padded with zeros!")
819 533 : IF (nmo_read > nmo) &
820 : CALL cp_warn(__LOCATION__, &
821 : "The number of MOs on the restart unit is greater than the number of "// &
822 7 : "the allocated MOs. The read MO set will be truncated!")
823 :
824 533 : READ (rst_unit) eig_read(1:nmo_read), occ_read(1:nmo_read)
825 2704 : mos(ispin)%eigenvalues(1:nmo) = eig_read(1:nmo)
826 2704 : mos(ispin)%occupation_numbers(1:nmo) = occ_read(1:nmo)
827 533 : DEALLOCATE (eig_read, occ_read)
828 :
829 533 : mos(ispin)%homo = homo_read
830 533 : mos(ispin)%lfomo = lfomo_read
831 533 : IF (homo_read > nmo) THEN
832 0 : IF (nelectron_read == mos(ispin)%nelectron) THEN
833 : CALL cp_warn(__LOCATION__, &
834 : "The number of occupied MOs on the restart unit is larger than "// &
835 0 : "the allocated MOs. The read MO set will be truncated and the occupation numbers recalculated!")
836 0 : CALL set_mo_occupation(mo_set=mos(ispin))
837 : ELSE
838 : ! can not make this a warning i.e. homo must be smaller than nmo
839 : ! otherwise e.g. set_mo_occupation will go out of bounds
840 0 : CPABORT("Number of occupied MOs on restart unit larger than allocated MOs. ")
841 : END IF
842 : END IF
843 : END IF
844 :
845 1011 : CALL para_env%bcast(nmo)
846 1011 : CALL para_env%bcast(mos(ispin)%homo)
847 1011 : CALL para_env%bcast(mos(ispin)%lfomo)
848 1011 : CALL para_env%bcast(mos(ispin)%nelectron)
849 9561 : CALL para_env%bcast(mos(ispin)%eigenvalues)
850 9561 : CALL para_env%bcast(mos(ispin)%occupation_numbers)
851 :
852 1011 : IF (PRESENT(rt_mos)) THEN
853 24 : DO imat = 2*ispin - 1, 2*ispin
854 40 : DO i = 1, nmo
855 16 : IF (para_env%is_source()) THEN
856 168 : READ (rst_unit) vecbuffer
857 : ELSE
858 88 : vecbuffer(1, :) = 0.0_dp
859 : END IF
860 656 : CALL para_env%bcast(vecbuffer)
861 : CALL cp_fm_set_submatrix(rt_mos(imat), &
862 32 : vecbuffer, 1, i, nao, 1, transpose=.TRUE.)
863 : END DO
864 : END DO
865 : ELSE
866 5246 : DO i = 1, nmo
867 4243 : IF (para_env%is_source()) THEN
868 145287 : READ (rst_unit) vecbuffer_read
869 : ! now, try to assign the read to the real vector
870 : ! in case the basis set changed this involves some guessing
871 2167 : irow = 1
872 9702 : DO iatom = 1, natom
873 7535 : NULLIFY (orb_basis_set, dftb_parameter, l, nshell)
874 7535 : CALL get_atomic_kind(particle_set(iatom)%atomic_kind, kind_number=ikind)
875 : CALL get_qs_kind(qs_kind_set(ikind), &
876 7535 : basis_set=orb_basis_set, dftb_parameter=dftb_parameter)
877 7535 : IF (ASSOCIATED(orb_basis_set)) THEN
878 : CALL get_gto_basis_set(gto_basis_set=orb_basis_set, &
879 : nset=nset, &
880 : nshell=nshell, &
881 7487 : l=l)
882 7487 : minbas = .FALSE.
883 48 : ELSEIF (ASSOCIATED(dftb_parameter)) THEN
884 48 : CALL get_dftb_atom_param(dftb_parameter, lmax=lmax)
885 48 : nset = 1
886 48 : minbas = .TRUE.
887 : ELSE
888 : ! assume an atom without basis set
889 : ! CPABORT("Unknown basis set type. ")
890 0 : nset = 0
891 : END IF
892 :
893 7535 : use_this = .TRUE.
894 7535 : iset_read = 1
895 35114 : DO iset = 1, nset
896 17877 : ishell_read = 1
897 17877 : IF (minbas) THEN
898 48 : nnshell = lmax + 1
899 : ELSE
900 17829 : nnshell = nshell(iset)
901 : END IF
902 56888 : DO ishell = 1, nnshell
903 31476 : IF (minbas) THEN
904 72 : lshell = ishell - 1
905 : ELSE
906 31404 : lshell = l(ishell, iset)
907 : END IF
908 31476 : IF (iset_read > nset_info(iatom)) use_this = .FALSE.
909 : IF (use_this) THEN ! avoids out of bound access of the lower line if false
910 31452 : IF (nso(lshell) == nso_info(ishell_read, iset_read, iatom)) THEN
911 31452 : offset_read = offset_info(ishell_read, iset_read, iatom)
912 31452 : ishell_read = ishell_read + 1
913 31452 : IF (ishell_read > nshell_info(iset, iatom)) THEN
914 17873 : ishell_read = 1
915 17873 : iset_read = iset_read + 1
916 : END IF
917 : ELSE
918 : use_this = .FALSE.
919 : END IF
920 : END IF
921 103180 : DO iso = 1, nso(lshell)
922 71704 : IF (use_this) THEN
923 71560 : IF (offset_read - 1 + iso < 1 .OR. offset_read - 1 + iso > nao_read) THEN
924 0 : vecbuffer(1, irow) = 0.0_dp
925 : ELSE
926 71560 : vecbuffer(1, irow) = vecbuffer_read(1, offset_read - 1 + iso)
927 : END IF
928 : ELSE
929 144 : vecbuffer(1, irow) = 0.0_dp
930 : END IF
931 103180 : irow = irow + 1
932 : END DO
933 49353 : use_this = .TRUE.
934 : END DO
935 : END DO
936 : END DO
937 :
938 : ELSE
939 :
940 72302 : vecbuffer(1, :) = 0.0_dp
941 :
942 : END IF
943 :
944 571963 : CALL para_env%bcast(vecbuffer)
945 : CALL cp_fm_set_submatrix(mos(ispin)%mo_coeff, &
946 5246 : vecbuffer, 1, i, nao, 1, transpose=.TRUE.)
947 : END DO
948 : END IF
949 : ! Skip extra MOs if there any
950 1011 : IF (para_env%is_source()) THEN
951 : !ignore nmo = 0
952 536 : IF (nmo > 0) THEN
953 563 : DO i = nmo + 1, nmo_read
954 1783 : READ (rst_unit) vecbuffer_read
955 : END DO
956 : END IF
957 : END IF
958 :
959 1596 : IF (.NOT. PRESENT(rt_mos)) THEN
960 1003 : IF (ispin == 1 .AND. nspin_read < nspin) THEN
961 :
962 0 : mos(ispin + 1)%homo = mos(ispin)%homo
963 0 : mos(ispin + 1)%lfomo = mos(ispin)%lfomo
964 0 : nelectron = mos(ispin)%nelectron
965 0 : IF (my_mult /= 1) THEN
966 : CALL cp_abort(__LOCATION__, &
967 0 : "Restarting an LSD calculation from an LDA wfn only works for multiplicity=1 (singlets).")
968 : END IF
969 0 : IF (mos(ispin + 1)%nelectron < 0) THEN
970 0 : CPABORT("LSD: too few electrons for this multiplisity. ")
971 : END IF
972 0 : mos(ispin + 1)%eigenvalues = mos(ispin)%eigenvalues
973 0 : mos(ispin)%occupation_numbers = mos(ispin)%occupation_numbers/2.0_dp
974 0 : mos(ispin + 1)%occupation_numbers = mos(ispin)%occupation_numbers
975 0 : CALL cp_fm_to_fm(mos(ispin)%mo_coeff, mos(ispin + 1)%mo_coeff)
976 0 : EXIT
977 : END IF
978 : END IF
979 : END DO ! ispin
980 :
981 585 : DEALLOCATE (vecbuffer)
982 :
983 585 : IF (para_env%is_source()) THEN
984 311 : DEALLOCATE (vecbuffer_read)
985 311 : DEALLOCATE (offset_info)
986 311 : DEALLOCATE (nso_info)
987 311 : DEALLOCATE (nshell_info)
988 311 : DEALLOCATE (nset_info)
989 : END IF
990 :
991 1170 : END SUBROUTINE read_mos_restart_low
992 :
993 : ! **************************************************************************************************
994 : !> \brief Write MO information to output file (eigenvalues, occupation numbers, coefficients)
995 : !> \param mo_set ...
996 : !> \param qs_kind_set ...
997 : !> \param particle_set ...
998 : !> \param dft_section ...
999 : !> \param before Digits before the dot
1000 : !> \param kpoint An integer that labels the current k point, e.g. its index
1001 : !> \param final_mos ...
1002 : !> \param spin ...
1003 : !> \param solver_method ...
1004 : !> \param rtp ...
1005 : !> \param cpart ...
1006 : !> \param sim_step ...
1007 : !> \param umo_set ...
1008 : !> \param qs_env ...
1009 : !> \date 15.05.2001
1010 : !> \par History:
1011 : !> - Optionally print Cartesian MOs (20.04.2005, MK)
1012 : !> - Revise printout of MO information (05.05.2021, MK)
1013 : !> \par Variables
1014 : !> - after : Number of digits after point.
1015 : !> - before: Number of digits before point.
1016 : !> \author Matthias Krack (MK)
1017 : !> \version 1.1
1018 : ! **************************************************************************************************
1019 6419 : SUBROUTINE write_mo_set_to_output_unit(mo_set, qs_kind_set, particle_set, &
1020 : dft_section, before, kpoint, final_mos, spin, &
1021 : solver_method, rtp, cpart, sim_step, umo_set, qs_env)
1022 :
1023 : TYPE(mo_set_type), INTENT(IN) :: mo_set
1024 : TYPE(qs_kind_type), DIMENSION(:), POINTER :: qs_kind_set
1025 : TYPE(particle_type), DIMENSION(:), POINTER :: particle_set
1026 : TYPE(section_vals_type), POINTER :: dft_section
1027 : INTEGER, INTENT(IN) :: before, kpoint
1028 : LOGICAL, INTENT(IN), OPTIONAL :: final_mos
1029 : CHARACTER(LEN=*), INTENT(IN), OPTIONAL :: spin
1030 : CHARACTER(LEN=2), INTENT(IN), OPTIONAL :: solver_method
1031 : LOGICAL, INTENT(IN), OPTIONAL :: rtp
1032 : INTEGER, INTENT(IN), OPTIONAL :: cpart, sim_step
1033 : TYPE(mo_set_type), INTENT(IN), OPTIONAL :: umo_set
1034 : TYPE(qs_environment_type), OPTIONAL, POINTER :: qs_env
1035 :
1036 : CHARACTER(LEN=12) :: symbol
1037 6419 : CHARACTER(LEN=12), DIMENSION(:), POINTER :: bcgf_symbol
1038 : CHARACTER(LEN=14) :: fmtstr5
1039 : CHARACTER(LEN=15) :: energy_str, orbital_str, step_string
1040 : CHARACTER(LEN=2) :: element_symbol, my_solver_method
1041 : CHARACTER(LEN=2*default_string_length) :: name
1042 : CHARACTER(LEN=21) :: vector_str
1043 : CHARACTER(LEN=22) :: fmtstr4
1044 : CHARACTER(LEN=24) :: fmtstr2
1045 : CHARACTER(LEN=25) :: fmtstr1
1046 : CHARACTER(LEN=29) :: fmtstr6
1047 : CHARACTER(LEN=4) :: reim
1048 : CHARACTER(LEN=40) :: fmtstr3
1049 6419 : CHARACTER(LEN=6), DIMENSION(:), POINTER :: bsgf_symbol
1050 : INTEGER :: after, first_mo, from, homo, iatom, icgf, ico, icol, ikind, imo, irow, iset, &
1051 : isgf, ishell, iso, iw, jcol, last_mo, left, lmax, lshell, nao, natom, ncgf, ncol, nkind, &
1052 : nmo, nset, nsgf, numo, right, scf_step, to, width
1053 6419 : INTEGER, DIMENSION(:), POINTER :: mo_index_range, nshell
1054 6419 : INTEGER, DIMENSION(:, :), POINTER :: l
1055 : LOGICAL :: ionode, my_final, my_rtp, omit_headers, print_cartesian, print_cartesian_overlap, &
1056 : print_eigvals, print_eigvecs, print_occup, should_output
1057 : REAL(KIND=dp) :: gap, maxocc
1058 6419 : REAL(KIND=dp), ALLOCATABLE, DIMENSION(:) :: mo_eigenvalues, mo_occupation_numbers
1059 6419 : REAL(KIND=dp), ALLOCATABLE, DIMENSION(:, :) :: cmatrix, smatrix
1060 6419 : REAL(KIND=dp), DIMENSION(:), POINTER :: eigenvalues, occupation_numbers
1061 : TYPE(cp_fm_type), POINTER :: mo_coeff, umo_coeff
1062 : TYPE(cp_logger_type), POINTER :: logger
1063 6419 : TYPE(dbcsr_p_type), DIMENSION(:), POINTER :: sro
1064 6419 : TYPE(gto_basis_set_p_type), DIMENSION(:), POINTER :: orb_basis_set_list
1065 : TYPE(gto_basis_set_type), POINTER :: orb_basis_set, orbbasis
1066 : TYPE(mp_para_env_type), POINTER :: para_env
1067 : TYPE(neighbor_list_set_p_type), DIMENSION(:), &
1068 6419 : POINTER :: sro_list
1069 : TYPE(qs_dftb_atom_type), POINTER :: dftb_parameter
1070 : TYPE(qs_kind_type), POINTER :: qs_kind
1071 : TYPE(qs_ks_env_type), POINTER :: ks_env
1072 :
1073 6419 : NULLIFY (bcgf_symbol)
1074 6419 : NULLIFY (bsgf_symbol)
1075 6419 : NULLIFY (logger)
1076 6419 : NULLIFY (mo_index_range)
1077 6419 : NULLIFY (nshell)
1078 6419 : NULLIFY (mo_coeff)
1079 :
1080 12838 : logger => cp_get_default_logger()
1081 6419 : ionode = logger%para_env%is_source()
1082 6419 : CALL section_vals_val_get(dft_section, "PRINT%MO%EIGENVALUES", l_val=print_eigvals)
1083 6419 : CALL section_vals_val_get(dft_section, "PRINT%MO%EIGENVECTORS", l_val=print_eigvecs)
1084 6419 : CALL section_vals_val_get(dft_section, "PRINT%MO%OCCUPATION_NUMBERS", l_val=print_occup)
1085 6419 : CALL section_vals_val_get(dft_section, "PRINT%MO%CARTESIAN", l_val=print_cartesian)
1086 6419 : CALL section_vals_val_get(dft_section, "PRINT%MO%MO_INDEX_RANGE", i_vals=mo_index_range)
1087 6419 : CALL section_vals_val_get(dft_section, "PRINT%MO%NDIGITS", i_val=after)
1088 6419 : CALL section_vals_val_get(dft_section, "PRINT%MO%CARTESIAN_OVERLAP", l_val=print_cartesian_overlap)
1089 6419 : after = MIN(MAX(after, 1), 16)
1090 :
1091 : ! Do we print the final MO information after SCF convergence is reached (default: no)
1092 6419 : IF (PRESENT(final_mos)) THEN
1093 6411 : my_final = final_mos
1094 : ELSE
1095 : my_final = .FALSE.
1096 : END IF
1097 :
1098 : ! complex MOS for RTP, no eigenvalues
1099 6419 : my_rtp = .FALSE.
1100 6419 : IF (PRESENT(rtp)) THEN
1101 8 : my_rtp = rtp
1102 : ! print the first time step if MO print required
1103 : should_output = BTEST(cp_print_key_should_output(logger%iter_info, dft_section, &
1104 : "PRINT%MO"), cp_p_file) &
1105 8 : .OR. (sim_step == 1)
1106 : ELSE
1107 : should_output = BTEST(cp_print_key_should_output(logger%iter_info, dft_section, &
1108 7338 : "PRINT%MO"), cp_p_file) .OR. my_final
1109 : END IF
1110 :
1111 6419 : IF ((.NOT. should_output) .OR. (.NOT. (print_eigvals .OR. print_eigvecs .OR. print_occup))) RETURN
1112 :
1113 5488 : IF (my_rtp) THEN
1114 8 : CPASSERT(PRESENT(sim_step))
1115 8 : CPASSERT(PRESENT(cpart))
1116 8 : scf_step = sim_step
1117 8 : IF (cpart == 0) THEN
1118 4 : reim = "IMAG"
1119 : ELSE
1120 4 : reim = "REAL"
1121 : END IF
1122 8 : print_eigvals = .FALSE.
1123 : ELSE
1124 5480 : scf_step = MAX(0, logger%iter_info%iteration(logger%iter_info%n_rlevel) - 1)
1125 : END IF
1126 :
1127 5488 : IF (.NOT. my_final) THEN
1128 4222 : IF (.NOT. my_rtp) THEN
1129 4214 : step_string = " AFTER SCF STEP"
1130 : ELSE
1131 8 : step_string = " AFTER RTP STEP"
1132 : END IF
1133 : END IF
1134 :
1135 5488 : IF (PRESENT(solver_method)) THEN
1136 5334 : my_solver_method = solver_method
1137 : ELSE
1138 : ! Traditional diagonalization is assumed as default solver method
1139 154 : my_solver_method = "TD"
1140 : END IF
1141 :
1142 : ! Retrieve MO information
1143 : CALL get_mo_set(mo_set=mo_set, &
1144 : mo_coeff=mo_coeff, &
1145 : eigenvalues=eigenvalues, &
1146 : occupation_numbers=occupation_numbers, &
1147 : homo=homo, &
1148 : maxocc=maxocc, &
1149 : nao=nao, &
1150 5488 : nmo=nmo)
1151 5488 : IF (PRESENT(umo_set)) THEN
1152 : CALL get_mo_set(mo_set=umo_set, &
1153 : mo_coeff=umo_coeff, &
1154 20 : nmo=numo)
1155 20 : nmo = nmo + numo
1156 : ELSE
1157 5468 : numo = 0
1158 : END IF
1159 16414 : ALLOCATE (mo_eigenvalues(nmo))
1160 5488 : mo_eigenvalues(:) = 0.0_dp
1161 48566 : mo_eigenvalues(1:nmo - numo) = eigenvalues(1:nmo - numo)
1162 16414 : ALLOCATE (mo_occupation_numbers(nmo))
1163 5488 : mo_occupation_numbers(:) = 0.0_dp
1164 48566 : mo_occupation_numbers(1:nmo - numo) = occupation_numbers(1:nmo - numo)
1165 5488 : IF (numo > 0) THEN
1166 : CALL get_mo_set(mo_set=umo_set, &
1167 20 : eigenvalues=eigenvalues)
1168 130 : mo_eigenvalues(nmo - numo + 1:nmo) = eigenvalues(1:numo)
1169 : END IF
1170 :
1171 5488 : IF (print_eigvecs) THEN
1172 13974 : ALLOCATE (smatrix(nao, nmo))
1173 3506 : CALL cp_fm_get_submatrix(mo_coeff, smatrix(1:nao, 1:nmo - numo))
1174 3506 : IF (numo > 0) THEN
1175 14 : CALL cp_fm_get_submatrix(umo_coeff, smatrix(1:nao, nmo - numo + 1:nmo))
1176 : END IF
1177 3506 : IF (.NOT. ionode) THEN
1178 1753 : DEALLOCATE (smatrix)
1179 : END IF
1180 : END IF
1181 :
1182 5488 : IF (PRESENT(qs_env)) THEN
1183 5334 : IF (ASSOCIATED(qs_env) .AND. my_final .AND. print_cartesian_overlap) THEN
1184 2 : NULLIFY (qs_kind_set)
1185 2 : CALL get_qs_env(qs_env=qs_env, qs_kind_set=qs_kind_set)
1186 2 : nkind = SIZE(qs_kind_set)
1187 :
1188 2 : IF (BTEST(cp_print_key_should_output(logger%iter_info, &
1189 : qs_env%input, "DFT%PRINT%AO_MATRICES/OVERLAP"), cp_p_file)) THEN
1190 8 : ALLOCATE (orb_basis_set_list(nkind))
1191 4 : DO ikind = 1, nkind
1192 2 : qs_kind => qs_kind_set(ikind)
1193 2 : NULLIFY (orb_basis_set_list(ikind)%gto_basis_set)
1194 2 : NULLIFY (orbbasis)
1195 2 : CALL get_qs_kind(qs_kind=qs_kind, basis_set=orbbasis, basis_type="ORB")
1196 4 : IF (ASSOCIATED(orbbasis)) orb_basis_set_list(ikind)%gto_basis_set => orbbasis
1197 : END DO
1198 2 : NULLIFY (sro_list)
1199 2 : CALL setup_neighbor_list(sro_list, orb_basis_set_list, qs_env=qs_env)
1200 2 : NULLIFY (sro)
1201 2 : NULLIFY (para_env)
1202 2 : CALL get_qs_env(qs_env, ks_env=ks_env, para_env=para_env)
1203 : CALL build_overlap_matrix_simple(ks_env, sro, &
1204 2 : orb_basis_set_list, orb_basis_set_list, sro_list, .TRUE.)
1205 2 : CALL release_neighbor_list_sets(sro_list)
1206 :
1207 : iw = cp_print_key_unit_nr(logger, qs_env%input, "DFT%PRINT%AO_MATRICES/OVERLAP", &
1208 2 : extension=".Log")
1209 2 : CALL section_vals_val_get(qs_env%input, "DFT%PRINT%AO_MATRICES%OMIT_HEADERS", l_val=omit_headers)
1210 2 : CALL section_vals_val_get(qs_env%input, "DFT%PRINT%AO_MATRICES%NDIGITS", i_val=after)
1211 2 : after = MIN(MAX(after, 1), 16)
1212 2 : IF (ASSOCIATED(sro)) THEN
1213 : CALL cp_dbcsr_write_sparse_matrix(sro(1)%matrix, 4, after, qs_env, para_env, &
1214 : output_unit=iw, omit_headers=omit_headers, &
1215 2 : cartesian_basis=.TRUE.)
1216 : END IF
1217 : CALL cp_print_key_finished_output(iw, logger, qs_env%input, &
1218 2 : "DFT%PRINT%AO_MATRICES/OVERLAP")
1219 2 : IF (ASSOCIATED(sro)) CALL dbcsr_deallocate_matrix_set(sro)
1220 4 : DEALLOCATE (orb_basis_set_list)
1221 : END IF
1222 : END IF
1223 : END IF
1224 :
1225 : iw = cp_print_key_unit_nr(logger, dft_section, "PRINT%MO", &
1226 : ignore_should_output=should_output, &
1227 5488 : extension=".MOLog")
1228 :
1229 5488 : IF (iw > 0) THEN
1230 :
1231 2744 : natom = SIZE(particle_set)
1232 2744 : CALL get_qs_kind_set(qs_kind_set, ncgf=ncgf, nsgf=nsgf)
1233 :
1234 : ! Definition of the variable formats
1235 :
1236 2744 : fmtstr1 = "(T2,A,21X, ( X,I5, X))"
1237 2744 : fmtstr2 = "(T2,A,21X, (1X,F . ))"
1238 2744 : fmtstr3 = "(T2,A,I5,1X,I5,1X,A,1X,A6, (1X,F . ))"
1239 :
1240 2744 : width = before + after + 3
1241 2744 : ncol = INT(56/width)
1242 :
1243 2744 : right = MAX((after - 2), 1)
1244 2744 : left = width - right - 5
1245 :
1246 2744 : WRITE (UNIT=fmtstr1(11:12), FMT="(I2)") ncol
1247 2744 : WRITE (UNIT=fmtstr1(14:15), FMT="(I2)") left
1248 2744 : WRITE (UNIT=fmtstr1(21:22), FMT="(I2)") right
1249 :
1250 2744 : WRITE (UNIT=fmtstr2(11:12), FMT="(I2)") ncol
1251 2744 : WRITE (UNIT=fmtstr2(18:19), FMT="(I2)") width - 1
1252 2744 : WRITE (UNIT=fmtstr2(21:22), FMT="(I2)") after
1253 :
1254 2744 : WRITE (UNIT=fmtstr3(27:28), FMT="(I2)") ncol
1255 2744 : WRITE (UNIT=fmtstr3(34:35), FMT="(I2)") width - 1
1256 2744 : WRITE (UNIT=fmtstr3(37:38), FMT="(I2)") after
1257 :
1258 2744 : IF (my_final .OR. (my_solver_method == "TD")) THEN
1259 2744 : energy_str = "EIGENVALUES"
1260 2744 : vector_str = "EIGENVECTORS"
1261 : ELSE
1262 0 : energy_str = "ENERGIES"
1263 0 : vector_str = "COEFFICIENTS"
1264 : END IF
1265 :
1266 2744 : IF (my_rtp) THEN
1267 4 : energy_str = "ZEROS"
1268 4 : vector_str = TRIM(reim)//" RTP COEFFICIENTS"
1269 : END IF
1270 :
1271 2744 : IF (print_eigvecs) THEN
1272 :
1273 1753 : IF (print_cartesian) THEN
1274 :
1275 97 : orbital_str = "CARTESIAN"
1276 :
1277 388 : ALLOCATE (cmatrix(ncgf, ncgf))
1278 97 : cmatrix = 0.0_dp
1279 :
1280 : ! Transform spherical MOs to Cartesian MOs
1281 97 : icgf = 1
1282 97 : isgf = 1
1283 307 : DO iatom = 1, natom
1284 210 : NULLIFY (orb_basis_set, dftb_parameter)
1285 210 : CALL get_atomic_kind(particle_set(iatom)%atomic_kind, kind_number=ikind)
1286 : CALL get_qs_kind(qs_kind_set(ikind), &
1287 : basis_set=orb_basis_set, &
1288 210 : dftb_parameter=dftb_parameter)
1289 517 : IF (ASSOCIATED(orb_basis_set)) THEN
1290 : CALL get_gto_basis_set(gto_basis_set=orb_basis_set, &
1291 : nset=nset, &
1292 : nshell=nshell, &
1293 174 : l=l)
1294 538 : DO iset = 1, nset
1295 1064 : DO ishell = 1, nshell(iset)
1296 526 : lshell = l(ishell, iset)
1297 : CALL dgemm("T", "N", nco(lshell), nmo, nso(lshell), 1.0_dp, &
1298 : orbtramat(lshell)%c2s, nso(lshell), &
1299 : smatrix(isgf, 1), nsgf, 0.0_dp, &
1300 526 : cmatrix(icgf, 1), ncgf)
1301 526 : icgf = icgf + nco(lshell)
1302 890 : isgf = isgf + nso(lshell)
1303 : END DO
1304 : END DO
1305 36 : ELSE IF (ASSOCIATED(dftb_parameter)) THEN
1306 36 : CALL get_dftb_atom_param(dftb_parameter, lmax=lmax)
1307 90 : DO ishell = 1, lmax + 1
1308 54 : lshell = ishell - 1
1309 : CALL dgemm("T", "N", nco(lshell), nsgf, nso(lshell), 1.0_dp, &
1310 : orbtramat(lshell)%c2s, nso(lshell), &
1311 : smatrix(isgf, 1), nsgf, 0.0_dp, &
1312 54 : cmatrix(icgf, 1), ncgf)
1313 54 : icgf = icgf + nco(lshell)
1314 90 : isgf = isgf + nso(lshell)
1315 : END DO
1316 : ELSE
1317 : ! assume atom without basis set
1318 : ! CPABORT("Unknown basis set type")
1319 : END IF
1320 : END DO ! iatom
1321 :
1322 : ELSE
1323 :
1324 1656 : orbital_str = "SPHERICAL"
1325 :
1326 : END IF ! print_cartesian
1327 :
1328 : name = TRIM(energy_str)//", OCCUPATION NUMBERS, AND "// &
1329 1753 : TRIM(orbital_str)//" "//TRIM(vector_str)
1330 :
1331 1753 : IF (.NOT. my_final) &
1332 1408 : WRITE (UNIT=name, FMT="(A,1X,I0)") TRIM(name)//step_string, scf_step
1333 :
1334 991 : ELSE IF (print_occup .OR. print_eigvals) THEN
1335 991 : name = TRIM(energy_str)//" AND OCCUPATION NUMBERS"
1336 :
1337 991 : IF (.NOT. my_final) &
1338 703 : WRITE (UNIT=name, FMT="(A,1X,I0)") TRIM(name)//step_string, scf_step
1339 : END IF ! print_eigvecs
1340 :
1341 : ! Print headline
1342 2744 : IF (PRESENT(spin) .AND. (kpoint > 0)) THEN
1343 : WRITE (UNIT=iw, FMT="(/,T2,A,I0)") &
1344 0 : "MO| "//TRIM(spin)//" "//TRIM(name)//" FOR K POINT ", kpoint
1345 : ELSE IF (PRESENT(spin)) THEN
1346 : WRITE (UNIT=iw, FMT="(/,T2,A)") &
1347 312 : "MO| "//TRIM(spin)//" "//TRIM(name)
1348 2432 : ELSE IF (kpoint > 0) THEN
1349 : WRITE (UNIT=iw, FMT="(/,T2,A,I0)") &
1350 293 : "MO| "//TRIM(name)//" FOR K POINT ", kpoint
1351 : ELSE
1352 : WRITE (UNIT=iw, FMT="(/,T2,A)") &
1353 2139 : "MO| "//TRIM(name)
1354 : END IF
1355 :
1356 : ! Check if only a subset of the MOs has to be printed
1357 3028 : IF (ALL(mo_index_range > 0)) THEN
1358 142 : IF (mo_index_range(2) > nmo) THEN
1359 : CALL cp_warn(__LOCATION__, &
1360 7 : "The last orbital index is larger than the number of orbitals.")
1361 : END IF
1362 142 : IF (mo_index_range(1) > mo_index_range(2)) THEN
1363 : CALL cp_warn(__LOCATION__, &
1364 0 : "The first orbital index is larger than the last orbital index.")
1365 : END IF
1366 142 : first_mo = MIN(MAX(1, mo_index_range(1)), nmo)
1367 142 : last_mo = MIN(MAX(first_mo, mo_index_range(2)), nmo)
1368 2602 : ELSE IF (mo_index_range(2) < 0) THEN
1369 0 : IF (mo_index_range(1) > nmo) THEN
1370 : CALL cp_warn(__LOCATION__, &
1371 0 : "The first orbital index is larger than the number of orbitals.")
1372 : END IF
1373 0 : first_mo = MIN(MAX(1, mo_index_range(1)), nmo)
1374 0 : last_mo = nmo
1375 : ELSE
1376 2602 : first_mo = 1
1377 2602 : last_mo = nmo
1378 : END IF
1379 :
1380 2744 : IF (print_eigvecs) THEN
1381 :
1382 : ! Print full MO information
1383 :
1384 3952 : DO icol = first_mo, last_mo, ncol
1385 :
1386 2199 : from = icol
1387 2199 : to = MIN((from + ncol - 1), last_mo)
1388 :
1389 2199 : WRITE (UNIT=iw, FMT="(T2,A)") "MO|"
1390 : WRITE (UNIT=iw, FMT=fmtstr1) &
1391 10206 : "MO|", (jcol, jcol=from, to)
1392 : WRITE (UNIT=iw, FMT=fmtstr2) &
1393 2199 : "MO|", (mo_eigenvalues(jcol), jcol=from, to)
1394 2199 : WRITE (UNIT=iw, FMT="(T2,A)") "MO|"
1395 : WRITE (UNIT=iw, FMT=fmtstr2) &
1396 2199 : "MO|", (mo_occupation_numbers(jcol), jcol=from, to)
1397 2199 : WRITE (UNIT=iw, FMT="(T2,A)") "MO|"
1398 :
1399 2199 : irow = 1
1400 :
1401 8687 : DO iatom = 1, natom
1402 :
1403 4735 : IF (iatom /= 1) WRITE (UNIT=iw, FMT="(T2,A)") "MO|"
1404 :
1405 4735 : NULLIFY (orb_basis_set, dftb_parameter)
1406 : CALL get_atomic_kind(particle_set(iatom)%atomic_kind, &
1407 4735 : element_symbol=element_symbol, kind_number=ikind)
1408 : CALL get_qs_kind(qs_kind_set(ikind), &
1409 : basis_set=orb_basis_set, &
1410 4735 : dftb_parameter=dftb_parameter)
1411 :
1412 11669 : IF (print_cartesian) THEN
1413 :
1414 876 : IF (ASSOCIATED(orb_basis_set)) THEN
1415 : CALL get_gto_basis_set(gto_basis_set=orb_basis_set, &
1416 : nset=nset, &
1417 : nshell=nshell, &
1418 : l=l, &
1419 768 : cgf_symbol=bcgf_symbol)
1420 :
1421 768 : icgf = 1
1422 2592 : DO iset = 1, nset
1423 5064 : DO ishell = 1, nshell(iset)
1424 2472 : lshell = l(ishell, iset)
1425 10968 : DO ico = 1, nco(lshell)
1426 : WRITE (UNIT=iw, FMT=fmtstr3) &
1427 6672 : "MO|", irow, iatom, ADJUSTR(element_symbol), bcgf_symbol(icgf), &
1428 13344 : (cmatrix(irow, jcol), jcol=from, to)
1429 6672 : icgf = icgf + 1
1430 9144 : irow = irow + 1
1431 : END DO
1432 : END DO
1433 : END DO
1434 108 : ELSE IF (ASSOCIATED(dftb_parameter)) THEN
1435 108 : CALL get_dftb_atom_param(dftb_parameter, lmax=lmax)
1436 108 : icgf = 1
1437 270 : DO ishell = 1, lmax + 1
1438 162 : lshell = ishell - 1
1439 540 : DO ico = 1, nco(lshell)
1440 270 : symbol = cgf_symbol(1, indco(1:3, icgf))
1441 270 : symbol(1:2) = " "
1442 : WRITE (UNIT=iw, FMT=fmtstr3) &
1443 270 : "MO|", irow, iatom, ADJUSTR(element_symbol), symbol, &
1444 540 : (cmatrix(irow, jcol), jcol=from, to)
1445 270 : icgf = icgf + 1
1446 432 : irow = irow + 1
1447 : END DO
1448 : END DO
1449 : ELSE
1450 : ! assume atom without basis set
1451 : ! CPABORT("Unknown basis set type")
1452 : END IF
1453 :
1454 : ELSE
1455 :
1456 3859 : IF (ASSOCIATED(orb_basis_set)) THEN
1457 : CALL get_gto_basis_set(gto_basis_set=orb_basis_set, &
1458 : nset=nset, &
1459 : nshell=nshell, &
1460 : l=l, &
1461 3859 : sgf_symbol=bsgf_symbol)
1462 3859 : isgf = 1
1463 11027 : DO iset = 1, nset
1464 20499 : DO ishell = 1, nshell(iset)
1465 9472 : lshell = l(ishell, iset)
1466 39040 : DO iso = 1, nso(lshell)
1467 : WRITE (UNIT=iw, FMT=fmtstr3) &
1468 22400 : "MO|", irow, iatom, ADJUSTR(element_symbol), bsgf_symbol(isgf), &
1469 44800 : (smatrix(irow, jcol), jcol=from, to)
1470 22400 : isgf = isgf + 1
1471 31872 : irow = irow + 1
1472 : END DO
1473 : END DO
1474 : END DO
1475 0 : ELSE IF (ASSOCIATED(dftb_parameter)) THEN
1476 0 : CALL get_dftb_atom_param(dftb_parameter, lmax=lmax)
1477 0 : isgf = 1
1478 0 : DO ishell = 1, lmax + 1
1479 0 : lshell = ishell - 1
1480 0 : DO iso = 1, nso(lshell)
1481 0 : symbol = sgf_symbol(1, lshell, -lshell + iso - 1)
1482 0 : symbol(1:2) = " "
1483 : WRITE (UNIT=iw, FMT=fmtstr3) &
1484 0 : "MO|", irow, iatom, ADJUSTR(element_symbol), symbol, &
1485 0 : (smatrix(irow, jcol), jcol=from, to)
1486 0 : isgf = isgf + 1
1487 0 : irow = irow + 1
1488 : END DO
1489 : END DO
1490 : ELSE
1491 : ! assume atom without basis set
1492 : ! CPABORT("Unknown basis set type")
1493 : END IF
1494 :
1495 : END IF ! print_cartesian
1496 :
1497 : END DO ! iatom
1498 :
1499 : END DO ! icol
1500 :
1501 1753 : WRITE (UNIT=iw, FMT="(T2,A)") "MO|"
1502 :
1503 : ! Release work storage
1504 :
1505 1753 : IF (print_cartesian) THEN
1506 97 : DEALLOCATE (cmatrix)
1507 : END IF
1508 1753 : DEALLOCATE (smatrix)
1509 :
1510 991 : ELSE IF (print_occup .OR. print_eigvals) THEN
1511 :
1512 991 : WRITE (UNIT=iw, FMT="(T2,A)") "MO|"
1513 991 : fmtstr4 = "(T2,A,I7,3(1X,F22. ))"
1514 991 : WRITE (UNIT=fmtstr4(19:20), FMT="(I2)") after
1515 991 : IF (my_final .OR. (my_solver_method == "TD")) THEN
1516 : WRITE (UNIT=iw, FMT="(A)") &
1517 991 : " MO| Index Eigenvalue [a.u.] Eigenvalue [eV] Occupation"
1518 : ELSE
1519 : WRITE (UNIT=iw, FMT="(A)") &
1520 0 : " MO| Index Energy [a.u.] Energy [eV] Occupation"
1521 : END IF
1522 14534 : DO imo = first_mo, last_mo
1523 : WRITE (UNIT=iw, FMT=fmtstr4) &
1524 13543 : "MO|", imo, mo_eigenvalues(imo), &
1525 13543 : mo_eigenvalues(imo)*evolt, &
1526 28077 : mo_occupation_numbers(imo)
1527 : END DO
1528 991 : fmtstr5 = "(A,T59,F22. )"
1529 991 : WRITE (UNIT=fmtstr5(12:13), FMT="(I2)") after
1530 : WRITE (UNIT=iw, FMT=fmtstr5) &
1531 991 : " MO| Sum:", accurate_sum(mo_occupation_numbers(:))
1532 :
1533 : END IF ! print_eigvecs
1534 :
1535 2744 : IF (.NOT. my_rtp) THEN
1536 2740 : fmtstr6 = "(A,T18,F17. ,A,T41,F17. ,A)"
1537 2740 : WRITE (UNIT=fmtstr6(12:13), FMT="(I2)") after
1538 2740 : WRITE (UNIT=fmtstr6(25:26), FMT="(I2)") after
1539 : WRITE (UNIT=iw, FMT=fmtstr6) &
1540 2740 : " MO| E(Fermi):", mo_set%mu, " a.u.", mo_set%mu*evolt, " eV"
1541 : END IF
1542 2744 : IF ((homo > 0) .AND. .NOT. my_rtp) THEN
1543 2715 : IF ((mo_occupation_numbers(homo) == maxocc) .AND. (last_mo > homo)) THEN
1544 : gap = mo_eigenvalues(homo + 1) - &
1545 409 : mo_eigenvalues(homo)
1546 : WRITE (UNIT=iw, FMT=fmtstr6) &
1547 409 : " MO| Band gap:", gap, " a.u.", gap*evolt, " eV"
1548 : END IF
1549 : END IF
1550 2744 : WRITE (UNIT=iw, FMT="(A)") ""
1551 :
1552 : END IF ! iw
1553 :
1554 5488 : IF (ALLOCATED(mo_eigenvalues)) DEALLOCATE (mo_eigenvalues)
1555 5488 : IF (ALLOCATED(mo_occupation_numbers)) DEALLOCATE (mo_occupation_numbers)
1556 :
1557 : CALL cp_print_key_finished_output(iw, logger, dft_section, "PRINT%MO", &
1558 5488 : ignore_should_output=should_output)
1559 :
1560 24745 : END SUBROUTINE write_mo_set_to_output_unit
1561 :
1562 : END MODULE qs_mo_io
|