LCOV - code coverage report
Current view: top level - src/motion - rt_propagation.F (source / functions) Coverage Total Hit
Test: CP2K Regtests (git:7641cd9) Lines: 97.9 % 282 276
Test Date: 2026-05-25 07:16:39 Functions: 100.0 % 8 8

            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 the real time propagation.
      10              : !> \author Florian Schiffmann (02.09)
      11              : ! **************************************************************************************************
      12              : 
      13              : MODULE rt_propagation
      14              :    USE bibliography,                    ONLY: Andermatt2016,&
      15              :                                               cite_reference
      16              :    USE cell_types,                      ONLY: cell_type
      17              :    USE cp_control_types,                ONLY: dft_control_type,&
      18              :                                               rtp_control_type
      19              :    USE cp_dbcsr_api,                    ONLY: dbcsr_copy,&
      20              :                                               dbcsr_create,&
      21              :                                               dbcsr_p_type,&
      22              :                                               dbcsr_release,&
      23              :                                               dbcsr_set
      24              :    USE cp_external_control,             ONLY: external_control
      25              :    USE cp_fm_types,                     ONLY: cp_fm_set_all,&
      26              :                                               cp_fm_to_fm,&
      27              :                                               cp_fm_type
      28              :    USE cp_log_handling,                 ONLY: cp_get_default_logger,&
      29              :                                               cp_logger_get_default_io_unit,&
      30              :                                               cp_logger_get_default_unit_nr,&
      31              :                                               cp_logger_type,&
      32              :                                               cp_to_string
      33              :    USE cp_output_handling,              ONLY: cp_add_iter_level,&
      34              :                                               cp_iterate,&
      35              :                                               cp_p_file,&
      36              :                                               cp_print_key_generate_filename,&
      37              :                                               cp_print_key_should_output,&
      38              :                                               cp_print_key_unit_nr,&
      39              :                                               cp_rm_iter_level
      40              :    USE efield_utils,                    ONLY: calculate_ecore_efield
      41              :    USE force_env_methods,               ONLY: force_env_calc_energy_force
      42              :    USE force_env_types,                 ONLY: force_env_get,&
      43              :                                               force_env_type
      44              :    USE global_types,                    ONLY: global_environment_type
      45              :    USE hfx_admm_utils,                  ONLY: hfx_admm_init
      46              :    USE input_constants,                 ONLY: real_time_propagation,&
      47              :                                               use_restart_wfn,&
      48              :                                               use_rt_restart,&
      49              :                                               use_scf_wfn
      50              :    USE input_cp2k_restarts,             ONLY: write_restart
      51              :    USE input_section_types,             ONLY: section_vals_get,&
      52              :                                               section_vals_get_subs_vals,&
      53              :                                               section_vals_type,&
      54              :                                               section_vals_val_get,&
      55              :                                               section_vals_val_set
      56              :    USE kinds,                           ONLY: default_path_length,&
      57              :                                               dp
      58              :    USE machine,                         ONLY: m_walltime
      59              :    USE md_environment_types,            ONLY: md_environment_type
      60              :    USE moments_utils,                   ONLY: get_reference_point
      61              :    USE pw_env_types,                    ONLY: pw_env_type
      62              :    USE qs_core_hamiltonian,             ONLY: qs_matrix_h_allocate_imag_from_real
      63              :    USE qs_energy_init,                  ONLY: qs_energies_init
      64              :    USE qs_energy_types,                 ONLY: qs_energy_type
      65              :    USE qs_environment_types,            ONLY: get_qs_env,&
      66              :                                               qs_environment_type
      67              :    USE qs_external_potential,           ONLY: external_c_potential,&
      68              :                                               external_e_potential
      69              :    USE qs_ks_methods,                   ONLY: qs_ks_allocate_basics,&
      70              :                                               qs_ks_update_qs_env
      71              :    USE qs_ks_types,                     ONLY: qs_ks_did_change,&
      72              :                                               qs_ks_env_type,&
      73              :                                               set_ks_env
      74              :    USE qs_mo_io,                        ONLY: wfn_restart_file_name
      75              :    USE qs_mo_types,                     ONLY: get_mo_set,&
      76              :                                               init_mo_set,&
      77              :                                               mo_set_type
      78              :    USE qs_moments,                      ONLY: build_local_moment_matrix
      79              :    USE qs_rho_methods,                  ONLY: allocate_rho_ao_imag_from_real
      80              :    USE qs_rho_types,                    ONLY: qs_rho_set,&
      81              :                                               qs_rho_type
      82              :    USE rt_delta_pulse,                  ONLY: apply_delta_pulse
      83              :    USE rt_hfx_utils,                    ONLY: rtp_hfx_rebuild
      84              :    USE rt_projection_mo_utils,          ONLY: init_mo_projection
      85              :    USE rt_propagation_methods,          ONLY: propagation_step,&
      86              :                                               rtp_localize
      87              :    USE rt_propagation_output,           ONLY: calc_local_moment,&
      88              :                                               print_ft,&
      89              :                                               print_moments,&
      90              :                                               rt_prop_output
      91              :    USE rt_propagation_types,            ONLY: get_rtp,&
      92              :                                               rt_prop_create,&
      93              :                                               rt_prop_type,&
      94              :                                               rtp_create_SinvH_imag,&
      95              :                                               rtp_history_create
      96              :    USE rt_propagation_utils,            ONLY: calc_S_derivs,&
      97              :                                               calc_update_rho,&
      98              :                                               calc_update_rho_sparse,&
      99              :                                               get_restart_wfn,&
     100              :                                               read_moments,&
     101              :                                               recalculate_fields,&
     102              :                                               warn_section_unused
     103              :    USE rt_propagation_velocity_gauge,   ONLY: velocity_gauge_ks_matrix
     104              :    USE rt_propagator_init,              ONLY: init_propagators,&
     105              :                                               rt_initialize_rho_from_mos
     106              : #include "../base/base_uses.f90"
     107              : 
     108              :    IMPLICIT NONE
     109              : 
     110              :    PRIVATE
     111              : 
     112              :    CHARACTER(len=*), PARAMETER, PRIVATE :: moduleN = 'rt_propagation'
     113              : 
     114              :    PUBLIC :: rt_prop_setup
     115              : 
     116              : CONTAINS
     117              : 
     118              : ! **************************************************************************************************
     119              : !> \brief creates rtp_type, gets the initial state, either by reading MO's
     120              : !>        from file or calling SCF run
     121              : !> \param force_env ...
     122              : !> \author Florian Schiffmann (02.09)
     123              : ! **************************************************************************************************
     124              : 
     125          612 :    SUBROUTINE rt_prop_setup(force_env)
     126              :       TYPE(force_env_type), POINTER                      :: force_env
     127              : 
     128              :       INTEGER                                            :: aspc_order
     129              :       LOGICAL                                            :: magnetic, vel_reprs
     130              :       TYPE(dft_control_type), POINTER                    :: dft_control
     131              :       TYPE(global_environment_type), POINTER             :: globenv
     132              :       TYPE(qs_energy_type), POINTER                      :: energy
     133              :       TYPE(qs_environment_type), POINTER                 :: qs_env
     134              :       TYPE(rt_prop_type), POINTER                        :: rtp
     135              :       TYPE(rtp_control_type), POINTER                    :: rtp_control
     136              :       TYPE(section_vals_type), POINTER :: hfx_sections, input, ls_scf_section, md_section, &
     137              :          motion_section, print_moments_section, rtp_print_section, rtp_section
     138              : 
     139          204 :       NULLIFY (qs_env, rtp_control, dft_control)
     140              : 
     141          204 :       CALL cite_reference(Andermatt2016)
     142              : 
     143          204 :       CALL force_env_get(force_env=force_env, qs_env=qs_env, globenv=globenv)
     144          204 :       CALL get_qs_env(qs_env, dft_control=dft_control, energy=energy)
     145          204 :       rtp_control => dft_control%rtp_control
     146              : 
     147              :       ! Takes care that an initial wavefunction/density is available
     148              :       ! Can either be by performing an scf loop or reading a restart
     149          204 :       CALL rt_initial_guess(qs_env, force_env, rtp_control)
     150              : 
     151              :       ! Initializes the extrapolation
     152          204 :       NULLIFY (rtp)
     153          204 :       CALL get_qs_env(qs_env=qs_env, rtp=rtp, input=input)
     154          204 :       aspc_order = rtp_control%aspc_order
     155          204 :       CALL rtp_history_create(rtp, aspc_order)
     156              : 
     157              :       ! Reads the simulation parameters from the input
     158          204 :       motion_section => section_vals_get_subs_vals(force_env%root_section, "MOTION")
     159          204 :       md_section => section_vals_get_subs_vals(motion_section, "MD")
     160          204 :       hfx_sections => section_vals_get_subs_vals(force_env%root_section, "FORCE_EVAL%DFT%XC%HF")
     161          204 :       rtp_section => section_vals_get_subs_vals(force_env%root_section, "FORCE_EVAL%DFT%REAL_TIME_PROPAGATION")
     162          204 :       print_moments_section => section_vals_get_subs_vals(force_env%root_section, "FORCE_EVAL%DFT%PRINT%MOMENTS")
     163          204 :       CALL section_vals_val_get(md_section, "TIMESTEP", r_val=qs_env%rtp%dt)
     164          204 :       CALL section_vals_val_get(md_section, "STEP_START_VAL", i_val=qs_env%rtp%i_start)
     165          204 :       CALL section_vals_val_get(md_section, "STEPS", i_val=rtp%nsteps)
     166          204 :       CALL section_vals_val_get(md_section, "MAX_STEPS", i_val=rtp%max_steps)
     167              : 
     168          204 :       ls_scf_section => section_vals_get_subs_vals(input, "DFT%LS_SCF")
     169          204 :       CALL section_vals_val_get(ls_scf_section, "EPS_FILTER", r_val=rtp%filter_eps)
     170          204 :       IF (.NOT. qs_env%rtp%linear_scaling) rtp%filter_eps = 0.0_dp
     171          204 :       IF (rtp_control%acc_ref < 1) rtp_control%acc_ref = 1
     172          204 :       rtp%filter_eps_small = rtp%filter_eps/rtp_control%acc_ref
     173          204 :       CALL section_vals_val_get(ls_scf_section, "EPS_LANCZOS", r_val=rtp%lanzcos_threshold)
     174          204 :       CALL section_vals_val_get(ls_scf_section, "MAX_ITER_LANCZOS", i_val=rtp%lanzcos_max_iter)
     175          204 :       CALL section_vals_val_get(ls_scf_section, "SIGN_SQRT_ORDER", i_val=rtp%newton_schulz_order)
     176          204 :       CALL section_vals_get(hfx_sections, explicit=rtp%do_hfx)
     177          204 :       CALL section_vals_val_get(print_moments_section, "MAGNETIC", l_val=magnetic)
     178          204 :       CALL section_vals_val_get(print_moments_section, "VEL_REPRS", l_val=vel_reprs)
     179              : 
     180              :       rtp%track_imag_density = (magnetic) .OR. (vel_reprs) .OR. (rtp_control%velocity_gauge) &
     181          204 :                                .OR. (rtp%do_hfx) .OR. (.NOT. rtp_control%fixed_ions)
     182          204 :       rtp%propagate_complex_ks = rtp%do_hfx .OR. rtp_control%velocity_gauge
     183              : 
     184              :       ! Marek : In case some print sections that apply so far only to RTBSE are present,
     185              :       !         warn the user that the quantities will not be in fact printed out
     186          204 :       rtp_print_section => section_vals_get_subs_vals(rtp_section, "PRINT")
     187              :       CALL warn_section_unused(rtp_print_section, "DENSITY_MATRIX", &
     188          204 :                                "DENSITY_MATRIX printing not implemented for non-RTBSE code.")
     189              : 
     190              :       CALL rt_init_complex_quantities(qs_env, imag_p=rtp%track_imag_density, &
     191          204 :                                       imag_ks=rtp%propagate_complex_ks, imag_h=rtp_control%velocity_gauge)
     192              : 
     193          204 :       IF (rtp_control%save_local_moments) CALL rt_init_local_moments(rtp, qs_env)
     194              : 
     195              :       ! Hmm, not really like to initialize with the structure of S but I reckon it is
     196              :       ! done everywhere like this
     197          204 :       IF (rtp%do_hfx) CALL rtp_hfx_rebuild(qs_env)
     198              : 
     199              :       ! Setup the MO projection environment if required
     200          204 :       IF (rtp_control%is_proj_mo) CALL init_mo_projection(qs_env, rtp_control)
     201              : 
     202          204 :       CALL init_propagation_run(qs_env)
     203          204 :       IF (.NOT. rtp_control%fixed_ions) THEN
     204              :          !derivativs of the overlap needed for EMD
     205           74 :          CALL calc_S_derivs(qs_env)
     206              :          ! a bit hidden, but computes SinvH and SinvB (calc_SinvH for CN,EM and ARNOLDI)
     207              :          ! make_etrs_exp in case of ETRS in combination with TAYLOR and PADE
     208              :       END IF
     209          204 :       CALL init_propagators(qs_env)
     210          204 :       IF (rtp_control%fixed_ions) THEN
     211          130 :          CALL run_propagation(qs_env, force_env, globenv)
     212              :       ELSE
     213           74 :          rtp_control%initial_step = .TRUE.
     214           74 :          CALL force_env_calc_energy_force(force_env, calc_force=.TRUE.)
     215           74 :          rtp_control%initial_step = .FALSE.
     216           74 :          rtp%energy_old = energy%total
     217              :       END IF
     218              : 
     219          204 :       IF (rtp_control%save_local_moments) THEN
     220              :          ! Call routines for outputs and deallocations of FT observables
     221           18 :          CALL final_ft_output(qs_env)
     222              :       END IF
     223              : 
     224          204 :       IF (ASSOCIATED(rtp_control%print_pol_elements)) DEALLOCATE (rtp_control%print_pol_elements)
     225              : 
     226          204 :    END SUBROUTINE rt_prop_setup
     227              : 
     228              : ! **************************************************************************************************
     229              : !> \brief calculates the matrices needed in the first step of EMD/RTP
     230              : !> \param qs_env ...
     231              : !> \author Florian Schiffmann (02.09)
     232              : ! **************************************************************************************************
     233              : 
     234          204 :    SUBROUTINE init_propagation_run(qs_env)
     235              :       TYPE(qs_environment_type), POINTER                 :: qs_env
     236              : 
     237              :       REAL(KIND=dp), PARAMETER                           :: zero = 0.0_dp
     238              : 
     239              :       INTEGER                                            :: i, ispin, re
     240              :       INTEGER, DIMENSION(2)                              :: nelectron_spin
     241          204 :       TYPE(cp_fm_type), DIMENSION(:), POINTER            :: mos_new, mos_old
     242          204 :       TYPE(dbcsr_p_type), DIMENSION(:), POINTER          :: matrix_ks, rho_new, rho_old
     243              :       TYPE(dft_control_type), POINTER                    :: dft_control
     244          204 :       TYPE(mo_set_type), DIMENSION(:), POINTER           :: mos
     245              :       TYPE(rt_prop_type), POINTER                        :: rtp
     246              :       TYPE(rtp_control_type), POINTER                    :: rtp_control
     247              : 
     248          204 :       NULLIFY (dft_control, rtp, rtp_control)
     249              : 
     250          204 :       CALL cite_reference(Andermatt2016)
     251              : 
     252              :       CALL get_qs_env(qs_env, &
     253              :                       rtp=rtp, &
     254          204 :                       dft_control=dft_control)
     255          204 :       rtp_control => dft_control%rtp_control
     256              : 
     257          204 :       IF (rtp_control%initial_wfn == use_scf_wfn) THEN
     258          168 :          IF (rtp_control%apply_delta_pulse .OR. rtp_control%apply_delta_pulse_mag) THEN
     259           58 :             CALL apply_delta_pulse(qs_env, rtp, rtp_control)
     260              :          ELSE
     261          110 :             IF (.NOT. rtp%linear_scaling) THEN
     262           76 :                CALL get_rtp(rtp=rtp, mos_old=mos_old)
     263           76 :                CALL get_qs_env(qs_env, mos=mos)
     264          168 :                DO i = 1, SIZE(mos)
     265           92 :                   CALL cp_fm_to_fm(mos(i)%mo_coeff, mos_old(2*i - 1))
     266          168 :                   CALL cp_fm_set_all(mos_old(2*i), zero, zero)
     267              :                END DO
     268              :             END IF
     269              :          END IF
     270              :       END IF
     271              : 
     272          204 :       IF (.NOT. rtp%linear_scaling) THEN
     273          114 :          CALL get_rtp(rtp=rtp, mos_old=mos_old, mos_new=mos_new)
     274          406 :          DO i = 1, SIZE(mos_old)
     275          406 :             CALL cp_fm_to_fm(mos_old(i), mos_new(i))
     276              :          END DO
     277          114 :          CALL calc_update_rho(qs_env)
     278              :       ELSE
     279           90 :          IF (rtp_control%initial_wfn == use_scf_wfn) THEN
     280              :             CALL get_qs_env(qs_env, &
     281              :                             matrix_ks=matrix_ks, &
     282              :                             mos=mos, &
     283           74 :                             nelectron_spin=nelectron_spin)
     284           74 :             IF (ASSOCIATED(mos)) THEN
     285              :                !The wavefunction was minimized by an mo based algorith. P is therefore calculated from the mos
     286           64 :                IF (ASSOCIATED(rtp%mos)) THEN
     287           40 :                   IF (ASSOCIATED(rtp%mos%old)) THEN
     288              :                      ! Delta kick was applied and the results is in rtp%mos%old
     289           40 :                      CALL rt_initialize_rho_from_mos(rtp, mos, mos_old=rtp%mos%old)
     290              :                   ELSE
     291            0 :                      CALL rt_initialize_rho_from_mos(rtp, mos)
     292              :                   END IF
     293              :                ELSE
     294           24 :                   CALL rt_initialize_rho_from_mos(rtp, mos)
     295              :                END IF
     296              :             ELSE
     297              :                !The wavefunction was minimized using a linear scaling method. The density matrix is therefore taken from the ls_scf_env.
     298           10 :                CALL get_rtp(rtp=rtp, rho_old=rho_old, rho_new=rho_new)
     299           24 :                DO ispin = 1, SIZE(rho_old)/2
     300           14 :                   re = 2*ispin - 1
     301           14 :                   CALL dbcsr_copy(rho_old(re)%matrix, qs_env%ls_scf_env%matrix_p(ispin))
     302           24 :                   CALL dbcsr_copy(rho_new(re)%matrix, qs_env%ls_scf_env%matrix_p(ispin))
     303              :                END DO
     304              :             END IF
     305           74 :             CALL calc_update_rho_sparse(qs_env)
     306              :          END IF
     307              :       END IF
     308              :       ! Modify KS matrix to include the additional terms in the velocity gauge
     309          204 :       IF (rtp_control%velocity_gauge) THEN
     310              :          ! As matrix_h and matrix_h_im are not updated by qs_ks_update_qs_env()
     311              :          ! the non-gauge transformed non-local part has to be subtracted here
     312            8 :          CALL velocity_gauge_ks_matrix(qs_env, subtract_nl_term=.TRUE.)
     313              :       END IF
     314          204 :       CALL qs_ks_update_qs_env(qs_env, calculate_forces=.FALSE.)
     315              : 
     316          204 :    END SUBROUTINE init_propagation_run
     317              : 
     318              : ! **************************************************************************************************
     319              : !> \brief performs the real RTP run, gets information from MD section
     320              : !>        uses MD as iteration level
     321              : !> \param qs_env ...
     322              : !> \param force_env ...
     323              : !> \param globenv ...
     324              : !> \author Florian Schiffmann (02.09)
     325              : ! **************************************************************************************************
     326              : 
     327          130 :    SUBROUTINE run_propagation(qs_env, force_env, globenv)
     328              :       TYPE(qs_environment_type), POINTER                 :: qs_env
     329              :       TYPE(force_env_type), POINTER                      :: force_env
     330              :       TYPE(global_environment_type), POINTER             :: globenv
     331              : 
     332              :       CHARACTER(len=*), PARAMETER                        :: routineN = 'run_propagation'
     333              : 
     334              :       INTEGER                                            :: aspc_order, handle, i_iter, i_step, &
     335              :                                                             max_iter, max_steps, output_unit, &
     336              :                                                             unit_nr
     337              :       LOGICAL                                            :: moments_read, should_stop
     338              :       REAL(Kind=dp)                                      :: eps_ener, time_iter_start, &
     339              :                                                             time_iter_stop, used_time
     340              :       TYPE(cp_logger_type), POINTER                      :: logger
     341          130 :       TYPE(dbcsr_p_type), DIMENSION(:), POINTER          :: rho_new
     342              :       TYPE(dft_control_type), POINTER                    :: dft_control
     343              :       TYPE(pw_env_type), POINTER                         :: pw_env
     344              :       TYPE(qs_energy_type), POINTER                      :: energy
     345              :       TYPE(rt_prop_type), POINTER                        :: rtp
     346              :       TYPE(rtp_control_type), POINTER                    :: rtp_control
     347              :       TYPE(section_vals_type), POINTER                   :: input, moments_section, rtp_section
     348              : 
     349          130 :       should_stop = .FALSE.
     350          130 :       CALL timeset(routineN, handle)
     351              : 
     352          130 :       CALL cite_reference(Andermatt2016)
     353              : 
     354          130 :       NULLIFY (logger, dft_control, energy, rtp, rtp_control, input, rtp_section)
     355          130 :       logger => cp_get_default_logger()
     356          130 :       IF (logger%para_env%is_source()) THEN
     357           65 :          unit_nr = cp_logger_get_default_unit_nr(logger, local=.TRUE.)
     358              :       ELSE
     359              :          unit_nr = -1
     360              :       END IF
     361              : 
     362          130 :       CALL get_qs_env(qs_env=qs_env, dft_control=dft_control, rtp=rtp, energy=energy, input=input)
     363              : 
     364          130 :       rtp_control => dft_control%rtp_control
     365          130 :       max_steps = MIN(rtp%nsteps, rtp%max_steps)
     366          130 :       max_iter = rtp_control%max_iter
     367          130 :       eps_ener = rtp_control%eps_ener
     368              : 
     369          130 :       aspc_order = rtp_control%aspc_order
     370              : 
     371          130 :       rtp%energy_old = energy%total
     372          130 :       time_iter_start = m_walltime()
     373          130 :       CALL cp_add_iter_level(logger%iter_info, "MD")
     374          130 :       CALL cp_iterate(logger%iter_info, iter_nr=0)
     375          130 :       IF (rtp%i_start >= max_steps) CALL cp_abort(__LOCATION__, &
     376            0 :                                                   "maximum step number smaller than initial step value")
     377              : 
     378          130 :       rtp_section => section_vals_get_subs_vals(input, "DFT%REAL_TIME_PROPAGATION")
     379              :       output_unit = cp_print_key_unit_nr(logger, rtp_section, "PRINT%PROGRAM_RUN_INFO", &
     380          130 :                                          extension=".scfLog")
     381              :       ! Add the zero iteration moments to the moment trace
     382          130 :       IF (rtp_control%save_local_moments) THEN
     383           18 :          CALL get_rtp(rtp, rho_new=rho_new)
     384           18 :          moments_section => section_vals_get_subs_vals(rtp_section, "PRINT%MOMENTS")
     385              :          ! TODO : Conditions on when not to read the files
     386           18 :          CALL read_moments(moments_section, 0, rtp%i_start, rtp%moments, rtp%times, moments_read)
     387              :          ! Recalculate the field at times in the trace/Read the field from the files
     388           18 :          CALL recalculate_fields(rtp%fields, rtp%times, 0, rtp%i_start, dft_control)
     389           18 :          IF (.NOT. moments_read) THEN
     390           18 :             CALL calc_local_moment(rtp%local_moments, rho_new, rtp%local_moments_work, rtp%moments(:, :, 1))
     391           18 :             qs_env%sim_time = REAL(rtp%i_start, dp)*rtp%dt
     392           18 :             rtp%times(1) = qs_env%sim_time
     393              :             CALL print_moments(moments_section, output_unit, rtp%moments(:, :, 1), &
     394           18 :                                qs_env%sim_time, rtp%track_imag_density)
     395              :          END IF
     396              :       END IF
     397              : 
     398          482 :       DO i_step = rtp%i_start + 1, max_steps
     399          352 :          IF (output_unit > 0) THEN
     400              :             WRITE (output_unit, FMT="(/,(T2,A,T40,I6))") &
     401          176 :                "Real time propagation step:", i_step
     402              :          END IF
     403          352 :          energy%efield_core = 0.0_dp
     404          352 :          qs_env%sim_time = REAL(i_step, dp)*rtp%dt
     405          352 :          CALL get_qs_env(qs_env, pw_env=pw_env)
     406          352 :          pw_env%poisson_env%parameters%dbc_params%time = qs_env%sim_time
     407          352 :          qs_env%sim_step = i_step
     408          352 :          rtp%istep = i_step - rtp%i_start
     409          352 :          CALL calculate_ecore_efield(qs_env, .FALSE.)
     410          352 :          IF (dft_control%apply_external_potential) THEN
     411            0 :             IF (.NOT. dft_control%expot_control%static) THEN
     412            0 :                dft_control%eval_external_potential = .TRUE.
     413              :             END IF
     414              :          END IF
     415          352 :          CALL external_c_potential(qs_env, calculate_forces=.FALSE.)
     416          352 :          CALL external_e_potential(qs_env)
     417          352 :          CALL cp_iterate(logger%iter_info, last=(i_step == max_steps), iter_nr=i_step)
     418          352 :          rtp%converged = .FALSE.
     419         1172 :          DO i_iter = 1, max_iter
     420         1172 :             IF (i_step == rtp%i_start + 1 .AND. i_iter == 2 .AND. rtp_control%hfx_redistribute) &
     421            0 :                CALL qs_ks_did_change(qs_env%ks_env, s_mstruct_changed=.TRUE.)
     422         1172 :             rtp%iter = i_iter
     423         1172 :             CALL propagation_step(qs_env, rtp, rtp_control)
     424         1172 :             CALL qs_ks_update_qs_env(qs_env, calculate_forces=.FALSE.)
     425         1172 :             rtp%energy_new = energy%total
     426         1172 :             IF (rtp%converged) EXIT
     427         1172 :             CALL rt_prop_output(qs_env, real_time_propagation, rtp%delta_iter)
     428              :          END DO
     429          482 :          IF (rtp%converged) THEN
     430          352 :             CALL external_control(should_stop, "MD", globenv=globenv)
     431          352 :             IF (should_stop) CALL cp_iterate(logger%iter_info, last=.TRUE., iter_nr=i_step)
     432          352 :             time_iter_stop = m_walltime()
     433          352 :             used_time = time_iter_stop - time_iter_start
     434          352 :             time_iter_start = time_iter_stop
     435          352 :             CALL rt_prop_output(qs_env, real_time_propagation, delta_iter=rtp%delta_iter, used_time=used_time)
     436          352 :             CALL rt_write_input_restart(force_env=force_env, qs_env=qs_env)
     437          352 :             IF (MODULO(i_step, dft_control%localize_each) == 0) THEN
     438          352 :                CALL rtp_localize(qs_env, rtp)
     439              :             END IF
     440          352 :             IF (should_stop) EXIT
     441              :          ELSE
     442              :             EXIT
     443              :          END IF
     444              :       END DO
     445          130 :       CALL cp_rm_iter_level(logger%iter_info, "MD")
     446              : 
     447          130 :       IF (.NOT. rtp%converged) &
     448              :          CALL cp_abort(__LOCATION__, "propagation did not converge, "// &
     449            0 :                        "either increase MAX_ITER or use a smaller TIMESTEP")
     450              : 
     451          130 :       CALL timestop(handle)
     452              : 
     453          130 :    END SUBROUTINE run_propagation
     454              : 
     455              : ! **************************************************************************************************
     456              : !> \brief overwrites some values in the input file such that the .restart
     457              : !>        file will contain the appropriate information
     458              : !> \param md_env ...
     459              : !> \param qs_env ...
     460              : !> \param force_env ...
     461              : !> \author Florian Schiffmann (02.09)
     462              : ! **************************************************************************************************
     463              : 
     464          352 :    SUBROUTINE rt_write_input_restart(md_env, qs_env, force_env)
     465              :       TYPE(md_environment_type), OPTIONAL, POINTER       :: md_env
     466              :       TYPE(qs_environment_type), OPTIONAL, POINTER       :: qs_env
     467              :       TYPE(force_env_type), POINTER                      :: force_env
     468              : 
     469              :       CHARACTER(len=default_path_length)                 :: file_name
     470          352 :       REAL(KIND=dp), DIMENSION(:), POINTER               :: tmp_vals
     471              :       TYPE(cp_logger_type), POINTER                      :: logger
     472              :       TYPE(dft_control_type), POINTER                    :: dft_control
     473              :       TYPE(section_vals_type), POINTER                   :: dft_section, efield_section, &
     474              :                                                             motion_section, print_key, &
     475              :                                                             root_section, rt_section
     476              : 
     477          352 :       CALL get_qs_env(qs_env=qs_env, dft_control=dft_control)
     478          352 :       root_section => force_env%root_section
     479          352 :       motion_section => section_vals_get_subs_vals(root_section, "MOTION")
     480          352 :       dft_section => section_vals_get_subs_vals(root_section, "FORCE_EVAL%DFT")
     481          352 :       rt_section => section_vals_get_subs_vals(root_section, "FORCE_EVAL%DFT%REAL_TIME_PROPAGATION")
     482              : 
     483          352 :       CALL section_vals_val_set(rt_section, "INITIAL_WFN", i_val=use_rt_restart)
     484          352 :       CALL section_vals_val_set(rt_section, "APPLY_DELTA_PULSE", l_val=.FALSE.)
     485          352 :       CALL section_vals_val_set(rt_section, "APPLY_DELTA_PULSE_MAG", l_val=.FALSE.)
     486          352 :       CALL section_vals_val_set(rt_section, "APPLY_WFN_MIX_INIT_RESTART", l_val=.FALSE.)
     487              : 
     488          352 :       logger => cp_get_default_logger()
     489              : 
     490              :       ! to continue propagating the TD wavefunction we need to read from the new .rtpwfn
     491          352 :       IF (BTEST(cp_print_key_should_output(logger%iter_info, &
     492              :                                            rt_section, "PRINT%RESTART"), cp_p_file)) THEN
     493          130 :          print_key => section_vals_get_subs_vals(rt_section, "PRINT%RESTART")
     494              :          file_name = cp_print_key_generate_filename(logger, print_key, &
     495          130 :                                                     extension=".rtpwfn", my_local=.FALSE.)
     496          130 :          CALL section_vals_val_set(dft_section, "WFN_RESTART_FILE_NAME", c_val=TRIM(file_name))
     497              :       END IF
     498              : 
     499              :       ! coming from RTP
     500          352 :       IF (.NOT. PRESENT(md_env)) THEN
     501          352 :          CALL section_vals_val_set(motion_section, "MD%STEP_START_VAL", i_val=force_env%qs_env%sim_step)
     502              :       END IF
     503              : 
     504          352 :       IF (dft_control%apply_vector_potential) THEN
     505           22 :          efield_section => section_vals_get_subs_vals(root_section, "FORCE_EVAL%DFT%EFIELD")
     506              :          NULLIFY (tmp_vals)
     507           22 :          ALLOCATE (tmp_vals(3))
     508           88 :          tmp_vals = dft_control%efield_fields(1)%efield%vec_pot_initial
     509              :          CALL section_vals_val_set(efield_section, "VEC_POT_INITIAL", &
     510              :                                    r_vals_ptr=tmp_vals, &
     511           22 :                                    i_rep_section=1)
     512              :       END IF
     513              : 
     514          352 :       CALL write_restart(md_env=md_env, root_section=root_section)
     515              : 
     516          352 :    END SUBROUTINE rt_write_input_restart
     517              : 
     518              : ! **************************************************************************************************
     519              : !> \brief Creates the initial electronic states and allocates the necessary
     520              : !>        matrices
     521              : !> \param qs_env ...
     522              : !> \param force_env ...
     523              : !> \param rtp_control ...
     524              : !> \author Florian Schiffmann (02.09)
     525              : ! **************************************************************************************************
     526              : 
     527          204 :    SUBROUTINE rt_initial_guess(qs_env, force_env, rtp_control)
     528              :       TYPE(qs_environment_type), POINTER                 :: qs_env
     529              :       TYPE(force_env_type), POINTER                      :: force_env
     530              :       TYPE(rtp_control_type), POINTER                    :: rtp_control
     531              : 
     532              :       INTEGER                                            :: homo, ispin
     533              :       LOGICAL                                            :: energy_consistency
     534              :       TYPE(cp_fm_type), POINTER                          :: mo_coeff
     535          204 :       TYPE(dbcsr_p_type), DIMENSION(:), POINTER          :: matrix_s
     536              :       TYPE(dft_control_type), POINTER                    :: dft_control
     537              : 
     538          204 :       NULLIFY (matrix_s, dft_control)
     539          204 :       CALL get_qs_env(qs_env, dft_control=dft_control)
     540          204 :       CPASSERT(ASSOCIATED(qs_env))
     541              : 
     542          372 :       SELECT CASE (rtp_control%initial_wfn)
     543              :       CASE (use_scf_wfn)
     544          168 :          qs_env%sim_time = 0.0_dp
     545          168 :          qs_env%sim_step = 0
     546          168 :          energy_consistency = .TRUE.
     547              :          !in the linear scaling case we need a correct kohn-sham matrix, which we cannot get with consistent energies
     548          168 :          IF (rtp_control%linear_scaling) energy_consistency = .FALSE.
     549              :          CALL force_env_calc_energy_force(force_env, calc_force=.FALSE., &
     550          168 :                                           consistent_energies=energy_consistency)
     551          168 :          qs_env%run_rtp = .TRUE.
     552          168 :          ALLOCATE (qs_env%rtp)
     553          168 :          CALL get_qs_env(qs_env, matrix_s=matrix_s)
     554          168 :          IF (dft_control%do_admm) THEN
     555            8 :             CALL hfx_admm_init(qs_env)
     556              :             CALL rt_prop_create(qs_env%rtp, qs_env%mos, qs_env%mpools, dft_control, matrix_s(1)%matrix, &
     557            8 :                                 rtp_control%linear_scaling, qs_env%admm_env%mos_aux_fit)
     558              :          ELSE
     559              :             CALL rt_prop_create(qs_env%rtp, qs_env%mos, qs_env%mpools, dft_control, matrix_s(1)%matrix, &
     560          160 :                                 rtp_control%linear_scaling)
     561              :          END IF
     562              : 
     563              :       CASE (use_restart_wfn, use_rt_restart)
     564           36 :          CALL qs_energies_init(qs_env, .FALSE.)
     565           36 :          IF (.NOT. rtp_control%linear_scaling .OR. rtp_control%initial_wfn == use_restart_wfn) THEN
     566           86 :             DO ispin = 1, SIZE(qs_env%mos)
     567           52 :                CALL get_mo_set(qs_env%mos(ispin), mo_coeff=mo_coeff, homo=homo)
     568           86 :                IF (.NOT. ASSOCIATED(mo_coeff)) THEN
     569              :                   CALL init_mo_set(qs_env%mos(ispin), &
     570              :                                    qs_env%mpools%ao_mo_fm_pools(ispin)%pool, &
     571           52 :                                    name="qs_env%mo"//TRIM(ADJUSTL(cp_to_string(ispin))))
     572              :                END IF
     573              :             END DO
     574           34 :             IF (dft_control%do_admm) CALL hfx_admm_init(qs_env)
     575              :          END IF
     576           36 :          ALLOCATE (qs_env%rtp)
     577           36 :          CALL get_qs_env(qs_env, matrix_s=matrix_s)
     578              :          CALL rt_prop_create(qs_env%rtp, qs_env%mos, qs_env%mpools, dft_control, matrix_s(1)%matrix, &
     579           36 :                              rtp_control%linear_scaling, qs_env%admm_env%mos_aux_fit)
     580           36 :          CALL get_restart_wfn(qs_env)
     581           36 :          CPASSERT(ASSOCIATED(qs_env))
     582              : 
     583          240 :          qs_env%run_rtp = .TRUE.
     584              :       END SELECT
     585              : 
     586          204 :    END SUBROUTINE rt_initial_guess
     587              : 
     588              : ! **************************************************************************************************
     589              : !> \brief ...
     590              : !> \param qs_env ...
     591              : !> \param imag_p ...
     592              : !> \param imag_ks ...
     593              : !> \param imag_h ...
     594              : ! **************************************************************************************************
     595          204 :    SUBROUTINE rt_init_complex_quantities(qs_env, imag_p, imag_ks, imag_h)
     596              :       TYPE(qs_environment_type), POINTER                 :: qs_env
     597              :       LOGICAL, INTENT(in)                                :: imag_p, imag_ks, imag_h
     598              : 
     599              :       TYPE(dft_control_type), POINTER                    :: dft_control
     600              :       TYPE(qs_ks_env_type), POINTER                      :: ks_env
     601              :       TYPE(qs_rho_type), POINTER                         :: rho
     602              :       TYPE(rt_prop_type), POINTER                        :: rtp
     603              : 
     604          204 :       NULLIFY (ks_env, rho, dft_control)
     605              : 
     606              :       CALL get_qs_env(qs_env, &
     607              :                       dft_control=dft_control, &
     608              :                       ks_env=ks_env, &
     609              :                       rho=rho, &
     610          204 :                       rtp=rtp)
     611              : 
     612              :       ! rho
     613          204 :       CALL qs_rho_set(rho, complex_rho_ao=imag_p)
     614          204 :       IF (imag_p) CALL allocate_rho_ao_imag_from_real(rho, qs_env)
     615              : 
     616              :       ! ks
     617          204 :       CALL set_ks_env(ks_env, complex_ks=imag_ks)
     618          204 :       IF (imag_ks) THEN
     619           42 :          CALL qs_ks_allocate_basics(qs_env, is_complex=imag_ks)
     620           42 :          IF (.NOT. dft_control%rtp_control%fixed_ions) &
     621           24 :             CALL rtp_create_SinvH_imag(rtp, dft_control%nspins)
     622              :       END IF
     623              : 
     624              :       ! h
     625          204 :       IF (imag_h) CALL qs_matrix_h_allocate_imag_from_real(qs_env)
     626              : 
     627          204 :    END SUBROUTINE rt_init_complex_quantities
     628              : 
     629              : ! **************************************************************************************************
     630              : !> \brief Allocates and fills the local moment matrices (only available for linear scaling)
     631              : !> \param rtp Real time propagtion properties - local moment matrices are stored there
     632              : !> \param qs_env QS environment necessary for moment matrix calculation
     633              : ! **************************************************************************************************
     634           18 :    SUBROUTINE rt_init_local_moments(rtp, qs_env)
     635              :       TYPE(rt_prop_type), POINTER                        :: rtp
     636              :       TYPE(qs_environment_type), POINTER                 :: qs_env
     637              : 
     638              :       INTEGER                                            :: k, nspin, output_unit
     639              :       REAL(kind=dp), DIMENSION(3)                        :: reference_point
     640              :       TYPE(cp_logger_type), POINTER                      :: logger
     641           18 :       TYPE(dbcsr_p_type), DIMENSION(:), POINTER          :: matrix_s, rho_old
     642              :       TYPE(dft_control_type), POINTER                    :: dft_control
     643              :       TYPE(rtp_control_type), POINTER                    :: rtc
     644              :       TYPE(section_vals_type), POINTER                   :: input, moments_section
     645              : 
     646           36 :       logger => cp_get_default_logger()
     647           18 :       output_unit = cp_logger_get_default_io_unit(logger)
     648              : 
     649           18 :       CALL get_qs_env(qs_env, dft_control=dft_control, matrix_s=matrix_s, input=input)
     650           18 :       rtc => dft_control%rtp_control
     651              :       moments_section => section_vals_get_subs_vals(input, &
     652           18 :                                                     "DFT%REAL_TIME_PROPAGATION%PRINT%MOMENTS")
     653              : 
     654              :       ! Construct the local moments matrix - copy from matrix_s structure
     655              :       ! NOTE : construction where blocks are allocated by neighbour lists does not seem to work,
     656              :       ! so doing a copy instead of:
     657              :       !    CALL dbcsr_create(rtp%local_moments(k)%matrix, template=matrix_s(1)%matrix, &
     658              :       !       name="Local moment")
     659              :       !    CALL cp_dbcsr_alloc_block_from_nbl(rtp%local_moments(k)%matrix, sab_all)
     660           18 :       NULLIFY (rtp%local_moments)
     661           72 :       ALLOCATE (rtp%local_moments(3))
     662           72 :       DO k = 1, 3
     663           54 :          NULLIFY (rtp%local_moments(k)%matrix)
     664           54 :          ALLOCATE (rtp%local_moments(k)%matrix)
     665           54 :          CALL dbcsr_create(rtp%local_moments(k)%matrix, template=matrix_s(1)%matrix, name="Local moment")
     666           54 :          CALL dbcsr_copy(rtp%local_moments(k)%matrix, matrix_s(1)%matrix)
     667           72 :          CALL dbcsr_set(rtp%local_moments(k)%matrix, 0.0_dp)
     668              :       END DO
     669              :       ! Workspace allocation
     670           18 :       NULLIFY (rtp%local_moments_work)
     671           18 :       ALLOCATE (rtp%local_moments_work)
     672           18 :       CALL dbcsr_create(rtp%local_moments_work, template=rtp%local_moments(1)%matrix, name="tmp")
     673           18 :       CALL dbcsr_copy(rtp%local_moments_work, rtp%local_moments(1)%matrix)
     674              : 
     675              :       CALL get_reference_point(rpoint=reference_point, qs_env=qs_env, &
     676           18 :                                reference=rtc%moment_trace_ref_type, ref_point=rtc%moment_trace_user_ref_point)
     677              : 
     678           18 :       CALL build_local_moment_matrix(qs_env, rtp%local_moments, 1, reference_point)
     679              : 
     680              :       ! Allocate the moments trace and output start moments
     681           18 :       CALL get_rtp(rtp, rho_old=rho_old)
     682           18 :       nspin = SIZE(rho_old)/2
     683          576 :       ALLOCATE (rtp%moments(SIZE(rho_old)/2, 3, rtp%nsteps + 1), source=CMPLX(0.0, 0.0, kind=dp))
     684           18 :       NULLIFY (rtp%times)
     685           54 :       ALLOCATE (rtp%times(rtp%nsteps + 1))
     686           18 :       NULLIFY (rtp%fields)
     687          270 :       ALLOCATE (rtp%fields(3, rtp%nsteps + 1), source=CMPLX(0.0, 0.0, kind=dp))
     688              : 
     689           18 :    END SUBROUTINE rt_init_local_moments
     690              : 
     691              : ! **************************************************************************************************
     692              : !> \brief Allocates and fills the local moment matrices (only available for linear scaling)
     693              : !> \param qs_env QS environment necessary for moment matrix calculation
     694              : ! **************************************************************************************************
     695           18 :    SUBROUTINE final_ft_output(qs_env)
     696              :       TYPE(qs_environment_type), POINTER                 :: qs_env
     697              : 
     698              :       INTEGER                                            :: k, unit_nr
     699              :       TYPE(cell_type), POINTER                           :: cell
     700              :       TYPE(cp_logger_type), POINTER                      :: logger
     701              :       TYPE(dft_control_type), POINTER                    :: dft_control
     702              :       TYPE(rt_prop_type), POINTER                        :: rtp
     703              :       TYPE(section_vals_type), POINTER                   :: input, rtp_section
     704              : 
     705           18 :       CALL get_qs_env(qs_env, cell=cell, rtp=rtp, input=input, dft_control=dft_control)
     706           18 :       rtp_section => section_vals_get_subs_vals(input, "DFT%REAL_TIME_PROPAGATION")
     707           18 :       logger => cp_get_default_logger()
     708           18 :       unit_nr = cp_logger_get_default_io_unit(logger)
     709              :       CALL print_ft(rtp_section, rtp%moments, rtp%times, rtp%fields, dft_control%rtp_control, &
     710           18 :                     info_opt=unit_nr, cell=cell)
     711              :       ! Deallocating the local moments matrices and array
     712           72 :       DO k = 1, 3
     713           54 :          CALL dbcsr_release(rtp%local_moments(k)%matrix)
     714           72 :          DEALLOCATE (rtp%local_moments(k)%matrix)
     715              :       END DO
     716           18 :       DEALLOCATE (rtp%local_moments)
     717           18 :       CALL dbcsr_release(rtp%local_moments_work)
     718           18 :       DEALLOCATE (rtp%local_moments_work)
     719           18 :       DEALLOCATE (rtp%moments)
     720           18 :       DEALLOCATE (rtp%times)
     721           18 :       DEALLOCATE (rtp%fields)
     722           18 :    END SUBROUTINE final_ft_output
     723              : 
     724              : END MODULE rt_propagation
        

Generated by: LCOV version 2.0-1