From debb03b6ae094afbcf50c89ba9a753973a355b7f Mon Sep 17 00:00:00 2001 From: Salvatore Filippone Date: Wed, 20 May 2020 09:44:17 +0200 Subject: [PATCH 1/3] Prepare for alternative code path --- base/internals/psi_crea_index.f90 | 79 ++++-- base/internals/psi_extrct_dl.F90 | 21 +- base/internals/psi_xtr_loc_dl.F90 | 394 ++++++++++++++++++++++++++++++ base/modules/psi_i_mod.F90 | 15 +- 4 files changed, 474 insertions(+), 35 deletions(-) create mode 100644 base/internals/psi_xtr_loc_dl.F90 diff --git a/base/internals/psi_crea_index.f90 b/base/internals/psi_crea_index.f90 index 126d54e2..51046737 100644 --- a/base/internals/psi_crea_index.f90 +++ b/base/internals/psi_crea_index.f90 @@ -107,36 +107,67 @@ subroutine psi_i_crea_index(desc_a,index_in,index_out,nxch,nsnd,nrcv,info) if (debug_level >= psb_debug_inner_) & & write(debug_unit,*) me,' ',trim(name),': calling extract_dep_list' mode = 1 - if (do_timings) call psb_tic(idx_phase1) + if (.false.) then + if (do_timings) call psb_tic(idx_phase1) - call psi_extract_dep_list(ictxt,& - & desc_a%is_bld(), desc_a%is_upd(),& - & index_in, dep_list,length_dl,dl_lda,mode,info) - if (info /= psb_success_) then - call psb_errpush(psb_err_from_subroutine_,name,a_err='extrct_dl') - goto 9999 - end if + call psi_extract_dep_list(ictxt,& + & desc_a%is_bld(), desc_a%is_upd(),& + & index_in, dep_list,length_dl,dl_lda,mode,info) + if (info /= psb_success_) then + call psb_errpush(psb_err_from_subroutine_,name,a_err='extrct_dl') + goto 9999 + end if - if (debug_level >= psb_debug_inner_) & - & write(debug_unit,*) me,' ',trim(name),': from extract_dep_list',& - & me,length_dl(0),index_in(1), ':',dep_list(:length_dl(me),me) - ! ...now process root contains dependence list of all processes... - if (debug_level >= psb_debug_inner_) & - & write(debug_unit,*) me,' ',trim(name),': root sorting dep list' - if (do_timings) call psb_toc(idx_phase1) - if (do_timings) call psb_tic(idx_phase2) + if (debug_level >= psb_debug_inner_) & + & write(debug_unit,*) me,' ',trim(name),': from extract_dep_list',& + & me,length_dl(0),index_in(1), ':',dep_list(:length_dl(me),me) + ! ...now process root contains dependence list of all processes... + if (debug_level >= psb_debug_inner_) & + & write(debug_unit,*) me,' ',trim(name),': root sorting dep list' + if (do_timings) call psb_toc(idx_phase1) + if (do_timings) call psb_tic(idx_phase2) - call psi_dl_check(dep_list,dl_lda,np,length_dl) + call psi_dl_check(dep_list,dl_lda,np,length_dl) - ! ....now i can sort dependency lists. - call psi_sort_dl(dep_list,length_dl,np,info) - if(info /= psb_success_) then - call psb_errpush(psb_err_from_subroutine_,name,a_err='psi_sort_dl') - goto 9999 + ! ....now i can sort dependency lists. + call psi_sort_dl(dep_list,length_dl,np,info) + if(info /= psb_success_) then + call psb_errpush(psb_err_from_subroutine_,name,a_err='psi_sort_dl') + goto 9999 + end if + if (do_timings) call psb_toc(idx_phase2) + else + if (do_timings) call psb_tic(idx_phase1) + + call psi_extract_dep_list(ictxt,& + & desc_a%is_bld(), desc_a%is_upd(),& + & index_in, dep_list,length_dl,dl_lda,mode,info) + if (info /= psb_success_) then + call psb_errpush(psb_err_from_subroutine_,name,a_err='extrct_dl') + goto 9999 + end if + + if (debug_level >= psb_debug_inner_) & + & write(debug_unit,*) me,' ',trim(name),': from extract_dep_list',& + & me,length_dl(0),index_in(1), ':',dep_list(:length_dl(me),me) + ! ...now process root contains dependence list of all processes... + if (debug_level >= psb_debug_inner_) & + & write(debug_unit,*) me,' ',trim(name),': root sorting dep list' + if (do_timings) call psb_toc(idx_phase1) + if (do_timings) call psb_tic(idx_phase2) + + call psi_dl_check(dep_list,dl_lda,np,length_dl) + + ! ....now i can sort dependency lists. + call psi_sort_dl(dep_list,length_dl,np,info) + if(info /= psb_success_) then + call psb_errpush(psb_err_from_subroutine_,name,a_err='psi_sort_dl') + goto 9999 + end if + if (do_timings) call psb_toc(idx_phase2) end if - if (do_timings) call psb_toc(idx_phase2) if (do_timings) call psb_tic(idx_phase3) - + if(debug_level >= psb_debug_inner_)& & write(debug_unit,*) me,' ',trim(name),': calling psi_desc_index' ! Do the actual format conversion. diff --git a/base/internals/psi_extrct_dl.F90 b/base/internals/psi_extrct_dl.F90 index 2ee8f2ce..dc2b0e14 100644 --- a/base/internals/psi_extrct_dl.F90 +++ b/base/internals/psi_extrct_dl.F90 @@ -252,6 +252,17 @@ subroutine psi_i_extract_dep_list(ictxt,is_bld,is_upd,desc_str,dep_list,& endif length_dl(me)=pointer_dep_list-1 + if ((profile).and.(me==0)) then + block + integer(psb_ipk_) :: dlmax, dlavg + dlmax = maxval(length_dl(:)) + dlavg = (sum(length_dl(:))+np-1)/np + if (dlmax>0) write(0,*) 'Dependency list : max:',dlmax,& + & ' avg:',dlavg, ((dlmax>np/3).or.((dlavg>=np/4).and.(np>128))) + + end block + end if + if (dist_symm_list) then call psb_realloc(length_dl(me),itmp,info) call psi_symm_dep_list(itmp,ictxt,info) @@ -367,16 +378,6 @@ subroutine psi_i_extract_dep_list(ictxt,is_bld,is_upd,desc_str,dep_list,& end if call psb_barrier(ictxt) end if - if ((profile).and.(me==0)) then - block - integer(psb_ipk_) :: dlmax, dlavg - dlmax = maxval(length_dl(:)) - dlavg = (sum(length_dl(:))+np-1)/np - if (dlmax>0) write(0,*) 'Dependency list : max:',dlmax,& - & ' avg:',dlavg, ((dlmax>np/2).or.((dlavg>=np/4).and.(np>128))) - - end block - end if call psb_erractionrestore(err_act) return diff --git a/base/internals/psi_xtr_loc_dl.F90 b/base/internals/psi_xtr_loc_dl.F90 new file mode 100644 index 00000000..4941982a --- /dev/null +++ b/base/internals/psi_xtr_loc_dl.F90 @@ -0,0 +1,394 @@ +! +! Parallel Sparse BLAS version 3.5 +! (C) Copyright 2006-2018 +! Salvatore Filippone +! Alfredo Buttari +! +! Redistribution and use in source and binary forms, with or without +! modification, are permitted provided that the following conditions +! are met: +! 1. Redistributions of source code must retain the above copyright +! notice, this list of conditions and the following disclaimer. +! 2. Redistributions in binary form must reproduce the above copyright +! notice, this list of conditions, and the following disclaimer in the +! documentation and/or other materials provided with the distribution. +! 3. The name of the PSBLAS group or the names of its contributors may +! not be used to endorse or promote products derived from this +! software without specific written permission. +! +! THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +! ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED +! TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR +! PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE PSBLAS GROUP OR ITS CONTRIBUTORS +! BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR +! CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF +! SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS +! INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN +! CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) +! ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE +! POSSIBILITY OF SUCH DAMAGE. +! +! +subroutine psi_i_xtr_loc_dl(ictxt,is_bld,is_upd,desc_str,loc_dl,& + & ldl,max_ldl,mode,info) + + ! internal routine + ! == = ============= + ! + ! _____called by psi_crea_halo and psi_crea_ovrlap ______ + ! + ! purpose + ! == = ==== + ! process root (pid=0) extracts for each process "k" the ordered list of process + ! to which "k" must communicate. this list with its order is extracted from + ! desc_str list + ! + ! + ! input + ! == = ==== + ! desc_data :integer array + ! explanation: + ! name explanation + ! ------------------ ------------------------------------------------------- + ! desc_data array of integer that contains some local and global + ! information of matrix. + ! + ! + ! now we explain each of the above vectors. + ! + ! let a be a generic sparse matrix. we denote with matdata_a the matrix_data + ! array for matrix a. + ! data stored in matrix_data array are: + ! + ! notation stored in explanation + ! --------------- ---------------------- ------------------------------------- + ! dec_type matdata_a[psb_dec_type_] decomposition type + ! m matdata_a[m_] total number of equations + ! n matdata_a[n_] total number of variables + ! n_row matdata_a[psb_n_row_] number of local equations + ! n_col matdata_a[psb_n_col_] number of local variables + ! psb_ctxt_a matdata_a[ctxt_] the blacs context handle, indicating + ! the global context of the operation + ! on the matrix. + ! the context itself is global. + ! desc_str integer array + ! explanation: + ! let desc_str_p be the array desc_str for local process. + !! this is composed of variable dimension blocks for each process to + ! communicate to. + ! each block contain indexes of local halo elements to exchange with other + ! process. + ! let p be the pointer to the first element of a block in desc_str_p. + ! this block is stored in desc_str_p as : + ! + ! notation stored in explanation + ! --------------- --------------------------- ----------------------------------- + ! process_id desc_str_p[p+psb_proc_id_] identifier of process which exchange + ! data with. + ! n_elements_recv desc_str_p[p+n_elem_recv_] number of elements to receive. + ! elements_recv desc_str_p[p+elem_recv_+i] indexes of local elements to + ! receive. these are stored in the + ! array from location p+elem_recv_ to + ! location p+elem_recv_+ + ! desc_str_p[p+n_elem_recv_]-1. + ! if desc_data(psb_dec_type_) == 0 + ! then also will be: + ! n_elements_send desc_str_p[p+n_elem_send_] number of elements to send. + ! elements_send desc_str_p[p+elem_send_+i] indexes of local elements to + ! send. these are stored in the + ! array from location p+elem_send_ to + ! location p+elem_send_+ + ! desc_str_p[p+n_elem_send_]-1. + ! list is ended by -1 value + ! + ! np integer (global input) + ! number of grid process. + ! + ! mode integer (global input) + ! if mode =0 then will be inserted also duplicate element in + ! a same dependence list + ! if mode =1 then not will be inserted duplicate element in + ! a same dependence list + ! output + ! == = == + ! only for root (pid=0) process: + ! dep_list integer array(dl_lda,0:np) + ! dependence list dep_list(*,i) is the list of process identifiers to which process i + ! must communicate with. this list with its order is extracted from + ! desc_str list. + ! length_dl integer array(0:np) + ! length_dl(i) is the length of dep_list(*,i) list + use psi_mod, psb_protect_name => psi_i_xtr_loc_dl +#ifdef MPI_MOD + use mpi +#endif + use psb_penv_mod + use psb_const_mod + use psb_error_mod + use psb_desc_mod + use psb_sort_mod + implicit none +#ifdef MPI_H + include 'mpif.h' +#endif + ! ....scalar parameters... + logical, intent(in) :: is_bld, is_upd + integer(psb_ipk_), intent(in) :: ictxt,mode + integer(psb_ipk_), intent(out) :: max_ldl, ldl + integer(psb_ipk_), intent(in) :: desc_str(*) + integer(psb_ipk_), allocatable, intent(out) :: loc_dl(:) + integer(psb_ipk_), intent(out) :: info + ! .....local arrays.... + integer(psb_ipk_) :: int_err(5) + integer(psb_ipk_), allocatable :: itmp(:) + + ! .....local scalars... + integer(psb_ipk_) :: i,pointer_dep_list,proc,j,err_act + integer(psb_ipk_) :: err + integer(psb_ipk_) :: debug_level, debug_unit + integer(psb_ipk_), allocatable :: length_dl(:) + integer(psb_mpk_) :: iictxt, icomm, me, np, minfo + logical, parameter :: dist_symm_list=.false., print_dl=.false., profile=.true. + character name*20 + name='psi_extrct_dl' + + call psb_erractionsave(err_act) + debug_unit = psb_get_debug_unit() + debug_level = psb_get_debug_level() + iictxt = ictxt + info = psb_success_ + + call psb_info(iictxt,me,np) + allocate(itmp(2*np+1),length_dl(0:np),stat=info) + if (info /= psb_success_) then + info=psb_err_alloc_dealloc_ + goto 9999 + end if + do i=0,np + length_dl(i) = 0 + itmp(i+1) = -1 + enddo + i=1 + if (debug_level >= psb_debug_inner_)& + & write(debug_unit,*) me,' ',trim(name),': start ',info + + pointer_dep_list=1 + if (is_bld) then + do while (desc_str(i) /= -1) + if (debug_level >= psb_debug_inner_)& + & write(debug_unit,*) me,' ',trim(name),' : looping ',i,& + & desc_str(i),desc_str(i+1),desc_str(i+2) + + ! ...with different decomposition type we have different + ! structure of indices lists............................ + if ((desc_str(i+1) /= 0).or.(desc_str(i+2) /= 0)) then + ! ..if number of element to be exchanged !=0 + proc=desc_str(i) + if ((proc < 0).or.(proc >= np)) then + if (debug_level >= psb_debug_inner_)& + & write(debug_unit,*) me,' ',trim(name),': error ',i,desc_str(i) + info = 9999 + int_err(1) = i + int_err(2) = desc_str(i) + goto 9999 + endif + ! if((me == 1).and.(proc == 3))write(psb_err_unit,*)'found 3' + if (mode == 1) then + ! ...search if already exist proc + ! in itmp(*)... + j=1 + do while ((j < pointer_dep_list).and.& + & (itmp(j) /= proc)) + j=j+1 + enddo + + if (j == pointer_dep_list) then + ! ...if not found..... + itmp(pointer_dep_list)=proc + pointer_dep_list=pointer_dep_list+1 + endif + else if (mode == 0) then + itmp(pointer_dep_list)=proc + pointer_dep_list=pointer_dep_list+1 + endif + endif + i=i+desc_str(i+1)+2 + enddo + + else if (is_upd) then + + do while (desc_str(i) /= -1) + if (debug_level >= psb_debug_inner_) & + & write(debug_unit,*) me,' ',trim(name),': looping ',i,desc_str(i) + + ! ...with different decomposition type we have different + ! structure of indices lists............................ + if (desc_str(i+1) /= 0) then + + proc=desc_str(i) + ! ..if number of element to be exchanged !=0 + + if (mode == 1) then + ! ...search if already exist proc.... + j=1 + do while ((j < pointer_dep_list).and.& + & (itmp(j) /= proc)) + j=j+1 + enddo + if (j == pointer_dep_list) then + ! ...if not found..... + itmp(pointer_dep_list)=proc + pointer_dep_list=pointer_dep_list+1 + endif + else if (mode == 0) then + itmp(pointer_dep_list)=proc + pointer_dep_list=pointer_dep_list+1 + endif + endif + i=i+desc_str(i+1)+2 + enddo + else + info = 2020 + goto 9999 + endif + + length_dl(me)=pointer_dep_list-1 + if ((profile).and.(me==0)) then + block + integer(psb_ipk_) :: dlmax, dlavg + dlmax = maxval(length_dl(:)) + dlavg = (sum(length_dl(:))+np-1)/np + if (dlmax>0) write(0,*) 'Dependency list : max:',dlmax,& + & ' avg:',dlavg, ((dlmax>np/3).or.((dlavg>=np/4).and.(np>128))) + + end block + end if + + if (dist_symm_list) then + call psb_realloc(length_dl(me),itmp,info) + call psi_symm_dep_list(itmp,ictxt,info) + dl_lda = max(size(itmp),1) + call psb_max(iictxt, dl_lda) + + if (debug_level >= psb_debug_inner_) & + & write(debug_unit,*) me,' ',trim(name),': Dep_list length ',length_dl(me),dl_lda + call psb_realloc(dl_lda,itmp,info) + ! dl_lda = min(np,2*dl_lda) + allocate(dep_list(dl_lda,0:np),stat=info) + if (info /= psb_success_) then + call psb_errpush(psb_err_from_subroutine_,name,a_err='Allocate') + goto 9999 + end if + + call psb_sum(iictxt,length_dl(0:np)) + icomm = psb_get_mpi_comm(iictxt) + call mpi_allgather(itmp,dl_lda,psb_mpi_ipk_,& + & dep_list,dl_lda,psb_mpi_ipk_,icomm,minfo) + info = minfo + if (info == 0) deallocate(itmp,stat=info) + if (info /= psb_success_) then + info=psb_err_alloc_dealloc_ + goto 9999 + endif + else + block + integer(psb_ipk_), allocatable :: list1(:,:), ldl2(:), list2(:,:) + integer(psb_ipk_) :: i,j,ip,dlsym, ldu, mdl, l1, l2 + + dl_lda = max(length_dl(me),1) + call psb_max(iictxt, dl_lda) + if (debug_level >= psb_debug_inner_) & + & write(debug_unit,*) me,' ',trim(name),': Dep_list length ',length_dl(me),dl_lda + allocate(dep_list(dl_lda,0:np),stat=info) + if (info /= psb_success_) then + call psb_errpush(psb_err_from_subroutine_,name,a_err='Allocate') + goto 9999 + end if + call psb_sum(iictxt,length_dl(0:np)) + icomm = psb_get_mpi_comm(iictxt) + call mpi_allgather(itmp,dl_lda,psb_mpi_ipk_,& + & dep_list,dl_lda,psb_mpi_ipk_,icomm,minfo) + info = minfo + if (info /= psb_success_) then + info=psb_err_alloc_dealloc_ + goto 9999 + endif + allocate(ldl2(0:np),stat=info) + ldl2 = 0 + do j=0, np-1 + do i=1,length_dl(j) + ip = dep_list(i,j) + ldl2(ip) = ldl2(ip) + 1 + end do + end do + dlsym = maxval(ldl2) + allocate(list2(dlsym,0:np),stat=info) + call move_alloc(dep_list,list1) + if (info /= psb_success_) then + call psb_errpush(psb_err_from_subroutine_,name,a_err='Allocate') + goto 9999 + end if + ldl2 = 0 + do j=0, np-1 + do i=1,length_dl(j) + ip = list1(i,j) + ldl2(ip) = ldl2(ip) + 1 + list2(ldl2(ip),ip) = j + end do + end do + mdl = 0 + do j = 0, np-1 + l1 = length_dl(j) + l2 = ldl2(j) + itmp(1:l1) = list1(1:l1,j) + itmp(l1+1:l1+l2) = list2(1:l2,j) + ldu = l1 + l2 + !if (me == 0) write(0,*) 'Iter ',j,':',l1,l2,':',itmp(1:l1),':',itmp(l1+1:l1+l2) + call psb_msort_unique(itmp(1:l1+l2),ldu) + mdl = max(mdl, ldu) + !if (me == 0) write(0,*) 'Iter ',j,':',ldu,':',itmp(1:ldu) + end do + dl_lda = mdl + allocate(dep_list(dl_lda,0:np),stat=info) + dep_list = -1 + if (info /= psb_success_) then + call psb_errpush(psb_err_from_subroutine_,name,a_err='Allocate') + goto 9999 + end if + do j = 0, np-1 + l1 = length_dl(j) + l2 = ldl2(j) + itmp(1:l1) = list1(1:l1,j) + itmp(l1+1:l1+l2) = list2(1:l2,j) + ldu = l1 + l2 + call psb_msort_unique(itmp(1:l1+l2),ldu) + length_dl(j) = ldu + dep_list(1:ldu,j) = itmp(1:ldu) + end do + + end block + endif + if (print_dl) then + if (me == 0) then + write(0,*) ' Dep_list ' + do i=0,np-1 + j = length_dl(i) + write(0,*) 'Proc ',i,':',dep_list(1:j,i) + end do + flush(0) + end if + call psb_barrier(ictxt) + end if + + call psb_erractionrestore(err_act) + return + + +9999 continue + + call psb_errpush(info,name,i_err=int_err) + call psb_error_handler(err_act) + + return + +end subroutine psi_i_xtr_loc_dl diff --git a/base/modules/psi_i_mod.F90 b/base/modules/psi_i_mod.F90 index ac2fcf06..eff2a05f 100644 --- a/base/modules/psi_i_mod.F90 +++ b/base/modules/psi_i_mod.F90 @@ -125,7 +125,20 @@ module psi_i_mod integer(psb_ipk_), allocatable, intent(out) :: dep_list(:,:), length_dl(:) integer(psb_ipk_), intent(out) :: info end subroutine psi_i_extract_dep_list - end interface + end interface psi_extract_dep_list + + interface psi_extract_loc_dl + subroutine psi_i_xtr_loc_dl(ictxt,is_bld,is_upd,desc_str,loc_dl,& + & ldl,max_ldl,mode,info) + import + logical, intent(in) :: is_bld, is_upd + integer(psb_ipk_), intent(in) :: ictxt,mode + integer(psb_ipk_), intent(out) :: max_ldl, ldl + integer(psb_ipk_), intent(in) :: desc_str(*) + integer(psb_ipk_), allocatable, intent(out) :: loc_dl(:) + integer(psb_ipk_), intent(out) :: info + end subroutine psi_i_xtr_loc_dl + end interface psi_extract_loc_dl interface psi_fnd_owner subroutine psi_i_fnd_owner(nv,idx,iprc,desc,info) From 2b295de9e762e6f8f5aba0fe9090100d287eaacf Mon Sep 17 00:00:00 2001 From: Salvatore Filippone Date: Wed, 20 May 2020 15:32:52 +0200 Subject: [PATCH 2/3] New crea_index internals. --- base/internals/Makefile | 3 +- base/internals/psi_crea_index.f90 | 99 ++++++++----- base/internals/psi_xtr_loc_dl.F90 | 222 ++++-------------------------- base/modules/psi_i_mod.F90 | 22 ++- 4 files changed, 116 insertions(+), 230 deletions(-) diff --git a/base/internals/Makefile b/base/internals/Makefile index 1ebdbcc0..cf499f2d 100644 --- a/base/internals/Makefile +++ b/base/internals/Makefile @@ -4,7 +4,8 @@ FOBJS = psi_compute_size.o psi_crea_bnd_elem.o psi_crea_index.o \ psi_crea_ovr_elem.o psi_bld_tmpovrl.o psi_dl_check.o \ psi_bld_tmphalo.o psi_sort_dl.o \ psi_indx_map_fnd_owner.o \ - psi_desc_impl.o psi_hash_impl.o psi_list_search.o psi_srtlist.o + psi_desc_impl.o psi_hash_impl.o psi_list_search.o psi_srtlist.o \ + psi_bld_glb_dep_list.o psi_xtr_loc_dl.o MPFOBJS = psi_desc_index.o psi_extrct_dl.o psi_fnd_owner.o psi_a2a_fnd_owner.o \ psi_graph_fnd_owner.o psi_adjcncy_fnd_owner.o psi_symm_dep_list.o diff --git a/base/internals/psi_crea_index.f90 b/base/internals/psi_crea_index.f90 index 51046737..eaa9f636 100644 --- a/base/internals/psi_crea_index.f90 +++ b/base/internals/psi_crea_index.f90 @@ -64,9 +64,10 @@ subroutine psi_i_crea_index(desc_a,index_in,index_out,nxch,nsnd,nrcv,info) integer(psb_ipk_), allocatable, intent(inout) :: index_out(:) ! ....local scalars... - integer(psb_ipk_) :: ictxt, me, np, mode, err_act, dl_lda + integer(psb_ipk_) :: ictxt, me, np, mode, err_act, dl_lda, ldl ! ...parameters... - integer(psb_ipk_), allocatable :: dep_list(:,:), length_dl(:) + integer(psb_ipk_), allocatable :: dep_list(:,:), length_dl(:), loc_dl(:) + integer(psb_ipk_) :: dlmax, dlavg integer(psb_ipk_),parameter :: root=psb_root_,no_comm=-1 integer(psb_ipk_) :: debug_level, debug_unit character(len=20) :: name @@ -136,54 +137,80 @@ subroutine psi_i_crea_index(desc_a,index_in,index_out,nxch,nsnd,nrcv,info) goto 9999 end if if (do_timings) call psb_toc(idx_phase2) + ldl = length_dl(me) + loc_dl = dep_list(1:ldl,me) + else + if (do_timings) call psb_tic(idx_phase1) - call psi_extract_dep_list(ictxt,& + call psi_extract_loc_dl(ictxt,& & desc_a%is_bld(), desc_a%is_upd(),& - & index_in, dep_list,length_dl,dl_lda,mode,info) - if (info /= psb_success_) then - call psb_errpush(psb_err_from_subroutine_,name,a_err='extrct_dl') - goto 9999 + & index_in, loc_dl,length_dl,info) + + dlmax = maxval(length_dl(:)) + dlavg = (sum(length_dl(:))+np-1)/np +!!$ if ((dlmax>0).and.(me==0)) write(0,*) 'Dependency list : max:',dlmax,& +!!$ & ' avg:',dlavg, ((dlmax>np/3).or.((dlavg>=np/4).and.(np>128))) + + if (choose_sorting(dlmax,dlavg,np)) then + + call psi_bld_glb_dep_list(ictxt,& + & loc_dl,length_dl,dep_list,dl_lda,info) + if (info /= psb_success_) then + call psb_errpush(psb_err_from_subroutine_,name,a_err='extrct_dl') + goto 9999 + end if + + if (debug_level >= psb_debug_inner_) & + & write(debug_unit,*) me,' ',trim(name),': from extract_dep_list',& + & me,length_dl(0),index_in(1), ':',dep_list(:length_dl(me),me) + ! ...now process root contains dependence list of all processes... + if (debug_level >= psb_debug_inner_) & + & write(debug_unit,*) me,' ',trim(name),': root sorting dep list' + if (do_timings) call psb_toc(idx_phase1) + if (do_timings) call psb_tic(idx_phase2) + + call psi_dl_check(dep_list,dl_lda,np,length_dl) + + ! ....now i can sort dependency lists. + call psi_sort_dl(dep_list,length_dl,np,info) + if(info /= psb_success_) then + call psb_errpush(psb_err_from_subroutine_,name,a_err='psi_sort_dl') + goto 9999 + end if + if (do_timings) call psb_toc(idx_phase2) + ldl = length_dl(me) + loc_dl = dep_list(1:ldl,me) + else + ! Do nothing + ldl = length_dl(me) + loc_dl = loc_dl(1:ldl) end if - if (debug_level >= psb_debug_inner_) & - & write(debug_unit,*) me,' ',trim(name),': from extract_dep_list',& - & me,length_dl(0),index_in(1), ':',dep_list(:length_dl(me),me) - ! ...now process root contains dependence list of all processes... - if (debug_level >= psb_debug_inner_) & - & write(debug_unit,*) me,' ',trim(name),': root sorting dep list' - if (do_timings) call psb_toc(idx_phase1) - if (do_timings) call psb_tic(idx_phase2) - - call psi_dl_check(dep_list,dl_lda,np,length_dl) - - ! ....now i can sort dependency lists. - call psi_sort_dl(dep_list,length_dl,np,info) - if(info /= psb_success_) then - call psb_errpush(psb_err_from_subroutine_,name,a_err='psi_sort_dl') - goto 9999 - end if - if (do_timings) call psb_toc(idx_phase2) end if - if (do_timings) call psb_tic(idx_phase3) + if (do_timings) call psb_tic(idx_phase3) if(debug_level >= psb_debug_inner_)& - & write(debug_unit,*) me,' ',trim(name),': calling psi_desc_index' + & write(debug_unit,*) me,' ',trim(name),': calling psi_desc_index',ldl,':',loc_dl(1:ldl) ! Do the actual format conversion. - call psi_desc_index(desc_a,index_in,dep_list(1:,me),& - & length_dl(me),nsnd,nrcv, index_out,info) + call psi_desc_index(desc_a,index_in,loc_dl,ldl,nsnd,nrcv,index_out,info) if(debug_level >= psb_debug_inner_) & & write(debug_unit,*) me,' ',trim(name),': out of psi_desc_index',& & size(index_out) - nxch = length_dl(me) + nxch = ldl if(info /= psb_success_) then call psb_errpush(psb_err_from_subroutine_,name,a_err='psi_desc_index') goto 9999 end if if (do_timings) call psb_toc(idx_phase3) - deallocate(dep_list,length_dl) + if (allocated(dep_list)) deallocate(dep_list,stat=info) + if ((info==0).and.allocated(length_dl)) deallocate(length_dl,stat=info) + if (info /= 0) then + info = psb_err_alloc_dealloc_ + goto 9999 + end if if(debug_level >= psb_debug_inner_) & & write(debug_unit,*) me,' ',trim(name),': done' @@ -193,4 +220,14 @@ subroutine psi_i_crea_index(desc_a,index_in,index_out,nxch,nsnd,nrcv,info) 9999 call psb_error_handler(ictxt,err_act) return +contains + function choose_sorting(dlmax,dlavg,np) result(val) + implicit none + integer(psb_ipk_), intent(in) :: dlmax,dlavg,np + logical :: val + + val = .not.(((dlmax>(27*4)).or.((dlavg>=(27*2)).and.(np>128)))) + val = .false. + end function choose_sorting + end subroutine psi_i_crea_index diff --git a/base/internals/psi_xtr_loc_dl.F90 b/base/internals/psi_xtr_loc_dl.F90 index 4941982a..c97cc7a5 100644 --- a/base/internals/psi_xtr_loc_dl.F90 +++ b/base/internals/psi_xtr_loc_dl.F90 @@ -29,8 +29,7 @@ ! POSSIBILITY OF SUCH DAMAGE. ! ! -subroutine psi_i_xtr_loc_dl(ictxt,is_bld,is_upd,desc_str,loc_dl,& - & ldl,max_ldl,mode,info) +subroutine psi_i_xtr_loc_dl(ictxt,is_bld,is_upd,desc_str,loc_dl,length_dl,info) ! internal routine ! == = ============= @@ -104,20 +103,11 @@ subroutine psi_i_xtr_loc_dl(ictxt,is_bld,is_upd,desc_str,loc_dl,& ! np integer (global input) ! number of grid process. ! - ! mode integer (global input) - ! if mode =0 then will be inserted also duplicate element in - ! a same dependence list - ! if mode =1 then not will be inserted duplicate element in - ! a same dependence list ! output ! == = == - ! only for root (pid=0) process: - ! dep_list integer array(dl_lda,0:np) - ! dependence list dep_list(*,i) is the list of process identifiers to which process i - ! must communicate with. this list with its order is extracted from - ! desc_str list. - ! length_dl integer array(0:np) - ! length_dl(i) is the length of dep_list(*,i) list + ! loc_dl integer array(:) + ! dependence list of current process + ! use psi_mod, psb_protect_name => psi_i_xtr_loc_dl #ifdef MPI_MOD use mpi @@ -133,22 +123,19 @@ subroutine psi_i_xtr_loc_dl(ictxt,is_bld,is_upd,desc_str,loc_dl,& #endif ! ....scalar parameters... logical, intent(in) :: is_bld, is_upd - integer(psb_ipk_), intent(in) :: ictxt,mode - integer(psb_ipk_), intent(out) :: max_ldl, ldl - integer(psb_ipk_), intent(in) :: desc_str(*) - integer(psb_ipk_), allocatable, intent(out) :: loc_dl(:) + integer(psb_ipk_), intent(in) :: ictxt + integer(psb_ipk_), intent(in) :: desc_str(:) + integer(psb_ipk_), allocatable, intent(out) :: loc_dl(:), length_dl(:) integer(psb_ipk_), intent(out) :: info ! .....local arrays.... integer(psb_ipk_) :: int_err(5) - integer(psb_ipk_), allocatable :: itmp(:) ! .....local scalars... - integer(psb_ipk_) :: i,pointer_dep_list,proc,j,err_act + integer(psb_ipk_) :: i,pdl,proc,j,err_act, ldu integer(psb_ipk_) :: err integer(psb_ipk_) :: debug_level, debug_unit - integer(psb_ipk_), allocatable :: length_dl(:) integer(psb_mpk_) :: iictxt, icomm, me, np, minfo - logical, parameter :: dist_symm_list=.false., print_dl=.false., profile=.true. + logical, parameter :: dist_symm_list=.true., print_dl=.false., profile=.true. character name*20 name='psi_extrct_dl' @@ -159,20 +146,19 @@ subroutine psi_i_xtr_loc_dl(ictxt,is_bld,is_upd,desc_str,loc_dl,& info = psb_success_ call psb_info(iictxt,me,np) - allocate(itmp(2*np+1),length_dl(0:np),stat=info) + pdl = size(desc_str) + allocate(loc_dl(pdl+1),length_dl(0:np),stat=info) if (info /= psb_success_) then info=psb_err_alloc_dealloc_ goto 9999 end if - do i=0,np - length_dl(i) = 0 - itmp(i+1) = -1 - enddo - i=1 + if (debug_level >= psb_debug_inner_)& & write(debug_unit,*) me,' ',trim(name),': start ',info - pointer_dep_list=1 + loc_dl = -1 + i = 1 + pdl = 0 if (is_bld) then do while (desc_str(i) /= -1) if (debug_level >= psb_debug_inner_)& @@ -193,24 +179,8 @@ subroutine psi_i_xtr_loc_dl(ictxt,is_bld,is_upd,desc_str,loc_dl,& goto 9999 endif ! if((me == 1).and.(proc == 3))write(psb_err_unit,*)'found 3' - if (mode == 1) then - ! ...search if already exist proc - ! in itmp(*)... - j=1 - do while ((j < pointer_dep_list).and.& - & (itmp(j) /= proc)) - j=j+1 - enddo - - if (j == pointer_dep_list) then - ! ...if not found..... - itmp(pointer_dep_list)=proc - pointer_dep_list=pointer_dep_list+1 - endif - else if (mode == 0) then - itmp(pointer_dep_list)=proc - pointer_dep_list=pointer_dep_list+1 - endif + pdl=pdl+1 + loc_dl(pdl)=proc endif i=i+desc_str(i+1)+2 enddo @@ -227,159 +197,25 @@ subroutine psi_i_xtr_loc_dl(ictxt,is_bld,is_upd,desc_str,loc_dl,& proc=desc_str(i) ! ..if number of element to be exchanged !=0 - - if (mode == 1) then - ! ...search if already exist proc.... - j=1 - do while ((j < pointer_dep_list).and.& - & (itmp(j) /= proc)) - j=j+1 - enddo - if (j == pointer_dep_list) then - ! ...if not found..... - itmp(pointer_dep_list)=proc - pointer_dep_list=pointer_dep_list+1 - endif - else if (mode == 0) then - itmp(pointer_dep_list)=proc - pointer_dep_list=pointer_dep_list+1 - endif + pdl=pdl+1 + loc_dl(pdl)=proc endif i=i+desc_str(i+1)+2 enddo + else info = 2020 goto 9999 endif - - length_dl(me)=pointer_dep_list-1 - if ((profile).and.(me==0)) then - block - integer(psb_ipk_) :: dlmax, dlavg - dlmax = maxval(length_dl(:)) - dlavg = (sum(length_dl(:))+np-1)/np - if (dlmax>0) write(0,*) 'Dependency list : max:',dlmax,& - & ' avg:',dlavg, ((dlmax>np/3).or.((dlavg>=np/4).and.(np>128))) - - end block - end if - - if (dist_symm_list) then - call psb_realloc(length_dl(me),itmp,info) - call psi_symm_dep_list(itmp,ictxt,info) - dl_lda = max(size(itmp),1) - call psb_max(iictxt, dl_lda) - - if (debug_level >= psb_debug_inner_) & - & write(debug_unit,*) me,' ',trim(name),': Dep_list length ',length_dl(me),dl_lda - call psb_realloc(dl_lda,itmp,info) - ! dl_lda = min(np,2*dl_lda) - allocate(dep_list(dl_lda,0:np),stat=info) - if (info /= psb_success_) then - call psb_errpush(psb_err_from_subroutine_,name,a_err='Allocate') - goto 9999 - end if - - call psb_sum(iictxt,length_dl(0:np)) - icomm = psb_get_mpi_comm(iictxt) - call mpi_allgather(itmp,dl_lda,psb_mpi_ipk_,& - & dep_list,dl_lda,psb_mpi_ipk_,icomm,minfo) - info = minfo - if (info == 0) deallocate(itmp,stat=info) - if (info /= psb_success_) then - info=psb_err_alloc_dealloc_ - goto 9999 - endif - else - block - integer(psb_ipk_), allocatable :: list1(:,:), ldl2(:), list2(:,:) - integer(psb_ipk_) :: i,j,ip,dlsym, ldu, mdl, l1, l2 - - dl_lda = max(length_dl(me),1) - call psb_max(iictxt, dl_lda) - if (debug_level >= psb_debug_inner_) & - & write(debug_unit,*) me,' ',trim(name),': Dep_list length ',length_dl(me),dl_lda - allocate(dep_list(dl_lda,0:np),stat=info) - if (info /= psb_success_) then - call psb_errpush(psb_err_from_subroutine_,name,a_err='Allocate') - goto 9999 - end if - call psb_sum(iictxt,length_dl(0:np)) - icomm = psb_get_mpi_comm(iictxt) - call mpi_allgather(itmp,dl_lda,psb_mpi_ipk_,& - & dep_list,dl_lda,psb_mpi_ipk_,icomm,minfo) - info = minfo - if (info /= psb_success_) then - info=psb_err_alloc_dealloc_ - goto 9999 - endif - allocate(ldl2(0:np),stat=info) - ldl2 = 0 - do j=0, np-1 - do i=1,length_dl(j) - ip = dep_list(i,j) - ldl2(ip) = ldl2(ip) + 1 - end do - end do - dlsym = maxval(ldl2) - allocate(list2(dlsym,0:np),stat=info) - call move_alloc(dep_list,list1) - if (info /= psb_success_) then - call psb_errpush(psb_err_from_subroutine_,name,a_err='Allocate') - goto 9999 - end if - ldl2 = 0 - do j=0, np-1 - do i=1,length_dl(j) - ip = list1(i,j) - ldl2(ip) = ldl2(ip) + 1 - list2(ldl2(ip),ip) = j - end do - end do - mdl = 0 - do j = 0, np-1 - l1 = length_dl(j) - l2 = ldl2(j) - itmp(1:l1) = list1(1:l1,j) - itmp(l1+1:l1+l2) = list2(1:l2,j) - ldu = l1 + l2 - !if (me == 0) write(0,*) 'Iter ',j,':',l1,l2,':',itmp(1:l1),':',itmp(l1+1:l1+l2) - call psb_msort_unique(itmp(1:l1+l2),ldu) - mdl = max(mdl, ldu) - !if (me == 0) write(0,*) 'Iter ',j,':',ldu,':',itmp(1:ldu) - end do - dl_lda = mdl - allocate(dep_list(dl_lda,0:np),stat=info) - dep_list = -1 - if (info /= psb_success_) then - call psb_errpush(psb_err_from_subroutine_,name,a_err='Allocate') - goto 9999 - end if - do j = 0, np-1 - l1 = length_dl(j) - l2 = ldl2(j) - itmp(1:l1) = list1(1:l1,j) - itmp(l1+1:l1+l2) = list2(1:l2,j) - ldu = l1 + l2 - call psb_msort_unique(itmp(1:l1+l2),ldu) - length_dl(j) = ldu - dep_list(1:ldu,j) = itmp(1:ldu) - end do - - end block - endif - if (print_dl) then - if (me == 0) then - write(0,*) ' Dep_list ' - do i=0,np-1 - j = length_dl(i) - write(0,*) 'Proc ',i,':',dep_list(1:j,i) - end do - flush(0) - end if - call psb_barrier(ictxt) - end if - + call psb_msort_unique(loc_dl(1:pdl),ldu) + pdl = ldu + call psb_realloc(pdl,loc_dl,info) + call psi_symm_dep_list(loc_dl,ictxt,info) + pdl = size(loc_dl) + length_dl = 0 + length_dl(me) = pdl + call psb_sum(ictxt,length_dl) + call psb_erractionrestore(err_act) return diff --git a/base/modules/psi_i_mod.F90 b/base/modules/psi_i_mod.F90 index eff2a05f..9239617d 100644 --- a/base/modules/psi_i_mod.F90 +++ b/base/modules/psi_i_mod.F90 @@ -126,16 +126,28 @@ module psi_i_mod integer(psb_ipk_), intent(out) :: info end subroutine psi_i_extract_dep_list end interface psi_extract_dep_list + + interface psi_bld_glb_dep_list + subroutine psi_i_bld_glb_dep_list(ictxt,loc_dl,& + & length_dl,dep_list,dl_lda,info) + import + implicit none + integer(psb_ipk_), intent(in) :: ictxt + integer(psb_ipk_), intent(out) :: dl_lda + integer(psb_ipk_), intent(in) :: loc_dl(:), length_dl(:) + integer(psb_ipk_), allocatable, intent(out) :: dep_list(:,:) + integer(psb_ipk_), intent(out) :: info + end subroutine psi_i_bld_glb_dep_list + end interface psi_bld_glb_dep_list interface psi_extract_loc_dl subroutine psi_i_xtr_loc_dl(ictxt,is_bld,is_upd,desc_str,loc_dl,& - & ldl,max_ldl,mode,info) + & length_dl,info) import logical, intent(in) :: is_bld, is_upd - integer(psb_ipk_), intent(in) :: ictxt,mode - integer(psb_ipk_), intent(out) :: max_ldl, ldl - integer(psb_ipk_), intent(in) :: desc_str(*) - integer(psb_ipk_), allocatable, intent(out) :: loc_dl(:) + integer(psb_ipk_), intent(in) :: ictxt + integer(psb_ipk_), intent(in) :: desc_str(:) + integer(psb_ipk_), allocatable, intent(out) :: loc_dl(:), length_dl(:) integer(psb_ipk_), intent(out) :: info end subroutine psi_i_xtr_loc_dl end interface psi_extract_loc_dl From d2f8d87204b91fbfb8f94d8f6762268b86ba7492 Mon Sep 17 00:00:00 2001 From: Salvatore Filippone Date: Thu, 21 May 2020 13:54:18 +0200 Subject: [PATCH 3/3] Defined a threshold for enacting simplified exchange list build --- base/internals/psi_crea_index.f90 | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/base/internals/psi_crea_index.f90 b/base/internals/psi_crea_index.f90 index eaa9f636..d40693ed 100644 --- a/base/internals/psi_crea_index.f90 +++ b/base/internals/psi_crea_index.f90 @@ -151,7 +151,7 @@ subroutine psi_i_crea_index(desc_a,index_in,index_out,nxch,nsnd,nrcv,info) dlmax = maxval(length_dl(:)) dlavg = (sum(length_dl(:))+np-1)/np !!$ if ((dlmax>0).and.(me==0)) write(0,*) 'Dependency list : max:',dlmax,& -!!$ & ' avg:',dlavg, ((dlmax>np/3).or.((dlavg>=np/4).and.(np>128))) +!!$ & ' avg:',dlavg, choose_sorting(dlmax,dlavg,np) if (choose_sorting(dlmax,dlavg,np)) then @@ -226,8 +226,7 @@ contains integer(psb_ipk_), intent(in) :: dlmax,dlavg,np logical :: val - val = .not.(((dlmax>(27*4)).or.((dlavg>=(27*2)).and.(np>128)))) - val = .false. + val = .not.(((dlmax>(26*4)).or.((dlavg>=(26*2)).and.(np>=128)))) end function choose_sorting end subroutine psi_i_crea_index