quantum-espresso/atomic/scf.f90

115 lines
3.3 KiB
Fortran
Raw Normal View History

!
! Copyright (C) 2004 PWSCF group
! This file is distributed under the terms of the
! GNU General Public License. See the file `License'
! in the root directory of the present distribution,
! or http://www.gnu.org/copyleft/gpl.txt .
!
!---------------------------------------------------------------
subroutine scf
!---------------------------------------------------------------
!
! this routine performs the atomic self-consistent procedure
! self-interaction-correction allowed
!
use constants, only: e2
use ld1inc
implicit none
logical:: conv
integer:: nerr, nstop, n, i, is, id, nin, mch
real(kind=dp) :: vnew(ndm,2), rhoc1(ndm), ze2
real(kind=dp), allocatable :: vsic(:,:), vsicnew(:), vhn1(:), egc(:)
integer, parameter :: maxter=200
real(kind=dp), parameter :: thresh=1.0e-10_dp
!
!
ze2 = - zed * e2
rhoc1=0.0_dp
id=3
psi=0.0_dp
!
if (isic /= 0) then
allocate(vsic(ndm,nwf), vsicnew(ndm), vhn1(ndm), egc(ndm))
vsic=0.0_dp
! id=1
endif
do iter=1,maxter
nerr=0
vnew=vpot
do n=1,nwf
if (oc(n) >= 0.0_dp) then
is=isw(n)
if (isic /= 0 .and. iter > 1) vnew(:,is)=vpot(:,is)-vsic(:,n)
if (rel == 0) then
call ascheq (nn(n),ll(n),enl(n),mesh,dx,r,r2, &
sqr,vnew(1,is),ze2,thresh,psi(1,1,n),nstop)
elseif (rel == 1) then
call lschps (1,zed,exp(dx),dx,mesh,nin,mch, &
nn(n),ll(n),enl(n),psi(1,1,n),r,vnew(1,is))
nstop=0
elseif (rel == 2) then
call dirsol (ndm,mesh,nn(n),ll(n),jj(n),iter,enl(n), &
thresh,dx,psi(1,1,n),r,rab,vnew(1,is))
nstop=0
else
call errore('scf','relativistic not programmed',1)
endif
! write(6,*) el(n),enl(n)
! if (nstop /= 0) write(6,'(4i6)') iter,nn(n),ll(n),nstop
nerr=nerr+nstop
else
enl(n)=0.0_dp
psi(:,:,n)=0.0_dp
endif
enddo
!
! calculate charge density (spherical approximation)
!
rho=0.0_dp
do n=1,nwf
do i=1,mesh
rho(i,isw(n))=rho(i,isw(n))+oc(n)*(psi(i,1,n)**2+psi(i,2,n)**2)
enddo
enddo
!
! calculate new potential
!
call new_potential(ndm,mesh,r,r2,sqr,dx,zed,vxt, &
lsd,.false.,latt,enne,rhoc1,rho,vh,vnew)
!
! calculate SIC correction potential (if present)
!
if (isic /= 0) then
do n=1,nwf
if (oc(n) >= 0.0_dp) then
is=isw(n)
call sic_correction(n,vhn1,vsicnew,egc)
!
! use simple mixing for SIC correction
!
vsic(:,n) = (1.0_dp-beta)*vsic(:,n)+beta*vsicnew(:)
end if
enddo
endif
!
! mix old and new potential
!
call vpack(mesh,ndm,nspin,vnew,vpot,1)
call dmixp(mesh*nspin,vnew,vpot,beta,tr2,iter,id,eps0,conv)
call vpack(mesh,ndm,nspin,vnew,vpot,-1)
! write(6,*) iter, eps0
!
if (conv) then
if (nerr /= 0) call errore('scf','errors in KS equations',-1)
goto 45
endif
enddo
call errore('scf','warning: convergence not achieved',-1)
45 if (isic /= 0) then
deallocate(egc, vhn1, vsicnew, vsic)
endif
return
end subroutine scf