You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
343 lines
10 KiB
Fortran
343 lines
10 KiB
Fortran
program zf_bench
|
|
use precd
|
|
use getp
|
|
use psb_base_mod
|
|
use psb_util_mod
|
|
use mld_prec_mod
|
|
use psb_krylov_mod
|
|
implicit none
|
|
|
|
! input parameters
|
|
character(len=20) :: cmethd
|
|
character(len=80) :: outf1, outf2, outf3
|
|
character(len=20), allocatable :: mtrx(:),rhs(:)
|
|
type(precdata), allocatable :: precs(:)
|
|
|
|
|
|
! sparse matrices
|
|
type(psb_zspmat_type) :: a, aux_a
|
|
|
|
! preconditioner data
|
|
type(mld_zprec_type) :: pre
|
|
integer :: igsmth, matop, novr
|
|
|
|
! dense matrices
|
|
complex(kind(1.d0)), allocatable, target :: aux_b(:,:), d(:)
|
|
complex(kind(1.d0)), allocatable , save :: b_col(:), x_col(:), r_col(:), &
|
|
& x_col_glob(:), r_col_glob(:)
|
|
complex(kind(1.d0)), pointer :: b_col_glob(:)
|
|
|
|
! communications data structure
|
|
type(psb_desc_type):: desc_a
|
|
|
|
! blacs variables
|
|
integer :: ictxt, iam, np, root
|
|
logical :: amroot, out1, out2
|
|
|
|
! solver paramters
|
|
integer :: iter, itmax, ierr, itrace, ircode, ipart,&
|
|
& methd, istopc, iprec, ml, irnum, irst, ntry, nmat, ilev,nlev
|
|
real(kind(1.d0)) :: err, eps
|
|
|
|
character(len=5) :: afmt
|
|
character(len=20) :: name
|
|
integer :: iparm(20)
|
|
|
|
! other variables
|
|
integer :: i,info,j,m_problem, nm, nt
|
|
integer :: internal, m,ii,nnzero, nprecs, pp
|
|
real(kind(1.d0)) :: t1, t2, tprec, r_amax, b_amax,&
|
|
&scale,resmx,resmxp, mttot, mtslv, mtprec
|
|
integer :: nrhs, nrow, n_row, dim, nv, ne
|
|
integer, allocatable :: ipv(:), neigh(:), ivg(:)
|
|
|
|
|
|
call psb_init(ictxt)
|
|
call psb_info(ictxt,iam,np)
|
|
|
|
if (iam < 0) then
|
|
! This should not happen, but just in case
|
|
call psb_exit(ictxt)
|
|
stop
|
|
endif
|
|
|
|
call enablecore()
|
|
|
|
root = 0
|
|
amroot = (iam==root)
|
|
|
|
name='df_sample'
|
|
if(psb_get_errstatus() /= 0) goto 9999
|
|
info=0
|
|
call psb_set_errverbosity(2)
|
|
call psb_set_erraction(psb_act_abort_)
|
|
!
|
|
! get parameters
|
|
!
|
|
call get_parms(ictxt,irst,irnum,ntry,nmat,mtrx,rhs,cmethd,nprecs,precs,&
|
|
& ipart,afmt,istopc,itmax,itrace,eps,outf1,outf2)
|
|
|
|
if(amroot) then
|
|
if(outf1 /= 'NONE') then
|
|
open(8,file=outf1,action='write')
|
|
out1=.true.
|
|
else
|
|
out1=.false.
|
|
end if
|
|
|
|
if(outf2 /= 'NONE') then
|
|
open(10,file=outf2,action='write')
|
|
out2=.true.
|
|
else
|
|
out2=.false.
|
|
end if
|
|
end if
|
|
|
|
do nm=1, nmat
|
|
|
|
if(amroot) write(*,'(25("=")," ",a20," ",25("="))')mtrx(nm)
|
|
call psb_barrier(ictxt)
|
|
t1 = psb_wtime()
|
|
! read the input matrix to be processed and (possibly) the rhs
|
|
nrhs = 1
|
|
|
|
if (amroot) then
|
|
call read_mat(mtrx(nm), aux_a, ictxt)
|
|
|
|
m_problem = aux_a%m
|
|
call psb_bcast(ictxt,m_problem,root)
|
|
|
|
if(rhs(nm) /= 'none') then
|
|
! reading an rhs
|
|
call read_rhs(rhs(nm),aux_b,ictxt)
|
|
end if
|
|
|
|
if (allocated(aux_b).and.size(aux_b,1)==m_problem) then
|
|
! if any rhs were present, broadcast the first one
|
|
b_col_glob =>aux_b(:,1)
|
|
else
|
|
allocate(aux_b(m_problem,1), stat=ircode)
|
|
if (ircode /= 0) then
|
|
call psb_errpush(4000,name)
|
|
goto 9999
|
|
endif
|
|
|
|
b_col_glob => aux_b(:,1)
|
|
do i=1, m_problem
|
|
b_col_glob(i) = 1.d0
|
|
enddo
|
|
endif
|
|
call psb_bcast(ictxt,b_col_glob(1:m_problem),root)
|
|
else
|
|
call psb_bcast(ictxt,m_problem,root)
|
|
allocate(aux_b(m_problem,1), stat=ircode)
|
|
if (ircode /= 0) then
|
|
call psb_errpush(4000,name)
|
|
goto 9999
|
|
endif
|
|
b_col_glob =>aux_b(:,1)
|
|
call psb_bcast(ictxt,b_col_glob(1:m_problem),root)
|
|
end if
|
|
|
|
! switch over different partition types
|
|
if (ipart == 0) then
|
|
call psb_barrier(ictxt)
|
|
! if (amroot) write(*,'("Partition type: block")')
|
|
allocate(ivg(m_problem),ipv(np))
|
|
do i=1,m_problem
|
|
call part_block(i,m_problem,np,ipv,nv)
|
|
ivg(i) = ipv(1)
|
|
enddo
|
|
call psb_matdist(aux_a, a, ivg, ictxt, &
|
|
& desc_a,b_col_glob,b_col,info,fmt=afmt)
|
|
else if (ipart == 2) then
|
|
if (amroot) then
|
|
call build_mtpart(aux_a%m,aux_a%fida,aux_a%ia1,aux_a%ia2,np)
|
|
endif
|
|
call psb_barrier(ictxt)
|
|
call distr_mtpart(0,ictxt)
|
|
call getv_mtpart(ivg)
|
|
call psb_matdist(aux_a, a, ivg, ictxt, &
|
|
& desc_a,b_col_glob,b_col,info,fmt=afmt)
|
|
call free_part(info)
|
|
else
|
|
call psb_matdist(aux_a, a, part_block, ictxt, &
|
|
& desc_a,b_col_glob,b_col,info,fmt=afmt)
|
|
end if
|
|
|
|
call psb_geall(x_col,desc_a,info)
|
|
x_col(:) =0.0
|
|
call psb_geasb(x_col,desc_a,info)
|
|
call psb_geall(r_col,desc_a,info)
|
|
r_col(:) =0.0
|
|
call psb_geasb(r_col,desc_a,info)
|
|
t2 = psb_wtime() - t1
|
|
|
|
!!$ call psb_csprt(10+iam,a,head='% (A)')
|
|
call psb_sp_free(aux_a,info)
|
|
|
|
call psb_amx(ictxt, t2)
|
|
|
|
!!$ call psb_csprt(20+iam,a,head='% (A)')
|
|
!
|
|
! prepare the preconditioning matrix. note the availability
|
|
! of optional parameters
|
|
!
|
|
|
|
|
|
do pp=1, nprecs
|
|
|
|
mttot=1.d300
|
|
|
|
do nt=1,ntry
|
|
|
|
if (precs(pp)%lv2(1:2) == 'ml') then
|
|
if (precs(pp)%nlev < 2) then
|
|
write(0,*) 'Inconsistent number of levels ',precs(pp)%nlev,&
|
|
& ' forcing 2'
|
|
precs(pp)%nlev = 2
|
|
end if
|
|
nlev = precs(pp)%nlev
|
|
call mld_precinit(pre,precs(pp)%lv2,info,nlev=nlev)
|
|
! Defaults are OK for all intermediate levels. Only fix last level.
|
|
if (precs(pp)%omega>=0.0) then
|
|
call mld_precset(pre,mld_aggr_damp_,precs(pp)%omega,info,ilev=nlev)
|
|
end if
|
|
call mld_precset(pre,mld_ml_type_, precs(pp)%mltype, info,ilev=nlev)
|
|
call mld_precset(pre,mld_aggr_alg_, precs(pp)%aggr, info,ilev=nlev)
|
|
call mld_precset(pre,mld_coarse_mat_, precs(pp)%cmat, info,ilev=nlev)
|
|
call mld_precset(pre,mld_smooth_pos_, precs(pp)%smthpos, info,ilev=nlev)
|
|
call mld_precset(pre,mld_sub_solve_, precs(pp)%ftype2, info,ilev=nlev)
|
|
call mld_precset(pre,mld_smooth_sweeps_, precs(pp)%jswp, info,ilev=nlev)
|
|
call mld_precset(pre,mld_aggr_kind_, precs(pp)%smthkind,info,ilev=nlev)
|
|
else
|
|
call mld_precinit(pre,precs(pp)%lv1,info)
|
|
end if
|
|
call mld_precset(pre,mld_n_ovr_, precs(pp)%novr,info ,ilev=1)
|
|
call mld_precset(pre,mld_sub_restr_, precs(pp)%restr,info ,ilev=1)
|
|
call mld_precset(pre,mld_sub_prol_, precs(pp)%prol,info ,ilev=1)
|
|
call mld_precset(pre,mld_sub_solve_, precs(pp)%ftype1,info ,ilev=1)
|
|
|
|
|
|
|
|
! setting initial guess to zero
|
|
call psb_geaxpby(zzero,b_col,zzero,x_col,desc_a,info)
|
|
|
|
! building the preconditioner
|
|
call psb_barrier(ictxt)
|
|
t1 = psb_wtime()
|
|
call mld_precbld(a,desc_a,pre,info)
|
|
tprec = psb_wtime()-t1
|
|
if (info /= 0) then
|
|
write(0,*) 'INFO from precbld ',info
|
|
call psb_error()
|
|
goto 9999
|
|
end if
|
|
|
|
|
|
call psb_amx(ictxt,tprec)
|
|
!!$ call psb_csprt(40+iam,a,head='% (A)')
|
|
!!$ if (amroot) then
|
|
!!$ write(*,'("Matrix : ",a)') mtrx(nm)
|
|
!!$ write(*,'("RHS : ",a)') rhs(nm)
|
|
!!$ write(*,'("Method : ",a)') cmethd
|
|
!!$ write(*,'("Preconditioner : ",a)') precs(pp)%descr
|
|
!!$ call mld_prec_descr(6,pre)
|
|
!!$ end if
|
|
iparm = 0
|
|
call psb_barrier(ictxt)
|
|
t1 = psb_wtime()
|
|
call psb_krylov(cmethd,a,pre,b_col,x_col,eps,desc_a,info,&
|
|
& itmax,iter,err,itrace,irst=ml,istop=istopc)
|
|
|
|
call psb_barrier(ictxt)
|
|
t2 = psb_wtime() - t1
|
|
call psb_amx(ictxt,t2)
|
|
if (info/=0) then
|
|
write(0,*) 'INFO from solver ',info
|
|
call psb_errpush(4010,name,a_err='psb_SOLVER')
|
|
goto 9999
|
|
end if
|
|
!!$ write(0,*) iam,'Done Solver'
|
|
!!$ call flush(0)
|
|
!!$ call flush(6)
|
|
!!$ call psb_barrier(ictxt)
|
|
if(amroot.and.out2) &
|
|
& write(10,'(a20,2(1x,i3),1x,i5,3(1x,g9.4),1x,a)') &
|
|
& mtrx(nm),np,precs(pp)%novr,iter,tprec,t2,t2+tprec,&
|
|
& trim(precs(pp)%descr)
|
|
if(amroot) &
|
|
& write(0,'(a20,2(1x,i3),1x,i5,3(1x,g9.4),1x,a)') &
|
|
& mtrx(nm),np,precs(pp)%novr,iter,tprec,t2,t2+tprec,&
|
|
& trim(precs(pp)%descr)
|
|
|
|
if(nt.lt.ntry) call mld_precfree(pre,info)
|
|
if((t2+tprec).lt.mttot) then
|
|
mtslv=t2
|
|
mtprec=tprec
|
|
mttot=t2+tprec
|
|
end if
|
|
|
|
end do
|
|
!!$ call psb_csprt(50+iam,a,head='% (A)')
|
|
|
|
call psb_geaxpby(zone,b_col,zzero,r_col,desc_a,info)
|
|
call psb_spmm(-zone,a,x_col,zone,r_col,desc_a,info)
|
|
call psb_genrm2s(resmx,r_col,desc_a,info)
|
|
call psb_geamaxs(resmxp,r_col,desc_a,info)
|
|
|
|
if (amroot) then
|
|
write(*,'("Matrix : ",a)') mtrx(nm)
|
|
write(*,'("RHS : ",a)') rhs(nm)
|
|
write(*,'("Method : ",a)') cmethd
|
|
write(*,'("Preconditioner : ",a)') precs(pp)%descr
|
|
call mld_prec_descr(pre)
|
|
write(*,'("Computed solution on ",i4," processors")')np
|
|
write(*,'(" ")')
|
|
write(*,'("Iterations to convergence: ",i6)') iter
|
|
write(*,'("Error indicator on exit : ",g9.4)') err
|
|
write(*,'("Time to buil prec. : ",es10.4)')mtprec
|
|
write(*,'("Time to solve matrix : ",es10.4)')mtslv
|
|
write(*,'("Time per iteration : ",es10.4)')mtslv/(iter)
|
|
write(*,'("Total time : ",es10.4)')mttot
|
|
write(*,'("Residual norm 2 : ",es10.4)')resmx
|
|
write(*,'("Residual norm inf : ",es10.4)')resmxp
|
|
write(*,'(72("="))')
|
|
write(*,'(" ")')
|
|
write(*,'(" ")')
|
|
write(*,'(" ")')
|
|
|
|
if(out1) write(8,'(a20,2(1x,i3),1x,i5,5(1x,g9.4),1x,a)') mtrx(nm),&
|
|
& np,precs(pp)%novr,&
|
|
& iter,mtprec,mtslv,mttot,resmx,resmxp,trim(precs(pp)%descr)
|
|
end if
|
|
|
|
call mld_precfree(pre,info)
|
|
|
|
end do
|
|
|
|
call psb_gefree(b_col, desc_a,info)
|
|
call psb_gefree(x_col, desc_a,info)
|
|
call psb_spfree(a, desc_a,info)
|
|
call psb_cdfree(desc_a,info)
|
|
deallocate(r_col,stat=info)
|
|
deallocate(aux_b,stat=info)
|
|
if (ipart==0) then
|
|
deallocate(ivg,ipv,stat=info)
|
|
endif
|
|
end do
|
|
|
|
9999 continue
|
|
if(info /= 0) then
|
|
call psb_error(ictxt)
|
|
end if
|
|
call psb_exit(ictxt)
|
|
stop
|
|
|
|
end program zf_bench
|
|
|
|
|
|
|
|
|
|
|