LCOV - code coverage report
Current view: top level - src - qs_dispersion_d2.F (source / functions) Hit Total Coverage
Test: CP2K Regtests (git:bce21c9) Lines: 84 94 89.4 %
Date: 2024-09-26 06:30:18 Functions: 2 2 100.0 %

          Line data    Source code
       1             : !--------------------------------------------------------------------------------------------------!
       2             : !   CP2K: A general program to perform molecular dynamics simulations                              !
       3             : !   Copyright 2000-2024 CP2K developers group <https://cp2k.org>                                   !
       4             : !                                                                                                  !
       5             : !   SPDX-License-Identifier: GPL-2.0-or-later                                                      !
       6             : !--------------------------------------------------------------------------------------------------!
       7             : 
       8             : ! **************************************************************************************************
       9             : !> \brief Calculation of D2 dispersion
      10             : !> \author JGH
      11             : ! **************************************************************************************************
      12             : MODULE qs_dispersion_d2
      13             : 
      14             :    USE atomic_kind_types,               ONLY: atomic_kind_type,&
      15             :                                               get_atomic_kind,&
      16             :                                               get_atomic_kind_set
      17             :    USE atprop_types,                    ONLY: atprop_array_init,&
      18             :                                               atprop_type
      19             :    USE cell_types,                      ONLY: cell_type
      20             :    USE kinds,                           ONLY: dp
      21             :    USE physcon,                         ONLY: bohr,&
      22             :                                               kjmol
      23             :    USE qs_dispersion_types,             ONLY: qs_atom_dispersion_type,&
      24             :                                               qs_dispersion_type
      25             :    USE qs_environment_types,            ONLY: get_qs_env,&
      26             :                                               qs_environment_type
      27             :    USE qs_force_types,                  ONLY: qs_force_type
      28             :    USE qs_kind_types,                   ONLY: get_qs_kind,&
      29             :                                               qs_kind_type
      30             :    USE qs_neighbor_list_types,          ONLY: get_iterator_info,&
      31             :                                               neighbor_list_iterate,&
      32             :                                               neighbor_list_iterator_create,&
      33             :                                               neighbor_list_iterator_p_type,&
      34             :                                               neighbor_list_iterator_release,&
      35             :                                               neighbor_list_set_p_type
      36             :    USE virial_methods,                  ONLY: virial_pair_force
      37             :    USE virial_types,                    ONLY: virial_type
      38             : #include "./base/base_uses.f90"
      39             : 
      40             :    IMPLICIT NONE
      41             : 
      42             :    PRIVATE
      43             : 
      44             :    CHARACTER(len=*), PARAMETER, PRIVATE :: moduleN = 'qs_dispersion_d2'
      45             : 
      46             :    PUBLIC :: calculate_dispersion_d2_pairpot, dftd2_param
      47             : 
      48             : ! **************************************************************************************************
      49             : 
      50             : CONTAINS
      51             : 
      52             : ! **************************************************************************************************
      53             : !> \brief ...
      54             : !> \param qs_env ...
      55             : !> \param dispersion_env ...
      56             : !> \param evdw ...
      57             : !> \param calculate_forces ...
      58             : !> \param atevdw ...
      59             : ! **************************************************************************************************
      60          58 :    SUBROUTINE calculate_dispersion_d2_pairpot(qs_env, dispersion_env, evdw, calculate_forces, atevdw)
      61             : 
      62             :       TYPE(qs_environment_type), POINTER                 :: qs_env
      63             :       TYPE(qs_dispersion_type), POINTER                  :: dispersion_env
      64             :       REAL(KIND=dp), INTENT(OUT)                         :: evdw
      65             :       LOGICAL, INTENT(IN)                                :: calculate_forces
      66             :       REAL(KIND=dp), DIMENSION(:), OPTIONAL              :: atevdw
      67             : 
      68             :       CHARACTER(LEN=*), PARAMETER :: routineN = 'calculate_dispersion_d2_pairpot'
      69             : 
      70             :       INTEGER                                            :: atom_a, atom_b, handle, iatom, ikind, &
      71             :                                                             jatom, jkind, mepos, natom, nkind, &
      72             :                                                             num_pe, za, zb
      73          58 :       INTEGER, ALLOCATABLE, DIMENSION(:)                 :: atom_of_kind, atomnumber, kind_of
      74             :       LOGICAL                                            :: atenergy, atex, atstress, floating_a, &
      75             :                                                             ghost_a, use_virial
      76          58 :       LOGICAL, ALLOCATABLE, DIMENSION(:)                 :: dodisp, floating, ghost
      77             :       REAL(KIND=dp)                                      :: c6, dd, devdw, dfdmp, dr, er, fac, fdmp, &
      78             :                                                             rcc, rcut, s6, xp
      79          58 :       REAL(KIND=dp), ALLOCATABLE, DIMENSION(:)           :: c6d2, radd2
      80             :       REAL(KIND=dp), DIMENSION(3)                        :: fdij, rij
      81             :       REAL(KIND=dp), DIMENSION(3, 3)                     :: pv_virial_thread
      82          58 :       REAL(KIND=dp), DIMENSION(:), POINTER               :: atener
      83          58 :       REAL(KIND=dp), DIMENSION(:, :, :), POINTER         :: atstr
      84          58 :       TYPE(atomic_kind_type), DIMENSION(:), POINTER      :: atomic_kind_set
      85             :       TYPE(atprop_type), POINTER                         :: atprop
      86             :       TYPE(cell_type), POINTER                           :: cell
      87             :       TYPE(neighbor_list_iterator_p_type), &
      88          58 :          DIMENSION(:), POINTER                           :: nl_iterator
      89             :       TYPE(neighbor_list_set_p_type), DIMENSION(:), &
      90          58 :          POINTER                                         :: sab_vdw
      91             :       TYPE(qs_atom_dispersion_type), POINTER             :: disp_a
      92          58 :       TYPE(qs_force_type), DIMENSION(:), POINTER         :: force
      93          58 :       TYPE(qs_kind_type), DIMENSION(:), POINTER          :: qs_kind_set
      94             :       TYPE(virial_type), POINTER                         :: virial
      95             : 
      96          58 :       CALL timeset(routineN, handle)
      97             : 
      98          58 :       evdw = 0._dp
      99             : 
     100          58 :       NULLIFY (atomic_kind_set, qs_kind_set, sab_vdw)
     101             : 
     102             :       CALL get_qs_env(qs_env=qs_env, nkind=nkind, natom=natom, atomic_kind_set=atomic_kind_set, &
     103          58 :                       qs_kind_set=qs_kind_set, cell=cell, virial=virial, atprop=atprop)
     104             : 
     105             :       ! atomic energy and stress arrays
     106          58 :       atenergy = atprop%energy
     107          58 :       IF (atenergy) THEN
     108           0 :          CALL atprop_array_init(atprop%atevdw, natom)
     109           0 :          atener => atprop%atevdw
     110             :       END IF
     111          58 :       atstress = atprop%stress
     112          58 :       atstr => atprop%atstress
     113             :       ! external atomic energy
     114          58 :       atex = .FALSE.
     115          58 :       IF (PRESENT(atevdw)) THEN
     116           0 :          atex = .TRUE.
     117             :       END IF
     118             : 
     119          58 :       NULLIFY (force)
     120          58 :       CALL get_qs_env(qs_env=qs_env, force=force)
     121          58 :       CALL get_atomic_kind_set(atomic_kind_set, atom_of_kind=atom_of_kind, kind_of=kind_of)
     122          58 :       use_virial = virial%pv_availability .AND. (.NOT. virial%pv_numer)
     123          58 :       pv_virial_thread(:, :) = 0._dp
     124             : 
     125         522 :       ALLOCATE (dodisp(nkind), ghost(nkind), floating(nkind), atomnumber(nkind), c6d2(nkind), radd2(nkind))
     126         188 :       DO ikind = 1, nkind
     127         130 :          CALL get_atomic_kind(atomic_kind_set(ikind), z=za)
     128         130 :          CALL get_qs_kind(qs_kind_set(ikind), dispersion=disp_a, ghost=ghost_a, floating=floating_a)
     129         130 :          dodisp(ikind) = disp_a%defined
     130         130 :          ghost(ikind) = ghost_a
     131         130 :          floating(ikind) = floating_a
     132         130 :          atomnumber(ikind) = za
     133         130 :          c6d2(ikind) = disp_a%c6
     134         318 :          radd2(ikind) = disp_a%vdw_radii
     135             :       END DO
     136             : 
     137          58 :       rcut = 2._dp*dispersion_env%rc_disp
     138          58 :       s6 = dispersion_env%scaling
     139          58 :       dd = dispersion_env%exp_pre
     140             : 
     141          58 :       sab_vdw => dispersion_env%sab_vdw
     142          58 :       num_pe = 1
     143          58 :       CALL neighbor_list_iterator_create(nl_iterator, sab_vdw, nthread=num_pe)
     144             : 
     145          58 :       mepos = 0
     146       32703 :       DO WHILE (neighbor_list_iterate(nl_iterator, mepos=mepos) == 0)
     147       32645 :          CALL get_iterator_info(nl_iterator, mepos=mepos, ikind=ikind, jkind=jkind, iatom=iatom, jatom=jatom, r=rij)
     148             : 
     149       32645 :          IF (ghost(ikind) .OR. ghost(jkind) .OR. floating(ikind) .OR. floating(jkind)) CYCLE
     150             : 
     151       32147 :          IF (.NOT. (dodisp(ikind) .AND. dodisp(jkind))) CYCLE
     152             : 
     153       32147 :          za = atomnumber(ikind)
     154       32147 :          zb = atomnumber(jkind)
     155             :          ! vdW potential
     156      128588 :          dr = SQRT(SUM(rij(:)**2))
     157       32205 :          IF (dr <= rcut) THEN
     158       32147 :             fac = 1._dp
     159       32147 :             IF (iatom == jatom) fac = 0.5_dp
     160       32147 :             IF (dr > 0.001_dp) THEN
     161       31997 :                c6 = SQRT(c6d2(ikind)*c6d2(jkind))
     162       31997 :                rcc = radd2(ikind) + radd2(jkind)
     163       31997 :                er = EXP(-dd*(dr/rcc - 1._dp))
     164       31997 :                fdmp = 1._dp/(1._dp + er)
     165       31997 :                xp = s6*c6/dr**6
     166       31997 :                evdw = evdw - xp*fdmp*fac
     167       31997 :                IF (calculate_forces) THEN
     168       31432 :                   dfdmp = dd/rcc*er*fdmp*fdmp
     169       31432 :                   devdw = -xp*(-6._dp*fdmp/dr + dfdmp)
     170      125728 :                   fdij(:) = devdw*rij(:)/dr*fac
     171       31432 :                   atom_a = atom_of_kind(iatom)
     172       31432 :                   atom_b = atom_of_kind(jatom)
     173      125728 :                   force(ikind)%dispersion(:, atom_a) = force(ikind)%dispersion(:, atom_a) - fdij(:)
     174      125728 :                   force(jkind)%dispersion(:, atom_b) = force(jkind)%dispersion(:, atom_b) + fdij(:)
     175       31432 :                   IF (use_virial) THEN
     176        9842 :                      CALL virial_pair_force(pv_virial_thread, -1._dp, fdij, rij)
     177             :                   END IF
     178       31432 :                   IF (atstress) THEN
     179           0 :                      CALL virial_pair_force(atstr(:, :, iatom), -0.5_dp, fdij, rij)
     180           0 :                      CALL virial_pair_force(atstr(:, :, jatom), -0.5_dp, fdij, rij)
     181             :                   END IF
     182             :                END IF
     183       31997 :                IF (atenergy) THEN
     184           0 :                   atener(iatom) = atener(iatom) - 0.5_dp*xp*fdmp*fac
     185           0 :                   atener(jatom) = atener(jatom) - 0.5_dp*xp*fdmp*fac
     186             :                END IF
     187       31997 :                IF (atex) THEN
     188           0 :                   atevdw(iatom) = atevdw(iatom) - 0.5_dp*xp*fdmp*fac
     189           0 :                   atevdw(jatom) = atevdw(jatom) - 0.5_dp*xp*fdmp*fac
     190             :                END IF
     191             :             END IF
     192             :          END IF
     193             : 
     194             :       END DO
     195             : 
     196         754 :       virial%pv_virial = virial%pv_virial + pv_virial_thread
     197             : 
     198          58 :       CALL neighbor_list_iterator_release(nl_iterator)
     199             : 
     200          58 :       DEALLOCATE (dodisp, ghost, floating, atomnumber, radd2, c6d2)
     201             : 
     202          58 :       CALL timestop(handle)
     203             : 
     204         116 :    END SUBROUTINE calculate_dispersion_d2_pairpot
     205             : 
     206             : ! **************************************************************************************************
     207             : !> \brief ...
     208             : !> \param z ...
     209             : !> \param c6 ...
     210             : !> \param r ...
     211             : !> \param found ...
     212             : ! **************************************************************************************************
     213          38 :    SUBROUTINE dftd2_param(z, c6, r, found)
     214             : 
     215             :       INTEGER, INTENT(in)                                :: z
     216             :       REAL(KIND=dp), INTENT(inout)                       :: c6, r
     217             :       LOGICAL, INTENT(inout)                             :: found
     218             : 
     219             :       REAL(KIND=dp), DIMENSION(54), PARAMETER :: c6val = (/0.14_dp, 0.08_dp, 1.61_dp, 1.61_dp, &
     220             :          3.13_dp, 1.75_dp, 1.23_dp, 0.70_dp, 0.75_dp, 0.63_dp, 5.71_dp, 5.71_dp, 10.79_dp, 9.23_dp,&
     221             :          7.84_dp, 5.57_dp, 5.07_dp, 4.61_dp, 10.80_dp, 10.80_dp, 10.80_dp, 10.80_dp, 10.80_dp, &
     222             :          10.80_dp, 10.80_dp, 10.80_dp, 10.80_dp, 10.80_dp, 10.80_dp, 10.80_dp, 16.99_dp, 17.10_dp, &
     223             :          16.37_dp, 12.64_dp, 12.47_dp, 12.01_dp, 24.67_dp, 24.67_dp, 24.67_dp, 24.67_dp, 24.67_dp, &
     224             :          24.67_dp, 24.67_dp, 24.67_dp, 24.67_dp, 24.67_dp, 24.67_dp, 24.67_dp, 37.32_dp, 38.71_dp, &
     225             :          38.44_dp, 31.74_dp, 31.50_dp, 29.99_dp/)
     226             :       REAL(KIND=dp), DIMENSION(54), PARAMETER :: rval = (/1.001_dp, 1.012_dp, 0.825_dp, 1.408_dp, &
     227             :          1.485_dp, 1.452_dp, 1.397_dp, 1.342_dp, 1.287_dp, 1.243_dp, 1.144_dp, 1.364_dp, 1.639_dp, &
     228             :          1.716_dp, 1.705_dp, 1.683_dp, 1.639_dp, 1.595_dp, 1.485_dp, 1.474_dp, 1.562_dp, 1.562_dp, &
     229             :          1.562_dp, 1.562_dp, 1.562_dp, 1.562_dp, 1.562_dp, 1.562_dp, 1.562_dp, 1.562_dp, 1.650_dp, &
     230             :          1.727_dp, 1.760_dp, 1.771_dp, 1.749_dp, 1.727_dp, 1.628_dp, 1.606_dp, 1.639_dp, 1.639_dp, &
     231             :          1.639_dp, 1.639_dp, 1.639_dp, 1.639_dp, 1.639_dp, 1.639_dp, 1.639_dp, 1.639_dp, 1.672_dp, &
     232             :          1.804_dp, 1.881_dp, 1.892_dp, 1.892_dp, 1.881_dp/)
     233             : 
     234             : !
     235             : ! GRIMME DISPERSION PARAMETERS
     236             : ! Stefan Grimme, Semiempirical GGA-Type Density Functional Constructed
     237             : !                with a Long-Range Dispersion Correction, J. Comp. Chem. 27: 1787-1799 (2006)
     238             : ! doi:10.1002/jcc.20495
     239             : !
     240             : ! Conversion factor [Jnm^6mol^-1] -> [a.u.] : 17.34527758021901
     241             : ! Conversion factor [A] -> [a.u.] : 1.889726132885643
     242             : !
     243             : ! C6 values in [Jnm^6/mol]
     244             : ! vdW radii [A]
     245             : 
     246          38 :       IF (z > 0 .AND. z <= 54) THEN
     247          38 :          found = .TRUE.
     248          38 :          c6 = c6val(z)*1000._dp*bohr**6/kjmol
     249          38 :          r = rval(z)*bohr
     250             :       ELSE
     251           0 :          found = .FALSE.
     252             :       END IF
     253             : 
     254          38 :    END SUBROUTINE dftd2_param
     255             : 
     256             : ! **************************************************************************************************
     257             : 
     258             : END MODULE qs_dispersion_d2

Generated by: LCOV version 1.15