From 1850ac446c5700a60adc2b4c5a9e762d2368a710 Mon Sep 17 00:00:00 2001 From: Jeremy E Kozdon Date: Wed, 4 Aug 2021 10:51:56 -0700 Subject: [PATCH] Add (basic) ksp functionality --- src/PETSc.jl | 2 +- src/ksp.jl | 205 ++++++++++++++++++++++++++++++----------------- src/sys.jl | 8 +- src/vec.jl | 7 ++ test/ksp.jl | 82 +++++++++++++++++++ test/runtests.jl | 3 +- 6 files changed, 231 insertions(+), 76 deletions(-) create mode 100644 test/ksp.jl diff --git a/src/PETSc.jl b/src/PETSc.jl index 87d53f93..9eb1a7ec 100644 --- a/src/PETSc.jl +++ b/src/PETSc.jl @@ -29,7 +29,7 @@ include("matshell.jl") # include("dm.jl") # include("dmda.jl") # include("dmstag.jl") -# include("ksp.jl") +include("ksp.jl") # include("pc.jl") # include("snes.jl") include("sys.jl") diff --git a/src/ksp.jl b/src/ksp.jl index f2a5ba13..3c417d0e 100644 --- a/src/ksp.jl +++ b/src/ksp.jl @@ -1,28 +1,146 @@ - const CKSP = Ptr{Cvoid} const CKSPType = Cstring -abstract type AbstractKSP{T, PetscLib} <: Factorization{T} end +abstract type AbstractKSP{PetscLib, PetscScalar} <: Factorization{PetscScalar} end -Base.@kwdef mutable struct KSP{T, PetscLib} <: AbstractKSP{T, PetscLib} +Base.@kwdef mutable struct KSP{PetscLib, PetscScalar} <: + AbstractKSP{PetscLib, PetscScalar} ptr::CKSP = C_NULL - opts::Options{PetscLib} - # Stuff to keep around so that they don't get gc'ed - _A = nothing - _P = nothing - _dm = nothing - # Function pointers - ComputeRHS! = nothing - ComputeOperators! = nothing + opts::Options{PetscLib} = Options(PetscLib) + A::Union{AbstractMat, Nothing} = nothing + P::Union{AbstractMat, Nothing} = nothing +end + +""" + KSP(A::AbstractMat, P::AbstractMat{PetscLib} = A; options...) + +Create a `KSP` using the matrix `A` and preconditioner construction matrix `P` +with the `options`. + +The communicator is obtained from `A` and if it has size `1` then the garbage +collector is set, otherwise the user is responsible for calling +[`destroy`](@ref). + +# External Links +$(_doc_external("KSP/KSPCreate")) +$(_doc_external("KSP/KSPSetOperators")) +$(_doc_external("KSP/KSPSetFromOptions")) +""" +function KSP( + A::AbstractMat{PetscLib}, + P::AbstractMat{PetscLib} = A; + options..., +) where {PetscLib} + @assert initialized(PetscLib) + opts = Options(PetscLib; options...) + PetscScalar = PetscLib.PetscScalar + ksp = KSP{PetscLib, PetscScalar}(opts = opts) + comm = getcomm(A) + + with(ksp.opts) do + LibPETSc.KSPCreate(PetscLib, comm, ksp) + end + + setoperators!(ksp, A, P) + setfromoptions!(ksp) + + # If there is only one rank we can finalize the KSP with GC + if MPI.Comm_size(comm) == 1 + finalizer(destroy, ksp) + end + + return ksp +end + +function setoperators!( + ksp::AbstractKSP{PetscLib}, + A::AbstractMat{PetscLib}, + P::AbstractMat{PetscLib} = A, +) where {PetscLib} + LibPETSc.KSPSetOperators(PetscLib, ksp, A, P) + ksp.A = A + ksp.P = P + return ksp +end + +function setfromoptions!(ksp::AbstractKSP{PetscLib}) where {PetscLib} + with(ksp.opts) do + LibPETSc.KSPSetFromOptions(PetscLib, ksp) + end +end + +function destroy(ksp::AbstractKSP{PetscLib}) where {PetscLib} + finalized(PetscLib) || LibPETSc.MatDestroy(PetscLib, ksp) + return nothing +end + +function solve!( + x::AbstractVec{PetscLib}, + ksp::AbstractKSP{PetscLib}, + b::AbstractVec{PetscLib}, +) where {PetscLib} + with(ksp.opts) do + LibPETSc.KSPSolve(PetscLib, ksp, b, x) + end + return x +end + +""" + createvecs(ksp::AbstractKSP; nright = 0, nleft = 0) + +Create `nright` right and `nleft` left vectors compatible with the `ksp`. +Returned object `V` has `Tuple` members `V.right` and `V.left` containing the +vectors. + +# External Links +$(_doc_external("KSP/KSPCreateVecs")) +""" +function createvecs( + ksp::AbstractKSP{PetscLib}; + nright = 0, + nleft = 0, +) where {PetscLib} + # pointer of pointers to the base vectors + r_right_vs = Ref{Ptr{CVec}}() + r_left_vs = Ref{Ptr{CVec}}() + + # create 1 right and left vector + LibPETSc.KSPCreateVecs(PetscLib, ksp, 1, r_right_vs, 1, r_left_vs) + + # create right vectors + a_v = unsafe_wrap(Array, r_right_vs[], 1; own = false) + v = VecPtr(PetscLib, a_v[1], false) + right = ntuple(i -> similar(v), nright) + + # create left vectors + a_v = unsafe_wrap(Array, r_left_vs[], 1; own = false) + v = VecPtr(PetscLib, a_v[1], false) + left = ntuple(i -> similar(v), nleft) + + LibPETSc.VecDestroyVecs(PetscLib, 1, r_right_vs) + LibPETSc.VecDestroyVecs(PetscLib, 1, r_left_vs) + + (right = right, left = left) +end + +function LinearAlgebra.ldiv!(x::AbstractVec, ksp::AbstractKSP, b::AbstractVec) + solve!(x, ksp, b) +end + +function Base.:\(ksp::AbstractKSP, b::AbstractVec) + x = createvecs(ksp; nleft = 1).left[1] + ldiv!(x, ksp, b) + return x end +#= +# +# OLD WRAPPERS +# struct WrappedKSP{T, PetscLib} <: AbstractKSP{T, PetscLib} ptr::CKSP end -scalartype(::KSP{T}) where {T} = T -Base.eltype(::KSP{T}) where {T} = T - LinearAlgebra.transpose(ksp) = LinearAlgebra.Transpose(ksp) LinearAlgebra.adjoint(ksp) = LinearAlgebra.Adjoint(ksp) @@ -70,32 +188,6 @@ struct Fn_KSPComputeOperators{T} end @for_libpetsc begin - function KSP{$PetscScalar}(comm::MPI.Comm; kwargs...) - @assert initialized($petsclib) - opts = Options($petsclib, kwargs...) - ksp = KSP{$PetscScalar, $PetscLib}(opts=opts) - with(ksp.opts) do - @chk ccall((:KSPCreate, $libpetsc), PetscErrorCode, (MPI.MPI_Comm, Ptr{CKSP}), comm, ksp) - end - if comm == MPI.COMM_SELF - finalizer(destroy, ksp) - end - return ksp - end - - function destroy(ksp::KSP{$PetscScalar}) - finalized($petsclib) || - @chk ccall((:KSPDestroy, $libpetsc), PetscErrorCode, (Ptr{CKSP},), ksp) - return nothing - end - - function setoperators!(ksp::KSP{$PetscScalar}, A::AbstractMat{$PetscScalar}, P::AbstractMat{$PetscScalar}) - @chk ccall((:KSPSetOperators, $libpetsc), PetscErrorCode, (CKSP, CMat, CMat), ksp, A, P) - ksp._A = A - ksp._P = P - return nothing - end - function (::Fn_KSPComputeRHS{$PetscScalar})( new_ksp_ptr::CKSP, cb::CVec, @@ -184,12 +276,6 @@ struct Fn_KSPComputeOperators{T} end return nothing end - function setfromoptions!(ksp::KSP{$PetscScalar}) - with(ksp.opts) do - @chk ccall((:KSPSetFromOptions, $libpetsc), PetscErrorCode, (CKSP,), ksp) - end - end - function gettype(ksp::KSP{$PetscScalar}) t_r = Ref{CKSPType}() @chk ccall((:KSPGetType, $libpetsc), PetscErrorCode, (CKSP, Ptr{CKSPType}), ksp, t_r) @@ -217,14 +303,6 @@ struct Fn_KSPComputeOperators{T} end return r_rnorm[] end - function solve!(x::AbstractVec{$PetscScalar}, ksp::KSP{$PetscScalar}, b::AbstractVec{$PetscScalar}) - with(ksp.opts) do - @chk ccall((:KSPSolve, $libpetsc), PetscErrorCode, - (CKSP, CVec, CVec), ksp, b, x) - end - return x - end - function solve!(ksp::KSP{$PetscScalar}) with(ksp.opts) do @chk ccall((:KSPSolve, $libpetsc), PetscErrorCode, @@ -256,21 +334,6 @@ function LinearAlgebra.ldiv!(x::AbstractVector{T}, ksp::KSPAT{T, LT}, b::Abstrac end Base.:\(ksp::KSPAT{T, LT}, b::AbstractVector{T}) where {T, LT} = ldiv!(similar(b), ksp, b) - -""" - KSP(A, P; options...) - -Construct a PETSc Krylov subspace solver. - -Any PETSc options prefixed with `ksp_` and `pc_` can be passed as keywords. -""" -function KSP(A::AbstractMat{T}, P::AbstractMat{T}=A; kwargs...) where {T} - ksp = KSP{T}(getcomm(A); kwargs...) - setoperators!(ksp, A, P) - setfromoptions!(ksp) - return ksp -end - """ KSP(da::AbstractDM; options...) @@ -290,7 +353,6 @@ end Base.show(io::IO, ksp::KSP) = _show(io, ksp) - """ iters(ksp::KSP) @@ -301,7 +363,6 @@ $(_doc_external("KSP/KSPGetIterationNumber")) """ iters - """ resnorm(ksp::KSP) @@ -311,4 +372,4 @@ Gets the last (approximate preconditioned) residual norm that has been computed. $(_doc_external("KSP/KSPGetResidualNorm")) """ resnorm - +=# diff --git a/src/sys.jl b/src/sys.jl index 88380c5f..37ad2842 100644 --- a/src/sys.jl +++ b/src/sys.jl @@ -1,6 +1,6 @@ const CPetscObject = Ptr{Cvoid} -const UnionPetscTypes = Union{Options, AbstractVec, AbstractMat} +const UnionPetscTypes = Union{Options, AbstractVec, AbstractMat, AbstractKSP} # allows us to pass PETSc_XXX objects directly into CXXX ccall signatures Base.cconvert(::Type{CPetscObject}, obj::UnionPetscTypes) = obj @@ -12,7 +12,11 @@ function Base.unsafe_convert(::Type{Ptr{CPetscObject}}, obj::UnionPetscTypes) end function getcomm( - obj::Union{AbstractVec{PetscLib}, AbstractMat{PetscLib}}, + obj::Union{ + AbstractVec{PetscLib}, + AbstractMat{PetscLib}, + AbstractKSP{PetscLib}, + }, ) where {PetscLib} comm = MPI.Comm() LibPETSc.PetscObjectGetComm(PetscLib, obj, comm) diff --git a/src/vec.jl b/src/vec.jl index 1d69e74c..21258134 100644 --- a/src/vec.jl +++ b/src/vec.jl @@ -612,3 +612,10 @@ function getpetsctype(vec::AbstractVec{PetscLib}) where {PetscLib} LibPETSc.VecGetType(PetscLib, vec, name_r) return unsafe_string(name_r[]) end + +function Base.similar(v::AbstractVec{PetscLib}) where {PetscLib} + r_x = Ref{CVec}() + LibPETSc.VecDuplicate(PetscLib, v, r_x) + x = VecPtr(PetscLib, r_x[], true) + return x +end diff --git a/test/ksp.jl b/test/ksp.jl new file mode 100644 index 00000000..cc477192 --- /dev/null +++ b/test/ksp.jl @@ -0,0 +1,82 @@ +using Test +using MPI +MPI.Initialized() || MPI.Init() +using PETSc +using LinearAlgebra: mul! + +@testset "KSP" begin + comm = MPI.COMM_WORLD + mpisize = MPI.Comm_size(comm) + mpirank = MPI.Comm_rank(comm) + + for petsclib in PETSc.petsclibs + PETSc.initialize(petsclib) + PetscScalar = petsclib.PetscScalar + PetscInt = petsclib.PetscInt + + loc_num_rows = 10 + loc_num_cols = 10 + diag_nonzeros = 3 + off_diag_non_zeros = 3 + + A = PETSc.MatAIJ( + petsclib, + comm, + loc_num_rows, + loc_num_cols, + diag_nonzeros, + off_diag_non_zeros, + ) + + # Get compatible vectors + (x, b) = PETSc.createvecs(A) + + row_rng = PETSc.ownershiprange(A, false) + for i in row_rng + if i == 0 + vals = [-2, 1] + row0idxs = [i] + col0idxs = [i, i + 1] + elseif i == mpisize * loc_num_rows - 1 + vals = [-2, 1] + row0idxs = [i] + col0idxs = [i, i - 1] + else + vals = [1, -2, 1] + row0idxs = [i] + col0idxs = [i - 1, i, i + 1] + end + PETSc.setvalues!( + A, + PetscInt.(row0idxs), + PetscInt.(col0idxs), + PetscScalar.(vals), + ) + x[i + 1] = (i + 1)^3 + end + PETSc.assemble!(A) + PETSc.assemble!(x) + + mul!(b, A, x) + y = similar(x) + + ksp = PETSc.KSP(A; ksp_rtol = 1e-16, pc_type = "jacobi") + PETSc.solve!(y, ksp, b) + PETSc.withlocalarray!(x, y) do x, y + @test x ≈ y + end + PETSc.destroy(x) + + x = ksp \ b + PETSc.withlocalarray!(x, y) do x, y + @test x ≈ y + end + PETSc.destroy(x) + + # PETSc.destroy(ksp) + PETSc.destroy(A) + PETSc.destroy(y) + PETSc.destroy(b) + PETSc.finalize(petsclib) + end +end diff --git a/test/runtests.jl b/test/runtests.jl index 85d81e9d..9c4a43ab 100644 --- a/test/runtests.jl +++ b/test/runtests.jl @@ -3,7 +3,7 @@ using MPI: mpiexec # Do the MPI tests first so we do not have mpi running inside MPI @testset "mpi tests" begin - for file in ("mpivec.jl", "mpimat.jl") + for file in ("mpivec.jl", "mpimat.jl", "ksp.jl") @test mpiexec() do mpi_cmd cmd = `$mpi_cmd -n 4 $(Base.julia_cmd()) --startup-file=no --project $file` @@ -18,3 +18,4 @@ include("options.jl") include("vec.jl") include("mat.jl") include("matshell.jl") +include("ksp.jl")