quantum-espresso/Modules/xsf.f90

214 lines
6.3 KiB
Fortran

!
! Copyright (C) 2003 Tone Kokalj
! 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 .
!
! This file holds XSF (=Xcrysden Structure File) utilities.
! Routines written by Tone Kokalj on Mon Jan 27 18:51:17 CET 2003
!
!-------------------------------------------------------------------
SUBROUTINE xsf_struct (alat, at, nat, tau, atm, ityp, ounit)
!-------------------------------------------------------------------
!! This routine writes the crystal structure in XSF format.
!
USE kinds, ONLY : DP
USE constants, ONLY : BOHR_RADIUS_ANGS
IMPLICIT NONE
INTEGER :: nat, ityp (nat), ounit
CHARACTER(len=3) :: atm(*)
real(DP) :: alat, tau (3, nat), at (3, 3)
! --
INTEGER :: i, j, n
real(DP) :: at1 (3, 3)
! convert lattice vectors to ANGSTROM units ...
DO i=1,3
DO j=1,3
at1(j,i) = at(j,i)*alat*BOHR_RADIUS_ANGS
ENDDO
ENDDO
WRITE(ounit,*) 'CRYSTAL'
WRITE(ounit,*) 'PRIMVEC'
WRITE(ounit,'(2(3F15.9/),3f15.9)') at1
WRITE(ounit,*) 'PRIMCOORD'
WRITE(ounit,*) nat, 1
DO n=1,nat
! positions are in Angstroms
WRITE(ounit,'(a3,3x,3f15.9)') atm(ityp(n)), &
tau(1,n)*alat*BOHR_RADIUS_ANGS, &
tau(2,n)*alat*BOHR_RADIUS_ANGS, &
tau(3,n)*alat*BOHR_RADIUS_ANGS
ENDDO
RETURN
END SUBROUTINE xsf_struct
!-------------------------------------------------------------------
SUBROUTINE xsf_fast_datagrid_3d &
(rho, nr1, nr2, nr3, nr1x, nr2x, nr3x, at, alat, ounit)
!-------------------------------------------------------------------
!! This routine writes the 3D scalar field (i.e. uniform mesh of points)
!! in XSF format using the FFT mesh (i.e. fast write).
!
USE kinds, ONLY : DP
USE constants, ONLY : BOHR_RADIUS_ANGS
IMPLICIT NONE
INTEGER :: nr1x, nr2x, nr3x, nr1, nr2, nr3, ounit
real(DP) :: alat, at (3, 3), rho(nr1x,nr2x,nr3x)
! --
INTEGER :: i1, i2, i3, ix, iy, iz, count, i, &
ind_x(10), ind_y(10),ind_z(10)
! XSF scalar-field header
WRITE(ounit,'(a)') 'BEGIN_BLOCK_DATAGRID_3D'
WRITE(ounit,'(a)') '3D_PWSCF'
WRITE(ounit,'(a)') 'BEGIN_DATAGRID_3D_UNKNOWN'
! number of points in each direction
WRITE(ounit,*) nr1+1, nr2+1, nr3+1
! origin
WRITE(ounit,'(3f10.6)') 0.0d0, 0.0d0, 0.0d0
! 1st spanning (=lattice) vector
WRITE(ounit,'(3f12.6)') (BOHR_RADIUS_ANGS*alat*at(i,1),i=1,3) ! in ANGSTROMS
! 2nd spanning (=lattice) vector
WRITE(ounit,'(3f12.6)') (BOHR_RADIUS_ANGS*alat*at(i,2),i=1,3)
! 3rd spanning (=lattice) vector
WRITE(ounit,'(3f12.6)') (BOHR_RADIUS_ANGS*alat*at(i,3),i=1,3)
count=0
DO i3=0,nr3
!iz = mod(i3,nr3)
iz = mod(i3,nr3) + 1
DO i2=0,nr2
!iy = mod(i2,nr2)
iy = mod(i2,nr2) + 1
DO i1=0,nr1
!ix = mod(i1,nr1)
ix = mod(i1,nr1) + 1
!ii = (1+ix) + iy*nr1x + iz*nr1x*nr2x
IF (count<6) THEN
count = count + 1
!ind(count) = ii
ELSE
WRITE(ounit,'(6e14.6)') &
(rho(ind_x(i),ind_y(i),ind_z(i)),i=1,6)
count=1
!ind(count) = ii
ENDIF
ind_x(count) = ix
ind_y(count) = iy
ind_z(count) = iz
ENDDO
ENDDO
ENDDO
WRITE(ounit,'(6e14.6:)') (rho(ind_x(i),ind_y(i),ind_z(i)),i=1,count)
WRITE(ounit,'(a)') 'END_DATAGRID_3D'
WRITE(ounit,'(a)') 'END_BLOCK_DATAGRID_3D'
RETURN
END SUBROUTINE xsf_fast_datagrid_3d
SUBROUTINE xsf_datagrid_2d (rho, nx, ny, m1, m2, x0, e1, e2, alat, ounit)
USE kinds, ONLY : DP
USE constants, ONLY : BOHR_RADIUS_ANGS
IMPLICIT NONE
INTEGER :: nx, ny, ounit
real(DP) :: m1, m2, alat, x0(3), e1(3), e2(3), rho(2, nx, ny)
! --
INTEGER :: ix, iy, count, i, ind_x(10), ind_y(10)
! XSF scalar-field header
WRITE(ounit,'(a)') 'BEGIN_BLOCK_DATAGRID_2D'
WRITE(ounit,'(a)') '2D_PWSCF'
WRITE(ounit,'(a)') 'DATAGRID_2D_UNKNOWN'
! number of points in each direction
WRITE(ounit,*) nx, ny
! origin
WRITE(ounit,'(3f10.6)') (BOHR_RADIUS_ANGS*alat*x0(i),i=1,3) ! in ANGSTROMS
! 1st spanning (=lattice) vector
WRITE(ounit,'(3f10.6)') (BOHR_RADIUS_ANGS*alat*e1(i)*m1,i=1,3) ! in ANGSTROMS
! 2nd spanning (=lattice) vector
WRITE(ounit,'(3f10.6)') (BOHR_RADIUS_ANGS*alat*e2(i)*m2,i=1,3) ! in ANGSTROMS
count=0
DO iy=1,ny
DO ix=1,nx
IF (count < 6) THEN
count = count + 1
ELSE
WRITE(ounit,'(6e14.6)') (rho(1,ind_x(i),ind_y(i)),i=1,6)
count=1
ENDIF
ind_x(count) = ix
ind_y(count) = iy
ENDDO
ENDDO
WRITE(ounit,'(6e14.6:)') (rho(1,ind_x(i),ind_y(i)),i=1,count)
WRITE(ounit,'(a)') 'END_DATAGRID_2D'
WRITE(ounit,'(a)') 'END_BLOCK_DATAGRID_2D'
RETURN
END SUBROUTINE xsf_datagrid_2d
SUBROUTINE xsf_datagrid_3d &
(rho, nx, ny, nz, m1, m2, m3, x0, e1, e2, e3, alat, ounit)
USE kinds, ONLY : DP
USE constants, ONLY : BOHR_RADIUS_ANGS
IMPLICIT NONE
INTEGER :: nx, ny, nz, ounit
real(DP) :: m1, m2, m3, alat, x0(3), e1(3),e2(3),e3(3), rho(nx, ny, nz)
! --
INTEGER :: ix, iy, iz, count, i, ind_x(10), ind_y(10), ind_z(10)
! XSF scalar-field header
WRITE(ounit,'(a)') 'BEGIN_BLOCK_DATAGRID_3D'
WRITE(ounit,'(a)') '3D_PWSCF'
WRITE(ounit,'(a)') 'DATAGRID_3D_UNKNOWN'
! number of points in each direction
WRITE(ounit,*) nx, ny, nz
! origin
WRITE(ounit,'(3f10.6)') (BOHR_RADIUS_ANGS*alat*x0(i),i=1,3) ! in ANGSTROMS
! 1st spanning (=lattice) vector
WRITE(ounit,'(3f10.6)') (BOHR_RADIUS_ANGS*alat*e1(i)*m1,i=1,3) ! in ANGSTROMS
! 2nd spanning (=lattice) vector
WRITE(ounit,'(3f10.6)') (BOHR_RADIUS_ANGS*alat*e2(i)*m2,i=1,3) ! in ANGSTROMS
! 3rd spanning (=lattice) vector
WRITE(ounit,'(3f10.6)') (BOHR_RADIUS_ANGS*alat*e3(i)*m3,i=1,3)
count=0
DO iz=1,nz
DO iy=1,ny
DO ix=1,nx
IF (count<6) THEN
count = count + 1
ELSE
WRITE(ounit,'(6e14.6)') (rho(ind_x(i),ind_y(i),ind_z(i)),i=1,6)
count=1
ENDIF
ind_x(count) = ix
ind_y(count) = iy
ind_z(count) = iz
ENDDO
ENDDO
ENDDO
WRITE(ounit,'(6e14.6:)') (rho(ind_x(i),ind_y(i),ind_z(i)),i=1,count)
WRITE(ounit,'(a)') 'END_DATAGRID_3D'
WRITE(ounit,'(a)') 'END_BLOCK_DATAGRID_3D'
RETURN
END SUBROUTINE xsf_datagrid_3d