mirror of https://gitlab.com/QEF/q-e.git
891 lines
26 KiB
Fortran
891 lines
26 KiB
Fortran
!
|
|
! Copyright (C) 2001-2008 Quantum ESPRESSO 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 ep_matrix_element_wannier()
|
|
!-----------------------------------------------------------------------
|
|
!
|
|
! Electron-phonon calculation from data saved in fildvscf
|
|
!
|
|
USE kinds, ONLY : DP
|
|
USE cell_base, ONLY : celldm, omega, ibrav
|
|
USE ions_base, ONLY : nat, ntyp => nsp, ityp, tau, amass
|
|
USE gvecs, ONLY: doublegrid
|
|
USE fft_base, ONLY : dfftp, dffts
|
|
USE fft_interfaces, ONLY : fft_interpolate
|
|
USE noncollin_module, ONLY : nspin_mag, noncolin
|
|
USE dynmat, ONLY : dyn, w2
|
|
USE modes, ONLY : npert, nirr, u
|
|
USE control_ph, ONLY : trans
|
|
USE units_ph, ONLY : iudyn, lrdrho, iudvscf
|
|
USE io_global, ONLY : stdout
|
|
USE mp_pools, ONLY : me_pool, root_pool
|
|
USE klist, ONLY : xk
|
|
USE el_phon, ONLY: elph_mat, kpq, g_kpq, igqg, xk_gamma
|
|
USE uspp, ONLY: okvan
|
|
USE paw_variables, ONLY : okpaw
|
|
USE uspp_param, ONLY : nhm
|
|
USE lsda_mod, ONLY : nspin
|
|
|
|
USE lrus, ONLY : int3, int3_nc, int3_paw
|
|
USE qpoint, ONLY : xq, nksq, ikks
|
|
!
|
|
IMPLICIT NONE
|
|
!
|
|
LOGICAL :: read_dvscf_cart, ascii_dvscf
|
|
INTEGER :: irr, imode0, ipert, is, ik
|
|
! counter on the representations
|
|
! counter on the modes
|
|
! the change of Vscf due to perturbations
|
|
COMPLEX(DP), POINTER :: dvscfin(:,:,:), dvscfins (:,:,:)
|
|
|
|
|
|
CALL start_clock ('elphon')
|
|
|
|
|
|
ascii_dvscf=.false.
|
|
if(elph_mat) read_dvscf_cart=.true.
|
|
if(read_dvscf_cart) then
|
|
write(stdout,*)
|
|
write(stdout,*) 'Reading dvscf in cartesian coordinates !'
|
|
write(stdout,*)
|
|
|
|
u=(0.d0,0.d0)
|
|
do irr=1,3*nat
|
|
u(irr,irr)=(1.d0,0.d0)
|
|
enddo
|
|
|
|
|
|
! if(ascii_dvscf) then
|
|
! ALLOCATE (dvrot ( nrxx , nspin , 3*nat) )
|
|
! fildvscf_asc=trim(tmp_dir)//trim(prefix)//"."//trim(fildvscf)//'1'
|
|
! open(unit=7899,file=fildvscf_asc,status='unknown')
|
|
! dvrot=(0.0,0.0)
|
|
! do na=1,nat
|
|
! do ipol=1,3
|
|
! irr=(na-1)*3+ipol
|
|
! do k = 1, dfftp%nr3
|
|
! do j = 1, dfftp%nr2
|
|
! do i = 1, dfftp%nr1
|
|
! read(7899,*) n, rep,imp
|
|
! dvrot(n,1,irr)=CMPLX(rep,imp,kind=dp)
|
|
! enddo
|
|
! enddo
|
|
! enddo
|
|
! enddo
|
|
! enddo
|
|
! close(7899)
|
|
! endif
|
|
|
|
endif
|
|
|
|
|
|
|
|
!
|
|
! read Delta Vscf and calculate electron-phonon coefficients
|
|
!
|
|
imode0 = 0
|
|
DO irr = 1, nirr
|
|
ALLOCATE (dvscfin (dfftp%nnr, nspin_mag , npert(irr)) )
|
|
IF (okvan) THEN
|
|
ALLOCATE (int3 ( nhm, nhm, nat, nspin_mag, npert(irr)))
|
|
IF (okpaw) ALLOCATE (int3_paw (nhm, nhm, nat, nspin_mag, npert(irr)))
|
|
IF (noncolin) ALLOCATE(int3_nc( nhm, nhm, nat, nspin, npert(irr)))
|
|
ENDIF
|
|
|
|
! if(ascii_dvscf) then
|
|
! DO ipert = 1, npert(irr)
|
|
! dvscfin(1:dfftp%nnr,:,ipert)=dvrot(1:dfftp%nnr,:,imode0+ipert)
|
|
! enddo
|
|
! else
|
|
DO ipert = 1, npert (irr)
|
|
CALL davcio_drho ( dvscfin(1,1,ipert), lrdrho, iudvscf, &
|
|
imode0 + ipert, -1 )
|
|
END DO
|
|
! endif
|
|
IF (doublegrid) THEN
|
|
ALLOCATE (dvscfins (dffts%nnr, nspin_mag , npert(irr)) )
|
|
DO is = 1, nspin_mag
|
|
DO ipert = 1, npert(irr)
|
|
CALL fft_interpolate (dfftp, dvscfin(:,is,ipert), dffts, dvscfins(:,is,ipert))
|
|
ENDDO
|
|
ENDDO
|
|
ELSE
|
|
dvscfins => dvscfin
|
|
ENDIF
|
|
CALL newdq (dvscfin, npert(irr))
|
|
CALL elphel_refolded (npert (irr), imode0, dvscfins)
|
|
!
|
|
imode0 = imode0 + npert (irr)
|
|
IF (doublegrid) DEALLOCATE (dvscfins)
|
|
DEALLOCATE (dvscfin)
|
|
|
|
IF (okvan) THEN
|
|
DEALLOCATE (int3)
|
|
IF (okpaw) DEALLOCATE (int3_paw)
|
|
IF (noncolin) DEALLOCATE(int3_nc)
|
|
ENDIF
|
|
|
|
ENDDO
|
|
!
|
|
! now read the eigenvalues and eigenvectors of the dynamical matrix
|
|
! calculated in a previous run
|
|
!
|
|
! IF (.NOT.trans) CALL readmat (iudyn, ibrav, celldm, nat, ntyp, &
|
|
! ityp, omega, amass, tau, xq, w2, dyn)
|
|
|
|
IF (.NOT.trans) CALL readmat_findq (iudyn, ibrav, celldm, nat, ntyp, &
|
|
ityp, omega, amass, tau, xq, w2, dyn)
|
|
!
|
|
|
|
|
|
deallocate(xk_gamma)
|
|
deallocate(kpq,g_kpq,igqg)
|
|
|
|
! CALL stop_clock ('elphon')
|
|
RETURN
|
|
END SUBROUTINE ep_matrix_element_wannier
|
|
|
|
!-----------------------------------------------------------------------
|
|
SUBROUTINE elphsum_wannier(q_index)
|
|
!-----------------------------------------------------------------------
|
|
!
|
|
! Sum over BZ of the electron-phonon matrix elements el_ph_mat
|
|
! Original routine written by Francesco Mauri
|
|
! Adapted to wannier functions by Matteo Calandra
|
|
! Dev. Comment: missing calc_sigma_yet
|
|
!-----------------------------------------------------------------------
|
|
USE kinds, ONLY : DP
|
|
USE ions_base, ONLY : nat, ityp, tau,amass,tau, ntyp => nsp, atm
|
|
USE cell_base, ONLY : at, bg, ibrav, celldm
|
|
USE symm_base, ONLY : s, sr, irt, nsym, time_reversal, invs, ftau, copy_sym, inverse_s
|
|
USE klist, ONLY : xk, nelec
|
|
USE wvfct, ONLY : nbnd, et
|
|
USE el_phon
|
|
USE mp_pools, ONLY : me_pool, root_pool, inter_pool_comm, npool
|
|
USE mp_bands, ONLY : intra_bgrp_comm
|
|
USE io_global, ONLY : stdout,ionode
|
|
USE io_files, ONLY : prefix
|
|
USE dynmat, ONLY : dyn, w2
|
|
USE modes, ONLY : u
|
|
USE lsda_mod, only : isk,nspin, current_spin,lsda
|
|
USE mp, ONLY: mp_sum
|
|
|
|
USE lr_symm_base, ONLY : irotmq, irgq, gimq, gi
|
|
USE qpoint, ONLY : xq, nksq, ikks, ikqs
|
|
USE control_lr, ONLY : lgamma
|
|
USE noncollin_module, ONLY : noncolin
|
|
!
|
|
IMPLICIT NONE
|
|
!
|
|
LOGICAL :: lborn
|
|
INTEGER :: q_index
|
|
!
|
|
!
|
|
logical :: minus_qloc,sym (48)
|
|
integer :: nq, imq, isq(48)
|
|
INTEGER :: ik, ikk, ikq, ibnd, jbnd, ipert, jpert, nu, mu, &
|
|
ios, iuelphmat,i,j,nt,k
|
|
INTEGER :: iu_sym,nmodes,nsymq
|
|
INTEGER :: io_file_unit
|
|
! for star_q
|
|
REAL(DP) :: rtauloc(3,48,nat)
|
|
! end of star_q definitions
|
|
real(DP) :: sxq (3, 48)
|
|
REAL(DP) xk_dummy(3)
|
|
character(len=80) :: filelph
|
|
CHARACTER(len=256) :: file_elphmat
|
|
!
|
|
COMPLEX(DP) :: el_ph_sum (3*nat,3*nat), dyn_corr(3*nat,3*nat)
|
|
|
|
INTEGER, EXTERNAL :: find_free_unit
|
|
CHARACTER (LEN=6), EXTERNAL :: int_to_char
|
|
|
|
nmodes=3*nat
|
|
|
|
write(filelph,'(A5,f9.6,A1,f9.6,A1,f9.6)') 'elph.',xq(1),'.',xq(2),'.',xq(3)
|
|
file_elphmat=trim(adjustl(prefix))//'_elph.mat.q_'// TRIM( int_to_char( q_index ) )
|
|
|
|
lborn=.false.
|
|
! parallel case: only first node writes
|
|
IF ( .not.ionode ) THEN
|
|
iuelphmat = 0
|
|
ELSE
|
|
!
|
|
! First I dump information for the electron-phonon interaction
|
|
!
|
|
|
|
!
|
|
iuelphmat = find_free_unit()
|
|
OPEN (unit = iuelphmat, file = file_elphmat, status = 'unknown', err = &
|
|
111, iostat = ios, form='unformatted')
|
|
111 CALL errore ('elphsum_wannier', 'opening file'//file_elphmat, ABS (ios) )
|
|
REWIND (iuelphmat)
|
|
xk_dummy(:)=xq(:)
|
|
call cryst_to_cart(1,xk_dummy,at,-1)
|
|
WRITE (iuelphmat) xk_dummy
|
|
WRITE (iuelphmat) noncolin, nspin, lborn
|
|
WRITE (iuelphmat) nelec
|
|
WRITE (iuelphmat) elph_nbnd_min,elph_nbnd_max,nbnd
|
|
WRITE (iuelphmat) nmodes, nksq, nat, ntyp
|
|
WRITE (iuelphmat) ibrav,(celldm(j),j=1,6)
|
|
WRITE(iuelphmat) (atm(j),j=1,ntyp),(amass(j),j=1,ntyp), &
|
|
(ityp(j),j=1,nat),((tau(j,i),j=1,3),i=1,nat)
|
|
WRITE (iuelphmat) (w2 (nu) , nu = 1, nmodes)
|
|
WRITE (iuelphmat) ((u(ipert,jpert),ipert=1,nmodes),jpert=1,nmodes)
|
|
WRITE (iuelphmat) ((dyn(ipert,jpert),ipert=1,3*nat),jpert=1,3*nat)
|
|
do ik=1,nksq
|
|
ikk=ikks(ik)
|
|
ikq=ikqs(ik)
|
|
xk_dummy(:)=xk(:,ikk)
|
|
call cryst_to_cart(1,xk_dummy,at,-1)
|
|
WRITE (iuelphmat) (xk_dummy(ipert),ipert=1,3)
|
|
WRITE (iuelphmat) (et(ibnd,ikk),ibnd=elph_nbnd_min,elph_nbnd_max)
|
|
|
|
do nu=1,nmodes
|
|
WRITE (iuelphmat) &
|
|
((el_ph_mat (jbnd, ibnd, ik, nu),jbnd=elph_nbnd_min,elph_nbnd_max),&
|
|
ibnd=elph_nbnd_min,elph_nbnd_max)
|
|
enddo
|
|
enddo
|
|
|
|
|
|
|
|
|
|
!
|
|
! Then I dump symmetry operations
|
|
!
|
|
minus_qloc = .true.
|
|
sym = .false.
|
|
sym(1:nsym) = .true.
|
|
|
|
call smallg_q (xq, 0, at, bg, nsym, s, ftau, sym, minus_qloc)
|
|
nsymq = copy_sym(nsym, sym)
|
|
! recompute the inverses as the order of sym.ops. has changed
|
|
CALL inverse_s ( )
|
|
|
|
! part 2: this redoes most of the above, plus it computes irgq, gi, gimq
|
|
CALL smallgq (xq, at, bg, s, nsym, irgq, nsymq, irotmq, &
|
|
minus_qloc, gi, gimq)
|
|
|
|
sym(1:nsym)=.true.
|
|
call sgam_ph (at, bg, nsym, s, irt, tau, rtauloc, nat, sym)
|
|
call star_q(xq, at, bg, nsym , s , invs , nq, sxq, &
|
|
isq, imq, .FALSE. )
|
|
|
|
|
|
do j=1,3
|
|
write(iuelphmat) (at(i,j),i=1,3)
|
|
enddo
|
|
do j=1,3
|
|
write(iuelphmat) (bg(i,j),i=1,3)
|
|
enddo
|
|
write(iuelphmat) nsym,nq,imq
|
|
do i=1,nsym
|
|
write(iuelphmat) i,invs(i),isq(i)
|
|
do j=1,3
|
|
do k=1,3
|
|
write(iuelphmat) k,j, s(k,j,i)
|
|
enddo
|
|
enddo
|
|
do j=1,nat
|
|
write(iuelphmat) j, irt(i,j)
|
|
enddo
|
|
do j=1,3
|
|
do k=1,nat
|
|
write(iuelphmat) j,i, rtauloc(j,i,k)
|
|
enddo
|
|
enddo
|
|
do j=1,3
|
|
write(iuelphmat) j, sxq(j,i)
|
|
enddo
|
|
enddo
|
|
|
|
close(iuelphmat)
|
|
endif
|
|
|
|
!
|
|
!
|
|
|
|
|
|
RETURN
|
|
|
|
|
|
END SUBROUTINE ELPHSUM_WANNIER
|
|
|
|
!
|
|
!-----------------------------------------------------------------------
|
|
SUBROUTINE elphel_refolded (npe, imode0, dvscfins)
|
|
!-----------------------------------------------------------------------
|
|
!
|
|
! Calculation of the electron-phonon matrix elements el_ph_mat
|
|
! <\psi(k+q)|dV_{SCF}/du^q_{i a}|\psi(k)>
|
|
! Original routine written by Francesco Mauri
|
|
!
|
|
USE kinds, ONLY : DP
|
|
USE fft_base, ONLY : dffts
|
|
USE wavefunctions, ONLY: evc
|
|
USE io_files, ONLY: prefix, diropn
|
|
USE klist, ONLY: xk, ngk, igk_k
|
|
USE lsda_mod, ONLY: lsda, current_spin, isk
|
|
USE noncollin_module, ONLY : noncolin, npol, nspin_mag
|
|
USE buffers, ONLY : get_buffer
|
|
USE wvfct, ONLY: nbnd, npwx
|
|
USE uspp, ONLY : vkb
|
|
USE el_phon, ONLY : el_ph_mat, iunwfcwann, igqg, kpq, g_kpq, &
|
|
xk_gamma, npwq_refolded, lrwfcr
|
|
USE modes, ONLY : u
|
|
USE units_ph, ONLY : iubar, lrbar
|
|
USE control_ph, ONLY : trans
|
|
USE mp_bands, ONLY: intra_bgrp_comm
|
|
USE mp_pools, ONLY: me_pool, root_pool
|
|
USE mp, ONLY: mp_sum
|
|
USE ions_base, ONLY : nat
|
|
USE io_global, ONLY : stdout
|
|
|
|
USE eqv, ONLY : dvpsi!, evq
|
|
USE qpoint, ONLY : nksq, ikks, ikqs
|
|
USE control_lr, ONLY : lgamma
|
|
|
|
IMPLICIT NONE
|
|
!
|
|
INTEGER :: npe, imode0
|
|
COMPLEX(DP) :: dvscfins (dffts%nnr, nspin_mag, npe)
|
|
COMPLEX(DP), allocatable :: evq(:,:)
|
|
|
|
|
|
! LOCAL variables
|
|
logical :: exst
|
|
INTEGER :: npw, npwq
|
|
INTEGER :: nrec, ik, ikk, ikq, ikqg,ipert, mode, ibnd, jbnd, ir, ig, &
|
|
ios
|
|
|
|
COMPLEX(DP) , ALLOCATABLE :: aux1 (:,:), elphmat (:,:,:)
|
|
COMPLEX(DP), EXTERNAL :: zdotc
|
|
INTEGER, EXTERNAL :: find_free_unit
|
|
!
|
|
allocate (evq(npol*npwx,nbnd))
|
|
ALLOCATE (aux1 (dffts%nnr, npol))
|
|
ALLOCATE (elphmat ( nbnd , nbnd , 3*nat))
|
|
|
|
|
|
! iunwfcwann=find_free_unit()
|
|
! CALL diropn (iunwfcwann, 'wfc', lrwfc, exst, dvscf_dir)
|
|
! IF (.NOT.exst) THEN
|
|
! CALL errore ('elphel_refolded', 'file '//trim(prefix)//'.wfc not found in Rotated_DVSCF', 1)
|
|
! END IF
|
|
!
|
|
! Start the loops over the k-points
|
|
!
|
|
|
|
DO ik = 1, nksq
|
|
!
|
|
! ik = counter of k-points with vector k
|
|
! ikk= index of k-point with vector k
|
|
! ikq= index of k-point with vector k+q
|
|
! k and k+q are alternated if q!=0, are the same if q=0
|
|
!
|
|
ikk = ikks(ik)
|
|
ikq = ikqs(ik)
|
|
ikqg = kpq(ik)
|
|
npw = ngk(ikk)
|
|
npwq= ngk(ikq)
|
|
IF (lsda) current_spin = isk (ikk)
|
|
!
|
|
CALL init_us_2 (npwq, igk_k(1,ikq), xk (1, ikq), vkb)
|
|
!
|
|
! read unperturbed wavefuctions psi(k) and psi(k+q)
|
|
!
|
|
evc=(0.d0,0.d0)
|
|
|
|
! Warning error in reading wfc, this could explain.
|
|
! We read here the wfc at the Gamma point, that is
|
|
! that saved by Wannier.
|
|
|
|
! CALL davcio (evc, lrwfc, iunwfcwann, ik, - 1)
|
|
! CALL davcio (evq, lrwfc, iunwfcwann, ikqg, - 1)
|
|
|
|
|
|
|
|
! IF (nksq.GT.1) THEN
|
|
! IF (lgamma) THEN
|
|
! CALL davcio (evc, lrwfc, iunwfcwann, ikk, - 1)
|
|
! ELSE
|
|
! CALL davcio (evc, lrwfc, iunwfcwann, ik, - 1)
|
|
! CALL davcio (evq, lrwfc, iunwfcwann, ikqg, - 1)
|
|
! ENDIF
|
|
! ENDIF
|
|
!
|
|
|
|
call read_wfc_rspace_and_fwfft( evc , ik , lrwfcr , iunwfcwann , npw , igk_k(1,ikk) )
|
|
|
|
|
|
call calculate_and_apply_phase(ik, ikqg, igqg, npwq_refolded, g_kpq,xk_gamma, evq, .true.)
|
|
|
|
|
|
DO ipert = 1, npe
|
|
nrec = (ipert - 1) * nksq + ik
|
|
!
|
|
! dvbare_q*psi_kpoint is read from file (if available) or recalculated
|
|
!
|
|
IF (trans) THEN
|
|
CALL get_buffer (dvpsi, lrbar, iubar, nrec)
|
|
ELSE
|
|
mode = imode0 + ipert
|
|
! FIXME : .false. or .true. ???
|
|
CALL dvqpsi_us (ik, u (1, mode), .FALSE. )
|
|
ENDIF
|
|
!
|
|
! calculate dvscf_q*psi_k
|
|
!
|
|
|
|
DO ibnd = 1, nbnd
|
|
CALL cft_wave (ik, evc(1, ibnd), aux1, +1)
|
|
CALL apply_dpot(dffts%nnr, aux1, dvscfins(1,1,ipert), current_spin)
|
|
CALL cft_wave (ik, dvpsi(1, ibnd), aux1, -1)
|
|
END DO
|
|
CALL adddvscf (ipert, ik)
|
|
!
|
|
! calculate elphmat(j,i)=<psi_{k+q,j}|dvscf_q*psi_{k,i}> for this pertur
|
|
!
|
|
DO ibnd =1, nbnd
|
|
DO jbnd = 1, nbnd
|
|
elphmat (jbnd, ibnd, ipert) = zdotc (npwq_refolded, evq (1, jbnd), 1, &
|
|
dvpsi (1, ibnd), 1)
|
|
IF (noncolin) &
|
|
elphmat (jbnd, ibnd, ipert) = elphmat (jbnd, ibnd, ipert)+ &
|
|
zdotc (npwq_refolded, evq(npwx+1,jbnd),1,dvpsi(npwx+1,ibnd), 1)
|
|
ENDDO
|
|
ENDDO
|
|
ENDDO
|
|
!
|
|
CALL mp_sum (elphmat, intra_bgrp_comm)
|
|
!
|
|
! save all e-ph matrix elements into el_ph_mat
|
|
!
|
|
DO ipert = 1, npe
|
|
DO jbnd = 1, nbnd
|
|
DO ibnd = 1, nbnd
|
|
el_ph_mat (ibnd, jbnd, ik, ipert + imode0) = elphmat (ibnd, jbnd, ipert)
|
|
ENDDO
|
|
ENDDO
|
|
ENDDO
|
|
ENDDO
|
|
|
|
|
|
! CLOSE( UNIT = iunwfcwann, STATUS = 'KEEP' )
|
|
!
|
|
DEALLOCATE (elphmat)
|
|
DEALLOCATE (aux1)
|
|
DEALLOCATE(evq)
|
|
!
|
|
RETURN
|
|
END SUBROUTINE elphel_refolded
|
|
!
|
|
subroutine get_equivalent_kpq(xk,xq,kpq,g_kpq, igqg)
|
|
!==================================================================!
|
|
! !
|
|
! Set up the k+q shell for electron-phonon coupling !
|
|
! !
|
|
! This routine finds the G vectors such that !
|
|
! k+q+G=k' with k and k' belonging to nksq !
|
|
! for each k, the G vector is stored in g_kpq !
|
|
! k'=kpq(ik) !
|
|
! and finally igqg(ik) is the index that allows to find !
|
|
! the g vector g_kpq in the list of all the G vectors !
|
|
! !
|
|
! Matteo Calandra !
|
|
!===================================================================
|
|
USE kinds, ONLY : DP
|
|
USE io_global, ONLY : stdout
|
|
USE cell_base, ONLY : at, bg
|
|
USE qpoint, ONLY : nksq, ikks
|
|
USE gvect, ONLY: g, gg
|
|
USE qpoint, ONLY : nksq
|
|
USE mp_bands, ONLY : intra_bgrp_comm
|
|
USE mp, ONLY : mp_sum
|
|
! WARNING g_kpq mesh is an integer
|
|
implicit none
|
|
|
|
! Variables that are private
|
|
|
|
integer :: iqx,iqy,iqz,i,j,k,n,nn,iq,ik, ig
|
|
integer :: kpq(nksq),g_kpq(3,nksq),igqg(nksq)
|
|
integer, allocatable :: ig_check(:)
|
|
real(kind=dp) :: gg_
|
|
real(kind=dp) :: xq(3), xk(3,*)
|
|
real(kind=dp) :: xkpq(3),Gvec(3),xq_crys(3)
|
|
real(kind=dp), allocatable :: xk_crys(:,:)
|
|
|
|
!
|
|
! nksq = number of k point per pool withour k+q
|
|
!
|
|
|
|
!
|
|
! The xk_point entering here must be the k and not
|
|
! the k+q
|
|
!
|
|
|
|
xq_crys=xq
|
|
|
|
call cryst_to_cart (1, xq_crys, at, -1)
|
|
|
|
|
|
allocate(xk_crys(3,nksq))
|
|
|
|
do ik=1,nksq
|
|
xk_crys(1:3,ik)=xk(1:3,ik)
|
|
enddo
|
|
call cryst_to_cart (nksq, xk_crys, at, -1)
|
|
|
|
!
|
|
! kpt_latt are the BZ vectors in crystalline coordinates
|
|
! xq is the q vector in crystalline coordinates
|
|
!
|
|
|
|
do iq=1,nksq
|
|
xkpq(:)=xk_crys(:,iq)+xq_crys(:)
|
|
do i=1,nksq
|
|
do iqx=-4,4
|
|
do iqy=-4,4
|
|
do iqz=-4,4
|
|
Gvec(1)=real(iqx,dp)+xkpq(1)
|
|
Gvec(2)=real(iqy,dp)+xkpq(2)
|
|
Gvec(3)=real(iqz,dp)+xkpq(3)
|
|
|
|
if(dabs(xk_crys(1,i)-Gvec(1)).lt.1.d-6.and. &
|
|
dabs(xk_crys(2,i)-Gvec(2)).lt.1.d-6.and. &
|
|
dabs(xk_crys(3,i)-Gvec(3)).lt.1.d-6) then
|
|
kpq(iq)=i
|
|
g_kpq(1,iq)=-iqx
|
|
g_kpq(2,iq)=-iqy
|
|
g_kpq(3,iq)=-iqz
|
|
goto 99
|
|
endif
|
|
enddo
|
|
enddo
|
|
enddo
|
|
enddo
|
|
CALL errore ('get_equivalent_kpq', 'cannot find index k+q ', 2 )
|
|
stop
|
|
99 continue
|
|
enddo
|
|
|
|
!
|
|
! here between all the g-vectors I find the index of that
|
|
! related to the translation in the Brillouin zone.
|
|
! Warning: if G does not belong to the processor igqg is zero.
|
|
!
|
|
|
|
igqg=0
|
|
do ik=1,nksq
|
|
Gvec(:) = REAL( g_kpq(:,ik),dp )
|
|
call cryst_to_cart (1, Gvec, bg, 1)
|
|
gg_ = Gvec(1)*Gvec(1) + Gvec(2)*Gvec(2) + Gvec(3)*Gvec(3)
|
|
igqg(ik)=0
|
|
ig=1
|
|
do while (gg(ig) <= gg_ + 1.d-6)
|
|
if ( (abs(g(1,ig)-Gvec(1)) < 1.d-6) .and. &
|
|
(abs(g(2,ig)-Gvec(2)) < 1.d-6) .and. &
|
|
(abs(g(3,ig)-Gvec(3)) < 1.d-6) ) then
|
|
igqg(ik) = ig
|
|
|
|
endif
|
|
ig= ig +1
|
|
end do
|
|
end do
|
|
|
|
allocate(ig_check(nksq))
|
|
ig_check=igqg
|
|
CALL mp_sum( ig_check, intra_bgrp_comm )
|
|
do ik=1,nksq
|
|
if(ig_check(ik).eq.0) &
|
|
CALL errore('get_equivalent_kpq', &
|
|
' g_kpq vector is not in the list of Gs', 100*ik )
|
|
|
|
enddo
|
|
deallocate(xk_crys)
|
|
|
|
|
|
end subroutine get_equivalent_kpq
|
|
|
|
subroutine calculate_and_apply_phase(ik, ikqg, igqg, npwq_refolded, g_kpq, xk_gamma, evq, lread)
|
|
USE kinds, ONLY : DP
|
|
USE fft_base, ONLY : dffts
|
|
USE fft_interfaces, ONLY : fwfft, invfft
|
|
USE wvfct, ONLY: nbnd, npwx
|
|
USE gvect, ONLY : ngm, g
|
|
USE gvecw, ONLY : gcutw
|
|
USE cell_base, ONLY : bg
|
|
USE qpoint, ONLY : nksq
|
|
USE wavefunctions, ONLY : evc
|
|
USE noncollin_module, ONLY : npol, noncolin
|
|
USE el_phon, ONLY:iunwfcwann, lrwfcr
|
|
|
|
IMPLICIT NONE
|
|
|
|
LOGICAL :: lread
|
|
INTEGER :: ik, ikqg, npwq_refolded
|
|
INTEGER :: igqg(nksq)
|
|
INTEGER :: g_kpq(3,nksq)
|
|
REAL (DP) :: xk_gamma(3,nksq)
|
|
complex(dp) :: evq(npwx*npol,nbnd)
|
|
! internal
|
|
INTEGER :: npw_, m,i
|
|
INTEGER, allocatable :: igk_(:), igkq_(:)
|
|
REAL(DP) :: xkqg(3), g_(3), g_scra(3,ngm)
|
|
REAL(DP), ALLOCATABLE :: gk(:)
|
|
COMPLEX (DP), allocatable :: psi_scratch(:)
|
|
complex(DP), allocatable :: phase(:)
|
|
|
|
allocate(igk_(npwx), igkq_(npwx), gk(npwx) )
|
|
allocate (psi_scratch ( dffts%nnr) )
|
|
allocate (phase(dffts%nnr))
|
|
FLUSH (6)
|
|
|
|
g_scra=g
|
|
|
|
g_(:)=real( g_kpq(:,ik), dp )
|
|
call cryst_to_cart (1, g_, bg, 1)
|
|
xkqg(:)=xk_gamma(:,ikqg)+g_(:)
|
|
|
|
npw_=0
|
|
npwq_refolded=0
|
|
igk_=0
|
|
igkq_=0
|
|
|
|
|
|
call gk_sort (xk_gamma(1,ikqg), ngm, g_scra, gcutw, npw_, igk_, gk)
|
|
|
|
if(lread) then
|
|
call read_wfc_rspace_and_fwfft( evq , ikqg , lrwfcr , iunwfcwann , npw_ , igk_ )
|
|
endif
|
|
|
|
call gk_sort (xkqg, ngm, g_scra, gcutw, npwq_refolded, igkq_, gk)
|
|
|
|
phase(:) = (0.d0,0.d0)
|
|
|
|
if ( igqg(ik)>0) then
|
|
phase( dffts%nl(igqg(ik)) ) = (1.d0,0.d0)
|
|
endif
|
|
|
|
|
|
CALL invfft ('Wave', phase, dffts)
|
|
! call cft3s (phase, nr1s, nr2s, nr3s, nrx1s, nrx2s, nrx3s, +2)
|
|
phase(:)=conjg(phase(:))
|
|
|
|
|
|
if(npwq_refolded.ne.npw_) call errore('calculate_and_apply_phase', 'Warning : npwq_refolded \= npw_',-1)
|
|
|
|
do m=1,nbnd
|
|
psi_scratch = (0.d0, 0.d0)
|
|
psi_scratch(dffts%nl (igk_ (1:npw_) ) ) = evq (1:npw_, m)
|
|
! psi_scratch(dffts%nl (igk_ (1:npw) ) ) = evq (1:npw, m)
|
|
CALL invfft ('Wave', psi_scratch, dffts)
|
|
! call cft3s (psic, nr1s, nr2s, nr3s, nrx1s, nrx2s, nrx3s, +2)
|
|
psi_scratch(1:dffts%nnr) = psi_scratch(1:dffts%nnr) * phase(1:dffts%nnr)
|
|
! call cft3s (psic, nr1s, nr2s, nr3s, nrx1s, nrx2s, nrx3s, -2)
|
|
CALL fwfft ('Wave', psi_scratch, dffts)
|
|
evq(1:npwq_refolded,m) = psi_scratch(dffts%nl (igkq_(1:npwq_refolded) ) )
|
|
enddo
|
|
|
|
if(noncolin) then
|
|
do m=1,nbnd
|
|
psi_scratch = (0.d0, 0.d0)
|
|
psi_scratch(dffts%nl (igk_ (1:npw_) ) ) = evq (npwx+1:npwx+npw_, m)
|
|
! psi_scratch(dffts%nl (igk_ (1:npw) ) ) = evq (1:npw, m)
|
|
CALL invfft ('Wave', psi_scratch, dffts)
|
|
! call cft3s (psic, nr1s, nr2s, nr3s, nrx1s, nrx2s, nrx3s, +2)
|
|
psi_scratch(1:dffts%nnr) = psi_scratch(1:dffts%nnr) * phase(1:dffts%nnr)
|
|
! call cft3s (psic, nr1s, nr2s, nr3s, nrx1s, nrx2s, nrx3s, -2)
|
|
CALL fwfft ('Wave', psi_scratch, dffts)
|
|
! evq(npwx+1:npwx+npwq_refolded,m) = psi_scratch(dffts%nl (igkq_(1:npwq_refolded) ) )
|
|
evq((npwx+1):(npwx+npwq_refolded),m) = psi_scratch(dffts%nl (igkq_(1:npwq_refolded) ) )
|
|
enddo
|
|
endif
|
|
|
|
deallocate(psi_scratch)
|
|
DEALLOCATE(phase)
|
|
deallocate(gk, igk_, igkq_)
|
|
|
|
return
|
|
end subroutine calculate_and_apply_phase
|
|
|
|
|
|
!-----------------------------------------------------------------------
|
|
SUBROUTINE readmat_findq (iudyn, ibrav, celldm, nat, ntyp, ityp, omega, &
|
|
amass, tau, q, w2, dyn)
|
|
!-----------------------------------------------------------------------
|
|
!
|
|
USE kinds, ONLY : DP
|
|
USE constants, ONLY : amu_ry
|
|
USE control_ph, ONLY : xmldyn
|
|
USE output, ONLY : fildyn
|
|
USE io_dyn_mat, ONLY : read_dyn_mat_param, read_dyn_mat_header, &
|
|
read_dyn_mat, read_dyn_mat_tail
|
|
IMPLICIT NONE
|
|
! Input
|
|
INTEGER :: iudyn, ibrav, nat, ntyp, ityp (nat)
|
|
REAL(DP) :: celldm (6), amass (ntyp), tau (3, nat), q (3), &
|
|
omega
|
|
! output
|
|
REAL(DP) :: w2 (3 * nat)
|
|
COMPLEX(DP) :: dyn (3 * nat, 3 * nat)
|
|
! local (control variables)
|
|
INTEGER :: ntyp_, nat_, ibrav_, ityp_
|
|
REAL(DP) :: celldm_ (6), amass_, tau_ (3), q_ (3)
|
|
! local
|
|
INTEGER :: nspin_mag, nqs
|
|
REAL(DP) :: at(3,3)
|
|
REAL(DP) :: bg(3,3)
|
|
REAL(DP) :: m_loc(3,nat)
|
|
INTEGER :: ityp__ (nat)
|
|
REAL(DP) :: amass__ (ntyp)
|
|
INTEGER :: iq
|
|
REAL(DP) :: xq(3)
|
|
COMPLEX(DP) :: u(3*nat,3*nat)
|
|
REAL(DP) :: dynr (2, 3, nat, 3, nat), err_q(3)
|
|
COMPLEX(DP) :: dynr_c(3,3,nat,nat)
|
|
|
|
CHARACTER(len=80) :: line
|
|
CHARACTER(len=3) :: atm
|
|
INTEGER :: nt, na, nb, naa, nbb, nu, mu, i, j
|
|
LOGICAL :: lfound
|
|
|
|
!
|
|
!
|
|
IF(xmldyn) THEN
|
|
CALL read_dyn_mat_param(fildyn, ntyp_, nat_ )
|
|
CALL read_dyn_mat_header(ntyp_, nat_, ibrav_, nspin_mag, &
|
|
celldm_, at, bg, omega, atm, amass__, tau_, ityp__, m_loc, &
|
|
nqs )
|
|
IF ( ntyp.NE.ntyp_ .OR. nat.NE.nat_ .OR.ibrav_.NE.ibrav .OR. &
|
|
ABS ( celldm_ (1) - celldm (1) ) > 1.0d-5) &
|
|
CALL errore ('readmat', 'inconsistent data a', 1)
|
|
DO nt = 1, ntyp
|
|
IF ( ABS (amass__ (nt) - amass (nt) ) > 1.0d-5) &
|
|
CALL errore ( 'readmat', 'inconsistent data b', 1 + nt)
|
|
ENDDO
|
|
DO na = 1, nat
|
|
IF (ityp__ (na).NE.ityp (na) ) CALL errore ('readmat', &
|
|
'inconsistent data c', na)
|
|
ENDDO
|
|
|
|
ELSE
|
|
REWIND (iudyn)
|
|
READ (iudyn, '(a)') line
|
|
READ (iudyn, '(a)') line
|
|
READ (iudyn, * ) ntyp_, nat_, ibrav_, celldm_
|
|
IF ( ntyp.NE.ntyp_ .OR. nat.NE.nat_ .OR.ibrav_.NE.ibrav .OR. &
|
|
ABS ( celldm_ (1) - celldm (1) ) > 1.0d-5) &
|
|
CALL errore ('readmat', 'inconsistent data', 1)
|
|
DO nt = 1, ntyp
|
|
READ (iudyn, * ) i, atm, amass_
|
|
IF ( nt.NE.i .OR. ABS (amass_ - amu_ry*amass (nt) ) > 1.0d-5) &
|
|
CALL errore ( 'readmat', 'inconsistent data', 1 + nt)
|
|
ENDDO
|
|
DO na = 1, nat
|
|
READ (iudyn, * ) i, ityp_, tau_
|
|
IF (na.NE.i.OR.ityp_.NE.ityp (na) ) CALL errore ('readmat', &
|
|
'inconsistent data', 10 + na)
|
|
ENDDO
|
|
|
|
ENDIF
|
|
|
|
lfound=.false.
|
|
iq=0
|
|
|
|
do while(.not.lfound)
|
|
|
|
IF(xmldyn) THEN
|
|
|
|
iq = iq+1
|
|
CALL read_dyn_mat(nat,iq,xq,dynr_c)
|
|
! CALL read_dyn_mat_tail(nat,omega,u)
|
|
err_q(1:3)=dabs(xq(1:3)-q(1:3))
|
|
|
|
if(err_q(1).lt.1.d-7.and.err_q(2).lt.1.d-7.and.err_q(3).lt.1.d-7) lfound=.true.
|
|
|
|
DO nb = 1, nat
|
|
DO j = 1, 3
|
|
DO na = 1, nat
|
|
DO i = 1, 3
|
|
dynr (1, i, na, j, nb) = REAL(dynr_c(i, j, na, nb))
|
|
dynr (2, i, na, j, nb) = AIMAG(dynr_c(i, j, na, nb))
|
|
ENDDO
|
|
ENDDO
|
|
ENDDO
|
|
ENDDO
|
|
|
|
ELSE
|
|
READ (iudyn, '(a)') line
|
|
READ (iudyn, '(a)') line
|
|
READ (iudyn, '(a)') line
|
|
READ (iudyn, '(a)') line
|
|
|
|
READ (line (11:80), * ) (q_ (i), i = 1, 3)
|
|
|
|
err_q(1:3)=dabs(q_(1:3)-q(1:3))
|
|
|
|
if(err_q(1).lt.1.d-7.and.err_q(2).lt.1.d-7.and.err_q(3).lt.1.d-7) lfound=.true.
|
|
|
|
|
|
|
|
READ (iudyn, '(a)') line
|
|
DO na = 1, nat
|
|
DO nb = 1, nat
|
|
READ (iudyn, * ) naa, nbb
|
|
IF (na.NE.naa.OR.nb.NE.nbb) CALL errore ('readmat', 'error reading &
|
|
&file', nb)
|
|
READ (iudyn, * ) ( (dynr (1, i, na, j, nb), dynr (2, i, na, j, nb) &
|
|
, j = 1, 3), i = 1, 3)
|
|
ENDDO
|
|
ENDDO
|
|
|
|
ENDIF
|
|
|
|
if(lfound) then
|
|
!
|
|
! divide the dynamical matrix by the (input) masses (in amu)
|
|
!
|
|
DO nb = 1, nat
|
|
DO j = 1, 3
|
|
DO na = 1, nat
|
|
DO i = 1, 3
|
|
dynr (1, i, na, j, nb) = dynr (1, i, na, j, nb) / SQRT (amass ( &
|
|
ityp (na) ) * amass (ityp (nb) ) ) / amu_ry
|
|
dynr (2, i, na, j, nb) = dynr (2, i, na, j, nb) / SQRT (amass ( &
|
|
ityp (na) ) * amass (ityp (nb) ) ) / amu_ry
|
|
ENDDO
|
|
ENDDO
|
|
ENDDO
|
|
ENDDO
|
|
!
|
|
! solve the eigenvalue problem.
|
|
! NOTA BENE: eigenvectors are overwritten on dyn
|
|
!
|
|
CALL cdiagh (3 * nat, dynr, 3 * nat, w2, dyn)
|
|
!
|
|
! divide by sqrt(mass) to get displacements
|
|
!
|
|
DO nu = 1, 3 * nat
|
|
DO mu = 1, 3 * nat
|
|
na = (mu - 1) / 3 + 1
|
|
dyn (mu, nu) = dyn (mu, nu) / SQRT ( amu_ry * amass (ityp (na) ) )
|
|
ENDDO
|
|
ENDDO
|
|
!
|
|
!
|
|
endif
|
|
enddo
|
|
|
|
|
|
RETURN
|
|
END SUBROUTINE readmat_findq
|
|
|