2019-10-28 16:34:09 +01:00
|
|
|
subroutine diagonalize_general_matrix(N,A,WR,VR)
|
2019-10-15 17:56:27 +02:00
|
|
|
|
|
|
|
! Diagonalize a non-symmetric square matrix
|
|
|
|
|
|
|
|
implicit none
|
|
|
|
|
|
|
|
! Input variables
|
|
|
|
|
2019-10-28 16:34:09 +01:00
|
|
|
integer :: i,j,k
|
2019-10-15 17:56:27 +02:00
|
|
|
integer,intent(in) :: N
|
|
|
|
double precision,intent(inout):: A(N,N)
|
2019-10-28 16:34:09 +01:00
|
|
|
double precision,intent(out) :: VR(N,N)
|
|
|
|
double precision,intent(out) :: WR(N)
|
2019-10-15 17:56:27 +02:00
|
|
|
|
|
|
|
! Local variables
|
|
|
|
|
|
|
|
integer :: lwork,info
|
2019-10-30 10:31:05 +01:00
|
|
|
double precision,allocatable :: work(:),WI(:),VL(:,:)
|
2019-10-15 17:56:27 +02:00
|
|
|
|
|
|
|
! Memory allocation
|
|
|
|
|
|
|
|
lwork = 4*N
|
2019-10-30 10:31:05 +01:00
|
|
|
allocate(WI(N),VL(N,N),work(lwork))
|
2020-03-25 21:10:21 +01:00
|
|
|
|
2019-10-28 16:34:09 +01:00
|
|
|
call dgeev('V','V',N,A,N,WR,WI,VL,N,VR,N,work,lwork,info)
|
2019-10-15 17:56:27 +02:00
|
|
|
|
|
|
|
if(info /= 0) then
|
2019-10-28 16:34:09 +01:00
|
|
|
print*,'Problem in diagonalize_general_matrix (dgeev)!!'
|
2019-10-15 17:56:27 +02:00
|
|
|
endif
|
|
|
|
|
|
|
|
end subroutine diagonalize_general_matrix
|
|
|
|
|
2019-03-20 13:38:42 +01:00
|
|
|
subroutine diagonalize_matrix(N,A,e)
|
|
|
|
|
|
|
|
! Diagonalize a square matrix
|
|
|
|
|
|
|
|
implicit none
|
|
|
|
|
|
|
|
! Input variables
|
|
|
|
|
|
|
|
integer,intent(in) :: N
|
|
|
|
double precision,intent(inout):: A(N,N)
|
|
|
|
double precision,intent(out) :: e(N)
|
|
|
|
|
|
|
|
! Local variables
|
|
|
|
|
|
|
|
integer :: lwork,info
|
|
|
|
double precision,allocatable :: work(:)
|
|
|
|
|
|
|
|
! Memory allocation
|
|
|
|
|
|
|
|
allocate(work(3*N))
|
|
|
|
lwork = size(work)
|
|
|
|
|
|
|
|
call dsyev('V','U',N,A,N,e,work,lwork,info)
|
|
|
|
|
|
|
|
if(info /= 0) then
|
|
|
|
print*,'Problem in diagonalize_matrix (dsyev)!!'
|
|
|
|
endif
|
2019-10-15 17:56:27 +02:00
|
|
|
|
2019-03-20 13:38:42 +01:00
|
|
|
end subroutine diagonalize_matrix
|
|
|
|
|
|
|
|
subroutine svd(N,A,U,D,Vt)
|
|
|
|
|
|
|
|
! Compute A = U.D.Vt
|
|
|
|
! Dimension of A is NxN
|
|
|
|
|
|
|
|
implicit none
|
|
|
|
|
|
|
|
integer, intent(in) :: N
|
|
|
|
double precision,intent(in) :: A(N,N)
|
|
|
|
double precision,intent(out) :: U(N,N)
|
|
|
|
double precision,intent(out) :: Vt(N,N)
|
|
|
|
double precision,intent(out) :: D(N)
|
|
|
|
double precision,allocatable :: work(:)
|
|
|
|
integer :: info,lwork
|
|
|
|
|
|
|
|
double precision,allocatable :: scr(:,:)
|
|
|
|
|
|
|
|
allocate (scr(N,N))
|
|
|
|
|
|
|
|
scr(:,:) = A(:,:)
|
|
|
|
|
|
|
|
! Find optimal size for temporary arrays
|
|
|
|
|
|
|
|
allocate(work(1))
|
|
|
|
|
|
|
|
lwork = -1
|
|
|
|
call dgesvd('A','A',N,N,scr,N,D,U,N,Vt,N,work,lwork,info)
|
|
|
|
lwork = int(work(1))
|
|
|
|
|
|
|
|
deallocate(work)
|
|
|
|
|
|
|
|
allocate(work(lwork))
|
|
|
|
|
|
|
|
call dgesvd('A','A',N,N,scr,N,D,U,N,Vt,N,work,lwork,info)
|
|
|
|
|
|
|
|
deallocate(work,scr)
|
|
|
|
|
|
|
|
if (info /= 0) then
|
|
|
|
print *, info, ': SVD failed'
|
|
|
|
stop
|
|
|
|
endif
|
|
|
|
|
|
|
|
end
|
|
|
|
|
|
|
|
subroutine inverse_matrix(N,A,B)
|
|
|
|
|
|
|
|
! Returns the inverse of the square matrix A in B
|
|
|
|
|
|
|
|
implicit none
|
|
|
|
|
|
|
|
integer,intent(in) :: N
|
|
|
|
double precision, intent(in) :: A(N,N)
|
|
|
|
double precision, intent(out) :: B(N,N)
|
|
|
|
|
|
|
|
integer :: info,lwork
|
|
|
|
integer, allocatable :: ipiv(:)
|
|
|
|
double precision,allocatable :: work(:)
|
|
|
|
|
|
|
|
allocate (ipiv(N),work(N*N))
|
|
|
|
lwork = size(work)
|
|
|
|
|
|
|
|
B(1:N,1:N) = A(1:N,1:N)
|
|
|
|
|
|
|
|
call dgetrf(N,N,B,N,ipiv,info)
|
|
|
|
|
|
|
|
if (info /= 0) then
|
|
|
|
|
|
|
|
print*,info
|
|
|
|
stop 'error in inverse (dgetrf)!!'
|
|
|
|
|
|
|
|
endif
|
|
|
|
|
|
|
|
call dgetri(N,B,N,ipiv,work,lwork,info)
|
|
|
|
|
|
|
|
if (info /= 0) then
|
|
|
|
|
|
|
|
print *, info
|
|
|
|
stop 'error in inverse (dgetri)!!'
|
|
|
|
|
|
|
|
endif
|
|
|
|
|
|
|
|
deallocate(ipiv,work)
|
|
|
|
|
|
|
|
end subroutine inverse_matrix
|
|
|
|
|
|
|
|
subroutine linear_solve(N,A,b,x,rcond)
|
|
|
|
|
|
|
|
! Solve the linear system A.x = b where A is a NxN matrix
|
|
|
|
! and x and x are vectors of size N
|
|
|
|
|
|
|
|
implicit none
|
|
|
|
|
|
|
|
integer,intent(in) :: N
|
|
|
|
double precision,intent(in) :: A(N,N),b(N),rcond
|
|
|
|
double precision,intent(out) :: x(N)
|
|
|
|
|
|
|
|
integer :: info,lwork
|
|
|
|
double precision :: ferr,berr
|
|
|
|
integer,allocatable :: ipiv(:),iwork(:)
|
|
|
|
double precision,allocatable :: AF(:,:),work(:)
|
|
|
|
|
|
|
|
lwork = 3*N
|
|
|
|
allocate(AF(N,N),ipiv(N),work(lwork),iwork(N))
|
|
|
|
|
|
|
|
call dsysvx('N','U',N,1,A,N,AF,N,ipiv,b,N,x,N,rcond,ferr,berr,work,lwork,iwork,info)
|
|
|
|
|
|
|
|
! if (info /= 0) then
|
|
|
|
|
|
|
|
! print *, info
|
|
|
|
! stop 'error in linear_solve (dsysvx)!!'
|
|
|
|
|
|
|
|
! endif
|
|
|
|
|
|
|
|
end subroutine linear_solve
|
|
|
|
|
|
|
|
subroutine easy_linear_solve(N,A,b,x)
|
|
|
|
|
|
|
|
! Solve the linear system A.x = b where A is a NxN matrix
|
|
|
|
! and x and x are vectors of size N
|
|
|
|
|
|
|
|
implicit none
|
|
|
|
|
|
|
|
integer,intent(in) :: N
|
|
|
|
double precision,intent(in) :: A(N,N),b(N)
|
|
|
|
double precision,intent(out) :: x(N)
|
|
|
|
|
|
|
|
integer :: info,lwork
|
|
|
|
integer,allocatable :: ipiv(:)
|
|
|
|
double precision,allocatable :: work(:)
|
|
|
|
|
|
|
|
allocate(ipiv(N),work(N*N))
|
|
|
|
lwork = size(work)
|
|
|
|
|
|
|
|
x = b
|
|
|
|
|
|
|
|
call dsysv('U',N,1,A,N,ipiv,x,N,work,lwork,info)
|
|
|
|
|
|
|
|
if (info /= 0) then
|
|
|
|
|
|
|
|
print *, info
|
|
|
|
stop 'error in linear_solve (dsysv)!!'
|
|
|
|
|
|
|
|
endif
|
|
|
|
|
|
|
|
end subroutine easy_linear_solve
|
|
|
|
|