LCOV - code coverage report
Current view: top level - src - qs_force.F (source / functions) Coverage Total Hit
Test: CP2K Regtests (git:06f838d) Lines: 89.6 % 326 292
Test Date: 2026-06-05 07:04:50 Functions: 100.0 % 3 3

            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 Quickstep force driver routine
      10              : !> \author MK (12.06.2002)
      11              : ! **************************************************************************************************
      12              : MODULE qs_force
      13              :    USE atomic_kind_types,               ONLY: atomic_kind_type,&
      14              :                                               get_atomic_kind_set
      15              :    USE cell_types,                      ONLY: cell_type
      16              :    USE cp_control_types,                ONLY: dft_control_type
      17              :    USE cp_dbcsr_api,                    ONLY: dbcsr_copy,&
      18              :                                               dbcsr_p_type,&
      19              :                                               dbcsr_set
      20              :    USE cp_dbcsr_operations,             ONLY: dbcsr_allocate_matrix_set,&
      21              :                                               dbcsr_deallocate_matrix_set
      22              :    USE cp_dbcsr_output,                 ONLY: cp_dbcsr_write_sparse_matrix
      23              :    USE cp_log_handling,                 ONLY: cp_get_default_logger,&
      24              :                                               cp_logger_get_default_io_unit,&
      25              :                                               cp_logger_type
      26              :    USE cp_output_handling,              ONLY: cp_p_file,&
      27              :                                               cp_print_key_finished_output,&
      28              :                                               cp_print_key_should_output,&
      29              :                                               cp_print_key_unit_nr
      30              :    USE dft_plus_u,                      ONLY: plus_u
      31              :    USE ec_env_types,                    ONLY: energy_correction_type
      32              :    USE efield_utils,                    ONLY: calculate_ecore_efield,&
      33              :                                               efield_potential_lengh_gauge
      34              :    USE energy_corrections,              ONLY: energy_correction
      35              :    USE excited_states,                  ONLY: excited_state_energy
      36              :    USE hfx_exx,                         ONLY: calculate_exx
      37              :    USE input_constants,                 ONLY: &
      38              :         do_method_gapw, do_method_gapw_xc, do_method_gpw, do_method_lrigpw, do_method_ofgpw, &
      39              :         do_method_rigpw, ri_mp2_laplace, ri_mp2_method_gpw, ri_rpa_method_gpw
      40              :    USE input_section_types,             ONLY: section_vals_get,&
      41              :                                               section_vals_get_subs_vals,&
      42              :                                               section_vals_type,&
      43              :                                               section_vals_val_get
      44              :    USE kinds,                           ONLY: dp
      45              :    USE lri_environment_types,           ONLY: lri_environment_type
      46              :    USE message_passing,                 ONLY: mp_para_env_type
      47              :    USE mp2_cphf,                        ONLY: update_mp2_forces
      48              :    USE mulliken,                        ONLY: mulliken_restraint
      49              :    USE particle_types,                  ONLY: particle_type
      50              :    USE qs_core_energies,                ONLY: calculate_ecore_overlap,&
      51              :                                               calculate_ecore_self
      52              :    USE qs_core_hamiltonian,             ONLY: build_core_hamiltonian_matrix
      53              :    USE qs_dftb_dispersion,              ONLY: calculate_dftb_dispersion
      54              :    USE qs_dftb_matrices,                ONLY: build_dftb_matrices
      55              :    USE qs_energy,                       ONLY: qs_energies
      56              :    USE qs_energy_types,                 ONLY: qs_energy_type
      57              :    USE qs_environment_methods,          ONLY: qs_env_rebuild_pw_env
      58              :    USE qs_environment_types,            ONLY: get_qs_env,&
      59              :                                               qs_environment_type
      60              :    USE qs_external_potential,           ONLY: external_c_potential,&
      61              :                                               external_e_potential
      62              :    USE qs_force_types,                  ONLY: allocate_qs_force,&
      63              :                                               qs_force_type,&
      64              :                                               replicate_qs_force,&
      65              :                                               zero_qs_force
      66              :    USE qs_ks_methods,                   ONLY: qs_ks_update_qs_env
      67              :    USE qs_ks_types,                     ONLY: qs_ks_env_type,&
      68              :                                               set_ks_env
      69              :    USE qs_rho_types,                    ONLY: qs_rho_get,&
      70              :                                               qs_rho_type
      71              :    USE qs_scf_post_scf,                 ONLY: qs_scf_compute_properties
      72              :    USE qs_subsys_types,                 ONLY: qs_subsys_set,&
      73              :                                               qs_subsys_type
      74              :    USE ri_environment_methods,          ONLY: build_ri_matrices
      75              :    USE rt_propagation_forces,           ONLY: calc_c_mat_force,&
      76              :                                               rt_admm_force
      77              :    USE rt_propagation_velocity_gauge,   ONLY: velocity_gauge_ks_matrix,&
      78              :                                               velocity_gauge_nl_force
      79              :    USE se_core_core,                    ONLY: se_core_core_interaction
      80              :    USE se_core_matrix,                  ONLY: build_se_core_matrix
      81              :    USE tblite_interface,                ONLY: build_tblite_matrices,&
      82              :                                               tb_reference_cli_compare
      83              :    USE virial_types,                    ONLY: project_virial_to_periodic_subspace,&
      84              :                                               symmetrize_virial,&
      85              :                                               virial_type
      86              :    USE xtb_matrices,                    ONLY: build_xtb_matrices
      87              : #include "./base/base_uses.f90"
      88              : 
      89              :    IMPLICIT NONE
      90              : 
      91              :    PRIVATE
      92              : 
      93              : ! *** Global parameters ***
      94              : 
      95              :    CHARACTER(len=*), PARAMETER, PRIVATE :: moduleN = 'qs_force'
      96              : 
      97              : ! *** Public subroutines ***
      98              : 
      99              :    PUBLIC :: qs_calc_energy_force
     100              : 
     101              : CONTAINS
     102              : 
     103              : ! **************************************************************************************************
     104              : !> \brief ...
     105              : !> \param qs_env ...
     106              : !> \param calc_force ...
     107              : !> \param consistent_energies ...
     108              : !> \param linres ...
     109              : ! **************************************************************************************************
     110        27845 :    SUBROUTINE qs_calc_energy_force(qs_env, calc_force, consistent_energies, linres)
     111              :       TYPE(qs_environment_type), POINTER                 :: qs_env
     112              :       LOGICAL                                            :: calc_force, consistent_energies, linres
     113              : 
     114        27845 :       qs_env%linres_run = linres
     115        27845 :       IF (calc_force) THEN
     116        11211 :          CALL qs_forces(qs_env)
     117              :       ELSE
     118              :          CALL qs_energies(qs_env, calc_forces=.FALSE., &
     119        16634 :                           consistent_energies=consistent_energies)
     120              :       END IF
     121              : 
     122        27845 :    END SUBROUTINE qs_calc_energy_force
     123              : 
     124              : ! **************************************************************************************************
     125              : !> \brief   Calculate the Quickstep forces.
     126              : !> \param qs_env ...
     127              : !> \date    29.10.2002
     128              : !> \author  MK
     129              : !> \version 1.0
     130              : ! **************************************************************************************************
     131        11211 :    SUBROUTINE qs_forces(qs_env)
     132              : 
     133              :       TYPE(qs_environment_type), POINTER                 :: qs_env
     134              : 
     135              :       CHARACTER(len=*), PARAMETER                        :: routineN = 'qs_forces'
     136              : 
     137              :       INTEGER                                            :: after, handle, i, iatom, ic, ikind, &
     138              :                                                             ispin, iw, natom, nkind, nspin, &
     139              :                                                             output_unit
     140        11211 :       INTEGER, ALLOCATABLE, DIMENSION(:)                 :: atom_of_kind, kind_of, natom_of_kind
     141              :       LOGICAL                                            :: do_admm, do_exx, do_gw, do_im_time, &
     142              :                                                             has_unit_metric, omit_headers, &
     143              :                                                             perform_ec, reuse_hfx
     144              :       REAL(dp)                                           :: dummy_real, dummy_real2(2)
     145        11211 :       TYPE(atomic_kind_type), DIMENSION(:), POINTER      :: atomic_kind_set
     146              :       TYPE(cell_type), POINTER                           :: cell
     147              :       TYPE(cp_logger_type), POINTER                      :: logger
     148        11211 :       TYPE(dbcsr_p_type), DIMENSION(:), POINTER          :: matrix_s, matrix_w, rho_ao
     149        11211 :       TYPE(dbcsr_p_type), DIMENSION(:, :), POINTER       :: matrix_w_kp
     150              :       TYPE(dft_control_type), POINTER                    :: dft_control
     151              :       TYPE(energy_correction_type), POINTER              :: ec_env
     152              :       TYPE(lri_environment_type), POINTER                :: lri_env
     153              :       TYPE(mp_para_env_type), POINTER                    :: para_env
     154        11211 :       TYPE(particle_type), DIMENSION(:), POINTER         :: particle_set
     155              :       TYPE(qs_energy_type), POINTER                      :: energy
     156        11211 :       TYPE(qs_force_type), DIMENSION(:), POINTER         :: force
     157              :       TYPE(qs_ks_env_type), POINTER                      :: ks_env
     158              :       TYPE(qs_rho_type), POINTER                         :: rho
     159              :       TYPE(qs_subsys_type), POINTER                      :: subsys
     160              :       TYPE(section_vals_type), POINTER                   :: hfx_sections, print_section
     161              :       TYPE(virial_type), POINTER                         :: virial
     162              : 
     163        11211 :       CALL timeset(routineN, handle)
     164        11211 :       NULLIFY (logger)
     165        11211 :       logger => cp_get_default_logger()
     166              : 
     167              :       ! rebuild plane wave environment
     168        11211 :       CALL qs_env_rebuild_pw_env(qs_env)
     169              : 
     170              :       ! zero out the forces in particle set
     171        11211 :       CALL get_qs_env(qs_env, particle_set=particle_set)
     172        11211 :       natom = SIZE(particle_set)
     173        81370 :       DO iatom = 1, natom
     174       291847 :          particle_set(iatom)%f = 0.0_dp
     175              :       END DO
     176              : 
     177              :       ! get atom mapping
     178        11211 :       NULLIFY (atomic_kind_set)
     179        11211 :       CALL get_qs_env(qs_env, atomic_kind_set=atomic_kind_set)
     180              :       CALL get_atomic_kind_set(atomic_kind_set=atomic_kind_set, &
     181              :                                atom_of_kind=atom_of_kind, &
     182        11211 :                                kind_of=kind_of)
     183              : 
     184        11211 :       NULLIFY (force, subsys, dft_control)
     185              :       CALL get_qs_env(qs_env, &
     186              :                       force=force, &
     187              :                       subsys=subsys, &
     188        11211 :                       dft_control=dft_control)
     189        11211 :       IF (.NOT. ASSOCIATED(force)) THEN
     190              :          !   *** Allocate the force data structure ***
     191         3405 :          nkind = SIZE(atomic_kind_set)
     192         3405 :          CALL get_atomic_kind_set(atomic_kind_set=atomic_kind_set, natom_of_kind=natom_of_kind)
     193         3405 :          CALL allocate_qs_force(force, natom_of_kind)
     194         3405 :          DEALLOCATE (natom_of_kind)
     195         3405 :          CALL qs_subsys_set(subsys, force=force)
     196              :       END IF
     197        11211 :       CALL zero_qs_force(force)
     198              : 
     199              :       ! Check if CDFT potential is needed and save it until forces have been calculated
     200        11211 :       IF (dft_control%qs_control%cdft) THEN
     201          118 :          dft_control%qs_control%cdft_control%save_pot = .TRUE.
     202              :       END IF
     203              : 
     204              :       ! recalculate energy and the response vector for the Z-vector linear equation system if calc_force = .true.
     205        11211 :       CALL qs_energies(qs_env, calc_forces=.TRUE.)
     206              : 
     207        11211 :       NULLIFY (para_env)
     208              :       CALL get_qs_env(qs_env, &
     209        11211 :                       para_env=para_env)
     210              : 
     211              :       ! Now we handle some special cases
     212              :       ! Maybe some of these would be better dealt with in qs_energies?
     213        11211 :       IF (qs_env%run_rtp) THEN
     214         1218 :          NULLIFY (matrix_w, matrix_s, ks_env)
     215              :          CALL get_qs_env(qs_env, &
     216              :                          ks_env=ks_env, &
     217              :                          matrix_w=matrix_w, &
     218         1218 :                          matrix_s=matrix_s)
     219         1218 :          CALL dbcsr_allocate_matrix_set(matrix_w, dft_control%nspins)
     220         2688 :          DO ispin = 1, dft_control%nspins
     221         1470 :             ALLOCATE (matrix_w(ispin)%matrix)
     222              :             CALL dbcsr_copy(matrix_w(ispin)%matrix, matrix_s(1)%matrix, &
     223         1470 :                             name="W MATRIX")
     224         2688 :             CALL dbcsr_set(matrix_w(ispin)%matrix, 0.0_dp)
     225              :          END DO
     226         1218 :          CALL set_ks_env(ks_env, matrix_w=matrix_w)
     227              : 
     228         1218 :          CALL calc_c_mat_force(qs_env)
     229         1218 :          IF (dft_control%do_admm) CALL rt_admm_force(qs_env)
     230         1218 :          IF (dft_control%rtp_control%velocity_gauge .AND. dft_control%rtp_control%nl_gauge_transform) &
     231           22 :             CALL velocity_gauge_nl_force(qs_env, particle_set)
     232              :       END IF
     233              :       ! from an eventual Mulliken restraint
     234        11211 :       IF (dft_control%qs_control%mulliken_restraint) THEN
     235            6 :          NULLIFY (matrix_w, matrix_s, rho)
     236              :          CALL get_qs_env(qs_env, &
     237              :                          matrix_w=matrix_w, &
     238              :                          matrix_s=matrix_s, &
     239            6 :                          rho=rho)
     240            6 :          NULLIFY (rho_ao)
     241            6 :          CALL qs_rho_get(rho, rho_ao=rho_ao)
     242              :          CALL mulliken_restraint(dft_control%qs_control%mulliken_restraint_control, &
     243            6 :                                  para_env, matrix_s(1)%matrix, rho_ao, w_matrix=matrix_w)
     244              :       END IF
     245              :       ! Add non-Pulay contribution of DFT+U to W matrix, since it has also to be
     246              :       ! digested with overlap matrix derivatives
     247        11211 :       IF (dft_control%dft_plus_u) THEN
     248           76 :          NULLIFY (matrix_w_kp)
     249           76 :          CALL get_qs_env(qs_env, matrix_w_kp=matrix_w_kp)
     250           76 :          CALL plus_u(qs_env=qs_env, matrix_w=matrix_w_kp)
     251              :       END IF
     252              : 
     253              :       ! Write W Matrix to output (if requested)
     254        11211 :       CALL get_qs_env(qs_env, has_unit_metric=has_unit_metric)
     255        11211 :       IF (.NOT. has_unit_metric) THEN
     256         8187 :          NULLIFY (matrix_w_kp)
     257         8187 :          CALL get_qs_env(qs_env, matrix_w_kp=matrix_w_kp)
     258         8187 :          nspin = SIZE(matrix_w_kp, 1)
     259        17448 :          DO ispin = 1, nspin
     260         9261 :             IF (BTEST(cp_print_key_should_output(logger%iter_info, &
     261         8187 :                                                  qs_env%input, "DFT%PRINT%AO_MATRICES/W_MATRIX"), cp_p_file)) THEN
     262              :                iw = cp_print_key_unit_nr(logger, qs_env%input, "DFT%PRINT%AO_MATRICES/W_MATRIX", &
     263            8 :                                          extension=".Log")
     264            8 :                CALL section_vals_val_get(qs_env%input, "DFT%PRINT%AO_MATRICES%NDIGITS", i_val=after)
     265            8 :                CALL section_vals_val_get(qs_env%input, "DFT%PRINT%AO_MATRICES%OMIT_HEADERS", l_val=omit_headers)
     266            8 :                after = MIN(MAX(after, 1), 16)
     267           16 :                DO ic = 1, SIZE(matrix_w_kp, 2)
     268              :                   CALL cp_dbcsr_write_sparse_matrix(matrix_w_kp(ispin, ic)%matrix, 4, after, qs_env, &
     269           16 :                                                     para_env, output_unit=iw, omit_headers=omit_headers)
     270              :                END DO
     271              :                CALL cp_print_key_finished_output(iw, logger, qs_env%input, &
     272            8 :                                                  "DFT%PRINT%AO_MATRICES/W_MATRIX")
     273              :             END IF
     274              :          END DO
     275              :       END IF
     276              : 
     277              :       ! Check if energy correction should be skipped
     278        11211 :       perform_ec = .FALSE.
     279        11211 :       IF (qs_env%energy_correction) THEN
     280          494 :          CALL get_qs_env(qs_env, ec_env=ec_env)
     281          494 :          IF (.NOT. ec_env%do_skip) perform_ec = .TRUE.
     282              :       END IF
     283              : 
     284              :       ! Compute core forces (also overwrites matrix_w)
     285        11211 :       IF (dft_control%qs_control%semi_empirical) THEN
     286              :          CALL build_se_core_matrix(qs_env=qs_env, para_env=para_env, &
     287         3024 :                                    calculate_forces=.TRUE.)
     288         3024 :          CALL se_core_core_interaction(qs_env, para_env, calculate_forces=.TRUE.)
     289         8187 :       ELSEIF (dft_control%qs_control%dftb) THEN
     290              :          CALL build_dftb_matrices(qs_env=qs_env, para_env=para_env, &
     291          786 :                                   calculate_forces=.TRUE.)
     292              :          CALL calculate_dftb_dispersion(qs_env=qs_env, para_env=para_env, &
     293          786 :                                         calculate_forces=.TRUE.)
     294         7401 :       ELSEIF (dft_control%qs_control%xtb) THEN
     295          708 :          IF (dft_control%qs_control%xtb_control%do_tblite) THEN
     296          114 :             CALL build_tblite_matrices(qs_env=qs_env, calculate_forces=.TRUE.)
     297              :          ELSE
     298          594 :             CALL build_xtb_matrices(qs_env=qs_env, calculate_forces=.TRUE.)
     299              :          END IF
     300         6693 :       ELSEIF (perform_ec) THEN
     301              :          ! Calculates core and grid based forces
     302          494 :          CALL energy_correction(qs_env, ec_init=.FALSE., calculate_forces=.TRUE.)
     303              :       ELSE
     304              :          ! Dispersion energy and forces are calculated in qs_energy?
     305         6199 :          CALL build_core_hamiltonian_matrix(qs_env=qs_env, calculate_forces=.TRUE.)
     306              :          ! The above line reset the core H, which should be re-updated in case a TD field is applied:
     307         6199 :          IF (qs_env%run_rtp) THEN
     308          814 :             IF (dft_control%apply_efield_field) &
     309          160 :                CALL efield_potential_lengh_gauge(qs_env)
     310          814 :             IF (dft_control%rtp_control%velocity_gauge) &
     311           22 :                CALL velocity_gauge_ks_matrix(qs_env, subtract_nl_term=.FALSE.)
     312              : 
     313              :          END IF
     314         6199 :          CALL calculate_ecore_self(qs_env)
     315         6199 :          CALL calculate_ecore_overlap(qs_env, para_env, calculate_forces=.TRUE.)
     316         6199 :          CALL calculate_ecore_efield(qs_env, calculate_forces=.TRUE.)
     317              :          !swap external_e_potential before external_c_potential, to ensure
     318              :          !that external potential on grid is loaded before calculating energy of cores
     319         6199 :          CALL external_e_potential(qs_env)
     320         6199 :          IF (.NOT. dft_control%qs_control%gapw) THEN
     321         5613 :             CALL external_c_potential(qs_env, calculate_forces=.TRUE.)
     322              :          END IF
     323              :          ! RIGPW  matrices
     324         6199 :          IF (dft_control%qs_control%rigpw) THEN
     325            2 :             CALL get_qs_env(qs_env=qs_env, lri_env=lri_env)
     326            2 :             CALL build_ri_matrices(lri_env, qs_env, calculate_forces=.TRUE.)
     327              :          END IF
     328              :       END IF
     329              : 
     330              :       ! MP2 Code
     331        11211 :       IF (ASSOCIATED(qs_env%mp2_env)) THEN
     332          322 :          NULLIFY (energy)
     333          322 :          CALL get_qs_env(qs_env, energy=energy)
     334          322 :          CALL qs_scf_compute_properties(qs_env, wf_type='MP2   ', do_mp2=.TRUE.)
     335          322 :          CALL qs_ks_update_qs_env(qs_env, just_energy=.TRUE.)
     336          322 :          energy%total = energy%total + energy%mp2
     337              : 
     338              :          IF ((qs_env%mp2_env%method == ri_mp2_method_gpw .OR. qs_env%mp2_env%method == ri_mp2_laplace .OR. &
     339              :               qs_env%mp2_env%method == ri_rpa_method_gpw) &
     340          322 :              .AND. .NOT. qs_env%mp2_env%do_im_time) THEN
     341          272 :             CALL update_mp2_forces(qs_env)
     342              :          END IF
     343              : 
     344              :          !RPA EXX energy and forces
     345          322 :          IF (qs_env%mp2_env%method == ri_rpa_method_gpw) THEN
     346              :             do_exx = .FALSE.
     347           52 :             hfx_sections => section_vals_get_subs_vals(qs_env%input, "DFT%XC%WF_CORRELATION%RI_RPA%HF")
     348           52 :             CALL section_vals_get(hfx_sections, explicit=do_exx)
     349           52 :             IF (do_exx) THEN
     350           26 :                do_gw = qs_env%mp2_env%ri_rpa%do_ri_g0w0
     351           26 :                do_admm = qs_env%mp2_env%ri_rpa%do_admm
     352           26 :                reuse_hfx = qs_env%mp2_env%ri_rpa%reuse_hfx
     353           26 :                do_im_time = qs_env%mp2_env%do_im_time
     354           26 :                output_unit = cp_logger_get_default_io_unit()
     355           26 :                dummy_real = 0.0_dp
     356              : 
     357              :                CALL calculate_exx(qs_env=qs_env, &
     358              :                                   unit_nr=output_unit, &
     359              :                                   hfx_sections=hfx_sections, &
     360              :                                   x_data=qs_env%mp2_env%ri_rpa%x_data, &
     361              :                                   do_gw=do_gw, &
     362              :                                   do_admm=do_admm, &
     363              :                                   calc_forces=.TRUE., &
     364              :                                   reuse_hfx=reuse_hfx, &
     365              :                                   do_im_time=do_im_time, &
     366              :                                   E_ex_from_GW=dummy_real, &
     367              :                                   E_admm_from_GW=dummy_real2, &
     368           26 :                                   t3=dummy_real)
     369              :             END IF
     370              :          END IF
     371        10889 :       ELSEIF (perform_ec) THEN
     372              :          ! energy correction forces postponed
     373        10395 :       ELSEIF (qs_env%harris_method) THEN
     374              :          ! Harris method forces already done in harris_energy_correction
     375              :       ELSE
     376              :          ! Compute grid-based forces
     377        10389 :          CALL qs_ks_update_qs_env(qs_env, calculate_forces=.TRUE.)
     378              :       END IF
     379              : 
     380              :       ! Excited state forces
     381              :       ! Solve the response linear equation system for the Z-vector method
     382              :       ! and calculate remaining terms of the force
     383        11211 :       CALL excited_state_energy(qs_env, calculate_forces=.TRUE.)
     384              : 
     385              :       ! replicate forces (get current pointer)
     386        11211 :       NULLIFY (force)
     387        11211 :       CALL get_qs_env(qs_env=qs_env, force=force)
     388        11211 :       CALL replicate_qs_force(force, para_env)
     389              : 
     390        81370 :       DO iatom = 1, natom
     391        70159 :          ikind = kind_of(iatom)
     392        70159 :          i = atom_of_kind(iatom)
     393              :          ! XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX
     394              :          ! the force is - dE/dR, what is called force is actually the gradient
     395              :          ! Things should have the right name
     396              :          ! The minus sign below is a hack
     397              :          ! XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX
     398       561272 :          force(ikind)%other(1:3, i) = -particle_set(iatom)%f(1:3) + force(ikind)%ch_pulay(1:3, i)
     399       280636 :          force(ikind)%total(1:3, i) = force(ikind)%total(1:3, i) + force(ikind)%other(1:3, i)
     400       572483 :          particle_set(iatom)%f = -force(ikind)%total(1:3, i)
     401              :       END DO
     402              : 
     403        11211 :       NULLIFY (cell, virial, energy)
     404        11211 :       CALL get_qs_env(qs_env=qs_env, cell=cell, virial=virial, energy=energy)
     405        11211 :       IF (virial%pv_availability) THEN
     406         1086 :          CALL para_env%sum(virial%pv_overlap)
     407         1086 :          CALL para_env%sum(virial%pv_ekinetic)
     408         1086 :          CALL para_env%sum(virial%pv_ppl)
     409         1086 :          CALL para_env%sum(virial%pv_ppnl)
     410         1086 :          CALL para_env%sum(virial%pv_ecore_overlap)
     411         1086 :          CALL para_env%sum(virial%pv_ehartree)
     412         1086 :          CALL para_env%sum(virial%pv_exc)
     413         1086 :          CALL para_env%sum(virial%pv_exx)
     414         1086 :          CALL para_env%sum(virial%pv_vdw)
     415         1086 :          CALL para_env%sum(virial%pv_mp2)
     416         1086 :          CALL para_env%sum(virial%pv_nlcc)
     417         1086 :          CALL para_env%sum(virial%pv_gapw)
     418         1086 :          CALL para_env%sum(virial%pv_lrigpw)
     419         1086 :          CALL para_env%sum(virial%pv_virial)
     420         1086 :          CALL symmetrize_virial(virial)
     421              :          ! Add the volume terms of the virial
     422         1086 :          IF ((.NOT. virial%pv_numer) .AND. &
     423              :              (.NOT. (dft_control%qs_control%dftb .OR. &
     424              :                      dft_control%qs_control%xtb .OR. &
     425              :                      dft_control%qs_control%semi_empirical))) THEN
     426              : 
     427              :             ! Harris energy correction requires volume terms from
     428              :             ! 1) Harris functional contribution, and
     429              :             ! 2) Linear Response solver
     430          642 :             IF (perform_ec) THEN
     431          172 :                CALL get_qs_env(qs_env, ec_env=ec_env)
     432          172 :                energy%hartree = ec_env%ehartree
     433          172 :                energy%exc = ec_env%exc
     434          172 :                IF (dft_control%do_admm) THEN
     435           38 :                   energy%exc_aux_fit = ec_env%exc_aux_fit
     436              :                END IF
     437              :             END IF
     438         2568 :             DO i = 1, 3
     439              :                virial%pv_ehartree(i, i) = virial%pv_ehartree(i, i) &
     440         1926 :                                           - 2.0_dp*(energy%hartree + energy%sccs_pol)
     441              :                virial%pv_virial(i, i) = virial%pv_virial(i, i) - energy%exc &
     442         1926 :                                         - 2.0_dp*(energy%hartree + energy%sccs_pol)
     443         1926 :                virial%pv_exc(i, i) = virial%pv_exc(i, i) - energy%exc
     444         2568 :                IF (dft_control%do_admm) THEN
     445          222 :                   virial%pv_exc(i, i) = virial%pv_exc(i, i) - energy%exc_aux_fit
     446          222 :                   virial%pv_virial(i, i) = virial%pv_virial(i, i) - energy%exc_aux_fit
     447              :                END IF
     448              :                ! The factor 2 is a hack. It compensates the plus sign in h_stress/pw_poisson_solve.
     449              :                ! The sign in pw_poisson_solve is correct for FIST, but not for QS.
     450              :                ! There should be a more elegant solution to that ...
     451              :             END DO
     452              :          END IF
     453         4344 :          IF ((.NOT. virial%pv_numer) .AND. COUNT(cell%perd /= 0) == 2) THEN
     454           46 :             SELECT CASE (dft_control%qs_control%method_id)
     455              :             CASE (do_method_gapw, do_method_gapw_xc, do_method_gpw, &
     456              :                   do_method_lrigpw, do_method_ofgpw, do_method_rigpw)
     457           30 :                CALL project_virial_to_periodic_subspace(virial, cell%perd)
     458              :             END SELECT
     459              :          END IF
     460              :       END IF
     461              : 
     462        11211 :       IF (dft_control%qs_control%xtb .AND. dft_control%qs_control%xtb_control%do_tblite) THEN
     463          114 :          CALL tb_reference_cli_compare(qs_env)
     464              :       END IF
     465              : 
     466              :       output_unit = cp_print_key_unit_nr(logger, qs_env%input, "DFT%PRINT%DERIVATIVES", &
     467        11211 :                                          extension=".Log")
     468        11211 :       print_section => section_vals_get_subs_vals(qs_env%input, "DFT%PRINT%DERIVATIVES")
     469        11211 :       IF (dft_control%qs_control%semi_empirical) THEN
     470              :          CALL write_forces(force, atomic_kind_set, 2, output_unit=output_unit, &
     471         3024 :                            print_section=print_section)
     472         8187 :       ELSE IF (dft_control%qs_control%dftb) THEN
     473              :          CALL write_forces(force, atomic_kind_set, 4, output_unit=output_unit, &
     474          786 :                            print_section=print_section)
     475         7401 :       ELSE IF (dft_control%qs_control%xtb) THEN
     476              :          CALL write_forces(force, atomic_kind_set, 4, output_unit=output_unit, &
     477          708 :                            print_section=print_section)
     478         6693 :       ELSE IF (dft_control%qs_control%gapw .OR. dft_control%qs_control%gapw_xc) THEN
     479              :          CALL write_forces(force, atomic_kind_set, 1, output_unit=output_unit, &
     480          748 :                            print_section=print_section)
     481              :       ELSE
     482              :          CALL write_forces(force, atomic_kind_set, 0, output_unit=output_unit, &
     483         5945 :                            print_section=print_section)
     484              :       END IF
     485              :       CALL cp_print_key_finished_output(output_unit, logger, qs_env%input, &
     486        11211 :                                         "DFT%PRINT%DERIVATIVES")
     487              : 
     488              :       ! deallocate W Matrix:
     489        11211 :       NULLIFY (ks_env, matrix_w_kp)
     490              :       CALL get_qs_env(qs_env=qs_env, &
     491              :                       matrix_w_kp=matrix_w_kp, &
     492        11211 :                       ks_env=ks_env)
     493        11211 :       CALL dbcsr_deallocate_matrix_set(matrix_w_kp)
     494        11211 :       NULLIFY (matrix_w_kp)
     495        11211 :       CALL set_ks_env(ks_env, matrix_w_kp=matrix_w_kp)
     496              : 
     497        11211 :       DEALLOCATE (atom_of_kind, kind_of)
     498              : 
     499        11211 :       CALL timestop(handle)
     500              : 
     501        22422 :    END SUBROUTINE qs_forces
     502              : 
     503              : ! **************************************************************************************************
     504              : !> \brief   Write a Quickstep force data structure to output unit
     505              : !> \param qs_force ...
     506              : !> \param atomic_kind_set ...
     507              : !> \param ftype ...
     508              : !> \param output_unit ...
     509              : !> \param print_section ...
     510              : !> \date    05.06.2002
     511              : !> \author  MK
     512              : !> \version 1.0
     513              : ! **************************************************************************************************
     514        11211 :    SUBROUTINE write_forces(qs_force, atomic_kind_set, ftype, output_unit, &
     515              :                            print_section)
     516              : 
     517              :       TYPE(qs_force_type), DIMENSION(:), POINTER         :: qs_force
     518              :       TYPE(atomic_kind_type), DIMENSION(:), POINTER      :: atomic_kind_set
     519              :       INTEGER, INTENT(IN)                                :: ftype, output_unit
     520              :       TYPE(section_vals_type), POINTER                   :: print_section
     521              : 
     522              :       CHARACTER(LEN=13)                                  :: fmtstr5
     523              :       CHARACTER(LEN=15)                                  :: fmtstr4
     524              :       CHARACTER(LEN=20)                                  :: fmtstr3
     525              :       CHARACTER(LEN=35)                                  :: fmtstr2
     526              :       CHARACTER(LEN=48)                                  :: fmtstr1
     527              :       INTEGER                                            :: i, iatom, ikind, my_ftype, natom, ndigits
     528        11211 :       INTEGER, ALLOCATABLE, DIMENSION(:)                 :: atom_of_kind, kind_of
     529              :       REAL(KIND=dp), DIMENSION(3)                        :: grand_total
     530              : 
     531        11211 :       IF (output_unit > 0) THEN
     532              : 
     533          181 :          IF (.NOT. ASSOCIATED(qs_force)) THEN
     534              :             CALL cp_abort(__LOCATION__, &
     535              :                           "The qs_force pointer is not associated "// &
     536            0 :                           "and cannot be printed")
     537              :          END IF
     538              : 
     539              :          CALL get_atomic_kind_set(atomic_kind_set=atomic_kind_set, atom_of_kind=atom_of_kind, &
     540          181 :                                   kind_of=kind_of, natom=natom)
     541              : 
     542              :          ! Variable precision output of the forces
     543              :          CALL section_vals_val_get(print_section, "NDIGITS", &
     544          181 :                                    i_val=ndigits)
     545              : 
     546          181 :          fmtstr1 = "(/,/,T2,A,/,/,T3,A,T11,A,T23,A,T40,A1,2(  X,A1))"
     547          181 :          WRITE (UNIT=fmtstr1(41:42), FMT="(I2)") ndigits + 5
     548              : 
     549          181 :          fmtstr2 = "(/,(T2,I5,4X,I4,T18,A,T34,3F  .  ))"
     550          181 :          WRITE (UNIT=fmtstr2(32:33), FMT="(I2)") ndigits
     551          181 :          WRITE (UNIT=fmtstr2(29:30), FMT="(I2)") ndigits + 6
     552              : 
     553          181 :          fmtstr3 = "(/,T3,A,T34,3F  .  )"
     554          181 :          WRITE (UNIT=fmtstr3(18:19), FMT="(I2)") ndigits
     555          181 :          WRITE (UNIT=fmtstr3(15:16), FMT="(I2)") ndigits + 6
     556              : 
     557          181 :          fmtstr4 = "((T34,3F  .  ))"
     558          181 :          WRITE (UNIT=fmtstr4(12:13), FMT="(I2)") ndigits
     559          181 :          WRITE (UNIT=fmtstr4(9:10), FMT="(I2)") ndigits + 6
     560              : 
     561              :          fmtstr5 = "(/T2,A//T3,A)"
     562              : 
     563              :          WRITE (UNIT=output_unit, FMT=fmtstr1) &
     564          181 :             "FORCES [a.u.]", "Atom", "Kind", "Component", "X", "Y", "Z"
     565              : 
     566          181 :          grand_total(:) = 0.0_dp
     567              : 
     568          181 :          my_ftype = ftype
     569              : 
     570            0 :          SELECT CASE (my_ftype)
     571              :          CASE DEFAULT
     572            0 :             DO iatom = 1, natom
     573            0 :                ikind = kind_of(iatom)
     574            0 :                i = atom_of_kind(iatom)
     575              :                WRITE (UNIT=output_unit, FMT=fmtstr2) &
     576            0 :                   iatom, ikind, "         total", qs_force(ikind)%total(1:3, i)
     577            0 :                grand_total(1:3) = grand_total(1:3) + qs_force(ikind)%total(1:3, i)
     578              :             END DO
     579              :          CASE (0)
     580          476 :             DO iatom = 1, natom
     581          342 :                ikind = kind_of(iatom)
     582          342 :                i = atom_of_kind(iatom)
     583              :                WRITE (UNIT=output_unit, FMT=fmtstr2) &
     584         1368 :                   iatom, ikind, "       overlap", qs_force(ikind)%overlap(1:3, i), &
     585         1368 :                   iatom, ikind, "  overlap_admm", qs_force(ikind)%overlap_admm(1:3, i), &
     586         1368 :                   iatom, ikind, "       kinetic", qs_force(ikind)%kinetic(1:3, i), &
     587         1368 :                   iatom, ikind, "       gth_ppl", qs_force(ikind)%gth_ppl(1:3, i), &
     588         1368 :                   iatom, ikind, "      gth_nlcc", qs_force(ikind)%gth_nlcc(1:3, i), &
     589         1368 :                   iatom, ikind, "      gth_ppnl", qs_force(ikind)%gth_ppnl(1:3, i), &
     590         1368 :                   iatom, ikind, "  core_overlap", qs_force(ikind)%core_overlap(1:3, i), &
     591         1368 :                   iatom, ikind, "      rho_core", qs_force(ikind)%rho_core(1:3, i), &
     592         1368 :                   iatom, ikind, "      rho_elec", qs_force(ikind)%rho_elec(1:3, i), &
     593         1368 :                   iatom, ikind, "  rho_lri_elec", qs_force(ikind)%rho_lri_elec(1:3, i), &
     594         1368 :                   iatom, ikind, "      ch_pulay", qs_force(ikind)%ch_pulay(1:3, i), &
     595         1368 :                   iatom, ikind, "    dispersion", qs_force(ikind)%dispersion(1:3, i), &
     596         1368 :                   iatom, ikind, "           gCP", qs_force(ikind)%gcp(1:3, i), &
     597         1368 :                   iatom, ikind, "         other", qs_force(ikind)%other(1:3, i), &
     598         1368 :                   iatom, ikind, "       fock_4c", qs_force(ikind)%fock_4c(1:3, i), &
     599         1368 :                   iatom, ikind, "     ehrenfest", qs_force(ikind)%ehrenfest(1:3, i), &
     600         1368 :                   iatom, ikind, "        efield", qs_force(ikind)%efield(1:3, i), &
     601         1368 :                   iatom, ikind, "           eev", qs_force(ikind)%eev(1:3, i), &
     602         1368 :                   iatom, ikind, "   mp2_non_sep", qs_force(ikind)%mp2_non_sep(1:3, i), &
     603         1710 :                   iatom, ikind, "         total", qs_force(ikind)%total(1:3, i)
     604         1502 :                grand_total(1:3) = grand_total(1:3) + qs_force(ikind)%total(1:3, i)
     605              :             END DO
     606              :          CASE (1)
     607           76 :             DO iatom = 1, natom
     608           55 :                ikind = kind_of(iatom)
     609           55 :                i = atom_of_kind(iatom)
     610              :                WRITE (UNIT=output_unit, FMT=fmtstr2) &
     611          220 :                   iatom, ikind, "       overlap", qs_force(ikind)%overlap(1:3, i), &
     612          220 :                   iatom, ikind, "  overlap_admm", qs_force(ikind)%overlap_admm(1:3, i), &
     613          220 :                   iatom, ikind, "       kinetic", qs_force(ikind)%kinetic(1:3, i), &
     614          220 :                   iatom, ikind, "       gth_ppl", qs_force(ikind)%gth_ppl(1:3, i), &
     615          220 :                   iatom, ikind, "      gth_nlcc", qs_force(ikind)%gth_nlcc(1:3, i), &
     616          220 :                   iatom, ikind, "      gth_ppnl", qs_force(ikind)%gth_ppnl(1:3, i), &
     617          220 :                   iatom, ikind, " all_potential", qs_force(ikind)%all_potential(1:3, i), &
     618          220 :                   iatom, ikind, "cneo_potential", qs_force(ikind)%cneo_potential(1:3, i), &
     619          220 :                   iatom, ikind, "  core_overlap", qs_force(ikind)%core_overlap(1:3, i), &
     620          220 :                   iatom, ikind, "      rho_core", qs_force(ikind)%rho_core(1:3, i), &
     621          220 :                   iatom, ikind, "      rho_elec", qs_force(ikind)%rho_elec(1:3, i), &
     622          220 :                   iatom, ikind, "  rho_lri_elec", qs_force(ikind)%rho_lri_elec(1:3, i), &
     623          220 :                   iatom, ikind, "  rho_cneo_nuc", qs_force(ikind)%rho_cneo_nuc(1:3, i), &
     624          220 :                   iatom, ikind, "     vhxc_atom", qs_force(ikind)%vhxc_atom(1:3, i), &
     625          220 :                   iatom, ikind, "   g0s_Vh_elec", qs_force(ikind)%g0s_Vh_elec(1:3, i), &
     626          220 :                   iatom, ikind, "      ch_pulay", qs_force(ikind)%ch_pulay(1:3, i), &
     627          220 :                   iatom, ikind, "    dispersion", qs_force(ikind)%dispersion(1:3, i), &
     628          220 :                   iatom, ikind, "           gCP", qs_force(ikind)%gcp(1:3, i), &
     629          220 :                   iatom, ikind, "       fock_4c", qs_force(ikind)%fock_4c(1:3, i), &
     630          220 :                   iatom, ikind, "     ehrenfest", qs_force(ikind)%ehrenfest(1:3, i), &
     631          220 :                   iatom, ikind, "        efield", qs_force(ikind)%efield(1:3, i), &
     632          220 :                   iatom, ikind, "           eev", qs_force(ikind)%eev(1:3, i), &
     633          220 :                   iatom, ikind, "   mp2_non_sep", qs_force(ikind)%mp2_non_sep(1:3, i), &
     634          275 :                   iatom, ikind, "         total", qs_force(ikind)%total(1:3, i)
     635          241 :                grand_total(1:3) = grand_total(1:3) + qs_force(ikind)%total(1:3, i)
     636              :             END DO
     637              :          CASE (2)
     638           75 :             DO iatom = 1, natom
     639           73 :                ikind = kind_of(iatom)
     640           73 :                i = atom_of_kind(iatom)
     641              :                WRITE (UNIT=output_unit, FMT=fmtstr2) &
     642          292 :                   iatom, ikind, " all_potential", qs_force(ikind)%all_potential(1:3, i), &
     643          292 :                   iatom, ikind, "      rho_elec", qs_force(ikind)%rho_elec(1:3, i), &
     644          365 :                   iatom, ikind, "         total", qs_force(ikind)%total(1:3, i)
     645          294 :                grand_total(1:3) = grand_total(1:3) + qs_force(ikind)%total(1:3, i)
     646              :             END DO
     647              :          CASE (3)
     648            0 :             DO iatom = 1, natom
     649            0 :                ikind = kind_of(iatom)
     650            0 :                i = atom_of_kind(iatom)
     651              :                WRITE (UNIT=output_unit, FMT=fmtstr2) &
     652            0 :                   iatom, ikind, "        overlap", qs_force(ikind)%overlap(1:3, i), &
     653            0 :                   iatom, ikind, "overlap_admm", qs_force(ikind)%overlap_admm(1:3, i), &
     654            0 :                   iatom, ikind, "        kinetic", qs_force(ikind)%kinetic(1:3, i), &
     655            0 :                   iatom, ikind, "        gth_ppl", qs_force(ikind)%gth_ppl(1:3, i), &
     656            0 :                   iatom, ikind, "       gth_nlcc", qs_force(ikind)%gth_nlcc(1:3, i), &
     657            0 :                   iatom, ikind, "       gth_ppnl", qs_force(ikind)%gth_ppnl(1:3, i), &
     658            0 :                   iatom, ikind, "   core_overlap", qs_force(ikind)%core_overlap(1:3, i), &
     659            0 :                   iatom, ikind, "       rho_core", qs_force(ikind)%rho_core(1:3, i), &
     660            0 :                   iatom, ikind, "       rho_elec", qs_force(ikind)%rho_elec(1:3, i), &
     661            0 :                   iatom, ikind, "       ch_pulay", qs_force(ikind)%ch_pulay(1:3, i), &
     662            0 :                   iatom, ikind, "        fock_4c", qs_force(ikind)%fock_4c(1:3, i), &
     663            0 :                   iatom, ikind, "   mp2_non_sep", qs_force(ikind)%mp2_non_sep(1:3, i), &
     664            0 :                   iatom, ikind, "          total", qs_force(ikind)%total(1:3, i)
     665            0 :                grand_total(1:3) = grand_total(1:3) + qs_force(ikind)%total(1:3, i)
     666              :             END DO
     667              :          CASE (4)
     668          188 :             DO iatom = 1, natom
     669          164 :                ikind = kind_of(iatom)
     670          164 :                i = atom_of_kind(iatom)
     671              :                WRITE (UNIT=output_unit, FMT=fmtstr2) &
     672          656 :                   iatom, ikind, "  all_potential", qs_force(ikind)%all_potential(1:3, i), &
     673          656 :                   iatom, ikind, "        overlap", qs_force(ikind)%overlap(1:3, i), &
     674          656 :                   iatom, ikind, "       rho_elec", qs_force(ikind)%rho_elec(1:3, i), &
     675          656 :                   iatom, ikind, "      repulsive", qs_force(ikind)%repulsive(1:3, i), &
     676          656 :                   iatom, ikind, "     dispersion", qs_force(ikind)%dispersion(1:3, i), &
     677          656 :                   iatom, ikind, "        efield", qs_force(ikind)%efield(1:3, i), &
     678          656 :                   iatom, ikind, "     ehrenfest", qs_force(ikind)%ehrenfest(1:3, i), &
     679          820 :                   iatom, ikind, "          total", qs_force(ikind)%total(1:3, i)
     680          680 :                grand_total(1:3) = grand_total(1:3) + qs_force(ikind)%total(1:3, i)
     681              :             END DO
     682              :          CASE (5)
     683          181 :             DO iatom = 1, natom
     684            0 :                ikind = kind_of(iatom)
     685            0 :                i = atom_of_kind(iatom)
     686              :                WRITE (UNIT=output_unit, FMT=fmtstr2) &
     687            0 :                   iatom, ikind, "       overlap", qs_force(ikind)%overlap(1:3, i), &
     688            0 :                   iatom, ikind, "       kinetic", qs_force(ikind)%kinetic(1:3, i), &
     689            0 :                   iatom, ikind, "      rho_elec", qs_force(ikind)%rho_elec(1:3, i), &
     690            0 :                   iatom, ikind, "    dispersion", qs_force(ikind)%dispersion(1:3, i), &
     691            0 :                   iatom, ikind, " all potential", qs_force(ikind)%all_potential(1:3, i), &
     692            0 :                   iatom, ikind, "         other", qs_force(ikind)%other(1:3, i), &
     693            0 :                   iatom, ikind, "         total", qs_force(ikind)%total(1:3, i)
     694            0 :                grand_total(1:3) = grand_total(1:3) + qs_force(ikind)%total(1:3, i)
     695              :             END DO
     696              :          END SELECT
     697              : 
     698          181 :          WRITE (UNIT=output_unit, FMT=fmtstr3) "Sum of total", grand_total(1:3)
     699              : 
     700          181 :          DEALLOCATE (atom_of_kind)
     701          181 :          DEALLOCATE (kind_of)
     702              : 
     703              :       END IF
     704              : 
     705        11211 :    END SUBROUTINE write_forces
     706              : 
     707              : END MODULE qs_force
        

Generated by: LCOV version 2.0-1