From d29131e59cb8d779b46e3d50e8960cb067b6c1a6 Mon Sep 17 00:00:00 2001 From: william-dawson Date: Thu, 20 Jun 2024 00:35:49 +0900 Subject: [PATCH] Feature for Scaling the Matrix by a Diagonal matrix (#233) * Local version of scale with a diagonal * Remove unused parameter * Fix API * First attempt at parallel, have to test somewhere else... * Arg fix * Test setup, now get to fixing it * More fixing for the test * Fix partitioning * Gather strategy seems to work * forgot a file * Seems to be working * Fix case of failing to upcast * Fix test conditions to handle the case an edge case * Try to simplify the code * Move the approach into the eigensolvers module * lint --- Source/C/PSMatrix_c.h | 2 + Source/C/SMatrix_c.h | 2 + Source/CPlusPlus/PSMatrix.cc | 10 +++ Source/CPlusPlus/PSMatrix.h | 8 ++- Source/CPlusPlus/SMatrix.cc | 10 +++ Source/CPlusPlus/SMatrix.h | 8 +++ Source/Fortran/EigenSolversModule.F90 | 28 ++++---- Source/Fortran/FermiOperatorModule.F90 | 55 +++++++++------- Source/Fortran/PSMatrixAlgebraModule.F90 | 42 +++++++++++- Source/Fortran/PSMatrixModule.F90 | 33 +++++++++- Source/Fortran/SMatrixAlgebraModule.F90 | 29 ++++++++ Source/Fortran/TripletListModule.F90 | 64 ++++++++++++++++++ .../ScaleDiagonal.f90 | 24 +++++++ .../distributed_includes/FilterMatrix.f90 | 12 +--- .../GatherMatrixTripletList.f90 | 3 + .../Fortran/sparse_includes/DiagonalScale.f90 | 9 +++ .../triplet_includes/GatherTripletList.f90 | 66 +++++++++++++++++++ Source/Wrapper/PSMatrixAlgebraModule_wrp.F90 | 32 ++++++++- Source/Wrapper/SMatrixAlgebraModule_wrp.F90 | 30 +++++++++ UnitTests/test_matrix.py | 27 ++++++++ UnitTests/test_psmatrixalgebra.py | 35 ++++++++++ 21 files changed, 478 insertions(+), 51 deletions(-) create mode 100644 Source/Fortran/distributed_algebra_includes/ScaleDiagonal.f90 create mode 100644 Source/Fortran/distributed_includes/GatherMatrixTripletList.f90 create mode 100644 Source/Fortran/sparse_includes/DiagonalScale.f90 create mode 100644 Source/Fortran/triplet_includes/GatherTripletList.f90 diff --git a/Source/C/PSMatrix_c.h b/Source/C/PSMatrix_c.h index b351b217..a211fe73 100644 --- a/Source/C/PSMatrix_c.h +++ b/Source/C/PSMatrix_c.h @@ -62,5 +62,7 @@ void ScaleMatrix_ps_wrp(int *ih_this, const double *constant); double MatrixNorm_ps_wrp(const int *ih_this); void MatrixTrace_ps_wrp(const int *ih_this, double *trace_val); int IsIdentity_ps_wrp(const int *ih_this); +void MatrixDiagonalScale_psr_wrp(int *ih_mat, const int *ih_tlist); +void MatrixDiagonalScale_psc_wrp(int *ih_mat, const int *ih_tlist); #endif diff --git a/Source/C/SMatrix_c.h b/Source/C/SMatrix_c.h index dcdb4fdc..9307c931 100644 --- a/Source/C/SMatrix_c.h +++ b/Source/C/SMatrix_c.h @@ -34,6 +34,7 @@ void PrintMatrix_lsr_wrp(const int *ih_this); void PrintMatrixF_lsr_wrp(const int *ih_this, const char *file_name, const int *name_size); void MatrixToTripletList_lsr_wrp(const int *ih_this, int *ih_triplet_list); +void MatrixDiagonalScale_lsr_wrp(int *ih_mat, const int *ih_tlist); void ConstructMatrixFromFile_lsc_wrp(int *ih_this, const char *file_name, const int *name_size); @@ -70,5 +71,6 @@ void PrintMatrix_lsc_wrp(const int *ih_this); void PrintMatrixF_lsc_wrp(const int *ih_this, const char *file_name, const int *name_size); void MatrixToTripletList_lsc_wrp(const int *ih_this, int *ih_triplet_list); +void MatrixDiagonalScale_lsc_wrp(int *ih_mat, const int *ih_tlist); #endif diff --git a/Source/CPlusPlus/PSMatrix.cc b/Source/CPlusPlus/PSMatrix.cc index c2c296bd..05f0d593 100644 --- a/Source/CPlusPlus/PSMatrix.cc +++ b/Source/CPlusPlus/PSMatrix.cc @@ -217,6 +217,16 @@ void Matrix_ps::Scale(double constant) { ScaleMatrix_ps_wrp(ih_this, &constant); } +////////////////////////////////////////////////////////////////////////////// +void Matrix_ps::DiagonalScale(const TripletList_r &tlist) { + MatrixDiagonalScale_psr_wrp(ih_this, tlist.ih_this); +} + +////////////////////////////////////////////////////////////////////////////// +void Matrix_ps::DiagonalScale(const TripletList_c &tlist) { + MatrixDiagonalScale_psc_wrp(ih_this, tlist.ih_this); +} + ////////////////////////////////////////////////////////////////////////////// double Matrix_ps::Norm() const { return MatrixNorm_ps_wrp(ih_this); } diff --git a/Source/CPlusPlus/PSMatrix.h b/Source/CPlusPlus/PSMatrix.h index eb2bd2ab..8aba171a 100644 --- a/Source/CPlusPlus/PSMatrix.h +++ b/Source/CPlusPlus/PSMatrix.h @@ -160,13 +160,19 @@ class Matrix_ps { void Gemm(const Matrix_ps &matA, const Matrix_ps &matB, PMatrixMemoryPool &memory_pool, double alpha = 1.0, double beta = 0.0, double threshold = 0.0); - //! scale the matrix by a constatn. + //! scale the matrix by a constant. //! constant the value to scale by. void Scale(double constant); //! compute the norm of a matrix. double Norm() const; //! compute the trace of a matrix. double Trace() const; + //!\param tlist the triplet list. + //!\param threshold for flushing small values. + void DiagonalScale(const NTPoly::TripletList_r &tlist); + //!\param tlist the triplet list. + //!\param threshold for flushing small values. + void DiagonalScale(const NTPoly::TripletList_c &tlist); public: //! Destructor. diff --git a/Source/CPlusPlus/SMatrix.cc b/Source/CPlusPlus/SMatrix.cc index 521d6546..4feda63b 100644 --- a/Source/CPlusPlus/SMatrix.cc +++ b/Source/CPlusPlus/SMatrix.cc @@ -175,6 +175,16 @@ void Matrix_lsc::Gemm(const Matrix_lsc &matA, const Matrix_lsc &matB, memory_pool.ih_this); } +//////////////////////////////////////////////////////////////////////////////// +void Matrix_lsr::DiagonalScale(const TripletList_r &tlist) { + MatrixDiagonalScale_lsr_wrp(ih_this, tlist.ih_this); +} + +//////////////////////////////////////////////////////////////////////////////// +void Matrix_lsc::DiagonalScale(const TripletList_c &tlist) { + MatrixDiagonalScale_lsc_wrp(ih_this, tlist.ih_this); +} + //////////////////////////////////////////////////////////////////////////////// void Matrix_lsr::Transpose(const Matrix_lsr &matA) { TransposeMatrix_lsr_wrp(matA.ih_this, ih_this); diff --git a/Source/CPlusPlus/SMatrix.h b/Source/CPlusPlus/SMatrix.h index fd5079c1..0accdb88 100644 --- a/Source/CPlusPlus/SMatrix.h +++ b/Source/CPlusPlus/SMatrix.h @@ -76,6 +76,10 @@ class Matrix_lsr { void Gemm(const NTPoly::Matrix_lsr &matA, const NTPoly::Matrix_lsr &matB, bool isATransposed, bool isBTransposed, double alpha, double beta, double threshold, NTPoly::MatrixMemoryPool_r &memory_pool); + //! Scale a matrix using a diagonal matrix (triplet list form). + //!\param tlist the triplet list. + //!\param threshold for flushing small values. + void DiagonalScale(const NTPoly::TripletList_r &tlist); public: //! Transpose a sparse matrix. @@ -170,6 +174,10 @@ class Matrix_lsc { void Gemm(const NTPoly::Matrix_lsc &matA, const NTPoly::Matrix_lsc &matB, bool isATransposed, bool isBTransposed, double alpha, double beta, double threshold, NTPoly::MatrixMemoryPool_c &memory_pool); + //! Scale a matrix using a diagonal matrix (triplet list form). + //!\param tlist the triplet list. + //!\param threshold for flushing small values. + void DiagonalScale(const NTPoly::TripletList_c &tlist); public: //! Transpose a sparse matrix. diff --git a/Source/Fortran/EigenSolversModule.F90 b/Source/Fortran/EigenSolversModule.F90 index 103e751c..fc820627 100644 --- a/Source/Fortran/EigenSolversModule.F90 +++ b/Source/Fortran/EigenSolversModule.F90 @@ -13,10 +13,10 @@ MODULE EigenSolversModule USE PMatrixMemoryPoolModule, ONLY : MatrixMemoryPool_p, & & DestructMatrixMemoryPool USE PSMatrixAlgebraModule, ONLY : IncrementMatrix, MatrixMultiply, & - & ScaleMatrix + & ScaleMatrix, MatrixDiagonalScale USE PSMatrixModule, ONLY : Matrix_ps, GatherMatrixToProcess, & & FillMatrixFromTripletList, ConstructEmptyMatrix, ConvertMatrixToReal, & - & DestructMatrix, CopyMatrix, GetMatrixTripletList, TransposeMatrix, & + & DestructMatrix, CopyMatrix, GatherMatrixTripletList, TransposeMatrix, & & ConjugateMatrix, FillMatrixIdentity, WriteMatrixToMatrixMarket USE SolverParametersModule, ONLY : SolverParameters_t, PrintParameters, & & DestructSolverParameters, ConstructSolverParameters, & @@ -24,7 +24,7 @@ MODULE EigenSolversModule USE SMatrixModule, ONLY : Matrix_lsr, Matrix_lsc, MatrixToTripletList, & & DestructMatrix USE TripletListModule, ONLY : TripletList_r, TripletList_c, & - & ConstructTripletList, DestructTripletList + & ConstructTripletList, CopyTripletList, DestructTripletList USE NTMPIModule IMPLICIT NONE PRIVATE @@ -107,6 +107,7 @@ END FUNCTION func !! Local Variables TYPE(Matrix_ps) :: vecs, vecsT, vals, temp TYPE(TripletList_r) :: tlist + TYPE(TripletList_c) :: tlist_c INTEGER :: II !! Optional Parameters @@ -121,26 +122,29 @@ END FUNCTION func & eigenvectors_in = vecs) !! Convert to a triplet list, map the triplet list, fill. - CALL GetMatrixTripletList(vals, tlist) + CALL GatherMatrixTripletList(vals, tlist) CALL DestructMatrix(vals) DO II = 1, tlist%CurrentSize tlist%DATA(II)%point_value = func(tlist%DATA(II)%point_value) END DO - !! Fill - CALL ConstructEmptyMatrix(ResultMat, this) - CALL FillMatrixFromTripletList(ResultMat, tlist, preduplicated_in = .TRUE.) - CALL DestructTripletList(tlist) - !! Multiply Back Together - CALL MatrixMultiply(vecs, ResultMat, temp, threshold_in = params%threshold) CALL TransposeMatrix(vecs, vecsT) - CALL DestructMatrix(vecs) CALL ConjugateMatrix(vecsT) - CALL MatrixMultiply(temp, vecsT, ResultMat, threshold_in = params%threshold) + IF (this%is_complex) THEN + CALL CopyTripletList(tlist, tlist_c) + CALL MatrixDiagonalScale(vecs, tlist_c) + ELSE + CALL MatrixDiagonalScale(vecs, tlist) + END IF + CALL MatrixMultiply(vecs, vecsT, ResultMat, threshold_in = params%threshold) !! Cleanup + CALL DestructMatrix(vecs) + CALL DestructMatrix(vecsT) CALL DestructMatrix(temp) + CALL DestructTripletList(tlist) + CALL DestructTripletList(tlist_c) CALL DestructSolverParameters(params) END SUBROUTINE DenseMatrixFunction diff --git a/Source/Fortran/FermiOperatorModule.F90 b/Source/Fortran/FermiOperatorModule.F90 index b43d6f0b..fafeb1c3 100644 --- a/Source/Fortran/FermiOperatorModule.F90 +++ b/Source/Fortran/FermiOperatorModule.F90 @@ -1,23 +1,26 @@ !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! !> A Module For Computing The Density Matrix Using the Fermi Operator Expansion MODULE FermiOperatorModule - USE DataTypesModule, ONLY : NTREAL, MPINTREAL + USE DataTypesModule, ONLY : NTREAL, MPINTREAL, NTCOMPLEX USE EigenSolversModule, ONLY : EigenDecomposition USE LoadBalancerModule, ONLY : PermuteMatrix, UndoPermuteMatrix USE LoggingModule, ONLY : WriteElement, WriteHeader, & & EnterSubLog, ExitSubLog, WriteListElement, WriteComment USE PSMatrixAlgebraModule, ONLY : MatrixMultiply, SimilarityTransform, & - & IncrementMatrix, MatrixNorm, ScaleMatrix, DotMatrix, MatrixTrace + & IncrementMatrix, MatrixNorm, ScaleMatrix, DotMatrix, MatrixTrace, & + & MatrixDiagonalScale USE PSMatrixModule, ONLY : Matrix_ps, ConstructEmptyMatrix, & & FillMatrixFromTripletList, GetMatrixTripletList, & - & TransposeMatrix, ConjugateMatrix, DestructMatrix, & - & FillMatrixIdentity, PrintMatrixInformation, CopyMatrix, GetMatrixSize + & TransposeMatrix, ConjugateMatrix, DestructMatrix, FilterMatrix, & + & FillMatrixIdentity, PrintMatrixInformation, CopyMatrix, & + & GatherMatrixTripletList, GetMatrixSize USE PMatrixMemoryPoolModule, ONLY : MatrixMemoryPool_p, & & DestructMatrixMemoryPool USE SolverParametersModule, ONLY : SolverParameters_t, & & PrintParameters, DestructSolverParameters, CopySolverParameters, & & ConstructSolverParameters - USE TripletListModule, ONLY : TripletList_r, DestructTripletList + USE TripletListModule, ONLY : TripletList_r, TripletList_c, & + & DestructTripletList, CopyTripletList USE NTMPIModule IMPLICIT NONE PRIVATE @@ -55,13 +58,13 @@ SUBROUTINE ComputeDenseFOE(H, ISQ, trace, K, inv_temp_in, & TYPE(Matrix_ps) :: vecs, vecsT, vals, Temp TYPE(MatrixMemoryPool_p) :: pool TYPE(TripletList_r) :: tlist + TYPE(TripletList_c) :: tlist_c REAL(NTREAL) :: chemical_potential, energy_value REAL(NTREAL), DIMENSION(:), ALLOCATABLE :: eigs, occ REAL(NTREAL) :: sval, sv, occ_temp REAL(NTREAL) :: left, right, homo, lumo INTEGER :: num_eigs INTEGER :: II, JJ - INTEGER :: ierr !! Optional Parameters IF (PRESENT(solver_parameters_in)) THEN @@ -100,15 +103,15 @@ SUBROUTINE ComputeDenseFOE(H, ISQ, trace, K, inv_temp_in, & & eigenvectors_in = vecs, solver_parameters_in = params) !! Gather the eigenvalues on to every process - CALL GetMatrixTripletList(vals, tlist) + CALL GatherMatrixTripletList(vals, tlist) + + !! Put them in an array for simplicity num_eigs = H%actual_matrix_dimension ALLOCATE(eigs(num_eigs)) eigs = 0 DO II = 1, tlist%CurrentSize - eigs(tlist%DATA(II)%index_column) = tlist%DATA(II)%point_value + eigs(II) = tlist%DATA(II)%point_value END DO - CALL MPI_ALLREDUCE(MPI_IN_PLACE, eigs, num_eigs, MPINTREAL, & - & MPI_SUM, H%process_grid%within_slice_comm, ierr) !! Compute MU By Bisection IF (do_smearing) THEN @@ -116,7 +119,7 @@ SUBROUTINE ComputeDenseFOE(H, ISQ, trace, K, inv_temp_in, & left = MINVAL(eigs) right = MAXVAL(eigs) DO JJ = 1, 10*params%max_iterations - chemical_potential = left + (right - left)/2 + chemical_potential = left + (right - left) / 2 DO II = 1, num_eigs sval = eigs(II) - chemical_potential ! occ(II) = 0.5_NTREAL * (1.0_NTREAL - ERF(inv_temp * sval)) @@ -148,7 +151,7 @@ SUBROUTINE ComputeDenseFOE(H, ISQ, trace, K, inv_temp_in, & CALL ExitSubLog END IF - !! Map + !! Map - note that we store the square root of the occupation numbers energy_value = 0.0_NTREAL DO II = 1, tlist%CurrentSize IF (.NOT. do_smearing) THEN @@ -159,7 +162,7 @@ SUBROUTINE ComputeDenseFOE(H, ISQ, trace, K, inv_temp_in, & occ_temp = trace - FLOOR(trace) energy_value = energy_value + & & occ_temp * tlist%DATA(II)%point_value - tlist%DATA(II)%point_value = occ_temp + tlist%DATA(II)%point_value = SQRT(occ_temp) ELSE tlist%DATA(II)%point_value = 0.0_NTREAL ENDIF @@ -167,22 +170,29 @@ SUBROUTINE ComputeDenseFOE(H, ISQ, trace, K, inv_temp_in, & sval = tlist%DATA(II)%point_value - chemical_potential ! occ_temp = 0.5_NTREAL * (1.0_NTREAL - ERF(inv_temp * sval)) occ_temp = 1.0_NTREAL / (1.0_NTREAL + EXP(inv_temp * sval)) - energy_value = energy_value + occ_temp * tlist%DATA(II)%point_value - tlist%DATA(II)%point_value = occ_temp + energy_value = energy_value + & + & occ_temp * tlist%DATA(II)%point_value + IF (occ_temp .LT. 0) THEN ! for safety + tlist%DATA(II)%point_value = 0 + ELSE + tlist%DATA(II)%point_value = SQRT(occ_temp) + END IF END IF END DO - CALL MPI_ALLREDUCE(MPI_IN_PLACE, energy_value, 1, MPINTREAL, MPI_SUM, & - & H%process_grid%within_slice_comm, ierr) - !! Fill - CALL ConstructEmptyMatrix(vals, H) - CALL FillMatrixFromTripletList(vals, tlist, preduplicated_in = .TRUE.) + !! Scale the eigenvectors + IF (vecs%is_complex) THEN + CALL CopyTripletList(tlist, tlist_c) + CALL MatrixDiagonalScale(vecs, tlist_c) + ELSE + CALL MatrixDiagonalScale(vecs, tlist) + END IF + CALL FilterMatrix(vecs, params%threshold) !! Multiply Back Together - CALL MatrixMultiply(vecs, vals, temp, threshold_in = params%threshold) CALL TransposeMatrix(vecs, vecsT) CALL ConjugateMatrix(vecsT) - CALL MatrixMultiply(temp, vecsT, WD, & + CALL MatrixMultiply(vecs, vecsT, WD, & & threshold_in = params%threshold) !! Compute the density matrix in the non-orthogonalized basis @@ -208,6 +218,7 @@ SUBROUTINE ComputeDenseFOE(H, ISQ, trace, K, inv_temp_in, & CALL DestructMatrix(vals) CALL DestructMatrix(temp) CALL DestructTripletList(tlist) + CALL DestructTripletList(tlist_c) CALL DestructMatrixMemoryPool(pool) IF (ALLOCATED(occ)) THEN DEALLOCATE(occ) diff --git a/Source/Fortran/PSMatrixAlgebraModule.F90 b/Source/Fortran/PSMatrixAlgebraModule.F90 index c2b13f86..6ecea2da 100644 --- a/Source/Fortran/PSMatrixAlgebraModule.F90 +++ b/Source/Fortran/PSMatrixAlgebraModule.F90 @@ -13,13 +13,16 @@ MODULE PSMatrixAlgebraModule & ConstructMatrixMemoryPool USE PSMatrixModule, ONLY : Matrix_ps, ConstructEmptyMatrix, CopyMatrix, & & DestructMatrix, ConvertMatrixToComplex, ConjugateMatrix, & - & MergeMatrixLocalBlocks, IsIdentity + & MergeMatrixLocalBlocks, IsIdentity, SplitMatrixToLocalBlocks USE SMatrixAlgebraModule, ONLY : MatrixMultiply, MatrixGrandSum, & & PairwiseMultiplyMatrix, IncrementMatrix, ScaleMatrix, & - & MatrixColumnNorm + & MatrixColumnNorm, MatrixDiagonalScale USE SMatrixModule, ONLY : Matrix_lsr, Matrix_lsc, DestructMatrix, CopyMatrix,& & TransposeMatrix, ComposeMatrixColumns, MatrixToTripletList - USE TripletListModule, ONLY : TripletList_r, TripletList_c + USE TripletListModule, ONLY : TripletList_r, TripletList_c, & + & ConstructTripletList, AppendToTripletList, DestructTripletList, & + & GetTripletAt + USE TripletModule, ONLY : Triplet_r, Triplet_c USE NTMPIModule IMPLICIT NONE PRIVATE @@ -34,6 +37,7 @@ MODULE PSMatrixAlgebraModule PUBLIC :: ScaleMatrix PUBLIC :: MatrixTrace PUBLIC :: SimilarityTransform + PUBLIC :: MatrixDiagonalScale !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! INTERFACE MatrixSigma MODULE PROCEDURE MatrixSigma_ps @@ -62,6 +66,10 @@ MODULE PSMatrixAlgebraModule MODULE PROCEDURE ScaleMatrix_psr MODULE PROCEDURE ScaleMatrix_psc END INTERFACE ScaleMatrix + INTERFACE MatrixDiagonalScale + MODULE PROCEDURE MatrixDiagonalScale_psr + MODULE PROCEDURE MatrixDiagonalScale_psc + END INTERFACE MatrixDiagonalScale INTERFACE MatrixTrace MODULE PROCEDURE MatrixTrace_psr END INTERFACE MatrixTrace @@ -491,6 +499,34 @@ RECURSIVE SUBROUTINE ScaleMatrix_psc(this, constant) END IF END SUBROUTINE ScaleMatrix_psc +!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !> Will scale a distributed sparse matrix by a constant. + SUBROUTINE MatrixDiagonalScale_psr(this, tlist) + !> Matrix to scale. + TYPE(Matrix_ps), INTENT(INOUT) :: this + !> A constant scale factor. + TYPE(TripletList_r), INTENT(IN) :: tlist + !! Local Data + TYPE(Matrix_lsr) :: lmat + TYPE(TripletList_r) :: filtered + TYPE(Triplet_r) :: trip + +#include "distributed_algebra_includes/ScaleDiagonal.f90" + END SUBROUTINE MatrixDiagonalScale_psr +!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !> Will scale a distributed sparse matrix by a constant. + RECURSIVE SUBROUTINE MatrixDiagonalScale_psc(this, tlist) + !> Matrix to scale. + TYPE(Matrix_ps), INTENT(INOUT) :: this + !> A constant scale factor. + TYPE(TripletList_c), INTENT(IN) :: tlist + !! Local Data + TYPE(Matrix_lsc) :: lmat + TYPE(TripletList_c) :: filtered + TYPE(Triplet_c) :: trip + +#include "distributed_algebra_includes/ScaleDiagonal.f90" + END SUBROUTINE MatrixDiagonalScale_psc !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! !> Compute the trace of the matrix. SUBROUTINE MatrixTrace_psr(this, trace_value) diff --git a/Source/Fortran/PSMatrixModule.F90 b/Source/Fortran/PSMatrixModule.F90 index 19812a28..75f60b1e 100644 --- a/Source/Fortran/PSMatrixModule.F90 +++ b/Source/Fortran/PSMatrixModule.F90 @@ -87,6 +87,7 @@ MODULE PSMatrixModule PUBLIC :: CommSplitMatrix PUBLIC :: ResizeMatrix PUBLIC :: GatherMatrixToProcess + PUBLIC :: GatherMatrixTripletList PUBLIC :: IsIdentity !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! INTERFACE ConstructEmptyMatrix @@ -180,6 +181,10 @@ MODULE PSMatrixModule MODULE PROCEDURE GatherMatrixToProcess_psc_id MODULE PROCEDURE GatherMatrixToProcess_psc_all END INTERFACE GatherMatrixToProcess + INTERFACE GatherMatrixTripletList + MODULE PROCEDURE GatherMatrixTripletList_r + MODULE PROCEDURE GatherMatrixTripletList_c + END INTERFACE GatherMatrixTripletList CONTAINS!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! !> Construct an empty sparse, distributed, matrix. SUBROUTINE ConstructEmptyMatrix_ps(this, matrix_dim, process_grid_in, & @@ -1443,7 +1448,7 @@ SUBROUTINE TransposeMatrix_psr(AMat, TransMat) !! Local Variables TYPE(TripletList_r) :: tlist TYPE(TripletList_r) :: new_list - TYPE(Triplet_r) :: trip, trip_t + TYPE(Triplet_r) :: trip #include "distributed_includes/TransposeMatrix.f90" @@ -1458,7 +1463,7 @@ SUBROUTINE TransposeMatrix_psc(AMat, TransMat) !! Local Variables TYPE(TripletList_c) :: tlist TYPE(TripletList_c) :: new_list - TYPE(Triplet_c) :: trip, trip_t + TYPE(Triplet_c) :: trip #include "distributed_includes/TransposeMatrix.f90" @@ -1843,5 +1848,29 @@ FUNCTION IsIdentity_psc(this) RESULT(is_identity) #include "distributed_includes/IsIdentity.f90" #undef ISCOMPLEX END FUNCTION IsIdentity_psc +!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !> This gathers the entire matrix into a triplet list on all processes. + SUBROUTINE GatherMatrixTripletList_r(this, tlist) + !> The matrix to gather. + TYPE(Matrix_ps), INTENT(IN) :: this + !> The full matrix, stored in a local matrix. + TYPE(TripletList_r), INTENT(INOUT) :: tlist + !! Local Variables + TYPE(Matrix_lsr) :: lmat + +#include "distributed_includes/GatherMatrixTripletList.f90" + END SUBROUTINE GatherMatrixTripletList_r +!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !> This gathers the entire matrix into a triplet list on all processes. + SUBROUTINE GatherMatrixTripletList_c(this, tlist) + !> The matrix to gather. + TYPE(Matrix_ps), INTENT(IN) :: this + !> The full matrix, stored in a local matrix. + TYPE(TripletList_c), INTENT(INOUT) :: tlist + !! Local Variables + TYPE(Matrix_lsc) :: lmat + +#include "distributed_includes/GatherMatrixTripletList.f90" + END SUBROUTINE GatherMatrixTripletList_c !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! END MODULE PSMatrixModule diff --git a/Source/Fortran/SMatrixAlgebraModule.F90 b/Source/Fortran/SMatrixAlgebraModule.F90 index cd790e76..ce6415c4 100644 --- a/Source/Fortran/SMatrixAlgebraModule.F90 +++ b/Source/Fortran/SMatrixAlgebraModule.F90 @@ -25,6 +25,7 @@ MODULE SMatrixAlgebraModule PUBLIC :: MatrixColumnNorm PUBLIC :: MatrixNorm PUBLIC :: MatrixGrandSum + PUBLIC :: MatrixDiagonalScale !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! INTERFACE ScaleMatrix MODULE PROCEDURE ScaleMatrix_lsr @@ -75,6 +76,10 @@ MODULE SMatrixAlgebraModule MODULE PROCEDURE DenseBranch_lsr MODULE PROCEDURE DenseBranch_lsc END INTERFACE DenseBranch + INTERFACE MatrixDiagonalScale + MODULE PROCEDURE MatrixDiagonalScale_lsr + MODULE PROCEDURE MatrixDiagonalScale_lsc + END INTERFACE MatrixDiagonalScale CONTAINS!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! !> Will scale a sparse matrix by a constant. PURE SUBROUTINE ScaleMatrix_lsr(matA,constant) @@ -526,5 +531,29 @@ PURE SUBROUTINE PruneList_lsc(memorypool,alpha,threshold, & #include "sparse_includes/PruneList.f90" END SUBROUTINE PruneList_lsc +!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !> Scale a matrix using a diagonal matrix (triplet list form). + SUBROUTINE MatrixDiagonalScale_lsr(mat, tlist) + !> The matrix to scale. + TYPE(Matrix_lsr), INTENT(INOUT) :: mat + !> The diagonal matrix. + TYPE(TripletList_r), INTENT(IN) :: tlist + !! Intermediate Data + REAL(NTREAL) :: val + +#include "sparse_includes/DiagonalScale.f90" + END SUBROUTINE MatrixDiagonalScale_lsr +!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !> Scale a matrix using a diagonal matrix (triplet list form). + SUBROUTINE MatrixDiagonalScale_lsc(mat, tlist) + !> The matrix to scale. + TYPE(Matrix_lsc), INTENT(INOUT) :: mat + !> The diagonal matrix. + TYPE(TripletList_c), INTENT(IN) :: tlist + !! Intermediate Data + COMPLEX(NTCOMPLEX) :: val + +#include "sparse_includes/DiagonalScale.f90" + END SUBROUTINE MatrixDiagonalScale_lsc !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! END MODULE SMatrixAlgebraModule diff --git a/Source/Fortran/TripletListModule.F90 b/Source/Fortran/TripletListModule.F90 index 29c0de68..5c967d2a 100644 --- a/Source/Fortran/TripletListModule.F90 +++ b/Source/Fortran/TripletListModule.F90 @@ -39,6 +39,7 @@ MODULE TripletListModule PUBLIC :: SymmetrizeTripletList PUBLIC :: GetTripletListSize PUBLIC :: RedistributeTripletLists + PUBLIC :: AllGatherTripletList PUBLIC :: ShiftTripletList PUBLIC :: ConvertTripletListType PUBLIC :: MergeTripletLists @@ -50,6 +51,7 @@ MODULE TripletListModule INTERFACE CopyTripletList MODULE PROCEDURE CopyTripletList_r MODULE PROCEDURE CopyTripletList_c + MODULE PROCEDURE CopyTripletList_rc END INTERFACE CopyTripletList INTERFACE DestructTripletList MODULE PROCEDURE DestructTripletList_r @@ -91,6 +93,10 @@ MODULE TripletListModule MODULE PROCEDURE RedistributeTripletLists_r MODULE PROCEDURE RedistributeTripletLists_c END INTERFACE RedistributeTripletLists + INTERFACE AllGatherTripletList + MODULE PROCEDURE AllGatherTripletList_r + MODULE PROCEDURE AllGatherTripletList_c + END INTERFACE AllGatherTripletList INTERFACE ShiftTripletList MODULE PROCEDURE ShiftTripletList_r MODULE PROCEDURE ShiftTripletList_c @@ -144,6 +150,7 @@ PURE SUBROUTINE DestructTripletList_c(this) END SUBROUTINE DestructTripletList_c !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !> Copy a triplet list (real). SUBROUTINE CopyTripletList_r(tripA, tripB) !> The triplet list to copy. TYPE(TripletList_r), INTENT(IN) :: tripA @@ -153,6 +160,7 @@ SUBROUTINE CopyTripletList_r(tripA, tripB) #include "triplet_includes/CopyTripletList.f90" END SUBROUTINE CopyTripletList_r !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !> Copy a triplet list (complex). SUBROUTINE CopyTripletList_c(tripA, tripB) !> The triplet list to copy. TYPE(TripletList_c), INTENT(IN) :: tripA @@ -161,6 +169,24 @@ SUBROUTINE CopyTripletList_c(tripA, tripB) #include "triplet_includes/CopyTripletList.f90" END SUBROUTINE CopyTripletList_c +!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !> Copy and upcast a triplet list (real -> complex). + SUBROUTINE CopyTripletList_rc(tripA, tripB) + !> The triplet list to copy. + TYPE(TripletList_r), INTENT(IN) :: tripA + !> tripB = tripA + TYPE(TripletList_c), INTENT(INOUT) :: tripB + !! Local varaibles + INTEGER II + + CALL ConstructTripletList(tripB, tripA%CurrentSize) + DO II = 1, tripA%CurrentSize + tripB%DATA(II)%index_row = tripA%DATA(II)%index_row + tripB%DATA(II)%index_column = tripA%DATA(II)%index_column + tripB%DATA(II)%point_value = & + & CMPLX(tripA%DATA(II)%point_value, KIND=NTCOMPLEX) + END DO + END SUBROUTINE CopyTripletList_rc !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! !> Increase the size of a triplet list. PURE SUBROUTINE ResizeTripletList_r(this, size) @@ -365,6 +391,44 @@ SUBROUTINE RedistributeTripletLists_c(triplet_lists, comm, local_data_out) #undef MPIDATATYPE END SUBROUTINE RedistributeTripletLists_c +!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !> Gather triplet lists from a set of processors. + SUBROUTINE AllGatherTripletList_r(triplet_in, comm, gathered_out) + !> Locally held triplet list + TYPE(TripletList_r), INTENT(IN) :: triplet_in + !> The mpi communicator to gather along. + INTEGER, INTENT(IN) :: comm + !> The resulting gathered triplet list. + TYPE(TripletList_r), INTENT(INOUT) :: gathered_out + !! Local data (type specific) + REAL(NTREAL), DIMENSION(:), ALLOCATABLE :: send_buffer_val + REAL(NTREAL), DIMENSION(:), ALLOCATABLE :: recv_buffer_val + TYPE(Triplet_r) :: temp_triplet + +#define MPIDATATYPE MPINTREAL +#include "triplet_includes/GatherTripletList.f90" +#undef MPIDATATYPE + + END SUBROUTINE AllGatherTripletList_r +!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !> Gather triplet lists from a set of processors. + SUBROUTINE AllGatherTripletList_c(triplet_in, comm, gathered_out) + !> Locally held triplet list + TYPE(TripletList_c), INTENT(IN) :: triplet_in + !> The mpi communicator to gather along. + INTEGER, INTENT(IN) :: comm + !> The resulting gathered triplet list. + TYPE(TripletList_c), INTENT(INOUT) :: gathered_out + !! Local data (type specific) + COMPLEX(NTCOMPLEX), DIMENSION(:), ALLOCATABLE :: send_buffer_val + COMPLEX(NTCOMPLEX), DIMENSION(:), ALLOCATABLE :: recv_buffer_val + TYPE(Triplet_c) :: temp_triplet + +#define MPIDATATYPE MPINTCOMPLEX +#include "triplet_includes/GatherTripletList.f90" +#undef MPIDATATYPE + + END SUBROUTINE AllGatherTripletList_c !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! !> Shift the rows and columns of a triplet list by set values. !> Frequently, we have a triplet list that comes from the global matrix which diff --git a/Source/Fortran/distributed_algebra_includes/ScaleDiagonal.f90 b/Source/Fortran/distributed_algebra_includes/ScaleDiagonal.f90 new file mode 100644 index 00000000..01f7c66e --- /dev/null +++ b/Source/Fortran/distributed_algebra_includes/ScaleDiagonal.f90 @@ -0,0 +1,24 @@ + INTEGER :: II, col + + !! Merge to the local block + CALL MergeMatrixLocalBlocks(this, lmat) + + !! Filter out the triplets that aren't stored locally + CALL ConstructTripletList(filtered) + DO II = 1, tlist%CurrentSize + CALL GetTripletAt(tlist, II, trip) + col = trip%index_column + IF (col .GE. this%start_column .AND. col .LT. this%end_column) THEN + trip%index_column = trip%index_column - this%start_column + 1 + trip%index_row = trip%index_column + CALL AppendToTripletList(filtered, trip) + END IF + END DO + + !! Scale + CALL MatrixDiagonalScale(lmat, filtered) + + !! Split + CALL SplitMatrixToLocalBlocks(this, lmat) + CALL DestructMatrix(lmat) + CALL DestructTripletList(filtered) diff --git a/Source/Fortran/distributed_includes/FilterMatrix.f90 b/Source/Fortran/distributed_includes/FilterMatrix.f90 index 0373f513..2530b267 100644 --- a/Source/Fortran/distributed_includes/FilterMatrix.f90 +++ b/Source/Fortran/distributed_includes/FilterMatrix.f90 @@ -1,7 +1,5 @@ !! Local Data - TYPE(ProcessGrid_t) :: grid_temp - LOGICAL :: is_complex_temp - INTEGER :: II, size_temp + INTEGER :: II CALL GetMatrixTripletList(this, tlist) CALL ConstructTripletList(new_list) @@ -13,10 +11,4 @@ END IF END DO - size_temp = this%actual_matrix_dimension - grid_temp = this%process_grid - is_complex_temp = this%is_complex - CALL DestructMatrix(this) - CALL ConstructEmptyMatrix(this, size_temp, grid_temp, is_complex_temp) - - CALL FillMatrixFromTripletList(this, new_list) + CALL FillMatrixFromTripletList(this, new_list, preduplicated_in=.TRUE.) diff --git a/Source/Fortran/distributed_includes/GatherMatrixTripletList.f90 b/Source/Fortran/distributed_includes/GatherMatrixTripletList.f90 new file mode 100644 index 00000000..ecb280ff --- /dev/null +++ b/Source/Fortran/distributed_includes/GatherMatrixTripletList.f90 @@ -0,0 +1,3 @@ + CALL GatherMatrixToProcess(this, lmat) + CALL MatrixToTripletList(lmat, tlist) + CALL DestructMatrix(lmat) diff --git a/Source/Fortran/sparse_includes/DiagonalScale.f90 b/Source/Fortran/sparse_includes/DiagonalScale.f90 new file mode 100644 index 00000000..2bb20fae --- /dev/null +++ b/Source/Fortran/sparse_includes/DiagonalScale.f90 @@ -0,0 +1,9 @@ + + INTEGER :: col, II + + DO II = 1, tlist%CurrentSize + col = tlist%DATA(II)%index_column + val = tlist%DATA(II)%point_value + mat%values(mat%outer_index(col) + 1:mat%outer_index(col + 1)) = & + val * mat%values(mat%outer_index(col) + 1:mat%outer_index(col + 1)) + END DO diff --git a/Source/Fortran/triplet_includes/GatherTripletList.f90 b/Source/Fortran/triplet_includes/GatherTripletList.f90 new file mode 100644 index 00000000..9b1587df --- /dev/null +++ b/Source/Fortran/triplet_includes/GatherTripletList.f90 @@ -0,0 +1,66 @@ + !! Local Data - Send/Recv Buffers + INTEGER, DIMENSION(:), ALLOCATABLE :: send_buffer_row + INTEGER, DIMENSION(:), ALLOCATABLE :: send_buffer_col + INTEGER, DIMENSION(:), ALLOCATABLE :: recv_buffer_row + INTEGER, DIMENSION(:), ALLOCATABLE :: recv_buffer_col + + !! Sizes help + INTEGER, DIMENSION(:), ALLOCATABLE :: recvcounts + INTEGER, DIMENSION(:), ALLOCATABLE :: displ + INTEGER :: gather_size + + !! Temporary variables + INTEGER :: num_processes, II, ierr + + !! Figure out the comm size + CALL MPI_COMM_SIZE(comm, num_processes, ierr) + + !! Get the count + ALLOCATE(recvcounts(num_processes)) + CALL MPI_Allgather(triplet_in%CurrentSize, 1, MPI_INTEGER, recvcounts, & + & 1, MPI_INTEGER, comm, ierr) + + !! Get the displacements + gather_size = SUM(recvcounts) + ALLOCATE(displ(num_processes)) + displ(1) = 0 + DO II = 2, num_processes + displ(II) = displ(II - 1) + recvcounts(II - 1) + END DO + + !! Prepare the send buffers + ALLOCATE(send_buffer_row(triplet_in%CurrentSize)) + ALLOCATE(send_buffer_col(triplet_in%CurrentSize)) + ALLOCATE(send_buffer_val(triplet_in%CurrentSize)) + DO II = 1, triplet_in%CurrentSize + CALL GetTripletAt(triplet_in, II, temp_triplet) + send_buffer_row(II) = temp_triplet%index_row + send_buffer_col(II) = temp_triplet%index_column + send_buffer_val(II) = temp_triplet%point_value + END DO + + !! Gather Call + ALLOCATE(recv_buffer_row(gather_size)) + ALLOCATE(recv_buffer_col(gather_size)) + ALLOCATE(recv_buffer_val(gather_size)) + CALL MPI_Allgatherv(send_buffer_row, triplet_in%CurrentSize, MPI_INTEGER, & + & recv_buffer_row, recvcounts, displ, MPI_INTEGER, comm, ierr) + CALL MPI_Allgatherv(send_buffer_col, triplet_in%CurrentSize, MPI_INTEGER, & + & recv_buffer_col, recvcounts, displ, MPI_INTEGER, comm, ierr) + CALL MPI_Allgatherv(send_buffer_val, triplet_in%CurrentSize, MPIDATATYPE, & + & recv_buffer_val, recvcounts, displ, MPIDATATYPE, comm, ierr) + + !! Unpack + CALL ConstructTripletList(gathered_out, gather_size) + DO II = 1, gather_size + gathered_out%DATA(II)%index_row = recv_buffer_row(II) + gathered_out%DATA(II)%index_column = recv_buffer_col(II) + gathered_out%DATA(II)%point_value = recv_buffer_val(II) + END DO + + !! Cleanup + DEALLOCATE(recvcounts) + DEALLOCATE(displ) + DEALLOCATE(send_buffer_row) + DEALLOCATE(send_buffer_col) + DEALLOCATE(send_buffer_val) diff --git a/Source/Wrapper/PSMatrixAlgebraModule_wrp.F90 b/Source/Wrapper/PSMatrixAlgebraModule_wrp.F90 index 3fe86b2d..f57aa77a 100644 --- a/Source/Wrapper/PSMatrixAlgebraModule_wrp.F90 +++ b/Source/Wrapper/PSMatrixAlgebraModule_wrp.F90 @@ -6,7 +6,7 @@ MODULE PSMatrixAlgebraModule_wrp USE PSMatrixAlgebraModule USE PSMatrixModule_wrp, ONLY : Matrix_ps_wrp USE PermutationModule_wrp, ONLY : Permutation_wrp - USE TripletListModule_wrp, ONLY : TripletList_r_wrp + USE TripletListModule_wrp, ONLY : TripletList_r_wrp, TripletList_c_wrp USE WrapperModule, ONLY : SIZE_wrp USE ISO_C_BINDING, ONLY : c_int, c_char, c_bool IMPLICIT NONE @@ -20,6 +20,8 @@ MODULE PSMatrixAlgebraModule_wrp PUBLIC :: ScaleMatrix_ps_wrp PUBLIC :: MatrixNorm_ps_wrp PUBLIC :: MatrixTrace_ps_wrp + PUBLIC :: MatrixDiagonalScale_psr_wrp + PUBLIC :: MatrixDiagonalScale_psc_wrp CONTAINS!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! !> Matrix B = alpha*Matrix A + Matrix B (AXPY) SUBROUTINE IncrementMatrix_ps_wrp(ih_matA, ih_matB, alpha_in,threshold_in) & @@ -143,5 +145,33 @@ SUBROUTINE MatrixTrace_ps_wrp(ih_this, trace_value) & h_this = TRANSFER(ih_this,h_this) CALL MatrixTrace(h_this%DATA, trace_value) END SUBROUTINE MatrixTrace_ps_wrp +!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !> Scale a matrix using a diagonal matrix (triplet list form). + SUBROUTINE MatrixDiagonalScale_psr_wrp(ih_mat, ih_tlist) & + & BIND(c,name="MatrixDiagonalScale_psr_wrp") + INTEGER(kind=c_int), INTENT(INOUT) :: ih_mat(SIZE_wrp) + INTEGER(kind=c_int), INTENT(IN) :: ih_tlist(SIZE_wrp) + TYPE(Matrix_ps_wrp) :: h_mat + TYPE(TripletList_r_wrp) :: h_tlist + + h_mat = TRANSFER(ih_mat, h_mat) + h_tlist = TRANSFER(ih_tlist, h_tlist) + + CALL MatrixDiagonalScale(h_mat%DATA, h_tlist%DATA) + END SUBROUTINE MatrixDiagonalScale_psr_wrp +!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !> Scale a matrix using a diagonal matrix (triplet list form). + SUBROUTINE MatrixDiagonalScale_psc_wrp(ih_mat, ih_tlist) & + & BIND(c,name="MatrixDiagonalScale_psc_wrp") + INTEGER(kind=c_int), INTENT(INOUT) :: ih_mat(SIZE_wrp) + INTEGER(kind=c_int), INTENT(IN) :: ih_tlist(SIZE_wrp) + TYPE(Matrix_ps_wrp) :: h_mat + TYPE(TripletList_c_wrp) :: h_tlist + + h_mat = TRANSFER(ih_mat, h_mat) + h_tlist = TRANSFER(ih_tlist, h_tlist) + + CALL MatrixDiagonalScale(h_mat%DATA, h_tlist%DATA) + END SUBROUTINE MatrixDiagonalScale_psc_wrp !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! END MODULE PSMatrixAlgebraModule_wrp diff --git a/Source/Wrapper/SMatrixAlgebraModule_wrp.F90 b/Source/Wrapper/SMatrixAlgebraModule_wrp.F90 index 0292018a..03b5d0ed 100644 --- a/Source/Wrapper/SMatrixAlgebraModule_wrp.F90 +++ b/Source/Wrapper/SMatrixAlgebraModule_wrp.F90 @@ -18,12 +18,14 @@ MODULE SMatrixAlgebraModule_wrp PUBLIC :: DotMatrix_lsr_wrp PUBLIC :: PairwiseMultiplyMatrix_lsr_wrp PUBLIC :: MatrixMultiply_lsr_wrp + PUBLIC :: MatrixDiagonalScale_lsr_wrp !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! PUBLIC :: ScaleMatrix_lsc_wrp PUBLIC :: IncrementMatrix_lsc_wrp PUBLIC :: DotMatrix_lsc_wrp PUBLIC :: PairwiseMultiplyMatrix_lsc_wrp PUBLIC :: MatrixMultiply_lsc_wrp + PUBLIC :: MatrixDiagonalScale_lsc_wrp CONTAINS!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! !> Wrap the scale a sparse matrix by a constant routine. SUBROUTINE ScaleMatrix_lsr_wrp(ih_this, constant) & @@ -110,6 +112,20 @@ SUBROUTINE MatrixMultiply_lsr_wrp(ih_matA, ih_matB, ih_matC, IsATransposed, & & LOGICAL(IsATransposed), LOGICAL(IsBTransposed), alpha, & & beta, threshold, h_blocked_memory_pool%DATA) END SUBROUTINE MatrixMultiply_lsr_wrp +!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !> Scale a matrix using a diagonal matrix (triplet list form). + SUBROUTINE MatrixDiagonalScale_lsr_wrp(ih_mat, ih_tlist) & + & BIND(c,name="MatrixDiagonalScale_lsr_wrp") + INTEGER(kind=c_int), INTENT(INOUT) :: ih_mat(SIZE_wrp) + INTEGER(kind=c_int), INTENT(IN) :: ih_tlist(SIZE_wrp) + TYPE(Matrix_lsr_wrp) :: h_mat + TYPE(TripletList_r_wrp) :: h_tlist + + h_mat = TRANSFER(ih_mat, h_mat) + h_tlist = TRANSFER(ih_tlist, h_tlist) + + CALL MatrixDiagonalScale(h_mat%DATA, h_tlist%DATA) + END SUBROUTINE MatrixDiagonalScale_lsr_wrp !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! !> Wrap the scale a sparse matrix by a constant routine. SUBROUTINE ScaleMatrix_lsc_wrp(ih_this, constant) & @@ -201,5 +217,19 @@ SUBROUTINE MatrixMultiply_lsc_wrp(ih_matA, ih_matB, ih_matC, IsATransposed, & & LOGICAL(IsATransposed), LOGICAL(IsBTransposed), alpha, & & beta, threshold, h_blocked_memory_pool%DATA) END SUBROUTINE MatrixMultiply_lsc_wrp +!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !> Scale a matrix using a diagonal matrix (triplet list form). + SUBROUTINE MatrixDiagonalScale_lsc_wrp(ih_mat, ih_tlist) & + & BIND(c,name="MatrixDiagonalScale_lsc_wrp") + INTEGER(kind=c_int), INTENT(INOUT) :: ih_mat(SIZE_wrp) + INTEGER(kind=c_int), INTENT(IN) :: ih_tlist(SIZE_wrp) + TYPE(Matrix_lsc_wrp) :: h_mat + TYPE(TripletList_c_wrp) :: h_tlist + + h_mat = TRANSFER(ih_mat, h_mat) + h_tlist = TRANSFER(ih_tlist, h_tlist) + + CALL MatrixDiagonalScale(h_mat%DATA, h_tlist%DATA) + END SUBROUTINE MatrixDiagonalScale_lsc_wrp !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! END MODULE SMatrixAlgebraModule_wrp diff --git a/UnitTests/test_matrix.py b/UnitTests/test_matrix.py index 585daa34..90ee3d14 100644 --- a/UnitTests/test_matrix.py +++ b/UnitTests/test_matrix.py @@ -57,6 +57,8 @@ class TestLocalMatrix(unittest.TestCase): SMatrix = nt.Matrix_lsr MatrixMemoryPool = nt.MatrixMemoryPool_r complex = False + TripletList = nt.TripletList_r + Triplet = nt.Triplet_r def _compare_mat(self, val1, val2): from helpers import THRESHOLD @@ -397,12 +399,37 @@ def test_get_column(self): ResultMat = mmread(self.file2) self._compare_mat(CheckMat, ResultMat) + def test_scalediag(self): + '''Test routines to scale by a diagonal matrix.''' + from copy import deepcopy + for param in self.parameters: + matrix = param.create_matrix(complex=self.complex) + mmwrite(self.file1, matrix) + CheckMat = deepcopy(matrix) + + tlist = self.TripletList() + for i in range(matrix.shape[1]): + t = self.Triplet() + t.index_column = i + 1 + t.index_row = i + 1 + t.point_value = i + tlist.Append(t) + CheckMat[:, i] *= i + ntmatrix = self.SMatrix(self.file1) + ntmatrix.DiagonalScale(tlist) + ntmatrix.WriteToMatrixMarket(self.file2) + + ResultMat = mmread(self.file2) + self._compare_mat(CheckMat, ResultMat) + class TestLocalMatrix_c(TestLocalMatrix): '''Specialization for complex matrices''' SMatrix = nt.Matrix_lsc MatrixMemoryPool = nt.MatrixMemoryPool_c complex = True + TripletList = nt.TripletList_c + Triplet = nt.Triplet_c def test_conjugatetranspose(self): '''Test routines to compute the conjugate transpose of a matrix.''' diff --git a/UnitTests/test_psmatrixalgebra.py b/UnitTests/test_psmatrixalgebra.py index 1d09c88b..052dff8c 100644 --- a/UnitTests/test_psmatrixalgebra.py +++ b/UnitTests/test_psmatrixalgebra.py @@ -284,6 +284,33 @@ def test_reverse(self): self.check_result() + def test_scalediag(self): + '''Test routines to scale by a diagonal matrix.''' + from copy import deepcopy + for param in self.parameters: + matrix = param.create_matrix(complex=self.complex1) + self.write_matrix(matrix, self.input_file1) + self.CheckMat = deepcopy(matrix) + + # Need a guard because otherwise we write a real matrix by mistake. + if self.complex1 and matrix.nnz == 0: + continue + + tlist = self.TripletList() + for i in range(matrix.shape[1]): + t = self.Triplet() + t.index_column = i + 1 + t.index_row = i + 1 + t.point_value = i + tlist.Append(t) + self.CheckMat[:, i] *= i + ntmatrix = nt.Matrix_ps(self.input_file1) + ntmatrix.DiagonalScale(tlist) + ntmatrix.WriteToMatrixMarket(self.result_file) + + comm.barrier() + self.check_result() + class TestPSMatrixAlgebra_r(TestPSMatrixAlgebra, unittest.TestCase): '''Special routines for real algebra''' @@ -291,6 +318,8 @@ class TestPSMatrixAlgebra_r(TestPSMatrixAlgebra, unittest.TestCase): complex1 = False # Whether the second matrix is complex or not complex2 = False + TripletList = nt.TripletList_r + Triplet = nt.Triplet_r def test_dot(self): '''Test routines to add together matrices.''' @@ -327,6 +356,8 @@ class TestPSMatrixAlgebra_c(TestPSMatrixAlgebra, unittest.TestCase): complex1 = True # Whether the second matrix is complex or not complex2 = True + TripletList = nt.TripletList_c + Triplet = nt.Triplet_c def test_dot(self): '''Test routines to add together matrices.''' @@ -366,6 +397,8 @@ class TestPSMatrixAlgebra_rc(TestPSMatrixAlgebra, unittest.TestCase): complex1 = True # Whether the second matrix is complex or not complex2 = False + TripletList = nt.TripletList_c + Triplet = nt.Triplet_c def setUp(self): '''Set up a specific test.''' @@ -381,6 +414,8 @@ class TestPSMatrixAlgebra_cr(TestPSMatrixAlgebra, unittest.TestCase): complex1 = False # Whether the second matrix is complex or not complex2 = True + TripletList = nt.TripletList_r + Triplet = nt.Triplet_r def setUp(self): '''Set up a specific test.'''