LCOV - code coverage report
Current view: top level - src - kpoint_coulomb_2c.F (source / functions) Hit Total Coverage
Test: CP2K Regtests (git:4dc10b3) Lines: 297 299 99.3 %
Date: 2024-11-21 06:45:46 Functions: 14 16 87.5 %

          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 Routines to compute the Coulomb integral V_(alpha beta)(k) for a k-point k using lattice
      10             : !>        summation in real space. These integrals are e.g. needed in periodic RI for RPA, GW
      11             : !> \par History
      12             : !>       2018.05 created [Jan Wilhelm]
      13             : !> \author Jan Wilhelm
      14             : ! **************************************************************************************************
      15             : MODULE kpoint_coulomb_2c
      16             :    USE atomic_kind_types,               ONLY: atomic_kind_type,&
      17             :                                               get_atomic_kind_set
      18             :    USE basis_set_types,                 ONLY: gto_basis_set_type
      19             :    USE cell_types,                      ONLY: cell_type,&
      20             :                                               get_cell,&
      21             :                                               pbc
      22             :    USE constants_operator,              ONLY: operator_coulomb
      23             :    USE cp_dbcsr_api,                    ONLY: &
      24             :         dbcsr_create, dbcsr_init_p, dbcsr_iterator_blocks_left, dbcsr_iterator_next_block, &
      25             :         dbcsr_iterator_start, dbcsr_iterator_stop, dbcsr_iterator_type, dbcsr_p_type, &
      26             :         dbcsr_release_p, dbcsr_reserve_all_blocks, dbcsr_set, dbcsr_type, dbcsr_type_no_symmetry
      27             :    USE generic_shg_integrals,           ONLY: int_operators_r12_ab_shg
      28             :    USE generic_shg_integrals_init,      ONLY: contraction_matrix_shg
      29             :    USE kinds,                           ONLY: dp
      30             :    USE kpoint_types,                    ONLY: get_kpoint_info,&
      31             :                                               kpoint_type
      32             :    USE mathconstants,                   ONLY: gaussi,&
      33             :                                               twopi,&
      34             :                                               z_one
      35             :    USE message_passing,                 ONLY: mp_para_env_type
      36             :    USE particle_types,                  ONLY: particle_type
      37             :    USE qs_environment_types,            ONLY: get_qs_env,&
      38             :                                               qs_environment_type
      39             :    USE qs_kind_types,                   ONLY: get_qs_kind,&
      40             :                                               qs_kind_type
      41             : #include "./base/base_uses.f90"
      42             : 
      43             :    IMPLICIT NONE
      44             : 
      45             :    PRIVATE
      46             : 
      47             :    CHARACTER(len=*), PARAMETER, PRIVATE :: moduleN = 'kpoint_coulomb_2c'
      48             : 
      49             :    PUBLIC :: build_2c_coulomb_matrix_kp, build_2c_coulomb_matrix_kp_small_cell
      50             : 
      51             : ! **************************************************************************************************
      52             : 
      53             :    TYPE two_d_util_type
      54             :       REAL(KIND=dp), ALLOCATABLE, DIMENSION(:, :)  :: block
      55             :    END TYPE
      56             : 
      57             : CONTAINS
      58             : 
      59             : ! **************************************************************************************************
      60             : !> \brief ...
      61             : !> \param matrix_v_kp ...
      62             : !> \param kpoints ...
      63             : !> \param basis_type ...
      64             : !> \param cell ...
      65             : !> \param particle_set ...
      66             : !> \param qs_kind_set ...
      67             : !> \param atomic_kind_set ...
      68             : !> \param size_lattice_sum ...
      69             : !> \param operator_type ...
      70             : !> \param ikp_start ...
      71             : !> \param ikp_end ...
      72             : ! **************************************************************************************************
      73         116 :    SUBROUTINE build_2c_coulomb_matrix_kp(matrix_v_kp, kpoints, basis_type, cell, particle_set, qs_kind_set, &
      74             :                                          atomic_kind_set, size_lattice_sum, operator_type, ikp_start, ikp_end)
      75             : 
      76             :       TYPE(dbcsr_p_type), DIMENSION(:, :), POINTER       :: matrix_v_kp
      77             :       TYPE(kpoint_type), POINTER                         :: kpoints
      78             :       CHARACTER(LEN=*), INTENT(IN)                       :: basis_type
      79             :       TYPE(cell_type), POINTER                           :: cell
      80             :       TYPE(particle_type), DIMENSION(:), POINTER         :: particle_set
      81             :       TYPE(qs_kind_type), DIMENSION(:), POINTER          :: qs_kind_set
      82             :       TYPE(atomic_kind_type), DIMENSION(:), POINTER      :: atomic_kind_set
      83             :       INTEGER                                            :: size_lattice_sum, operator_type, &
      84             :                                                             ikp_start, ikp_end
      85             : 
      86             :       CHARACTER(LEN=*), PARAMETER :: routineN = 'build_2c_coulomb_matrix_kp'
      87             : 
      88             :       INTEGER                                            :: handle
      89             :       TYPE(dbcsr_type), POINTER                          :: matrix_v_L_tmp
      90             : 
      91         116 :       CALL timeset(routineN, handle)
      92             : 
      93         116 :       CALL allocate_tmp(matrix_v_L_tmp, matrix_v_kp, ikp_start)
      94             : 
      95             :       CALL lattice_sum(matrix_v_kp, kpoints, basis_type, cell, particle_set, &
      96             :                        qs_kind_set, atomic_kind_set, size_lattice_sum, matrix_v_L_tmp, &
      97         116 :                        operator_type, ikp_start, ikp_end)
      98             : 
      99         116 :       CALL deallocate_tmp(matrix_v_L_tmp)
     100             : 
     101         116 :       CALL timestop(handle)
     102             : 
     103         116 :    END SUBROUTINE build_2c_coulomb_matrix_kp
     104             : 
     105             : ! **************************************************************************************************
     106             : !> \brief ...
     107             : !> \param matrix_v_kp ...
     108             : !> \param kpoints ...
     109             : !> \param basis_type ...
     110             : !> \param cell ...
     111             : !> \param particle_set ...
     112             : !> \param qs_kind_set ...
     113             : !> \param atomic_kind_set ...
     114             : !> \param size_lattice_sum ...
     115             : !> \param matrix_v_L_tmp ...
     116             : !> \param operator_type ...
     117             : !> \param ikp_start ...
     118             : !> \param ikp_end ...
     119             : ! **************************************************************************************************
     120         116 :    SUBROUTINE lattice_sum(matrix_v_kp, kpoints, basis_type, cell, particle_set, &
     121             :                           qs_kind_set, atomic_kind_set, size_lattice_sum, matrix_v_L_tmp, &
     122             :                           operator_type, ikp_start, ikp_end)
     123             : 
     124             :       TYPE(dbcsr_p_type), DIMENSION(:, :), POINTER       :: matrix_v_kp
     125             :       TYPE(kpoint_type), POINTER                         :: kpoints
     126             :       CHARACTER(LEN=*), INTENT(IN)                       :: basis_type
     127             :       TYPE(cell_type), POINTER                           :: cell
     128             :       TYPE(particle_type), DIMENSION(:), POINTER         :: particle_set
     129             :       TYPE(qs_kind_type), DIMENSION(:), POINTER          :: qs_kind_set
     130             :       TYPE(atomic_kind_type), DIMENSION(:), POINTER      :: atomic_kind_set
     131             :       INTEGER                                            :: size_lattice_sum
     132             :       TYPE(dbcsr_type), POINTER                          :: matrix_v_L_tmp
     133             :       INTEGER                                            :: operator_type, ikp_start, ikp_end
     134             : 
     135             :       CHARACTER(LEN=*), PARAMETER                        :: routineN = 'lattice_sum'
     136             : 
     137             :       INTEGER :: factor, handle, handle2, i_block, i_x, i_x_inner, i_x_outer, ik, j_y, j_y_inner, &
     138             :          j_y_outer, k_z, k_z_inner, k_z_outer, x_max, x_min, y_max, y_min, z_max, z_min
     139             :       INTEGER, DIMENSION(3)                              :: nkp_grid
     140             :       REAL(KIND=dp)                                      :: coskl, sinkl
     141             :       REAL(KIND=dp), DIMENSION(3)                        :: vec_L, vec_s
     142             :       REAL(KIND=dp), DIMENSION(3, 3)                     :: hmat
     143         116 :       TYPE(two_d_util_type), ALLOCATABLE, DIMENSION(:)   :: blocks_v_L, blocks_v_L_store
     144             :       TYPE(two_d_util_type), ALLOCATABLE, &
     145         116 :          DIMENSION(:, :, :)                              :: blocks_v_kp
     146             : 
     147         116 :       CALL timeset(routineN, handle)
     148             : 
     149             :       CALL get_factor_and_xyz_min_max(cell, kpoints, size_lattice_sum, factor, hmat, &
     150         116 :                                       x_min, x_max, y_min, y_max, z_min, z_max, nkp_grid)
     151             : 
     152         116 :       CALL allocate_blocks_v_kp(blocks_v_kp, matrix_v_kp, ikp_start, ikp_end)
     153         116 :       CALL allocate_blocks_v_L(blocks_v_L, matrix_v_L_tmp)
     154         116 :       CALL allocate_blocks_v_L(blocks_v_L_store, matrix_v_L_tmp)
     155             : 
     156         536 :       DO i_x_inner = 0, 2*nkp_grid(1) - 1
     157        4208 :          DO j_y_inner = 0, 2*nkp_grid(2) - 1
     158       28716 :             DO k_z_inner = 0, 2*nkp_grid(3) - 1
     159             : 
     160       59360 :                DO i_x_outer = x_min, x_max + nkp_grid(1), 2*nkp_grid(1)
     161      187024 :                   DO j_y_outer = y_min, y_max + nkp_grid(2), 2*nkp_grid(2)
     162      658448 :                      DO k_z_outer = z_min, z_max + nkp_grid(3), 2*nkp_grid(3)
     163             : 
     164      496048 :                         i_x = i_x_inner + i_x_outer
     165      496048 :                         j_y = j_y_inner + j_y_outer
     166      496048 :                         k_z = k_z_inner + k_z_outer
     167             : 
     168             :                         IF (i_x > x_max .OR. i_x < x_min .OR. &
     169             :                             j_y > y_max .OR. j_y < y_min .OR. &
     170      496048 :                             k_z > z_max .OR. k_z < z_min) CYCLE
     171             : 
     172      766520 :                         vec_s = [REAL(i_x, dp), REAL(j_y, dp), REAL(k_z, dp)]
     173             : 
     174     2491190 :                         vec_L = MATMUL(hmat, vec_s)
     175             : 
     176             :                         ! Compute (P 0 | Q vec_L) and store it in matrix_v_L_tmp
     177             :                         CALL compute_v_transl(matrix_v_L_tmp, blocks_v_L, vec_L, particle_set, &
     178             :                                               qs_kind_set, atomic_kind_set, basis_type, cell, &
     179      191630 :                                               operator_type)
     180             : 
     181      971874 :                         DO i_block = 1, SIZE(blocks_v_L)
     182             :                            blocks_v_L_store(i_block)%block(:, :) = blocks_v_L_store(i_block)%block(:, :) &
     183   141431196 :                                                                    + blocks_v_L(i_block)%block(:, :)
     184             :                         END DO
     185             : 
     186             :                      END DO
     187             :                   END DO
     188             :                END DO
     189             : 
     190       24624 :                CALL timeset(routineN//"_R_to_k", handle2)
     191             : 
     192             :                ! add exp(iq*vec_L) * (P 0 | Q vec_L) to V_PQ(q)
     193      198752 :                DO ik = ikp_start, ikp_end
     194             : 
     195             :                   ! coskl and sinkl are identical for all i_x_outer, j_y_outer, k_z_outer
     196      696512 :                   coskl = COS(twopi*DOT_PRODUCT(vec_s(1:3), kpoints%xkp(1:3, ik)))
     197      696512 :                   sinkl = SIN(twopi*DOT_PRODUCT(vec_s(1:3), kpoints%xkp(1:3, ik)))
     198             : 
     199      858048 :                   DO i_block = 1, SIZE(blocks_v_L)
     200             : 
     201             :                      blocks_v_kp(ik, 1, i_block)%block(:, :) = blocks_v_kp(ik, 1, i_block)%block(:, :) &
     202   294907424 :                                                                + coskl*blocks_v_L_store(i_block)%block(:, :)
     203             :                      blocks_v_kp(ik, 2, i_block)%block(:, :) = blocks_v_kp(ik, 2, i_block)%block(:, :) &
     204   295081552 :                                                                + sinkl*blocks_v_L_store(i_block)%block(:, :)
     205             : 
     206             :                   END DO
     207             : 
     208             :                END DO
     209             : 
     210      107792 :                DO i_block = 1, SIZE(blocks_v_L)
     211             : 
     212    20592560 :                   blocks_v_L_store(i_block)%block(:, :) = 0.0_dp
     213             : 
     214             :                END DO
     215             : 
     216       52920 :                CALL timestop(handle2)
     217             : 
     218             :             END DO
     219             :          END DO
     220             :       END DO
     221             : 
     222         116 :       CALL set_blocks_to_matrix_v_kp(matrix_v_kp, blocks_v_kp, ikp_start, ikp_end)
     223             : 
     224         116 :       CALL deallocate_blocks_v_kp(blocks_v_kp)
     225         116 :       CALL deallocate_blocks_v_L(blocks_v_L)
     226         116 :       CALL deallocate_blocks_v_L(blocks_v_L_store)
     227             : 
     228         116 :       CALL timestop(handle)
     229             : 
     230         116 :    END SUBROUTINE lattice_sum
     231             : 
     232             : ! **************************************************************************************************
     233             : !> \brief ...
     234             : !> \param matrix_v_kp ...
     235             : !> \param blocks_v_kp ...
     236             : !> \param ikp_start ...
     237             : !> \param ikp_end ...
     238             : ! **************************************************************************************************
     239         116 :    SUBROUTINE set_blocks_to_matrix_v_kp(matrix_v_kp, blocks_v_kp, ikp_start, ikp_end)
     240             : 
     241             :       TYPE(dbcsr_p_type), DIMENSION(:, :), POINTER       :: matrix_v_kp
     242             :       TYPE(two_d_util_type), ALLOCATABLE, &
     243             :          DIMENSION(:, :, :)                              :: blocks_v_kp
     244             :       INTEGER                                            :: ikp_start, ikp_end
     245             : 
     246             :       CHARACTER(LEN=*), PARAMETER :: routineN = 'set_blocks_to_matrix_v_kp'
     247             : 
     248             :       INTEGER                                            :: col, handle, i_block, i_real_im, ik, row
     249         116 :       REAL(KIND=dp), DIMENSION(:, :), POINTER            :: data_block
     250             :       TYPE(dbcsr_iterator_type)                          :: iter
     251             : 
     252         116 :       CALL timeset(routineN, handle)
     253             : 
     254         858 :       DO ik = ikp_start, ikp_end
     255             : 
     256        2342 :          DO i_real_im = 1, 2
     257             : 
     258        1484 :             i_block = 1
     259             : 
     260        1484 :             CALL dbcsr_iterator_start(iter, matrix_v_kp(ik, i_real_im)%matrix)
     261             : 
     262        7092 :             DO WHILE (dbcsr_iterator_blocks_left(iter))
     263             : 
     264        5608 :                CALL dbcsr_iterator_next_block(iter, row, col, data_block)
     265             : 
     266     2522828 :                data_block(:, :) = blocks_v_kp(ik, i_real_im, i_block)%block(:, :)
     267             : 
     268        5608 :                i_block = i_block + 1
     269             : 
     270             :             END DO
     271             : 
     272        3710 :             CALL dbcsr_iterator_stop(iter)
     273             : 
     274             :          END DO
     275             : 
     276             :       END DO
     277             : 
     278         116 :       CALL timestop(handle)
     279             : 
     280         116 :    END SUBROUTINE set_blocks_to_matrix_v_kp
     281             : 
     282             : ! **************************************************************************************************
     283             : !> \brief ...
     284             : !> \param matrix_v_L_tmp ...
     285             : !> \param blocks_v_L ...
     286             : !> \param vec_L ...
     287             : !> \param particle_set ...
     288             : !> \param qs_kind_set ...
     289             : !> \param atomic_kind_set ...
     290             : !> \param basis_type ...
     291             : !> \param cell ...
     292             : !> \param operator_type ...
     293             : ! **************************************************************************************************
     294      191630 :    SUBROUTINE compute_v_transl(matrix_v_L_tmp, blocks_v_L, vec_L, particle_set, &
     295             :                                qs_kind_set, atomic_kind_set, basis_type, cell, operator_type)
     296             : 
     297             :       TYPE(dbcsr_type), POINTER                          :: matrix_v_L_tmp
     298             :       TYPE(two_d_util_type), ALLOCATABLE, DIMENSION(:)   :: blocks_v_L
     299             :       REAL(KIND=dp), DIMENSION(3)                        :: vec_L
     300             :       TYPE(particle_type), DIMENSION(:), POINTER         :: particle_set
     301             :       TYPE(qs_kind_type), DIMENSION(:), POINTER          :: qs_kind_set
     302             :       TYPE(atomic_kind_type), DIMENSION(:), POINTER      :: atomic_kind_set
     303             :       CHARACTER(LEN=*), INTENT(IN)                       :: basis_type
     304             :       TYPE(cell_type), POINTER                           :: cell
     305             :       INTEGER                                            :: operator_type
     306             : 
     307             :       CHARACTER(LEN=*), PARAMETER                        :: routineN = 'compute_v_transl'
     308             : 
     309             :       INTEGER                                            :: col, handle, i_block, kind_a, kind_b, row
     310      191630 :       INTEGER, ALLOCATABLE, DIMENSION(:)                 :: kind_of
     311             :       REAL(dp), DIMENSION(3)                             :: ra, rab_L, rb
     312      191630 :       REAL(KIND=dp), DIMENSION(:, :), POINTER            :: data_block
     313      191630 :       REAL(KIND=dp), DIMENSION(:, :, :), POINTER         :: contr_a, contr_b
     314             :       TYPE(dbcsr_iterator_type)                          :: iter
     315             :       TYPE(gto_basis_set_type), POINTER                  :: basis_set_a, basis_set_b
     316             : 
     317      191630 :       CALL timeset(routineN, handle)
     318             : 
     319      191630 :       NULLIFY (basis_set_a, basis_set_b, data_block)
     320             : 
     321      191630 :       CALL get_atomic_kind_set(atomic_kind_set, kind_of=kind_of)
     322             : 
     323      191630 :       CALL dbcsr_set(matrix_v_L_tmp, 0.0_dp)
     324             : 
     325      191630 :       i_block = 1
     326             : 
     327      191630 :       CALL dbcsr_iterator_start(iter, matrix_v_L_tmp)
     328             : 
     329      844210 :       DO WHILE (dbcsr_iterator_blocks_left(iter))
     330             : 
     331      652580 :          CALL dbcsr_iterator_next_block(iter, row, col, data_block)
     332             : 
     333      652580 :          kind_a = kind_of(row)
     334      652580 :          kind_b = kind_of(col)
     335             : 
     336      652580 :          CALL get_qs_kind(qs_kind=qs_kind_set(kind_a), basis_set=basis_set_a, basis_type=basis_type)
     337      652580 :          CALL get_qs_kind(qs_kind=qs_kind_set(kind_b), basis_set=basis_set_b, basis_type=basis_type)
     338             : 
     339      652580 :          ra(1:3) = pbc(particle_set(row)%r(1:3), cell)
     340      652580 :          rb(1:3) = pbc(particle_set(col)%r(1:3), cell)
     341             : 
     342     2610320 :          rab_L(1:3) = rb(1:3) - ra(1:3) + vec_L(1:3)
     343             : 
     344      652580 :          CALL contraction_matrix_shg(basis_set_a, contr_a)
     345      652580 :          CALL contraction_matrix_shg(basis_set_b, contr_b)
     346             : 
     347   140935148 :          blocks_v_L(i_block)%block = 0.0_dp
     348             : 
     349             :          CALL int_operators_r12_ab_shg(operator_type, blocks_v_L(i_block)%block, rab=rab_L, &
     350             :                                        fba=basis_set_a, fbb=basis_set_b, scona_shg=contr_a, sconb_shg=contr_b, &
     351      652580 :                                        calculate_forces=.FALSE.)
     352             : 
     353      652580 :          i_block = i_block + 1
     354             : 
     355      652580 :          DEALLOCATE (contr_a, contr_b)
     356             : 
     357             :       END DO
     358             : 
     359      191630 :       CALL dbcsr_iterator_stop(iter)
     360             : 
     361      191630 :       DEALLOCATE (kind_of)
     362             : 
     363      191630 :       CALL timestop(handle)
     364             : 
     365      191630 :    END SUBROUTINE compute_v_transl
     366             : 
     367             : ! **************************************************************************************************
     368             : !> \brief ...
     369             : !> \param blocks_v_kp ...
     370             : ! **************************************************************************************************
     371         116 :    SUBROUTINE deallocate_blocks_v_kp(blocks_v_kp)
     372             :       TYPE(two_d_util_type), ALLOCATABLE, &
     373             :          DIMENSION(:, :, :)                              :: blocks_v_kp
     374             : 
     375             :       CHARACTER(LEN=*), PARAMETER :: routineN = 'deallocate_blocks_v_kp'
     376             : 
     377             :       INTEGER                                            :: handle, i_block, i_real_img, ik
     378             : 
     379         116 :       CALL timeset(routineN, handle)
     380             : 
     381        1090 :       DO ik = LBOUND(blocks_v_kp, 1), UBOUND(blocks_v_kp, 1)
     382        2342 :          DO i_real_img = 1, SIZE(blocks_v_kp, 2)
     383        7834 :             DO i_block = 1, SIZE(blocks_v_kp, 3)
     384        7092 :                DEALLOCATE (blocks_v_kp(ik, i_real_img, i_block)%block)
     385             :             END DO
     386             :          END DO
     387             :       END DO
     388             : 
     389        5724 :       DEALLOCATE (blocks_v_kp)
     390             : 
     391         116 :       CALL timestop(handle)
     392             : 
     393         116 :    END SUBROUTINE
     394             : 
     395             : ! **************************************************************************************************
     396             : !> \brief ...
     397             : !> \param blocks_v_L ...
     398             : ! **************************************************************************************************
     399         232 :    SUBROUTINE deallocate_blocks_v_L(blocks_v_L)
     400             :       TYPE(two_d_util_type), ALLOCATABLE, DIMENSION(:)   :: blocks_v_L
     401             : 
     402             :       CHARACTER(LEN=*), PARAMETER :: routineN = 'deallocate_blocks_v_L'
     403             : 
     404             :       INTEGER                                            :: handle, i_block
     405             : 
     406         232 :       CALL timeset(routineN, handle)
     407             : 
     408        1016 :       DO i_block = 1, SIZE(blocks_v_L, 1)
     409        1016 :          DEALLOCATE (blocks_v_L(i_block)%block)
     410             :       END DO
     411             : 
     412        1016 :       DEALLOCATE (blocks_v_L)
     413             : 
     414         232 :       CALL timestop(handle)
     415             : 
     416         232 :    END SUBROUTINE
     417             : 
     418             : ! **************************************************************************************************
     419             : !> \brief ...
     420             : !> \param blocks_v_L ...
     421             : !> \param matrix_v_L_tmp ...
     422             : ! **************************************************************************************************
     423         232 :    SUBROUTINE allocate_blocks_v_L(blocks_v_L, matrix_v_L_tmp)
     424             :       TYPE(two_d_util_type), ALLOCATABLE, DIMENSION(:)   :: blocks_v_L
     425             :       TYPE(dbcsr_type), POINTER                          :: matrix_v_L_tmp
     426             : 
     427             :       CHARACTER(LEN=*), PARAMETER :: routineN = 'allocate_blocks_v_L'
     428             : 
     429             :       INTEGER                                            :: col, handle, i_block, nblocks, row
     430         232 :       REAL(KIND=dp), DIMENSION(:, :), POINTER            :: data_block
     431             :       TYPE(dbcsr_iterator_type)                          :: iter
     432             : 
     433         232 :       CALL timeset(routineN, handle)
     434             : 
     435         232 :       nblocks = 0
     436             : 
     437         232 :       CALL dbcsr_iterator_start(iter, matrix_v_L_tmp)
     438             : 
     439        1016 :       DO WHILE (dbcsr_iterator_blocks_left(iter))
     440             : 
     441         784 :          CALL dbcsr_iterator_next_block(iter, row, col, data_block)
     442             : 
     443         784 :          nblocks = nblocks + 1
     444             : 
     445             :       END DO
     446             : 
     447         232 :       CALL dbcsr_iterator_stop(iter)
     448             : 
     449        1480 :       ALLOCATE (blocks_v_L(nblocks))
     450             : 
     451         232 :       i_block = 1
     452             : 
     453         232 :       CALL dbcsr_iterator_start(iter, matrix_v_L_tmp)
     454             : 
     455        1016 :       DO WHILE (dbcsr_iterator_blocks_left(iter))
     456             : 
     457         784 :          CALL dbcsr_iterator_next_block(iter, row, col, data_block)
     458             : 
     459        3136 :          ALLOCATE (blocks_v_L(i_block)%block(SIZE(data_block, 1), SIZE(data_block, 2)))
     460      224310 :          blocks_v_L(i_block)%block = 0.0_dp
     461             : 
     462        1800 :          i_block = i_block + 1
     463             : 
     464             :       END DO
     465             : 
     466         232 :       CALL dbcsr_iterator_stop(iter)
     467             : 
     468         232 :       CALL timestop(handle)
     469             : 
     470         464 :    END SUBROUTINE
     471             : 
     472             : ! **************************************************************************************************
     473             : !> \brief ...
     474             : !> \param blocks_v_kp ...
     475             : !> \param matrix_v_kp ...
     476             : !> \param ikp_start ...
     477             : !> \param ikp_end ...
     478             : ! **************************************************************************************************
     479         116 :    SUBROUTINE allocate_blocks_v_kp(blocks_v_kp, matrix_v_kp, ikp_start, ikp_end)
     480             :       TYPE(two_d_util_type), ALLOCATABLE, &
     481             :          DIMENSION(:, :, :)                              :: blocks_v_kp
     482             :       TYPE(dbcsr_p_type), DIMENSION(:, :), POINTER       :: matrix_v_kp
     483             :       INTEGER                                            :: ikp_start, ikp_end
     484             : 
     485             :       CHARACTER(LEN=*), PARAMETER :: routineN = 'allocate_blocks_v_kp'
     486             : 
     487             :       INTEGER                                            :: col, handle, i_block, i_real_img, ik, &
     488             :                                                             nblocks, row
     489         116 :       REAL(KIND=dp), DIMENSION(:, :), POINTER            :: data_block
     490             :       TYPE(dbcsr_iterator_type)                          :: iter
     491             : 
     492         116 :       CALL timeset(routineN, handle)
     493             : 
     494         116 :       nblocks = 0
     495             : 
     496         116 :       CALL dbcsr_iterator_start(iter, matrix_v_kp(ikp_start, 1)%matrix)
     497             : 
     498         508 :       DO WHILE (dbcsr_iterator_blocks_left(iter))
     499             : 
     500         392 :          CALL dbcsr_iterator_next_block(iter, row, col, data_block)
     501             : 
     502         392 :          nblocks = nblocks + 1
     503             : 
     504             :       END DO
     505             : 
     506         116 :       CALL dbcsr_iterator_stop(iter)
     507             : 
     508        7364 :       ALLOCATE (blocks_v_kp(ikp_start:ikp_end, SIZE(matrix_v_kp, 2), nblocks))
     509             : 
     510         858 :       DO ik = ikp_start, ikp_end
     511             : 
     512        2342 :          DO i_real_img = 1, SIZE(matrix_v_kp, 2)
     513             : 
     514        1484 :             i_block = 1
     515             : 
     516        1484 :             CALL dbcsr_iterator_start(iter, matrix_v_kp(ik, i_real_img)%matrix)
     517             : 
     518        7092 :             DO WHILE (dbcsr_iterator_blocks_left(iter))
     519             : 
     520        5608 :                CALL dbcsr_iterator_next_block(iter, row, col, data_block)
     521             : 
     522           0 :                ALLOCATE (blocks_v_kp(ik, i_real_img, i_block)%block(SIZE(data_block, 1), &
     523       22432 :                                                                     SIZE(data_block, 2)))
     524     2522828 :                blocks_v_kp(ik, i_real_img, i_block)%block = 0.0_dp
     525             : 
     526       12700 :                i_block = i_block + 1
     527             : 
     528             :             END DO
     529             : 
     530        3710 :             CALL dbcsr_iterator_stop(iter)
     531             : 
     532             :          END DO
     533             : 
     534             :       END DO
     535             : 
     536         116 :       CALL timestop(handle)
     537             : 
     538         116 :    END SUBROUTINE
     539             : 
     540             : ! **************************************************************************************************
     541             : !> \brief ...
     542             : !> \param cell ...
     543             : !> \param kpoints ...
     544             : !> \param size_lattice_sum ...
     545             : !> \param factor ...
     546             : !> \param hmat ...
     547             : !> \param x_min ...
     548             : !> \param x_max ...
     549             : !> \param y_min ...
     550             : !> \param y_max ...
     551             : !> \param z_min ...
     552             : !> \param z_max ...
     553             : !> \param nkp_grid ...
     554             : ! **************************************************************************************************
     555         128 :    SUBROUTINE get_factor_and_xyz_min_max(cell, kpoints, size_lattice_sum, factor, hmat, &
     556             :                                          x_min, x_max, y_min, y_max, z_min, z_max, nkp_grid)
     557             : 
     558             :       TYPE(cell_type), POINTER                           :: cell
     559             :       TYPE(kpoint_type), POINTER                         :: kpoints
     560             :       INTEGER                                            :: size_lattice_sum, factor
     561             :       REAL(KIND=dp), DIMENSION(3, 3)                     :: hmat
     562             :       INTEGER                                            :: x_min, x_max, y_min, y_max, z_min, z_max
     563             :       INTEGER, DIMENSION(3)                              :: nkp_grid
     564             : 
     565             :       CHARACTER(LEN=*), PARAMETER :: routineN = 'get_factor_and_xyz_min_max'
     566             : 
     567             :       INTEGER                                            :: handle, nkp
     568             :       INTEGER, DIMENSION(3)                              :: periodic
     569             : 
     570         128 :       CALL timeset(routineN, handle)
     571             : 
     572         128 :       CALL get_kpoint_info(kpoints, nkp_grid=nkp_grid, nkp=nkp)
     573         128 :       CALL get_cell(cell=cell, h=hmat, periodic=periodic)
     574             : 
     575         128 :       IF (periodic(1) == 0) THEN
     576          98 :          CPASSERT(nkp_grid(1) == 1)
     577             :       END IF
     578         128 :       IF (periodic(2) == 0) THEN
     579          18 :          CPASSERT(nkp_grid(2) == 1)
     580             :       END IF
     581         128 :       IF (periodic(3) == 0) THEN
     582          28 :          CPASSERT(nkp_grid(3) == 1)
     583             :       END IF
     584             : 
     585         128 :       IF (MODULO(nkp_grid(1), 2) == 1) THEN
     586          98 :          factor = 3**(size_lattice_sum - 1)
     587          30 :       ELSE IF (MODULO(nkp_grid(1), 2) == 0) THEN
     588          30 :          factor = 2**(size_lattice_sum - 1)
     589             :       END IF
     590             : 
     591         128 :       IF (MODULO(nkp_grid(1), 2) == 1) THEN
     592          98 :          x_min = -(factor*nkp_grid(1) - 1)/2
     593          98 :          x_max = (factor*nkp_grid(1) - 1)/2
     594          30 :       ELSE IF (MODULO(nkp_grid(1), 2) == 0) THEN
     595          30 :          x_min = -factor*nkp_grid(1)/2
     596          30 :          x_max = factor*nkp_grid(1)/2 - 1
     597             :       END IF
     598         128 :       IF (periodic(1) == 0) THEN
     599          98 :          x_min = 0
     600          98 :          x_max = 0
     601             :       END IF
     602             : 
     603         128 :       IF (MODULO(nkp_grid(2), 2) == 1) THEN
     604          18 :          y_min = -(factor*nkp_grid(2) - 1)/2
     605          18 :          y_max = (factor*nkp_grid(2) - 1)/2
     606         110 :       ELSE IF (MODULO(nkp_grid(2), 2) == 0) THEN
     607         110 :          y_min = -factor*nkp_grid(2)/2
     608         110 :          y_max = factor*nkp_grid(2)/2 - 1
     609             :       END IF
     610         128 :       IF (periodic(2) == 0) THEN
     611          18 :          y_min = 0
     612          18 :          y_max = 0
     613             :       END IF
     614             : 
     615         128 :       IF (MODULO(nkp_grid(3), 2) == 1) THEN
     616          28 :          z_min = -(factor*nkp_grid(3) - 1)/2
     617          28 :          z_max = (factor*nkp_grid(3) - 1)/2
     618         100 :       ELSE IF (MODULO(nkp_grid(3), 2) == 0) THEN
     619         100 :          z_min = -factor*nkp_grid(3)/2
     620         100 :          z_max = factor*nkp_grid(3)/2 - 1
     621             :       END IF
     622         128 :       IF (periodic(3) == 0) THEN
     623          28 :          z_min = 0
     624          28 :          z_max = 0
     625             :       END IF
     626             : 
     627         128 :       CALL timestop(handle)
     628             : 
     629         128 :    END SUBROUTINE
     630             : 
     631             : ! **************************************************************************************************
     632             : !> \brief ...
     633             : !> \param matrix_v_L_tmp ...
     634             : !> \param matrix_v_kp ...
     635             : !> \param ikp_start ...
     636             : ! **************************************************************************************************
     637         116 :    SUBROUTINE allocate_tmp(matrix_v_L_tmp, matrix_v_kp, ikp_start)
     638             : 
     639             :       TYPE(dbcsr_type), POINTER                          :: matrix_v_L_tmp
     640             :       TYPE(dbcsr_p_type), DIMENSION(:, :), POINTER       :: matrix_v_kp
     641             :       INTEGER                                            :: ikp_start
     642             : 
     643             :       CHARACTER(LEN=*), PARAMETER                        :: routineN = 'allocate_tmp'
     644             : 
     645             :       INTEGER                                            :: handle
     646             : 
     647         116 :       CALL timeset(routineN, handle)
     648             : 
     649         116 :       NULLIFY (matrix_v_L_tmp)
     650         116 :       CALL dbcsr_init_p(matrix_v_L_tmp)
     651             :       CALL dbcsr_create(matrix=matrix_v_L_tmp, &
     652             :                         template=matrix_v_kp(ikp_start, 1)%matrix, &
     653         116 :                         matrix_type=dbcsr_type_no_symmetry)
     654         116 :       CALL dbcsr_reserve_all_blocks(matrix_v_L_tmp)
     655         116 :       CALL dbcsr_set(matrix_v_L_tmp, 0.0_dp)
     656             : 
     657         116 :       CALL timestop(handle)
     658             : 
     659         116 :    END SUBROUTINE
     660             : 
     661             : ! **************************************************************************************************
     662             : !> \brief ...
     663             : !> \param matrix_v_L_tmp ...
     664             : ! **************************************************************************************************
     665         116 :    SUBROUTINE deallocate_tmp(matrix_v_L_tmp)
     666             : 
     667             :       TYPE(dbcsr_type), POINTER                          :: matrix_v_L_tmp
     668             : 
     669             :       CHARACTER(LEN=*), PARAMETER                        :: routineN = 'deallocate_tmp'
     670             : 
     671             :       INTEGER                                            :: handle
     672             : 
     673         116 :       CALL timeset(routineN, handle)
     674             : 
     675         116 :       CALL dbcsr_release_p(matrix_v_L_tmp)
     676             : 
     677         116 :       CALL timestop(handle)
     678             : 
     679         116 :    END SUBROUTINE
     680             : 
     681             : ! **************************************************************************************************
     682             : !> \brief ...
     683             : !> \param V_k ...
     684             : !> \param qs_env ...
     685             : !> \param kpoints ...
     686             : !> \param size_lattice_sum ...
     687             : !> \param basis_type ...
     688             : !> \param ikp_start ...
     689             : !> \param ikp_end ...
     690             : ! **************************************************************************************************
     691          12 :    SUBROUTINE build_2c_coulomb_matrix_kp_small_cell(V_k, qs_env, kpoints, size_lattice_sum, &
     692             :                                                     basis_type, ikp_start, ikp_end)
     693             :       COMPLEX(KIND=dp), DIMENSION(:, :, :)               :: V_k
     694             :       TYPE(qs_environment_type), POINTER                 :: qs_env
     695             :       TYPE(kpoint_type), POINTER                         :: kpoints
     696             :       INTEGER                                            :: size_lattice_sum
     697             :       CHARACTER(LEN=*), INTENT(IN)                       :: basis_type
     698             :       INTEGER                                            :: ikp_start, ikp_end
     699             : 
     700             :       CHARACTER(LEN=*), PARAMETER :: routineN = 'build_2c_coulomb_matrix_kp_small_cell'
     701             : 
     702             :       INTEGER :: factor, handle, handle2, i_cell, i_x, i_x_inner, i_x_outer, ik, ikp_local, j_y, &
     703             :          j_y_inner, j_y_outer, k_z, k_z_inner, k_z_outer, n_atom, n_bf, x_max, x_min, y_max, &
     704             :          y_min, z_max, z_min
     705          12 :       INTEGER, ALLOCATABLE, DIMENSION(:)                 :: bf_end_from_atom, bf_start_from_atom
     706             :       INTEGER, DIMENSION(3)                              :: nkp_grid
     707             :       REAL(KIND=dp)                                      :: coskl, sinkl
     708             :       REAL(KIND=dp), ALLOCATABLE, DIMENSION(:, :)        :: V_L
     709             :       REAL(KIND=dp), DIMENSION(3)                        :: vec_L, vec_s
     710             :       REAL(KIND=dp), DIMENSION(3, 3)                     :: hmat
     711          12 :       TYPE(atomic_kind_type), DIMENSION(:), POINTER      :: atomic_kind_set
     712             :       TYPE(cell_type), POINTER                           :: cell
     713             :       TYPE(mp_para_env_type), POINTER                    :: para_env
     714          12 :       TYPE(particle_type), DIMENSION(:), POINTER         :: particle_set
     715          12 :       TYPE(qs_kind_type), DIMENSION(:), POINTER          :: qs_kind_set
     716             : 
     717          12 :       CALL timeset(routineN, handle)
     718             : 
     719             :       CALL get_qs_env(qs_env=qs_env, &
     720             :                       para_env=para_env, &
     721             :                       particle_set=particle_set, &
     722             :                       cell=cell, &
     723             :                       qs_kind_set=qs_kind_set, &
     724          12 :                       atomic_kind_set=atomic_kind_set)
     725             : 
     726             :       CALL get_factor_and_xyz_min_max(cell, kpoints, size_lattice_sum, factor, hmat, &
     727          12 :                                       x_min, x_max, y_min, y_max, z_min, z_max, nkp_grid)
     728             : 
     729          12 :       CALL get_basis_sizes(qs_env, n_atom, basis_type, bf_start_from_atom, bf_end_from_atom, n_bf)
     730             : 
     731          48 :       ALLOCATE (V_L(n_bf, n_bf))
     732             : 
     733         404 :       DO i_x_inner = 0, 2*nkp_grid(1) - 1
     734       21268 :          DO j_y_inner = 0, 2*nkp_grid(2) - 1
     735       82696 :             DO k_z_inner = 0, 2*nkp_grid(3) - 1
     736             : 
     737     2150400 :                V_L(:, :) = 0.0_dp
     738       61440 :                i_cell = 0
     739             : 
     740      204800 :                DO i_x_outer = x_min, x_max + nkp_grid(1), 2*nkp_grid(1)
     741      675840 :                   DO j_y_outer = y_min, y_max + nkp_grid(2), 2*nkp_grid(2)
     742     1495040 :                      DO k_z_outer = z_min, z_max + nkp_grid(3), 2*nkp_grid(3)
     743             : 
     744      880640 :                         i_x = i_x_inner + i_x_outer
     745      880640 :                         j_y = j_y_inner + j_y_outer
     746      880640 :                         k_z = k_z_inner + k_z_outer
     747             : 
     748             :                         IF (i_x > x_max .OR. i_x < x_min .OR. &
     749             :                             j_y > y_max .OR. j_y < y_min .OR. &
     750      880640 :                             k_z > z_max .OR. k_z < z_min) CYCLE
     751             : 
     752      289280 :                         i_cell = i_cell + 1
     753             : 
     754     1157120 :                         vec_s = [REAL(i_x, dp), REAL(j_y, dp), REAL(k_z, dp)]
     755             : 
     756      289280 :                         IF (MODULO(i_cell, para_env%num_pe) .NE. para_env%mepos) CYCLE
     757             : 
     758     1880320 :                         vec_L = MATMUL(hmat, vec_s)
     759             : 
     760             :                         ! Compute (P 0 | Q vec_L) and add it to V_R
     761             :                         CALL add_V_L(V_L, vec_L, n_atom, bf_start_from_atom, bf_end_from_atom, &
     762     1351680 :                                      particle_set, qs_kind_set, atomic_kind_set, basis_type, cell)
     763             : 
     764             :                      END DO
     765             :                   END DO
     766             :                END DO
     767             : 
     768       61440 :                CALL para_env%sync()
     769       61440 :                CALL para_env%sum(V_L)
     770             : 
     771       61440 :                CALL timeset(routineN//"_R_to_k", handle2)
     772             : 
     773       61440 :                ikp_local = 0
     774             : 
     775             :                ! add exp(iq*vec_L) * (P 0 | Q vec_L) to V_PQ(q)
     776    33091584 :                DO ik = 1, ikp_end
     777             : 
     778    33030144 :                   IF (MODULO(ik, para_env%num_pe) .NE. para_env%mepos) CYCLE
     779             : 
     780    16515072 :                   ikp_local = ikp_local + 1
     781             : 
     782    16515072 :                   IF (ik < ikp_start) CYCLE
     783             : 
     784             :                   ! coskl and sinkl are identical for all i_x_outer, j_y_outer, k_z_outer
     785    53477376 :                   coskl = COS(twopi*DOT_PRODUCT(vec_s(1:3), kpoints%xkp(1:3, ik)))
     786    53477376 :                   sinkl = SIN(twopi*DOT_PRODUCT(vec_s(1:3), kpoints%xkp(1:3, ik)))
     787             : 
     788             :                   V_k(:, :, ikp_local) = V_k(:, :, ikp_local) + z_one*coskl*V_L(:, :) + &
     789   471134208 :                                          gaussi*sinkl*V_L(:, :)
     790             : 
     791             :                END DO
     792             : 
     793      143744 :                CALL timestop(handle2)
     794             : 
     795             :             END DO
     796             :          END DO
     797             :       END DO
     798             : 
     799          12 :       CALL timestop(handle)
     800             : 
     801          24 :    END SUBROUTINE build_2c_coulomb_matrix_kp_small_cell
     802             : 
     803             : ! **************************************************************************************************
     804             : !> \brief ...
     805             : !> \param qs_env ...
     806             : !> \param n_atom ...
     807             : !> \param basis_type ...
     808             : !> \param bf_start_from_atom ...
     809             : !> \param bf_end_from_atom ...
     810             : !> \param n_bf ...
     811             : ! **************************************************************************************************
     812          12 :    SUBROUTINE get_basis_sizes(qs_env, n_atom, basis_type, bf_start_from_atom, bf_end_from_atom, n_bf)
     813             : 
     814             :       TYPE(qs_environment_type), POINTER                 :: qs_env
     815             :       INTEGER                                            :: n_atom
     816             :       CHARACTER(LEN=*), INTENT(IN)                       :: basis_type
     817             :       INTEGER, ALLOCATABLE, DIMENSION(:)                 :: bf_start_from_atom, bf_end_from_atom
     818             :       INTEGER                                            :: n_bf
     819             : 
     820             :       CHARACTER(LEN=*), PARAMETER                        :: routineN = 'get_basis_sizes'
     821             : 
     822             :       INTEGER                                            :: handle, iatom, ikind, n_kind, nsgf
     823          12 :       INTEGER, ALLOCATABLE, DIMENSION(:)                 :: kind_of
     824          12 :       TYPE(atomic_kind_type), DIMENSION(:), POINTER      :: atomic_kind_set
     825             :       TYPE(gto_basis_set_type), POINTER                  :: basis_set_a
     826          12 :       TYPE(particle_type), DIMENSION(:), POINTER         :: particle_set
     827          12 :       TYPE(qs_kind_type), DIMENSION(:), POINTER          :: qs_kind_set
     828             : 
     829          12 :       CALL timeset(routineN, handle)
     830             : 
     831             :       CALL get_qs_env(qs_env=qs_env, particle_set=particle_set, &
     832          12 :                       qs_kind_set=qs_kind_set, atomic_kind_set=atomic_kind_set)
     833          12 :       CALL get_atomic_kind_set(atomic_kind_set, kind_of=kind_of)
     834             : 
     835          12 :       n_atom = SIZE(particle_set)
     836          12 :       n_kind = SIZE(qs_kind_set)
     837             : 
     838          36 :       DO ikind = 1, n_kind
     839             :          CALL get_qs_kind(qs_kind=qs_kind_set(ikind), basis_set=basis_set_a, &
     840          24 :                           basis_type=basis_type)
     841          36 :          CPASSERT(ASSOCIATED(basis_set_a))
     842             :       END DO
     843             : 
     844          60 :       ALLOCATE (bf_start_from_atom(n_atom), bf_end_from_atom(n_atom))
     845             : 
     846          12 :       n_bf = 0
     847          40 :       DO iatom = 1, n_atom
     848          28 :          bf_start_from_atom(iatom) = n_bf + 1
     849          28 :          ikind = kind_of(iatom)
     850          28 :          CALL get_qs_kind(qs_kind=qs_kind_set(ikind), nsgf=nsgf, basis_type=basis_type)
     851          28 :          n_bf = n_bf + nsgf
     852          40 :          bf_end_from_atom(iatom) = n_bf
     853             :       END DO
     854             : 
     855          12 :       CALL timestop(handle)
     856             : 
     857          24 :    END SUBROUTINE get_basis_sizes
     858             : 
     859             : ! **************************************************************************************************
     860             : !> \brief ...
     861             : !> \param V_L ...
     862             : !> \param vec_L ...
     863             : !> \param n_atom ...
     864             : !> \param bf_start_from_atom ...
     865             : !> \param bf_end_from_atom ...
     866             : !> \param particle_set ...
     867             : !> \param qs_kind_set ...
     868             : !> \param atomic_kind_set ...
     869             : !> \param basis_type ...
     870             : !> \param cell ...
     871             : ! **************************************************************************************************
     872      144640 :    SUBROUTINE add_V_L(V_L, vec_L, n_atom, bf_start_from_atom, bf_end_from_atom, &
     873             :                       particle_set, qs_kind_set, atomic_kind_set, basis_type, cell)
     874             : 
     875             :       REAL(KIND=dp), ALLOCATABLE, DIMENSION(:, :)        :: V_L
     876             :       REAL(KIND=dp), DIMENSION(3)                        :: vec_L
     877             :       INTEGER                                            :: n_atom
     878             :       INTEGER, ALLOCATABLE, DIMENSION(:)                 :: bf_start_from_atom, bf_end_from_atom
     879             :       TYPE(particle_type), DIMENSION(:), POINTER         :: particle_set
     880             :       TYPE(qs_kind_type), DIMENSION(:), POINTER          :: qs_kind_set
     881             :       TYPE(atomic_kind_type), DIMENSION(:), POINTER      :: atomic_kind_set
     882             :       CHARACTER(LEN=*), INTENT(IN)                       :: basis_type
     883             :       TYPE(cell_type), POINTER                           :: cell
     884             : 
     885             :       CHARACTER(LEN=*), PARAMETER                        :: routineN = 'add_V_L'
     886             : 
     887             :       INTEGER                                            :: a_1, a_2, atom_a, atom_b, b_1, b_2, &
     888             :                                                             handle, kind_a, kind_b
     889      144640 :       INTEGER, ALLOCATABLE, DIMENSION(:)                 :: kind_of
     890             :       REAL(dp), DIMENSION(3)                             :: ra, rab_L, rb
     891      144640 :       REAL(KIND=dp), DIMENSION(:, :), POINTER            :: V_L_ab
     892      144640 :       REAL(KIND=dp), DIMENSION(:, :, :), POINTER         :: contr_a, contr_b
     893             :       TYPE(gto_basis_set_type), POINTER                  :: basis_set_a, basis_set_b
     894             : 
     895      144640 :       CALL timeset(routineN, handle)
     896             : 
     897      144640 :       NULLIFY (basis_set_a, basis_set_b)
     898             : 
     899      144640 :       CALL get_atomic_kind_set(atomic_kind_set, kind_of=kind_of)
     900             : 
     901      537600 :       DO atom_a = 1, n_atom
     902             : 
     903     1634560 :          DO atom_b = 1, n_atom
     904             : 
     905     1096960 :             kind_a = kind_of(atom_a)
     906     1096960 :             kind_b = kind_of(atom_b)
     907             : 
     908             :             CALL get_qs_kind(qs_kind=qs_kind_set(kind_a), basis_set=basis_set_a, &
     909     1096960 :                              basis_type=basis_type)
     910             :             CALL get_qs_kind(qs_kind=qs_kind_set(kind_b), basis_set=basis_set_b, &
     911     1096960 :                              basis_type=basis_type)
     912             : 
     913     1096960 :             ra(1:3) = pbc(particle_set(atom_a)%r(1:3), cell)
     914     1096960 :             rb(1:3) = pbc(particle_set(atom_b)%r(1:3), cell)
     915             : 
     916     4387840 :             rab_L(1:3) = rb(1:3) - ra(1:3) + vec_L(1:3)
     917             : 
     918     1096960 :             CALL contraction_matrix_shg(basis_set_a, contr_a)
     919     1096960 :             CALL contraction_matrix_shg(basis_set_b, contr_b)
     920             : 
     921     1096960 :             a_1 = bf_start_from_atom(atom_a)
     922     1096960 :             a_2 = bf_end_from_atom(atom_a)
     923     1096960 :             b_1 = bf_start_from_atom(atom_b)
     924     1096960 :             b_2 = bf_end_from_atom(atom_b)
     925             : 
     926     4387840 :             ALLOCATE (V_L_ab(a_2 - a_1 + 1, b_2 - b_1 + 1))
     927             : 
     928             :             CALL int_operators_r12_ab_shg(operator_coulomb, V_L_ab, rab=rab_L, &
     929             :                                           fba=basis_set_a, fbb=basis_set_b, &
     930             :                                           scona_shg=contr_a, sconb_shg=contr_b, &
     931     1096960 :                                           calculate_forces=.FALSE.)
     932             : 
     933     8129280 :             V_L(a_1:a_2, b_1:b_2) = V_L(a_1:a_2, b_1:b_2) + V_L_ab(:, :)
     934             : 
     935     1489920 :             DEALLOCATE (contr_a, contr_b, V_L_ab)
     936             : 
     937             :          END DO
     938             : 
     939             :       END DO
     940             : 
     941      144640 :       DEALLOCATE (kind_of)
     942             : 
     943      144640 :       CALL timestop(handle)
     944             : 
     945      144640 :    END SUBROUTINE add_V_L
     946             : 
     947           0 : END MODULE kpoint_coulomb_2c

Generated by: LCOV version 1.15