9
1
mirror of https://github.com/QuantumPackage/qp2.git synced 2025-01-05 09:58:42 +01:00

add maximum overlap method in hartree fock

This commit is contained in:
Antoine Marie 2024-09-18 09:43:11 +02:00
parent df01997603
commit b68fe9feb2
3 changed files with 146 additions and 38 deletions

View File

@ -45,6 +45,12 @@ type: double precision
doc: Calculated HF energy doc: Calculated HF energy
interface: ezfio interface: ezfio
[do_mom]
type: logical
doc: If true, this will run a MOM calculation. The overlap will be computed at each step with respect to the initial MOs. After an initial Hartree-Fock calculation, the guess can be created by swapping molecular orbitals through the qp run swap_mos command.
interface: ezfio,provider,ocaml
default: False
[frozen_orb_scf] [frozen_orb_scf]
type: logical type: logical
doc: If true, leave untouched all the orbitals defined as core and optimize all the orbitals defined as active with qp_set_mo_class doc: If true, leave untouched all the orbitals defined as core and optimize all the orbitals defined as active with qp_set_mo_class

View File

@ -0,0 +1,88 @@
subroutine reorder_mo_max_overlap
implicit none
BEGIN_DOC
! routines that compute the projection of each MO of the current `mo_coef` on the space spanned by the occupied orbitals of `mo_coef_begin_iteration`
END_DOC
integer :: i,j,k,l
double precision, allocatable :: overlap(:,:)
double precision, allocatable :: proj(:)
integer, allocatable :: iorder(:)
double precision, allocatable :: mo_coef_tmp(:,:)
allocate(overlap(mo_num,mo_num),proj(mo_num),iorder(mo_num),mo_coef_tmp(ao_num,mo_num))
overlap(:,:) = 0d0
mo_coef_tmp(:,:) = 0d0
proj(:) = 0d0
iorder(:) = 0d0
! this loop compute the overlap between the initial and the current MOS
do i = 1, mo_num ! old mo
do j = 1, mo_num ! curent mo
do k = 1, ao_num
do l = 1, ao_num
overlap(i,j) += mo_coef_begin_iteration(k,i)* ao_overlap(k,l) * mo_coef(l,j)
enddo
enddo
enddo
enddo
! for each orbital compute the best overlap
do i = 1, mo_num
iorder(i) = i ! initialize the iorder list as we need it to sort later
do j = 1, elec_alpha_num
proj(i) += overlap(j,i)*overlap(j,i) ! compute the projection of current orbital i on the occupied space of the initial orbitals
enddo
proj(i) = dsqrt(proj(i))
enddo
! sort the list of projection to find the mos with the largest overlap
call dsort(proj(:),iorder(:),mo_num)
! reorder orbitals according to projection
do i=1,mo_num
mo_coef_tmp(:,i) = mo_coef(:,iorder(mo_num+1-i))
enddo
! update the orbitals
mo_coef(:,:) = mo_coef_tmp(:,:)
! if the determinant is open-shell we need to make sure that the singly occupied orbital correspond to the initial ones
if (elec_alpha_num > elec_beta_num) then
double precision, allocatable :: overlap_alpha(:,:)
double precision, allocatable :: proj_alpha(:)
integer, allocatable :: iorder_alpha(:)
allocate(overlap_alpha(mo_num,elec_alpha_num),proj_alpha(elec_alpha_num),iorder_alpha(elec_alpha_num))
overlap_alpha(:,:) = 0d0
proj_alpha(:) = 0d0
iorder_alpha(:) = 0d0
do i = 1, mo_num ! old mo
do j = 1, elec_alpha_num ! curent mo
do k = 1, ao_num
do l = 1, ao_num
overlap_alpha(i,j) += mo_coef_begin_iteration(k,i) * ao_overlap(k,l) * mo_coef(l,j)
enddo
enddo
enddo
enddo
do i = 1, elec_alpha_num
iorder_alpha(i) = i ! initialize the iorder list as we need it to sort later
do j = 1, elec_beta_num
proj_alpha(i) += overlap_alpha(j,i)*overlap_alpha(j,i) ! compute the projection of current orbital i on the beta occupied space of the initial orbitals
enddo
proj_alpha(i) = dsqrt(proj_alpha(i))
enddo
! sort the list of projection to find the mos with the largest overlap
call dsort(proj_alpha(:),iorder_alpha(:),elec_alpha_num)
! reorder orbitals according to projection
do i=1,elec_alpha_num
mo_coef_tmp(:,i) = mo_coef(:,iorder_alpha(elec_alpha_num+1-i))
enddo
do i=1,elec_alpha_num
mo_coef(:,i) = mo_coef_tmp(:,i)
enddo
endif
deallocate(overlap, proj, iorder, mo_coef_tmp)
end

View File

