LCOV - code coverage report
Current view: top level - src - qs_active_space_methods.F (source / functions) Hit Total Coverage
Test: CP2K Regtests (git:4dc10b3) Lines: 1085 1492 72.7 %
Date: 2024-11-21 06:45:46 Functions: 16 26 61.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 Determine active space Hamiltonian
      10             : !> \par History
      11             : !>      04.2016 created [JGH]
      12             : !> \author JGH
      13             : ! **************************************************************************************************
      14             : MODULE qs_active_space_methods
      15             :    USE atomic_kind_types, ONLY: atomic_kind_type
      16             :    USE basis_set_types, ONLY: allocate_sto_basis_set, &
      17             :                               create_gto_from_sto_basis, &
      18             :                               deallocate_sto_basis_set, &
      19             :                               gto_basis_set_type, &
      20             :                               init_orb_basis_set, &
      21             :                               set_sto_basis_set, &
      22             :                               srules, &
      23             :                               sto_basis_set_type
      24             :    USE cell_types, ONLY: cell_type
      25             :    USE cp_blacs_env, ONLY: cp_blacs_env_type, cp_blacs_env_create, cp_blacs_env_release, BLACS_GRID_SQUARE
      26             :    USE cp_control_types, ONLY: dft_control_type, qs_control_type
      27             :    USE cp_dbcsr_operations, ONLY: cp_dbcsr_plus_fm_fm_t, &
      28             :                                   cp_dbcsr_sm_fm_multiply, &
      29             :                                   dbcsr_allocate_matrix_set, &
      30             :                                   cp_dbcsr_m_by_n_from_template, copy_dbcsr_to_fm
      31             :    USE cp_files, ONLY: close_file, &
      32             :                        file_exists, &
      33             :                        open_file
      34             :    USE cp_fm_struct, ONLY: cp_fm_struct_create, &
      35             :                            cp_fm_struct_release, &
      36             :                            cp_fm_struct_type
      37             :    USE cp_fm_types, ONLY: &
      38             :       cp_fm_create, cp_fm_get_element, cp_fm_get_info, cp_fm_init_random, cp_fm_release, &
      39             :       cp_fm_set_all, cp_fm_set_element, cp_fm_to_fm, cp_fm_type
      40             :    USE cp_log_handling, ONLY: cp_get_default_logger, &
      41             :                               cp_logger_get_default_io_unit, &
      42             :                               cp_logger_type
      43             :    USE cp_output_handling, ONLY: &
      44             :       cp_p_file, cp_print_key_finished_output, cp_print_key_should_output, cp_print_key_unit_nr, &
      45             :       debug_print_level, high_print_level, low_print_level, medium_print_level, &
      46             :       silent_print_level
      47             :    USE cp_realspace_grid_cube, ONLY: cp_pw_to_cube
      48             :    USE cp_dbcsr_api, ONLY: &
      49             :       dbcsr_copy, dbcsr_csr_create, dbcsr_csr_type, dbcsr_p_type, dbcsr_type, dbcsr_release, &
      50             :       dbcsr_type_no_symmetry, dbcsr_create, dbcsr_set, dbcsr_multiply, dbcsr_iterator_next_block, &
      51             :       dbcsr_iterator_start, dbcsr_iterator_stop, dbcsr_iterator_blocks_left, &
      52             :       dbcsr_iterator_type, dbcsr_type_symmetric, dbcsr_get_occupation, dbcsr_get_info
      53             :    USE group_dist_types, ONLY: get_group_dist, release_group_dist, group_dist_d1_type
      54             :    USE input_constants, ONLY: &
      55             :       casci_canonical, dmft_model, eri_method_full_gpw, eri_method_gpw_ht, eri_operator_coulomb, &
      56             :       eri_operator_trunc, eri_operator_erf, eri_operator_erfc, eri_operator_gaussian, eri_operator_yukawa, hf_model, &
      57             :       manual_selection, mao_projection, no_solver, qiskit_solver, rsdft_model, wannier_projection
      58             :    USE input_section_types, ONLY: section_vals_get, &
      59             :                                   section_vals_get_subs_vals, &
      60             :                                   section_vals_type, &
      61             :                                   section_vals_val_get
      62             :    USE ISO_C_BINDING, ONLY: c_null_char
      63             :    USE kinds, ONLY: default_path_length, &
      64             :                     default_string_length, &
      65             :                     dp, &
      66             :                     int_8
      67             :    USE machine, ONLY: m_walltime
      68             :    USE mathconstants, ONLY: fourpi
      69             :    USE memory_utilities, ONLY: reallocate
      70             :    USE message_passing, ONLY: mp_comm_type, &
      71             :                               mp_para_env_type, &
      72             :                               mp_para_env_release
      73             :    USE mp2_gpw, ONLY: create_mat_munu, grep_rows_in_subgroups, build_dbcsr_from_rows
      74             :    USE parallel_gemm_api, ONLY: parallel_gemm
      75             :    USE particle_list_types, ONLY: particle_list_type
      76             :    USE particle_types, ONLY: particle_type
      77             :    USE periodic_table, ONLY: ptable
      78             :    USE preconditioner_types, ONLY: preconditioner_type
      79             :    USE pw_env_methods, ONLY: pw_env_create, &
      80             :                              pw_env_rebuild
      81             :    USE pw_env_types, ONLY: pw_env_get, &
      82             :                            pw_env_release, &
      83             :                            pw_env_type
      84             :    USE pw_methods, ONLY: pw_integrate_function, &
      85             :                          pw_multiply, &
      86             :                          pw_multiply_with, &
      87             :                          pw_transfer, &
      88             :                          pw_zero, pw_integral_ab, pw_scale
      89             :    USE pw_poisson_methods, ONLY: pw_poisson_rebuild, &
      90             :                                  pw_poisson_solve
      91             :    USE pw_poisson_types, ONLY: ANALYTIC0D, &
      92             :                                PERIODIC3D, &
      93             :                                greens_fn_type, &
      94             :                                pw_poisson_analytic, &
      95             :                                pw_poisson_periodic, &
      96             :                                pw_poisson_type
      97             :    USE pw_pool_types, ONLY: &
      98             :       pw_pool_type
      99             :    USE pw_types, ONLY: &
     100             :       pw_c1d_gs_type, &
     101             :       pw_r3d_rs_type
     102             :    USE qcschema, ONLY: qcschema_env_create, &
     103             :                        qcschema_env_release, &
     104             :                        qcschema_to_hdf5, &
     105             :                        qcschema_type
     106             :    USE qs_active_space_types, ONLY: active_space_type, &
     107             :                                     create_active_space_type, &
     108             :                                     csr_idx_from_combined, &
     109             :                                     csr_idx_to_combined, &
     110             :                                     eri_type, &
     111             :                                     eri_type_eri_element_func, &
     112             :                                     get_irange_csr
     113             :    USE qs_active_space_utils, ONLY: eri_to_array, &
     114             :                                     subspace_matrix_to_array
     115             :    USE qs_collocate_density, ONLY: calculate_wavefunction
     116             :    USE qs_density_matrices, ONLY: calculate_density_matrix
     117             :    USE qs_energy_types, ONLY: qs_energy_type
     118             :    USE qs_environment_types, ONLY: get_qs_env, &
     119             :                                    qs_environment_type, &
     120             :                                    set_qs_env
     121             :    USE qs_integrate_potential, ONLY: integrate_v_rspace
     122             :    USE qs_kind_types, ONLY: qs_kind_type
     123             :    USE qs_ks_methods, ONLY: qs_ks_update_qs_env
     124             :    USE qs_ks_types, ONLY: qs_ks_did_change, &
     125             :                           qs_ks_env_type
     126             :    USE qs_mo_io, ONLY: write_mo_set_to_output_unit
     127             :    USE qs_mo_methods, ONLY: calculate_subspace_eigenvalues
     128             :    USE qs_mo_types, ONLY: allocate_mo_set, &
     129             :                           get_mo_set, &
     130             :                           init_mo_set, &
     131             :                           mo_set_type
     132             :    USE qs_neighbor_list_types, ONLY: neighbor_list_set_p_type, release_neighbor_list_sets
     133             :    USE qs_ot_eigensolver, ONLY: ot_eigensolver
     134             :    USE qs_rho_methods, ONLY: qs_rho_update_rho
     135             :    USE qs_rho_types, ONLY: qs_rho_get, &
     136             :                            qs_rho_type
     137             :    USE qs_subsys_types, ONLY: qs_subsys_get, &
     138             :                               qs_subsys_type
     139             :    USE scf_control_types, ONLY: scf_control_type
     140             : #ifndef __NO_SOCKETS
     141             :    USE sockets_interface, ONLY: accept_socket, &
     142             :                                 close_socket, &
     143             :                                 listen_socket, &
     144             :                                 open_bind_socket, &
     145             :                                 readbuffer, &
     146             :                                 remove_socket_file, &
     147             :                                 writebuffer
     148             : #endif
     149             :    USE task_list_methods, ONLY: generate_qs_task_list
     150             :    USE task_list_types, ONLY: allocate_task_list, &
     151             :                               deallocate_task_list, &
     152             :                               task_list_type
     153             :    USE util, ONLY: get_limit
     154             : #include "./base/base_uses.f90"
     155             : 
     156             :    IMPLICIT NONE
     157             :    PRIVATE
     158             : 
     159             :    CHARACTER(len=*), PARAMETER, PRIVATE :: moduleN = 'qs_active_space_methods'
     160             : 
     161             :    PUBLIC :: active_space_main
     162             : 
     163             :    TYPE, EXTENDS(eri_type_eri_element_func) :: eri_fcidump_print
     164             :       INTEGER :: unit_nr = -1, bra_start = -1, ket_start = -1
     165             :    CONTAINS
     166             :       PROCEDURE :: func => eri_fcidump_print_func
     167             :    END TYPE
     168             : 
     169             :    TYPE, EXTENDS(eri_type_eri_element_func) :: eri_fcidump_checksum
     170             :       INTEGER :: bra_start = 0, ket_start = 0
     171             :       REAL(KIND=dp) :: checksum = 0.0_dp
     172             :    CONTAINS
     173             :       PROCEDURE, PASS :: set => eri_fcidump_set
     174             :       PROCEDURE :: func => eri_fcidump_checksum_func
     175             :    END TYPE eri_fcidump_checksum
     176             : 
     177             : CONTAINS
     178             : 
     179             : ! **************************************************************************************************
     180             : !> \brief Sets the starting indices of the bra and ket.
     181             : !> \param this object reference
     182             : !> \param bra_start starting index of the bra
     183             : !> \param ket_start starting index of the ket
     184             : ! **************************************************************************************************
     185         142 :    SUBROUTINE eri_fcidump_set(this, bra_start, ket_start)
     186             :       CLASS(eri_fcidump_checksum) :: this
     187             :       INTEGER, INTENT(IN) :: bra_start, ket_start
     188         142 :       this%bra_start = bra_start
     189         142 :       this%ket_start = ket_start
     190         142 :    END SUBROUTINE eri_fcidump_set
     191             : 
     192             : ! **************************************************************************************************
     193             : !> \brief Main method for determining the active space Hamiltonian
     194             : !> \param qs_env ...
     195             : ! **************************************************************************************************
     196       19703 :    SUBROUTINE active_space_main(qs_env)
     197             :       TYPE(qs_environment_type), POINTER                 :: qs_env
     198             : 
     199             :       CHARACTER(len=*), PARAMETER                        :: routineN = 'active_space_main'
     200             : 
     201             :       CHARACTER(len=10)                                  :: cshell, lnam(5)
     202             :       CHARACTER(len=default_path_length)                 :: qcschema_filename
     203             :       INTEGER :: as_solver, eri_method, eri_operator, eri_print, group_size, handle, i, iatom, &
     204             :          ishell, isp, ispin, iw, j, jm, m, max_orb_ind, mselect, n1, n2, nao, natom, nel, &
     205             :          nelec_active, nelec_inactive, nelec_total, nmo, nmo_active, nmo_available, nmo_inactive, &
     206             :          nmo_inactive_remaining, nmo_occ, nmo_virtual, nn1, nn2, nrow_global, nspins
     207             :       INTEGER, DIMENSION(5)                              :: nshell
     208       19703 :       INTEGER, DIMENSION(:), POINTER                     :: invals
     209             :       LOGICAL                                            :: do_kpoints, ex_operator, ex_perd, &
     210             :                                                             explicit, isolated, stop_after_print, &
     211             :                                                             store_wfn
     212             :       REAL(KIND=dp) :: eri_eps_filter, eri_eps_grid, eri_eps_int, eri_gpw_cutoff, eri_op_param, &
     213             :          eri_rcut, eri_rel_cutoff, fel, focc, maxocc, nze_percentage
     214       19703 :       REAL(KIND=dp), ALLOCATABLE, DIMENSION(:, :)        :: eigenvalues
     215       19703 :       REAL(KIND=dp), DIMENSION(:), POINTER               :: evals_virtual
     216             :       TYPE(active_space_type), POINTER                   :: active_space_env
     217             :       TYPE(cell_type), POINTER                           :: cell
     218             :       TYPE(cp_blacs_env_type), POINTER                   :: context
     219             :       TYPE(cp_fm_struct_type), POINTER                   :: fm_struct_tmp
     220             :       TYPE(cp_fm_type)                                   :: fm_dummy, mo_virtual
     221             :       TYPE(cp_fm_type), POINTER                          :: fm_target_active, fm_target_inactive, &
     222             :                                                             fmat, mo_coeff, mo_ref, mo_target
     223             :       TYPE(cp_logger_type), POINTER                      :: logger
     224             :       TYPE(dbcsr_csr_type), POINTER                      :: eri_mat
     225       39406 :       TYPE(dbcsr_p_type), DIMENSION(:), POINTER          :: ks_matrix, rho_ao, s_matrix
     226             :       TYPE(dbcsr_type), POINTER                          :: denmat
     227             :       TYPE(dft_control_type), POINTER                    :: dft_control
     228       19703 :       TYPE(mo_set_type), DIMENSION(:), POINTER           :: mos
     229             :       TYPE(mo_set_type), POINTER                         :: mo_set, mo_set_active, mo_set_inactive
     230             :       TYPE(mp_para_env_type), POINTER                    :: para_env
     231             :       TYPE(preconditioner_type), POINTER                 :: local_preconditioner
     232       78812 :       TYPE(qcschema_type)                                :: qcschema_env
     233             :       TYPE(qs_energy_type), POINTER                      :: energy
     234             :       TYPE(qs_rho_type), POINTER                         :: rho
     235             :       TYPE(scf_control_type), POINTER                    :: scf_control
     236             :       TYPE(section_vals_type), POINTER                   :: as_input, input, loc_print, loc_section, &
     237             :                                                             print_orb
     238             : 
     239             :       !--------------------------------------------------------------------------------------------!
     240             : 
     241       19703 :       CALL get_qs_env(qs_env, input=input)
     242       19703 :       as_input => section_vals_get_subs_vals(input, "DFT%ACTIVE_SPACE")
     243       19703 :       CALL section_vals_get(as_input, explicit=explicit)
     244       19703 :       IF (.NOT. explicit) RETURN
     245         106 :       CALL timeset(routineN, handle)
     246             : 
     247         106 :       logger => cp_get_default_logger()
     248         106 :       iw = cp_logger_get_default_io_unit(logger)
     249             : 
     250         106 :       IF (iw > 0) THEN
     251             :          WRITE (iw, '(/,T2,A)') &
     252          53 :             '!-----------------------------------------------------------------------------!'
     253          53 :          WRITE (iw, '(T26,A)') "Generate Active Space Hamiltonian"
     254          53 :          WRITE (iw, '(T27,A)') "Interface to CAS-CI and DMRG-CI"
     255             :          WRITE (iw, '(T2,A)') &
     256          53 :             '!-----------------------------------------------------------------------------!'
     257             :       END IF
     258             : 
     259             :       ! k-points?
     260         106 :       CALL get_qs_env(qs_env, do_kpoints=do_kpoints)
     261         106 :       IF (do_kpoints) THEN
     262           0 :          CALL cp_abort(__LOCATION__, "K-points not allowd for Active Space Interface")
     263             :       END IF
     264             : 
     265         106 :       NULLIFY (active_space_env)
     266         106 :       CALL create_active_space_type(active_space_env)
     267         106 :       active_space_env%energy_total = 0.0_dp
     268         106 :       active_space_env%energy_ref = 0.0_dp
     269         106 :       active_space_env%energy_inactive = 0.0_dp
     270         106 :       active_space_env%energy_active = 0.0_dp
     271             : 
     272             :       ! input options
     273             : 
     274             :       ! figure out what needs to be printed/stored
     275         106 :       IF (BTEST(cp_print_key_should_output(logger%iter_info, as_input, "FCIDUMP"), cp_p_file)) THEN
     276         106 :          active_space_env%fcidump = .TRUE.
     277             :       END IF
     278             : 
     279         106 :       CALL section_vals_val_get(as_input, "QCSCHEMA", c_val=qcschema_filename, explicit=explicit)
     280         106 :       IF (explicit) THEN
     281           0 :          active_space_env%qcschema = .TRUE.
     282           0 :          active_space_env%qcschema_filename = qcschema_filename
     283             :       END IF
     284             : 
     285             :       ! model
     286         106 :       CALL section_vals_val_get(as_input, "MODEL", i_val=active_space_env%model)
     287         106 :       IF (iw > 0) THEN
     288         106 :          SELECT CASE (active_space_env%model)
     289             :          CASE (hf_model)
     290          53 :             WRITE (iw, '(T3,A)') "Hartree-Fock model for interaction Hamiltonian"
     291             :          CASE (rsdft_model)
     292           0 :             WRITE (iw, '(T3,A)') "Range-separated DFT model for interaction Hamiltonian"
     293             :          CASE (dmft_model)
     294           0 :             WRITE (iw, '(T3,A)') "DMFT model for interaction Hamiltonian"
     295             :          CASE DEFAULT
     296          53 :             CPABORT("Unknown Model")
     297             :          END SELECT
     298             :       END IF
     299             : 
     300             :       ! isolated (molecular) system?
     301         106 :       CALL section_vals_val_get(as_input, "ISOLATED_SYSTEM", l_val=isolated)
     302         106 :       active_space_env%molecule = isolated
     303         106 :       IF (iw > 0) THEN
     304          53 :          IF (active_space_env%molecule) THEN
     305          51 :             WRITE (iw, '(T3,A)') "System is treated without periodicity"
     306             :          END IF
     307             :       END IF
     308             : 
     309         106 :       CALL section_vals_val_get(as_input, "ACTIVE_ELECTRONS", i_val=nelec_active)
     310             :       ! actually nelec_spin tells me the number of electrons per spin channel from qs_env
     311             :       ! CALL get_qs_env(qs_env, nelectron_total=nelec_total, nelectron_spin=nelec_spin)
     312         106 :       CALL get_qs_env(qs_env, nelectron_total=nelec_total)
     313             : 
     314         106 :       IF (nelec_active <= 0) CPABORT("Specify a positive number of active electrons.")
     315         106 :       IF (nelec_active > nelec_total) CPABORT("More active electrons than total electrons.")
     316             : 
     317         106 :       nelec_inactive = nelec_total - nelec_active
     318         106 :       IF (MOD(nelec_inactive, 2) /= 0) THEN
     319           0 :          CPABORT("The remaining number of inactive electrons has to be even.")
     320             :       END IF
     321             : 
     322         106 :       CALL get_qs_env(qs_env, dft_control=dft_control)
     323         106 :       nspins = dft_control%nspins
     324         106 :       IF (iw > 0) THEN
     325          53 :          WRITE (iw, '(T3,A,T70,I10)') "Total number of electrons", nelec_total
     326          53 :          WRITE (iw, '(T3,A,T70,I10)') "Number of inactive electrons", nelec_inactive
     327          53 :          WRITE (iw, '(T3,A,T70,I10)') "Number of active electrons", nelec_active
     328             :       END IF
     329             : 
     330         106 :       active_space_env%nelec_active = nelec_active
     331         106 :       active_space_env%nelec_inactive = nelec_inactive
     332         106 :       active_space_env%nelec_total = nelec_total
     333         106 :       active_space_env%nspins = nspins
     334         106 :       active_space_env%multiplicity = dft_control%multiplicity
     335             : 
     336             :       ! define the active/inactive space orbitals
     337         106 :       CALL section_vals_val_get(as_input, "ACTIVE_ORBITALS", explicit=explicit, i_val=nmo_active)
     338         106 :       IF (.NOT. explicit) THEN
     339           0 :          CALL cp_abort(__LOCATION__, "Number of Active Orbitals has to be specified.")
     340             :       END IF
     341         106 :       active_space_env%nmo_active = nmo_active
     342             :       ! this is safe because nelec_inactive is always even
     343         106 :       nmo_inactive = nelec_inactive/2
     344         106 :       active_space_env%nmo_inactive = nmo_inactive
     345             : 
     346         106 :       CALL section_vals_val_get(as_input, "ORBITAL_SELECTION", i_val=mselect)
     347         106 :       IF (iw > 0) THEN
     348           0 :          SELECT CASE (mselect)
     349             :          CASE DEFAULT
     350           0 :             CPABORT("Unknown orbital selection method")
     351             :          CASE (casci_canonical)
     352             :             WRITE (iw, '(/,T3,A)') &
     353          47 :                "Active space orbitals selected using energy ordered canonical orbitals"
     354             :          CASE (wannier_projection)
     355             :             WRITE (iw, '(/,T3,A)') &
     356           0 :                "Active space orbitals selected using projected Wannier orbitals"
     357             :          CASE (mao_projection)
     358             :             WRITE (iw, '(/,T3,A)') &
     359           0 :                "Active space orbitals selected using modified atomic orbitals (MAO)"
     360             :          CASE (manual_selection)
     361             :             WRITE (iw, '(/,T3,A)') &
     362          53 :                "Active space orbitals selected manually"
     363             :          END SELECT
     364             : 
     365          53 :          WRITE (iw, '(T3,A,T70,I10)') "Number of inactive orbitals", nmo_inactive
     366          53 :          WRITE (iw, '(T3,A,T70,I10)') "Number of active orbitals", nmo_active
     367             :       END IF
     368             : 
     369             :       ! get projection spaces
     370         106 :       CALL section_vals_val_get(as_input, "SUBSPACE_ATOM", i_val=iatom, explicit=explicit)
     371         106 :       IF (explicit) THEN
     372           0 :          CALL get_qs_env(qs_env, natom=natom)
     373           0 :          IF (iatom <= 0 .OR. iatom > natom) THEN
     374           0 :             IF (iw > 0) THEN
     375           0 :                WRITE (iw, '(/,T3,A,I3)') "ERROR: SUBSPACE_ATOM number is not valid", iatom
     376             :             END IF
     377           0 :             CPABORT("Select a valid SUBSPACE_ATOM")
     378             :          END IF
     379             :       END IF
     380         106 :       CALL section_vals_val_get(as_input, "SUBSPACE_SHELL", c_val=cshell, explicit=explicit)
     381         106 :       nshell = 0
     382         636 :       lnam = ""
     383         106 :       IF (explicit) THEN
     384           0 :          cshell = ADJUSTL(cshell)
     385           0 :          n1 = 1
     386           0 :          DO i = 1, 5
     387           0 :             ishell = i
     388           0 :             IF (cshell(n1:n1) == " ") THEN
     389         106 :                ishell = ishell - 1
     390             :                EXIT
     391             :             END IF
     392           0 :             READ (cshell(n1:), "(I1,A1)") nshell(i), lnam(i)
     393           0 :             n1 = n1 + 2
     394             :          END DO
     395             :       END IF
     396             : 
     397             :       ! generate orbitals
     398           0 :       SELECT CASE (mselect)
     399             :       CASE DEFAULT
     400           0 :          CPABORT("Unknown orbital selection method")
     401             :       CASE (casci_canonical)
     402          94 :          CALL get_qs_env(qs_env, mos=mos)
     403             : 
     404             :          ! total number of occupied orbitals, i.e. inactive plus active MOs
     405          94 :          nmo_occ = nmo_inactive + nmo_active
     406             : 
     407             :          ! set inactive orbital indices, these are trivially 1...nmo_inactive
     408         300 :          ALLOCATE (active_space_env%inactive_orbitals(nmo_inactive, nspins))
     409         200 :          DO ispin = 1, nspins
     410         276 :             DO i = 1, nmo_inactive
     411         182 :                active_space_env%inactive_orbitals(i, ispin) = i
     412             :             END DO
     413             :          END DO
     414             : 
     415             :          ! set active orbital indices, these are shifted by nmo_inactive
     416         376 :          ALLOCATE (active_space_env%active_orbitals(nmo_active, nspins))
     417         200 :          DO ispin = 1, nspins
     418         482 :             DO i = 1, nmo_active
     419         388 :                active_space_env%active_orbitals(i, ispin) = nmo_inactive + i
     420             :             END DO
     421             :          END DO
     422             : 
     423             :          ! allocate and initialize inactive and active mo coefficients.
     424             :          ! These are stored in a data structure for the full occupied space:
     425             :          ! for inactive mos, the active subset is set to zero, vice versa for the active mos
     426             :          ! TODO: allocate data structures only for the eaxct number MOs
     427          94 :          maxocc = 2.0_dp
     428          94 :          IF (nspins > 1) maxocc = 1.0_dp
     429         388 :          ALLOCATE (active_space_env%mos_active(nspins))
     430         294 :          ALLOCATE (active_space_env%mos_inactive(nspins))
     431         200 :          DO ispin = 1, nspins
     432         106 :             CALL get_mo_set(mos(ispin), mo_coeff=mo_ref, nao=nao)
     433         106 :             CALL cp_fm_get_info(mo_ref, context=context, para_env=para_env, nrow_global=nrow_global)
     434             :             ! the right number of active electrons per spin channel is initialized further down
     435         106 :             CALL allocate_mo_set(active_space_env%mos_active(ispin), nao, nmo_occ, 0, 0.0_dp, maxocc, 0.0_dp)
     436             :             CALL cp_fm_struct_create(fm_struct_tmp, para_env=para_env, context=context, &
     437         106 :                                      nrow_global=nrow_global, ncol_global=nmo_occ)
     438         106 :             CALL init_mo_set(active_space_env%mos_active(ispin), fm_struct=fm_struct_tmp, name="Active Space MO")
     439         106 :             CALL cp_fm_struct_release(fm_struct_tmp)
     440         106 :             IF (nspins == 2) THEN
     441          24 :                nel = nelec_inactive/2
     442             :             ELSE
     443          82 :                nel = nelec_inactive
     444             :             END IF
     445             :             CALL allocate_mo_set(active_space_env%mos_inactive(ispin), nao, nmo_occ, nel, &
     446         106 :                                  REAL(nel, KIND=dp), maxocc, 0.0_dp)
     447             :             CALL cp_fm_struct_create(fm_struct_tmp, para_env=para_env, context=context, &
     448         106 :                                      nrow_global=nrow_global, ncol_global=nmo_occ)
     449         106 :             CALL init_mo_set(active_space_env%mos_inactive(ispin), fm_struct=fm_struct_tmp, name="Inactive Space MO")
     450         306 :             CALL cp_fm_struct_release(fm_struct_tmp)
     451             :          END DO
     452             : 
     453             :          ! create canonical orbitals
     454          94 :          IF (dft_control%restricted) THEN
     455           0 :             CPABORT("Unclear how we define MOs in the restricted case ... stopping")
     456             :          ELSE
     457          94 :             IF (dft_control%do_admm) THEN
     458           0 :                CPABORT("ADMM currently not possible for canonical orbital options")
     459             :             END IF
     460             : 
     461         376 :             ALLOCATE (eigenvalues(nmo_occ, nspins))
     462         558 :             eigenvalues = 0.0_dp
     463          94 :             CALL get_qs_env(qs_env, matrix_ks=ks_matrix, matrix_s=s_matrix, scf_control=scf_control)
     464             : 
     465             :             ! calculate virtual MOs and copy inactive and active orbitals
     466          94 :             IF (iw > 0) THEN
     467          47 :                WRITE (iw, '(/,T3,A)') "Calculating virtual MOs..."
     468             :             END IF
     469         200 :             DO ispin = 1, nspins
     470             :                ! nmo_available is the number of MOs available from the SCF calculation:
     471             :                ! this is at least the number of occupied orbitals in the SCF, plus
     472             :                ! any number of added MOs (virtuals) requested in the SCF section
     473         106 :                CALL get_mo_set(mos(ispin), mo_coeff=mo_ref, nmo=nmo_available)
     474             : 
     475             :                ! calculate how many extra MOs we still have to compute
     476         106 :                nmo_virtual = nmo_occ - nmo_available
     477         106 :                nmo_virtual = MAX(nmo_virtual, 0)
     478             : 
     479             :                NULLIFY (evals_virtual)
     480         212 :                ALLOCATE (evals_virtual(nmo_virtual))
     481             : 
     482             :                CALL cp_fm_get_info(mo_ref, context=context, para_env=para_env, &
     483         106 :                                    nrow_global=nrow_global)
     484             : 
     485             :                CALL cp_fm_struct_create(fm_struct_tmp, para_env=para_env, context=context, &
     486         106 :                                         nrow_global=nrow_global, ncol_global=nmo_virtual)
     487         106 :                CALL cp_fm_create(mo_virtual, fm_struct_tmp, name="virtual")
     488         106 :                CALL cp_fm_struct_release(fm_struct_tmp)
     489         106 :                CALL cp_fm_init_random(mo_virtual, nmo_virtual)
     490             : 
     491         106 :                NULLIFY (local_preconditioner)
     492             : 
     493             :                ! compute missing virtual MOs
     494             :                CALL ot_eigensolver(matrix_h=ks_matrix(ispin)%matrix, matrix_s=s_matrix(1)%matrix, &
     495             :                                    matrix_c_fm=mo_virtual, matrix_orthogonal_space_fm=mo_ref, &
     496             :                                    eps_gradient=scf_control%eps_lumos, &
     497             :                                    preconditioner=local_preconditioner, &
     498             :                                    iter_max=scf_control%max_iter_lumos, &
     499         106 :                                    size_ortho_space=nmo_available)
     500             : 
     501             :                ! get the eigenvalues
     502         106 :                CALL calculate_subspace_eigenvalues(mo_virtual, ks_matrix(ispin)%matrix, evals_virtual)
     503             : 
     504             :                ! TODO: double check that we really need this.
     505             :                ! we need to send the copy of MOs to preserve the sign
     506         106 :                CALL cp_fm_create(fm_dummy, mo_ref%matrix_struct)
     507         106 :                CALL cp_fm_to_fm(mo_ref, fm_dummy)
     508             :                CALL calculate_subspace_eigenvalues(fm_dummy, ks_matrix(ispin)%matrix, &
     509         106 :                                                    evals_arg=eigenvalues(:, ispin), do_rotation=.TRUE.)
     510             : 
     511             :                ! copy inactive orbitals
     512         106 :                mo_set => active_space_env%mos_inactive(ispin)
     513         106 :                CALL get_mo_set(mo_set, mo_coeff=mo_target)
     514         182 :                DO i = 1, SIZE(active_space_env%inactive_orbitals, 1)
     515          76 :                   m = active_space_env%inactive_orbitals(i, ispin)
     516          76 :                   CALL cp_fm_to_fm(mo_ref, mo_target, 1, m, m)
     517          76 :                   mo_set%eigenvalues(m) = eigenvalues(m, ispin)
     518         182 :                   IF (nspins > 1) THEN
     519          56 :                      mo_set%occupation_numbers(m) = 1.0
     520             :                   ELSE
     521          20 :                      mo_set%occupation_numbers(m) = 2.0
     522             :                   END IF
     523             :                END DO
     524             : 
     525             :                ! copy active orbitals
     526         106 :                mo_set => active_space_env%mos_active(ispin)
     527         106 :                CALL get_mo_set(mo_set, mo_coeff=mo_target)
     528             :                ! for mult > 1, put the polarized electrons in the alpha channel
     529         106 :                IF (nspins == 2) THEN
     530          24 :                   IF (ispin == 1) THEN
     531          12 :                      nel = (nelec_active + active_space_env%multiplicity - 1)/2
     532             :                   ELSE
     533          12 :                      nel = (nelec_active - active_space_env%multiplicity + 1)/2
     534             :                   END IF
     535             :                ELSE
     536          82 :                   nel = nelec_active
     537             :                END IF
     538         106 :                mo_set%nelectron = nel
     539         106 :                mo_set%n_el_f = REAL(nel, KIND=dp)
     540         388 :                DO i = 1, nmo_active
     541         282 :                   m = active_space_env%active_orbitals(i, ispin)
     542         282 :                   IF (m > nmo_available) THEN
     543           0 :                      CALL cp_fm_to_fm(mo_virtual, mo_target, 1, m - nmo_available, m)
     544           0 :                      eigenvalues(m, ispin) = evals_virtual(m - nmo_available)
     545           0 :                      mo_set%occupation_numbers(m) = 0.0
     546             :                   ELSE
     547         282 :                      CALL cp_fm_to_fm(mo_ref, mo_target, 1, m, m)
     548         282 :                      mo_set%occupation_numbers(m) = mos(ispin)%occupation_numbers(m)
     549             :                   END IF
     550         388 :                   mo_set%eigenvalues(m) = eigenvalues(m, ispin)
     551             :                END DO
     552             :                ! Release
     553         106 :                DEALLOCATE (evals_virtual)
     554         106 :                CALL cp_fm_release(fm_dummy)
     555         624 :                CALL cp_fm_release(mo_virtual)
     556             :             END DO
     557             : 
     558          94 :             IF (iw > 0) THEN
     559         100 :                DO ispin = 1, nspins
     560          53 :                   WRITE (iw, '(/,T3,A,I3,T66,A)') "Canonical Orbital Selection for spin", ispin, &
     561         106 :                      "[atomic units]"
     562          68 :                   DO i = 1, nmo_inactive, 4
     563          15 :                      jm = MIN(3, nmo_inactive - i)
     564         106 :                      WRITE (iw, '(T3,4(F14.6,A5))') (eigenvalues(i + j, ispin), " [I]", j=0, jm)
     565             :                   END DO
     566         107 :                   DO i = nmo_inactive + 1, nmo_inactive + nmo_active, 4
     567          54 :                      jm = MIN(3, nmo_inactive + nmo_active - i)
     568         248 :                      WRITE (iw, '(T3,4(F14.6,A5))') (eigenvalues(i + j, ispin), " [A]", j=0, jm)
     569             :                   END DO
     570          53 :                   WRITE (iw, '(/,T3,A,I3)') "Active Orbital Indices for spin", ispin
     571         154 :                   DO i = 1, SIZE(active_space_env%active_orbitals, 1), 4
     572          54 :                      jm = MIN(3, SIZE(active_space_env%active_orbitals, 1) - i)
     573         248 :                      WRITE (iw, '(T3,4(I4))') (active_space_env%active_orbitals(i + j, ispin), j=0, jm)
     574             :                   END DO
     575             :                END DO
     576             :             END IF
     577          94 :             DEALLOCATE (eigenvalues)
     578             :          END IF
     579             : 
     580             :       CASE (manual_selection)
     581             :          ! create canonical orbitals
     582          12 :          IF (dft_control%restricted) THEN
     583           0 :             CPABORT("Unclear how we define MOs in the restricted case ... stopping")
     584             :          ELSE
     585          12 :             IF (dft_control%do_admm) THEN
     586           0 :                CPABORT("ADMM currently not possible for manual orbitals selection")
     587             :             END IF
     588             : 
     589          12 :             CALL section_vals_val_get(as_input, "ACTIVE_ORBITAL_INDICES", explicit=explicit, i_vals=invals)
     590          12 :             IF (.NOT. explicit) THEN
     591             :                CALL cp_abort(__LOCATION__, "Manual orbital selection requires to explicitly "// &
     592           0 :                              "set the active orbital indices via ACTIVE_ORBITAL_INDICES")
     593             :             END IF
     594             : 
     595          12 :             IF (nspins == 1) THEN
     596           6 :                CPASSERT(SIZE(invals) == nmo_active)
     597             :             ELSE
     598           6 :                CPASSERT(SIZE(invals) == 2*nmo_active)
     599             :             END IF
     600          36 :             ALLOCATE (active_space_env%inactive_orbitals(nmo_inactive, nspins))
     601          48 :             ALLOCATE (active_space_env%active_orbitals(nmo_active, nspins))
     602             : 
     603          30 :             DO ispin = 1, nspins
     604          66 :                DO i = 1, nmo_active
     605          54 :                   active_space_env%active_orbitals(i, ispin) = invals(i + (ispin - 1)*nmo_active)
     606             :                END DO
     607             :             END DO
     608             : 
     609          12 :             CALL get_qs_env(qs_env, mos=mos)
     610             : 
     611             :             ! include MOs up to the largest index in the list
     612          48 :             max_orb_ind = MAXVAL(invals)
     613          12 :             maxocc = 2.0_dp
     614          12 :             IF (nspins > 1) maxocc = 1.0_dp
     615          54 :             ALLOCATE (active_space_env%mos_active(nspins))
     616          42 :             ALLOCATE (active_space_env%mos_inactive(nspins))
     617          30 :             DO ispin = 1, nspins
     618             :                ! init active orbitals
     619          18 :                CALL get_mo_set(mos(ispin), mo_coeff=mo_ref, nao=nao)
     620          18 :                CALL cp_fm_get_info(mo_ref, context=context, para_env=para_env, nrow_global=nrow_global)
     621          18 :                CALL allocate_mo_set(active_space_env%mos_active(ispin), nao, max_orb_ind, 0, 0.0_dp, maxocc, 0.0_dp)
     622             :                CALL cp_fm_struct_create(fm_struct_tmp, para_env=para_env, context=context, &
     623          18 :                                         nrow_global=nrow_global, ncol_global=max_orb_ind)
     624          18 :                CALL init_mo_set(active_space_env%mos_active(ispin), fm_struct=fm_struct_tmp, name="Active Space MO")
     625          18 :                CALL cp_fm_struct_release(fm_struct_tmp)
     626             : 
     627             :                ! init inactive orbitals
     628          18 :                IF (nspins == 2) THEN
     629          12 :                   nel = nelec_inactive/2
     630             :                ELSE
     631           6 :                   nel = nelec_inactive
     632             :                END IF
     633          18 :                CALL allocate_mo_set(active_space_env%mos_inactive(ispin), nao, max_orb_ind, nel, REAL(nel, KIND=dp), maxocc, 0.0_dp)
     634             :                CALL cp_fm_struct_create(fm_struct_tmp, para_env=para_env, context=context, &
     635          18 :                                         nrow_global=nrow_global, ncol_global=max_orb_ind)
     636          18 :                CALL init_mo_set(active_space_env%mos_inactive(ispin), fm_struct=fm_struct_tmp, name="Inactive Space MO")
     637             :                ! small hack: set the correct inactive occupations down below
     638          68 :                active_space_env%mos_inactive(ispin)%occupation_numbers = 0.0_dp
     639          48 :                CALL cp_fm_struct_release(fm_struct_tmp)
     640             :             END DO
     641             : 
     642          48 :             ALLOCATE (eigenvalues(max_orb_ind, nspins))
     643          80 :             eigenvalues = 0.0_dp
     644          12 :             CALL get_qs_env(qs_env, matrix_ks=ks_matrix, matrix_s=s_matrix, scf_control=scf_control)
     645             : 
     646             :             ! calculate virtual MOs and copy inactive and active orbitals
     647          12 :             IF (iw > 0) THEN
     648           6 :                WRITE (iw, '(/,T3,A)') "Calculating virtual MOs..."
     649             :             END IF
     650          30 :             DO ispin = 1, nspins
     651          18 :                CALL get_mo_set(mos(ispin), mo_coeff=mo_ref, nmo=nmo_available)
     652          18 :                nmo_virtual = max_orb_ind - nmo_available
     653          18 :                nmo_virtual = MAX(nmo_virtual, 0)
     654             : 
     655             :                NULLIFY (evals_virtual)
     656          36 :                ALLOCATE (evals_virtual(nmo_virtual))
     657             : 
     658             :                CALL cp_fm_get_info(mo_ref, context=context, para_env=para_env, &
     659          18 :                                    nrow_global=nrow_global)
     660             : 
     661             :                CALL cp_fm_struct_create(fm_struct_tmp, para_env=para_env, context=context, &
     662          18 :                                         nrow_global=nrow_global, ncol_global=nmo_virtual)
     663          18 :                CALL cp_fm_create(mo_virtual, fm_struct_tmp, name="virtual")
     664          18 :                CALL cp_fm_struct_release(fm_struct_tmp)
     665          18 :                CALL cp_fm_init_random(mo_virtual, nmo_virtual)
     666             : 
     667          18 :                NULLIFY (local_preconditioner)
     668             : 
     669             :                CALL ot_eigensolver(matrix_h=ks_matrix(ispin)%matrix, matrix_s=s_matrix(1)%matrix, &
     670             :                                    matrix_c_fm=mo_virtual, matrix_orthogonal_space_fm=mo_ref, &
     671             :                                    eps_gradient=scf_control%eps_lumos, &
     672             :                                    preconditioner=local_preconditioner, &
     673             :                                    iter_max=scf_control%max_iter_lumos, &
     674          18 :                                    size_ortho_space=nmo_available)
     675             : 
     676             :                CALL calculate_subspace_eigenvalues(mo_virtual, ks_matrix(ispin)%matrix, &
     677          18 :                                                    evals_virtual)
     678             : 
     679             :                ! We need to send the copy of MOs to preserve the sign
     680          18 :                CALL cp_fm_create(fm_dummy, mo_ref%matrix_struct)
     681          18 :                CALL cp_fm_to_fm(mo_ref, fm_dummy)
     682             : 
     683             :                CALL calculate_subspace_eigenvalues(fm_dummy, ks_matrix(ispin)%matrix, &
     684          18 :                                                    evals_arg=eigenvalues(:, ispin), do_rotation=.TRUE.)
     685             : 
     686          18 :                mo_set_active => active_space_env%mos_active(ispin)
     687          18 :                CALL get_mo_set(mo_set_active, mo_coeff=fm_target_active)
     688          18 :                mo_set_inactive => active_space_env%mos_inactive(ispin)
     689          18 :                CALL get_mo_set(mo_set_inactive, mo_coeff=fm_target_inactive)
     690             : 
     691             :                ! copy orbitals
     692          18 :                nmo_inactive_remaining = nmo_inactive
     693          68 :                DO i = 1, max_orb_ind
     694             :                   ! case for i being an active orbital
     695         114 :                   IF (ANY(active_space_env%active_orbitals(:, ispin) == i)) THEN
     696          36 :                      IF (i > nmo_available) THEN
     697           0 :                         CALL cp_fm_to_fm(mo_virtual, fm_target_active, 1, i - nmo_available, i)
     698           0 :                         eigenvalues(i, ispin) = evals_virtual(i - nmo_available)
     699           0 :                         mo_set_active%occupation_numbers(i) = 0.0
     700             :                      ELSE
     701          36 :                         CALL cp_fm_to_fm(fm_dummy, fm_target_active, 1, i, i)
     702          36 :                         mo_set_active%occupation_numbers(i) = mos(ispin)%occupation_numbers(i)
     703             :                      END IF
     704          36 :                      mo_set_active%eigenvalues(i) = eigenvalues(i, ispin)
     705             :                      ! if it was not an active orbital, check whether it is an inactive orbital
     706          14 :                   ELSEIF (nmo_inactive_remaining > 0) THEN
     707           0 :                      CALL cp_fm_to_fm(fm_dummy, fm_target_inactive, 1, i, i)
     708             :                      ! store on the fly the mapping of inactive orbitals
     709           0 :                      active_space_env%inactive_orbitals(nmo_inactive - nmo_inactive_remaining + 1, ispin) = i
     710           0 :                      mo_set_inactive%eigenvalues(i) = eigenvalues(i, ispin)
     711           0 :                      mo_set_inactive%occupation_numbers(i) = mos(ispin)%occupation_numbers(i)
     712             :                      ! hack: set homo and lumo manually
     713           0 :                      IF (nmo_inactive_remaining == 1) THEN
     714           0 :                         mo_set_inactive%homo = i
     715           0 :                         mo_set_inactive%lfomo = i + 1
     716             :                      END IF
     717           0 :                      nmo_inactive_remaining = nmo_inactive_remaining - 1
     718             :                   ELSE
     719          14 :                      CYCLE
     720             :                   END IF
     721             :                END DO
     722             : 
     723             :                ! Release
     724          18 :                DEALLOCATE (evals_virtual)
     725          18 :                CALL cp_fm_release(fm_dummy)
     726         102 :                CALL cp_fm_release(mo_virtual)
     727             :             END DO
     728             : 
     729          12 :             IF (iw > 0) THEN
     730          15 :                DO ispin = 1, nspins
     731           9 :                   WRITE (iw, '(/,T3,A,I3,T66,A)') "Orbital Energies and Selection for spin", ispin, "[atomic units]"
     732             : 
     733          18 :                   DO i = 1, max_orb_ind, 4
     734           9 :                      jm = MIN(3, max_orb_ind - i)
     735           9 :                      WRITE (iw, '(T4)', advance="no")
     736          34 :                      DO j = 0, jm
     737          57 :                         IF (ANY(active_space_env%active_orbitals(:, ispin) == i + j)) THEN
     738          18 :                            WRITE (iw, '(T3,F12.6,A5)', advance="no") eigenvalues(i + j, ispin), " [A]"
     739           7 :                         ELSEIF (ANY(active_space_env%inactive_orbitals(:, ispin) == i + j)) THEN
     740           0 :                            WRITE (iw, '(T3,F12.6,A5)', advance="no") eigenvalues(i + j, ispin), " [I]"
     741             :                         ELSE
     742           7 :                            WRITE (iw, '(T3,F12.6,A5)', advance="no") eigenvalues(i + j, ispin), " [V]"
     743             :                         END IF
     744             :                      END DO
     745          18 :                      WRITE (iw, *)
     746             :                   END DO
     747           9 :                   WRITE (iw, '(/,T3,A,I3)') "Active Orbital Indices for spin", ispin
     748          24 :                   DO i = 1, SIZE(active_space_env%active_orbitals, 1), 4
     749           9 :                      jm = MIN(3, SIZE(active_space_env%active_orbitals, 1) - i)
     750          36 :                      WRITE (iw, '(T3,4(I4))') (active_space_env%active_orbitals(i + j, ispin), j=0, jm)
     751             :                   END DO
     752             :                END DO
     753             :             END IF
     754          24 :             DEALLOCATE (eigenvalues)
     755             :          END IF
     756             : 
     757             :       CASE (wannier_projection)
     758           0 :          NULLIFY (loc_section, loc_print)
     759           0 :          loc_section => section_vals_get_subs_vals(as_input, "LOCALIZE")
     760           0 :          CPASSERT(ASSOCIATED(loc_section))
     761           0 :          loc_print => section_vals_get_subs_vals(as_input, "LOCALIZE%PRINT")
     762             :          !
     763           0 :          CPABORT("not yet available")
     764             :          !
     765             :       CASE (mao_projection)
     766             :          !
     767         106 :          CPABORT("not yet available")
     768             :          !
     769             :       END SELECT
     770             : 
     771             :       ! Print orbitals on Cube files
     772         106 :       print_orb => section_vals_get_subs_vals(as_input, "PRINT_ORBITAL_CUBES")
     773         106 :       CALL section_vals_get(print_orb, explicit=explicit)
     774         106 :       CALL section_vals_val_get(print_orb, "STOP_AFTER_CUBES", l_val=stop_after_print)
     775         106 :       IF (explicit) THEN
     776             :          !
     777           2 :          CALL print_orbital_cubes(print_orb, qs_env, active_space_env%mos_active)
     778             :          !
     779           2 :          IF (stop_after_print) THEN
     780             : 
     781           0 :             IF (iw > 0) THEN
     782             :                WRITE (iw, '(/,T2,A)') &
     783           0 :                   '!----------------- Early End of Active Space Interface -----------------------!'
     784             :             END IF
     785             : 
     786           0 :             CALL timestop(handle)
     787             : 
     788           0 :             RETURN
     789             :          END IF
     790             :       END IF
     791             : 
     792             :       ! calculate inactive density matrix
     793         106 :       CALL get_qs_env(qs_env, rho=rho)
     794         106 :       CALL qs_rho_get(rho, rho_ao=rho_ao)
     795         106 :       CPASSERT(ASSOCIATED(rho_ao))
     796         106 :       CALL dbcsr_allocate_matrix_set(active_space_env%pmat_inactive, nspins)
     797         230 :       DO ispin = 1, nspins
     798         124 :          ALLOCATE (denmat)
     799         124 :          CALL dbcsr_copy(denmat, rho_ao(ispin)%matrix)
     800         124 :          mo_set => active_space_env%mos_inactive(ispin)
     801         124 :          CALL calculate_density_matrix(mo_set, denmat)
     802         230 :          active_space_env%pmat_inactive(ispin)%matrix => denmat
     803             :       END DO
     804             : 
     805             :       ! generate integrals
     806             :       ! make sure that defaults are set correctly (from basic calculation)
     807             :       ! make sure that periodicity is consistent
     808         106 :       CALL section_vals_val_get(as_input, "ERI%METHOD", i_val=eri_method)
     809         106 :       active_space_env%eri%method = eri_method
     810         106 :       CALL section_vals_val_get(as_input, "ERI%OPERATOR", i_val=eri_operator, explicit=ex_operator)
     811         106 :       active_space_env%eri%operator = eri_operator
     812         106 :       CALL section_vals_val_get(as_input, "ERI%OPERATOR_PARAMETER", r_val=eri_op_param)
     813         106 :       active_space_env%eri%operator_parameter = eri_op_param
     814         106 :       CALL section_vals_val_get(as_input, "ERI%CUTOFF_RADIUS", r_val=eri_rcut)
     815         106 :       active_space_env%eri%cutoff_radius = eri_rcut
     816         106 :       CALL section_vals_val_get(as_input, "ERI%PERIODICITY", i_vals=invals, explicit=ex_perd)
     817         106 :       CALL section_vals_val_get(as_input, "ERI%EPS_INTEGRAL", r_val=eri_eps_int)
     818         106 :       active_space_env%eri%eps_integral = eri_eps_int
     819         106 :       IF (active_space_env%molecule) THEN
     820             :          ! check that we are in a non-periodic setting
     821         102 :          CALL get_qs_env(qs_env, cell=cell)
     822         408 :          IF (SUM(cell%perd) /= 0) THEN
     823           0 :             CPABORT("Active space option ISOLATED_SYSTEM requires non-periodic setting")
     824             :          END IF
     825         102 :          IF (ex_perd) THEN
     826         408 :             IF (SUM(invals) /= 0) THEN
     827           0 :                CPABORT("Active space option ISOLATED_SYSTEM requires non-periodic setting")
     828             :             END IF
     829             :          END IF
     830         408 :          active_space_env%eri%periodicity(1:3) = 0
     831           4 :       ELSE IF (ex_perd) THEN
     832           0 :          IF (SIZE(invals) == 1) THEN
     833           0 :             active_space_env%eri%periodicity(1:3) = invals(1)
     834             :          ELSE
     835           0 :             active_space_env%eri%periodicity(1:3) = invals(1:3)
     836             :          END IF
     837             :       ELSE
     838           4 :          CALL get_qs_env(qs_env, cell=cell)
     839          28 :          active_space_env%eri%periodicity(1:3) = cell%perd(1:3)
     840             :       END IF
     841         106 :       IF (iw > 0) THEN
     842          53 :          WRITE (iw, '(/,T3,A)') "Calculation of Electron Repulsion Integrals"
     843          41 :          SELECT CASE (eri_method)
     844             :          CASE (eri_method_full_gpw)
     845          41 :             WRITE (iw, '(T3,A,T50,A)') "Integration method", "GPW Fourier transform over MOs"
     846             :          CASE (eri_method_gpw_ht)
     847          12 :             WRITE (iw, '(T3,A,T44,A)') "Integration method", "Half transformed integrals from GPW"
     848             :          CASE DEFAULT
     849          53 :             CPABORT("Unknown ERI method")
     850             :          END SELECT
     851          46 :          SELECT CASE (eri_operator)
     852             :          CASE (eri_operator_coulomb)
     853          46 :             WRITE (iw, '(T3,A,T66,A)') "ERI operator", "Coulomb <1/R>"
     854             :          CASE (eri_operator_yukawa)
     855           0 :             WRITE (iw, '(T3,A,T59,A)') "ERI operator", "Yukawa <EXP(-a*R)/R>"
     856           0 :             WRITE (iw, '(T3,A,T66,F14.3)') "ERI operator parameter", eri_op_param
     857             :          CASE (eri_operator_erf)
     858           7 :             WRITE (iw, '(T3,A,T53,A)') "ERI operator", "Error function <ERF(a*R)/R>"
     859           7 :             WRITE (iw, '(T3,A,T66,F14.3)') "ERI operator parameter", eri_op_param
     860             :          CASE (eri_operator_erfc)
     861           0 :             WRITE (iw, '(T3,A,T45,A)') "ERI operator", "Compl. error function <ERFC(a*R)/R>"
     862           0 :             WRITE (iw, '(T3,A,T66,F14.3)') "ERI operator parameter", eri_op_param
     863             :          CASE (eri_operator_gaussian)
     864           0 :             WRITE (iw, '(T3,A,T41,A)') "ERI operator", "Gaussian attenuated <EXP(-a*R^2)/R>"
     865           0 :             WRITE (iw, '(T3,A,T66,F14.3)') "ERI operator parameter", eri_op_param
     866             :          CASE (eri_operator_trunc)
     867           0 :             WRITE (iw, '(T3,A,T53,A)') "ERI operator", "Truncated Coulomb <H(a-R)/R>"
     868           0 :             WRITE (iw, '(T3,A,T66,F14.3)') "ERI operator parameter", eri_op_param
     869             :          CASE DEFAULT
     870          53 :             CPABORT("Unknown ERI operator")
     871             :          END SELECT
     872          53 :          WRITE (iw, '(T3,A,T68,E12.4)') "Accuracy of ERI", eri_eps_int
     873          53 :          WRITE (iw, '(T3,A,T71,3I3)') "Periodicity", active_space_env%eri%periodicity(1:3)
     874         212 :          IF (PRODUCT(active_space_env%eri%periodicity(1:3)) == 0) THEN
     875          52 :             IF (eri_rcut > 0.0_dp) WRITE (iw, '(T3,A,T65,F14.6)') "Periodicity (Cutoff)", eri_rcut
     876             :          END IF
     877          53 :          IF (nspins < 2) THEN
     878          44 :             WRITE (iw, '(T3,A,T68,I12)') "Total Number of ERI", (nmo_active**4)/8
     879             :          ELSE
     880           9 :             WRITE (iw, '(T3,A,T68,I12)') "Total Number of ERI (aa|aa)", (nmo_active**4)/8
     881           9 :             WRITE (iw, '(T3,A,T68,I12)') "Total Number of ERI (bb|bb)", (nmo_active**4)/8
     882           9 :             WRITE (iw, '(T3,A,T68,I12)') "Total Number of ERI (aa|bb)", (nmo_active**4)/4
     883             :          END IF
     884             :       END IF
     885             : 
     886             :       ! allocate container for integrals (CSR matrix)
     887         106 :       CALL get_qs_env(qs_env, para_env=para_env)
     888         106 :       m = (nspins*(nspins + 1))/2
     889         460 :       ALLOCATE (active_space_env%eri%eri(m))
     890         248 :       DO i = 1, m
     891         142 :          CALL get_mo_set(active_space_env%mos_active(1), nmo=nmo)
     892         142 :          ALLOCATE (active_space_env%eri%eri(i)%csr_mat)
     893         142 :          eri_mat => active_space_env%eri%eri(i)%csr_mat
     894         142 :          IF (i == 1) THEN
     895         106 :             n1 = nmo
     896         106 :             n2 = nmo
     897          36 :          ELSEIF (i == 2) THEN
     898          18 :             n1 = nmo
     899          18 :             n2 = nmo
     900             :          ELSE
     901          18 :             n1 = nmo
     902          18 :             n2 = nmo
     903             :          END IF
     904         142 :          nn1 = (n1*(n1 + 1))/2
     905         142 :          nn2 = (n2*(n2 + 1))/2
     906         142 :          CALL dbcsr_csr_create(eri_mat, nn1, nn2, 0_int_8, 0, 0, para_env%get_handle())
     907         390 :          active_space_env%eri%norb = nmo
     908             :       END DO
     909             : 
     910         106 :       SELECT CASE (eri_method)
     911             :       CASE (eri_method_full_gpw, eri_method_gpw_ht)
     912         106 :          CALL section_vals_val_get(as_input, "ERI_GPW%EPS_GRID", r_val=eri_eps_grid)
     913         106 :          active_space_env%eri%eri_gpw%eps_grid = eri_eps_grid
     914         106 :          CALL section_vals_val_get(as_input, "ERI_GPW%EPS_FILTER", r_val=eri_eps_filter)
     915         106 :          active_space_env%eri%eri_gpw%eps_filter = eri_eps_filter
     916         106 :          CALL section_vals_val_get(as_input, "ERI_GPW%CUTOFF", r_val=eri_gpw_cutoff)
     917         106 :          active_space_env%eri%eri_gpw%cutoff = eri_gpw_cutoff
     918         106 :          CALL section_vals_val_get(as_input, "ERI_GPW%REL_CUTOFF", r_val=eri_rel_cutoff)
     919         106 :          active_space_env%eri%eri_gpw%rel_cutoff = eri_rel_cutoff
     920         106 :          CALL section_vals_val_get(as_input, "ERI_GPW%PRINT_LEVEL", i_val=eri_print)
     921         106 :          active_space_env%eri%eri_gpw%print_level = eri_print
     922         106 :          CALL section_vals_val_get(as_input, "ERI_GPW%STORE_WFN", l_val=store_wfn)
     923         106 :          active_space_env%eri%eri_gpw%store_wfn = store_wfn
     924         106 :          CALL section_vals_val_get(as_input, "ERI_GPW%GROUP_SIZE", i_val=group_size)
     925         106 :          active_space_env%eri%eri_gpw%group_size = group_size
     926             :          ! Always redo Poisson solver for now
     927         106 :          active_space_env%eri%eri_gpw%redo_poisson = .TRUE.
     928             :          ! active_space_env%eri%eri_gpw%redo_poisson = (ex_operator .OR. ex_perd)
     929         106 :          IF (iw > 0) THEN
     930          53 :             WRITE (iw, '(/,T2,A,T71,F10.1)') "ERI_GPW| Energy cutoff [Ry]", eri_gpw_cutoff
     931          53 :             WRITE (iw, '(T2,A,T71,F10.1)') "ERI_GPW| Relative energy cutoff [Ry]", eri_rel_cutoff
     932             :          END IF
     933             :          !
     934         106 :          CALL calculate_eri_gpw(active_space_env%mos_active, active_space_env%active_orbitals, active_space_env%eri, qs_env, iw)
     935             :          !
     936             :       CASE DEFAULT
     937         106 :          CPABORT("Unknown ERI method")
     938             :       END SELECT
     939         106 :       IF (iw > 0) THEN
     940         124 :          DO isp = 1, SIZE(active_space_env%eri%eri)
     941          71 :             eri_mat => active_space_env%eri%eri(isp)%csr_mat
     942             :             nze_percentage = 100.0_dp*(REAL(eri_mat%nze_total, KIND=dp) &
     943          71 :                                        /REAL(eri_mat%nrows_total, KIND=dp))/REAL(eri_mat%ncols_total, KIND=dp)
     944          71 :             WRITE (iw, '(/,T2,A,I2,T30,A,T68,I12)') "ERI_GPW| Spinmatrix:", isp, &
     945         142 :                "Number of  CSR non-zero elements:", eri_mat%nze_total
     946          71 :             WRITE (iw, '(T2,A,I2,T30,A,T68,F12.4)') "ERI_GPW| Spinmatrix:", isp, &
     947         142 :                "Percentage CSR non-zero elements:", nze_percentage
     948          71 :             WRITE (iw, '(T2,A,I2,T30,A,T68,I12)') "ERI_GPW| Spinmatrix:", isp, &
     949         142 :                "nrows_total", eri_mat%nrows_total
     950          71 :             WRITE (iw, '(T2,A,I2,T30,A,T68,I12)') "ERI_GPW| Spinmatrix:", isp, &
     951         142 :                "ncols_total", eri_mat%ncols_total
     952          71 :             WRITE (iw, '(T2,A,I2,T30,A,T68,I12)') "ERI_GPW| Spinmatrix:", isp, &
     953         195 :                "nrows_local", eri_mat%nrows_local
     954             :          END DO
     955             :       END IF
     956             : 
     957             :       ! set the reference active space density matrix
     958         106 :       nspins = active_space_env%nspins
     959         442 :       ALLOCATE (active_space_env%p_active(nspins))
     960         230 :       DO isp = 1, nspins
     961         124 :          mo_set => active_space_env%mos_active(isp)
     962         124 :          CALL get_mo_set(mo_set, mo_coeff=mo_coeff, nmo=nmo)
     963         230 :          CALL create_subspace_matrix(mo_coeff, active_space_env%p_active(isp), nmo)
     964             :       END DO
     965           0 :       SELECT CASE (mselect)
     966             :       CASE DEFAULT
     967           0 :          CPABORT("Unknown orbital selection method")
     968             :       CASE (casci_canonical, manual_selection)
     969         106 :          focc = 2.0_dp
     970         106 :          IF (nspins == 2) focc = 1.0_dp
     971         230 :          DO isp = 1, nspins
     972         124 :             fmat => active_space_env%p_active(isp)
     973         124 :             CALL cp_fm_set_all(fmat, alpha=0.0_dp)
     974         124 :             IF (nspins == 2) THEN
     975          36 :                IF (isp == 1) THEN
     976          18 :                   nel = (active_space_env%nelec_active + active_space_env%multiplicity - 1)/2
     977             :                ELSE
     978          18 :                   nel = (active_space_env%nelec_active - active_space_env%multiplicity + 1)/2
     979             :                END IF
     980             :             ELSE
     981          88 :                nel = active_space_env%nelec_active
     982             :             END IF
     983         548 :             DO i = 1, nmo_active
     984         318 :                m = active_space_env%active_orbitals(i, isp)
     985         318 :                fel = MIN(focc, REAL(nel, KIND=dp))
     986         318 :                CALL cp_fm_set_element(fmat, m, m, fel)
     987         318 :                nel = nel - NINT(fel)
     988         442 :                nel = MAX(nel, 0)
     989             :             END DO
     990             :          END DO
     991             :       CASE (wannier_projection)
     992           0 :          CPABORT("NOT IMPLEMENTED")
     993             :       CASE (mao_projection)
     994         106 :          CPABORT("NOT IMPLEMENTED")
     995             :       END SELECT
     996             : 
     997             :       ! transform KS/Fock, Vxc and Hcore to AS MO basis
     998         106 :       CALL calculate_operators(active_space_env%mos_active, qs_env, active_space_env)
     999             :       ! set the reference energy in the active space
    1000         106 :       CALL get_qs_env(qs_env, energy=energy)
    1001         106 :       active_space_env%energy_ref = energy%total
    1002             :       ! calculate inactive energy and embedding potential
    1003         106 :       CALL subspace_fock_matrix(active_space_env)
    1004             : 
    1005             :       ! associate the active space environment with the qs environment
    1006         106 :       CALL set_qs_env(qs_env, active_space=active_space_env)
    1007             : 
    1008             :       ! Perform the embedding calculation only if qiskit is specified
    1009         106 :       CALL section_vals_val_get(as_input, "AS_SOLVER", i_val=as_solver)
    1010         106 :       SELECT CASE (as_solver)
    1011             :       CASE (no_solver)
    1012         106 :          IF (iw > 0) THEN
    1013          53 :             WRITE (iw, '(/,T3,A)') "No active space solver specified, skipping embedding calculation"
    1014             :          END IF
    1015             :       CASE (qiskit_solver)
    1016           0 :          CALL rsdft_embedding(qs_env, active_space_env, as_input)
    1017             :       CASE DEFAULT
    1018         106 :          CPABORT("Unknown active space solver")
    1019             :       END SELECT
    1020             : 
    1021             :       ! Output a FCIDUMP file if requested
    1022         106 :       IF (active_space_env%fcidump) CALL fcidump(active_space_env, as_input)
    1023             : 
    1024             :       ! Output a QCSchema file if requested
    1025         106 :       IF (active_space_env%qcschema) THEN
    1026           0 :          CALL qcschema_env_create(qcschema_env, qs_env)
    1027           0 :          CALL qcschema_to_hdf5(qcschema_env, active_space_env%qcschema_filename)
    1028           0 :          CALL qcschema_env_release(qcschema_env)
    1029             :       END IF
    1030             : 
    1031         106 :       IF (iw > 0) THEN
    1032             :          WRITE (iw, '(/,T2,A)') &
    1033          53 :             '!-------------------- End of Active Space Interface --------------------------!'
    1034             :       END IF
    1035             : 
    1036         106 :       CALL timestop(handle)
    1037             : 
    1038       79978 :    END SUBROUTINE active_space_main
    1039             : 
    1040             : ! **************************************************************************************************
    1041             : !> \brief computes the one-electron operators in the subspace of the provided orbital set
    1042             : !> \param mos the molecular orbital set within the active subspace
    1043             : !> \param qs_env ...
    1044             : !> \param active_space_env ...
    1045             : !> \par History
    1046             : !>      04.2016 created [JGH]
    1047             : ! **************************************************************************************************
    1048         106 :    SUBROUTINE calculate_operators(mos, qs_env, active_space_env)
    1049             : 
    1050             :       TYPE(mo_set_type), DIMENSION(:), INTENT(IN)        :: mos
    1051             :       TYPE(qs_environment_type), POINTER                 :: qs_env
    1052             :       TYPE(active_space_type), POINTER                   :: active_space_env
    1053             : 
    1054             :       CHARACTER(len=*), PARAMETER :: routineN = 'calculate_operators'
    1055             : 
    1056             :       INTEGER                                            :: handle, is, nmo, nspins
    1057             :       TYPE(cp_fm_type), POINTER                          :: mo_coeff
    1058         106 :       TYPE(dbcsr_p_type), DIMENSION(:), POINTER          :: matrix_vxc
    1059         106 :       TYPE(dbcsr_p_type), DIMENSION(:, :), POINTER       :: h_matrix, ks_matrix
    1060             : 
    1061         106 :       CALL timeset(routineN, handle)
    1062             : 
    1063         106 :       nspins = active_space_env%nspins
    1064             : 
    1065             :       ! Kohn-Sham / Fock operator
    1066         106 :       CALL cp_fm_release(active_space_env%ks_sub)
    1067         106 :       CALL get_qs_env(qs_env, matrix_ks_kp=ks_matrix)
    1068         106 :       IF (SIZE(ks_matrix, 2) > 1) THEN
    1069           0 :          CPABORT("No k-points allowed at this point")
    1070             :       END IF
    1071         442 :       ALLOCATE (active_space_env%ks_sub(nspins))
    1072         230 :       DO is = 1, nspins
    1073         124 :          CALL get_mo_set(mo_set=mos(is), mo_coeff=mo_coeff, nmo=nmo)
    1074         230 :          CALL subspace_operator(mo_coeff, nmo, ks_matrix(is, 1)%matrix, active_space_env%ks_sub(is))
    1075             :       END DO
    1076             : 
    1077             :       ! Vxc matrix
    1078         106 :       CALL cp_fm_release(active_space_env%vxc_sub)
    1079             : 
    1080         106 :       NULLIFY (matrix_vxc)
    1081         106 :       CALL get_qs_env(qs_env, matrix_vxc=matrix_vxc)
    1082         106 :       IF (ASSOCIATED(matrix_vxc)) THEN
    1083           0 :          ALLOCATE (active_space_env%vxc_sub(nspins))
    1084           0 :          DO is = 1, nspins
    1085           0 :             CALL get_mo_set(mo_set=mos(is), mo_coeff=mo_coeff, nmo=nmo)
    1086           0 :             CALL subspace_operator(mo_coeff, nmo, matrix_vxc(is)%matrix, active_space_env%vxc_sub(is))
    1087             :          END DO
    1088             :       END IF
    1089             : 
    1090             :       ! Core Hamiltonian
    1091         106 :       CALL cp_fm_release(active_space_env%h_sub)
    1092             : 
    1093         106 :       NULLIFY (h_matrix)
    1094         106 :       CALL get_qs_env(qs_env=qs_env, matrix_h_kp=h_matrix)
    1095         106 :       IF (SIZE(h_matrix, 2) > 1) THEN
    1096           0 :          CPABORT("No k-points allowed at this point")
    1097             :       END IF
    1098         442 :       ALLOCATE (active_space_env%h_sub(nspins))
    1099         230 :       DO is = 1, nspins
    1100         124 :          CALL get_mo_set(mo_set=mos(is), mo_coeff=mo_coeff, nmo=nmo)
    1101         230 :          CALL subspace_operator(mo_coeff, nmo, h_matrix(1, 1)%matrix, active_space_env%h_sub(is))
    1102             :       END DO
    1103             : 
    1104         106 :       CALL timestop(handle)
    1105             : 
    1106         106 :    END SUBROUTINE calculate_operators
    1107             : 
    1108             : ! **************************************************************************************************
    1109             : !> \brief computes a one-electron operator in the subspace of the provided orbital set
    1110             : !> \param orbitals the orbital coefficient matrix
    1111             : !> \param nmo the number of orbitals
    1112             : !> \param op_matrix operator matrix in AO basis
    1113             : !> \param op_sub operator in orbital basis
    1114             : !> \par History
    1115             : !>      04.2016 created [JGH]
    1116             : ! **************************************************************************************************
    1117         496 :    SUBROUTINE subspace_operator(orbitals, nmo, op_matrix, op_sub)
    1118             : 
    1119             :       TYPE(cp_fm_type), INTENT(IN)                       :: orbitals
    1120             :       INTEGER, INTENT(IN)                                :: nmo
    1121             :       TYPE(dbcsr_type), POINTER                          :: op_matrix
    1122             :       TYPE(cp_fm_type), INTENT(INOUT)                    :: op_sub
    1123             : 
    1124             :       CHARACTER(len=*), PARAMETER                        :: routineN = 'subspace_operator'
    1125             : 
    1126             :       INTEGER                                            :: handle, ncol, nrow
    1127             :       TYPE(cp_fm_type)                                   :: vectors
    1128             : 
    1129         248 :       CALL timeset(routineN, handle)
    1130             : 
    1131         248 :       CALL cp_fm_get_info(matrix=orbitals, ncol_global=ncol, nrow_global=nrow)
    1132         248 :       CPASSERT(nmo <= ncol)
    1133             : 
    1134         248 :       IF (nmo > 0) THEN
    1135             : 
    1136         248 :          CALL cp_fm_create(vectors, orbitals%matrix_struct, "vectors")
    1137             : 
    1138         248 :          CALL create_subspace_matrix(orbitals, op_sub, nmo)
    1139             : 
    1140         248 :          CALL cp_dbcsr_sm_fm_multiply(op_matrix, orbitals, vectors, nmo)
    1141             : 
    1142         248 :          CALL parallel_gemm('T', 'N', nmo, nmo, nrow, 1.0_dp, orbitals, vectors, 0.0_dp, op_sub)
    1143             : 
    1144         248 :          CALL cp_fm_release(vectors)
    1145             : 
    1146             :       END IF
    1147             : 
    1148         248 :       CALL timestop(handle)
    1149             : 
    1150         248 :    END SUBROUTINE subspace_operator
    1151             : 
    1152             : ! **************************************************************************************************
    1153             : !> \brief creates a matrix of subspace size
    1154             : !> \param orbitals the orbital coefficient matrix
    1155             : !> \param op_sub operator in orbital basis
    1156             : !> \param n the number of orbitals
    1157             : !> \par History
    1158             : !>      04.2016 created [JGH]
    1159             : ! **************************************************************************************************
    1160         372 :    SUBROUTINE create_subspace_matrix(orbitals, op_sub, n)
    1161             : 
    1162             :       TYPE(cp_fm_type), INTENT(IN)                       :: orbitals
    1163             :       TYPE(cp_fm_type), INTENT(OUT)                      :: op_sub
    1164             :       INTEGER, INTENT(IN)                                :: n
    1165             : 
    1166             :       TYPE(cp_fm_struct_type), POINTER                   :: fm_struct
    1167             : 
    1168         372 :       IF (n > 0) THEN
    1169             : 
    1170         372 :          NULLIFY (fm_struct)
    1171             :          CALL cp_fm_struct_create(fm_struct, nrow_global=n, ncol_global=n, &
    1172             :                                   para_env=orbitals%matrix_struct%para_env, &
    1173         372 :                                   context=orbitals%matrix_struct%context)
    1174         372 :          CALL cp_fm_create(op_sub, fm_struct, name="Subspace operator")
    1175         372 :          CALL cp_fm_struct_release(fm_struct)
    1176             : 
    1177             :       END IF
    1178             : 
    1179         372 :    END SUBROUTINE create_subspace_matrix
    1180             : 
    1181             : ! **************************************************************************************************
    1182             : !> \brief computes a electron repulsion integrals using GPW technology
    1183             : !> \param mos the molecular orbital set within the active subspace
    1184             : !> \param orbitals ...
    1185             : !> \param eri_env ...
    1186             : !> \param qs_env ...
    1187             : !> \param iw ...
    1188             : !> \par History
    1189             : !>      04.2016 created [JGH]
    1190             : ! **************************************************************************************************
    1191         106 :    SUBROUTINE calculate_eri_gpw(mos, orbitals, eri_env, qs_env, iw)
    1192             : 
    1193             :       TYPE(mo_set_type), DIMENSION(:), INTENT(IN)        :: mos
    1194             :       INTEGER, DIMENSION(:, :), POINTER                  :: orbitals
    1195             :       TYPE(eri_type)                                     :: eri_env
    1196             :       TYPE(qs_environment_type), POINTER                 :: qs_env
    1197             :       INTEGER, INTENT(IN)                                :: iw
    1198             : 
    1199             :       CHARACTER(len=*), PARAMETER                        :: routineN = 'calculate_eri_gpw'
    1200             : 
    1201             :       INTEGER :: col_local, color, handle, i1, i2, i3, i4, i_multigrid, icount2, intcount, &
    1202             :          irange(2), irange_sub(2), isp, isp1, isp2, ispin, iwa1, iwa12, iwa2, iwb1, iwb12, iwb2, &
    1203             :          iwbs, iwbt, iwfn, n_multigrid, ncol_global, ncol_local, nmm, nmo, nmo1, nmo2, &
    1204             :          nrow_global, nrow_local, nspins, number_of_subgroups, nx, row_local, stored_integrals
    1205         106 :       INTEGER, ALLOCATABLE, DIMENSION(:)                 :: eri_index
    1206         106 :       INTEGER, DIMENSION(:), POINTER                     :: col_indices, row_indices
    1207             :       LOGICAL                                            :: print1, print2, &
    1208             :                                                             skip_load_balance_distributed
    1209             :       REAL(KIND=dp)                                      :: dvol, erint, pair_int, &
    1210             :                                                             progression_factor, rc, rsize
    1211         106 :       REAL(KIND=dp), ALLOCATABLE, DIMENSION(:)           :: eri
    1212         106 :       TYPE(atomic_kind_type), DIMENSION(:), POINTER      :: atomic_kind_set
    1213             :       TYPE(cell_type), POINTER                           :: cell
    1214             :       TYPE(cp_blacs_env_type), POINTER                   :: blacs_env, blacs_env_sub
    1215             :       TYPE(cp_fm_struct_type), POINTER                   :: fm_struct
    1216         106 :       TYPE(cp_fm_type), ALLOCATABLE, DIMENSION(:)        :: fm_matrix_pq_rnu, fm_matrix_pq_rs, &
    1217         106 :                                                             fm_mo_coeff_as
    1218             :       TYPE(cp_fm_type), POINTER                          :: mo_coeff, mo_coeff1, mo_coeff2
    1219             :       TYPE(dbcsr_p_type)                                 :: mat_munu
    1220         106 :       TYPE(dbcsr_type), ALLOCATABLE, DIMENSION(:)        :: matrix_pq_rnu, mo_coeff_as
    1221             :       TYPE(dft_control_type), POINTER                    :: dft_control
    1222             :       TYPE(mp_comm_type)                                 :: mp_group
    1223             :       TYPE(mp_para_env_type), POINTER                    :: para_env, para_env_sub
    1224             :       TYPE(neighbor_list_set_p_type), DIMENSION(:), &
    1225         106 :          POINTER                                         :: sab_orb_sub
    1226         106 :       TYPE(particle_type), DIMENSION(:), POINTER         :: particle_set
    1227             :       TYPE(pw_c1d_gs_type)                               :: pot_g, rho_g
    1228             :       TYPE(pw_env_type), POINTER                         :: pw_env_sub
    1229             :       TYPE(pw_poisson_type), POINTER                     :: poisson_env
    1230             :       TYPE(pw_pool_type), POINTER                        :: auxbas_pw_pool
    1231             :       TYPE(pw_r3d_rs_type)                               :: rho_r, wfn_r
    1232             :       TYPE(pw_r3d_rs_type), ALLOCATABLE, &
    1233         106 :          DIMENSION(:, :), TARGET                         :: wfn_a
    1234             :       TYPE(pw_r3d_rs_type), POINTER                      :: wfn1, wfn2, wfn3, wfn4
    1235             :       TYPE(qs_control_type), POINTER                     :: qs_control, qs_control_old
    1236         106 :       TYPE(qs_kind_type), DIMENSION(:), POINTER          :: qs_kind_set
    1237             :       TYPE(qs_ks_env_type), POINTER                      :: ks_env
    1238             :       TYPE(task_list_type), POINTER                      :: task_list_sub
    1239             : 
    1240         106 :       CALL timeset(routineN, handle)
    1241             : 
    1242             :       ! print levels
    1243         208 :       SELECT CASE (eri_env%eri_gpw%print_level)
    1244             :       CASE (silent_print_level)
    1245         102 :          print1 = .FALSE.
    1246         102 :          print2 = .FALSE.
    1247             :       CASE (low_print_level)
    1248           2 :          print1 = .FALSE.
    1249           2 :          print2 = .FALSE.
    1250             :       CASE (medium_print_level)
    1251           2 :          print1 = .TRUE.
    1252           2 :          print2 = .FALSE.
    1253             :       CASE (high_print_level)
    1254           0 :          print1 = .TRUE.
    1255           0 :          print2 = .TRUE.
    1256             :       CASE (debug_print_level)
    1257           0 :          print1 = .TRUE.
    1258         106 :          print2 = .TRUE.
    1259             :       CASE DEFAULT
    1260             :          ! do nothing
    1261             :       END SELECT
    1262             : 
    1263             :       ! Check the inpuz group
    1264         106 :       CALL get_qs_env(qs_env, para_env=para_env, blacs_env=blacs_env)
    1265         106 :       IF (eri_env%eri_gpw%group_size <= 1) eri_env%eri_gpw%group_size = para_env%num_pe
    1266         106 :       IF (MOD(para_env%num_pe, eri_env%eri_gpw%group_size) /= 0) &
    1267           0 :          CPABORT("Group size must be a divisor of the total number of processes!")
    1268             :       ! Create a new para_env or reuse the old one
    1269         106 :       IF (eri_env%eri_gpw%group_size == para_env%num_pe) THEN
    1270         106 :          para_env_sub => para_env
    1271         106 :          CALL para_env_sub%retain()
    1272         106 :          IF (eri_env%method == eri_method_gpw_ht) THEN
    1273          24 :             blacs_env_sub => blacs_env
    1274          24 :             CALL blacs_env_sub%retain()
    1275             :          END IF
    1276         106 :          number_of_subgroups = 1
    1277         106 :          color = 0
    1278             :       ELSE
    1279           0 :          number_of_subgroups = para_env%num_pe/eri_env%eri_gpw%group_size
    1280           0 :          color = para_env%mepos/eri_env%eri_gpw%group_size
    1281           0 :          ALLOCATE (para_env_sub)
    1282           0 :          CALL para_env_sub%from_split(para_env, color)
    1283           0 :          IF (eri_env%method == eri_method_gpw_ht) THEN
    1284           0 :             NULLIFY (blacs_env_sub)
    1285           0 :             CALL cp_blacs_env_create(blacs_env_sub, para_env_sub, BLACS_GRID_SQUARE, .TRUE.)
    1286             :          END IF
    1287             :       END IF
    1288             : 
    1289             :       ! This should be done differently! Copied from MP2 code
    1290         106 :       CALL get_qs_env(qs_env, dft_control=dft_control)
    1291         318 :       ALLOCATE (qs_control)
    1292         106 :       qs_control_old => dft_control%qs_control
    1293         106 :       qs_control = qs_control_old
    1294         106 :       dft_control%qs_control => qs_control
    1295         106 :       progression_factor = qs_control%progression_factor
    1296         106 :       n_multigrid = SIZE(qs_control%e_cutoff)
    1297         106 :       nspins = SIZE(mos)
    1298             :       ! Allocate new cutoffs (just in private qs_control, not in qs_control_old)
    1299         318 :       ALLOCATE (qs_control%e_cutoff(n_multigrid))
    1300             : 
    1301         106 :       qs_control%cutoff = eri_env%eri_gpw%cutoff*0.5_dp
    1302         106 :       qs_control%e_cutoff(1) = qs_control%cutoff
    1303         424 :       DO i_multigrid = 2, n_multigrid
    1304             :          qs_control%e_cutoff(i_multigrid) = qs_control%e_cutoff(i_multigrid - 1) &
    1305         424 :                                             /progression_factor
    1306             :       END DO
    1307         106 :       qs_control%relative_cutoff = eri_env%eri_gpw%rel_cutoff*0.5_dp
    1308             : 
    1309         106 :       IF (eri_env%method == eri_method_gpw_ht) THEN
    1310             :          ! For now, we will distribute neighbor lists etc. within the global communicator
    1311          24 :          CALL get_qs_env(qs_env, ks_env=ks_env)
    1312             :          CALL create_mat_munu(mat_munu, qs_env, eri_env%eri_gpw%eps_grid, blacs_env_sub, sab_orb_sub=sab_orb_sub, &
    1313          24 :                               do_alloc_blocks_from_nbl=.TRUE., dbcsr_sym_type=dbcsr_type_symmetric)
    1314          24 :          CALL dbcsr_set(mat_munu%matrix, 0.0_dp)
    1315             :       END IF
    1316             : 
    1317             :       ! Generate the appropriate  pw_env
    1318         106 :       NULLIFY (pw_env_sub)
    1319         106 :       CALL pw_env_create(pw_env_sub)
    1320         106 :       CALL pw_env_rebuild(pw_env_sub, qs_env, external_para_env=para_env_sub)
    1321             :       CALL pw_env_get(pw_env_sub, auxbas_pw_pool=auxbas_pw_pool, &
    1322         106 :                       poisson_env=poisson_env)
    1323         106 :       IF (eri_env%eri_gpw%redo_poisson) THEN
    1324             :          ! We need to rebuild the Poisson solver on the fly
    1325         424 :          IF (SUM(eri_env%periodicity) /= 0) THEN
    1326           2 :             poisson_env%parameters%solver = pw_poisson_periodic
    1327             :          ELSE
    1328         104 :             poisson_env%parameters%solver = pw_poisson_analytic
    1329             :          END IF
    1330         424 :          poisson_env%parameters%periodic = eri_env%periodicity
    1331         106 :          CALL pw_poisson_rebuild(poisson_env)
    1332         106 :          IF (eri_env%cutoff_radius > 0.0_dp) THEN
    1333           0 :             poisson_env%green_fft%radius = eri_env%cutoff_radius
    1334             :          ELSE
    1335         106 :             CALL get_qs_env(qs_env, cell=cell)
    1336         106 :             rc = cell%hmat(1, 1)
    1337         424 :             DO iwa1 = 1, 3
    1338         424 :                rc = MIN(rc, 0.5_dp*cell%hmat(iwa1, iwa1))
    1339             :             END DO
    1340         106 :             poisson_env%green_fft%radius = rc
    1341             :          END IF
    1342             : 
    1343         106 :          CALL pw_eri_green_create(poisson_env%green_fft, eri_env)
    1344             : 
    1345         106 :          IF (iw > 0) THEN
    1346          53 :             CALL get_qs_env(qs_env, cell=cell)
    1347         371 :             IF (SUM(cell%perd) /= SUM(eri_env%periodicity)) THEN
    1348           0 :                IF (SUM(eri_env%periodicity) /= 0) THEN
    1349             :                   WRITE (UNIT=iw, FMT="(/,T2,A,T51,A30)") &
    1350           0 :                      "ERI_GPW| Switching Poisson solver to", "PERIODIC"
    1351             :                ELSE
    1352             :                   WRITE (UNIT=iw, FMT="(/,T2,A,T51,A30)") &
    1353           0 :                      "ERI_GPW| Switching Poisson solver to", "ANALYTIC"
    1354             :                END IF
    1355             :             END IF
    1356             :             ! print out the Greens function to check it matches the Poisson solver
    1357          54 :             SELECT CASE (poisson_env%green_fft%method)
    1358             :             CASE (PERIODIC3D)
    1359             :                WRITE (UNIT=iw, FMT="(T2,A,T51,A30)") &
    1360           1 :                   "ERI_GPW| Poisson Greens function", "PERIODIC"
    1361             :             CASE (ANALYTIC0D)
    1362             :                WRITE (UNIT=iw, FMT="(T2,A,T51,A30)") &
    1363          52 :                   "ERI_GPW| Poisson Greens function", "ANALYTIC"
    1364             :             CASE DEFAULT
    1365          53 :                CPABORT("Wrong Greens function setup")
    1366             :             END SELECT
    1367             :          END IF
    1368             :       END IF
    1369             : 
    1370         106 :       IF (eri_env%method == eri_method_gpw_ht) THEN
    1371             :          ! We need a task list
    1372          24 :          NULLIFY (task_list_sub)
    1373          24 :          skip_load_balance_distributed = dft_control%qs_control%skip_load_balance_distributed
    1374          24 :          CALL allocate_task_list(task_list_sub)
    1375             :          CALL generate_qs_task_list(ks_env, task_list_sub, &
    1376             :                                     reorder_rs_grid_ranks=.TRUE., soft_valid=.FALSE., &
    1377             :                                     skip_load_balance_distributed=skip_load_balance_distributed, &
    1378          24 :                                     pw_env_external=pw_env_sub, sab_orb_external=sab_orb_sub)
    1379             : 
    1380             :          ! Create sparse matrices carrying the matrix products, Code borrowed from the MP2 GPW method
    1381             :          ! Create equal distributions for them (no sparsity present)
    1382             :          ! We use the routines from mp2 suggesting that one may replicate the grids later for better performance
    1383             :          ALLOCATE (mo_coeff_as(nspins), matrix_pq_rnu(nspins), &
    1384         312 :                    fm_matrix_pq_rnu(nspins), fm_mo_coeff_as(nspins), fm_matrix_pq_rs(nspins))
    1385          48 :          DO ispin = 1, nspins
    1386          72 :             BLOCK
    1387          24 :                REAL(KIND=dp), DIMENSION(:, :), ALLOCATABLE :: C
    1388          24 :                TYPE(group_dist_d1_type) :: gd_array
    1389             :                TYPE(cp_fm_type), POINTER :: mo_coeff
    1390          24 :                CALL get_mo_set(mos(ispin), mo_coeff=mo_coeff)
    1391             :                CALL grep_rows_in_subgroups(para_env, para_env_sub, mo_coeff, gd_array, C)
    1392             : 
    1393             :                CALL build_dbcsr_from_rows(para_env_sub, mo_coeff_as(ispin), &
    1394             :                                           C(:, MINVAL(orbitals(:, ispin)):MAXVAL(orbitals(:, ispin))), &
    1395         120 :                                           mat_munu%matrix, gd_array, eri_env%eri_gpw%eps_filter)
    1396          24 :                CALL release_group_dist(gd_array)
    1397          72 :                DEALLOCATE (C)
    1398             :             END BLOCK
    1399             : 
    1400          24 :             CALL dbcsr_create(matrix_pq_rnu(ispin), template=mo_coeff_as(ispin))
    1401          24 :             CALL dbcsr_set(matrix_pq_rnu(ispin), 0.0_dp)
    1402             : 
    1403          24 :             CALL dbcsr_get_info(matrix_pq_rnu(ispin), nfullrows_total=nrow_global, nfullcols_total=ncol_global)
    1404             : 
    1405          24 :             NULLIFY (fm_struct)
    1406             :             CALL cp_fm_struct_create(fm_struct, context=blacs_env, para_env=para_env_sub, &
    1407          24 :                                      nrow_global=nrow_global, ncol_global=ncol_global)
    1408          24 :             CALL cp_fm_create(fm_matrix_pq_rnu(ispin), fm_struct)
    1409          24 :             CALL cp_fm_create(fm_mo_coeff_as(ispin), fm_struct)
    1410          24 :             CALL cp_fm_struct_release(fm_struct)
    1411             : 
    1412          24 :             CALL copy_dbcsr_to_fm(mo_coeff_as(ispin), fm_mo_coeff_as(ispin))
    1413             : 
    1414          24 :             NULLIFY (fm_struct)
    1415             :             CALL cp_fm_struct_create(fm_struct, context=blacs_env, para_env=para_env_sub, &
    1416          24 :                                      nrow_global=ncol_global, ncol_global=ncol_global)
    1417          24 :             CALL cp_fm_create(fm_matrix_pq_rs(ispin), fm_struct)
    1418          72 :             CALL cp_fm_struct_release(fm_struct)
    1419             :          END DO
    1420             : 
    1421             :          ! Copy the active space of the MOs into DBCSR matrices
    1422             :       END IF
    1423             : 
    1424         106 :       CALL auxbas_pw_pool%create_pw(wfn_r)
    1425         106 :       CALL auxbas_pw_pool%create_pw(rho_g)
    1426             :       CALL get_qs_env(qs_env, qs_kind_set=qs_kind_set, cell=cell, &
    1427         106 :                       particle_set=particle_set, atomic_kind_set=atomic_kind_set)
    1428             : 
    1429         106 :       IF (eri_env%eri_gpw%store_wfn) THEN
    1430             :          ! pre-calculate wavefunctions on reals space grid
    1431          82 :          rsize = 0.0_dp
    1432          82 :          nmo = 0
    1433         182 :          DO ispin = 1, nspins
    1434         100 :             CALL get_mo_set(mo_set=mos(ispin), nmo=nx)
    1435         100 :             nmo = MAX(nmo, nx)
    1436         482 :             rsize = REAL(SIZE(wfn_r%array), KIND=dp)*nx
    1437             :          END DO
    1438          82 :          IF (print1 .AND. iw > 0) THEN
    1439           1 :             rsize = rsize*8._dp/1000000._dp
    1440           1 :             WRITE (iw, "(T4,'ERI_GPW|',' Store active orbitals on real space grid ',T63,F12.3,' MB')") rsize
    1441             :          END IF
    1442         788 :          ALLOCATE (wfn_a(nmo, nspins))
    1443         182 :          DO ispin = 1, nspins
    1444         100 :             CALL get_mo_set(mo_set=mos(ispin), mo_coeff=mo_coeff, nmo=nmo)
    1445         452 :             DO i1 = 1, SIZE(orbitals, 1)
    1446         270 :                iwfn = orbitals(i1, ispin)
    1447         270 :                CALL auxbas_pw_pool%create_pw(wfn_a(iwfn, ispin))
    1448             :                CALL calculate_wavefunction(mo_coeff, iwfn, wfn_a(iwfn, ispin), rho_g, atomic_kind_set, &
    1449         270 :                                            qs_kind_set, cell, dft_control, particle_set, pw_env_sub)
    1450         370 :                IF (print2 .AND. iw > 0) THEN
    1451           0 :                   WRITE (iw, "(T4,'ERI_GPW|',' Orbital stored ',I4,'  Spin ',i1)") iwfn, ispin
    1452             :                END IF
    1453             :             END DO
    1454             :          END DO
    1455             :       ELSE
    1456             :          ! Even if we do not store all WFNs, we still need containers for the functions to store
    1457          24 :          ALLOCATE (wfn1, wfn2)
    1458          24 :          CALL auxbas_pw_pool%create_pw(wfn1)
    1459          24 :          CALL auxbas_pw_pool%create_pw(wfn2)
    1460          24 :          IF (eri_env%method /= eri_method_gpw_ht) THEN
    1461          12 :             ALLOCATE (wfn3, wfn4)
    1462          12 :             CALL auxbas_pw_pool%create_pw(wfn3)
    1463          12 :             CALL auxbas_pw_pool%create_pw(wfn4)
    1464             :          END IF
    1465             :       END IF
    1466             : 
    1467             :       ! get some of the grids ready
    1468         106 :       CALL auxbas_pw_pool%create_pw(rho_r)
    1469         106 :       CALL auxbas_pw_pool%create_pw(pot_g)
    1470             : 
    1471             :       ! run the FFT once, to set up buffers and to take into account the memory
    1472         106 :       CALL pw_zero(rho_r)
    1473         106 :       CALL pw_transfer(rho_r, rho_g)
    1474         106 :       dvol = rho_r%pw_grid%dvol
    1475         106 :       CALL mp_group%set_handle(eri_env%eri(1)%csr_mat%mp_group%get_handle())
    1476             : 
    1477             :       ! calculate the integrals
    1478         106 :       stored_integrals = 0
    1479         230 :       DO isp1 = 1, nspins
    1480         124 :          CALL get_mo_set(mo_set=mos(isp1), nmo=nmo1, mo_coeff=mo_coeff1)
    1481         124 :          nmm = (nmo1*(nmo1 + 1))/2
    1482         124 :          IF (eri_env%method == eri_method_gpw_ht) THEN
    1483          72 :             irange = [1, nmm]
    1484             :          ELSE
    1485         100 :             irange = get_irange_csr(nmm, para_env)
    1486             :          END IF
    1487         124 :          irange_sub = get_limit(nmm, number_of_subgroups, color)
    1488         672 :          DO i1 = 1, SIZE(orbitals, 1)
    1489         318 :             iwa1 = orbitals(i1, isp1)
    1490         318 :             IF (eri_env%eri_gpw%store_wfn) THEN
    1491         270 :                wfn1 => wfn_a(iwa1, isp1)
    1492             :             ELSE
    1493             :                CALL calculate_wavefunction(mo_coeff1, iwa1, wfn1, rho_g, atomic_kind_set, &
    1494          48 :                                            qs_kind_set, cell, dft_control, particle_set, pw_env_sub)
    1495             :             END IF
    1496        1048 :             DO i2 = i1, SIZE(orbitals, 1)
    1497         606 :                iwa2 = orbitals(i2, isp1)
    1498         606 :                iwa12 = csr_idx_to_combined(iwa1, iwa2, nmo1)
    1499             :                ! Skip calculation directly if the pair is not part of our subgroup
    1500         606 :                IF (iwa12 < irange_sub(1) .OR. iwa12 > irange_sub(2)) CYCLE
    1501         606 :                IF (iwa12 >= irange(1) .AND. iwa12 <= irange(2)) THEN
    1502         339 :                   iwa12 = iwa12 - irange(1) + 1
    1503             :                ELSE
    1504         267 :                   iwa12 = 0
    1505             :                END IF
    1506         606 :                IF (eri_env%eri_gpw%store_wfn) THEN
    1507         534 :                   wfn2 => wfn_a(iwa2, isp1)
    1508             :                ELSE
    1509             :                   CALL calculate_wavefunction(mo_coeff1, iwa2, wfn2, rho_g, atomic_kind_set, &
    1510          72 :                                               qs_kind_set, cell, dft_control, particle_set, pw_env_sub)
    1511             :                END IF
    1512             :                ! calculate charge distribution and potential
    1513         606 :                CALL pw_zero(rho_r)
    1514         606 :                CALL pw_multiply(rho_r, wfn1, wfn2)
    1515         606 :                IF (print2) THEN
    1516           0 :                   erint = pw_integrate_function(rho_r)/dvol
    1517           0 :                   IF (iw > 0) THEN
    1518             :                      WRITE (iw, "(T4,'ERI_GPW| Integral rho_ab ',T32,2I4,' [',I1,']',T58,G20.14)") &
    1519           0 :                         iwa1, iwa2, isp1, erint
    1520             :                   END IF
    1521             :                END IF
    1522         606 :                CALL pw_transfer(rho_r, rho_g)
    1523         606 :                CALL pw_poisson_solve(poisson_env, rho_g, pair_int, pot_g)
    1524             :                ! screening using pair_int
    1525         606 :                IF (pair_int < eri_env%eps_integral) CYCLE
    1526         606 :                CALL pw_transfer(pot_g, rho_r)
    1527             :                !
    1528        1530 :                IF (eri_env%method == eri_method_gpw_ht) THEN
    1529          72 :                   CALL pw_scale(rho_r, dvol)
    1530         144 :                   DO isp2 = isp1, nspins
    1531          72 :                      CALL get_mo_set(mo_set=mos(isp2), nmo=nmo2)
    1532          72 :                      nx = (nmo2*(nmo2 + 1))/2
    1533         360 :                      ALLOCATE (eri(nx), eri_index(nx))
    1534          72 :                      CALL dbcsr_set(mat_munu%matrix, 0.0_dp)
    1535             :                      CALL integrate_v_rspace(rho_r, hmat=mat_munu, qs_env=qs_env, &
    1536             :                                              calculate_forces=.FALSE., compute_tau=.FALSE., gapw=.FALSE., &
    1537          72 :                                              pw_env_external=pw_env_sub, task_list_external=task_list_sub)
    1538             : 
    1539             :                      CALL dbcsr_multiply("N", "N", 1.0_dp, mat_munu%matrix, mo_coeff_as(isp2), &
    1540          72 :                                          0.0_dp, matrix_pq_rnu(isp2), filter_eps=eri_env%eri_gpw%eps_filter)
    1541          72 :                      CALL copy_dbcsr_to_fm(matrix_pq_rnu(isp2), fm_matrix_pq_rnu(isp2))
    1542             : 
    1543          72 :                      CALL cp_fm_get_info(fm_matrix_pq_rnu(isp2), ncol_global=ncol_global, nrow_global=nrow_global)
    1544             : 
    1545             :                      CALL parallel_gemm("T", "N", ncol_global, ncol_global, nrow_global, 0.5_dp, &
    1546             :                                         fm_matrix_pq_rnu(isp2), fm_mo_coeff_as(isp2), &
    1547          72 :                                         0.0_dp, fm_matrix_pq_rs(isp2))
    1548             :                      CALL parallel_gemm("T", "N", ncol_global, ncol_global, nrow_global, 0.5_dp, &
    1549             :                                         fm_mo_coeff_as(isp2), fm_matrix_pq_rnu(isp2), &
    1550          72 :                                         1.0_dp, fm_matrix_pq_rs(isp2))
    1551             : 
    1552             :                      CALL cp_fm_get_info(fm_matrix_pq_rs(isp2), ncol_local=ncol_local, nrow_local=nrow_local, &
    1553          72 :                                          col_indices=col_indices, row_indices=row_indices)
    1554             : 
    1555          72 :                      icount2 = 0
    1556         216 :                      DO col_local = 1, ncol_local
    1557         144 :                         iwb2 = orbitals(col_indices(col_local), isp2)
    1558         360 :                         DO row_local = 1, nrow_local
    1559         144 :                            iwb1 = orbitals(row_indices(row_local), isp2)
    1560             : 
    1561         288 :                            IF (iwb1 <= iwb2) THEN
    1562         108 :                               iwb12 = csr_idx_to_combined(iwb1, iwb2, nmo2)
    1563         108 :                               erint = fm_matrix_pq_rs(isp2)%local_data(row_local, col_local)
    1564         108 :                               IF (ABS(erint) > eri_env%eps_integral .AND. (iwa12 <= iwb12 .OR. isp1 /= isp2)) THEN
    1565          60 :                                  icount2 = icount2 + 1
    1566          60 :                                  eri(icount2) = erint
    1567          60 :                                  eri_index(icount2) = iwb12
    1568             :                               END IF
    1569             :                            END IF
    1570             :                         END DO
    1571             :                      END DO
    1572          72 :                      stored_integrals = stored_integrals + icount2
    1573             :                      !
    1574          72 :                      isp = (isp1 - 1)*isp2 + (isp2 - isp1 + 1)
    1575          72 :                      CALL update_csr_matrix(eri_env%eri(isp)%csr_mat, icount2, eri, eri_index, iwa12)
    1576             :                      !
    1577         360 :                      DEALLOCATE (eri, eri_index)
    1578             :                   END DO
    1579         534 :                ELSEIF (eri_env%method == eri_method_full_gpw) THEN
    1580        1158 :                   DO isp2 = isp1, nspins
    1581         624 :                      CALL get_mo_set(mo_set=mos(isp2), nmo=nmo2, mo_coeff=mo_coeff2)
    1582         624 :                      nx = (nmo2*(nmo2 + 1))/2
    1583        3120 :                      ALLOCATE (eri(nx), eri_index(nx))
    1584         624 :                      icount2 = 0
    1585         624 :                      iwbs = 1
    1586         624 :                      IF (isp1 == isp2) iwbs = i1
    1587         624 :                      isp = (isp1 - 1)*isp2 + (isp2 - isp1 + 1)
    1588        2190 :                      DO i3 = iwbs, SIZE(orbitals, 1)
    1589        1566 :                         iwb1 = orbitals(i3, isp2)
    1590        1566 :                         IF (eri_env%eri_gpw%store_wfn) THEN
    1591        1506 :                            wfn3 => wfn_a(iwb1, isp2)
    1592             :                         ELSE
    1593             :                            CALL calculate_wavefunction(mo_coeff2, iwb1, wfn3, rho_g, atomic_kind_set, &
    1594          60 :                                                        qs_kind_set, cell, dft_control, particle_set, pw_env_sub)
    1595             :                         END IF
    1596        1566 :                         CALL pw_zero(wfn_r)
    1597        1566 :                         CALL pw_multiply(wfn_r, rho_r, wfn3)
    1598        1566 :                         iwbt = i3
    1599        1566 :                         IF (isp1 == isp2 .AND. i1 == i3) iwbt = i2
    1600        4884 :                         DO i4 = iwbt, SIZE(orbitals, 1)
    1601        2694 :                            iwb2 = orbitals(i4, isp2)
    1602        2694 :                            IF (eri_env%eri_gpw%store_wfn) THEN
    1603        2622 :                               wfn4 => wfn_a(iwb2, isp2)
    1604             :                            ELSE
    1605             :                               CALL calculate_wavefunction(mo_coeff2, iwb2, wfn4, rho_g, atomic_kind_set, &
    1606          72 :                                                           qs_kind_set, cell, dft_control, particle_set, pw_env_sub)
    1607             :                            END IF
    1608             :                            ! We reduce the amount of communication by collecting the local sums first and sum globally later
    1609        2694 :                            erint = pw_integral_ab(wfn_r, wfn4, local_only=.TRUE.)
    1610        2694 :                            icount2 = icount2 + 1
    1611        2694 :                            eri(icount2) = erint
    1612        4260 :                            eri_index(icount2) = csr_idx_to_combined(iwb1, iwb2, nmo2)
    1613             :                         END DO
    1614             :                      END DO
    1615             :                      ! Now, we sum the integrals globally
    1616         624 :                      CALL wfn_r%pw_grid%para%group%sum(eri)
    1617             :                      ! and we reorder the integrals to prevent storing too small integrals
    1618         624 :                      intcount = 0
    1619         624 :                      icount2 = 0
    1620             :                      iwbs = 1
    1621             :                      IF (isp1 == isp2) iwbs = i1
    1622         624 :                      isp = (isp1 - 1)*isp2 + (isp2 - isp1 + 1)
    1623        2190 :                      DO i3 = iwbs, SIZE(orbitals, 1)
    1624        1566 :                         iwb1 = orbitals(i3, isp2)
    1625        1566 :                         iwbt = i3
    1626        1566 :                         IF (isp1 == isp2 .AND. i1 == i3) iwbt = i2
    1627        4884 :                         DO i4 = iwbt, SIZE(orbitals, 1)
    1628        2694 :                            iwb2 = orbitals(i4, isp2)
    1629        2694 :                            intcount = intcount + 1
    1630        2694 :                            erint = eri(intcount)
    1631        4260 :                            IF (ABS(erint) > eri_env%eps_integral) THEN
    1632        2092 :                               icount2 = icount2 + 1
    1633        2092 :                               eri(icount2) = erint
    1634        2092 :                               eri_index(icount2) = eri_index(intcount)
    1635             :                            END IF
    1636             :                         END DO
    1637             :                      END DO
    1638         624 :                      stored_integrals = stored_integrals + icount2
    1639             :                      !
    1640         624 :                      CALL update_csr_matrix(eri_env%eri(isp)%csr_mat, icount2, eri, eri_index, iwa12)
    1641             :                      !
    1642        1782 :                      DEALLOCATE (eri, eri_index)
    1643             :                   END DO
    1644             :                ELSE
    1645           0 :                   CPABORT("Unknown option")
    1646             :                END IF
    1647             :             END DO
    1648             :          END DO
    1649             :       END DO
    1650             : 
    1651         106 :       IF (print1 .AND. iw > 0) THEN
    1652           1 :          WRITE (iw, "(T4,'ERI_GPW|',' Number of Integrals stored ',T68,I10)") stored_integrals
    1653             :       END IF
    1654             : 
    1655         106 :       IF (eri_env%eri_gpw%store_wfn) THEN
    1656         182 :          DO ispin = 1, nspins
    1657         452 :             DO i1 = 1, SIZE(orbitals, 1)
    1658         270 :                iwfn = orbitals(i1, ispin)
    1659         370 :                CALL wfn_a(iwfn, ispin)%release()
    1660             :             END DO
    1661             :          END DO
    1662          82 :          DEALLOCATE (wfn_a)
    1663             :       ELSE
    1664          24 :          CALL wfn1%release()
    1665          24 :          CALL wfn2%release()
    1666          24 :          DEALLOCATE (wfn1, wfn2)
    1667          24 :          IF (eri_env%method /= eri_method_gpw_ht) THEN
    1668          12 :             CALL wfn3%release()
    1669          12 :             CALL wfn4%release()
    1670          12 :             DEALLOCATE (wfn3, wfn4)
    1671             :          END IF
    1672             :       END IF
    1673         106 :       CALL auxbas_pw_pool%give_back_pw(wfn_r)
    1674         106 :       CALL auxbas_pw_pool%give_back_pw(rho_g)
    1675         106 :       CALL auxbas_pw_pool%give_back_pw(rho_r)
    1676         106 :       CALL auxbas_pw_pool%give_back_pw(pot_g)
    1677         106 :       CALL mp_para_env_release(para_env_sub)
    1678             : 
    1679         106 :       IF (eri_env%method == eri_method_gpw_ht) THEN
    1680          48 :          DO ispin = 1, nspins
    1681          24 :             CALL dbcsr_release(mo_coeff_as(ispin))
    1682          24 :             CALL dbcsr_release(matrix_pq_rnu(ispin))
    1683          24 :             CALL cp_fm_release(fm_mo_coeff_as(ispin))
    1684          24 :             CALL cp_fm_release(fm_matrix_pq_rnu(ispin))
    1685          48 :             CALL cp_fm_release(fm_matrix_pq_rs(ispin))
    1686             :          END DO
    1687           0 :          DEALLOCATE (mo_coeff_as, matrix_pq_rnu, &
    1688          24 :                      fm_mo_coeff_as, fm_matrix_pq_rnu, fm_matrix_pq_rs)
    1689          24 :          CALL deallocate_task_list(task_list_sub)
    1690          24 :          CALL dbcsr_release(mat_munu%matrix)
    1691          24 :          DEALLOCATE (mat_munu%matrix)
    1692          24 :          CALL release_neighbor_list_sets(sab_orb_sub)
    1693          24 :          CALL cp_blacs_env_release(blacs_env_sub)
    1694             :       END IF
    1695         106 :       CALL pw_env_release(pw_env_sub)
    1696             :       ! Return to the old qs_control
    1697         106 :       dft_control%qs_control => qs_control_old
    1698         106 :       DEALLOCATE (qs_control%e_cutoff)
    1699         106 :       DEALLOCATE (qs_control)
    1700             : 
    1701         106 :       CALL timestop(handle)
    1702             : 
    1703         212 :    END SUBROUTINE calculate_eri_gpw
    1704             : 
    1705             : ! **************************************************************************************************
    1706             : !> \brief Sets the Green's function
    1707             : !> \param green ...
    1708             : !> \param eri_env ...
    1709             : !> \par History
    1710             : !>      04.2016 created [JGH]
    1711             : ! **************************************************************************************************
    1712         106 :    SUBROUTINE pw_eri_green_create(green, eri_env)
    1713             : 
    1714             :       TYPE(greens_fn_type), INTENT(INOUT)                :: green
    1715             :       TYPE(eri_type)                                     :: eri_env
    1716             : 
    1717             :       INTEGER                                            :: ig
    1718             :       REAL(KIND=dp)                                      :: a, ea, g2, g3d, ga, gg, rg, rlength
    1719             : 
    1720             :       ! initialize influence function
    1721             :       ASSOCIATE (gf => green%influence_fn, grid => green%influence_fn%pw_grid)
    1722         108 :          SELECT CASE (green%method)
    1723             :          CASE (PERIODIC3D)
    1724             : 
    1725         108 :             SELECT CASE (eri_env%operator)
    1726             :             CASE (eri_operator_coulomb)
    1727      262145 :                DO ig = grid%first_gne0, grid%ngpts_cut_local
    1728      262143 :                   g2 = grid%gsq(ig)
    1729      262145 :                   gf%array(ig) = fourpi/g2
    1730             :                END DO
    1731           2 :                IF (grid%have_g0) gf%array(1) = 0.0_dp
    1732             :             CASE (eri_operator_yukawa)
    1733           0 :                a = eri_env%operator_parameter**2
    1734           0 :                DO ig = grid%first_gne0, grid%ngpts_cut_local
    1735           0 :                   g2 = grid%gsq(ig)
    1736           0 :                   gf%array(ig) = fourpi/(a + g2)
    1737             :                END DO
    1738           0 :                IF (grid%have_g0) gf%array(1) = fourpi/a
    1739             :             CASE (eri_operator_erf)
    1740           0 :                a = eri_env%operator_parameter**2
    1741           0 :                DO ig = grid%first_gne0, grid%ngpts_cut_local
    1742           0 :                   g2 = grid%gsq(ig)
    1743           0 :                   ga = -0.25_dp*g2/a
    1744           0 :                   gf%array(ig) = fourpi/g2*EXP(ga)
    1745             :                END DO
    1746           0 :                IF (grid%have_g0) gf%array(1) = 0.0_dp
    1747             :             CASE (eri_operator_erfc)
    1748           0 :                a = eri_env%operator_parameter**2
    1749           0 :                DO ig = grid%first_gne0, grid%ngpts_cut_local
    1750           0 :                   g2 = grid%gsq(ig)
    1751           0 :                   ga = -0.25_dp*g2/a
    1752           0 :                   gf%array(ig) = fourpi/g2*(1._dp - EXP(ga))
    1753             :                END DO
    1754           0 :                IF (grid%have_g0) gf%array(1) = 0.25_dp*fourpi/a
    1755             :             CASE (eri_operator_trunc)
    1756           0 :                a = eri_env%operator_parameter
    1757           0 :                DO ig = grid%first_gne0, grid%ngpts_cut_local
    1758           0 :                   g2 = grid%gsq(ig)
    1759           0 :                   ga = SQRT(g2)*a
    1760           0 :                   IF (ga >= 0.005_dp) THEN
    1761           0 :                      gf%array(ig) = fourpi/g2*(1.0_dp - COS(ga))
    1762             :                   ELSE
    1763           0 :                      gf%array(ig) = fourpi/g2*ga**2/2.0_dp*(1.0_dp - ga**2/12.0_dp)
    1764             :                   END IF
    1765             :                END DO
    1766           0 :                IF (grid%have_g0) gf%array(1) = 0.0_dp
    1767             :             CASE (eri_operator_gaussian)
    1768           0 :                CPABORT("")
    1769             :             CASE DEFAULT
    1770           2 :                CPABORT("")
    1771             :             END SELECT
    1772             : 
    1773             :          CASE (ANALYTIC0D)
    1774             : 
    1775         194 :             SELECT CASE (eri_env%operator)
    1776             :             CASE (eri_operator_coulomb)
    1777          90 :                rlength = green%radius
    1778    24643161 :                DO ig = grid%first_gne0, grid%ngpts_cut_local
    1779    24643071 :                   g2 = grid%gsq(ig)
    1780    24643071 :                   gg = SQRT(g2)
    1781    24643071 :                   g3d = fourpi/g2
    1782    24643161 :                   gf%array(ig) = g3d*(1.0_dp - COS(rlength*gg))
    1783             :                END DO
    1784          90 :                IF (grid%have_g0) gf%array(1) = 0.5_dp*fourpi*rlength*rlength
    1785             :             CASE (eri_operator_yukawa)
    1786           0 :                rlength = green%radius
    1787           0 :                a = eri_env%operator_parameter
    1788           0 :                ea = EXP(-a*rlength)
    1789           0 :                DO ig = grid%first_gne0, grid%ngpts_cut_local
    1790           0 :                   g2 = grid%gsq(ig)
    1791           0 :                   gg = SQRT(g2)
    1792           0 :                   g3d = fourpi/(a*a + g2)
    1793           0 :                   rg = rlength*gg
    1794           0 :                   gf%array(ig) = g3d*(1.0_dp - ea*(COS(rg) + a/gg*SIN(rg)))
    1795             :                END DO
    1796           0 :                IF (grid%have_g0) gf%array(1) = fourpi/(a*a)*(1.0_dp - ea*(1._dp + a*rlength))
    1797             :             CASE (eri_operator_erf)
    1798          14 :                rlength = green%radius
    1799          14 :                a = eri_env%operator_parameter**2
    1800     1512007 :                DO ig = grid%first_gne0, grid%ngpts_cut_local
    1801     1511993 :                   g2 = grid%gsq(ig)
    1802     1511993 :                   gg = SQRT(g2)
    1803     1511993 :                   ga = -0.25_dp*g2/a
    1804     1512007 :                   gf%array(ig) = fourpi/g2*EXP(ga)*(1.0_dp - COS(rlength*gg))
    1805             :                END DO
    1806          14 :                IF (grid%have_g0) gf%array(1) = 0.5_dp*fourpi*rlength*rlength
    1807             :             CASE (eri_operator_erfc)
    1808           0 :                rlength = green%radius
    1809           0 :                a = eri_env%operator_parameter**2
    1810           0 :                DO ig = grid%first_gne0, grid%ngpts_cut_local
    1811           0 :                   g2 = grid%gsq(ig)
    1812           0 :                   gg = SQRT(g2)
    1813           0 :                   ga = -0.25_dp*g2/a
    1814           0 :                   gf%array(ig) = fourpi/g2*(1._dp - EXP(ga))*(1.0_dp - COS(rlength*gg))
    1815             :                END DO
    1816           0 :                IF (grid%have_g0) gf%array(1) = 0._dp
    1817             :             CASE (eri_operator_trunc)
    1818           0 :                a = eri_env%operator_parameter
    1819           0 :                DO ig = grid%first_gne0, grid%ngpts_cut_local
    1820           0 :                   g2 = grid%gsq(ig)
    1821           0 :                   ga = SQRT(g2)*a
    1822           0 :                   IF (ga >= 0.005_dp) THEN
    1823           0 :                      gf%array(ig) = fourpi/g2*(1.0_dp - COS(ga))
    1824             :                   ELSE
    1825           0 :                      gf%array(ig) = fourpi/g2*ga**2/2.0_dp*(1.0_dp - ga**2/12.0_dp)
    1826             :                   END IF
    1827             :                END DO
    1828           0 :                IF (grid%have_g0) gf%array(1) = 0.0_dp
    1829             :             CASE (eri_operator_gaussian)
    1830           0 :                CPABORT("")
    1831             :             CASE DEFAULT
    1832         104 :                CPABORT("")
    1833             :             END SELECT
    1834             : 
    1835             :          CASE DEFAULT
    1836         106 :             CPABORT("")
    1837             :          END SELECT
    1838             :       END ASSOCIATE
    1839             : 
    1840         106 :    END SUBROUTINE pw_eri_green_create
    1841             : 
    1842             : ! **************************************************************************************************
    1843             : !> \brief Adds data for a new row to the csr matrix
    1844             : !> \param csr_mat ...
    1845             : !> \param nnz ...
    1846             : !> \param rdat ...
    1847             : !> \param rind ...
    1848             : !> \param irow ...
    1849             : !> \par History
    1850             : !>      04.2016 created [JGH]
    1851             : ! **************************************************************************************************
    1852         696 :    SUBROUTINE update_csr_matrix(csr_mat, nnz, rdat, rind, irow)
    1853             : 
    1854             :       TYPE(dbcsr_csr_type), INTENT(INOUT)                :: csr_mat
    1855             :       INTEGER, INTENT(IN)                                :: nnz
    1856             :       REAL(KIND=dp), DIMENSION(:), INTENT(IN)            :: rdat
    1857             :       INTEGER, DIMENSION(:), INTENT(IN)                  :: rind
    1858             :       INTEGER, INTENT(IN)                                :: irow
    1859             : 
    1860             :       INTEGER                                            :: k, nrow, nze, nze_new
    1861             : 
    1862         696 :       IF (irow /= 0) THEN
    1863         384 :          nze = csr_mat%nze_local
    1864         384 :          nze_new = nze + nnz
    1865             :          ! values
    1866         384 :          CALL reallocate(csr_mat%nzval_local%r_dp, 1, nze_new)
    1867        1490 :          csr_mat%nzval_local%r_dp(nze + 1:nze_new) = rdat(1:nnz)
    1868             :          ! col indices
    1869         384 :          CALL reallocate(csr_mat%colind_local, 1, nze_new)
    1870        1490 :          csr_mat%colind_local(nze + 1:nze_new) = rind(1:nnz)
    1871             :          ! rows
    1872         384 :          nrow = csr_mat%nrows_local
    1873         384 :          CALL reallocate(csr_mat%rowptr_local, 1, irow + 1)
    1874         851 :          csr_mat%rowptr_local(nrow + 1:irow) = nze + 1
    1875         384 :          csr_mat%rowptr_local(irow + 1) = nze_new + 1
    1876             :          ! nzerow
    1877         384 :          CALL reallocate(csr_mat%nzerow_local, 1, irow)
    1878         851 :          DO k = nrow + 1, irow
    1879         851 :             csr_mat%nzerow_local(k) = csr_mat%rowptr_local(k + 1) - csr_mat%rowptr_local(k)
    1880             :          END DO
    1881         384 :          csr_mat%nrows_local = irow
    1882         384 :          csr_mat%nze_local = csr_mat%nze_local + nnz
    1883             :       END IF
    1884         696 :       csr_mat%nze_total = csr_mat%nze_total + nnz
    1885         696 :       csr_mat%has_indices = .TRUE.
    1886             : 
    1887         696 :    END SUBROUTINE update_csr_matrix
    1888             : 
    1889             : ! **************************************************************************************************
    1890             : !> \brief Computes and prints the active orbitals on Cube Files
    1891             : !> \param input ...
    1892             : !> \param qs_env the qs_env in which the qs_env lives
    1893             : !> \param mos ...
    1894             : ! **************************************************************************************************
    1895           2 :    SUBROUTINE print_orbital_cubes(input, qs_env, mos)
    1896             :       TYPE(section_vals_type), POINTER                   :: input
    1897             :       TYPE(qs_environment_type), POINTER                 :: qs_env
    1898             :       TYPE(mo_set_type), DIMENSION(:), INTENT(IN)        :: mos
    1899             : 
    1900             :       CHARACTER(LEN=default_path_length)                 :: filebody, filename, title
    1901             :       INTEGER                                            :: i, imo, isp, nmo, str(3), unit_nr
    1902           2 :       INTEGER, DIMENSION(:), POINTER                     :: alist, blist, istride
    1903             :       LOGICAL                                            :: do_mo, explicit_a, explicit_b
    1904           2 :       TYPE(atomic_kind_type), DIMENSION(:), POINTER      :: atomic_kind_set
    1905             :       TYPE(cell_type), POINTER                           :: cell
    1906             :       TYPE(cp_fm_type), POINTER                          :: mo_coeff
    1907             :       TYPE(dft_control_type), POINTER                    :: dft_control
    1908             :       TYPE(mp_para_env_type), POINTER                    :: para_env
    1909             :       TYPE(particle_list_type), POINTER                  :: particles
    1910           2 :       TYPE(particle_type), DIMENSION(:), POINTER         :: particle_set
    1911             :       TYPE(pw_c1d_gs_type)                               :: wf_g
    1912             :       TYPE(pw_env_type), POINTER                         :: pw_env
    1913             :       TYPE(pw_pool_type), POINTER                        :: auxbas_pw_pool
    1914             :       TYPE(pw_r3d_rs_type)                               :: wf_r
    1915           2 :       TYPE(qs_kind_type), DIMENSION(:), POINTER          :: qs_kind_set
    1916             :       TYPE(qs_subsys_type), POINTER                      :: subsys
    1917             :       TYPE(section_vals_type), POINTER                   :: dft_section, scf_input
    1918             : 
    1919           2 :       CALL section_vals_val_get(input, "FILENAME", c_val=filebody)
    1920           2 :       CALL section_vals_val_get(input, "STRIDE", i_vals=istride)
    1921           2 :       IF (SIZE(istride) == 1) THEN
    1922           8 :          str(1:3) = istride(1)
    1923           0 :       ELSEIF (SIZE(istride) == 3) THEN
    1924           0 :          str(1:3) = istride(1:3)
    1925             :       ELSE
    1926           0 :          CPABORT("STRIDE arguments inconsistent")
    1927             :       END IF
    1928           2 :       CALL section_vals_val_get(input, "ALIST", i_vals=alist, explicit=explicit_a)
    1929           2 :       CALL section_vals_val_get(input, "BLIST", i_vals=blist, explicit=explicit_b)
    1930             : 
    1931             :       CALL get_qs_env(qs_env=qs_env, &
    1932             :                       dft_control=dft_control, &
    1933             :                       para_env=para_env, &
    1934             :                       subsys=subsys, &
    1935             :                       atomic_kind_set=atomic_kind_set, &
    1936             :                       qs_kind_set=qs_kind_set, &
    1937             :                       cell=cell, &
    1938             :                       particle_set=particle_set, &
    1939             :                       pw_env=pw_env, &
    1940           2 :                       input=scf_input)
    1941             : 
    1942           2 :       CALL qs_subsys_get(subsys, particles=particles)
    1943             :       !
    1944           2 :       CALL pw_env_get(pw_env, auxbas_pw_pool=auxbas_pw_pool)
    1945           2 :       CALL auxbas_pw_pool%create_pw(wf_r)
    1946           2 :       CALL auxbas_pw_pool%create_pw(wf_g)
    1947             :       !
    1948           2 :       dft_section => section_vals_get_subs_vals(scf_input, "DFT")
    1949             :       !
    1950           4 :       DO isp = 1, SIZE(mos)
    1951           2 :          CALL get_mo_set(mo_set=mos(isp), mo_coeff=mo_coeff, nmo=nmo)
    1952             : 
    1953           2 :          IF (SIZE(mos) > 1) THEN
    1954           0 :             SELECT CASE (isp)
    1955             :             CASE (1)
    1956             :                CALL write_mo_set_to_output_unit(mos(isp), atomic_kind_set, qs_kind_set, particle_set, &
    1957           0 :                                                 dft_section, 4, 0, final_mos=.TRUE., spin="ALPHA")
    1958             :             CASE (2)
    1959             :                CALL write_mo_set_to_output_unit(mos(isp), atomic_kind_set, qs_kind_set, particle_set, &
    1960           0 :                                                 dft_section, 4, 0, final_mos=.TRUE., spin="BETA")
    1961             :             CASE DEFAULT
    1962           0 :                CPABORT("Invalid spin")
    1963             :             END SELECT
    1964             :          ELSE
    1965             :             CALL write_mo_set_to_output_unit(mos(isp), atomic_kind_set, qs_kind_set, particle_set, &
    1966           2 :                                              dft_section, 4, 0, final_mos=.TRUE.)
    1967             :          END IF
    1968             : 
    1969          22 :          DO imo = 1, nmo
    1970          16 :             IF (isp == 1 .AND. explicit_a) THEN
    1971          16 :                IF (alist(1) == -1) THEN
    1972             :                   do_mo = .TRUE.
    1973             :                ELSE
    1974          16 :                   do_mo = .FALSE.
    1975          64 :                   DO i = 1, SIZE(alist)
    1976          64 :                      IF (imo == alist(i)) do_mo = .TRUE.
    1977             :                   END DO
    1978             :                END IF
    1979           0 :             ELSE IF (isp == 2 .AND. explicit_b) THEN
    1980           0 :                IF (blist(1) == -1) THEN
    1981             :                   do_mo = .TRUE.
    1982             :                ELSE
    1983           0 :                   do_mo = .FALSE.
    1984           0 :                   DO i = 1, SIZE(blist)
    1985           0 :                      IF (imo == blist(i)) do_mo = .TRUE.
    1986             :                   END DO
    1987             :                END IF
    1988             :             ELSE
    1989             :                do_mo = .TRUE.
    1990             :             END IF
    1991          16 :             IF (.NOT. do_mo) CYCLE
    1992             :             CALL calculate_wavefunction(mo_coeff, imo, wf_r, wf_g, atomic_kind_set, &
    1993           6 :                                         qs_kind_set, cell, dft_control, particle_set, pw_env)
    1994           6 :             IF (para_env%is_source()) THEN
    1995           3 :                WRITE (filename, '(A,A1,I4.4,A1,I1.1,A)') TRIM(filebody), "_", imo, "_", isp, ".cube"
    1996           3 :                CALL open_file(filename, unit_number=unit_nr, file_status="UNKNOWN", file_action="WRITE")
    1997           3 :                WRITE (title, *) "Active Orbital ", imo, " spin ", isp
    1998             :             ELSE
    1999           3 :                unit_nr = -1
    2000             :             END IF
    2001           6 :             CALL cp_pw_to_cube(wf_r, unit_nr, title, particles=particles, stride=istride)
    2002           8 :             IF (para_env%is_source()) THEN
    2003          13 :                CALL close_file(unit_nr)
    2004             :             END IF
    2005             :          END DO
    2006             :       END DO
    2007             : 
    2008           2 :       CALL auxbas_pw_pool%give_back_pw(wf_r)
    2009           2 :       CALL auxbas_pw_pool%give_back_pw(wf_g)
    2010             : 
    2011           2 :    END SUBROUTINE print_orbital_cubes
    2012             : 
    2013             : ! **************************************************************************************************
    2014             : !> \brief Writes a FCIDUMP file
    2015             : !> \param active_space_env ...
    2016             : !> \param as_input ...
    2017             : !> \par History
    2018             : !>      04.2016 created [JGH]
    2019             : ! **************************************************************************************************
    2020         106 :    SUBROUTINE fcidump(active_space_env, as_input)
    2021             : 
    2022             :       TYPE(active_space_type), POINTER                   :: active_space_env
    2023             :       TYPE(section_vals_type), POINTER                   :: as_input
    2024             : 
    2025             :       INTEGER                                            :: i, i1, i2, i3, i4, isym, iw, m1, m2, &
    2026             :                                                             nmo, norb, nspins
    2027             :       REAL(KIND=dp)                                      :: checksum, esub
    2028         106 :       REAL(KIND=dp), ALLOCATABLE, DIMENSION(:, :)        :: fmat
    2029             :       TYPE(cp_logger_type), POINTER                      :: logger
    2030             :       TYPE(eri_fcidump_checksum)                         :: eri_checksum
    2031             : 
    2032         106 :       checksum = 0.0_dp
    2033             : 
    2034         212 :       logger => cp_get_default_logger()
    2035             :       iw = cp_print_key_unit_nr(logger, as_input, "FCIDUMP", &
    2036         106 :                                 extension=".fcidump", file_status="REPLACE", file_action="WRITE", file_form="FORMATTED")
    2037             :       !
    2038         106 :       nspins = active_space_env%nspins
    2039         106 :       norb = SIZE(active_space_env%active_orbitals, 1)
    2040         106 :       IF (nspins == 1) THEN
    2041             :          ASSOCIATE (ms2 => active_space_env%multiplicity, &
    2042             :                     nelec => active_space_env%nelec_active)
    2043             : 
    2044          88 :             IF (iw > 0) THEN
    2045          44 :                WRITE (iw, "(A,A,I4,A,I4,A,I2,A)") "&FCI", " NORB=", norb, ",NELEC=", nelec, ",MS2=", ms2, ","
    2046          44 :                isym = 1
    2047         155 :                WRITE (iw, "(A,1000(I1,','))") "  ORBSYM=", (isym, i=1, norb)
    2048          44 :                isym = 0
    2049          44 :                WRITE (iw, "(A,I1,A)") "  ISYM=", isym, ","
    2050          44 :                WRITE (iw, "(A)") " /"
    2051             :             END IF
    2052             :             !
    2053             :             ! Print integrals: ERI
    2054             :             CALL active_space_env%eri%eri_foreach(1, active_space_env%active_orbitals, &
    2055          88 :                                                   eri_fcidump_print(iw, 1, 1), 1, 1)
    2056          88 :             CALL eri_checksum%set(1, 1)
    2057          88 :             CALL active_space_env%eri%eri_foreach(1, active_space_env%active_orbitals, eri_checksum, 1, 1)
    2058             : 
    2059             :             ! Print integrals: Fij
    2060             :             ! replicate Fock matrix
    2061          88 :             nmo = active_space_env%eri%norb
    2062         352 :             ALLOCATE (fmat(nmo, nmo))
    2063          88 :             CALL replicate_and_symmetrize_matrix(nmo, active_space_env%fock_sub(1), fmat)
    2064          88 :             IF (iw > 0) THEN
    2065          44 :                i3 = 0; i4 = 0
    2066         155 :                DO m1 = 1, SIZE(active_space_env%active_orbitals, 1)
    2067         111 :                   i1 = active_space_env%active_orbitals(m1, 1)
    2068         368 :                   DO m2 = m1, SIZE(active_space_env%active_orbitals, 1)
    2069         213 :                      i2 = active_space_env%active_orbitals(m2, 1)
    2070         213 :                      checksum = checksum + ABS(fmat(i1, i2))
    2071         324 :                      WRITE (iw, "(ES23.16,4I4)") fmat(i1, i2), m1, m2, i3, i4
    2072             :                   END DO
    2073             :                END DO
    2074             :             END IF
    2075          88 :             DEALLOCATE (fmat)
    2076             :             ! Print energy
    2077          88 :             esub = active_space_env%energy_inactive
    2078          88 :             i1 = 0; i2 = 0; i3 = 0; i4 = 0
    2079          88 :             checksum = checksum + ABS(esub)
    2080         176 :             IF (iw > 0) WRITE (iw, "(ES23.16,4I4)") esub, i1, i2, i3, i4
    2081             :          END ASSOCIATE
    2082             : 
    2083             :       ELSE
    2084             :          ASSOCIATE (ms2 => active_space_env%multiplicity, &
    2085             :                     nelec => active_space_env%nelec_active)
    2086             : 
    2087          18 :             IF (iw > 0) THEN
    2088           9 :                WRITE (iw, "(A,A,I4,A,I4,A,I2,A)") "&FCI", " NORB=", norb, ",NELEC=", nelec, ",MS2=", ms2, ","
    2089           9 :                isym = 1
    2090          33 :                WRITE (iw, "(A,1000(I1,','))") "  ORBSYM=", (isym, i=1, norb)
    2091           9 :                isym = 0
    2092           9 :                WRITE (iw, "(A,I1,A)") "  ISYM=", isym, ","
    2093           9 :                WRITE (iw, "(A,I1,A)") "  UHF=", 1, ","
    2094           9 :                WRITE (iw, "(A)") " /"
    2095             :             END IF
    2096             :             !
    2097             :             ! Print integrals: ERI
    2098             :             ! alpha-alpha
    2099             :             CALL active_space_env%eri%eri_foreach(1, active_space_env%active_orbitals, &
    2100          18 :                                                   eri_fcidump_print(iw, 1, 1), 1, 1)
    2101          18 :             CALL eri_checksum%set(1, 1)
    2102          18 :             CALL active_space_env%eri%eri_foreach(1, active_space_env%active_orbitals, eri_checksum, 1, 1)
    2103             :             ! alpha-beta
    2104             :             CALL active_space_env%eri%eri_foreach(2, active_space_env%active_orbitals, &
    2105          18 :                                                   eri_fcidump_print(iw, 1, norb + 1), 1, 2)
    2106          18 :             CALL eri_checksum%set(1, norb + 1)
    2107          18 :             CALL active_space_env%eri%eri_foreach(2, active_space_env%active_orbitals, eri_checksum, 1, 2)
    2108             :             ! beta-beta
    2109             :             CALL active_space_env%eri%eri_foreach(3, active_space_env%active_orbitals, &
    2110          18 :                                                   eri_fcidump_print(iw, norb + 1, norb + 1), 2, 2)
    2111          18 :             CALL eri_checksum%set(norb + 1, norb + 1)
    2112          18 :             CALL active_space_env%eri%eri_foreach(3, active_space_env%active_orbitals, eri_checksum, 2, 2)
    2113             :             ! Print integrals: Fij
    2114             :             ! alpha
    2115          18 :             nmo = active_space_env%eri%norb
    2116          72 :             ALLOCATE (fmat(nmo, nmo))
    2117          18 :             CALL replicate_and_symmetrize_matrix(nmo, active_space_env%fock_sub(1), fmat)
    2118          18 :             IF (iw > 0) THEN
    2119           9 :                i3 = 0; i4 = 0
    2120          33 :                DO m1 = 1, norb
    2121          24 :                   i1 = active_space_env%active_orbitals(m1, 1)
    2122          78 :                   DO m2 = m1, norb
    2123          45 :                      i2 = active_space_env%active_orbitals(m2, 1)
    2124          45 :                      checksum = checksum + ABS(fmat(i1, i2))
    2125          69 :                      WRITE (iw, "(ES23.16,4I4)") fmat(i1, i2), m1, m2, i3, i4
    2126             :                   END DO
    2127             :                END DO
    2128             :             END IF
    2129          18 :             DEALLOCATE (fmat)
    2130             :             ! beta
    2131          72 :             ALLOCATE (fmat(nmo, nmo))
    2132          18 :             CALL replicate_and_symmetrize_matrix(nmo, active_space_env%fock_sub(2), fmat)
    2133          18 :             IF (iw > 0) THEN
    2134           9 :                i3 = 0; i4 = 0
    2135          33 :                DO m1 = 1, SIZE(active_space_env%active_orbitals, 1)
    2136          24 :                   i1 = active_space_env%active_orbitals(m1, 2)
    2137          78 :                   DO m2 = m1, SIZE(active_space_env%active_orbitals, 1)
    2138          45 :                      i2 = active_space_env%active_orbitals(m2, 2)
    2139          45 :                      checksum = checksum + ABS(fmat(i1, i2))
    2140          69 :                      WRITE (iw, "(ES23.16,4I4)") fmat(i1, i2), m1 + norb, m2 + norb, i3, i4
    2141             :                   END DO
    2142             :                END DO
    2143             :             END IF
    2144          18 :             DEALLOCATE (fmat)
    2145             :             ! Print energy
    2146          18 :             esub = active_space_env%energy_inactive
    2147          18 :             i1 = 0; i2 = 0; i3 = 0; i4 = 0
    2148          18 :             checksum = checksum + ABS(esub)
    2149          36 :             IF (iw > 0) WRITE (iw, "(ES23.16,4I4)") esub, i1, i2, i3, i4
    2150             :          END ASSOCIATE
    2151             :       END IF
    2152             :       !
    2153         106 :       CALL cp_print_key_finished_output(iw, logger, as_input, "FCIDUMP")
    2154             : 
    2155             :       !>>
    2156         106 :       iw = cp_logger_get_default_io_unit(logger)
    2157         106 :       IF (iw > 0) WRITE (iw, '(T4,A,T66,F12.8)') "FCIDUMP| Checksum:", eri_checksum%checksum + checksum
    2158             :       !<<
    2159             : 
    2160         212 :    END SUBROUTINE fcidump
    2161             : 
    2162             : ! **************************************************************************************************
    2163             : !> \brief replicate and symmetrize a matrix
    2164             : !> \param norb the number of orbitals
    2165             : !> \param distributed_matrix ...
    2166             : !> \param replicated_matrix ...
    2167             : ! **************************************************************************************************
    2168         372 :    SUBROUTINE replicate_and_symmetrize_matrix(norb, distributed_matrix, replicated_matrix)
    2169             :       INTEGER, INTENT(IN)                                :: norb
    2170             :       TYPE(cp_fm_type), INTENT(IN)                       :: distributed_matrix
    2171             :       REAL(dp), DIMENSION(:, :), INTENT(INOUT)           :: replicated_matrix
    2172             : 
    2173             :       INTEGER                                            :: i1, i2
    2174             :       REAL(dp)                                           :: mval
    2175             : 
    2176        6468 :       replicated_matrix(:, :) = 0.0_dp
    2177        1596 :       DO i1 = 1, norb
    2178        4644 :          DO i2 = i1, norb
    2179        3048 :             CALL cp_fm_get_element(distributed_matrix, i1, i2, mval)
    2180        3048 :             replicated_matrix(i1, i2) = mval
    2181        4272 :             replicated_matrix(i2, i1) = mval
    2182             :          END DO
    2183             :       END DO
    2184         372 :    END SUBROUTINE replicate_and_symmetrize_matrix
    2185             : 
    2186             : ! **************************************************************************************************
    2187             : !> \brief Calculates active space Fock matrix and inactive energy
    2188             : !> \param active_space_env ...
    2189             : !> \par History
    2190             : !>      06.2016 created [JGH]
    2191             : ! **************************************************************************************************
    2192         106 :    SUBROUTINE subspace_fock_matrix(active_space_env)
    2193             : 
    2194             :       TYPE(active_space_type), POINTER                   :: active_space_env
    2195             : 
    2196             :       INTEGER                                            :: i1, i2, is, norb, nspins
    2197             :       REAL(KIND=dp)                                      :: eeri, eref, esub, mval
    2198         106 :       REAL(KIND=dp), ALLOCATABLE, DIMENSION(:, :)        :: ks_a_mat, ks_a_ref, ks_b_mat, ks_b_ref, &
    2199         106 :                                                             ks_mat, ks_ref, p_a_mat, p_b_mat, p_mat
    2200             :       TYPE(cp_fm_type), POINTER                          :: matrix, mo_coef
    2201             :       TYPE(dbcsr_csr_type), POINTER                      :: eri, eri_aa, eri_ab, eri_bb
    2202             : 
    2203         106 :       eref = active_space_env%energy_ref
    2204         106 :       nspins = active_space_env%nspins
    2205             : 
    2206         106 :       IF (nspins == 1) THEN
    2207          88 :          CALL get_mo_set(active_space_env%mos_active(1), nmo=norb, mo_coeff=mo_coef)
    2208             :          !
    2209             :          ! Loop over ERI, calculate subspace HF energy and Fock matrix
    2210             :          !
    2211             :          ! replicate KS, Core, and P matrices
    2212         880 :          ALLOCATE (ks_mat(norb, norb), ks_ref(norb, norb), p_mat(norb, norb))
    2213        1184 :          ks_ref = 0.0_dp
    2214             : 
    2215             :          ! ks_mat contains the KS/Fock matrix (of full density) projected onto the AS MO subspace (f_ref in eq. 19)
    2216          88 :          CALL replicate_and_symmetrize_matrix(norb, active_space_env%ks_sub(1), ks_mat)
    2217          88 :          CALL replicate_and_symmetrize_matrix(norb, active_space_env%p_active(1), p_mat)
    2218             : 
    2219             :          ! compute ks_ref = V_H[rho^A] + V_HFX[rho^A]
    2220          88 :          eri => active_space_env%eri%eri(1)%csr_mat
    2221          88 :          CALL build_subspace_fock_matrix(active_space_env%active_orbitals, eri, p_mat, ks_ref, active_space_env%eri%method)
    2222             : 
    2223             :          ! compute eeri = E_H[rho^A] + E_HFX[rho^A] as
    2224             :          ! eeri = 1/2 * (SUM_pq (V_H[rho^A] + V_HFX[rho^A])_pq * D^A_pq)
    2225        1184 :          eeri = 0.5_dp*SUM(ks_ref*p_mat)
    2226             : 
    2227             :          ! now calculate the inactive energy acoording to eq. 19, that is
    2228             :          ! esub = E^I = E_ref - f_ref .* D^A + E_H[rho^A] + E_HFX[rho^A]
    2229             :          ! where f^ref = ks_mat, which is the KS/Fock matrix in MO basis, transformed previously
    2230             :          ! and is equal to ks_mat = h^0 + V_core + V_H[rho] + V_HFX[rho]
    2231        1184 :          esub = eref - SUM(ks_mat(1:norb, 1:norb)*p_mat(1:norb, 1:norb)) + eeri
    2232             : 
    2233             :          ! reuse ks_mat to store f^I = f^ref - (V_H[rho^A] + V_HFX[rho^A]) according to eq. 20
    2234        1184 :          ks_mat(1:norb, 1:norb) = ks_mat(1:norb, 1:norb) - ks_ref(1:norb, 1:norb)
    2235             :          ! this is now the embedding potential for the AS calculation!
    2236             : 
    2237          88 :          active_space_env%energy_inactive = esub
    2238             : 
    2239          88 :          CALL cp_fm_release(active_space_env%fock_sub)
    2240         352 :          ALLOCATE (active_space_env%fock_sub(nspins))
    2241         176 :          DO is = 1, nspins
    2242          88 :             matrix => active_space_env%ks_sub(is)
    2243             :             CALL cp_fm_create(active_space_env%fock_sub(is), matrix%matrix_struct, &
    2244         176 :                               name="Active Fock operator")
    2245             :          END DO
    2246          88 :          matrix => active_space_env%fock_sub(1)
    2247         424 :          DO i1 = 1, norb
    2248        1184 :             DO i2 = 1, norb
    2249         848 :                mval = ks_mat(i1, i2)
    2250        1096 :                CALL cp_fm_set_element(matrix, i1, i2, mval)
    2251             :             END DO
    2252             :          END DO
    2253             :       ELSE
    2254             : 
    2255          18 :          CALL get_mo_set(active_space_env%mos_active(1), nmo=norb)
    2256             :          !
    2257             :          ! Loop over ERI, calculate subspace HF energy and Fock matrix
    2258             :          !
    2259             :          ! replicate KS, Core, and P matrices
    2260             :          ALLOCATE (ks_a_mat(norb, norb), ks_b_mat(norb, norb), &
    2261             :               &    ks_a_ref(norb, norb), ks_b_ref(norb, norb), &
    2262         342 :               &     p_a_mat(norb, norb), p_b_mat(norb, norb))
    2263         972 :          ks_a_ref(:, :) = 0.0_dp; ks_b_ref(:, :) = 0.0_dp
    2264             : 
    2265          18 :          CALL replicate_and_symmetrize_matrix(norb, active_space_env%p_active(1), p_a_mat)
    2266          18 :          CALL replicate_and_symmetrize_matrix(norb, active_space_env%p_active(2), p_b_mat)
    2267          18 :          CALL replicate_and_symmetrize_matrix(norb, active_space_env%ks_sub(1), ks_a_mat)
    2268          18 :          CALL replicate_and_symmetrize_matrix(norb, active_space_env%ks_sub(2), ks_b_mat)
    2269             :          !
    2270             :          !
    2271          18 :          eri_aa => active_space_env%eri%eri(1)%csr_mat
    2272          18 :          eri_ab => active_space_env%eri%eri(2)%csr_mat
    2273          18 :          eri_bb => active_space_env%eri%eri(3)%csr_mat
    2274             :          CALL build_subspace_spin_fock_matrix(active_space_env%active_orbitals, eri_aa, eri_ab, p_a_mat, p_b_mat, ks_a_ref, &
    2275          18 :                                               tr_mixed_eri=.FALSE., eri_method=active_space_env%eri%method)
    2276             :          CALL build_subspace_spin_fock_matrix(active_space_env%active_orbitals, eri_bb, eri_ab, p_b_mat, p_a_mat, ks_b_ref, &
    2277          18 :                                               tr_mixed_eri=.TRUE., eri_method=active_space_env%eri%method)
    2278             :          !
    2279             :          ! calculate energy
    2280          18 :          eeri = 0.0_dp
    2281         972 :          eeri = 0.5_dp*(SUM(ks_a_ref*p_a_mat) + SUM(ks_b_ref*p_b_mat))
    2282         972 :          esub = eref - SUM(ks_a_mat*p_a_mat) - SUM(ks_b_mat*p_b_mat) + eeri
    2283         486 :          ks_a_mat(:, :) = ks_a_mat(:, :) - ks_a_ref(:, :)
    2284         486 :          ks_b_mat(:, :) = ks_b_mat(:, :) - ks_b_ref(:, :)
    2285             :          !
    2286          18 :          active_space_env%energy_inactive = esub
    2287             :          !
    2288          18 :          CALL cp_fm_release(active_space_env%fock_sub)
    2289          90 :          ALLOCATE (active_space_env%fock_sub(nspins))
    2290          54 :          DO is = 1, nspins
    2291          36 :             matrix => active_space_env%ks_sub(is)
    2292             :             CALL cp_fm_create(active_space_env%fock_sub(is), matrix%matrix_struct, &
    2293          54 :                               name="Active Fock operator")
    2294             :          END DO
    2295             : 
    2296          18 :          matrix => active_space_env%fock_sub(1)
    2297          98 :          DO i1 = 1, norb
    2298         486 :             DO i2 = 1, norb
    2299         388 :                mval = ks_a_mat(i1, i2)
    2300         468 :                CALL cp_fm_set_element(matrix, i1, i2, mval)
    2301             :             END DO
    2302             :          END DO
    2303          18 :          matrix => active_space_env%fock_sub(2)
    2304         116 :          DO i1 = 1, norb
    2305         486 :             DO i2 = 1, norb
    2306         388 :                mval = ks_b_mat(i1, i2)
    2307         468 :                CALL cp_fm_set_element(matrix, i1, i2, mval)
    2308             :             END DO
    2309             :          END DO
    2310             : 
    2311             :       END IF
    2312             : 
    2313         106 :    END SUBROUTINE subspace_fock_matrix
    2314             : 
    2315             : ! **************************************************************************************************
    2316             : !> \brief build subspace fockian
    2317             : !> \param active_orbitals the active orbital indices
    2318             : !> \param eri two electon integrals in MO
    2319             : !> \param p_mat density matrix
    2320             : !> \param ks_ref fockian matrix
    2321             : !> \param eri_method ...
    2322             : ! **************************************************************************************************
    2323          88 :    SUBROUTINE build_subspace_fock_matrix(active_orbitals, eri, p_mat, ks_ref, eri_method)
    2324             :       INTEGER, DIMENSION(:, :), INTENT(IN)               :: active_orbitals
    2325             :       TYPE(dbcsr_csr_type), INTENT(IN)                   :: eri
    2326             :       REAL(dp), DIMENSION(:, :), INTENT(IN)              :: p_mat
    2327             :       REAL(dp), DIMENSION(:, :), INTENT(INOUT)           :: ks_ref
    2328             :       INTEGER, INTENT(IN)                                :: eri_method
    2329             : 
    2330             :       INTEGER                                            :: i1, i12, i12l, i2, i3, i34, i34l, i4, &
    2331             :                                                             irptr, m1, m2, nindex, nmo_total, norb
    2332             :       INTEGER, DIMENSION(2)                              :: irange
    2333             :       REAL(dp)                                           :: erint
    2334             :       TYPE(mp_comm_type)                                 :: mp_group
    2335             : 
    2336             :       ! Nothing to do
    2337          88 :       norb = SIZE(active_orbitals, 1)
    2338          88 :       nmo_total = SIZE(p_mat, 1)
    2339          88 :       nindex = (nmo_total*(nmo_total + 1))/2
    2340          88 :       CALL mp_group%set_handle(eri%mp_group%get_handle())
    2341          88 :       IF (eri_method == eri_method_gpw_ht) THEN
    2342          72 :          irange = [1, nindex]
    2343             :       ELSE
    2344          64 :          irange = get_irange_csr(nindex, mp_group)
    2345             :       END IF
    2346         310 :       DO m1 = 1, norb
    2347         222 :          i1 = active_orbitals(m1, 1)
    2348         736 :          DO m2 = m1, norb
    2349         426 :             i2 = active_orbitals(m2, 1)
    2350         426 :             i12 = csr_idx_to_combined(i1, i2, nmo_total)
    2351         648 :             IF (i12 >= irange(1) .AND. i12 <= irange(2)) THEN
    2352         249 :                i12l = i12 - irange(1) + 1
    2353         249 :                irptr = eri%rowptr_local(i12l) - 1
    2354         902 :                DO i34l = 1, eri%nzerow_local(i12l)
    2355         653 :                   i34 = eri%colind_local(irptr + i34l)
    2356         653 :                   CALL csr_idx_from_combined(i34, nmo_total, i3, i4)
    2357         653 :                   erint = eri%nzval_local%r_dp(irptr + i34l)
    2358             :                   ! Coulomb
    2359         653 :                   ks_ref(i1, i2) = ks_ref(i1, i2) + erint*p_mat(i3, i4)
    2360         653 :                   IF (i3 /= i4) THEN
    2361         323 :                      ks_ref(i1, i2) = ks_ref(i1, i2) + erint*p_mat(i3, i4)
    2362             :                   END IF
    2363         653 :                   IF (i12 /= i34) THEN
    2364         440 :                      ks_ref(i3, i4) = ks_ref(i3, i4) + erint*p_mat(i1, i2)
    2365         440 :                      IF (i1 /= i2) THEN
    2366         272 :                         ks_ref(i3, i4) = ks_ref(i3, i4) + erint*p_mat(i1, i2)
    2367             :                      END IF
    2368             :                   END IF
    2369             :                   ! Exchange
    2370         653 :                   erint = -0.5_dp*erint
    2371         653 :                   ks_ref(i1, i3) = ks_ref(i1, i3) + erint*p_mat(i2, i4)
    2372         653 :                   IF (i1 /= i2) THEN
    2373         374 :                      ks_ref(i2, i3) = ks_ref(i2, i3) + erint*p_mat(i1, i4)
    2374             :                   END IF
    2375         653 :                   IF (i3 /= i4) THEN
    2376         323 :                      ks_ref(i1, i4) = ks_ref(i1, i4) + erint*p_mat(i2, i3)
    2377             :                   END IF
    2378        1555 :                   IF (i1 /= i2 .AND. i3 /= i4) THEN
    2379         257 :                      ks_ref(i2, i4) = ks_ref(i2, i4) + erint*p_mat(i1, i3)
    2380             :                   END IF
    2381             :                END DO
    2382             :             END IF
    2383             :          END DO
    2384             :       END DO
    2385             :       !
    2386         310 :       DO m1 = 1, norb
    2387         222 :          i1 = active_orbitals(m1, 1)
    2388         736 :          DO m2 = m1, norb
    2389         426 :             i2 = active_orbitals(m2, 1)
    2390         648 :             ks_ref(i2, i1) = ks_ref(i1, i2)
    2391             :          END DO
    2392             :       END DO
    2393        2280 :       CALL mp_group%sum(ks_ref)
    2394             : 
    2395          88 :    END SUBROUTINE build_subspace_fock_matrix
    2396             : 
    2397             : ! **************************************************************************************************
    2398             : !> \brief build subspace fockian for unrestricted spins
    2399             : !> \param active_orbitals the active orbital indices
    2400             : !> \param eri_aa two electon integrals in MO with parallel spins
    2401             : !> \param eri_ab two electon integrals in MO with anti-parallel spins
    2402             : !> \param p_a_mat density matrix for up-spin
    2403             : !> \param p_b_mat density matrix for down-spin
    2404             : !> \param ks_a_ref fockian matrix for up-spin
    2405             : !> \param tr_mixed_eri boolean to indicate Coulomb interaction alignment
    2406             : !> \param eri_method ...
    2407             : ! **************************************************************************************************
    2408          36 :    SUBROUTINE build_subspace_spin_fock_matrix(active_orbitals, eri_aa, eri_ab, p_a_mat, p_b_mat, ks_a_ref, tr_mixed_eri, eri_method)
    2409             :       INTEGER, DIMENSION(:, :), INTENT(IN)               :: active_orbitals
    2410             :       TYPE(dbcsr_csr_type), INTENT(IN)                   :: eri_aa, eri_ab
    2411             :       REAL(dp), DIMENSION(:, :), INTENT(IN)              :: p_a_mat, p_b_mat
    2412             :       REAL(dp), DIMENSION(:, :), INTENT(INOUT)           :: ks_a_ref
    2413             :       LOGICAL, INTENT(IN)                                :: tr_mixed_eri
    2414             :       INTEGER, INTENT(IN)                                :: eri_method
    2415             : 
    2416             :       INTEGER                                            :: i1, i12, i12l, i2, i3, i34, i34l, i4, &
    2417             :                                                             irptr, m1, m2, nindex, nmo_total, &
    2418             :                                                             norb, spin1, spin2
    2419             :       INTEGER, DIMENSION(2)                              :: irange
    2420             :       REAL(dp)                                           :: erint
    2421             :       TYPE(mp_comm_type)                                 :: mp_group
    2422             : 
    2423          36 :       norb = SIZE(active_orbitals, 1)
    2424          36 :       nmo_total = SIZE(p_a_mat, 1)
    2425          36 :       nindex = (nmo_total*(nmo_total + 1))/2
    2426          36 :       CALL mp_group%set_handle(eri_aa%mp_group%get_handle())
    2427          36 :       IF (eri_method == eri_method_gpw_ht) THEN
    2428           0 :          irange = [1, nindex]
    2429             :       ELSE
    2430          36 :          irange = get_irange_csr(nindex, mp_group)
    2431             :       END IF
    2432          36 :       IF (tr_mixed_eri) THEN
    2433             :          spin1 = 2
    2434          36 :          spin2 = 1
    2435             :       ELSE
    2436          18 :          spin1 = 1
    2437          18 :          spin2 = 2
    2438             :       END IF
    2439         132 :       DO m1 = 1, norb
    2440          96 :          i1 = active_orbitals(m1, spin1)
    2441         312 :          DO m2 = m1, norb
    2442         180 :             i2 = active_orbitals(m2, spin1)
    2443         180 :             i12 = csr_idx_to_combined(i1, i2, nmo_total)
    2444         276 :             IF (i12 >= irange(1) .AND. i12 <= irange(2)) THEN
    2445          90 :                i12l = i12 - irange(1) + 1
    2446          90 :                irptr = eri_aa%rowptr_local(i12l) - 1
    2447         322 :                DO i34l = 1, eri_aa%nzerow_local(i12l)
    2448         232 :                   i34 = eri_aa%colind_local(irptr + i34l)
    2449         232 :                   CALL csr_idx_from_combined(i34, nmo_total, i3, i4)
    2450         232 :                   erint = eri_aa%nzval_local%r_dp(irptr + i34l)
    2451             :                   ! Coulomb
    2452             :                   !F_ij += (ij|kl)*d_kl
    2453         232 :                   ks_a_ref(i1, i2) = ks_a_ref(i1, i2) + erint*p_a_mat(i3, i4)
    2454         232 :                   IF (i12 /= i34) THEN
    2455             :                      !F_kl += (ij|kl)*d_ij
    2456         142 :                      ks_a_ref(i3, i4) = ks_a_ref(i3, i4) + erint*p_a_mat(i1, i2)
    2457             :                   END IF
    2458             :                   ! Exchange
    2459         232 :                   erint = -1.0_dp*erint
    2460             :                   !F_ik -= (ij|kl)*d_jl
    2461         232 :                   ks_a_ref(i1, i3) = ks_a_ref(i1, i3) + erint*p_a_mat(i2, i4)
    2462         232 :                   IF (i1 /= i2) THEN
    2463             :                      !F_jk -= (ij|kl)*d_il
    2464          98 :                      ks_a_ref(i2, i3) = ks_a_ref(i2, i3) + erint*p_a_mat(i1, i4)
    2465             :                   END IF
    2466         232 :                   IF (i3 /= i4) THEN
    2467             :                      !F_il -= (ij|kl)*d_jk
    2468         104 :                      ks_a_ref(i1, i4) = ks_a_ref(i1, i4) + erint*p_a_mat(i2, i3)
    2469             :                   END IF
    2470         554 :                   IF (i1 /= i2 .AND. i3 /= i4) THEN
    2471             :                      !F_jl -= (ij|kl)*d_ik
    2472          60 :                      ks_a_ref(i2, i4) = ks_a_ref(i2, i4) + erint*p_a_mat(i1, i3)
    2473             :                   END IF
    2474             :                END DO
    2475             :             END IF
    2476             :          END DO
    2477             :       END DO
    2478             :       !
    2479             : 
    2480          36 :       CALL mp_group%set_handle(eri_ab%mp_group%get_handle())
    2481          36 :       IF (eri_method == eri_method_gpw_ht) THEN
    2482           0 :          irange = [1, nindex]
    2483             :       ELSE
    2484          36 :          irange = get_irange_csr(nindex, mp_group)
    2485             :       END IF
    2486         132 :       DO m1 = 1, norb
    2487          96 :          i1 = active_orbitals(m1, 1)
    2488         312 :          DO m2 = m1, norb
    2489         180 :             i2 = active_orbitals(m2, 1)
    2490         180 :             i12 = csr_idx_to_combined(i1, i2, nmo_total)
    2491         276 :             IF (i12 >= irange(1) .AND. i12 <= irange(2)) THEN
    2492          90 :                i12l = i12 - irange(1) + 1
    2493          90 :                irptr = eri_ab%rowptr_local(i12l) - 1
    2494         532 :                DO i34l = 1, eri_ab%nzerow_local(i12l)
    2495         442 :                   i34 = eri_ab%colind_local(irptr + i34l)
    2496         442 :                   CALL csr_idx_from_combined(i34, nmo_total, i3, i4)
    2497         442 :                   erint = eri_ab%nzval_local%r_dp(irptr + i34l)
    2498             :                   ! Coulomb
    2499         532 :                   IF (tr_mixed_eri) THEN
    2500             :                      !F_kl += (kl beta|ij alpha )*d_alpha_ij
    2501         221 :                      ks_a_ref(i3, i4) = ks_a_ref(i3, i4) + erint*p_b_mat(i1, i2)
    2502             :                   ELSE
    2503             :                      !F_ij += (ij alpha|kl beta )*d_beta_kl
    2504         221 :                      ks_a_ref(i1, i2) = ks_a_ref(i1, i2) + erint*p_b_mat(i3, i4)
    2505             :                   END IF
    2506             :                END DO
    2507             :             END IF
    2508             :          END DO
    2509             :       END DO
    2510             :       !
    2511         132 :       DO m1 = 1, norb
    2512          96 :          i1 = active_orbitals(m1, spin1)
    2513         312 :          DO m2 = m1, norb
    2514         180 :             i2 = active_orbitals(m2, spin1)
    2515         276 :             ks_a_ref(i2, i1) = ks_a_ref(i1, i2)
    2516             :          END DO
    2517             :       END DO
    2518          36 :       CALL mp_group%set_handle(eri_aa%mp_group%get_handle())
    2519        1908 :       CALL mp_group%sum(ks_a_ref)
    2520             : 
    2521          36 :    END SUBROUTINE build_subspace_spin_fock_matrix
    2522             : 
    2523             : ! **************************************************************************************************
    2524             : !> \brief Creates a local basis
    2525             : !> \param pro_basis_set ...
    2526             : !> \param zval ...
    2527             : !> \param ishell ...
    2528             : !> \param nshell ...
    2529             : !> \param lnam ...
    2530             : !> \par History
    2531             : !>      05.2016 created [JGH]
    2532             : ! **************************************************************************************************
    2533           0 :    SUBROUTINE create_pro_basis(pro_basis_set, zval, ishell, nshell, lnam)
    2534             :       TYPE(gto_basis_set_type), POINTER                  :: pro_basis_set
    2535             :       INTEGER, INTENT(IN)                                :: zval, ishell
    2536             :       INTEGER, DIMENSION(:), INTENT(IN)                  :: nshell
    2537             :       CHARACTER(len=*), DIMENSION(:), INTENT(IN)         :: lnam
    2538             : 
    2539           0 :       CHARACTER(len=6), DIMENSION(:), POINTER            :: sym
    2540             :       INTEGER                                            :: i, l, nj
    2541             :       INTEGER, DIMENSION(4, 7)                           :: ne
    2542           0 :       INTEGER, DIMENSION(:), POINTER                     :: lq, nq
    2543           0 :       REAL(KIND=dp), DIMENSION(:), POINTER               :: zet
    2544             :       TYPE(sto_basis_set_type), POINTER                  :: sto_basis_set
    2545             : 
    2546           0 :       CPASSERT(.NOT. ASSOCIATED(pro_basis_set))
    2547           0 :       NULLIFY (sto_basis_set)
    2548             : 
    2549             :       ! electronic configuration
    2550           0 :       ne = 0
    2551           0 :       DO l = 1, 4 !lq(1)+1
    2552           0 :          nj = 2*(l - 1) + 1
    2553           0 :          DO i = l, 7 ! nq(1)
    2554           0 :             ne(l, i) = ptable(zval)%e_conv(l - 1) - 2*nj*(i - l)
    2555           0 :             ne(l, i) = MAX(ne(l, i), 0)
    2556           0 :             ne(l, i) = MIN(ne(l, i), 2*nj)
    2557             :          END DO
    2558             :       END DO
    2559           0 :       ALLOCATE (nq(ishell), lq(ishell), zet(ishell), sym(ishell))
    2560           0 :       DO i = 1, ishell
    2561           0 :          nq(i) = nshell(i)
    2562           0 :          SELECT CASE (lnam(i))
    2563             :          CASE ('S', 's')
    2564           0 :             lq(i) = 0
    2565             :          CASE ('P', 'p')
    2566           0 :             lq(i) = 1
    2567             :          CASE ('D', 'd')
    2568           0 :             lq(i) = 2
    2569             :          CASE ('F', 'f')
    2570           0 :             lq(i) = 3
    2571             :          CASE DEFAULT
    2572           0 :             CPABORT("Wrong l QN")
    2573             :          END SELECT
    2574           0 :          sym(i) = lnam(i)
    2575           0 :          zet(i) = srules(zval, ne, nq(1), lq(1))
    2576             :       END DO
    2577           0 :       CALL allocate_sto_basis_set(sto_basis_set)
    2578           0 :       CALL set_sto_basis_set(sto_basis_set, nshell=1, nq=nq, lq=lq, zet=zet, symbol=sym)
    2579           0 :       CALL create_gto_from_sto_basis(sto_basis_set, pro_basis_set, 6)
    2580           0 :       pro_basis_set%norm_type = 2
    2581           0 :       CALL init_orb_basis_set(pro_basis_set)
    2582           0 :       CALL deallocate_sto_basis_set(sto_basis_set)
    2583             : 
    2584           0 :    END SUBROUTINE create_pro_basis
    2585             : 
    2586             : ! **************************************************************************************************
    2587             : !> \brief Update the density matrix in AO basis with the active density contribution
    2588             : !> \param active_space_env the active space environment
    2589             : !> \param rho_ao the density matrix in AO basis
    2590             : ! **************************************************************************************************
    2591           0 :    SUBROUTINE update_density_ao(active_space_env, rho_ao)
    2592             :       TYPE(active_space_type), POINTER                   :: active_space_env
    2593             :       TYPE(dbcsr_p_type), DIMENSION(:), POINTER          :: rho_ao
    2594             : 
    2595             :       INTEGER                                            :: ispin, nao, nmo, nspins
    2596             :       TYPE(cp_fm_type)                                   :: R, U
    2597             :       TYPE(cp_fm_type), POINTER                          :: C_active, p_active_mo
    2598             :       TYPE(dbcsr_type), POINTER                          :: p_inactive_ao
    2599           0 :       TYPE(mo_set_type), DIMENSION(:), POINTER           :: mos_active
    2600             : 
    2601             :       ! Transform the AS density matrix P_MO to the atomic orbital basis,
    2602             :       ! this is simply C * P_MO * C^T
    2603           0 :       nspins = active_space_env%nspins
    2604           0 :       mos_active => active_space_env%mos_active
    2605           0 :       DO ispin = 1, nspins
    2606             :          ! size of p_inactive_ao is (nao x nao)
    2607           0 :          p_inactive_ao => active_space_env%pmat_inactive(ispin)%matrix
    2608             : 
    2609             :          ! copy p_inactive_ao to rho_ao
    2610           0 :          CALL dbcsr_copy(rho_ao(ispin)%matrix, p_inactive_ao)
    2611             : 
    2612             :          ! size of p_active_mo is (nmo x nmo)
    2613           0 :          p_active_mo => active_space_env%p_active(ispin)
    2614             : 
    2615             :          ! calculate R = p_mo
    2616           0 :          CALL cp_fm_create(R, p_active_mo%matrix_struct)
    2617           0 :          CALL cp_fm_to_fm(p_active_mo, R)
    2618             : 
    2619             :          ! calculate U = C * p_mo
    2620           0 :          CALL get_mo_set(mos_active(ispin), mo_coeff=C_active, nao=nao, nmo=nmo)
    2621           0 :          CALL cp_fm_create(U, C_active%matrix_struct)
    2622           0 :          CALL parallel_gemm("N", "N", nao, nmo, nmo, 1.0_dp, C_active, R, 0.0_dp, U)
    2623             : 
    2624             :          CALL cp_dbcsr_plus_fm_fm_t(sparse_matrix=rho_ao(ispin)%matrix, &
    2625           0 :                                     matrix_v=U, matrix_g=C_active, ncol=nmo, alpha=1.0_dp)
    2626             : 
    2627           0 :          CALL cp_fm_release(R)
    2628           0 :          CALL cp_fm_release(U)
    2629             :       END DO
    2630             : 
    2631           0 :    END SUBROUTINE update_density_ao
    2632             : 
    2633             : ! **************************************************************************************************
    2634             : !> \brief Print each value on the master node
    2635             : !> \param this object reference
    2636             : !> \param i i-index
    2637             : !> \param j j-index
    2638             : !> \param k k-index
    2639             : !> \param l l-index
    2640             : !> \param val value of the integral at (i,j,k.l)
    2641             : !> \return always true to dump all integrals
    2642             : ! **************************************************************************************************
    2643        2212 :    LOGICAL FUNCTION eri_fcidump_print_func(this, i, j, k, l, val) RESULT(cont)
    2644             :       CLASS(eri_fcidump_print), INTENT(inout) :: this
    2645             :       INTEGER, INTENT(in)                     :: i, j, k, l
    2646             :       REAL(KIND=dp), INTENT(in)               :: val
    2647             : 
    2648             :       ! write to the actual file only on the master
    2649        2212 :       IF (this%unit_nr > 0) THEN
    2650        1106 :          WRITE (this%unit_nr, "(ES23.16,4I4)") val, i + this%bra_start - 1, j + this%bra_start - 1, &
    2651        2212 :               &                                     k + this%ket_start - 1, l + this%ket_start - 1
    2652             :       END IF
    2653             : 
    2654        2212 :       cont = .TRUE.
    2655        2212 :    END FUNCTION eri_fcidump_print_func
    2656             : 
    2657             : ! **************************************************************************************************
    2658             : !> \brief checksum each value on the master node
    2659             : !> \param this object reference
    2660             : !> \param i i-index
    2661             : !> \param j j-index
    2662             : !> \param k k-index
    2663             : !> \param l l-index
    2664             : !> \param val value of the integral at (i,j,k.l)
    2665             : !> \return always true to dump all integrals
    2666             : ! **************************************************************************************************
    2667        2212 :    LOGICAL FUNCTION eri_fcidump_checksum_func(this, i, j, k, l, val) RESULT(cont)
    2668             :       CLASS(eri_fcidump_checksum), INTENT(inout) :: this
    2669             :       INTEGER, INTENT(in)                     :: i, j, k, l
    2670             :       REAL(KIND=dp), INTENT(in)               :: val
    2671             :       MARK_USED(i)
    2672             :       MARK_USED(j)
    2673             :       MARK_USED(k)
    2674             :       MARK_USED(l)
    2675             : 
    2676        2212 :       this%checksum = this%checksum + ABS(val)
    2677             : 
    2678        2212 :       cont = .TRUE.
    2679        2212 :    END FUNCTION eri_fcidump_checksum_func
    2680             : 
    2681             : ! **************************************************************************************************
    2682             : !> \brief Update active space density matrix from a fortran array
    2683             : !> \param p_act_mo density matrix in active space MO basis
    2684             : !> \param active_space_env active space environment
    2685             : !> \param ispin spin index
    2686             : !> \author Vladimir Rybkin
    2687             : ! **************************************************************************************************
    2688           0 :    SUBROUTINE update_active_density(p_act_mo, active_space_env, ispin)
    2689             :       REAL(KIND=dp), DIMENSION(:)                        :: p_act_mo
    2690             :       TYPE(active_space_type), POINTER                   :: active_space_env
    2691             :       INTEGER                                            :: ispin
    2692             : 
    2693             :       INTEGER                                            :: i1, i2, m1, m2, nmo_active
    2694             :       REAL(KIND=dp)                                      :: mval
    2695             :       TYPE(cp_fm_type), POINTER                          :: p_active
    2696             : 
    2697           0 :       p_active => active_space_env%p_active(ispin)
    2698           0 :       nmo_active = active_space_env%nmo_active
    2699             : 
    2700           0 :       DO i1 = 1, nmo_active
    2701           0 :          m1 = active_space_env%active_orbitals(i1, ispin)
    2702           0 :          DO i2 = 1, nmo_active
    2703           0 :             m2 = active_space_env%active_orbitals(i2, ispin)
    2704           0 :             mval = p_act_mo(i2 + (i1 - 1)*nmo_active)
    2705           0 :             CALL cp_fm_set_element(p_active, m1, m2, mval)
    2706             :          END DO
    2707             :       END DO
    2708             : 
    2709           0 :    END SUBROUTINE update_active_density
    2710             : 
    2711             : ! **************************************************************************************************
    2712             : !> \brief ...
    2713             : !> \param qs_env ...
    2714             : !> \param active_space_env ...
    2715             : !> \param as_input ...
    2716             : ! **************************************************************************************************
    2717           0 :    SUBROUTINE rsdft_embedding(qs_env, active_space_env, as_input)
    2718             :       TYPE(qs_environment_type), POINTER                 :: qs_env
    2719             :       TYPE(active_space_type), POINTER                   :: active_space_env
    2720             :       TYPE(section_vals_type), POINTER                   :: as_input
    2721             : 
    2722             :       CHARACTER(len=*), PARAMETER                        :: routineN = 'rsdft_embedding'
    2723             :       INTEGER                                            :: handle
    2724             : 
    2725             : #ifdef __NO_SOCKETS
    2726             :       CALL timeset(routineN, handle)
    2727             :       CPABORT("CP2K was compiled with the __NO_SOCKETS option!")
    2728             :       MARK_USED(qs_env)
    2729             :       MARK_USED(active_space_env)
    2730             :       MARK_USED(as_input)
    2731             : #else
    2732             : 
    2733             :       INTEGER                                            :: iw, client_fd, socket_fd, iter, max_iter
    2734             :       LOGICAL                                            :: converged, do_scf_embedding, ionode
    2735             :       REAL(KIND=dp)                                      :: alpha, delta_E, energy_corr, energy_new, &
    2736             :                                                             energy_old, energy_scf, eps_iter, t1, &
    2737             :                                                             t2
    2738             :       TYPE(cp_logger_type), POINTER                      :: logger
    2739           0 :       TYPE(dbcsr_p_type), DIMENSION(:), POINTER          :: rho_ao
    2740           0 :       TYPE(mo_set_type), DIMENSION(:), POINTER           :: mos_active
    2741             :       TYPE(mp_para_env_type), POINTER                    :: para_env
    2742             :       TYPE(qs_energy_type), POINTER                      :: energy
    2743             :       TYPE(qs_rho_type), POINTER                         :: rho
    2744             : 
    2745           0 :       CALL timeset(routineN, handle)
    2746             : 
    2747           0 :       t1 = m_walltime()
    2748             : 
    2749           0 :       logger => cp_get_default_logger()
    2750           0 :       iw = cp_logger_get_default_io_unit(logger)
    2751             : 
    2752           0 :       CALL get_qs_env(qs_env, para_env=para_env)
    2753           0 :       ionode = para_env%is_source()
    2754             : 
    2755             :       ! get info from the input
    2756           0 :       CALL section_vals_val_get(as_input, "SCF_EMBEDDING", l_val=do_scf_embedding)
    2757           0 :       active_space_env%do_scf_embedding = do_scf_embedding
    2758           0 :       CALL section_vals_val_get(as_input, "MAX_ITER", i_val=max_iter)
    2759           0 :       CALL section_vals_val_get(as_input, "EPS_ITER", r_val=eps_iter)
    2760           0 :       alpha = 0.0
    2761             : 
    2762             :       ! create the socket and wait for the client to connect
    2763           0 :       CALL initialize_socket(socket_fd, client_fd, as_input, ionode)
    2764           0 :       CALL para_env%sync()
    2765             : 
    2766             :       ! send two-electron integrals to the client
    2767           0 :       CALL send_eri_to_client(client_fd, active_space_env, para_env)
    2768             : 
    2769             :       ! get pointer to density in ao basis
    2770           0 :       CALL get_qs_env(qs_env, rho=rho, energy=energy)
    2771           0 :       CALL qs_rho_get(rho, rho_ao=rho_ao)
    2772             : 
    2773           0 :       IF ((iw > 0)) THEN
    2774             :          WRITE (UNIT=iw, FMT="(/,T3,A,T11,A,T21,A,T34,A,T55,A,T75,A,/,T3,A)") &
    2775           0 :             "Iter", "Update", "Time", "Corr. energy", "Total energy", "Change", REPEAT("-", 78)
    2776             :       END IF
    2777             :       ! CALL cp_add_iter_level(logger%iter_info, "QS_SCF")
    2778             : 
    2779           0 :       iter = 0
    2780           0 :       converged = .FALSE.
    2781             :       ! store the scf energy
    2782           0 :       energy_scf = active_space_env%energy_ref
    2783           0 :       energy_new = energy_scf
    2784           0 :       mos_active => active_space_env%mos_active
    2785             :       ! CALL set_qs_env(qs_env, active_space=active_space_env)
    2786             : 
    2787             :       ! start the self-consistent embedding loop
    2788           0 :       DO WHILE (iter < max_iter)
    2789           0 :          iter = iter + 1
    2790             : 
    2791             :          ! send V_emb and E_ina to the active space solver and update
    2792             :          ! the active space environment with the new active energy and density
    2793           0 :          CALL send_fock_to_client(client_fd, active_space_env, para_env)
    2794             : 
    2795             :          ! update energies
    2796           0 :          energy_old = energy_new
    2797           0 :          energy_new = active_space_env%energy_total
    2798           0 :          energy_corr = energy_new - energy_scf
    2799           0 :          delta_E = energy_new - energy_old
    2800             : 
    2801             :          ! get timer
    2802           0 :          t2 = m_walltime()
    2803             :          ! print out progress
    2804           0 :          IF ((iw > 0)) THEN
    2805             :             WRITE (UNIT=iw, &
    2806             :                    FMT="(T3,I4,T11,A,T21,F4.2,T28,F18.10,T49,F18.10,T70,ES11.2)") &
    2807           0 :                iter, 'P_Mix', t2 - t1, energy_corr, energy_new, delta_E
    2808             :          END IF
    2809             : 
    2810             :          ! update total density in AO basis with the AS contribution
    2811           0 :          CALL update_density_ao(active_space_env, rho_ao) ! rho_ao is updated
    2812             : 
    2813             :          ! calculate F_ks in AO basis (which contains Vxc) with the new density
    2814           0 :          qs_env%requires_matrix_vxc = .TRUE.
    2815           0 :          CALL qs_rho_update_rho(rho, qs_env) ! updates rho_r and rho_g using rho_ao
    2816           0 :          CALL qs_ks_did_change(qs_env%ks_env, rho_changed=.TRUE.) ! set flags about the change
    2817           0 :          CALL qs_ks_update_qs_env(qs_env) ! this call actually calculates F_ks
    2818             : 
    2819             :          ! update the reference energy
    2820           0 :          active_space_env%energy_ref = energy%total
    2821             : 
    2822             :          ! transform KS/Fock, Vxc and Hcore from AO to MO basis
    2823           0 :          CALL calculate_operators(mos_active, qs_env, active_space_env)
    2824             : 
    2825             :          ! calculate the new inactive energy and embedding potential
    2826           0 :          CALL subspace_fock_matrix(active_space_env)
    2827             : 
    2828             :          ! check if it is a one-shot correction
    2829           0 :          IF (.NOT. active_space_env%do_scf_embedding) THEN
    2830           0 :             IF (iw > 0) THEN
    2831             :                WRITE (UNIT=iw, FMT="(/,T3,A,I5,A/)") &
    2832           0 :                   "*** one-shot embedding correction finished ***"
    2833             :             END IF
    2834             :             converged = .TRUE.
    2835             :             EXIT
    2836             :             ! check for convergence
    2837           0 :          ELSEIF (ABS(delta_E) <= eps_iter) THEN
    2838           0 :             IF (iw > 0) THEN
    2839             :                WRITE (UNIT=iw, FMT="(/,T3,A,I5,A/)") &
    2840           0 :                   "*** rsDFT embedding run converged in ", iter, " iteration(s) ***"
    2841             :             END IF
    2842             :             converged = .TRUE.
    2843             :             EXIT
    2844             :          END IF
    2845             : 
    2846           0 :          t1 = m_walltime()
    2847             :       END DO
    2848             : 
    2849             :       IF (.NOT. converged) THEN
    2850           0 :          IF (iw > 0) THEN
    2851             :             WRITE (UNIT=iw, FMT="(/,T3,A,I5,A/)") &
    2852           0 :                "*** rsDFT embedding did not converged after ", iter, " iteration(s) ***"
    2853             :          END IF
    2854             :       END IF
    2855             : 
    2856             :       ! update qs total energy to the final embedding energy
    2857           0 :       energy%total = active_space_env%energy_total
    2858             : 
    2859           0 :       CALL finalize_socket(socket_fd, client_fd, as_input, ionode)
    2860           0 :       CALL para_env%sync()
    2861             : #endif
    2862             : 
    2863           0 :       CALL timestop(handle)
    2864             : 
    2865           0 :    END SUBROUTINE rsdft_embedding
    2866             : 
    2867             : #ifndef __NO_SOCKETS
    2868             : ! **************************************************************************************************
    2869             : !> \brief Creates the socket, spawns the client and connects to it
    2870             : !> \param socket_fd the socket file descriptor
    2871             : !> \param client_fd the client file descriptor
    2872             : !> \param as_input active space inpute section
    2873             : !> \param ionode logical flag indicating if the process is the master
    2874             : ! **************************************************************************************************
    2875           0 :    SUBROUTINE initialize_socket(socket_fd, client_fd, as_input, ionode)
    2876             :       INTEGER, INTENT(OUT)                               :: socket_fd, client_fd
    2877             :       TYPE(section_vals_type), INTENT(IN), POINTER       :: as_input
    2878             :       LOGICAL, INTENT(IN)                                :: ionode
    2879             : 
    2880             :       CHARACTER(len=*), PARAMETER                        :: routineN = 'initialize_socket'
    2881             :       INTEGER, PARAMETER                                 :: backlog = 10
    2882             : 
    2883             :       CHARACTER(len=default_path_length)                 :: hostname
    2884             :       INTEGER                                            :: handle, iw, port, protocol
    2885             :       LOGICAL                                            :: inet
    2886             :       TYPE(cp_logger_type), POINTER                      :: logger
    2887             : 
    2888           0 :       CALL timeset(routineN, handle)
    2889             : 
    2890           0 :       logger => cp_get_default_logger()
    2891           0 :       iw = cp_logger_get_default_io_unit(logger)
    2892             : 
    2893             :       ! protocol == 0 for UNIX, protocol > 0 for INET
    2894           0 :       CALL section_vals_val_get(as_input, "SOCKET%INET", l_val=inet)
    2895           0 :       IF (inet) THEN
    2896           0 :          protocol = 1
    2897             :       ELSE
    2898           0 :          protocol = 0
    2899             :       END IF
    2900           0 :       CALL section_vals_val_get(as_input, "SOCKET%HOST", c_val=hostname)
    2901           0 :       CALL section_vals_val_get(as_input, "SOCKET%PORT", i_val=port)
    2902             : 
    2903           0 :       IF (ionode) THEN
    2904           0 :          CALL open_bind_socket(socket_fd, protocol, port, TRIM(hostname)//C_NULL_CHAR)
    2905           0 :          WRITE (iw, '(/,T3,A,A)') "@SERVER: Created socket with address ", TRIM(hostname)
    2906           0 :          CALL listen_socket(socket_fd, backlog)
    2907             : 
    2908             :          ! wait until a connetion request arrives
    2909           0 :          WRITE (iw, '(T3,A)') "@SERVER: Waiting for requests..."
    2910           0 :          CALL accept_socket(socket_fd, client_fd)
    2911           0 :          WRITE (iw, '(T3,A,I2)') "@SERVER: Accepted socket with fd ", client_fd
    2912             :       END IF
    2913             : 
    2914           0 :       CALL timestop(handle)
    2915             : 
    2916           0 :    END SUBROUTINE initialize_socket
    2917             : 
    2918             : ! **************************************************************************************************
    2919             : !> \brief Closes the connection to the socket and deletes the file
    2920             : !> \param socket_fd the socket file descriptor
    2921             : !> \param client_fd the client file descriptor
    2922             : !> \param as_input active space inpute section
    2923             : !> \param ionode logical flag indicating if the process is the master
    2924             : ! **************************************************************************************************
    2925           0 :    SUBROUTINE finalize_socket(socket_fd, client_fd, as_input, ionode)
    2926             :       INTEGER, INTENT(IN)                                :: socket_fd, client_fd
    2927             :       TYPE(section_vals_type), INTENT(IN), POINTER       :: as_input
    2928             :       LOGICAL, INTENT(IN)                                :: ionode
    2929             : 
    2930             :       CHARACTER(len=*), PARAMETER                        :: routineN = 'finalize_socket'
    2931             :       INTEGER, PARAMETER                                 :: header_len = 12
    2932             : 
    2933             :       CHARACTER(len=default_path_length)                 :: hostname
    2934             :       INTEGER                                            :: handle
    2935             : 
    2936           0 :       CALL timeset(routineN, handle)
    2937             : 
    2938           0 :       CALL section_vals_val_get(as_input, "SOCKET%HOST", c_val=hostname)
    2939             : 
    2940           0 :       IF (ionode) THEN
    2941             :          ! signal the client to quit
    2942           0 :          CALL writebuffer(client_fd, "QUIT        ", header_len)
    2943             :          ! close the connection
    2944           0 :          CALL close_socket(client_fd)
    2945           0 :          CALL close_socket(socket_fd)
    2946             : 
    2947             :          ! delete the socket file
    2948           0 :          IF (file_exists(TRIM(hostname))) THEN
    2949           0 :             CALL remove_socket_file(TRIM(hostname)//C_NULL_CHAR)
    2950             :          END IF
    2951             :       END IF
    2952             : 
    2953           0 :       CALL timestop(handle)
    2954             : 
    2955           0 :    END SUBROUTINE finalize_socket
    2956             : 
    2957             : ! **************************************************************************************************
    2958             : !> \brief Sends the two-electron integrals to the client vie the socket
    2959             : !> \param client_fd the client file descriptor
    2960             : !> \param active_space_env active space environment
    2961             : !> \param para_env parallel environment
    2962             : ! **************************************************************************************************
    2963           0 :    SUBROUTINE send_eri_to_client(client_fd, active_space_env, para_env)
    2964             :       INTEGER, INTENT(IN)                                :: client_fd
    2965             :       TYPE(active_space_type), INTENT(IN), POINTER       :: active_space_env
    2966             :       TYPE(mp_para_env_type), INTENT(IN), POINTER        :: para_env
    2967             : 
    2968             :       CHARACTER(len=*), PARAMETER :: routineN = 'send_eri_to_client'
    2969             :       INTEGER, PARAMETER                                 :: header_len = 12
    2970             : 
    2971             :       CHARACTER(len=default_string_length)               :: header
    2972             :       INTEGER                                            :: handle, iw
    2973             :       LOGICAL                                            :: ionode
    2974           0 :       REAL(KIND=dp), ALLOCATABLE, DIMENSION(:)           :: eri_aa, eri_ab, eri_bb
    2975             :       TYPE(cp_logger_type), POINTER                      :: logger
    2976             : 
    2977           0 :       CALL timeset(routineN, handle)
    2978             : 
    2979           0 :       logger => cp_get_default_logger()
    2980           0 :       iw = cp_logger_get_default_io_unit(logger)
    2981           0 :       ionode = para_env%is_source()
    2982             : 
    2983             :       ! TODO: do we really need to allocate the arrays on every process?
    2984           0 :       ALLOCATE (eri_aa(active_space_env%nmo_active**4))
    2985           0 :       CALL eri_to_array(active_space_env%eri, eri_aa, active_space_env%active_orbitals, 1, 1)
    2986           0 :       IF (active_space_env%nspins == 2) THEN
    2987           0 :          ALLOCATE (eri_ab(active_space_env%nmo_active**4))
    2988           0 :          CALL eri_to_array(active_space_env%eri, eri_ab, active_space_env%active_orbitals, 1, 2)
    2989           0 :          ALLOCATE (eri_bb(active_space_env%nmo_active**4))
    2990           0 :          CALL eri_to_array(active_space_env%eri, eri_bb, active_space_env%active_orbitals, 2, 2)
    2991             :       END IF
    2992             : 
    2993             :       ! ask the status of the client
    2994           0 :       IF (ionode) CALL writebuffer(client_fd, "STATUS      ", header_len)
    2995             :       DO
    2996           0 :          header = ""
    2997           0 :          CALL para_env%sync()
    2998           0 :          IF (ionode) THEN
    2999             :             ! IF (iw > 0) WRITE(iw, *) "@SERVER: Waiting for messages..."
    3000           0 :             CALL readbuffer(client_fd, header, header_len)
    3001             :          END IF
    3002           0 :          CALL para_env%bcast(header, para_env%source)
    3003             : 
    3004             :          ! IF (iw > 0) WRITE(iw, *) "@SERVER: Message from client: ", TRIM(header)
    3005             : 
    3006           0 :          IF (TRIM(header) == "READY") THEN
    3007             :             ! if the client is ready, send the data
    3008           0 :             CALL para_env%sync()
    3009           0 :             IF (ionode) THEN
    3010           0 :                CALL writebuffer(client_fd, "TWOBODY     ", header_len)
    3011           0 :                CALL writebuffer(client_fd, active_space_env%nspins)
    3012           0 :                CALL writebuffer(client_fd, active_space_env%nmo_active)
    3013           0 :                CALL writebuffer(client_fd, active_space_env%nelec_active)
    3014           0 :                CALL writebuffer(client_fd, active_space_env%multiplicity)
    3015             :                ! send the alpha component
    3016           0 :                CALL writebuffer(client_fd, eri_aa, SIZE(eri_aa))
    3017             :                ! send the beta part for unrestricted calculations
    3018           0 :                IF (active_space_env%nspins == 2) THEN
    3019           0 :                   CALL writebuffer(client_fd, eri_ab, SIZE(eri_ab))
    3020           0 :                   CALL writebuffer(client_fd, eri_bb, SIZE(eri_bb))
    3021             :                END IF
    3022             :             END IF
    3023           0 :          ELSE IF (TRIM(header) == "RECEIVED") THEN
    3024             :             EXIT
    3025             :          END IF
    3026             :       END DO
    3027             : 
    3028           0 :       DEALLOCATE (eri_aa)
    3029           0 :       IF (active_space_env%nspins == 2) THEN
    3030           0 :          DEALLOCATE (eri_ab)
    3031           0 :          DEALLOCATE (eri_bb)
    3032             :       END IF
    3033             : 
    3034           0 :       CALL para_env%sync()
    3035             : 
    3036           0 :       CALL timestop(handle)
    3037             : 
    3038           0 :    END SUBROUTINE send_eri_to_client
    3039             : 
    3040             : ! **************************************************************************************************
    3041             : !> \brief Sends the one-electron embedding potential and the inactive energy to the client
    3042             : !> \param client_fd the client file descriptor
    3043             : !> \param active_space_env active space environment
    3044             : !> \param para_env parallel environment
    3045             : ! **************************************************************************************************
    3046           0 :    SUBROUTINE send_fock_to_client(client_fd, active_space_env, para_env)
    3047             :       INTEGER, INTENT(IN)                                :: client_fd
    3048             :       TYPE(active_space_type), INTENT(IN), POINTER       :: active_space_env
    3049             :       TYPE(mp_para_env_type), INTENT(IN), POINTER        :: para_env
    3050             : 
    3051             :       CHARACTER(len=*), PARAMETER :: routineN = 'send_fock_to_client'
    3052             :       INTEGER, PARAMETER                                 :: header_len = 12
    3053             : 
    3054             :       CHARACTER(len=default_string_length)               :: header
    3055             :       INTEGER                                            :: handle, iw
    3056             :       LOGICAL                                            :: debug, ionode
    3057           0 :       REAL(KIND=dp), ALLOCATABLE, DIMENSION(:)           :: fock_a, fock_b, p_act_mo_a, p_act_mo_b
    3058             :       TYPE(cp_logger_type), POINTER                      :: logger
    3059             : 
    3060           0 :       CALL timeset(routineN, handle)
    3061             : 
    3062             :       ! Set to .TRUE. to activate debug output
    3063           0 :       debug = .FALSE.
    3064             : 
    3065           0 :       logger => cp_get_default_logger()
    3066           0 :       iw = cp_logger_get_default_io_unit(logger)
    3067           0 :       ionode = para_env%is_source()
    3068             : 
    3069           0 :       ALLOCATE (p_act_mo_a(active_space_env%nmo_active**2))
    3070           0 :       ALLOCATE (fock_a(active_space_env%nmo_active**2))
    3071           0 :       IF (active_space_env%nspins == 2) THEN
    3072           0 :          ALLOCATE (p_act_mo_b(active_space_env%nmo_active**2))
    3073           0 :          ALLOCATE (fock_b(active_space_env%nmo_active**2))
    3074             :       END IF
    3075             : 
    3076             :       ! get the fock matrix into Fortran arrays
    3077             :       ASSOCIATE (act_indices => active_space_env%active_orbitals(:, 1))
    3078           0 :          CALL subspace_matrix_to_array(active_space_env%fock_sub(1), fock_a, act_indices, act_indices)
    3079             :       END ASSOCIATE
    3080             : 
    3081           0 :       IF (active_space_env%nspins == 2) THEN
    3082             :          ASSOCIATE (act_indices => active_space_env%active_orbitals(:, 2))
    3083           0 :             CALL subspace_matrix_to_array(active_space_env%fock_sub(2), fock_b, act_indices, act_indices)
    3084             :          END ASSOCIATE
    3085             :       END IF
    3086             : 
    3087             :       ! ask the status of the client
    3088           0 :       IF (ionode) CALL writebuffer(client_fd, "STATUS      ", header_len)
    3089             :       DO
    3090           0 :          header = ""
    3091             : 
    3092           0 :          CALL para_env%sync()
    3093           0 :          IF (ionode) THEN
    3094             :             IF (debug .AND. iw > 0) WRITE (iw, *) "@SERVER: Waiting for messages..."
    3095           0 :             CALL readbuffer(client_fd, header, header_len)
    3096             :          END IF
    3097           0 :          CALL para_env%bcast(header, para_env%source)
    3098             : 
    3099             :          IF (debug .AND. iw > 0) WRITE (iw, *) "@SERVER: Message from client: ", TRIM(header)
    3100             : 
    3101           0 :          IF (TRIM(header) == "READY") THEN
    3102             :             ! if the client is ready, send the data
    3103           0 :             CALL para_env%sync()
    3104           0 :             IF (ionode) THEN
    3105           0 :                CALL writebuffer(client_fd, "ONEBODY     ", header_len)
    3106           0 :                CALL writebuffer(client_fd, active_space_env%energy_inactive)
    3107             :                ! send the alpha component
    3108           0 :                CALL writebuffer(client_fd, fock_a, SIZE(fock_a))
    3109             :                ! send the beta part for unrestricted calculations
    3110           0 :                IF (active_space_env%nspins == 2) THEN
    3111           0 :                   CALL writebuffer(client_fd, fock_b, SIZE(fock_b))
    3112             :                END IF
    3113             :             END IF
    3114             : 
    3115           0 :          ELSE IF (TRIM(header) == "HAVEDATA") THEN
    3116             :             ! qiskit has data to transfer, let them know we want it and wait for it
    3117           0 :             CALL para_env%sync()
    3118           0 :             IF (ionode) THEN
    3119             :                IF (debug .AND. iw > 0) WRITE (iw, *) "@SERVER: Qiskit has data to transfer"
    3120           0 :                CALL writebuffer(client_fd, "GETDENSITY  ", header_len)
    3121             : 
    3122             :                ! read the active energy and density
    3123           0 :                CALL readbuffer(client_fd, active_space_env%energy_active)
    3124           0 :                CALL readbuffer(client_fd, p_act_mo_a, SIZE(p_act_mo_a))
    3125           0 :                IF (active_space_env%nspins == 2) THEN
    3126           0 :                   CALL readbuffer(client_fd, p_act_mo_b, SIZE(p_act_mo_b))
    3127             :                END IF
    3128             :             END IF
    3129             : 
    3130             :             ! broadcast the data to all processors
    3131           0 :             CALL para_env%bcast(active_space_env%energy_active, para_env%source)
    3132           0 :             CALL para_env%bcast(p_act_mo_a, para_env%source)
    3133           0 :             IF (active_space_env%nspins == 2) THEN
    3134           0 :                CALL para_env%bcast(p_act_mo_b, para_env%source)
    3135             :             END IF
    3136             : 
    3137             :             ! update total and reference energies in active space enviornment
    3138           0 :             active_space_env%energy_total = active_space_env%energy_inactive + active_space_env%energy_active
    3139             : 
    3140             :             ! update the active density matrix in the active space environment
    3141           0 :             CALL update_active_density(p_act_mo_a, active_space_env, 1)
    3142           0 :             IF (active_space_env%nspins == 2) THEN
    3143           0 :                CALL update_active_density(p_act_mo_b, active_space_env, 2)
    3144             :             END IF
    3145             : 
    3146             :             ! the non-iterative part is done, we can continue
    3147             :             EXIT
    3148             :          END IF
    3149             : 
    3150             :       END DO
    3151             : 
    3152           0 :       DEALLOCATE (p_act_mo_a)
    3153           0 :       DEALLOCATE (fock_a)
    3154           0 :       IF (active_space_env%nspins == 2) THEN
    3155           0 :          DEALLOCATE (p_act_mo_b)
    3156           0 :          DEALLOCATE (fock_b)
    3157             :       END IF
    3158             : 
    3159           0 :       CALL para_env%sync()
    3160             : 
    3161           0 :       CALL timestop(handle)
    3162             : 
    3163           0 :    END SUBROUTINE send_fock_to_client
    3164             : #endif
    3165             : 
    3166           0 : END MODULE qs_active_space_methods

Generated by: LCOV version 1.15