2014-06-20 18:35:26 +02:00
|
|
|
BEGIN_PROVIDER [ double precision, one_body_dm_mo_alpha, (mo_tot_num_align,mo_tot_num) ]
|
|
|
|
&BEGIN_PROVIDER [ double precision, one_body_dm_mo_beta, (mo_tot_num_align,mo_tot_num) ]
|
|
|
|
implicit none
|
|
|
|
BEGIN_DOC
|
|
|
|
! Alpha and beta one-body density matrix for each state
|
|
|
|
END_DOC
|
|
|
|
|
|
|
|
integer :: j,k,l,m
|
|
|
|
integer :: occ(N_int*bit_kind_size,2)
|
|
|
|
double precision :: ck, cl, ckl
|
|
|
|
double precision :: phase
|
|
|
|
integer :: h1,h2,p1,p2,s1,s2, degree
|
|
|
|
integer :: exc(0:2,2,2),n_occ_alpha
|
|
|
|
double precision, allocatable :: tmp_a(:,:), tmp_b(:,:)
|
|
|
|
|
2015-03-19 21:14:52 +01:00
|
|
|
if(only_single_double_dm)then
|
|
|
|
print*,'ONLY DOUBLE DM'
|
|
|
|
one_body_dm_mo_alpha = one_body_single_double_dm_mo_alpha
|
|
|
|
one_body_dm_mo_beta = one_body_single_double_dm_mo_beta
|
|
|
|
else
|
|
|
|
one_body_dm_mo_alpha = 0.d0
|
|
|
|
one_body_dm_mo_beta = 0.d0
|
|
|
|
!$OMP PARALLEL DEFAULT(NONE) &
|
|
|
|
!$OMP PRIVATE(j,k,l,m,occ,ck, cl, ckl,phase,h1,h2,p1,p2,s1,s2, degree,exc, &
|
|
|
|
!$OMP tmp_a, tmp_b, n_occ_alpha)&
|
|
|
|
!$OMP SHARED(psi_det,psi_coef,N_int,N_states,state_average_weight,elec_alpha_num,&
|
|
|
|
!$OMP elec_beta_num,one_body_dm_mo_alpha,one_body_dm_mo_beta,N_det,mo_tot_num_align,&
|
|
|
|
!$OMP mo_tot_num)
|
|
|
|
allocate(tmp_a(mo_tot_num_align,mo_tot_num), tmp_b(mo_tot_num_align,mo_tot_num) )
|
|
|
|
tmp_a = 0.d0
|
|
|
|
tmp_b = 0.d0
|
|
|
|
!$OMP DO SCHEDULE(dynamic)
|
|
|
|
do k=1,N_det
|
|
|
|
call bitstring_to_list(psi_det(1,1,k), occ(1,1), n_occ_alpha, N_int)
|
|
|
|
call bitstring_to_list(psi_det(1,2,k), occ(1,2), n_occ_alpha, N_int)
|
|
|
|
do m=1,N_states
|
|
|
|
ck = psi_coef(k,m)*psi_coef(k,m) * state_average_weight(m)
|
|
|
|
do l=1,elec_alpha_num
|
|
|
|
j = occ(l,1)
|
|
|
|
tmp_a(j,j) += ck
|
|
|
|
enddo
|
|
|
|
do l=1,elec_beta_num
|
|
|
|
j = occ(l,2)
|
|
|
|
tmp_b(j,j) += ck
|
|
|
|
enddo
|
|
|
|
enddo
|
|
|
|
do l=1,k-1
|
|
|
|
call get_excitation_degree(psi_det(1,1,k),psi_det(1,1,l),degree,N_int)
|
|
|
|
if (degree /= 1) then
|
|
|
|
cycle
|
|
|
|
endif
|
|
|
|
call get_mono_excitation(psi_det(1,1,k),psi_det(1,1,l),exc,phase,N_int)
|
|
|
|
call decode_exc(exc,degree,h1,p1,h2,p2,s1,s2)
|
|
|
|
do m=1,N_states
|
|
|
|
ckl = psi_coef(k,m) * psi_coef(l,m) * phase * state_average_weight(m)
|
|
|
|
if (s1==1) then
|
|
|
|
tmp_a(h1,p1) += ckl
|
|
|
|
tmp_a(p1,h1) += ckl
|
|
|
|
else
|
|
|
|
tmp_b(h1,p1) += ckl
|
|
|
|
tmp_b(p1,h1) += ckl
|
|
|
|
endif
|
|
|
|
enddo
|
|
|
|
enddo
|
|
|
|
enddo
|
|
|
|
!$OMP END DO NOWAIT
|
|
|
|
!$OMP CRITICAL
|
|
|
|
one_body_dm_mo_alpha = one_body_dm_mo_alpha + tmp_a
|
|
|
|
!$OMP END CRITICAL
|
|
|
|
!$OMP CRITICAL
|
|
|
|
one_body_dm_mo_beta = one_body_dm_mo_beta + tmp_b
|
|
|
|
!$OMP END CRITICAL
|
|
|
|
deallocate(tmp_a,tmp_b)
|
|
|
|
!$OMP BARRIER
|
|
|
|
!$OMP END PARALLEL
|
|
|
|
|
|
|
|
endif
|
|
|
|
END_PROVIDER
|
|
|
|
|
|
|
|
BEGIN_PROVIDER [ double precision, one_body_single_double_dm_mo_alpha, (mo_tot_num_align,mo_tot_num) ]
|
|
|
|
&BEGIN_PROVIDER [ double precision, one_body_single_double_dm_mo_beta, (mo_tot_num_align,mo_tot_num) ]
|
|
|
|
implicit none
|
|
|
|
BEGIN_DOC
|
|
|
|
! Alpha and beta one-body density matrix for each state
|
|
|
|
END_DOC
|
|
|
|
|
|
|
|
integer :: j,k,l,m
|
|
|
|
integer :: occ(N_int*bit_kind_size,2)
|
|
|
|
double precision :: ck, cl, ckl
|
|
|
|
double precision :: phase
|
|
|
|
integer :: h1,h2,p1,p2,s1,s2, degree
|
|
|
|
integer :: exc(0:2,2,2),n_occ_alpha
|
|
|
|
double precision, allocatable :: tmp_a(:,:), tmp_b(:,:)
|
|
|
|
integer :: degree_respect_to_HF_k
|
|
|
|
integer :: degree_respect_to_HF_l
|
|
|
|
|
|
|
|
PROVIDE elec_alpha_num elec_beta_num
|
|
|
|
|
|
|
|
one_body_single_double_dm_mo_alpha = 0.d0
|
|
|
|
one_body_single_double_dm_mo_beta = 0.d0
|
2014-06-20 18:35:26 +02:00
|
|
|
!$OMP PARALLEL DEFAULT(NONE) &
|
|
|
|
!$OMP PRIVATE(j,k,l,m,occ,ck, cl, ckl,phase,h1,h2,p1,p2,s1,s2, degree,exc, &
|
2015-03-19 21:14:52 +01:00
|
|
|
!$OMP tmp_a, tmp_b, n_occ_alpha,degree_respect_to_HF_k,degree_respect_to_HF_l)&
|
|
|
|
!$OMP SHARED(ref_bitmask,psi_det,psi_coef,N_int,N_states,state_average_weight,elec_alpha_num,&
|
|
|
|
!$OMP elec_beta_num,one_body_single_double_dm_mo_alpha,one_body_single_double_dm_mo_beta,N_det,mo_tot_num_align,&
|
2014-06-20 18:35:26 +02:00
|
|
|
!$OMP mo_tot_num)
|
|
|
|
allocate(tmp_a(mo_tot_num_align,mo_tot_num), tmp_b(mo_tot_num_align,mo_tot_num) )
|
|
|
|
tmp_a = 0.d0
|
|
|
|
tmp_b = 0.d0
|
|
|
|
!$OMP DO SCHEDULE(dynamic)
|
|
|
|
do k=1,N_det
|
|
|
|
call bitstring_to_list(psi_det(1,1,k), occ(1,1), n_occ_alpha, N_int)
|
|
|
|
call bitstring_to_list(psi_det(1,2,k), occ(1,2), n_occ_alpha, N_int)
|
2015-03-19 21:14:52 +01:00
|
|
|
call get_excitation_degree(ref_bitmask,psi_det(1,1,k),degree_respect_to_HF_k,N_int)
|
|
|
|
|
2014-06-20 18:35:26 +02:00
|
|
|
do m=1,N_states
|
|
|
|
ck = psi_coef(k,m)*psi_coef(k,m) * state_average_weight(m)
|
2015-03-19 21:14:52 +01:00
|
|
|
call get_excitation_degree(ref_bitmask,psi_det(1,1,k),degree_respect_to_HF_l,N_int)
|
|
|
|
if(degree_respect_to_HF_l.le.0)then
|
|
|
|
do l=1,elec_alpha_num
|
|
|
|
j = occ(l,1)
|
|
|
|
tmp_a(j,j) += ck
|
|
|
|
enddo
|
|
|
|
do l=1,elec_beta_num
|
|
|
|
j = occ(l,2)
|
|
|
|
tmp_b(j,j) += ck
|
|
|
|
enddo
|
|
|
|
endif
|
2014-06-20 18:35:26 +02:00
|
|
|
enddo
|
|
|
|
do l=1,k-1
|
2015-03-19 21:14:52 +01:00
|
|
|
call get_excitation_degree(ref_bitmask,psi_det(1,1,l),degree_respect_to_HF_l,N_int)
|
|
|
|
if(degree_respect_to_HF_k.ne.0)cycle
|
|
|
|
if(degree_respect_to_HF_l.eq.2.and.degree_respect_to_HF_k.ne.2)cycle
|
2014-06-20 18:35:26 +02:00
|
|
|
call get_excitation_degree(psi_det(1,1,k),psi_det(1,1,l),degree,N_int)
|
|
|
|
if (degree /= 1) then
|
|
|
|
cycle
|
|
|
|
endif
|
|
|
|
call get_mono_excitation(psi_det(1,1,k),psi_det(1,1,l),exc,phase,N_int)
|
|
|
|
call decode_exc(exc,degree,h1,p1,h2,p2,s1,s2)
|
|
|
|
do m=1,N_states
|
|
|
|
ckl = psi_coef(k,m) * psi_coef(l,m) * phase * state_average_weight(m)
|
|
|
|
if (s1==1) then
|
|
|
|
tmp_a(h1,p1) += ckl
|
|
|
|
tmp_a(p1,h1) += ckl
|
|
|
|
else
|
|
|
|
tmp_b(h1,p1) += ckl
|
|
|
|
tmp_b(p1,h1) += ckl
|
|
|
|
endif
|
|
|
|
enddo
|
2015-03-19 21:14:52 +01:00
|
|
|
enddo
|
2014-06-20 18:35:26 +02:00
|
|
|
enddo
|
|
|
|
!$OMP END DO NOWAIT
|
|
|
|
!$OMP CRITICAL
|
2015-03-19 21:14:52 +01:00
|
|
|
one_body_single_double_dm_mo_alpha = one_body_single_double_dm_mo_alpha + tmp_a
|
2014-06-20 18:35:26 +02:00
|
|
|
!$OMP END CRITICAL
|
|
|
|
!$OMP CRITICAL
|
2015-03-19 21:14:52 +01:00
|
|
|
one_body_single_double_dm_mo_beta = one_body_single_double_dm_mo_beta + tmp_b
|
2014-06-20 18:35:26 +02:00
|
|
|
!$OMP END CRITICAL
|
|
|
|
deallocate(tmp_a,tmp_b)
|
|
|
|
!$OMP BARRIER
|
|
|
|
!$OMP END PARALLEL
|
|
|
|
END_PROVIDER
|
|
|
|
|
|
|
|
BEGIN_PROVIDER [ double precision, one_body_dm_mo, (mo_tot_num_align,mo_tot_num) ]
|
|
|
|
implicit none
|
|
|
|
BEGIN_DOC
|
|
|
|
! One-body density matrix
|
|
|
|
END_DOC
|
|
|
|
one_body_dm_mo = one_body_dm_mo_alpha + one_body_dm_mo_beta
|
|
|
|
END_PROVIDER
|
|
|
|
|
2015-03-19 21:14:52 +01:00
|
|
|
BEGIN_PROVIDER [ double precision, one_body_spin_density_mo, (mo_tot_num_align,mo_tot_num) ]
|
|
|
|
implicit none
|
|
|
|
BEGIN_DOC
|
|
|
|
! rho(alpha) - rho(beta)
|
|
|
|
END_DOC
|
|
|
|
one_body_spin_density_mo = one_body_dm_mo_alpha - one_body_dm_mo_beta
|
|
|
|
END_PROVIDER
|
|
|
|
|
2014-06-25 14:58:58 +02:00
|
|
|
subroutine set_natural_mos
|
2014-06-20 18:35:26 +02:00
|
|
|
implicit none
|
|
|
|
BEGIN_DOC
|
2014-06-25 14:58:58 +02:00
|
|
|
! Set natural orbitals, obtained by diagonalization of the one-body density matrix in the MO basis
|
2014-06-20 18:35:26 +02:00
|
|
|
END_DOC
|
|
|
|
character*(64) :: label
|
|
|
|
double precision, allocatable :: tmp(:,:)
|
|
|
|
allocate(tmp(size(one_body_dm_mo,1),size(one_body_dm_mo,2)))
|
|
|
|
|
|
|
|
! Negation to have the occupied MOs first after the diagonalization
|
|
|
|
tmp = -one_body_dm_mo
|
|
|
|
label = "Natural"
|
|
|
|
call mo_as_eigvectors_of_mo_matrix(tmp,size(tmp,1),size(tmp,2),label)
|
|
|
|
deallocate(tmp)
|
|
|
|
|
|
|
|
end
|
2014-06-25 14:58:58 +02:00
|
|
|
subroutine save_natural_mos
|
|
|
|
implicit none
|
|
|
|
BEGIN_DOC
|
|
|
|
! Save natural orbitals, obtained by diagonalization of the one-body density matrix in the MO basis
|
|
|
|
END_DOC
|
|
|
|
call set_natural_mos
|
2015-03-19 21:14:52 +01:00
|
|
|
call save_mos
|
2014-06-25 14:58:58 +02:00
|
|
|
|
|
|
|
end
|
|
|
|
|
2014-06-20 18:35:26 +02:00
|
|
|
|
|
|
|
BEGIN_PROVIDER [ double precision, state_average_weight, (N_states) ]
|
|
|
|
implicit none
|
|
|
|
BEGIN_DOC
|
|
|
|
! Weights in the state-average calculation of the density matrix
|
|
|
|
END_DOC
|
|
|
|
state_average_weight = 1.d0/dble(N_states)
|
|
|
|
END_PROVIDER
|
|
|
|
|