@ -51,6 +51,11 @@ END_DOC
! !
PROVIDE FPS_SPF_matrix_AO Fock_matrix_AO PROVIDE FPS_SPF_matrix_AO Fock_matrix_AO
! Initialize MO to run IMOM
if(do_mom)then
call initialize_mo_coef_begin_iteration
endif
converged = .False. converged = .False.
do while ( .not.converged .and. (iteration_SCF < n_it_SCF_max) ) do while ( .not.converged .and. (iteration_SCF < n_it_SCF_max) )
@ -88,16 +93,17 @@ END_DOC
Fock_matrix_AO_beta = Fock_matrix_AO*0.5d0 Fock_matrix_AO_beta = Fock_matrix_AO*0.5d0
TOUCH Fock_matrix_AO_alpha Fock_matrix_AO_beta TOUCH Fock_matrix_AO_alpha Fock_matrix_AO_beta
endif endif
MO_coef = eigenvectors_Fock_matrix_MO MO_coef = eigenvectors_Fock_matrix_MO
if(do_mom)then
call reorder_mo_max_overlap
endif
if(frozen_orb_scf)then if(frozen_orb_scf)then
call reorder_core_orb call reorder_core_orb
call initialize_mo_coef_begin_iteration call initialize_mo_coef_begin_iteration
endif endif
TOUCH MO_coef TOUCH MO_coef
! Calculate error vectors ! Calculate error vectors
max_error_DIIS = maxval(Abs(FPS_SPF_Matrix_MO)) max_error_DIIS = maxval(Abs(FPS_SPF_Matrix_MO))
@ -106,41 +112,46 @@ END_DOC
energy_SCF = SCF_energy energy_SCF = SCF_energy
Delta_Energy_SCF = energy_SCF - energy_SCF_previous Delta_Energy_SCF = energy_SCF - energy_SCF_previous
if ( (SCF_algorithm == 'DIIS').and.(Delta_Energy_SCF > 0.d0) ) then ! if ( (SCF_algorithm == 'DIIS').and.(Delta_Energy_SCF > 0.d0) ) then
Fock_matrix_AO(1:ao_num,1:ao_num) = Fock_matrix_DIIS (1:ao_num,1:ao_num,index_dim_DIIS) ! Fock_matrix_AO(1:ao_num,1:ao_num) = Fock_matrix_DIIS (1:ao_num,1:ao_num,index_dim_DIIS)
Fock_matrix_AO_alpha = Fock_matrix_AO*0.5d0 ! Fock_matrix_AO_alpha = Fock_matrix_AO*0.5d0
Fock_matrix_AO_beta = Fock_matrix_AO*0.5d0 ! Fock_matrix_AO_beta = Fock_matrix_AO*0.5d0
TOUCH Fock_matrix_AO_alpha Fock_matrix_AO_beta ! TOUCH Fock_matrix_AO_alpha Fock_matrix_AO_beta
endif ! endif
double precision :: level_shift_save if (.not.do_mom) then
level_shift_save = level_shift double precision :: level_shift_save
mo_coef_save(1:ao_num,1:mo_num) = mo_coef(1:ao_num,1:mo_num) level_shift_save = level_shift
do while (Delta_energy_SCF > 0.d0) mo_coef_save(1:ao_num,1:mo_num) = mo_coef(1:ao_num,1:mo_num)
mo_coef(1:ao_num,1:mo_num) = mo_coef_save do while (Delta_energy_SCF > 0.d0)
if (level_shift <= .1d0) then mo_coef(1:ao_num,1:mo_num) = mo_coef_save
level_shift = 1.d0 if (level_shift <= .1d0) then
else level_shift = 1.d0
level_shift = level_shift * 3.0d0 else
endif level_shift = level_shift * 3.0d0
TOUCH mo_coef level_shift endif
mo_coef(1:ao_num,1:mo_num) = eigenvectors_Fock_matrix_MO(1:ao_num,1:mo_num) TOUCH mo_coef level_shift
if(frozen_orb_scf)then mo_coef(1:ao_num,1:mo_num) = eigenvectors_Fock_matrix_MO(1:ao_num,1:mo_num)
call reorder_core_orb if(do_mom)then
call initialize_mo_coef_begin_iteration call reorder_mo_max_overlap
endif endif
TOUCH mo_coef if(frozen_orb_scf)then
Delta_Energy_SCF = SCF_energy - energy_SCF_previous call reorder_core_orb
energy_SCF = SCF_energy call initialize_mo_coef_begin_iteration
if (level_shift-level_shift_save > 40.d0) then endif
level_shift = level_shift_save * 4.d0 TOUCH mo_coef
SOFT_TOUCH level_shift Delta_Energy_SCF = SCF_energy - energy_SCF_previous
exit energy_SCF = SCF_energy
endif if (level_shift-level_shift_save > 40.d0) then
dim_DIIS=0 level_shift = level_shift_save * 4.d0
enddo SOFT_TOUCH level_shift
level_shift = level_shift * 0.5d0 exit
SOFT_TOUCH level_shift endif
dim_DIIS=0
enddo
level_shift = level_shift * 0.5d0
SOFT_TOUCH level_shift
endif
energy_SCF_previous = energy_SCF energy_SCF_previous = energy_SCF
converged = ( (max_error_DIIS <= threshold_DIIS_nonzero) .and. & converged = ( (max_error_DIIS <= threshold_DIIS_nonzero) .and. &
@ -205,7 +216,7 @@ END_DOC
if(.not.frozen_orb_scf)then if(.not.frozen_orb_scf)then
call mo_as_eigvectors_of_mo_matrix(Fock_matrix_mo,size(Fock_matrix_mo,1), & call mo_as_eigvectors_of_mo_matrix(Fock_matrix_mo,size(Fock_matrix_mo,1), &
size(Fock_matrix_mo,2),mo_label,1,.true.) size(Fock_matrix_mo,2),mo_label,1,.true.)
call restore_symmetry(ao_num, mo_num, mo_coef, size(mo_coef,1), 1.d-10) call restore_symmetry(ao_num, mo_num, mo_coef, size(mo_coef,1), 1.d-10)
call orthonormalize_mos call orthonormalize_mos
endif endif
@ -228,6 +239,9 @@ END_DOC
i = j+1 i = j+1
enddo enddo
if(do_mom)then
call reorder_mo_max_overlap
endif
call save_mos call save_mos