1
0
mirror of https://gitlab.com/scemama/qp_plugins_scemama.git synced 2025-01-10 13:08:15 +01:00
qp_plugins_scemama/devel/svdwf/psiSVD_naivBbyB_v2.irp.f

335 lines
9.8 KiB
FortranFixed
Raw Normal View History

2021-11-02 16:18:07 +01:00
program psiSVD_naivBbyB_v2
BEGIN_DOC
! diagonal H in the FSVD space
END_DOC
implicit none
read_wf = .True.
TOUCH read_wf
PROVIDE N_int
call run()
end
subroutine run
implicit none
integer(bit_kind) :: det1(N_int,2), det2(N_int,2)
integer :: degree, i_state
double precision :: h12, tol_h
integer :: na, nb
integer :: n_FSVD, n_TSVD
integer :: i, j, k, l
double precision, allocatable :: Uref(:,:), Dref(:), Vtref(:,:), Aref(:,:), Vref(:,:)
double precision :: t_beg, t_end
call wall_time(t_beg)
i_state = 1
det1(:,1) = psi_det_alpha_unique(:,1)
det2(:,1) = psi_det_alpha_unique(:,1)
det1(:,2) = psi_det_beta_unique(:,1)
det2(:,2) = psi_det_beta_unique(:,1)
call get_excitation_degree_spin(det1(1,1),det2(1,1),degree,N_int)
call get_excitation_degree(det1,det2,degree,N_int)
call i_H_j(det1, det2, N_int, h12)
na = n_det_alpha_unique
nb = n_det_beta_unique
n_FSVD = min(na,nb)
! ---------------------------------------------------------------------------------------
! construct the initial CI matrix
!
print *, ' ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~'
print *, ' CI matrix:', na, 'x', nb
print *, ' N det :', N_det
print *, ' ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~'
allocate( Aref(na,nb) )
Aref(:,:) = 0.d0
do k = 1, N_det
i = psi_bilinear_matrix_rows(k)
j = psi_bilinear_matrix_columns(k)
Aref(i,j) = psi_bilinear_matrix_values(k,i_state)
enddo
!
! ---------------------------------------------------------------------------------------
! ---------------------------------------------------------------------------------------
! perform a Full SVD
!
allocate( Uref(na,na), Dref(n_FSVD), Vtref(nb,nb) )
call svd_s(Aref, size(Aref,1), Uref, size(Uref,1), Dref, Vtref, size(Vtref,1), na, nb)
deallocate( Aref )
allocate( Vref(n_det_beta_unique,n_det_beta_unique) )
do l = 1, nb
do i = 1, nb
Vref(i,l) = Vtref(l,i)
enddo
enddo
deallocate( Vtref )
!
! ---------------------------------------------------------------------------------------
tol_h = 1d-12
do i = 2, n_FSVD, 10
call diag_hsvd(i, n_FSVD, tol_h, Uref, Vref, Dref)
enddo
call diag_hsvd(n_FSVD, n_FSVD, tol_h, Uref, Vref, Dref)
deallocate( Dref, Uref, Vref )
call wall_time(t_end)
print *, ' '
print *, " total elapsed time (min) = ", (t_end-t_beg)/60.d0
end
! ____________________________________________________________________________________________________________
!
! H_svd(a,b,c,d) = sum_{i,j,k,l} H_det(i,j,k,l) U(i,a) V(j,b) U(k,c) V(l,d)
!
subroutine diag_hsvd(n_TSVD, n_FSVD, tol_h, Uref, Vref, Dref)
implicit none
integer, intent(in) :: n_TSVD, n_FSVD
double precision, intent(in) :: tol_h
double precision, intent(in) :: Uref(n_det_alpha_unique,n_det_alpha_unique)
double precision, intent(in) :: Vref(n_det_beta_unique ,n_det_beta_unique)
double precision, intent(in) :: Dref(n_FSVD)
integer(bit_kind) :: det1(N_int,2), det2(N_int,2)
integer :: degree
double precision :: h12, x
integer :: na, nb, n_selec
integer :: i, j, k, l
integer :: a, b, c, d
double precision, allocatable :: H_SVD(:,:,:,:)
double precision, allocatable :: tmp1(:,:,:,:), tmp2(:,:,:,:)
integer :: ii, ind_gs
double precision :: E0, overlop
double precision, allocatable :: eigvec(:,:), eigval(:), check_ov(:)
double precision :: t1, t2, t3
na = n_det_alpha_unique
nb = n_det_beta_unique
n_selec = n_TSVD * n_TSVD
! ---------------------------------------------------------------------------------------
!
call wall_time(t1)
print *, ''
print *, ' start const psiHpsi in SVD space ...'
allocate( tmp1(nb,nb,n_TSVD,n_TSVD) )
tmp1(:,:,:,:) = 0.d0
!$OMP PARALLEL DEFAULT(NONE) &
!$OMP PRIVATE(a, c, i, j, k, l, h12, x, det1, det2, degree, tmp2) &
!$OMP SHARED(na, nb, psi_det_alpha_unique, psi_det_beta_unique, &
!$OMP N_int, n_TSVD, tol_h, Uref, tmp1)
allocate( tmp2(nb,nb,n_TSVD,n_TSVD) )
tmp2(:,:,:,:) = 0.d0
!$OMP DO COLLAPSE(2) SCHEDULE(DYNAMIC,8)
do l = 1, nb
do j = 1, nb
det2(:,2) = psi_det_beta_unique(:,l)
det1(:,2) = psi_det_beta_unique(:,j)
call get_excitation_degree_spin(det1(1,2),det2(1,2),degree,N_int)
if(degree .gt. 2) cycle
do k = 1, na
det2(:,1) = psi_det_alpha_unique(:,k)
do i = 1, na
det1(:,1) = psi_det_alpha_unique(:,i)
call get_excitation_degree(det1,det2,degree,N_int)
if(degree .gt. 2) cycle
call i_H_j(det1, det2, N_int, h12)
if(dabs(h12) .lt. tol_h) cycle
! tmp2(j,l,a,c) = sum_{i,k} H(i,j,k,l) U(i,a) U(k,c)
do c = 1, n_TSVD
x = Uref(k,c) * h12
if(dabs(x) .lt. tol_h) cycle
do a = 1, n_TSVD
tmp2(j,l,a,c) += Uref(i,a) * x
enddo
enddo
enddo
enddo
enddo
enddo
!$OMP END DO
! tmp1 = tmp2
!$OMP CRITICAL
do a = 1, n_TSVD
do c = 1, n_TSVD
do l = 1, nb
do j = 1, nb
tmp1(j,l,a,c) += tmp2(j,l,a,c)
enddo
enddo
enddo
enddo
!$OMP END CRITICAL
deallocate( tmp2 )
!$OMP END PARALLEL
! tmp2_{l,a,c,b} = sum_{j} tmp1_{j,l,a,c} V_{j,b}
allocate( tmp2(nb,n_TSVD,n_TSVD,n_TSVD) )
call DGEMM('T', 'N', nb*n_TSVD*n_TSVD, n_TSVD, nb, 1.d0 &
, tmp1, size(tmp1,1) &
, Vref, size(Vref,1) &
, 0.d0, tmp2, size(tmp2,1)*size(tmp2,2)*size(tmp2,3) )
deallocate(tmp1)
! tmp1_{d,a,c,b} = sum_{l} tmp2_{l,a,c,b} V_{l,d}
allocate( tmp1(n_TSVD,n_TSVD,n_TSVD,n_TSVD) )
call DGEMM('T', 'N', n_TSVD*n_TSVD*n_TSVD, n_TSVD, nb, 1.d0 &
, tmp2, size(tmp2,1) &
, Vref, size(Vref,1) &
, 0.d0, tmp1, size(tmp1,1)*size(tmp1,2)*size(tmp1,3) )
deallocate( tmp2 )
allocate( H_SVD(n_TSVD,n_TSVD,n_TSVD,n_TSVD) )
H_SVD(:,:,:,:) = 0.d0
do d = 1, n_TSVD
do c = 1, n_TSVD
do b = 1, n_TSVD
do a = 1, n_TSVD
H_SVD(a,b,c,d) = tmp1(d,a,c,b)
enddo
enddo
enddo
enddo
deallocate( tmp1 )
call wall_time(t2)
print *, ' end const psiHpsi in SVD space after (min)', (t2-t1)/60.d0
print *, ''
!
! ---------------------------------------------------------------------------------------
! ---------------------------------------------------------------------------------------
!
print *, ''
print *, ' diagonalize psiHpsi in SVD space ...'
allocate( eigvec(n_selec,n_selec), eigval(n_selec) )
call lapack_diag(eigval, eigvec, H_SVD, n_selec, n_selec)
deallocate(H_SVD)
! get the postsvd ground state
allocate( check_ov(n_selec) )
do l = 1, n_selec
overlop = 0.d0
do i = 1, n_TSVD
ii = i + (i-1) * n_TSVD
overlop = overlop + eigvec(ii,l) * Dref(i)
enddo
check_ov(l) = dabs(overlop)
enddo
ind_gs = MAXLOC( check_ov, DIM=1 )
overlop = check_ov(ind_gs)
E0 = eigval(ind_gs) + nuclear_repulsion
print *, ' - - - - - - - - - - - - '
print *, ' n_TSVD = ', n_TSVD
print *, ' space dimen = ', n_selec
print *, ' gs num = ', ind_gs
print *, ' overlop = ', overlop
print *, ' diag energy = ', E0
print *, ' eigen vector: '
do i = 1, n_selec
print *, eigvec(i,ind_gs)
enddo
print *, ' - - - - - - - - - - - - '
write(222, *) n_TSVD, n_selec, E0, overlop
deallocate( check_ov, eigval, eigvec )
call wall_time(t3)
print *, ' end diag in SVD space after (min)', (t3-t2)/60.d0
print *, ''
!
! ---------------------------------------------------------------------------------------
return
end subroutine diag_hsvd
! ____________________________________________________________________________________________________________
! ____________________________________________________________________________________________________________
! ! ---------------------------------------------------------------------------------------
! ! H0 in det basis
!
! call wall_time(ti)
! print *, ''
! print *, ' start const psiHpsi in det space ...'
!
! tol_h = 1d-15
! allocate( H0_det(na,nb,na,nb) )
! H0_det(:,:,:,:) = 0.d0
!
! !$OMP PARALLEL DEFAULT(NONE) &
! !$OMP PRIVATE(i, j, k, l, h12, det1, det2, degree) &
! !$OMP SHARED(na, nb, psi_det_alpha_unique,psi_det_beta_unique, &
! !$OMP N_int, H0, tol_h)
! !$OMP DO COLLAPSE(2) SCHEDULE(DYNAMIC,8)
! do l = 1, nb
! do j = 1, nb
! det2(:,2) = psi_det_beta_unique(:,l)
! det1(:,2) = psi_det_beta_unique(:,j)
! call get_excitation_degree_spin(det1(1,2),det2(1,2),degree,N_int)
! if(degree .gt. 2) cycle
! do k = 1, na
! det2(:,1) = psi_det_alpha_unique(:,k)
! do i = 1, na
! det1(:,1) = psi_det_alpha_unique(:,i)
! call get_excitation_degree(det1,det2,degree,N_int)
! if(degree .gt. 2) cycle
! call i_H_j(det1, det2, N_int, h12)
! if(dabs(h12) .lt. tol_h) cycle
! H0(i,j,k,l) = h12
! enddo
! enddo
! enddo
! enddo
! !$OMP END DO
! !$OMP END PARALLEL
!
! call wall_time(tf)
! print *, ' end const psiHpsi in det space after (min)', (ti-tf)/60.d0
! print *, ''
! ! ---------------------------------------------------------------------------------------