From 13f1a97edc152cba0ff6ddf898d13d40719a183f Mon Sep 17 00:00:00 2001 From: Katharine Hyatt Date: Tue, 7 Jan 2025 17:11:05 -0500 Subject: [PATCH 1/6] Split linalg tests into multiple files --- test/libraries/cublas.jl | 2330 --------------------------- test/libraries/cublas_extensions.jl | 490 ++++++ test/libraries/cublas_level1.jl | 97 ++ test/libraries/cublas_level2.jl | 640 ++++++++ test/libraries/cublas_level3.jl | 1432 ++++++++++++++++ 5 files changed, 2659 insertions(+), 2330 deletions(-) delete mode 100644 test/libraries/cublas.jl create mode 100644 test/libraries/cublas_extensions.jl create mode 100644 test/libraries/cublas_level1.jl create mode 100644 test/libraries/cublas_level2.jl create mode 100644 test/libraries/cublas_level3.jl diff --git a/test/libraries/cublas.jl b/test/libraries/cublas.jl deleted file mode 100644 index 1ef268309d..0000000000 --- a/test/libraries/cublas.jl +++ /dev/null @@ -1,2330 +0,0 @@ -using CUDA.CUBLAS -using CUDA.CUBLAS: band, bandex - -using LinearAlgebra - -using BFloat16s -using StaticArrays - -@test CUBLAS.version() isa VersionNumber -@test CUBLAS.version().major == CUBLAS.cublasGetProperty(CUDA.MAJOR_VERSION) -@test CUBLAS.version().minor == CUBLAS.cublasGetProperty(CUDA.MINOR_VERSION) -@test CUBLAS.version().patch == CUBLAS.cublasGetProperty(CUDA.PATCH_LEVEL) - -m = 20 -n = 35 -k = 13 - -############################################################################################ - -@testset "level 1" begin - @testset for T in [Float32, Float64, ComplexF32, ComplexF64] - A = CUDA.rand(T, m) - B = CuArray{T}(undef, m) - CUBLAS.copy!(m,A,B) - @test Array(A) == Array(B) - - @test testf(rmul!, rand(T, 6, 9, 3), Ref(rand())) - @test testf(dot, rand(T, m), rand(T, m)) - @test testf(*, transpose(rand(T, m)), rand(T, m)) - @test testf(*, rand(T, m)', rand(T, m)) - @test testf(norm, rand(T, m)) - @test testf(BLAS.asum, rand(T, m)) - @test testf(axpy!, Ref(rand()), rand(T, m), rand(T, m)) - @test testf(axpby!, Ref(rand()), rand(T, m), Ref(rand()), rand(T, m)) - - if T <: Complex - @test testf(dot, rand(T, m), rand(T, m)) - x = rand(T, m) - y = rand(T, m) - dx = CuArray(x) - dy = CuArray(y) - dz = dot(dx, dy) - z = dot(x, y) - @test dz ≈ z - end - - @test testf(rotate!, rand(T, m), rand(T, m), rand(real(T)), rand(real(T))) - @test testf(rotate!, rand(T, m), rand(T, m), rand(real(T)), rand(T)) - - @test testf(reflect!, rand(T, m), rand(T, m), rand(real(T)), rand(real(T))) - @test testf(reflect!, rand(T, m), rand(T, m), rand(real(T)), rand(T)) - - # swap is an extension - x = rand(T, m) - y = rand(T, m) - dx = CuArray(x) - dy = CuArray(y) - CUBLAS.swap!(m, dx, dy) - h_x = collect(dx) - h_y = collect(dy) - @test h_x ≈ y - @test h_y ≈ x - - a = convert.(T, [1.0, 2.0, -0.8, 5.0, 3.0]) - ca = CuArray(a) - @test BLAS.iamax(a) == CUBLAS.iamax(ca) - @test CUBLAS.iamin(ca) == 3 - end # level 1 testset - @testset for T in [Float16, ComplexF16] - A = CuVector(rand(T, m)) # CUDA.rand doesn't work with 16 bit types yet - B = CuArray{T}(undef, m) - CUBLAS.copy!(m,A,B) - @test Array(A) == Array(B) - - @test testf(dot, rand(T, m), rand(T, m)) - @test testf(*, transpose(rand(T, m)), rand(T, m)) - @test testf(*, rand(T, m)', rand(T, m)) - @test testf(norm, rand(T, m)) - @test testf(axpy!, Ref(rand()), rand(T, m), rand(T, m)) - @test testf(axpby!, Ref(rand()), rand(T, m), Ref(rand()), rand(T, m)) - - if T <: Complex - @test testf(dot, rand(T, m), rand(T, m)) - x = rand(T, m) - y = rand(T, m) - dx = CuArray(x) - dy = CuArray(y) - dz = dot(dx, dy) - z = dot(x, y) - @test dz ≈ z - end - end # level 1 testset -end - -############################################################################################ - -@testset "level 2" begin - @testset for elty in [Float32, Float64, ComplexF32, ComplexF64] - alpha = rand(elty) - beta = rand(elty) - - @testset "gemv" begin - @test testf(*, rand(elty, m, n), rand(elty, n)) - @test testf(*, transpose(rand(elty, m, n)), rand(elty, m)) - @test testf(*, rand(elty, m, n)', rand(elty, m)) - x = rand(elty, m) - A = rand(elty, m, m + 1 ) - y = rand(elty, m) - dx = CuArray(x) - dA = CuArray(A) - dy = CuArray(y) - @test_throws DimensionMismatch mul!(dy, dA, dx) - A = rand(elty, m + 1, m ) - dA = CuArray(A) - @test_throws DimensionMismatch mul!(dy, dA, dx) - x = rand(elty, m) - A = rand(elty, n, m) - dx = CuArray(x) - dA = CuArray(A) - alpha = rand(elty) - dy = CUBLAS.gemv('N', alpha, dA, dx) - hy = collect(dy) - @test hy ≈ alpha * A * x - dy = CUBLAS.gemv('N', dA, dx) - hy = collect(dy) - @test hy ≈ A * x - end - - if CUBLAS.version() >= v"11.9" - @testset "gemv_batched" begin - x = [rand(elty, m) for i=1:10] - A = [rand(elty, n, m) for i=1:10] - y = [rand(elty, n) for i=1:10] - dx = CuArray{elty, 1}[] - dA = CuArray{elty, 2}[] - dy = CuArray{elty, 1}[] - dbad = CuArray{elty, 1}[] - for i=1:length(A) - push!(dA, CuArray(A[i])) - push!(dx, CuArray(x[i])) - push!(dy, CuArray(y[i])) - if i < length(A) - 2 - push!(dbad,CuArray(dx[i])) - end - end - @test_throws DimensionMismatch CUBLAS.gemv_batched!('N', alpha, dA, dx, beta, dbad) - CUBLAS.gemv_batched!('N', alpha, dA, dx, beta, dy) - for i=1:length(A) - hy = collect(dy[i]) - y[i] = alpha * A[i] * x[i] + beta * y[i] - @test y[i] ≈ hy - end - end - end - - if CUBLAS.version() >= v"11.9" - @testset "gemv_strided_batched" begin - x = rand(elty, m, 10) - A = rand(elty, n, m, 10) - y = rand(elty, n, 10) - bad = rand(elty, m, 10) - dx = CuArray(x) - dA = CuArray(A) - dy = CuArray(y) - dbad = CuArray(bad) - @test_throws DimensionMismatch CUBLAS.gemv_strided_batched!('N', alpha, dA, dx, beta, dbad) - bad = rand(elty, n, 2) - dbad = CuArray(bad) - @test_throws DimensionMismatch CUBLAS.gemv_strided_batched!('N', alpha, dA, dx, beta, dbad) - CUBLAS.gemv_strided_batched!('N', alpha, dA, dx, beta, dy) - for i=1:size(A, 3) - hy = collect(dy[:, i]) - y[:, i] = alpha * A[:, :, i] * x[:, i] + beta * y[:, i] - @test y[:, i] ≈ hy - end - end - end - - @testset "mul! y = $f(A) * x * $Ts(a) + y * $Ts(b)" for f in (identity, transpose, adjoint), Ts in (Int, elty) - y, A, x = rand(elty, 5), rand(elty, 5, 5), rand(elty, 5) - dy, dA, dx = CuArray(y), CuArray(A), CuArray(x) - mul!(dy, f(dA), dx, Ts(1), Ts(2)) - mul!(y, f(A), x, Ts(1), Ts(2)) - @test Array(dy) ≈ y - end - - @testset "hermitian" begin - y, A, x = rand(elty, 5), Hermitian(rand(elty, 5, 5)), rand(elty, 5) - dy, dA, dx = CuArray(y), Hermitian(CuArray(A)), CuArray(x) - mul!(dy, dA, dx) - mul!(y, A, x) - @test Array(dy) ≈ y - end - - @testset "banded methods" begin - # bands - ku = 2 - kl = 3 - # generate banded matrix - A = rand(elty,m,n) - A = bandex(A,kl,ku) - # get packed format - Ab = band(A,kl,ku) - d_Ab = CuArray(Ab) - x = rand(elty,n) - d_x = CuArray(x) - @testset "gbmv!" begin - # test y = alpha*A*x + beta*y - y = rand(elty,m) - d_y = CuArray(y) - CUBLAS.gbmv!('N',m,kl,ku,alpha,d_Ab,d_x,beta,d_y) - BLAS.gbmv!('N',m,kl,ku,alpha,Ab,x,beta,y) - h_y = Array(d_y) - @test y ≈ h_y - # test y = alpha*transpose(A)*x + beta*y - x = rand(elty,n) - d_x = CuArray(x) - y = rand(elty,m) - d_y = CuArray(y) - CUBLAS.gbmv!('T',m,kl,ku,alpha,d_Ab,d_y,beta,d_x) - BLAS.gbmv!('T',m,kl,ku,alpha,Ab,y,beta,x) - h_x = Array(d_x) - @test x ≈ h_x - # test y = alpha*A'*x + beta*y - x = rand(elty,n) - d_x = CuArray(x) - y = rand(elty,m) - d_y = CuArray(y) - CUBLAS.gbmv!('C',m,kl,ku,alpha,d_Ab,d_y,beta,d_x) - BLAS.gbmv!('C',m,kl,ku,alpha,Ab,y,beta,x) - h_x = Array(d_x) - @test x ≈ h_x - # test alpha=1 version without y - d_y = CUBLAS.gbmv('N',m,kl,ku,d_Ab,d_x) - y = BLAS.gbmv('N',m,kl,ku,Ab,x) - h_y = Array(d_y) - @test y ≈ h_y - end - x = rand(elty,n) - d_x = CuArray(x) - @testset "gbmv" begin - # test y = alpha*A*x - d_y = CUBLAS.gbmv('N',m,kl,ku,alpha,d_Ab,d_x) - y = zeros(elty,m) - y = BLAS.gbmv('N',m,kl,ku,alpha,Ab,x) - h_y = Array(d_y) - @test y ≈ h_y - end - A = rand(elty,m,m) - A = A + A' - nbands = 3 - @test m >= 1+nbands - A = bandex(A,nbands,nbands) - # convert to 'upper' banded storage format - AB = band(A,0,nbands) - # construct x - x = rand(elty,m) - d_AB = CuArray(AB) - d_x = CuArray(x) - if elty <: Real - @testset "sbmv!" begin - y = rand(elty,m) - d_y = CuArray(y) - # sbmv! - CUBLAS.sbmv!('U',nbands,alpha,d_AB,d_x,beta,d_y) - y = alpha*(A*x) + beta*y - # compare - h_y = Array(d_y) - @test y ≈ h_y - end - @testset "sbmv" begin - d_y = CUBLAS.sbmv('U',nbands,d_AB,d_x) - y = A*x - # compare - h_y = Array(d_y) - @test y ≈ h_y - end - else - @testset "hbmv!" begin - y = rand(elty,m) - d_y = CuArray(y) - # hbmv! - CUBLAS.hbmv!('U',nbands,alpha,d_AB,d_x,beta,d_y) - y = alpha*(A*x) + beta*y - # compare - h_y = Array(d_y) - @test y ≈ h_y - end - @testset "hbmv" begin - d_y = CUBLAS.hbmv('U',nbands,d_AB,d_x) - y = A*x - # compare - h_y = Array(d_y) - @test y ≈ h_y - end - end - # generate triangular matrix - A = rand(elty,m,m) - # restrict to 3 bands - nbands = 3 - @test m >= 1+nbands - A = bandex(A,0,nbands) - # convert to 'upper' banded storage format - AB = band(A,0,nbands) - d_AB = CuArray(AB) - @testset "tbmv!" begin - y = rand(elty, m) - # move to host - d_y = CuArray(y) - # tbmv! - CUBLAS.tbmv!('U','N','N',nbands,d_AB,d_y) - y = A*y - # compare - h_y = Array(d_y) - @test y ≈ h_y - end - @testset "tbmv" begin - # tbmv - d_y = CUBLAS.tbmv('U','N','N',nbands,d_AB,d_x) - y = A*x - # compare - h_y = Array(d_y) - @test y ≈ h_y - end - @testset "tbsv!" begin - d_y = copy(d_x) - #tbsv! - CUBLAS.tbsv!('U','N','N',nbands,d_AB,d_y) - y = A\x - # compare - h_y = Array(d_y) - @test y ≈ h_y - end - @testset "tbsv" begin - d_y = CUBLAS.tbsv('U','N','N',nbands,d_AB,d_x) - y = A\x - # compare - h_y = Array(d_y) - @test y ≈ h_y - end - end - A = rand(elty,m,n) - dA = CuArray(A) - sA = rand(elty,m,m) - sA = sA + transpose(sA) - dsA = CuArray(sA) - hA = rand(elty,m,m) - hA = hA + hA' - dhA = CuArray(hA) - x = rand(elty,m) - dx = CuArray(x) - - function pack(A, uplo) - AP = Vector{elty}(undef, (n*(n+1))>>1) - k = 1 - for j in 1:n - for i in (uplo==:L ? (j:n) : (1:j)) - AP[k] = A[i,j] - k += 1 - end - end - return AP - end - - if elty in ["Float32", "Float64"] - # pack matrices - sAPU = pack(sA, :U) - dsAPU = CuVector(sAPU) - sAPL = pack(sA, :L) - dsAPL = CuVector(sAPL) - - @testset "spmv!" begin - # generate vectors - y = rand(elty,m) - # copy to device - dy = CuArray(y) - # execute on host - BLAS.spmv!('U',alpha,sAPU,x,beta,y) - # execute on device - CUBLAS.spmv!('U',alpha,dsAPU,dx,beta,dy) - # compare results - hy = Array(dy) - @test y ≈ hy - # execute on host - BLAS.spmv!('U',alpha,sAPL,x,beta,y) - # execute on device - CUBLAS.spmv!('U',alpha,dsAPL,dx,beta,dy) - # compare results - hy = Array(dy) - @test y ≈ hy - end - - @testset "spr!" begin - # execute on host - BLAS.spr!('U',alpha,x,sAPU) - # execute on device - CUBLAS.spr!('U',alpha,dx,dsAPU) - # compare results - hsAPU = Array(dsAPU) - @test sAPU ≈ hsAPU - # execute on host - BLAS.spr!('U',alpha,x,sAPL) - # execute on device - CUBLAS.spr!('U',alpha,dx,dsAPL) - # compare results - hAPL = Array(dAPL) - @test sAPL ≈ hAPL - end - end - - @testset "symv!" begin - # generate vectors - y = rand(elty,m) - # copy to device - dy = CuArray(y) - # execute on host - BLAS.symv!('U',alpha,sA,x,beta,y) - # execute on device - CUBLAS.symv!('U',alpha,dsA,dx,beta,dy) - # compare results - hy = Array(dy) - @test y ≈ hy - end - - @testset "symv" begin - y = BLAS.symv('U',sA,x) - # execute on device - dy = CUBLAS.symv('U',dsA,dx) - # compare results - hy = Array(dy) - @test y ≈ hy - end - if elty <: Complex - @testset "hemv!" begin - y = rand(elty,m) - dy = CuArray(y) - # execute on host - BLAS.hemv!('U',alpha,hA,x,beta,y) - # execute on device - CUBLAS.hemv!('U',alpha,dhA,dx,beta,dy) - # compare results - hy = Array(dy) - @test y ≈ hy - end - @testset "hemv" begin - y = BLAS.hemv('U',hA,x) - # execute on device - dy = CUBLAS.hemv('U',dhA,dx) - # compare results - hy = Array(dy) - @test y ≈ hy - end - end - A = triu(sA) - dA = CuArray(A) - @testset "trmv!" begin - d_y = copy(dx) - # execute trmv! - CUBLAS.trmv!('U','N','N',dA,d_y) - y = A*x - # compare - h_y = Array(d_y) - @test y ≈ h_y - @test_throws DimensionMismatch CUBLAS.trmv!('U','N','N',dA,CUDA.rand(elty,m+1)) - end - - @testset "trmv" begin - d_y = CUBLAS.trmv('U','N','N',dA,dx) - y = A*x - # compare - h_y = Array(d_y) - @test y ≈ h_y - end - - @testset "lmul!(::UpperTriangular)" begin - dy = copy(dx) - lmul!(UpperTriangular(dA), dy) - y = UpperTriangular(A) * x - @test y ≈ Array(dy) - end - @testset "lmul!(::UpperTriangular{Adjoint})" begin - dy = copy(dx) - lmul!(adjoint(UpperTriangular(dA)), dy) - y = adjoint(UpperTriangular(A)) * x - @test y ≈ Array(dy) - end - @testset "lmul!(::UpperTriangular{Transpose})" begin - dy = copy(dx) - lmul!(transpose(UpperTriangular(dA)), dy) - y = transpose(UpperTriangular(A)) * x - @test y ≈ Array(dy) - end - @testset "lmul!(::LowerTriangular)" begin - dy = copy(dx) - lmul!(LowerTriangular(dA), dy) - y = LowerTriangular(A) * x - @test y ≈ Array(dy) - end - @testset "lmul!(::LowerTriangular{Adjoint})" begin - dy = copy(dx) - lmul!(adjoint(LowerTriangular(dA)), dy) - y = adjoint(LowerTriangular(A)) * x - @test y ≈ Array(dy) - end - @testset "lmul!(::LowerTriangular{Transpose})" begin - dy = copy(dx) - lmul!(transpose(LowerTriangular(dA)), dy) - y = transpose(LowerTriangular(A)) * x - @test y ≈ Array(dy) - end - - @testset "trsv!" begin - d_y = copy(dx) - # execute trsv! - CUBLAS.trsv!('U','N','N',dA,d_y) - y = A\x - # compare - h_y = Array(d_y) - @test y ≈ h_y - @test_throws DimensionMismatch CUBLAS.trsv!('U','N','N',dA,CUDA.rand(elty,m+1)) - end - - @testset "trsv" begin - d_y = CUBLAS.trsv('U','N','N',dA,dx) - y = A\x - # compare - h_y = Array(d_y) - @test y ≈ h_y - end - @testset "trsv (adjoint)" begin - d_y = CUBLAS.trsv('U','C','N',dA,dx) - y = adjoint(A)\x - # compare - h_y = Array(d_y) - @test y ≈ h_y - end - @testset "trsv (transpose)" begin - d_y = CUBLAS.trsv('U','T','N',dA,dx) - y = transpose(A)\x - # compare - h_y = Array(d_y) - @test y ≈ h_y - end - - @testset "ldiv!(::UpperTriangular)" begin - A = copy(sA) - dA = CuArray(A) - dy = copy(dx) - ldiv!(UpperTriangular(dA), dy) - y = UpperTriangular(A) \ x - @test y ≈ Array(dy) - end - @testset "ldiv!(::UpperTriangular{Adjoint})" begin - A = copy(sA) - dA = CuArray(A) - dy = copy(dx) - ldiv!(adjoint(UpperTriangular(dA)), dy) - y = adjoint(UpperTriangular(A)) \ x - @test y ≈ Array(dy) - end - @testset "ldiv!(::UpperTriangular{Transpose})" begin - A = copy(sA) - dA = CuArray(A) - dy = copy(dx) - ldiv!(transpose(UpperTriangular(dA)), dy) - y = transpose(UpperTriangular(A)) \ x - @test y ≈ Array(dy) - end - @testset "ldiv!(::LowerTriangular)" begin - A = copy(sA) - dA = CuArray(A) - dy = copy(dx) - ldiv!(LowerTriangular(dA), dy) - y = LowerTriangular(A) \ x - @test y ≈ Array(dy) - end - @testset "ldiv!(::LowerTriangular{Adjoint})" begin - A = copy(sA) - dA = CuArray(A) - dy = copy(dx) - ldiv!(adjoint(LowerTriangular(dA)), dy) - y = adjoint(LowerTriangular(A)) \ x - @test y ≈ Array(dy) - end - @testset "ldiv!(::LowerTriangular{Transpose})" begin - A = copy(sA) - dA = CuArray(A) - dy = copy(dx) - ldiv!(transpose(LowerTriangular(dA)), dy) - y = transpose(LowerTriangular(A)) \ x - @test y ≈ Array(dy) - end - - @testset "inv($TR)" for TR in (UpperTriangular, LowerTriangular, UnitUpperTriangular, UnitLowerTriangular) - @test testf(x -> inv(TR(x)), rand(elty, m, m)) - end - - A = rand(elty,m,m) - x = rand(elty,m) - y = rand(elty,m) - dA = CuArray(A) - dx = CuArray(x) - dy = CuArray(y) - @testset "ger!" begin - # perform rank one update - dB = copy(dA) - CUBLAS.ger!(alpha,dx,dy,dB) - B = (alpha*x)*y' + A - # move to host and compare - hB = Array(dB) - @test B ≈ hB - end - - @testset "syr!" begin - dB = copy(dsA) - CUBLAS.syr!('U',alpha,dx,dB) - B = (alpha*x)*transpose(x) + sA - # move to host and compare upper triangles - hB = Array(dB) - B = triu(B) - hB = triu(hB) - @test B ≈ hB - end - if elty <: Complex - @testset "her!" begin - dB = copy(dhA) - # perform rank one update - CUBLAS.her!('U',real(alpha),dx,dB) - B = (real(alpha)*x)*x' + hA - # move to host and compare upper triangles - hB = Array(dB) - B = triu(B) - hB = triu(hB) - @test B ≈ hB - end - - @testset "her2!" begin - dB = copy(dhA) - CUBLAS.her2!('U',real(alpha),dx,dy,dB) - B = (real(alpha)*x)*y' + y*(real(alpha)*x)' + hA - # move to host and compare upper triangles - hB = Array(dB) - B = triu(B) - hB = triu(hB) - @test B ≈ hB - end - end - end - - @testset "gemv! with strided inputs" begin # JuliaGPU/CUDA.jl#445 - testf(rand(16), rand(4)) do p, b - W = @view p[reshape(1:(16),4,4)] - W*b - end - end - - @testset "StaticArray eltype" begin - A = CuArray(rand(SVector{2, Float64}, 3, 3)) - B = CuArray(rand(Float64, 3, 1)) - C = A * B - hC = Array(A) * Array(B) - @test Array(C) ≈ hC - end -end - -############################################################################################ - -@testset "level 3" begin - @testset for elty in [Float32, Float64, ComplexF32, ComplexF64] - alpha = rand(elty) - beta = rand(elty) - - @testset "mul! C = $f(A) * $g(B) * $Ts(a) + C * $Ts(b)" for f in (identity, transpose, adjoint), g in (identity, transpose, adjoint), Ts in (Int, elty) - C, A, B = rand(elty, 5, 5), rand(elty, 5, 5), rand(elty, 5, 5) - dC, dA, dB = CuArray(C), CuArray(A), CuArray(B) - mul!(dC, f(dA), g(dB), Ts(1), Ts(2)) - mul!(C, f(A), g(B), Ts(1), Ts(2)) - @test Array(dC) ≈ C - end - - @testset "hermitian" begin - C, A, B = rand(elty, 5, 5), Hermitian(rand(elty, 5, 5)), rand(elty, 5, 5) - dC, dA, dB = CuArray(C), Hermitian(CuArray(A)), CuArray(B) - mul!(dC, dA, dB) - mul!(C, A, B) - @test Array(dC) ≈ C - end - - A = rand(elty,m,k) - B = rand(elty,k,n) - Bbad = rand(elty,k+1,n+1) - C1 = rand(elty,m,n) - C2 = copy(C1) - d_A = CuArray(A) - d_B = CuArray(B) - d_Bbad = CuArray(Bbad) - d_C1 = CuArray(C1) - d_C2 = CuArray(C2) - hA = rand(elty,m,m) - hA = hA + hA' - dhA = CuArray(hA) - sA = rand(elty,m,m) - sA = sA + transpose(sA) - dsA = CuArray(sA) - @testset "gemm!" begin - CUBLAS.gemm!('N','N',alpha,d_A,d_B,beta,d_C1) - mul!(d_C2, d_A, d_B) - h_C1 = Array(d_C1) - h_C2 = Array(d_C2) - C1 = (alpha*A)*B + beta*C1 - C2 = A*B - # compare - @test C1 ≈ h_C1 - @test C2 ≈ h_C2 - @test_throws ArgumentError mul!(dhA, dhA, dsA) - @test_throws DimensionMismatch mul!(d_C1, d_A, dsA) - end - @testset "strided gemm!" begin - denseA = CUDA.rand(elty, 4,4) - denseB = CUDA.rand(elty, 4,4) - denseC = CUDA.zeros(elty, 4,4) - - stridedA = view(denseA, 1:2, 1:2)::SubArray - stridedB = view(denseB, 1:2, 1:2)::SubArray - stridedC = view(denseC, 1:2, 1:2)::SubArray - - CUBLAS.gemm!('N', 'N', true, stridedA, stridedB, false, stridedC) - @test Array(stridedC) ≈ Array(stridedA) * Array(stridedB) - - stridedC .= 0 - mul!(stridedC, stridedA, stridedB, true, false) - @test Array(stridedC) ≈ Array(stridedA) * Array(stridedB) - end - if capability(device()) > v"5.0" - @testset "gemmEx!" begin - α = rand(elty) - β = rand(elty) - CUBLAS.gemmEx!('N','N',α,d_A,d_B,β,d_C1) - h_C1 = Array(d_C1) - C1 = (α*A)*B + β*C1 - # compare - @test C1 ≈ h_C1 - end - end - @testset "gemm" begin - d_C = CUBLAS.gemm('N','N',d_A,d_B) - C = A*B - C2 = d_A * d_B - # compare - h_C = Array(d_C) - h_C2 = Array(C2) - @test C ≈ h_C - @test C ≈ h_C2 - end - @testset "xt_gemm! gpu" begin - synchronize() - CUBLAS.xt_gemm!('N','N',alpha,d_A,d_B,beta,d_C1) - mul!(d_C2, d_A, d_B) - h_C1 = Array(d_C1) - h_C2 = Array(d_C2) - C1 = (alpha*A)*B + beta*C1 - C2 = A*B - # compare - @test C1 ≈ h_C1 - @test C2 ≈ h_C2 - @test_throws DimensionMismatch CUBLAS.xt_gemm!('N','N',alpha,d_A,d_Bbad,beta,d_C1) - end - @testset "xt_gemm! cpu" begin - h_C1 = Array(d_C1) - CUBLAS.xt_gemm!('N','N',alpha,Array(d_A),Array(d_B),beta,h_C1) - mul!(d_C2, d_A, d_B) - h_C2 = Array(d_C2) - C1 = (alpha*A)*B + beta*C1 - C2 = A*B - # compare - @test C1 ≈ h_C1 - @test C2 ≈ h_C2 - end - - @testset "xt_gemm gpu" begin - synchronize() - d_C = CUBLAS.xt_gemm('N','N',d_A,d_B) - C = A*B - C2 = d_A * d_B - # compare - @test d_C isa CuArray - h_C = Array(d_C) - h_C2 = Array(C2) - @test C ≈ h_C - @test C ≈ h_C2 - end - @testset "xt_gemm cpu" begin - h_C = CUBLAS.xt_gemm('N','N',Array(d_A),Array(d_B)) - C = A*B - C2 = d_A * d_B - # compare - @test h_C isa Array - h_C2 = Array(C2) - @test C ≈ h_C - @test C ≈ h_C2 - end - - B = rand(elty,m,n) - C = rand(elty,m,n) - Bbad = rand(elty,m+1,n+1) - d_B = CuArray(B) - d_C = CuArray(C) - d_Bbad = CuArray(Bbad) - @testset "symm!" begin - CUBLAS.symm!('L','U',alpha,dsA,d_B,beta,d_C) - C = (alpha*sA)*B + beta*C - # compare - h_C = Array(d_C) - @test C ≈ h_C - @test_throws DimensionMismatch CUBLAS.symm!('L','U',alpha,dsA,d_Bbad,beta,d_C) - end - - @testset "symm" begin - d_C = CUBLAS.symm('L','U',dsA,d_B) - C = sA*B - # compare - h_C = Array(d_C) - @test C ≈ h_C - @test_throws DimensionMismatch CUBLAS.symm('L','U',dsA,d_Bbad) - end - @testset "xt_symm! gpu" begin - synchronize() - CUBLAS.xt_symm!('L','U',alpha,dsA,d_B,beta,d_C) - C = (alpha*sA)*B + beta*C - # compare - h_C = Array(d_C) - @test C ≈ h_C - end - @testset "xt_symm! cpu" begin - h_C = Array(d_C) - CUBLAS.xt_symm!('L','U',alpha,Array(dsA),Array(d_B),beta,h_C) - C = (alpha*sA)*B + beta*C - # compare - @test C ≈ h_C - end - - @testset "xt_symm gpu" begin - synchronize() - d_C = CUBLAS.xt_symm('L','U',dsA,d_B) - C = sA*B - # compare - @test d_C isa CuArray - h_C = Array(d_C) - @test C ≈ h_C - end - @testset "xt_symm cpu" begin - h_C = CUBLAS.xt_symm('L','U',Array(dsA),Array(d_B)) - C = sA*B - # compare - @test h_C isa Array - @test C ≈ h_C - end - A = triu(rand(elty, m, m)) - B = rand(elty,m,n) - C = zeros(elty,m,n) - dA = CuArray(A) - dB = CuArray(B) - dC = CuArray(C) - @testset "trmm!" begin - C = alpha*A*B - CUBLAS.trmm!('L','U','N','N',alpha,dA,dB,dC) - # move to host and compare - h_C = Array(dC) - @test C ≈ h_C - end - @testset "trmm" begin - C = alpha*A*B - d_C = CUBLAS.trmm('L','U','N','N',alpha,dA,dB) - # move to host and compare - h_C = Array(d_C) - @test C ≈ h_C - end - @testset "xt_trmm! gpu" begin - C = alpha*A*B - synchronize() - CUBLAS.xt_trmm!('L','U','N','N',alpha,dA,dB,dC) - # move to host and compare - h_C = Array(dC) - @test C ≈ h_C - end - @testset "xt_trmm! cpu" begin - C = alpha*A*B - h_C = Array(dC) - CUBLAS.xt_trmm!('L','U','N','N',alpha,Array(dA),Array(dB),h_C) - @test C ≈ h_C - end - @testset "xt_trmm gpu" begin - C = alpha*A*B - synchronize() - d_C = CUBLAS.xt_trmm('L','U','N','N',alpha,dA,dB) - # move to host and compare - @test d_C isa CuArray - h_C = Array(d_C) - @test C ≈ h_C - end - @testset "xt_trmm cpu" begin - C = alpha*A*B - h_C = CUBLAS.xt_trmm('L','U','N','N',alpha,Array(dA),Array(dB)) - @test h_C isa Array - @test C ≈ h_C - end - - @testset "xt_trsm! gpu" begin - C = alpha*(A\B) - dC = copy(dB) - synchronize() - CUBLAS.xt_trsm!('L','U','N','N',alpha,dA,dC) - # move to host and compare - h_C = Array(dC) - @test C ≈ h_C - end - @testset "xt_trsm! cpu" begin - C = alpha*(A\B) - dC = copy(dB) - h_C = Array(dC) - CUBLAS.xt_trsm!('L','U','N','N',alpha,Array(dA),h_C) - @test C ≈ h_C - end - @testset "xt_trsm gpu" begin - C = alpha*(A\B) - synchronize() - dC = CUBLAS.xt_trsm('L','U','N','N',alpha,dA,dB) - # move to host and compare - @test dC isa CuArray - h_C = Array(dC) - @test C ≈ h_C - end - @testset "xt_trsm cpu" begin - C = alpha*(A\B) - h_C = CUBLAS.xt_trsm('L','U','N','N',alpha,Array(dA),Array(dB)) - @test h_C isa Array - @test C ≈ h_C - end - @testset "trsm" begin - Br = rand(elty,m,n) - Bl = rand(elty,n,m) - d_Br = CuArray(Br) - d_Bl = CuArray(Bl) - # compute - @testset "adjtype=$adjtype, uplotype=$uplotype" for - adjtype in (identity, adjoint, transpose), - uplotype in (UpperTriangular, UnitUpperTriangular, LowerTriangular, UnitLowerTriangular) - - @test adjtype(uplotype(A))\Br ≈ Array(adjtype(uplotype(dA))\d_Br) - @test Bl/adjtype(uplotype(A)) ≈ Array(d_Bl/adjtype(uplotype(dA))) - end - # Check also that scaling parameter works - @test BLAS.trsm('L','U','N','N',alpha,A,Br) ≈ Array(CUBLAS.trsm('L','U','N','N',alpha,dA,d_Br)) - end - - @testset "trsm_batched!" begin - bA = [rand(elty,m,m) for i in 1:10] - map!((x) -> triu(x), bA, bA) - bB = [rand(elty,m,n) for i in 1:10] - bBbad = [rand(elty,m,n) for i in 1:9] - # move to device - bd_A = CuArray{elty, 2}[] - bd_B = CuArray{elty, 2}[] - bd_Bbad = CuArray{elty, 2}[] - for i in 1:length(bA) - push!(bd_A,CuArray(bA[i])) - push!(bd_B,CuArray(bB[i])) - end - for i in 1:length(bBbad) - push!(bd_Bbad,CuArray(bBbad[i])) - end - # compute - CUBLAS.trsm_batched!('L','U','N','N',alpha,bd_A,bd_B) - @test_throws DimensionMismatch CUBLAS.trsm_batched!('L','U','N','N',alpha,bd_A,bd_Bbad) - # move to host and compare - for i in 1:length(bd_B) - bC = alpha*(bA[i]\bB[i]) - h_C = Array(bd_B[i]) - #compare - @test bC ≈ h_C - end - end - - @testset "trsm_batched" begin - # generate parameter alpha = rand(elty) - # generate matrices - bA = [rand(elty,m,m) for i in 1:10] - map!((x) -> triu(x), bA, bA) - bB = [rand(elty,m,n) for i in 1:10] - # move to device - bd_A = CuArray{elty, 2}[] - bd_B = CuArray{elty, 2}[] - for i in 1:length(bA) - push!(bd_A,CuArray(bA[i])) - push!(bd_B,CuArray(bB[i])) - end - # compute - bd_C = CUBLAS.trsm_batched('L','U','N','N',alpha,bd_A,bd_B) - # move to host and compare - for i in 1:length(bd_C) - bC = alpha*(bA[i]\bB[i]) - h_C = Array(bd_C[i]) - @test bC ≈ h_C - end - end - - let A = triu(rand(elty, m, m)), B = rand(elty,m,n), alpha = rand(elty) - dA = CuArray(A) - dB = CuArray(B) - - @testset "left trsm!" begin - C = alpha*(A\B) - dC = copy(dB) - CUBLAS.trsm!('L','U','N','N',alpha,dA,dC) - @test C ≈ Array(dC) - end - - @testset "left trsm" begin - C = alpha*(A\B) - dC = CUBLAS.trsm('L','U','N','N',alpha,dA,dB) - @test C ≈ Array(dC) - end - @testset "left trsm (adjoint)" begin - C = alpha*(adjoint(A)\B) - dC = CUBLAS.trsm('L','U','C','N',alpha,dA,dB) - @test C ≈ Array(dC) - end - @testset "left trsm (transpose)" begin - C = alpha*(transpose(A)\B) - dC = CUBLAS.trsm('L','U','T','N',alpha,dA,dB) - @test C ≈ Array(dC) - end - end - - @testset "triangular ldiv!" begin - A = triu(rand(elty, m, m)) - B = rand(elty, m,m) - - dA = CuArray(A) - dB = CuArray(B) - - for t in (identity, transpose, adjoint), TR in (UpperTriangular, LowerTriangular, UnitUpperTriangular, UnitLowerTriangular) - dC = copy(dB) - ldiv!(t(TR(dA)), dC) - C = t(TR(A)) \ B - @test C ≈ Array(dC) - end - end - - let A = rand(elty, m,m), B = triu(rand(elty, m, m)), alpha = rand(elty) - dA = CuArray(A) - dB = CuArray(B) - - @testset "right trsm!" begin - C = alpha*(A/B) - dC = copy(dA) - CUBLAS.trsm!('R','U','N','N',alpha,dB,dC) - @test C ≈ Array(dC) - end - - @testset "right trsm" begin - C = alpha*(A/B) - dC = CUBLAS.trsm('R','U','N','N',alpha,dB,dA) - @test C ≈ Array(dC) - end - @testset "right trsm (adjoint)" begin - C = alpha*(A/adjoint(B)) - dC = CUBLAS.trsm('R','U','C','N',alpha,dB,dA) - @test C ≈ Array(dC) - end - @testset "right trsm (transpose)" begin - C = alpha*(A/transpose(B)) - dC = CUBLAS.trsm('R','U','T','N',alpha,dB,dA) - @test C ≈ Array(dC) - end - end - - @testset "triangular rdiv!" begin - A = rand(elty, m,m) - B = triu(rand(elty, m, m)) - - dA = CuArray(A) - dB = CuArray(B) - - for t in (identity, transpose, adjoint), TR in (UpperTriangular, LowerTriangular, UnitUpperTriangular, UnitLowerTriangular) - dC = copy(dA) - rdiv!(dC, t(TR(dB))) - C = A / t(TR(B)) - @test C ≈ Array(dC) - end - end - - @testset "Diagonal rdiv!" begin - A = rand(elty, m,m) - B = Diagonal(rand(elty, m)) - - dA = CuArray(A) - dB = CuArray(B) - - C = A / B - rdiv!(dA, dB) - @test C ≈ Array(dA) - end - - @testset "triangular-dense mul!" begin - A = triu(rand(elty, m, m)) - B = rand(elty,m,n) - C = zeros(elty,m,n) - - sA = rand(elty,m,m) - sA = sA + transpose(sA) - - for t in (identity, transpose, adjoint), TR in (UpperTriangular, LowerTriangular, UnitUpperTriangular, UnitLowerTriangular) - A = copy(sA) |> TR - B_L = copy(B) - B_R = copy(B') - C_L = copy(C) - C_R = copy(C') - dA = CuArray(parent(A)) |> TR - dB_L = CuArray(parent(B_L)) - dB_R = CuArray(parent(B_R)) - dC_L = CuArray(C_L) - dC_R = CuArray(C_R) - - D_L = mul!(C_L, t(A), B_L) - dD_L = mul!(dC_L, t(dA), dB_L) - - D_R = mul!(C_R, B_R, t(A)) - dD_R = mul!(dC_R, dB_R, t(dA)) - - @test C_L ≈ Array(dC_L) - @test D_L ≈ Array(dD_L) - @test C_R ≈ Array(dC_R) - @test D_R ≈ Array(dD_R) - end - end - - @testset "triangular-triangular mul!" begin - A = triu(rand(elty, m, m)) - B = triu(rand(elty, m, m)) - C0 = zeros(elty,m,m) - - sA = rand(elty,m,m) - sA = sA + transpose(sA) - sB = rand(elty,m,m) - sB = sB + transpose(sB) - - for (TRa, ta, TRb, tb, TRc) in ( - (UpperTriangular, identity, LowerTriangular, identity, Matrix), - (LowerTriangular, identity, UpperTriangular, identity, Matrix), - (UpperTriangular, identity, UpperTriangular, transpose, Matrix), - (UpperTriangular, transpose, UpperTriangular, identity, Matrix), - (LowerTriangular, identity, LowerTriangular, transpose, Matrix), - (LowerTriangular, transpose, LowerTriangular, identity, Matrix), - ) - - A = copy(sA) |> TRa - B = copy(sB) |> TRb - C = copy(C0) |> TRc - dA = CuArray(parent(sA)) |> TRa - dB = CuArray(parent(sB)) |> TRb - dC = if TRc == Matrix - CuArray(C0) |> DenseCuMatrix - else - CuArray(C0) |> TRc - end - - D = mul!(C, ta(A), tb(B)) - dD = mul!(dC, ta(dA), tb(dB)) - - @test C ≈ Array(dC) - @test D ≈ Array(dD) - end - end - - B = rand(elty,m,n) - C = rand(elty,m,n) - d_B = CuArray(B) - d_C = CuArray(C) - if elty <: Complex - @testset "hemm!" begin - # compute - C = alpha*(hA*B) + beta*C - CUBLAS.hemm!('L','L',alpha,dhA,d_B,beta,d_C) - # move to host and compare - h_C = Array(d_C) - @test C ≈ h_C - end - @testset "hemm" begin - C = hA*B - d_C = CUBLAS.hemm('L','U',dhA,d_B) - # move to host and compare - h_C = Array(d_C) - @test C ≈ h_C - end - @testset "xt_hemm! gpu" begin - # compute - C = alpha*(hA*B) + beta*C - synchronize() - CUBLAS.xt_hemm!('L','L',alpha,dhA,d_B,beta,d_C) - # move to host and compare - h_C = Array(d_C) - @test C ≈ h_C - end - @testset "xt_hemm! cpu" begin - # compute - C = alpha*(hA*B) + beta*C - h_C = Array(d_C) - CUBLAS.xt_hemm!('L','L',alpha,Array(dhA),Array(d_B),beta,h_C) - @test C ≈ h_C - end - @testset "xt_hemm gpu" begin - C = hA*B - synchronize() - d_C = CUBLAS.xt_hemm('L','U',dhA, d_B) - # move to host and compare - @test d_C isa CuArray - h_C = Array(d_C) - @test C ≈ h_C - end - @testset "xt_hemm cpu" begin - C = hA*B - h_C = CUBLAS.xt_hemm('L','U',Array(dhA), Array(d_B)) - # move to host and compare - @test h_C isa Array - @test C ≈ h_C - end - end - A = rand(elty,m,n) - d_A = CuArray(A) - @testset "geam!" begin - # compute - D = alpha*A + beta*B - CUBLAS.geam!('N','N',alpha,d_A,beta,d_B,d_C) - # move to host and compare - h_C = Array(d_C) - @test D ≈ h_C - - #test in place versions too - d_C = CuArray(C) - D = alpha*C + beta*B - CUBLAS.geam!('N','N',alpha,d_C,beta,d_B,d_C) - # move to host and compare - h_C = Array(d_C) - @test D ≈ h_C - - d_C = CuArray(C) - D = alpha*A + beta*C - CUBLAS.geam!('N','N',alpha,d_A,beta,d_C,d_C) - # move to host and compare - h_C = Array(d_C) - @test D ≈ h_C - - #test setting C to zero - CUBLAS.geam!('N','N',zero(elty),d_A,zero(elty),d_B,d_C) - h_C = Array(d_C) - @test h_C ≈ zeros(elty,m,n) - - # bounds checking - @test_throws DimensionMismatch CUBLAS.geam!('N','T',alpha,d_A,beta,d_B,d_C) - @test_throws DimensionMismatch CUBLAS.geam!('T','T',alpha,d_A,beta,d_B,d_C) - @test_throws DimensionMismatch CUBLAS.geam!('T','N',alpha,d_A,beta,d_B,d_C) - end - - @testset "geam" begin - D = alpha*A + beta*B - d_C = CUBLAS.geam('N','N',alpha,d_A,beta,d_B) - # move to host and compare - h_C = Array(d_C) - @test D ≈ h_C - end - @testset "CuMatrix -- A ± B -- $elty" begin - for opa in (identity, transpose, adjoint) - for opb in (identity, transpose, adjoint) - n = 10 - m = 20 - geam_A = opa == identity ? rand(elty, n, m) : rand(elty, m, n) - geam_B = opb == identity ? rand(elty, n, m) : rand(elty, m, n) - - geam_dA = CuMatrix{elty}(geam_A) - geam_dB = CuMatrix{elty}(geam_B) - - geam_C = opa(geam_A) + opb(geam_B) - geam_dC = opa(geam_dA) + opb(geam_dB) - @test geam_C ≈ collect(geam_dC) - - geam_C = opa(geam_A) - opb(geam_B) - geam_dC = opa(geam_dA) - opb(geam_dB) - @test geam_C ≈ collect(geam_dC) - end - end - end - A = rand(elty,m,k) - d_A = CuArray(A) - @testset "syrkx!" begin - # generate matrices - syrkx_A = rand(elty, n, k) - syrkx_B = rand(elty, n, k) - syrkx_C = rand(elty, n, n) - syrkx_C += syrkx_C' - d_syrkx_A = CuArray(syrkx_A) - d_syrkx_B = CuArray(syrkx_B) - d_syrkx_C = CuArray(syrkx_C) - # C = (alpha*A)*transpose(B) + beta*C - d_syrkx_C = CUBLAS.syrkx!('U','N',alpha,d_syrkx_A,d_syrkx_B,beta,d_syrkx_C) - final_C = (alpha*syrkx_A)*transpose(syrkx_B) + beta*syrkx_C - # move to host and compare - h_C = Array(d_syrkx_C) - @test triu(final_C) ≈ triu(h_C) - # C = A*transpose(B) - d_syrkx_C = CUBLAS.syrkx('U','N',d_syrkx_A,d_syrkx_B) - final_C = syrkx_A*transpose(syrkx_B) - # move to host and compare - h_C = Array(d_syrkx_C) - @test triu(final_C) ≈ triu(h_C) - badC = rand(elty, m, n) - d_badC = CuArray(badC) - @test_throws DimensionMismatch CUBLAS.syrkx!('U','N',alpha,d_syrkx_A,d_syrkx_B,beta,d_badC) - badC = rand(elty, n+1, n+1) - d_badC = CuArray(badC) - @test_throws DimensionMismatch CUBLAS.syrkx!('U','N',alpha,d_syrkx_A,d_syrkx_B,beta,d_badC) - end - @testset "xt_syrkx! gpu" begin - # generate matrices - syrkx_A = rand(elty, n, k) - syrkx_B = rand(elty, n, k) - syrkx_C = rand(elty, n, n) - syrkx_C += syrkx_C' - d_syrkx_A = CuArray(syrkx_A) - d_syrkx_B = CuArray(syrkx_B) - d_syrkx_C = CuArray(syrkx_C) - # C = (alpha*A)*transpose(B) + beta*C - synchronize() - d_syrkx_C = CUBLAS.xt_syrkx!('U','N',alpha,d_syrkx_A,d_syrkx_B,beta,d_syrkx_C) - final_C = (alpha*syrkx_A)*transpose(syrkx_B) + beta*syrkx_C - # move to host and compare - h_C = Array(d_syrkx_C) - @test triu(final_C) ≈ triu(h_C) - badC = rand(elty, m, n) - d_badC = CuArray(badC) - @test_throws DimensionMismatch CUBLAS.xt_syrkx!('U','N',alpha,d_syrkx_A,d_syrkx_B,beta,d_badC) - badC = rand(elty, n+1, n+1) - d_badC = CuArray(badC) - @test_throws DimensionMismatch CUBLAS.xt_syrkx!('U','N',alpha,d_syrkx_A,d_syrkx_B,beta,d_badC) - end - @testset "xt_syrkx! cpu" begin - # generate matrices - syrkx_A = rand(elty, n, k) - syrkx_B = rand(elty, n, k) - syrkx_C = rand(elty, n, n) - syrkx_C += syrkx_C' - final_C = (alpha*syrkx_A)*transpose(syrkx_B) + beta*syrkx_C - CUBLAS.xt_syrkx!('U','N',alpha,syrkx_A,syrkx_B,beta,syrkx_C) - # move to host and compare - @test triu(final_C) ≈ triu(syrkx_C) - end - @testset "xt_syrkx gpu" begin - # generate matrices - syrkx_A = rand(elty, n, k) - syrkx_B = rand(elty, n, k) - d_syrkx_A = CuArray(syrkx_A) - d_syrkx_B = CuArray(syrkx_B) - synchronize() - d_syrkx_C = CUBLAS.xt_syrkx('U','N',d_syrkx_A,d_syrkx_B) - final_C = syrkx_A*transpose(syrkx_B) - # move to host and compare - @test d_syrkx_C isa CuArray - h_C = Array(d_syrkx_C) - @test triu(final_C) ≈ triu(h_C) - end - @testset "xt_syrkx cpu" begin - # generate matrices - syrkx_A = rand(elty, n, k) - syrkx_B = rand(elty, n, k) - h_C = CUBLAS.xt_syrkx('U','N',syrkx_A,syrkx_B) - final_C = syrkx_A*transpose(syrkx_B) - @test h_C isa Array - @test triu(final_C) ≈ triu(h_C) - end - @testset "syrk" begin - # C = A*transpose(A) - d_C = CUBLAS.syrk('U','N',d_A) - C = A*transpose(A) - C = triu(C) - # move to host and compare - h_C = Array(d_C) - h_C = triu(C) - @test C ≈ h_C - end - @testset "xt_syrk gpu" begin - # C = A*transpose(A) - synchronize() - d_C = CUBLAS.xt_syrk('U','N',d_A) - C = A*transpose(A) - C = triu(C) - # move to host and compare - @test d_C isa CuArray - h_C = Array(d_C) - h_C = triu(C) - @test C ≈ h_C - end - @testset "xt_syrk cpu" begin - # C = A*transpose(A) - h_C = CUBLAS.xt_syrk('U','N',Array(d_A)) - C = A*transpose(A) - C = triu(C) - # move to host and compare - @test h_C isa Array - h_C = triu(C) - @test C ≈ h_C - end - if elty <: Complex - @testset "herk!" begin - d_C = CuArray(dhA) - CUBLAS.herk!('U','N',real(alpha),d_A,real(beta),d_C) - C = real(alpha)*(A*A') + real(beta)*hA - C = triu(C) - # move to host and compare - h_C = Array(d_C) - h_C = triu(C) - @test C ≈ h_C - end - @testset "herk" begin - d_C = CUBLAS.herk('U','N',d_A) - C = A*A' - C = triu(C) - # move to host and compare - h_C = Array(d_C) - h_C = triu(C) - @test C ≈ h_C - end - @testset "xt_herk! gpu" begin - d_C = CuArray(dhA) - C = real(alpha)*(A*A') + real(beta)*Array(d_C) - synchronize() - CUBLAS.xt_herk!('U','N',real(alpha),d_A,real(beta),d_C) - C = triu(C) - # move to host and compare - h_C = Array(d_C) - h_C = triu(h_C) - @test C ≈ h_C - end - @testset "xt_herk! cpu" begin - h_C = Array(dhA) - CUBLAS.xt_herk!('U','N',real(alpha),Array(d_A),real(beta),h_C) - C = real(alpha)*(A*A') + real(beta)*Array(dhA) - C = triu(C) - # move to host and compare - h_C = triu(h_C) - @test C ≈ h_C - end - @testset "xt_herk gpu" begin - synchronize() - d_C = CUBLAS.xt_herk('U','N',d_A) - C = A*A' - C = triu(C) - # move to host and compare - @test d_C isa CuArray - h_C = Array(d_C) - h_C = triu(h_C) - @test C ≈ h_C - end - @testset "xt_herk cpu" begin - h_C = CUBLAS.xt_herk('U','N',Array(d_A)) - C = A*A' - C = triu(C) - # move to host and compare - @test h_C isa Array - h_C = triu(h_C) - @test C ≈ h_C - end - end - A = rand(elty,m,k) - B = rand(elty,m,k) - Bbad = rand(elty,m+1,k+1) - C = rand(elty,m,m) - C = C + transpose(C) - # move to device - d_A = CuArray(A) - d_B = CuArray(B) - d_Bbad = CuArray(Bbad) - d_C = CuArray(C) - @testset "syr2k!" begin - # compute - C = alpha*(A*transpose(B) + B*transpose(A)) + beta*C - CUBLAS.syr2k!('U','N',alpha,d_A,d_B,beta,d_C) - # move back to host and compare - C = triu(C) - h_C = Array(d_C) - h_C = triu(h_C) - @test C ≈ h_C - @test_throws DimensionMismatch CUBLAS.syr2k!('U','N',alpha,d_A,d_Bbad,beta,d_C) - end - - @testset "syr2k" begin - C = alpha*(A*transpose(B) + B*transpose(A)) - d_C = CUBLAS.syr2k('U','N',alpha,d_A,d_B) - # move back to host and compare - C = triu(C) - h_C = Array(d_C) - h_C = triu(h_C) - @test C ≈ h_C - end - if elty <: Complex - @testset "her2k!" begin - elty1 = elty - elty2 = real(elty) - # generate parameters - α = rand(elty1) - β = rand(elty2) - C = C + C' - d_C = CuArray(C) - C = α*(A*B') + conj(α)*(B*A') + β*C - CUBLAS.her2k!('U','N',α,d_A,d_B,β,d_C) - # move back to host and compare - C = triu(C) - h_C = Array(d_C) - h_C = triu(h_C) - @test C ≈ h_C - @test_throws DimensionMismatch CUBLAS.her2k!('U','N',α,d_A,d_Bbad,β,d_C) - end - - @testset "her2k" begin - C = A*B' + B*A' - d_C = CUBLAS.her2k('U','N',d_A,d_B) - # move back to host and compare - C = triu(C) - h_C = Array(d_C) - h_C = triu(h_C) - @test C ≈ h_C - end - @testset "xt_her2k! gpu" begin - elty1 = elty - elty2 = real(elty) - # generate parameters - α = rand(elty1) - β = rand(elty2) - C = C + C' - d_C = CuArray(C) - C = α*(A*B') + conj(α)*(B*A') + β*C - synchronize() - CUBLAS.xt_her2k!('U','N',α,d_A,d_B,β,d_C) - # move back to host and compare - C = triu(C) - h_C = Array(d_C) - h_C = triu(h_C) - @test C ≈ h_C - end - @testset "xt_her2k! cpu" begin - elty1 = elty - elty2 = real(elty) - # generate parameters - α = rand(elty1) - β = rand(elty2) - C = C + C' - h_C = copy(C) - C = α*(A*B') + conj(α)*(B*A') + β*C - CUBLAS.xt_her2k!('U','N',α,A,B,β,h_C) - # move back to host and compare - C = triu(C) - h_C = triu(h_C) - @test C ≈ h_C - end - @testset "xt_her2k gpu" begin - # generate parameters - C = C + C' - C = (A*B') + (B*A') - synchronize() - d_C = CUBLAS.xt_her2k('U','N',d_A,d_B) - # move back to host and compare - C = triu(C) - @test d_C isa CuArray - h_C = Array(d_C) - h_C = triu(h_C) - @test C ≈ h_C - end - @testset "xt_her2k cpu" begin - # generate parameters - C = C + C' - C = (A*B') + (B*A') - h_C = CUBLAS.xt_her2k('U','N',A,B) - # move back to host and compare - @test h_C isa Array - C = triu(C) - h_C = triu(h_C) - @test C ≈ h_C - end - @testset "her2k" begin - C = A*B' + B*A' - d_C = CUBLAS.her2k('U','N',d_A,d_B) - # move back to host and compare - C = triu(C) - h_C = Array(d_C) - h_C = triu(h_C) - @test C ≈ h_C - end - end - end - - @testset "elty = $elty" for elty in [Float16, Float32, Float64, ComplexF32, ComplexF64] - elty == Float16 && capability(device()) < v"5.3" && continue - - alpha = rand(elty) - beta = rand(elty) - # generate matrices - bA = [rand(elty,m,k) for i in 1:10] - bB = [rand(elty,k,n) for i in 1:10] - bC = [rand(elty,m,n) for i in 1:10] - # move to device - bd_A = CuArray{elty, 2}[] - bd_B = CuArray{elty, 2}[] - bd_C = CuArray{elty, 2}[] - bd_bad = CuArray{elty, 2}[] - for i in 1:length(bA) - push!(bd_A,CuArray(bA[i])) - push!(bd_B,CuArray(bB[i])) - push!(bd_C,CuArray(bC[i])) - if i < length(bA) - 2 - push!(bd_bad,CuArray(bC[i])) - end - end - - @testset "gemm_batched!" begin - # C = (alpha*A)*B + beta*C - CUBLAS.gemm_batched!('N','N',alpha,bd_A,bd_B,beta,bd_C) - for i in 1:length(bd_C) - bC[i] = (alpha*bA[i])*bB[i] + beta*bC[i] - h_C = Array(bd_C[i]) - #compare - @test bC[i] ≈ h_C - end - @test_throws DimensionMismatch CUBLAS.gemm_batched!('N','N',alpha,bd_A,bd_bad,beta,bd_C) - end - - @testset "gemm_batched" begin - bd_C = CUBLAS.gemm_batched('N','N',bd_A,bd_B) - for i in 1:length(bA) - bC[i] = bA[i]*bB[i] - h_C = Array(bd_C[i]) - @test bC[i] ≈ h_C - end - @test_throws DimensionMismatch CUBLAS.gemm_batched('N','N',alpha,bd_A,bd_bad) - end - - @testset "gemmBatchedEx!" begin - # C = (alpha*A)*B + beta*C - CUBLAS.gemmBatchedEx!('N','N',alpha,bd_A,bd_B,beta,bd_C) - for i in 1:length(bd_C) - bC[i] = (alpha*bA[i])*bB[i] + beta*bC[i] - h_C = Array(bd_C[i]) - #compare - @test bC[i] ≈ h_C - end - @test_throws DimensionMismatch CUBLAS.gemmBatchedEx!('N','N',alpha,bd_A,bd_bad,beta,bd_C) - end - - nbatch = 10 - bA = rand(elty, m, k, nbatch) - bB = rand(elty, k, n, nbatch) - bC = rand(elty, m, n, nbatch) - bbad = rand(elty, m+1, n+1, nbatch) - # move to device - bd_A = CuArray{elty, 3}(bA) - bd_B = CuArray{elty, 3}(bB) - bd_C = CuArray{elty, 3}(bC) - bd_bad = CuArray{elty, 3}(bbad) - - @testset "gemm_strided_batched!" begin - CUBLAS.gemm_strided_batched!('N', 'N', alpha, bd_A, bd_B, beta, bd_C) - for i in 1:nbatch - bC[:, :, i] = (alpha * bA[:, :, i]) * bB[:, :, i] + beta * bC[:, :, i] - end - h_C = Array(bd_C) - @test bC ≈ h_C - @test_throws DimensionMismatch CUBLAS.gemm_strided_batched!('N', 'N', alpha, bd_A, bd_B, beta, bd_bad) - end - - @testset "gemmStridedBatchedEx!" begin - CUBLAS.gemmStridedBatchedEx!('N', 'N', alpha, bd_A, bd_B, beta, bd_C) - for i in 1:nbatch - bC[:, :, i] = (alpha * bA[:, :, i]) * bB[:, :, i] + beta * bC[:, :, i] - end - h_C = Array(bd_C) - @test bC ≈ h_C - @test_throws DimensionMismatch CUBLAS.gemmStridedBatchedEx!('N', 'N', alpha, bd_A, bd_B, beta, bd_bad) - end - - @testset "gemm_strided_batched" begin - bd_C = CUBLAS.gemm_strided_batched('N', 'N', bd_A, bd_B) - - for i in 1:nbatch - bC[:, :, i] = bA[:, :, i] * bB[:, :, i] - end - h_C = Array(bd_C) - @test bC ≈ h_C - # generate matrices - bA = rand(elty, k, m, nbatch) - bB = rand(elty, k, n, nbatch) - bC = zeros(elty, m, n, nbatch) - # move to device - bd_A = CuArray{elty, 3}(bA) - bd_B = CuArray{elty, 3}(bB) - - bd_C = CUBLAS.gemm_strided_batched('T', 'N', bd_A, bd_B) - for i in 1:nbatch - bC[:, :, i] = transpose(bA[:, :, i]) * bB[:, :, i] - end - h_C = Array(bd_C) - @test bC ≈ h_C - @test_throws DimensionMismatch CUBLAS.gemm_strided_batched('N', 'N', alpha, bd_A, bd_bad) - end - end - - if CUDA.CUBLAS.version() >= v"12.4.2" - @testset "elty = $elty" for elty in [Float32, Float64] - num_groups = 10 - group_sizes = collect(1:num_groups) - transA = ['N' for i in 1:num_groups] - transB = ['N' for i in 1:num_groups] - alpha = rand(elty, num_groups) - beta = rand(elty, num_groups) - # generate matrices - bA = [[rand(elty,3*i,2*i) for j in 1:group_sizes[i]] for i in 1:num_groups] - bB = [[rand(elty,2*i,5*i) for j in 1:group_sizes[i]] for i in 1:num_groups] - bC = [[rand(elty,3*i,5*i) for j in 1:group_sizes[i]] for i in 1:num_groups] - # move to device - bd_A = [[CuArray(bA[i][j]) for j in 1:group_sizes[i]] for i in 1:num_groups] - bd_B = [[CuArray(bB[i][j]) for j in 1:group_sizes[i]] for i in 1:num_groups] - bd_C = [[CuArray(bC[i][j]) for j in 1:group_sizes[i]] for i in 1:num_groups] - @testset "gemm_grouped_batched!" begin - # C = (alpha*A)*B + beta*C - CUBLAS.gemm_grouped_batched!(transA,transB,alpha,bd_A,bd_B,beta,bd_C) - for i in 1:num_groups, j in 1:group_sizes[i] - bC[i][j] = alpha[i] * bA[i][j] * bB[i][j] + beta[i] * bC[i][j] - h_C = Array(bd_C[i][j]) - @test bC[i][j] ≈ h_C - end - end - - @testset "gemm_grouped_batched" begin - bd_C = CUBLAS.gemm_grouped_batched(transA,transB,bd_A,bd_B) - for i in 1:num_groups, j in 1:group_sizes[i] - bC[i][j] = bA[i][j] * bB[i][j] - h_C = Array(bd_C[i][j]) - @test bC[i][j] ≈ h_C - end - end - end - end - - # Group size hardcoded to one - if CUDA.CUBLAS.version() >= v"12.4.2" - @testset "elty = $elty" for elty in [Float32, Float64] - - transA = ['N' for i in 1:10] - transB = ['N' for i in 1:10] - alpha = rand(elty, 10) - beta = rand(elty, 10) - # generate matrices - bA = [rand(elty,3*i,2*i) for i in 1:10] - bB = [rand(elty,2*i,5*i) for i in 1:10] - bC = [rand(elty,3*i,5*i) for i in 1:10] - # move to device - bd_A = CuArray{elty, 2}[] - bd_B = CuArray{elty, 2}[] - bd_C = CuArray{elty, 2}[] - for i in 1:length(bA) - push!(bd_A,CuArray(bA[i])) - push!(bd_B,CuArray(bB[i])) - push!(bd_C,CuArray(bC[i])) - end - - @testset "gemm_grouped_batched!" begin - # C = (alpha*A)*B + beta*C - CUBLAS.gemm_grouped_batched!(transA,transB,alpha,bd_A,bd_B,beta,bd_C) - for i in 1:length(bd_C) - bC[i] = alpha[i] * bA[i] * bB[i] + beta[i] * bC[i] - h_C = Array(bd_C[i]) - @test bC[i] ≈ h_C - end - end - - @testset "gemm_grouped_batched" begin - bd_C = CUBLAS.gemm_grouped_batched(transA,transB,bd_A,bd_B) - for i in 1:length(bd_C) - bC[i] = bA[i] * bB[i] - h_C = Array(bd_C[i]) - @test bC[i] ≈ h_C - end - end - end - end - - @testset "mixed-precision matmul" begin - m,k,n = 4,4,4 - cudaTypes = (Float16, Complex{Float16}, BFloat16, Complex{BFloat16}, Float32, Complex{Float32}, - Float64, Complex{Float64}, Int8, Complex{Int8}, UInt8, Complex{UInt8}, - Int16, Complex{Int16}, UInt16, Complex{UInt16}, Int32, Complex{Int32}, - UInt32, Complex{UInt32}, Int64, Complex{Int64}, UInt64, Complex{UInt64}) - - for AT in cudaTypes, CT in cudaTypes - BT = AT # gemmEx requires identical A and B types - - # we only test combinations of types that are supported by gemmEx - if CUBLAS.gemmExComputeType(AT, BT, CT, m,k,n) !== nothing - A = AT <: BFloat16 ? AT.(rand(m,k)) : rand(AT, m,k) - B = BT <: BFloat16 ? BT.(rand(k,n)) : rand(BT, k,n) - C = similar(B, CT) - mul!(C, A, B) - - # Base can't do Int8*Int8 without losing accuracy - if (AT == Int8 && BT == Int8) || (AT == Complex{Int8} && BT == Complex{Int8}) - C = CT.(A) * CT.(B) - end - - dA = CuArray(A) - dB = CuArray(B) - dC = similar(dB, CT) - mul!(dC, dA, dB) - - rtol = Base.rtoldefault(AT, BT, 0) - @test C ≈ Array(dC) rtol=rtol - end - end - - # also test an unsupported combination (falling back to GPUArrays) - if VERSION < v"1.11-" # JuliaGPU/CUDA.jl#2441 - AT=BFloat16 - BT=Int32 - CT=Float64 - - A = AT.(rand(m,k)) - B = rand(BT, k,n) - C = similar(B, CT) - mul!(C, A, B) - - dA = CuArray(A) - dB = CuArray(B) - dC = similar(dB, CT) - mul!(dC, dA, dB) - - rtol = Base.rtoldefault(AT, BT, 0) - @test C ≈ Array(dC) rtol=rtol - end - end - - @testset "gemm! with strided inputs" begin # JuliaGPU/CUDA.jl#78 - inn = 784; out = 32 - testf(randn(784*100), rand(Float32, 784, 100)) do p, x - p[reshape(1:(out*inn),out,inn)] * x - @view(p[reshape(1:(out*inn),out,inn)]) * x - end - end -end - -############################################################################################ - -@testset "extensions" begin - @testset for elty in [Float32, Float64, ComplexF32, ComplexF64] - @testset "getrf_batched!" begin - Random.seed!(1) - local k - # generate matrices - A = [rand(elty,m,m) for i in 1:10] - # move to device - d_A = CuArray{elty, 2}[] - for i in 1:length(A) - push!(d_A,CuArray(A[i])) - end - # testing without pivoting quickly results in inaccuracies along the diagonal - # test with pivoting - pivot, info = CUBLAS.getrf_batched!(d_A, true) - h_info = Array(info) - h_pivot = Array(pivot) - for As in 1:length(d_A) - C = lu(A[As]) - h_A = Array(d_A[As]) - #reconstruct L,U - dL = Matrix(one(elty)*I, m, m) - dU = zeros(elty,(m,m)) - k = h_info[As] - if( k >= 0 ) - dL += tril(h_A,-k-1) - dU += triu(h_A,k) - end - #compare pivots - @test length(setdiff(h_pivot[:,As],C.p)) == 0 - #make device pivot matrix - P = Matrix(1.0*I, m, m) - for row in 1:m - temp = copy(P[row,:]) - P[row,:] = P[h_pivot[row,As],:] - P[h_pivot[row,As],:] = temp - end - @test inv(P)*dL*dU ≈ inv(C.P) * C.L * C.U - end - end - - @testset "getrf_batched" begin - local k - # generate matrices - A = [rand(elty,m,m) for i in 1:10] - # move to device - d_A = CuArray{elty, 2}[] - for i in 1:length(A) - push!(d_A,CuArray(A[i])) - end - # testing without pivoting quickly results in inaccuracies along the diagonal - # test with pivoting - pivot, info, d_B = CUBLAS.getrf_batched(d_A, true) - h_info = Array(info) - h_pivot = Array(pivot) - for Bs in 1:length(d_B) - C = lu(A[Bs]) - h_A = Array(d_B[Bs]) - #reconstruct L,U - dL = Matrix(one(elty)*I, m, m) - dU = zeros(elty,(m,m)) - k = h_info[Bs] - if( k >= 0 ) - dL += tril(h_A,-k-1) - dU += triu(h_A,k) - end - #compare pivots - @test length(setdiff(h_pivot[:,Bs],C.p)) == 0 - #make device pivot matrix - P = Matrix(1.0*I, m, m) - for row in 1:m - temp = copy(P[row,:]) - P[row,:] = P[h_pivot[row,Bs],:] - P[h_pivot[row,Bs],:] = temp - end - @test inv(P)*dL*dU ≈ inv(C.P) * C.L * C.U - end - end - - @testset "getrf_strided_batched!" begin - Random.seed!(1) - local k - # generate strided matrix - A = rand(elty,m,m,10) - # move to device - d_A = CuArray(A) - # testing without pivoting quickly results in inaccuracies along the diagonal - # test with pivoting - pivot, info = CUBLAS.getrf_strided_batched!(d_A, true) - h_info = Array(info) - h_pivot = Array(pivot) - for As in 1:size(d_A, 3) - C = lu(A[:,:,As]) - h_A = Array(d_A[:,:,As]) - #reconstruct L,U - dL = Matrix(one(elty)*I, m, m) - dU = zeros(elty,(m,m)) - k = h_info[As] - if( k >= 0 ) - dL += tril(h_A,-k-1) - dU += triu(h_A,k) - end - #compare pivots - @test length(setdiff(h_pivot[:,As],C.p)) == 0 - #make device pivot matrix - P = Matrix(1.0*I, m, m) - for row in 1:m - temp = copy(P[row,:]) - P[row,:] = P[h_pivot[row,As],:] - P[h_pivot[row,As],:] = temp - end - @test inv(P)*dL*dU ≈ inv(C.P) * C.L * C.U - end - end - - @testset "getrf_strided_batched" begin - local k - # generate strided matrix - A = rand(elty,m,m,10) - # move to device - d_A = CuArray(A) - # testing without pivoting quickly results in inaccuracies along the diagonal - # test with pivoting - pivot, info, d_B = CUBLAS.getrf_strided_batched(d_A, true) - h_info = Array(info) - h_pivot = Array(pivot) - for Bs in 1:size(d_B, 3) - C = lu(A[:,:,Bs]) - h_A = Array(d_B[:,:,Bs]) - #reconstruct L,U - dL = Matrix(one(elty)*I, m, m) - dU = zeros(elty,(m,m)) - k = h_info[Bs] - if( k >= 0 ) - dL += tril(h_A,-k-1) - dU += triu(h_A,k) - end - #compare pivots - @test length(setdiff(h_pivot[:,Bs],C.p)) == 0 - #make device pivot matrix - P = Matrix(1.0*I, m, m) - for row in 1:m - temp = copy(P[row,:]) - P[row,:] = P[h_pivot[row,Bs],:] - P[h_pivot[row,Bs],:] = temp - end - @test inv(P)*dL*dU ≈ inv(C.P) * C.L * C.U - end - end - - for (opchar,opfun) in (('N',identity), ('T',transpose), ('C',adjoint)) - - @testset "getrs_batched!" begin - A = [rand(elty,n,n) for _ in 1:k]; - d_A = [CuArray(a) for a in A]; - d_A2 = deepcopy(d_A); - d_pivot, info, d_LU = CUDA.CUBLAS.getrf_batched!(d_A, true); - @test d_LU == d_A - d_pivot2 = similar(d_pivot); - info2 = similar(info); - CUDA.CUBLAS.getrf_batched!(d_A2, d_pivot2, info2); - @test isapprox(d_pivot, d_pivot2) - @test isapprox(info, info2) - B = [rand(elty,n,m) for _ in 1:k]; - d_B = [CuArray(b) for b in B]; - info2, d_Bhat = CUDA.CUBLAS.getrs_batched!(opchar, d_LU, d_B, d_pivot); - @test d_Bhat == d_B - h_Bhat = [collect(bh) for bh in d_Bhat]; - for i in 1:k - @test h_Bhat[i] ≈ opfun(A[i]) \ B[i] - end - end - - @testset "getrs_batched" begin - A = [rand(elty,n,n) for _ in 1:k]; - d_A = [CuArray(a) for a in A]; - d_A2 = deepcopy(d_A); - d_pivot, info, d_LU = CUDA.CUBLAS.getrf_batched(d_A, true); - @test d_LU != d_A - d_pivot2 = similar(d_pivot); - info2 = similar(info); - CUDA.CUBLAS.getrf_batched(d_A2, d_pivot2, info2); - @test isapprox(d_pivot, d_pivot2) - @test isapprox(info, info2) - B = [rand(elty,n,m) for _ in 1:k]; - d_B = [CuArray(b) for b in B]; - info2, d_Bhat = CUDA.CUBLAS.getrs_batched(opchar, d_LU, d_B, d_pivot); - @test d_Bhat != d_B - h_Bhat = [collect(bh) for bh in d_Bhat]; - for i in 1:k - @test h_Bhat[i] ≈ opfun(A[i]) \ B[i] - end - end - - @testset "getrs_strided_batched!" begin - A = rand(elty,n,n,k); - d_A = CuArray(A); - d_A2 = copy(d_A); - d_pivot, info, d_LU = CUDA.CUBLAS.getrf_strided_batched!(d_A, true); - @test d_LU == d_A - d_pivot2 = similar(d_pivot); - info2 = similar(info); - CUDA.CUBLAS.getrf_strided_batched!(d_A2, d_pivot2, info2); - @test isapprox(d_pivot, d_pivot2) - @test isapprox(info, info2) - B = rand(elty,n,m,k); - d_B = CuArray(B); - info2, d_Bhat = CUDA.CUBLAS.getrs_strided_batched!(opchar, d_LU, d_B, d_pivot); - @test d_Bhat == d_B - h_Bhat = collect(d_Bhat); - for i in 1:k - @test h_Bhat[:,:,i] ≈ opfun(A[:,:,i]) \ B[:,:,i] - end - end - - @testset "getrs_strided_batched" begin - A = rand(elty,n,n,k); - d_A = CuArray(A); - d_A2 = copy(d_A); - d_pivot, info, d_LU = CUDA.CUBLAS.getrf_strided_batched(d_A, true); - @test d_LU != d_A - d_pivot2 = similar(d_pivot); - info2 = similar(info); - CUDA.CUBLAS.getrf_strided_batched(d_A2, d_pivot2, info2); - @test isapprox(d_pivot, d_pivot2) - @test isapprox(info, info2) - B = rand(elty,n,m,k); - d_B = CuArray(B); - info2, d_Bhat = CUDA.CUBLAS.getrs_strided_batched(opchar, d_LU, d_B, d_pivot); - @test d_Bhat != d_B - h_Bhat = collect(d_Bhat); - for i in 1:k - @test h_Bhat[:,:,i] ≈ opfun(A[:,:,i]) \ B[:,:,i] - end - end - - end - - @testset "getri_strided_batched" begin - # generate strided matrix - A = rand(elty,m,m,10) - # move to device - d_A = CuArray(A) - d_B = similar(d_A) - pivot, info = CUBLAS.getrf_strided_batched!(d_A, true) - info = CUBLAS.getri_strided_batched!(d_A, d_B, pivot) - h_info = Array(info) - for Cs in 1:size(d_A,3) - B = inv(A[:,:,Cs]) - @test h_info[Cs] == 0 - @test B ≈ Array(d_B[:,:,Cs]) rtol=1e-3 - end - end - - @testset "getri_batched" begin - # generate matrices - A = [rand(elty,m,m) for i in 1:10] - # move to device - d_A = CuArray{elty, 2}[] - for i in 1:length(A) - push!(d_A,CuArray(A[i])) - end - pivot, info = CUBLAS.getrf_batched!(d_A, true) - h_info = Array(info) - for Cs in 1:length(h_info) - @test h_info[Cs] == 0 - end - pivot, info, d_C = CUBLAS.getri_batched(d_A, pivot) - h_info = Array(info) - for Cs in 1:length(d_C) - C = inv(A[Cs]) - h_C = Array(d_C[Cs]) - @test h_info[Cs] == 0 - @test C ≈ h_C rtol=1e-2 - end - end - - @testset "matinv_batched" begin - # generate matrices - A = [rand(elty,m,m) for i in 1:10] - # move to device - d_A = CuArray{elty, 2}[] - for i in 1:length(A) - push!(d_A,CuArray(A[i])) - end - info, d_C = CUBLAS.matinv_batched(d_A) - for Cs in 1:length(d_C) - C = inv(A[Cs]) - h_C = Array(d_C[Cs]) - @test C ≈ h_C - end - push!(d_A, CUDA.rand(elty, m, m+1)) - @test_throws DimensionMismatch CUBLAS.matinv_batched(d_A) - end - - @testset "geqrf_batched!" begin - # generate matrices - A = [rand(elty,m,n) for i in 1:10] - # move to device - d_A = CuArray{elty, 2}[] - for i in 1:length(A) - push!(d_A,CuArray(A[i])) - end - tau, d_A = CUBLAS.geqrf_batched!(d_A) - for As in 1:length(d_A) - C = qr(A[As]) - h_A = Array(d_A[As]) - h_tau = Array(tau[As]) - # build up Q - Q = Matrix(one(elty)*I, min(m,n), min(m,n)) - for i in 1:min(m,n) - v = zeros(elty,m) - v[i] = one(elty) - v[i+1:m] = h_A[i+1:m,i] - Q *= I - h_tau[i] * v * v' - end - @test Q ≈ Array(C.Q) rtol=1e-2 - end - end - - @testset "geqrf_batched" begin - # generate matrices - A = [rand(elty,m,n) for i in 1:10] - # move to device - d_A = CuArray{elty, 2}[] - for i in 1:length(A) - push!(d_A,CuArray(A[i])) - end - tau, d_B = CUBLAS.geqrf_batched!(d_A) - for Bs in 1:length(d_B) - C = qr(A[Bs]) - h_B = Array(d_B[Bs]) - h_tau = Array(tau[Bs]) - # build up Q - Q = Matrix(one(elty)*I, min(m,n), min(m,n)) - for i in 1:min(m,n) - v = zeros(elty,m) - v[i] = one(elty) - v[i+1:m] = h_B[i+1:m,i] - Q *= I - h_tau[i] * v * v' - end - @test Q ≈ Array(C.Q) rtol=1e-2 - end - end - - @testset "gels_batched!" begin - # generate matrices - A = [rand(elty,n,k) for i in 1:10] - C = [rand(elty,n,k) for i in 1:10] - # move to device - d_A = CuArray{elty, 2}[] - d_C = CuArray{elty, 2}[] - for i in 1:length(A) - push!(d_A,CuArray(A[i])) - push!(d_C,CuArray(C[i])) - end - d_A, d_C, info = CUBLAS.gels_batched!('N',d_A, d_C) - for Cs in 1:length(d_C) - X = A[Cs]\C[Cs] - h_C = Array(d_C[Cs])[1:k,1:k] - @test X ≈ h_C rtol=1e-2 - end - push!(d_C,CUDA.rand(elty,n,k-1)) - @test_throws DimensionMismatch CUBLAS.gels_batched!('N',d_A, d_C) - push!(d_A,CUDA.rand(elty,n,k-1)) - @test_throws DimensionMismatch CUBLAS.gels_batched!('N',d_A, d_C) - A = [rand(elty,k-1,k) for i in 1:10] - C = [rand(elty,k-1,k) for i in 1:10] - # move to device - d_A = CuArray{elty, 2}[] - d_C = CuArray{elty, 2}[] - for i in 1:length(A) - push!(d_A,CuArray(A[i])) - push!(d_C,CuArray(C[i])) - end - # system is now not overdetermined - @test_throws ArgumentError CUBLAS.gels_batched!('N',d_A, d_C) - end - - @testset "gels_batched" begin - # generate matrices - A = [rand(elty,n,k) for i in 1:10] - C = [rand(elty,n,k) for i in 1:10] - # move to device - d_A = CuArray{elty, 2}[] - d_C = CuArray{elty, 2}[] - for i in 1:length(A) - push!(d_A,CuArray(A[i])) - push!(d_C,CuArray(C[i])) - end - d_B, d_D, info = CUBLAS.gels_batched('N',d_A, d_C) - for Ds in 1:length(d_D) - X = A[Ds]\C[Ds] - h_D = Array(d_D[Ds])[1:k,1:k] - @test X ≈ h_D rtol=1e-2 - end - end - # generate matrices - A = rand(elty,m,n) - C = rand(elty,m,n) - x = rand(elty,m) - y = rand(elty,n) - # move to device - d_A = CuArray(A) - d_C = CuArray(C) - d_x = CuArray(x) - d_y = CuArray(y) - C = diagm(0 => x) * A - @testset "dgmm!" begin - # compute - CUBLAS.dgmm!('L', d_A, d_x, d_C) - # move to host and compare - h_C = Array(d_C) - @test C ≈ h_C - # bounds checking - @test_throws DimensionMismatch CUBLAS.dgmm!('R', d_A, d_x, d_C) - E = rand(elty,m,m) - d_E = CuArray(E) - @test_throws DimensionMismatch CUBLAS.dgmm!('L', d_E, d_x, d_C) - end - @testset "dgmm" begin - d_C = CUBLAS.dgmm('L', d_A, d_x) - # move to host and compare - h_C = Array(d_C) - @test C ≈ h_C - end - @testset "diagonal -- mul!" begin - XA = rand(elty,m,n) - d_XA = CuArray(XA) - d_X = Diagonal(d_x) - mul!(d_XA, d_X, d_A) - Array(d_XA) ≈ Diagonal(x) * A - - AY = rand(elty,m,n) - d_AY = CuArray(AY) - d_Y = Diagonal(d_y) - mul!(d_AY, d_A, d_Y) - Array(d_AY) ≈ A * Diagonal(y) - - YA = rand(elty,n,m) - d_YA = CuArray(YA) - d_Y = Diagonal(d_y) - mul!(d_YA, d_Y, transpose(d_A)) - Array(d_YA) ≈ Diagonal(y) * transpose(A) - - AX = rand(elty,n,m) - d_AX = CuArray(AX) - d_X = Diagonal(d_x) - mul!(d_AX, transpose(d_A), d_X) - Array(d_AX) ≈ transpose(A) * Diagonal(x) - - YA = rand(elty,n,m) - d_YA = CuArray(YA) - d_Y = Diagonal(d_y) - mul!(d_YA, d_Y, d_A') - Array(d_YA) ≈ Diagonal(y) * A' - - AX = rand(elty,n,m) - d_AX = CuArray(AX) - d_X = Diagonal(d_x) - mul!(d_AX, d_A', d_X) - Array(d_AX) ≈ A' * Diagonal(x) - end - end # extensions - - @testset "StaticArray eltype" begin - A = CuArray(rand(SVector{2, Float32}, 3, 3)) - B = CuArray(rand(Float32, 3, 3)) - C = A * B - hC = Array(A) * Array(B) - @test Array(C) ≈ hC - end -end # elty diff --git a/test/libraries/cublas_extensions.jl b/test/libraries/cublas_extensions.jl new file mode 100644 index 0000000000..ccfd47f35d --- /dev/null +++ b/test/libraries/cublas_extensions.jl @@ -0,0 +1,490 @@ +using CUDA.CUBLAS +using CUDA.CUBLAS: band, bandex + +using LinearAlgebra + +using BFloat16s +using StaticArrays + +@test CUBLAS.version() isa VersionNumber +@test CUBLAS.version().major == CUBLAS.cublasGetProperty(CUDA.MAJOR_VERSION) +@test CUBLAS.version().minor == CUBLAS.cublasGetProperty(CUDA.MINOR_VERSION) +@test CUBLAS.version().patch == CUBLAS.cublasGetProperty(CUDA.PATCH_LEVEL) + +m = 20 +n = 35 +k = 13 + +@testset "extensions" begin + @testset for elty in [Float32, Float64, ComplexF32, ComplexF64] + @testset "getrf_batched!" begin + Random.seed!(1) + local k + # generate matrices + A = [rand(elty,m,m) for i in 1:10] + # move to device + d_A = CuArray{elty, 2}[] + for i in 1:length(A) + push!(d_A,CuArray(A[i])) + end + # testing without pivoting quickly results in inaccuracies along the diagonal + # test with pivoting + pivot, info = CUBLAS.getrf_batched!(d_A, true) + h_info = Array(info) + h_pivot = Array(pivot) + for As in 1:length(d_A) + C = lu(A[As]) + h_A = Array(d_A[As]) + #reconstruct L,U + dL = Matrix(one(elty)*I, m, m) + dU = zeros(elty,(m,m)) + k = h_info[As] + if( k >= 0 ) + dL += tril(h_A,-k-1) + dU += triu(h_A,k) + end + #compare pivots + @test length(setdiff(h_pivot[:,As],C.p)) == 0 + #make device pivot matrix + P = Matrix(1.0*I, m, m) + for row in 1:m + temp = copy(P[row,:]) + P[row,:] = P[h_pivot[row,As],:] + P[h_pivot[row,As],:] = temp + end + @test inv(P)*dL*dU ≈ inv(C.P) * C.L * C.U + end + end + + @testset "getrf_batched" begin + local k + # generate matrices + A = [rand(elty,m,m) for i in 1:10] + # move to device + d_A = CuArray{elty, 2}[] + for i in 1:length(A) + push!(d_A,CuArray(A[i])) + end + # testing without pivoting quickly results in inaccuracies along the diagonal + # test with pivoting + pivot, info, d_B = CUBLAS.getrf_batched(d_A, true) + h_info = Array(info) + h_pivot = Array(pivot) + for Bs in 1:length(d_B) + C = lu(A[Bs]) + h_A = Array(d_B[Bs]) + #reconstruct L,U + dL = Matrix(one(elty)*I, m, m) + dU = zeros(elty,(m,m)) + k = h_info[Bs] + if( k >= 0 ) + dL += tril(h_A,-k-1) + dU += triu(h_A,k) + end + #compare pivots + @test length(setdiff(h_pivot[:,Bs],C.p)) == 0 + #make device pivot matrix + P = Matrix(1.0*I, m, m) + for row in 1:m + temp = copy(P[row,:]) + P[row,:] = P[h_pivot[row,Bs],:] + P[h_pivot[row,Bs],:] = temp + end + @test inv(P)*dL*dU ≈ inv(C.P) * C.L * C.U + end + end + + @testset "getrf_strided_batched!" begin + Random.seed!(1) + local k + # generate strided matrix + A = rand(elty,m,m,10) + # move to device + d_A = CuArray(A) + # testing without pivoting quickly results in inaccuracies along the diagonal + # test with pivoting + pivot, info = CUBLAS.getrf_strided_batched!(d_A, true) + h_info = Array(info) + h_pivot = Array(pivot) + for As in 1:size(d_A, 3) + C = lu(A[:,:,As]) + h_A = Array(d_A[:,:,As]) + #reconstruct L,U + dL = Matrix(one(elty)*I, m, m) + dU = zeros(elty,(m,m)) + k = h_info[As] + if( k >= 0 ) + dL += tril(h_A,-k-1) + dU += triu(h_A,k) + end + #compare pivots + @test length(setdiff(h_pivot[:,As],C.p)) == 0 + #make device pivot matrix + P = Matrix(1.0*I, m, m) + for row in 1:m + temp = copy(P[row,:]) + P[row,:] = P[h_pivot[row,As],:] + P[h_pivot[row,As],:] = temp + end + @test inv(P)*dL*dU ≈ inv(C.P) * C.L * C.U + end + end + + @testset "getrf_strided_batched" begin + local k + # generate strided matrix + A = rand(elty,m,m,10) + # move to device + d_A = CuArray(A) + # testing without pivoting quickly results in inaccuracies along the diagonal + # test with pivoting + pivot, info, d_B = CUBLAS.getrf_strided_batched(d_A, true) + h_info = Array(info) + h_pivot = Array(pivot) + for Bs in 1:size(d_B, 3) + C = lu(A[:,:,Bs]) + h_A = Array(d_B[:,:,Bs]) + #reconstruct L,U + dL = Matrix(one(elty)*I, m, m) + dU = zeros(elty,(m,m)) + k = h_info[Bs] + if( k >= 0 ) + dL += tril(h_A,-k-1) + dU += triu(h_A,k) + end + #compare pivots + @test length(setdiff(h_pivot[:,Bs],C.p)) == 0 + #make device pivot matrix + P = Matrix(1.0*I, m, m) + for row in 1:m + temp = copy(P[row,:]) + P[row,:] = P[h_pivot[row,Bs],:] + P[h_pivot[row,Bs],:] = temp + end + @test inv(P)*dL*dU ≈ inv(C.P) * C.L * C.U + end + end + + for (opchar,opfun) in (('N',identity), ('T',transpose), ('C',adjoint)) + + @testset "getrs_batched!" begin + A = [rand(elty,n,n) for _ in 1:k]; + d_A = [CuArray(a) for a in A]; + d_A2 = deepcopy(d_A); + d_pivot, info, d_LU = CUDA.CUBLAS.getrf_batched!(d_A, true); + @test d_LU == d_A + d_pivot2 = similar(d_pivot); + info2 = similar(info); + CUDA.CUBLAS.getrf_batched!(d_A2, d_pivot2, info2); + @test isapprox(d_pivot, d_pivot2) + @test isapprox(info, info2) + B = [rand(elty,n,m) for _ in 1:k]; + d_B = [CuArray(b) for b in B]; + info2, d_Bhat = CUDA.CUBLAS.getrs_batched!(opchar, d_LU, d_B, d_pivot); + @test d_Bhat == d_B + h_Bhat = [collect(bh) for bh in d_Bhat]; + for i in 1:k + @test h_Bhat[i] ≈ opfun(A[i]) \ B[i] + end + end + + @testset "getrs_batched" begin + A = [rand(elty,n,n) for _ in 1:k]; + d_A = [CuArray(a) for a in A]; + d_A2 = deepcopy(d_A); + d_pivot, info, d_LU = CUDA.CUBLAS.getrf_batched(d_A, true); + @test d_LU != d_A + d_pivot2 = similar(d_pivot); + info2 = similar(info); + CUDA.CUBLAS.getrf_batched(d_A2, d_pivot2, info2); + @test isapprox(d_pivot, d_pivot2) + @test isapprox(info, info2) + B = [rand(elty,n,m) for _ in 1:k]; + d_B = [CuArray(b) for b in B]; + info2, d_Bhat = CUDA.CUBLAS.getrs_batched(opchar, d_LU, d_B, d_pivot); + @test d_Bhat != d_B + h_Bhat = [collect(bh) for bh in d_Bhat]; + for i in 1:k + @test h_Bhat[i] ≈ opfun(A[i]) \ B[i] + end + end + + @testset "getrs_strided_batched!" begin + A = rand(elty,n,n,k); + d_A = CuArray(A); + d_A2 = copy(d_A); + d_pivot, info, d_LU = CUDA.CUBLAS.getrf_strided_batched!(d_A, true); + @test d_LU == d_A + d_pivot2 = similar(d_pivot); + info2 = similar(info); + CUDA.CUBLAS.getrf_strided_batched!(d_A2, d_pivot2, info2); + @test isapprox(d_pivot, d_pivot2) + @test isapprox(info, info2) + B = rand(elty,n,m,k); + d_B = CuArray(B); + info2, d_Bhat = CUDA.CUBLAS.getrs_strided_batched!(opchar, d_LU, d_B, d_pivot); + @test d_Bhat == d_B + h_Bhat = collect(d_Bhat); + for i in 1:k + @test h_Bhat[:,:,i] ≈ opfun(A[:,:,i]) \ B[:,:,i] + end + end + + @testset "getrs_strided_batched" begin + A = rand(elty,n,n,k); + d_A = CuArray(A); + d_A2 = copy(d_A); + d_pivot, info, d_LU = CUDA.CUBLAS.getrf_strided_batched(d_A, true); + @test d_LU != d_A + d_pivot2 = similar(d_pivot); + info2 = similar(info); + CUDA.CUBLAS.getrf_strided_batched(d_A2, d_pivot2, info2); + @test isapprox(d_pivot, d_pivot2) + @test isapprox(info, info2) + B = rand(elty,n,m,k); + d_B = CuArray(B); + info2, d_Bhat = CUDA.CUBLAS.getrs_strided_batched(opchar, d_LU, d_B, d_pivot); + @test d_Bhat != d_B + h_Bhat = collect(d_Bhat); + for i in 1:k + @test h_Bhat[:,:,i] ≈ opfun(A[:,:,i]) \ B[:,:,i] + end + end + + end + + @testset "getri_strided_batched" begin + # generate strided matrix + A = rand(elty,m,m,10) + # move to device + d_A = CuArray(A) + d_B = similar(d_A) + pivot, info = CUBLAS.getrf_strided_batched!(d_A, true) + info = CUBLAS.getri_strided_batched!(d_A, d_B, pivot) + h_info = Array(info) + for Cs in 1:size(d_A,3) + B = inv(A[:,:,Cs]) + @test h_info[Cs] == 0 + @test B ≈ Array(d_B[:,:,Cs]) rtol=1e-3 + end + end + + @testset "getri_batched" begin + # generate matrices + A = [rand(elty,m,m) for i in 1:10] + # move to device + d_A = CuArray{elty, 2}[] + for i in 1:length(A) + push!(d_A,CuArray(A[i])) + end + pivot, info = CUBLAS.getrf_batched!(d_A, true) + h_info = Array(info) + for Cs in 1:length(h_info) + @test h_info[Cs] == 0 + end + pivot, info, d_C = CUBLAS.getri_batched(d_A, pivot) + h_info = Array(info) + for Cs in 1:length(d_C) + C = inv(A[Cs]) + h_C = Array(d_C[Cs]) + @test h_info[Cs] == 0 + @test C ≈ h_C rtol=1e-2 + end + end + + @testset "matinv_batched" begin + # generate matrices + A = [rand(elty,m,m) for i in 1:10] + # move to device + d_A = CuArray{elty, 2}[] + for i in 1:length(A) + push!(d_A,CuArray(A[i])) + end + info, d_C = CUBLAS.matinv_batched(d_A) + for Cs in 1:length(d_C) + C = inv(A[Cs]) + h_C = Array(d_C[Cs]) + @test C ≈ h_C + end + push!(d_A, CUDA.rand(elty, m, m+1)) + @test_throws DimensionMismatch CUBLAS.matinv_batched(d_A) + end + + @testset "geqrf_batched!" begin + # generate matrices + A = [rand(elty,m,n) for i in 1:10] + # move to device + d_A = CuArray{elty, 2}[] + for i in 1:length(A) + push!(d_A,CuArray(A[i])) + end + tau, d_A = CUBLAS.geqrf_batched!(d_A) + for As in 1:length(d_A) + C = qr(A[As]) + h_A = Array(d_A[As]) + h_tau = Array(tau[As]) + # build up Q + Q = Matrix(one(elty)*I, min(m,n), min(m,n)) + for i in 1:min(m,n) + v = zeros(elty,m) + v[i] = one(elty) + v[i+1:m] = h_A[i+1:m,i] + Q *= I - h_tau[i] * v * v' + end + @test Q ≈ Array(C.Q) rtol=1e-2 + end + end + + @testset "geqrf_batched" begin + # generate matrices + A = [rand(elty,m,n) for i in 1:10] + # move to device + d_A = CuArray{elty, 2}[] + for i in 1:length(A) + push!(d_A,CuArray(A[i])) + end + tau, d_B = CUBLAS.geqrf_batched!(d_A) + for Bs in 1:length(d_B) + C = qr(A[Bs]) + h_B = Array(d_B[Bs]) + h_tau = Array(tau[Bs]) + # build up Q + Q = Matrix(one(elty)*I, min(m,n), min(m,n)) + for i in 1:min(m,n) + v = zeros(elty,m) + v[i] = one(elty) + v[i+1:m] = h_B[i+1:m,i] + Q *= I - h_tau[i] * v * v' + end + @test Q ≈ Array(C.Q) rtol=1e-2 + end + end + + @testset "gels_batched!" begin + # generate matrices + A = [rand(elty,n,k) for i in 1:10] + C = [rand(elty,n,k) for i in 1:10] + # move to device + d_A = CuArray{elty, 2}[] + d_C = CuArray{elty, 2}[] + for i in 1:length(A) + push!(d_A,CuArray(A[i])) + push!(d_C,CuArray(C[i])) + end + d_A, d_C, info = CUBLAS.gels_batched!('N',d_A, d_C) + for Cs in 1:length(d_C) + X = A[Cs]\C[Cs] + h_C = Array(d_C[Cs])[1:k,1:k] + @test X ≈ h_C rtol=1e-2 + end + push!(d_C,CUDA.rand(elty,n,k-1)) + @test_throws DimensionMismatch CUBLAS.gels_batched!('N',d_A, d_C) + push!(d_A,CUDA.rand(elty,n,k-1)) + @test_throws DimensionMismatch CUBLAS.gels_batched!('N',d_A, d_C) + A = [rand(elty,k-1,k) for i in 1:10] + C = [rand(elty,k-1,k) for i in 1:10] + # move to device + d_A = CuArray{elty, 2}[] + d_C = CuArray{elty, 2}[] + for i in 1:length(A) + push!(d_A,CuArray(A[i])) + push!(d_C,CuArray(C[i])) + end + # system is now not overdetermined + @test_throws ArgumentError CUBLAS.gels_batched!('N',d_A, d_C) + end + + @testset "gels_batched" begin + # generate matrices + A = [rand(elty,n,k) for i in 1:10] + C = [rand(elty,n,k) for i in 1:10] + # move to device + d_A = CuArray{elty, 2}[] + d_C = CuArray{elty, 2}[] + for i in 1:length(A) + push!(d_A,CuArray(A[i])) + push!(d_C,CuArray(C[i])) + end + d_B, d_D, info = CUBLAS.gels_batched('N',d_A, d_C) + for Ds in 1:length(d_D) + X = A[Ds]\C[Ds] + h_D = Array(d_D[Ds])[1:k,1:k] + @test X ≈ h_D rtol=1e-2 + end + end + # generate matrices + A = rand(elty,m,n) + C = rand(elty,m,n) + x = rand(elty,m) + y = rand(elty,n) + # move to device + d_A = CuArray(A) + d_C = CuArray(C) + d_x = CuArray(x) + d_y = CuArray(y) + C = diagm(0 => x) * A + @testset "dgmm!" begin + # compute + CUBLAS.dgmm!('L', d_A, d_x, d_C) + # move to host and compare + h_C = Array(d_C) + @test C ≈ h_C + # bounds checking + @test_throws DimensionMismatch CUBLAS.dgmm!('R', d_A, d_x, d_C) + E = rand(elty,m,m) + d_E = CuArray(E) + @test_throws DimensionMismatch CUBLAS.dgmm!('L', d_E, d_x, d_C) + end + @testset "dgmm" begin + d_C = CUBLAS.dgmm('L', d_A, d_x) + # move to host and compare + h_C = Array(d_C) + @test C ≈ h_C + end + @testset "diagonal -- mul!" begin + XA = rand(elty,m,n) + d_XA = CuArray(XA) + d_X = Diagonal(d_x) + mul!(d_XA, d_X, d_A) + Array(d_XA) ≈ Diagonal(x) * A + + AY = rand(elty,m,n) + d_AY = CuArray(AY) + d_Y = Diagonal(d_y) + mul!(d_AY, d_A, d_Y) + Array(d_AY) ≈ A * Diagonal(y) + + YA = rand(elty,n,m) + d_YA = CuArray(YA) + d_Y = Diagonal(d_y) + mul!(d_YA, d_Y, transpose(d_A)) + Array(d_YA) ≈ Diagonal(y) * transpose(A) + + AX = rand(elty,n,m) + d_AX = CuArray(AX) + d_X = Diagonal(d_x) + mul!(d_AX, transpose(d_A), d_X) + Array(d_AX) ≈ transpose(A) * Diagonal(x) + + YA = rand(elty,n,m) + d_YA = CuArray(YA) + d_Y = Diagonal(d_y) + mul!(d_YA, d_Y, d_A') + Array(d_YA) ≈ Diagonal(y) * A' + + AX = rand(elty,n,m) + d_AX = CuArray(AX) + d_X = Diagonal(d_x) + mul!(d_AX, d_A', d_X) + Array(d_AX) ≈ A' * Diagonal(x) + end + end # extensions + + @testset "StaticArray eltype" begin + A = CuArray(rand(SVector{2, Float32}, 3, 3)) + B = CuArray(rand(Float32, 3, 3)) + C = A * B + hC = Array(A) * Array(B) + @test Array(C) ≈ hC + end +end # elty diff --git a/test/libraries/cublas_level1.jl b/test/libraries/cublas_level1.jl new file mode 100644 index 0000000000..0d52febb2b --- /dev/null +++ b/test/libraries/cublas_level1.jl @@ -0,0 +1,97 @@ +using CUDA.CUBLAS + +using LinearAlgebra + +using BFloat16s +using StaticArrays + +@test CUBLAS.version() isa VersionNumber +@test CUBLAS.version().major == CUBLAS.cublasGetProperty(CUDA.MAJOR_VERSION) +@test CUBLAS.version().minor == CUBLAS.cublasGetProperty(CUDA.MINOR_VERSION) +@test CUBLAS.version().patch == CUBLAS.cublasGetProperty(CUDA.PATCH_LEVEL) + +m = 20 +n = 35 +k = 13 + +@testset "level 1" begin + @testset for T in [Float32, Float64, ComplexF32, ComplexF64] + A = CUDA.rand(T, m) + B = CuArray{T}(undef, m) + CUBLAS.copy!(m,A,B) + @test Array(A) == Array(B) + + @test testf(rmul!, rand(T, 6, 9, 3), Ref(rand())) + @test testf(dot, rand(T, m), rand(T, m)) + @test testf(*, transpose(rand(T, m)), rand(T, m)) + @test testf(*, rand(T, m)', rand(T, m)) + @test testf(norm, rand(T, m)) + @test testf(BLAS.asum, rand(T, m)) + @test testf(axpy!, Ref(rand()), rand(T, m), rand(T, m)) + @test testf(axpby!, Ref(rand()), rand(T, m), Ref(rand()), rand(T, m)) + + if T <: Complex + @test testf(dot, rand(T, m), rand(T, m)) + x = rand(T, m) + y = rand(T, m) + dx = CuArray(x) + dy = CuArray(y) + dz = dot(dx, dy) + z = dot(x, y) + @test dz ≈ z + end + + @testset "rotate!" begin + @test testf(rotate!, rand(T, m), rand(T, m), rand(real(T)), rand(real(T))) + @test testf(rotate!, rand(T, m), rand(T, m), rand(real(T)), rand(T)) + end + @testset "reflect!" begin + @test testf(reflect!, rand(T, m), rand(T, m), rand(real(T)), rand(real(T))) + @test testf(reflect!, rand(T, m), rand(T, m), rand(real(T)), rand(T)) + end + + @testset "swap!" begin + # swap is an extension + x = rand(T, m) + y = rand(T, m) + dx = CuArray(x) + dy = CuArray(y) + CUBLAS.swap!(m, dx, dy) + h_x = collect(dx) + h_y = collect(dy) + @test h_x ≈ y + @test h_y ≈ x + end + + @testset "iamax/iamin" begin + a = convert.(T, [1.0, 2.0, -0.8, 5.0, 3.0]) + ca = CuArray(a) + @test BLAS.iamax(a) == CUBLAS.iamax(ca) + @test CUBLAS.iamin(ca) == 3 + end + end # level 1 testset + @testset for T in [Float16, ComplexF16] + A = CuVector(rand(T, m)) # CUDA.rand doesn't work with 16 bit types yet + B = CuArray{T}(undef, m) + CUBLAS.copy!(m,A,B) + @test Array(A) == Array(B) + + @test testf(dot, rand(T, m), rand(T, m)) + @test testf(*, transpose(rand(T, m)), rand(T, m)) + @test testf(*, rand(T, m)', rand(T, m)) + @test testf(norm, rand(T, m)) + @test testf(axpy!, Ref(rand()), rand(T, m), rand(T, m)) + @test testf(axpby!, Ref(rand()), rand(T, m), Ref(rand()), rand(T, m)) + + if T <: Complex + @test testf(dot, rand(T, m), rand(T, m)) + x = rand(T, m) + y = rand(T, m) + dx = CuArray(x) + dy = CuArray(y) + dz = dot(dx, dy) + z = dot(x, y) + @test dz ≈ z + end + end # level 1 testset +end diff --git a/test/libraries/cublas_level2.jl b/test/libraries/cublas_level2.jl new file mode 100644 index 0000000000..368df82204 --- /dev/null +++ b/test/libraries/cublas_level2.jl @@ -0,0 +1,640 @@ +using CUDA.CUBLAS +using CUDA.CUBLAS: band, bandex + +using LinearAlgebra + +using BFloat16s +using StaticArrays + +@test CUBLAS.version() isa VersionNumber +@test CUBLAS.version().major == CUBLAS.cublasGetProperty(CUDA.MAJOR_VERSION) +@test CUBLAS.version().minor == CUBLAS.cublasGetProperty(CUDA.MINOR_VERSION) +@test CUBLAS.version().patch == CUBLAS.cublasGetProperty(CUDA.PATCH_LEVEL) + +m = 20 +n = 35 +k = 13 + +@testset "level 2" begin + @testset for elty in [Float32, Float64, ComplexF32, ComplexF64] + @testset "gemv" begin + alpha = rand(elty) + beta = rand(elty) + @test testf(*, rand(elty, m, n), rand(elty, n)) + @test testf(*, transpose(rand(elty, m, n)), rand(elty, m)) + @test testf(*, rand(elty, m, n)', rand(elty, m)) + x = rand(elty, m) + A = rand(elty, m, m + 1 ) + y = rand(elty, m) + dx = CuArray(x) + dA = CuArray(A) + dy = CuArray(y) + @test_throws DimensionMismatch mul!(dy, dA, dx) + A = rand(elty, m + 1, m ) + dA = CuArray(A) + @test_throws DimensionMismatch mul!(dy, dA, dx) + x = rand(elty, m) + A = rand(elty, n, m) + dx = CuArray(x) + dA = CuArray(A) + alpha = rand(elty) + dy = CUBLAS.gemv('N', alpha, dA, dx) + hy = collect(dy) + @test hy ≈ alpha * A * x + dy = CUBLAS.gemv('N', dA, dx) + hy = collect(dy) + @test hy ≈ A * x + end + + if CUBLAS.version() >= v"11.9" + @testset "gemv_batched" begin + alpha = rand(elty) + beta = rand(elty) + x = [rand(elty, m) for i=1:10] + A = [rand(elty, n, m) for i=1:10] + y = [rand(elty, n) for i=1:10] + dx = CuArray{elty, 1}[] + dA = CuArray{elty, 2}[] + dy = CuArray{elty, 1}[] + dbad = CuArray{elty, 1}[] + for i=1:length(A) + push!(dA, CuArray(A[i])) + push!(dx, CuArray(x[i])) + push!(dy, CuArray(y[i])) + if i < length(A) - 2 + push!(dbad,CuArray(dx[i])) + end + end + @test_throws DimensionMismatch CUBLAS.gemv_batched!('N', alpha, dA, dx, beta, dbad) + CUBLAS.gemv_batched!('N', alpha, dA, dx, beta, dy) + for i=1:length(A) + hy = collect(dy[i]) + y[i] = alpha * A[i] * x[i] + beta * y[i] + @test y[i] ≈ hy + end + end + end + + if CUBLAS.version() >= v"11.9" + @testset "gemv_strided_batched" begin + alpha = rand(elty) + beta = rand(elty) + x = rand(elty, m, 10) + A = rand(elty, n, m, 10) + y = rand(elty, n, 10) + bad = rand(elty, m, 10) + dx = CuArray(x) + dA = CuArray(A) + dy = CuArray(y) + dbad = CuArray(bad) + @test_throws DimensionMismatch CUBLAS.gemv_strided_batched!('N', alpha, dA, dx, beta, dbad) + bad = rand(elty, n, 2) + dbad = CuArray(bad) + @test_throws DimensionMismatch CUBLAS.gemv_strided_batched!('N', alpha, dA, dx, beta, dbad) + CUBLAS.gemv_strided_batched!('N', alpha, dA, dx, beta, dy) + for i=1:size(A, 3) + hy = collect(dy[:, i]) + y[:, i] = alpha * A[:, :, i] * x[:, i] + beta * y[:, i] + @test y[:, i] ≈ hy + end + end + end + + @testset "mul! y = $f(A) * x * $Ts(a) + y * $Ts(b)" for f in (identity, transpose, adjoint), Ts in (Int, elty) + y, A, x = rand(elty, 5), rand(elty, 5, 5), rand(elty, 5) + dy, dA, dx = CuArray(y), CuArray(A), CuArray(x) + mul!(dy, f(dA), dx, Ts(1), Ts(2)) + mul!(y, f(A), x, Ts(1), Ts(2)) + @test Array(dy) ≈ y + end + + @testset "hermitian" begin + y, A, x = rand(elty, 5), Hermitian(rand(elty, 5, 5)), rand(elty, 5) + dy, dA, dx = CuArray(y), Hermitian(CuArray(A)), CuArray(x) + mul!(dy, dA, dx) + mul!(y, A, x) + @test Array(dy) ≈ y + end + + @testset "banded methods" begin + # bands + ku = 2 + kl = 3 + # generate banded matrix + A = rand(elty,m,n) + A = bandex(A,kl,ku) + # get packed format + Ab = band(A,kl,ku) + d_Ab = CuArray(Ab) + x = rand(elty,n) + d_x = CuArray(x) + @testset "gbmv!" begin + alpha = rand(elty) + beta = rand(elty) + # test y = alpha*A*x + beta*y + y = rand(elty,m) + d_y = CuArray(y) + CUBLAS.gbmv!('N',m,kl,ku,alpha,d_Ab,d_x,beta,d_y) + BLAS.gbmv!('N',m,kl,ku,alpha,Ab,x,beta,y) + h_y = Array(d_y) + @test y ≈ h_y + # test y = alpha*transpose(A)*x + beta*y + x = rand(elty,n) + d_x = CuArray(x) + y = rand(elty,m) + d_y = CuArray(y) + CUBLAS.gbmv!('T',m,kl,ku,alpha,d_Ab,d_y,beta,d_x) + BLAS.gbmv!('T',m,kl,ku,alpha,Ab,y,beta,x) + h_x = Array(d_x) + @test x ≈ h_x + # test y = alpha*A'*x + beta*y + x = rand(elty,n) + d_x = CuArray(x) + y = rand(elty,m) + d_y = CuArray(y) + CUBLAS.gbmv!('C',m,kl,ku,alpha,d_Ab,d_y,beta,d_x) + BLAS.gbmv!('C',m,kl,ku,alpha,Ab,y,beta,x) + h_x = Array(d_x) + @test x ≈ h_x + # test alpha=1 version without y + d_y = CUBLAS.gbmv('N',m,kl,ku,d_Ab,d_x) + y = BLAS.gbmv('N',m,kl,ku,Ab,x) + h_y = Array(d_y) + @test y ≈ h_y + end + @testset "gbmv" begin + x = rand(elty,n) + d_x = CuArray(x) + alpha = rand(elty) + beta = rand(elty) + # test y = alpha*A*x + d_y = CUBLAS.gbmv('N',m,kl,ku,alpha,d_Ab,d_x) + y = zeros(elty,m) + y = BLAS.gbmv('N',m,kl,ku,alpha,Ab,x) + h_y = Array(d_y) + @test y ≈ h_y + end + A = rand(elty,m,m) + A = A + A' + nbands = 3 + @test m >= 1+nbands + A = bandex(A,nbands,nbands) + # convert to 'upper' banded storage format + AB = band(A,0,nbands) + # construct x + x = rand(elty,m) + d_AB = CuArray(AB) + d_x = CuArray(x) + if elty <: Real + @testset "sbmv!" begin + alpha = rand(elty) + beta = rand(elty) + y = rand(elty,m) + d_y = CuArray(y) + # sbmv! + CUBLAS.sbmv!('U',nbands,alpha,d_AB,d_x,beta,d_y) + y = alpha*(A*x) + beta*y + # compare + h_y = Array(d_y) + @test y ≈ h_y + end + @testset "sbmv" begin + d_y = CUBLAS.sbmv('U',nbands,d_AB,d_x) + y = A*x + # compare + h_y = Array(d_y) + @test y ≈ h_y + end + else + @testset "hbmv!" begin + alpha = rand(elty) + beta = rand(elty) + y = rand(elty,m) + d_y = CuArray(y) + # hbmv! + CUBLAS.hbmv!('U',nbands,alpha,d_AB,d_x,beta,d_y) + y = alpha*(A*x) + beta*y + # compare + h_y = Array(d_y) + @test y ≈ h_y + end + @testset "hbmv" begin + d_y = CUBLAS.hbmv('U',nbands,d_AB,d_x) + y = A*x + # compare + h_y = Array(d_y) + @test y ≈ h_y + end + end + # generate triangular matrix + A = rand(elty,m,m) + # restrict to 3 bands + nbands = 3 + @test m >= 1+nbands + A = bandex(A,0,nbands) + # convert to 'upper' banded storage format + AB = band(A,0,nbands) + d_AB = CuArray(AB) + @testset "tbmv!" begin + y = rand(elty, m) + # move to host + d_y = CuArray(y) + # tbmv! + CUBLAS.tbmv!('U','N','N',nbands,d_AB,d_y) + y = A*y + # compare + h_y = Array(d_y) + @test y ≈ h_y + end + @testset "tbmv" begin + # tbmv + d_y = CUBLAS.tbmv('U','N','N',nbands,d_AB,d_x) + y = A*x + # compare + h_y = Array(d_y) + @test y ≈ h_y + end + @testset "tbsv!" begin + d_y = copy(d_x) + #tbsv! + CUBLAS.tbsv!('U','N','N',nbands,d_AB,d_y) + y = A\x + # compare + h_y = Array(d_y) + @test y ≈ h_y + end + @testset "tbsv" begin + d_y = CUBLAS.tbsv('U','N','N',nbands,d_AB,d_x) + y = A\x + # compare + h_y = Array(d_y) + @test y ≈ h_y + end + end + A = rand(elty,m,n) + dA = CuArray(A) + sA = rand(elty,m,m) + sA = sA + transpose(sA) + dsA = CuArray(sA) + hA = rand(elty,m,m) + hA = hA + hA' + dhA = CuArray(hA) + x = rand(elty,m) + dx = CuArray(x) + + function pack(A, uplo) + AP = Vector{elty}(undef, (n*(n+1))>>1) + k = 1 + for j in 1:n + for i in (uplo==:L ? (j:n) : (1:j)) + AP[k] = A[i,j] + k += 1 + end + end + return AP + end + + if elty in ["Float32", "Float64"] + # pack matrices + sAPU = pack(sA, :U) + dsAPU = CuVector(sAPU) + sAPL = pack(sA, :L) + dsAPL = CuVector(sAPL) + + @testset "spmv!" begin + alpha = rand(elty) + beta = rand(elty) + # generate vectors + y = rand(elty,m) + # copy to device + dy = CuArray(y) + # execute on host + BLAS.spmv!('U',alpha,sAPU,x,beta,y) + # execute on device + CUBLAS.spmv!('U',alpha,dsAPU,dx,beta,dy) + # compare results + hy = Array(dy) + @test y ≈ hy + # execute on host + BLAS.spmv!('U',alpha,sAPL,x,beta,y) + # execute on device + CUBLAS.spmv!('U',alpha,dsAPL,dx,beta,dy) + # compare results + hy = Array(dy) + @test y ≈ hy + end + + @testset "spr!" begin + alpha = rand(elty) + beta = rand(elty) + # execute on host + BLAS.spr!('U',alpha,x,sAPU) + # execute on device + CUBLAS.spr!('U',alpha,dx,dsAPU) + # compare results + hsAPU = Array(dsAPU) + @test sAPU ≈ hsAPU + # execute on host + BLAS.spr!('U',alpha,x,sAPL) + # execute on device + CUBLAS.spr!('U',alpha,dx,dsAPL) + # compare results + hAPL = Array(dAPL) + @test sAPL ≈ hAPL + end + end + + @testset "symv!" begin + alpha = rand(elty) + beta = rand(elty) + # generate vectors + y = rand(elty,m) + # copy to device + dy = CuArray(y) + # execute on host + BLAS.symv!('U',alpha,sA,x,beta,y) + # execute on device + CUBLAS.symv!('U',alpha,dsA,dx,beta,dy) + # compare results + hy = Array(dy) + @test y ≈ hy + end + + @testset "symv" begin + y = BLAS.symv('U',sA,x) + # execute on device + dy = CUBLAS.symv('U',dsA,dx) + # compare results + hy = Array(dy) + @test y ≈ hy + end + if elty <: Complex + @testset "hemv!" begin + alpha = rand(elty) + beta = rand(elty) + y = rand(elty,m) + dy = CuArray(y) + # execute on host + BLAS.hemv!('U',alpha,hA,x,beta,y) + # execute on device + CUBLAS.hemv!('U',alpha,dhA,dx,beta,dy) + # compare results + hy = Array(dy) + @test y ≈ hy + end + @testset "hemv" begin + y = BLAS.hemv('U',hA,x) + # execute on device + dy = CUBLAS.hemv('U',dhA,dx) + # compare results + hy = Array(dy) + @test y ≈ hy + end + end + A = triu(sA) + dA = CuArray(A) + @testset "trmv!" begin + d_y = copy(dx) + # execute trmv! + CUBLAS.trmv!('U','N','N',dA,d_y) + y = A*x + # compare + h_y = Array(d_y) + @test y ≈ h_y + @test_throws DimensionMismatch CUBLAS.trmv!('U','N','N',dA,CUDA.rand(elty,m+1)) + end + + @testset "trmv" begin + d_y = CUBLAS.trmv('U','N','N',dA,dx) + y = A*x + # compare + h_y = Array(d_y) + @test y ≈ h_y + end + + @testset "lmul!(::UpperTriangular)" begin + dy = copy(dx) + lmul!(UpperTriangular(dA), dy) + y = UpperTriangular(A) * x + @test y ≈ Array(dy) + end + @testset "lmul!(::UpperTriangular{Adjoint})" begin + dy = copy(dx) + lmul!(adjoint(UpperTriangular(dA)), dy) + y = adjoint(UpperTriangular(A)) * x + @test y ≈ Array(dy) + end + @testset "lmul!(::UpperTriangular{Transpose})" begin + dy = copy(dx) + lmul!(transpose(UpperTriangular(dA)), dy) + y = transpose(UpperTriangular(A)) * x + @test y ≈ Array(dy) + end + @testset "lmul!(::LowerTriangular)" begin + dy = copy(dx) + lmul!(LowerTriangular(dA), dy) + y = LowerTriangular(A) * x + @test y ≈ Array(dy) + end + @testset "lmul!(::LowerTriangular{Adjoint})" begin + dy = copy(dx) + lmul!(adjoint(LowerTriangular(dA)), dy) + y = adjoint(LowerTriangular(A)) * x + @test y ≈ Array(dy) + end + @testset "lmul!(::LowerTriangular{Transpose})" begin + dy = copy(dx) + lmul!(transpose(LowerTriangular(dA)), dy) + y = transpose(LowerTriangular(A)) * x + @test y ≈ Array(dy) + end + + @testset "trsv!" begin + d_y = copy(dx) + # execute trsv! + CUBLAS.trsv!('U','N','N',dA,d_y) + y = A\x + # compare + h_y = Array(d_y) + @test y ≈ h_y + @test_throws DimensionMismatch CUBLAS.trsv!('U','N','N',dA,CUDA.rand(elty,m+1)) + end + + @testset "trsv" begin + d_y = CUBLAS.trsv('U','N','N',dA,dx) + y = A\x + # compare + h_y = Array(d_y) + @test y ≈ h_y + end + @testset "trsv (adjoint)" begin + d_y = CUBLAS.trsv('U','C','N',dA,dx) + y = adjoint(A)\x + # compare + h_y = Array(d_y) + @test y ≈ h_y + end + @testset "trsv (transpose)" begin + d_y = CUBLAS.trsv('U','T','N',dA,dx) + y = transpose(A)\x + # compare + h_y = Array(d_y) + @test y ≈ h_y + end + @testset "ldiv! of triangular types" begin + @testset "ldiv!(::UpperTriangular)" begin + A = rand(elty,m,m) + A = A + transpose(A) + dA = CuArray(A) + x = rand(elty,m) + dx = CuArray(x) + dy = copy(dx) + CUBLAS.ldiv!(UpperTriangular(dA), dy) + y = UpperTriangular(A) \ x + @test y ≈ Array(dy) + end + @testset "ldiv!(::UpperTriangular{Adjoint})" begin + A = rand(elty,m,m) + A = A + transpose(A) + dA = CuArray(A) + x = rand(elty,m) + dx = CuArray(x) + dy = copy(dx) + CUBLAS.ldiv!(adjoint(UpperTriangular(dA)), dy) + y = adjoint(UpperTriangular(A)) \ x + @test y ≈ Array(dy) + end + @testset "ldiv!(::UpperTriangular{Transpose})" begin + A = rand(elty,m,m) + A = A + transpose(A) + dA = CuArray(A) + x = rand(elty,m) + dx = CuArray(x) + dy = copy(dx) + CUBLAS.ldiv!(transpose(UpperTriangular(dA)), dy) + y = transpose(UpperTriangular(A)) \ x + @test y ≈ Array(dy) + end + @testset "ldiv!(::LowerTriangular)" begin + A = rand(elty,m,m) + A = A + transpose(A) + dA = CuArray(A) + x = rand(elty,m) + dx = CuArray(x) + dy = copy(dx) + CUBLAS.ldiv!(LowerTriangular(dA), dy) + y = LowerTriangular(A) \ x + @test y ≈ Array(dy) + end + @testset "ldiv!(::LowerTriangular{Adjoint})" begin + A = rand(elty,m,m) + A = A + transpose(A) + dA = CuArray(A) + x = rand(elty,m) + dx = CuArray(x) + dy = copy(dx) + CUBLAS.ldiv!(adjoint(LowerTriangular(dA)), dy) + y = adjoint(LowerTriangular(A)) \ x + @test y ≈ Array(dy) + end + @testset "ldiv!(::LowerTriangular{Transpose})" begin + A = rand(elty,m,m) + A = A + transpose(A) + dA = CuArray(A) + x = rand(elty,m) + dx = CuArray(x) + dy = copy(dx) + CUBLAS.ldiv!(transpose(LowerTriangular(dA)), dy) + y = transpose(LowerTriangular(A)) \ x + @test y ≈ Array(dy) + end + end + @testset "inv($TR)" for TR in (UpperTriangular, LowerTriangular, UnitUpperTriangular, UnitLowerTriangular) + @test testf(x -> inv(TR(x)), rand(elty, m, m)) + end + + @testset "ger!" begin + alpha = rand(elty) + A = rand(elty,m,m) + x = rand(elty,m) + y = rand(elty,m) + dA = CuArray(A) + dx = CuArray(x) + dy = CuArray(y) + # perform rank one update + dB = copy(dA) + CUBLAS.ger!(alpha,dx,dy,dB) + B = (alpha*x)*y' + A + # move to host and compare + hB = Array(dB) + @test B ≈ hB + end + + @testset "syr!" begin + alpha = rand(elty) + sA = rand(elty,m,m) + sA = sA + transpose(sA) + x = rand(elty,m) + dx = CuArray(x) + dB = CuArray(sA) + CUBLAS.syr!('U',alpha,dx,dB) + B = (alpha*x)*transpose(x) + sA + # move to host and compare upper triangles + hB = Array(dB) + B = triu(B) + hB = triu(hB) + @test B ≈ hB + end + if elty <: Complex + @testset "her!" begin + alpha = rand(elty) + hA = rand(elty,m,m) + hA = hA + adjoint(hA) + dB = CuArray(hA) + x = rand(elty,m) + dx = CuArray(x) + # perform rank one update + CUBLAS.her!('U',real(alpha),dx,dB) + B = (real(alpha)*x)*x' + hA + # move to host and compare upper triangles + hB = Array(dB) + B = triu(B) + hB = triu(hB) + @test B ≈ hB + end + + @testset "her2!" begin + alpha = rand(elty) + hA = rand(elty,m,m) + hA = hA + adjoint(hA) + dB = CuArray(hA) + x = rand(elty,m) + dx = CuArray(x) + y = rand(elty,m) + dy = CuArray(y) + CUBLAS.her2!('U',real(alpha),dx,dy,dB) + B = (real(alpha)*x)*y' + y*(real(alpha)*x)' + hA + # move to host and compare upper triangles + hB = Array(dB) + B = triu(B) + hB = triu(hB) + @test B ≈ hB + end + end + end + + @testset "gemv! with strided inputs" begin # JuliaGPU/CUDA.jl#445 + testf(rand(16), rand(4)) do p, b + W = @view p[reshape(1:(16),4,4)] + W*b + end + end + + @testset "StaticArray eltype" begin + A = CuArray(rand(SVector{2, Float64}, 3, 3)) + B = CuArray(rand(Float64, 3, 1)) + C = A * B + hC = Array(A) * Array(B) + @test Array(C) ≈ hC + end +end diff --git a/test/libraries/cublas_level3.jl b/test/libraries/cublas_level3.jl new file mode 100644 index 0000000000..043677a013 --- /dev/null +++ b/test/libraries/cublas_level3.jl @@ -0,0 +1,1432 @@ +using CUDA.CUBLAS +using CUDA.CUBLAS: band, bandex + +using LinearAlgebra + +using BFloat16s +using StaticArrays + +@test CUBLAS.version() isa VersionNumber +@test CUBLAS.version().major == CUBLAS.cublasGetProperty(CUDA.MAJOR_VERSION) +@test CUBLAS.version().minor == CUBLAS.cublasGetProperty(CUDA.MINOR_VERSION) +@test CUBLAS.version().patch == CUBLAS.cublasGetProperty(CUDA.PATCH_LEVEL) + +m = 20 +n = 35 +k = 13 + +@testset "level 3" begin + @testset for elty in [Float32, Float64, ComplexF32, ComplexF64] + + @testset "mul! C = $f(A) * $g(B) * $Ts(a) + C * $Ts(b)" for f in (identity, transpose, adjoint), g in (identity, transpose, adjoint), Ts in (Int, elty) + C, A, B = rand(elty, 5, 5), rand(elty, 5, 5), rand(elty, 5, 5) + dC, dA, dB = CuArray(C), CuArray(A), CuArray(B) + mul!(dC, f(dA), g(dB), Ts(1), Ts(2)) + mul!(C, f(A), g(B), Ts(1), Ts(2)) + @test Array(dC) ≈ C + end + + @testset "hermitian" begin + C, A, B = rand(elty, 5, 5), Hermitian(rand(elty, 5, 5)), rand(elty, 5, 5) + dC, dA, dB = CuArray(C), Hermitian(CuArray(A)), CuArray(B) + mul!(dC, dA, dB) + mul!(C, A, B) + @test Array(dC) ≈ C + end + + @testset "gemm!" begin + alpha = rand(elty) + beta = rand(elty) + A = rand(elty,m,k) + B = rand(elty,k,n) + C1 = rand(elty,m,n) + C2 = copy(C1) + d_A = CuArray(A) + d_B = CuArray(B) + d_C1 = CuArray(C1) + d_C2 = CuArray(C2) + hA = rand(elty,m,m) + hA = hA + hA' + dhA = CuArray(hA) + sA = rand(elty,m,m) + sA = sA + transpose(sA) + dsA = CuArray(sA) + + CUBLAS.gemm!('N','N',alpha,d_A,d_B,beta,d_C1) + mul!(d_C2, d_A, d_B) + h_C1 = Array(d_C1) + h_C2 = Array(d_C2) + C1 = (alpha*A)*B + beta*C1 + C2 = A*B + # compare + @test C1 ≈ h_C1 + @test C2 ≈ h_C2 + @test_throws ArgumentError mul!(dhA, dhA, dsA) + @test_throws DimensionMismatch mul!(d_C1, d_A, dsA) + end + @testset "strided gemm!" begin + denseA = CUDA.rand(elty, 4,4) + denseB = CUDA.rand(elty, 4,4) + denseC = CUDA.zeros(elty, 4,4) + + stridedA = view(denseA, 1:2, 1:2)::SubArray + stridedB = view(denseB, 1:2, 1:2)::SubArray + stridedC = view(denseC, 1:2, 1:2)::SubArray + + CUBLAS.gemm!('N', 'N', true, stridedA, stridedB, false, stridedC) + @test Array(stridedC) ≈ Array(stridedA) * Array(stridedB) + + stridedC .= 0 + mul!(stridedC, stridedA, stridedB, true, false) + @test Array(stridedC) ≈ Array(stridedA) * Array(stridedB) + end + if capability(device()) > v"5.0" + @testset "gemmEx!" begin + A = rand(elty,m,k) + B = rand(elty,k,n) + C1 = rand(elty,m,n) + d_A = CuArray(A) + d_B = CuArray(B) + d_C1 = CuArray(C1) + α = rand(elty) + β = rand(elty) + CUBLAS.gemmEx!('N','N',α,d_A,d_B,β,d_C1) + h_C1 = Array(d_C1) + C1 = (α*A)*B + β*C1 + # compare + @test C1 ≈ h_C1 + end + end + @testset "gemm" begin + A = rand(elty,m,k) + B = rand(elty,k,n) + d_A = CuArray(A) + d_B = CuArray(B) + d_C1 = CUBLAS.gemm('N','N',d_A,d_B) + C1 = A*B + C2 = d_A * d_B + # compare + h_C1 = Array(d_C1) + h_C2 = Array(C2) + @test C1 ≈ h_C1 + @test C1 ≈ h_C2 + end + @testset "xt_gemm! gpu" begin + alpha = rand(elty) + beta = rand(elty) + A = rand(elty,m,k) + B = rand(elty,k,n) + C1 = rand(elty,m,n) + C2 = copy(C1) + d_A = CuArray(A) + d_B = CuArray(B) + Bbad = rand(elty,k+1,n+1) + d_Bbad = CuArray(Bbad) + d_C1 = CuArray(C1) + d_C2 = CuArray(C2) + @test_throws DimensionMismatch CUBLAS.xt_gemm!('N','N',alpha,d_A,d_Bbad,beta,d_C1) + CUBLAS.xt_gemm!('N','N',alpha,d_A,d_B,beta,d_C1) + mul!(d_C2, d_A, d_B) + h_C1 = Array(d_C1) + h_C2 = Array(d_C2) + C1 = (alpha*A)*B + beta*C1 + C2 = A*B + # compare + @test C1 ≈ h_C1 + @test C2 ≈ h_C2 + end + @testset "xt_gemm! cpu" begin + alpha = rand(elty) + beta = rand(elty) + A = rand(elty,m,k) + B = rand(elty,k,n) + C1 = rand(elty,m,n) + C2 = copy(C1) + C3 = copy(C1) + C4 = copy(C2) + CUBLAS.xt_gemm!('N','N',alpha,A,B,beta,C1) + mul!(C2, A, B) + C3 = (alpha*A)*B + beta*C3 + C4 = A*B + # compare + @test C1 ≈ C3 + @test C2 ≈ C4 + end + @testset "xt_gemm gpu" begin + A = rand(elty,m,k) + B = rand(elty,k,n) + d_A = CuArray(A) + d_B = CuArray(B) + d_C = CUBLAS.xt_gemm('N','N',d_A,d_B) + C = A*B + C2 = d_A * d_B + # compare + @test d_C isa CuArray + h_C = Array(d_C) + h_C2 = Array(C2) + @test C ≈ h_C + @test C ≈ h_C2 + end + @testset "xt_gemm cpu" begin + A = rand(elty,m,k) + B = rand(elty,k,n) + d_A = CuArray(A) + d_B = CuArray(B) + C = CUBLAS.xt_gemm('N','N',A,B) + d_C2 = d_A * d_B + # compare + @test C isa Array + h_C2 = Array(C2) + @test C ≈ A*B + @test C ≈ Array(d_C2) + end + + @testset "symm!" begin + alpha = rand(elty) + beta = rand(elty) + sA = rand(elty,m,m) + sA = sA + transpose(sA) + dsA = CuArray(sA) + B = rand(elty,m,n) + C = rand(elty,m,n) + Bbad = rand(elty,m+1,n+1) + d_B = CuArray(B) + d_C = CuArray(C) + d_Bbad = CuArray(Bbad) + CUBLAS.symm!('L','U',alpha,dsA,d_B,beta,d_C) + C = (alpha*sA)*B + beta*C + # compare + h_C = Array(d_C) + @test C ≈ h_C + @test_throws DimensionMismatch CUBLAS.symm!('L','U',alpha,dsA,d_Bbad,beta,d_C) + end + + @testset "symm" begin + sA = rand(elty,m,m) + sA = sA + transpose(sA) + dsA = CuArray(sA) + B = rand(elty,m,n) + C = rand(elty,m,n) + Bbad = rand(elty,m+1,n+1) + d_B = CuArray(B) + d_C = CuArray(C) + d_Bbad = CuArray(Bbad) + d_C = CUBLAS.symm('L','U',dsA,d_B) + C = sA*B + # compare + h_C = Array(d_C) + @test C ≈ h_C + @test_throws DimensionMismatch CUBLAS.symm('L','U',dsA,d_Bbad) + end + @testset "xt_symm! gpu" begin + alpha = rand(elty) + beta = rand(elty) + sA = rand(elty,m,m) + sA = sA + transpose(sA) + dsA = CuArray(sA) + B = rand(elty,m,n) + C = rand(elty,m,n) + Bbad = rand(elty,m+1,n+1) + d_B = CuArray(B) + d_C = CuArray(C) + CUBLAS.xt_symm!('L','U',alpha,dsA,d_B,beta,d_C) + C = (alpha*sA)*B + beta*C + # compare + h_C = Array(d_C) + @test C ≈ h_C + end + @testset "xt_symm! cpu" begin + alpha = rand(elty) + beta = rand(elty) + sA = rand(elty,m,m) + sA = sA + transpose(sA) + B = rand(elty,m,n) + C = rand(elty,m,n) + h_C = copy(C) + CUBLAS.xt_symm!('L','U',alpha,copy(sA),copy(B),beta,h_C) + C = (alpha*sA)*B + beta*C + # compare + @test C ≈ h_C + end + + @testset "xt_symm gpu" begin + sA = rand(elty,m,m) + sA = sA + transpose(sA) + dsA = CuArray(sA) + B = rand(elty,m,n) + d_B = CuArray(B) + d_C = CUBLAS.xt_symm('L','U',dsA,d_B) + C = sA*B + # compare + @test d_C isa CuArray + h_C = Array(d_C) + @test C ≈ h_C + end + @testset "xt_symm cpu" begin + sA = rand(elty,m,m) + sA = sA + transpose(sA) + B = rand(elty,m,n) + h_C = CUBLAS.xt_symm('L','U',copy(sA),copy(B)) + C = sA*B + # compare + @test h_C isa Array + @test C ≈ h_C + end + @testset "trmm!" begin + alpha = rand(elty) + A = triu(rand(elty, m, m)) + B = rand(elty,m,n) + C = zeros(elty,m,n) + dA = CuArray(A) + dB = CuArray(B) + dC = CuArray(C) + C = alpha*A*B + CUBLAS.trmm!('L','U','N','N',alpha,dA,dB,dC) + # move to host and compare + h_C = Array(dC) + @test C ≈ h_C + end + @testset "trmm" begin + alpha = rand(elty) + A = triu(rand(elty, m, m)) + B = rand(elty,m,n) + C = zeros(elty,m,n) + dA = CuArray(A) + dB = CuArray(B) + dC = CuArray(C) + C = alpha*A*B + d_C = CUBLAS.trmm('L','U','N','N',alpha,dA,dB) + # move to host and compare + h_C = Array(d_C) + @test C ≈ h_C + end + @testset "xt_trmm! gpu" begin + alpha = rand(elty) + A = triu(rand(elty, m, m)) + B = rand(elty,m,n) + C = zeros(elty,m,n) + dA = CuArray(A) + dB = CuArray(B) + dC = CuArray(C) + C = alpha*A*B + CUBLAS.xt_trmm!('L','U','N','N',alpha,dA,dB,dC) + # move to host and compare + h_C = Array(dC) + @test C ≈ h_C + end + @testset "xt_trmm! cpu" begin + alpha = rand(elty) + A = triu(rand(elty, m, m)) + B = rand(elty,m,n) + C = alpha*A*B + h_C = zeros(C) + CUBLAS.xt_trmm!('L','U','N','N',alpha,copy(A),copy(B),h_C) + @test C ≈ h_C + end + @testset "xt_trmm gpu" begin + alpha = rand(elty) + A = triu(rand(elty, m, m)) + B = rand(elty,m,n) + C = zeros(elty,m,n) + dA = CuArray(A) + dB = CuArray(B) + dC = CuArray(C) + C = alpha*A*B + d_C = CUBLAS.xt_trmm('L','U','N','N',alpha,dA,dB) + # move to host and compare + @test d_C isa CuArray + h_C = Array(d_C) + @test C ≈ h_C + end + @testset "xt_trmm cpu" begin + alpha = rand(elty) + A = triu(rand(elty, m, m)) + B = rand(elty,m,n) + C = alpha*A*B + h_C = CUBLAS.xt_trmm('L','U','N','N',alpha,copy(A),copy(B)) + @test h_C isa Array + @test C ≈ h_C + end + + @testset "xt_trsm! gpu" begin + alpha = rand(elty) + A = triu(rand(elty, m, m)) + B = rand(elty,m,n) + dA = CuArray(A) + dB = CuArray(B) + C = alpha*(A\B) + dC = copy(dB) + synchronize() + CUBLAS.xt_trsm!('L','U','N','N',alpha,dA,dC) + # move to host and compare + h_C = Array(dC) + @test C ≈ h_C + end + @testset "xt_trsm! cpu" begin + alpha = rand(elty) + A = triu(rand(elty, m, m)) + B = rand(elty,m,n) + C = alpha*(A\B) + h_C = copy(B) + CUBLAS.xt_trsm!('L','U','N','N',alpha,copy(A),h_C) + @test C ≈ h_C + end + @testset "xt_trsm gpu" begin + alpha = rand(elty) + A = triu(rand(elty, m, m)) + B = rand(elty,m,n) + dA = CuArray(A) + dB = CuArray(B) + C = alpha*(A\B) + dC = CUBLAS.xt_trsm('L','U','N','N',alpha,dA,dB) + # move to host and compare + @test dC isa CuArray + h_C = Array(dC) + @test C ≈ h_C + end + @testset "xt_trsm cpu" begin + alpha = rand(elty) + A = triu(rand(elty, m, m)) + B = rand(elty,m,n) + C = alpha*(A\B) + h_C = CUBLAS.xt_trsm('L','U','N','N',alpha,copy(A),copy(B)) + @test h_C isa Array + @test C ≈ h_C + end + @testset "trsm" begin + # compute + @testset "adjtype=$adjtype, uplotype=$uplotype" for + adjtype in (identity, adjoint, transpose), + uplotype in (UpperTriangular, UnitUpperTriangular, LowerTriangular, UnitLowerTriangular) + + A = triu(rand(elty, m, m)) + dA = CuArray(A) + Br = rand(elty,m,n) + Bl = rand(elty,n,m) + d_Br = CuArray(Br) + d_Bl = CuArray(Bl) + @test adjtype(uplotype(A))\Br ≈ Array(adjtype(uplotype(dA))\d_Br) + @test Bl/adjtype(uplotype(A)) ≈ Array(d_Bl/adjtype(uplotype(dA))) + end + # Check also that scaling parameter works + alpha = rand(elty) + A = triu(rand(elty, m, m)) + dA = CuArray(A) + Br = rand(elty,m,n) + d_Br = CuArray(Br) + @test BLAS.trsm('L','U','N','N',alpha,A,Br) ≈ Array(CUBLAS.trsm('L','U','N','N',alpha,dA,d_Br)) + end + + @testset "trsm_batched!" begin + alpha = rand(elty) + bA = [rand(elty,m,m) for i in 1:10] + map!((x) -> triu(x), bA, bA) + bB = [rand(elty,m,n) for i in 1:10] + bBbad = [rand(elty,m,n) for i in 1:9] + # move to device + bd_A = CuArray{elty, 2}[] + bd_B = CuArray{elty, 2}[] + bd_Bbad = CuArray{elty, 2}[] + for i in 1:length(bA) + push!(bd_A,CuArray(bA[i])) + push!(bd_B,CuArray(bB[i])) + end + for i in 1:length(bBbad) + push!(bd_Bbad,CuArray(bBbad[i])) + end + # compute + CUBLAS.trsm_batched!('L','U','N','N',alpha,bd_A,bd_B) + @test_throws DimensionMismatch CUBLAS.trsm_batched!('L','U','N','N',alpha,bd_A,bd_Bbad) + # move to host and compare + for i in 1:length(bd_B) + bC = alpha*(bA[i]\bB[i]) + h_C = Array(bd_B[i]) + #compare + @test bC ≈ h_C + end + end + + @testset "trsm_batched" begin + # generate parameter + alpha = rand(elty) + # generate matrices + bA = [rand(elty,m,m) for i in 1:10] + map!((x) -> triu(x), bA, bA) + bB = [rand(elty,m,n) for i in 1:10] + # move to device + bd_A = CuArray{elty, 2}[] + bd_B = CuArray{elty, 2}[] + for i in 1:length(bA) + push!(bd_A,CuArray(bA[i])) + push!(bd_B,CuArray(bB[i])) + end + # compute + bd_C = CUBLAS.trsm_batched('L','U','N','N',alpha,bd_A,bd_B) + # move to host and compare + for i in 1:length(bd_C) + bC = alpha*(bA[i]\bB[i]) + h_C = Array(bd_C[i]) + @test bC ≈ h_C + end + end + + let A = triu(rand(elty, m, m)), B = rand(elty,m,n), alpha = rand(elty) + dA = CuArray(A) + dB = CuArray(B) + + @testset "left trsm!" begin + C = alpha*(A\B) + dC = copy(dB) + CUBLAS.trsm!('L','U','N','N',alpha,dA,dC) + @test C ≈ Array(dC) + end + + @testset "left trsm" begin + C = alpha*(A\B) + dC = CUBLAS.trsm('L','U','N','N',alpha,dA,dB) + @test C ≈ Array(dC) + end + @testset "left trsm (adjoint)" begin + C = alpha*(adjoint(A)\B) + dC = CUBLAS.trsm('L','U','C','N',alpha,dA,dB) + @test C ≈ Array(dC) + end + @testset "left trsm (transpose)" begin + C = alpha*(transpose(A)\B) + dC = CUBLAS.trsm('L','U','T','N',alpha,dA,dB) + @test C ≈ Array(dC) + end + end + + @testset "triangular ldiv!" begin + A = triu(rand(elty, m, m)) + B = rand(elty, m,m) + + dA = CuArray(A) + dB = CuArray(B) + + for t in (identity, transpose, adjoint), TR in (UpperTriangular, LowerTriangular, UnitUpperTriangular, UnitLowerTriangular) + dC = copy(dB) + ldiv!(t(TR(dA)), dC) + C = t(TR(A)) \ B + @test C ≈ Array(dC) + end + end + + let A = rand(elty, m,m), B = triu(rand(elty, m, m)), alpha = rand(elty) + dA = CuArray(A) + dB = CuArray(B) + + @testset "right trsm!" begin + C = alpha*(A/B) + dC = copy(dA) + CUBLAS.trsm!('R','U','N','N',alpha,dB,dC) + @test C ≈ Array(dC) + end + + @testset "right trsm" begin + C = alpha*(A/B) + dC = CUBLAS.trsm('R','U','N','N',alpha,dB,dA) + @test C ≈ Array(dC) + end + @testset "right trsm (adjoint)" begin + C = alpha*(A/adjoint(B)) + dC = CUBLAS.trsm('R','U','C','N',alpha,dB,dA) + @test C ≈ Array(dC) + end + @testset "right trsm (transpose)" begin + C = alpha*(A/transpose(B)) + dC = CUBLAS.trsm('R','U','T','N',alpha,dB,dA) + @test C ≈ Array(dC) + end + end + + @testset "triangular rdiv!" begin + A = rand(elty, m,m) + B = triu(rand(elty, m, m)) + + dA = CuArray(A) + dB = CuArray(B) + + for t in (identity, transpose, adjoint), TR in (UpperTriangular, LowerTriangular, UnitUpperTriangular, UnitLowerTriangular) + dC = copy(dA) + rdiv!(dC, t(TR(dB))) + C = A / t(TR(B)) + @test C ≈ Array(dC) + end + end + + @testset "Diagonal rdiv!" begin + A = rand(elty, m,m) + B = Diagonal(rand(elty, m)) + + dA = CuArray(A) + dB = CuArray(B) + + C = A / B + rdiv!(dA, dB) + @test C ≈ Array(dA) + end + + @testset "triangular-dense mul!" begin + A = triu(rand(elty, m, m)) + B = rand(elty,m,n) + C = zeros(elty,m,n) + + sA = rand(elty,m,m) + sA = sA + transpose(sA) + + for t in (identity, transpose, adjoint), TR in (UpperTriangular, LowerTriangular, UnitUpperTriangular, UnitLowerTriangular) + A = copy(sA) |> TR + B_L = copy(B) + B_R = copy(B') + C_L = copy(C) + C_R = copy(C') + dA = CuArray(parent(A)) |> TR + dB_L = CuArray(parent(B_L)) + dB_R = CuArray(parent(B_R)) + dC_L = CuArray(C_L) + dC_R = CuArray(C_R) + + D_L = mul!(C_L, t(A), B_L) + dD_L = mul!(dC_L, t(dA), dB_L) + + D_R = mul!(C_R, B_R, t(A)) + dD_R = mul!(dC_R, dB_R, t(dA)) + + @test C_L ≈ Array(dC_L) + @test D_L ≈ Array(dD_L) + @test C_R ≈ Array(dC_R) + @test D_R ≈ Array(dD_R) + end + end + + @testset "triangular-triangular mul!" begin + A = triu(rand(elty, m, m)) + B = triu(rand(elty, m, m)) + C0 = zeros(elty,m,m) + + sA = rand(elty,m,m) + sA = sA + transpose(sA) + sB = rand(elty,m,m) + sB = sB + transpose(sB) + + for (TRa, ta, TRb, tb, TRc) in ( + (UpperTriangular, identity, LowerTriangular, identity, Matrix), + (LowerTriangular, identity, UpperTriangular, identity, Matrix), + (UpperTriangular, identity, UpperTriangular, transpose, Matrix), + (UpperTriangular, transpose, UpperTriangular, identity, Matrix), + (LowerTriangular, identity, LowerTriangular, transpose, Matrix), + (LowerTriangular, transpose, LowerTriangular, identity, Matrix), + ) + + A = copy(sA) |> TRa + B = copy(sB) |> TRb + C = copy(C0) |> TRc + dA = CuArray(parent(sA)) |> TRa + dB = CuArray(parent(sB)) |> TRb + dC = if TRc == Matrix + CuArray(C0) |> DenseCuMatrix + else + CuArray(C0) |> TRc + end + + D = mul!(C, ta(A), tb(B)) + dD = mul!(dC, ta(dA), tb(dB)) + + @test C ≈ Array(dC) + @test D ≈ Array(dD) + end + end + + if elty <: Complex + @testset "hemm!" begin + alpha = rand(elty) + beta = rand(elty) + hA = rand(elty,m,m) + hA = hA + hA' + dhA = CuArray(hA) + B = rand(elty,m,n) + C = rand(elty,m,n) + d_B = CuArray(B) + d_C = CuArray(C) + # compute + C = alpha*(hA*B) + beta*C + CUBLAS.hemm!('L','L',alpha,dhA,d_B,beta,d_C) + # move to host and compare + h_C = Array(d_C) + @test C ≈ h_C + end + @testset "hemm" begin + hA = rand(elty,m,m) + hA = hA + hA' + dhA = CuArray(hA) + B = rand(elty,m,n) + d_B = CuArray(B) + C = hA*B + d_C = CUBLAS.hemm('L','U',dhA,d_B) + # move to host and compare + h_C = Array(d_C) + @test C ≈ h_C + end + @testset "xt_hemm! gpu" begin + alpha = rand(elty) + beta = rand(elty) + hA = rand(elty,m,m) + hA = hA + hA' + dhA = CuArray(hA) + B = rand(elty,m,n) + C = rand(elty,m,n) + d_B = CuArray(B) + d_C = CuArray(C) + # compute + C = alpha*(hA*B) + beta*C + CUBLAS.xt_hemm!('L','L',alpha,dhA,d_B,beta,d_C) + # move to host and compare + h_C = Array(d_C) + @test C ≈ h_C + end + @testset "xt_hemm! cpu" begin + alpha = rand(elty) + beta = rand(elty) + hA = rand(elty,m,m) + hA = hA + hA' + B = rand(elty,m,n) + C = rand(elty,m,n) + # compute + h_C = copy(C) + C = alpha*(hA*B) + beta*C + CUBLAS.xt_hemm!('L','L',alpha,copy(hA),copy(B),beta,h_C) + @test C ≈ h_C + end + @testset "xt_hemm gpu" begin + hA = rand(elty,m,m) + hA = hA + hA' + dhA = CuArray(hA) + B = rand(elty,m,n) + d_B = CuArray(B) + C = hA*B + d_C = CUBLAS.xt_hemm('L','U',dhA, d_B) + # move to host and compare + @test d_C isa CuArray + h_C = Array(d_C) + @test C ≈ h_C + end + @testset "xt_hemm cpu" begin + hA = rand(elty,m,m) + hA = hA + hA' + B = rand(elty,m,n) + C = hA*B + h_C = CUBLAS.xt_hemm('L','U',copy(hA), copy(B)) + # move to host and compare + @test h_C isa Array + @test C ≈ h_C + end + end + @testset "geam!" begin + alpha = rand(elty) + beta = rand(elty) + A = rand(elty,m,n) + B = rand(elty,m,n) + C = zeros(elty,m,n) + d_A = CuArray(A) + d_B = CuArray(B) + d_C = CuArray(C) + # compute + D = alpha*A + beta*B + CUBLAS.geam!('N','N',alpha,d_A,beta,d_B,d_C) + # move to host and compare + h_C = Array(d_C) + @test D ≈ h_C + + #test in place versions too + d_C = CuArray(C) + D = alpha*C + beta*B + CUBLAS.geam!('N','N',alpha,d_C,beta,d_B,d_C) + # move to host and compare + h_C = Array(d_C) + @test D ≈ h_C + + d_C = CuArray(C) + D = alpha*A + beta*C + CUBLAS.geam!('N','N',alpha,d_A,beta,d_C,d_C) + # move to host and compare + h_C = Array(d_C) + @test D ≈ h_C + + #test setting C to zero + CUBLAS.geam!('N','N',zero(elty),d_A,zero(elty),d_B,d_C) + h_C = Array(d_C) + @test h_C ≈ zeros(elty,m,n) + + # bounds checking + @test_throws DimensionMismatch CUBLAS.geam!('N','T',alpha,d_A,beta,d_B,d_C) + @test_throws DimensionMismatch CUBLAS.geam!('T','T',alpha,d_A,beta,d_B,d_C) + @test_throws DimensionMismatch CUBLAS.geam!('T','N',alpha,d_A,beta,d_B,d_C) + end + + @testset "geam" begin + alpha = rand(elty) + beta = rand(elty) + A = rand(elty, m, n) + d_A = CuArray(A) + B = rand(elty, m, n) + d_B = CuArray(B) + D = alpha*A + beta*B + d_C = CUBLAS.geam('N','N',alpha,d_A,beta,d_B) + # move to host and compare + h_C = Array(d_C) + @test D ≈ h_C + end + @testset "CuMatrix -- A ± B -- $elty" begin + for opa in (identity, transpose, adjoint) + for opb in (identity, transpose, adjoint) + n = 10 + m = 20 + geam_A = opa == identity ? rand(elty, n, m) : rand(elty, m, n) + geam_B = opb == identity ? rand(elty, n, m) : rand(elty, m, n) + + geam_dA = CuMatrix{elty}(geam_A) + geam_dB = CuMatrix{elty}(geam_B) + + geam_C = opa(geam_A) + opb(geam_B) + geam_dC = opa(geam_dA) + opb(geam_dB) + @test geam_C ≈ collect(geam_dC) + + geam_C = opa(geam_A) - opb(geam_B) + geam_dC = opa(geam_dA) - opb(geam_dB) + @test geam_C ≈ collect(geam_dC) + end + end + end + #A = rand(elty,m,k) + #d_A = CuArray(A) + @testset "syrkx!" begin + alpha = rand(elty) + beta = rand(elty) + # generate matrices + syrkx_A = rand(elty, n, k) + syrkx_B = rand(elty, n, k) + syrkx_C = rand(elty, n, n) + syrkx_C += syrkx_C' + d_syrkx_A = CuArray(syrkx_A) + d_syrkx_B = CuArray(syrkx_B) + d_syrkx_C = CuArray(syrkx_C) + # C = (alpha*A)*transpose(B) + beta*C + d_syrkx_C = CUBLAS.syrkx!('U','N',alpha,d_syrkx_A,d_syrkx_B,beta,d_syrkx_C) + final_C = (alpha*syrkx_A)*transpose(syrkx_B) + beta*syrkx_C + # move to host and compare + h_C = Array(d_syrkx_C) + @test triu(final_C) ≈ triu(h_C) + # C = A*transpose(B) + d_syrkx_C = CUBLAS.syrkx('U','N',d_syrkx_A,d_syrkx_B) + final_C = syrkx_A*transpose(syrkx_B) + # move to host and compare + h_C = Array(d_syrkx_C) + @test triu(final_C) ≈ triu(h_C) + badC = rand(elty, m, n) + d_badC = CuArray(badC) + @test_throws DimensionMismatch CUBLAS.syrkx!('U','N',alpha,d_syrkx_A,d_syrkx_B,beta,d_badC) + badC = rand(elty, n+1, n+1) + d_badC = CuArray(badC) + @test_throws DimensionMismatch CUBLAS.syrkx!('U','N',alpha,d_syrkx_A,d_syrkx_B,beta,d_badC) + end + @testset "xt_syrkx! gpu" begin + alpha = rand(elty) + beta = rand(elty) + # generate matrices + syrkx_A = rand(elty, n, k) + syrkx_B = rand(elty, n, k) + syrkx_C = rand(elty, n, n) + syrkx_C += syrkx_C' + d_syrkx_A = CuArray(syrkx_A) + d_syrkx_B = CuArray(syrkx_B) + d_syrkx_C = CuArray(syrkx_C) + # C = (alpha*A)*transpose(B) + beta*C + synchronize() + d_syrkx_C = CUBLAS.xt_syrkx!('U','N',alpha,d_syrkx_A,d_syrkx_B,beta,d_syrkx_C) + final_C = (alpha*syrkx_A)*transpose(syrkx_B) + beta*syrkx_C + # move to host and compare + h_C = Array(d_syrkx_C) + @test triu(final_C) ≈ triu(h_C) + badC = rand(elty, m, n) + d_badC = CuArray(badC) + @test_throws DimensionMismatch CUBLAS.xt_syrkx!('U','N',alpha,d_syrkx_A,d_syrkx_B,beta,d_badC) + badC = rand(elty, n+1, n+1) + d_badC = CuArray(badC) + @test_throws DimensionMismatch CUBLAS.xt_syrkx!('U','N',alpha,d_syrkx_A,d_syrkx_B,beta,d_badC) + end + @testset "xt_syrkx! cpu" begin + alpha = rand(elty) + beta = rand(elty) + # generate matrices + syrkx_A = rand(elty, n, k) + syrkx_B = rand(elty, n, k) + syrkx_C = rand(elty, n, n) + syrkx_C += syrkx_C' + final_C = (alpha*syrkx_A)*transpose(syrkx_B) + beta*syrkx_C + CUBLAS.xt_syrkx!('U','N',alpha,syrkx_A,syrkx_B,beta,syrkx_C) + # move to host and compare + @test triu(final_C) ≈ triu(syrkx_C) + end + @testset "xt_syrkx gpu" begin + # generate matrices + syrkx_A = rand(elty, n, k) + syrkx_B = rand(elty, n, k) + d_syrkx_A = CuArray(syrkx_A) + d_syrkx_B = CuArray(syrkx_B) + synchronize() + d_syrkx_C = CUBLAS.xt_syrkx('U','N',d_syrkx_A,d_syrkx_B) + final_C = syrkx_A*transpose(syrkx_B) + # move to host and compare + @test d_syrkx_C isa CuArray + h_C = Array(d_syrkx_C) + @test triu(final_C) ≈ triu(h_C) + end + @testset "xt_syrkx cpu" begin + # generate matrices + syrkx_A = rand(elty, n, k) + syrkx_B = rand(elty, n, k) + h_C = CUBLAS.xt_syrkx('U','N',syrkx_A,syrkx_B) + final_C = syrkx_A*transpose(syrkx_B) + @test h_C isa Array + @test triu(final_C) ≈ triu(h_C) + end + @testset "syrk" begin + A = rand(elty,m,k) + d_A = CuArray(A) + # C = A*transpose(A) + d_C = CUBLAS.syrk('U','N',d_A) + C = A*transpose(A) + C = triu(C) + # move to host and compare + h_C = Array(d_C) + h_C = triu(C) + @test C ≈ h_C + end + @testset "xt_syrk gpu" begin + # C = A*transpose(A) + A = rand(elty,m,k) + d_A = CuArray(A) + d_C = CUBLAS.xt_syrk('U','N',d_A) + C = A*transpose(A) + C = triu(C) + # move to host and compare + @test d_C isa CuArray + h_C = Array(d_C) + h_C = triu(C) + @test C ≈ h_C + end + @testset "xt_syrk cpu" begin + A = rand(elty,m,k) + # C = A*transpose(A) + h_C = CUBLAS.xt_syrk('U','N',copy(A)) + C = A*transpose(A) + C = triu(C) + # move to host and compare + @test h_C isa Array + h_C = triu(C) + @test C ≈ h_C + end + if elty <: Complex + @testset "herk!" begin + alpha = rand(elty) + beta = rand(elty) + A = rand(elty,m,m) + hA = A + A' + d_A = CuArray(A) + d_C = CuArray(hA) + CUBLAS.herk!('U','N',real(alpha),d_A,real(beta),d_C) + C = real(alpha)*(A*A') + real(beta)*hA + C = triu(C) + # move to host and compare + h_C = Array(d_C) + h_C = triu(C) + @test C ≈ h_C + end + @testset "herk" begin + A = rand(elty,m,m) + d_A = CuArray(A) + d_C = CUBLAS.herk('U','N',d_A) + C = A*A' + C = triu(C) + # move to host and compare + h_C = Array(d_C) + h_C = triu(C) + @test C ≈ h_C + end + @testset "xt_herk! gpu" begin + alpha = rand(elty) + beta = rand(elty) + A = rand(elty,m,m) + hA = A + A' + C = real(alpha)*(A*A') + real(beta)*copy(hA) + d_A = CuArray(A) + d_C = CuArray(hA) + synchronize() + CUBLAS.xt_herk!('U','N',real(alpha),d_A,real(beta),d_C) + C = triu(C) + # move to host and compare + h_C = Array(d_C) + h_C = triu(h_C) + @test C ≈ h_C + end + @testset "xt_herk! cpu" begin + alpha = rand(elty) + beta = rand(elty) + A = rand(elty,m,m) + hA = A + A' + h_C = copy(hA) + CUBLAS.xt_herk!('U','N',real(alpha),copy(A),real(beta),h_C) + C = real(alpha)*(A*A') + real(beta)*copy(hA) + C = triu(C) + # move to host and compare + h_C = triu(h_C) + @test C ≈ h_C + end + @testset "xt_herk gpu" begin + A = rand(elty,m,m) + d_A = CuArray(A) + synchronize() + d_C = CUBLAS.xt_herk('U','N',d_A) + C = A*A' + C = triu(C) + # move to host and compare + @test d_C isa CuArray + h_C = Array(d_C) + h_C = triu(h_C) + @test C ≈ h_C + end + @testset "xt_herk cpu" begin + A = rand(elty,m,m) + h_C = CUBLAS.xt_herk('U','N',copy(A)) + C = A*A' + C = triu(C) + # move to host and compare + @test h_C isa Array + h_C = triu(h_C) + @test C ≈ h_C + end + end + @testset "syr2k!" begin + alpha = rand(elty) + beta = rand(elty) + A = rand(elty,m,k) + B = rand(elty,m,k) + Bbad = rand(elty,m+1,k+1) + C = rand(elty,m,m) + C = C + transpose(C) + # move to device + d_A = CuArray(A) + d_B = CuArray(B) + d_Bbad = CuArray(Bbad) + d_C = CuArray(C) + # compute + C = alpha*(A*transpose(B) + B*transpose(A)) + beta*C + CUBLAS.syr2k!('U','N',alpha,d_A,d_B,beta,d_C) + # move back to host and compare + C = triu(C) + h_C = Array(d_C) + h_C = triu(h_C) + @test C ≈ h_C + @test_throws DimensionMismatch CUBLAS.syr2k!('U','N',alpha,d_A,d_Bbad,beta,d_C) + end + + @testset "syr2k" begin + alpha = rand(elty) + A = rand(elty,m,k) + B = rand(elty,m,k) + # move to device + d_A = CuArray(A) + d_B = CuArray(B) + C = alpha*(A*transpose(B) + B*transpose(A)) + d_C = CUBLAS.syr2k('U','N',alpha,d_A,d_B) + # move back to host and compare + C = triu(C) + h_C = Array(d_C) + h_C = triu(h_C) + @test C ≈ h_C + end + if elty <: Complex + @testset "her2k!" begin + elty1 = elty + elty2 = real(elty) + # generate parameters + α = rand(elty1) + β = rand(elty2) + A = rand(elty,m,k) + B = rand(elty,m,k) + Bbad = rand(elty,m+1,k+1) + C = rand(elty,m,m) + # move to device + d_A = CuArray(A) + d_B = CuArray(B) + d_Bbad = CuArray(Bbad) + C = C + C' + d_C = CuArray(C) + C = α*(A*B') + conj(α)*(B*A') + β*C + CUBLAS.her2k!('U','N',α,d_A,d_B,β,d_C) + # move back to host and compare + C = triu(C) + h_C = Array(d_C) + h_C = triu(h_C) + @test C ≈ h_C + @test_throws DimensionMismatch CUBLAS.her2k!('U','N',α,d_A,d_Bbad,β,d_C) + end + + @testset "her2k" begin + A = rand(elty,m,k) + B = rand(elty,m,k) + d_A = CuArray(A) + d_B = CuArray(B) + C = A*B' + B*A' + d_C = CUBLAS.her2k('U','N',d_A,d_B) + # move back to host and compare + C = triu(C) + h_C = Array(d_C) + h_C = triu(h_C) + @test C ≈ h_C + end + @testset "xt_her2k! gpu" begin + elty1 = elty + elty2 = real(elty) + # generate parameters + α = rand(elty1) + β = rand(elty2) + A = rand(elty,m,k) + B = rand(elty,m,k) + d_A = CuArray(A) + d_B = CuArray(B) + C = rand(elty,m,m) + C = C + C' + d_C = CuArray(C) + C = α*(A*B') + conj(α)*(B*A') + β*C + synchronize() + CUBLAS.xt_her2k!('U','N',α,d_A,d_B,β,d_C) + # move back to host and compare + C = triu(C) + h_C = Array(d_C) + h_C = triu(h_C) + @test C ≈ h_C + end + @testset "xt_her2k! cpu" begin + elty1 = elty + elty2 = real(elty) + # generate parameters + α = rand(elty1) + β = rand(elty2) + A = rand(elty,m,k) + B = rand(elty,m,k) + C = rand(elty,m,m) + C = C + C' + h_C = copy(C) + C = α*(A*B') + conj(α)*(B*A') + β*C + CUBLAS.xt_her2k!('U','N',α,A,B,β,h_C) + # move back to host and compare + C = triu(C) + h_C = triu(h_C) + @test C ≈ h_C + end + @testset "xt_her2k gpu" begin + # generate parameters + A = rand(elty,m,k) + B = rand(elty,m,k) + d_A = CuArray(A) + d_B = CuArray(B) + C = rand(elty,m,m) + C = C + C' + C = (A*B') + (B*A') + synchronize() + d_C = CUBLAS.xt_her2k('U','N',d_A,d_B) + # move back to host and compare + C = triu(C) + @test d_C isa CuArray + h_C = Array(d_C) + h_C = triu(h_C) + @test C ≈ h_C + end + @testset "xt_her2k cpu" begin + A = rand(elty,m,k) + B = rand(elty,m,k) + C = rand(elty,m,m) + # generate parameters + C = C + C' + C = (A*B') + (B*A') + h_C = CUBLAS.xt_her2k('U','N',A,B) + # move back to host and compare + @test h_C isa Array + C = triu(C) + h_C = triu(h_C) + @test C ≈ h_C + end + @testset "her2k" begin + A = rand(elty,m,k) + B = rand(elty,m,k) + d_A = CuArray(A) + d_B = CuArray(B) + C = A*B' + B*A' + d_C = CUBLAS.her2k('U','N',d_A,d_B) + # move back to host and compare + C = triu(C) + h_C = Array(d_C) + h_C = triu(h_C) + @test C ≈ h_C + end + end + end + + @testset "elty = $elty" for elty in [Float16, Float32, Float64, ComplexF32, ComplexF64] + elty == Float16 && capability(device()) < v"5.3" && continue + + alpha = rand(elty) + beta = rand(elty) + # generate matrices + bA = [rand(elty,m,k) for i in 1:10] + bB = [rand(elty,k,n) for i in 1:10] + bC = [rand(elty,m,n) for i in 1:10] + # move to device + bd_A = CuArray{elty, 2}[] + bd_B = CuArray{elty, 2}[] + bd_C = CuArray{elty, 2}[] + bd_bad = CuArray{elty, 2}[] + for i in 1:length(bA) + push!(bd_A,CuArray(bA[i])) + push!(bd_B,CuArray(bB[i])) + push!(bd_C,CuArray(bC[i])) + if i < length(bA) - 2 + push!(bd_bad,CuArray(bC[i])) + end + end + + @testset "gemm_batched!" begin + # C = (alpha*A)*B + beta*C + CUBLAS.gemm_batched!('N','N',alpha,bd_A,bd_B,beta,bd_C) + for i in 1:length(bd_C) + bC[i] = (alpha*bA[i])*bB[i] + beta*bC[i] + h_C = Array(bd_C[i]) + #compare + @test bC[i] ≈ h_C + end + @test_throws DimensionMismatch CUBLAS.gemm_batched!('N','N',alpha,bd_A,bd_bad,beta,bd_C) + end + + @testset "gemm_batched" begin + bd_C = CUBLAS.gemm_batched('N','N',bd_A,bd_B) + for i in 1:length(bA) + bC[i] = bA[i]*bB[i] + h_C = Array(bd_C[i]) + @test bC[i] ≈ h_C + end + @test_throws DimensionMismatch CUBLAS.gemm_batched('N','N',alpha,bd_A,bd_bad) + end + + @testset "gemmBatchedEx!" begin + # C = (alpha*A)*B + beta*C + CUBLAS.gemmBatchedEx!('N','N',alpha,bd_A,bd_B,beta,bd_C) + for i in 1:length(bd_C) + bC[i] = (alpha*bA[i])*bB[i] + beta*bC[i] + h_C = Array(bd_C[i]) + #compare + @test bC[i] ≈ h_C + end + @test_throws DimensionMismatch CUBLAS.gemmBatchedEx!('N','N',alpha,bd_A,bd_bad,beta,bd_C) + end + + nbatch = 10 + bA = rand(elty, m, k, nbatch) + bB = rand(elty, k, n, nbatch) + bC = rand(elty, m, n, nbatch) + bbad = rand(elty, m+1, n+1, nbatch) + # move to device + bd_A = CuArray{elty, 3}(bA) + bd_B = CuArray{elty, 3}(bB) + bd_C = CuArray{elty, 3}(bC) + bd_bad = CuArray{elty, 3}(bbad) + + @testset "gemm_strided_batched!" begin + CUBLAS.gemm_strided_batched!('N', 'N', alpha, bd_A, bd_B, beta, bd_C) + for i in 1:nbatch + bC[:, :, i] = (alpha * bA[:, :, i]) * bB[:, :, i] + beta * bC[:, :, i] + end + h_C = Array(bd_C) + @test bC ≈ h_C + @test_throws DimensionMismatch CUBLAS.gemm_strided_batched!('N', 'N', alpha, bd_A, bd_B, beta, bd_bad) + end + + @testset "gemmStridedBatchedEx!" begin + CUBLAS.gemmStridedBatchedEx!('N', 'N', alpha, bd_A, bd_B, beta, bd_C) + for i in 1:nbatch + bC[:, :, i] = (alpha * bA[:, :, i]) * bB[:, :, i] + beta * bC[:, :, i] + end + h_C = Array(bd_C) + @test bC ≈ h_C + @test_throws DimensionMismatch CUBLAS.gemmStridedBatchedEx!('N', 'N', alpha, bd_A, bd_B, beta, bd_bad) + end + + @testset "gemm_strided_batched" begin + bd_C = CUBLAS.gemm_strided_batched('N', 'N', bd_A, bd_B) + + for i in 1:nbatch + bC[:, :, i] = bA[:, :, i] * bB[:, :, i] + end + h_C = Array(bd_C) + @test bC ≈ h_C + # generate matrices + bA = rand(elty, k, m, nbatch) + bB = rand(elty, k, n, nbatch) + bC = zeros(elty, m, n, nbatch) + # move to device + bd_A = CuArray{elty, 3}(bA) + bd_B = CuArray{elty, 3}(bB) + + bd_C = CUBLAS.gemm_strided_batched('T', 'N', bd_A, bd_B) + for i in 1:nbatch + bC[:, :, i] = transpose(bA[:, :, i]) * bB[:, :, i] + end + h_C = Array(bd_C) + @test bC ≈ h_C + @test_throws DimensionMismatch CUBLAS.gemm_strided_batched('N', 'N', alpha, bd_A, bd_bad) + end + end + + if CUDA.CUBLAS.version() >= v"12.4.2" + @testset "elty = $elty" for elty in [Float32, Float64] + num_groups = 10 + group_sizes = collect(1:num_groups) + transA = ['N' for i in 1:num_groups] + transB = ['N' for i in 1:num_groups] + alpha = rand(elty, num_groups) + beta = rand(elty, num_groups) + # generate matrices + bA = [[rand(elty,3*i,2*i) for j in 1:group_sizes[i]] for i in 1:num_groups] + bB = [[rand(elty,2*i,5*i) for j in 1:group_sizes[i]] for i in 1:num_groups] + bC = [[rand(elty,3*i,5*i) for j in 1:group_sizes[i]] for i in 1:num_groups] + # move to device + bd_A = [[CuArray(bA[i][j]) for j in 1:group_sizes[i]] for i in 1:num_groups] + bd_B = [[CuArray(bB[i][j]) for j in 1:group_sizes[i]] for i in 1:num_groups] + bd_C = [[CuArray(bC[i][j]) for j in 1:group_sizes[i]] for i in 1:num_groups] + @testset "gemm_grouped_batched!" begin + # C = (alpha*A)*B + beta*C + CUBLAS.gemm_grouped_batched!(transA,transB,alpha,bd_A,bd_B,beta,bd_C) + for i in 1:num_groups, j in 1:group_sizes[i] + bC[i][j] = alpha[i] * bA[i][j] * bB[i][j] + beta[i] * bC[i][j] + h_C = Array(bd_C[i][j]) + @test bC[i][j] ≈ h_C + end + end + + @testset "gemm_grouped_batched" begin + bd_C = CUBLAS.gemm_grouped_batched(transA,transB,bd_A,bd_B) + for i in 1:num_groups, j in 1:group_sizes[i] + bC[i][j] = bA[i][j] * bB[i][j] + h_C = Array(bd_C[i][j]) + @test bC[i][j] ≈ h_C + end + end + end + end + + # Group size hardcoded to one + if CUDA.CUBLAS.version() >= v"12.4.2" + @testset "elty = $elty" for elty in [Float32, Float64] + + transA = ['N' for i in 1:10] + transB = ['N' for i in 1:10] + alpha = rand(elty, 10) + beta = rand(elty, 10) + # generate matrices + bA = [rand(elty,3*i,2*i) for i in 1:10] + bB = [rand(elty,2*i,5*i) for i in 1:10] + bC = [rand(elty,3*i,5*i) for i in 1:10] + # move to device + bd_A = CuArray{elty, 2}[] + bd_B = CuArray{elty, 2}[] + bd_C = CuArray{elty, 2}[] + for i in 1:length(bA) + push!(bd_A,CuArray(bA[i])) + push!(bd_B,CuArray(bB[i])) + push!(bd_C,CuArray(bC[i])) + end + + @testset "gemm_grouped_batched!" begin + # C = (alpha*A)*B + beta*C + CUBLAS.gemm_grouped_batched!(transA,transB,alpha,bd_A,bd_B,beta,bd_C) + for i in 1:length(bd_C) + bC[i] = alpha[i] * bA[i] * bB[i] + beta[i] * bC[i] + h_C = Array(bd_C[i]) + @test bC[i] ≈ h_C + end + end + + @testset "gemm_grouped_batched" begin + bd_C = CUBLAS.gemm_grouped_batched(transA,transB,bd_A,bd_B) + for i in 1:length(bd_C) + bC[i] = bA[i] * bB[i] + h_C = Array(bd_C[i]) + @test bC[i] ≈ h_C + end + end + end + end + + @testset "mixed-precision matmul" begin + m,k,n = 4,4,4 + cudaTypes = (Float16, Complex{Float16}, BFloat16, Complex{BFloat16}, Float32, Complex{Float32}, + Float64, Complex{Float64}, Int8, Complex{Int8}, UInt8, Complex{UInt8}, + Int16, Complex{Int16}, UInt16, Complex{UInt16}, Int32, Complex{Int32}, + UInt32, Complex{UInt32}, Int64, Complex{Int64}, UInt64, Complex{UInt64}) + + for AT in cudaTypes, CT in cudaTypes + BT = AT # gemmEx requires identical A and B types + + # we only test combinations of types that are supported by gemmEx + if CUBLAS.gemmExComputeType(AT, BT, CT, m,k,n) !== nothing + A = AT <: BFloat16 ? AT.(rand(m,k)) : rand(AT, m,k) + B = BT <: BFloat16 ? BT.(rand(k,n)) : rand(BT, k,n) + C = similar(B, CT) + mul!(C, A, B) + + # Base can't do Int8*Int8 without losing accuracy + if (AT == Int8 && BT == Int8) || (AT == Complex{Int8} && BT == Complex{Int8}) + C = CT.(A) * CT.(B) + end + + dA = CuArray(A) + dB = CuArray(B) + dC = similar(dB, CT) + mul!(dC, dA, dB) + + rtol = Base.rtoldefault(AT, BT, 0) + @test C ≈ Array(dC) rtol=rtol + end + end + + # also test an unsupported combination (falling back to GPUArrays) + if VERSION < v"1.11-" # JuliaGPU/CUDA.jl#2441 + AT=BFloat16 + BT=Int32 + CT=Float64 + + A = AT.(rand(m,k)) + B = rand(BT, k,n) + C = similar(B, CT) + mul!(C, A, B) + + dA = CuArray(A) + dB = CuArray(B) + dC = similar(dB, CT) + mul!(dC, dA, dB) + + rtol = Base.rtoldefault(AT, BT, 0) + @test C ≈ Array(dC) rtol=rtol + end + end + + @testset "gemm! with strided inputs" begin # JuliaGPU/CUDA.jl#78 + inn = 784; out = 32 + testf(randn(784*100), rand(Float32, 784, 100)) do p, x + p[reshape(1:(out*inn),out,inn)] * x + @view(p[reshape(1:(out*inn),out,inn)]) * x + end + end +end From eba40e55f48cd75bf26c1d1ef7464db0a3259ddf Mon Sep 17 00:00:00 2001 From: Katharine Hyatt Date: Tue, 7 Jan 2025 18:13:25 -0500 Subject: [PATCH 2/6] fix missing C2 --- test/libraries/cublas_level3.jl | 9 +++------ 1 file changed, 3 insertions(+), 6 deletions(-) diff --git a/test/libraries/cublas_level3.jl b/test/libraries/cublas_level3.jl index 043677a013..057284c51d 100644 --- a/test/libraries/cublas_level3.jl +++ b/test/libraries/cublas_level3.jl @@ -170,15 +170,12 @@ k = 13 @testset "xt_gemm cpu" begin A = rand(elty,m,k) B = rand(elty,k,n) - d_A = CuArray(A) - d_B = CuArray(B) - C = CUBLAS.xt_gemm('N','N',A,B) - d_C2 = d_A * d_B + C = CUBLAS.xt_gemm('N','N',A,B) + C2 = A*B # compare @test C isa Array - h_C2 = Array(C2) @test C ≈ A*B - @test C ≈ Array(d_C2) + @test C ≈ C2 end @testset "symm!" begin From 6be7992b36734896bc19b84717c16ee5a71b6445 Mon Sep 17 00:00:00 2001 From: Katharine Hyatt Date: Tue, 7 Jan 2025 19:40:59 -0500 Subject: [PATCH 3/6] Fix bad zeros --- test/libraries/cublas_level3.jl | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/test/libraries/cublas_level3.jl b/test/libraries/cublas_level3.jl index 057284c51d..fd52ef3ea8 100644 --- a/test/libraries/cublas_level3.jl +++ b/test/libraries/cublas_level3.jl @@ -316,7 +316,7 @@ k = 13 A = triu(rand(elty, m, m)) B = rand(elty,m,n) C = alpha*A*B - h_C = zeros(C) + h_C = zeros(elty, m, n) CUBLAS.xt_trmm!('L','U','N','N',alpha,copy(A),copy(B),h_C) @test C ≈ h_C end From 6fef63294229e31481ef73317d3aa78a9a4344bb Mon Sep 17 00:00:00 2001 From: Katharine Hyatt Date: Tue, 7 Jan 2025 20:26:34 -0500 Subject: [PATCH 4/6] Prepend LinearAlgebra to axpby --- test/Project.toml | 1 + test/libraries/cublas_level1.jl | 2 +- 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/test/Project.toml b/test/Project.toml index 5d6ea83e88..97a2e028fa 100644 --- a/test/Project.toml +++ b/test/Project.toml @@ -2,6 +2,7 @@ AbstractFFTs = "621f4979-c628-5d54-868e-fcf4e3e8185c" Adapt = "79e6a3ab-5dfb-504d-930d-738a2a938a0e" BFloat16s = "ab4f0b2a-ad5b-11e8-123f-65d77653426b" +CUDA = "052768ef-5323-5732-b1bb-66c8b64840ba" CUDA_Driver_jll = "4ee394cb-3365-5eb0-8335-949819d2adfc" CUDA_Runtime_jll = "76a88914-d11a-5bdc-97e0-2f5a05c973a2" ChainRulesCore = "d360d2e6-b24c-11e9-a2a3-2a2ae2dbcce4" diff --git a/test/libraries/cublas_level1.jl b/test/libraries/cublas_level1.jl index 0d52febb2b..9bcb81d72f 100644 --- a/test/libraries/cublas_level1.jl +++ b/test/libraries/cublas_level1.jl @@ -81,7 +81,7 @@ k = 13 @test testf(*, rand(T, m)', rand(T, m)) @test testf(norm, rand(T, m)) @test testf(axpy!, Ref(rand()), rand(T, m), rand(T, m)) - @test testf(axpby!, Ref(rand()), rand(T, m), Ref(rand()), rand(T, m)) + @test testf(LinearAlgebra.axpby!, Ref(rand()), rand(T, m), Ref(rand()), rand(T, m)) if T <: Complex @test testf(dot, rand(T, m), rand(T, m)) From 7f8cc3f23b82025cbfb4576a472e0dc2c720e0bb Mon Sep 17 00:00:00 2001 From: Katharine Hyatt Date: Tue, 7 Jan 2025 20:59:59 -0500 Subject: [PATCH 5/6] Fix Project.toml and restore prepend --- test/Project.toml | 1 - test/libraries/cublas_level1.jl | 2 +- 2 files changed, 1 insertion(+), 2 deletions(-) diff --git a/test/Project.toml b/test/Project.toml index 97a2e028fa..5d6ea83e88 100644 --- a/test/Project.toml +++ b/test/Project.toml @@ -2,7 +2,6 @@ AbstractFFTs = "621f4979-c628-5d54-868e-fcf4e3e8185c" Adapt = "79e6a3ab-5dfb-504d-930d-738a2a938a0e" BFloat16s = "ab4f0b2a-ad5b-11e8-123f-65d77653426b" -CUDA = "052768ef-5323-5732-b1bb-66c8b64840ba" CUDA_Driver_jll = "4ee394cb-3365-5eb0-8335-949819d2adfc" CUDA_Runtime_jll = "76a88914-d11a-5bdc-97e0-2f5a05c973a2" ChainRulesCore = "d360d2e6-b24c-11e9-a2a3-2a2ae2dbcce4" diff --git a/test/libraries/cublas_level1.jl b/test/libraries/cublas_level1.jl index 9bcb81d72f..c803de7a3c 100644 --- a/test/libraries/cublas_level1.jl +++ b/test/libraries/cublas_level1.jl @@ -28,7 +28,7 @@ k = 13 @test testf(norm, rand(T, m)) @test testf(BLAS.asum, rand(T, m)) @test testf(axpy!, Ref(rand()), rand(T, m), rand(T, m)) - @test testf(axpby!, Ref(rand()), rand(T, m), Ref(rand()), rand(T, m)) + @test testf(LinearAlgebra.axpby!, Ref(rand()), rand(T, m), Ref(rand()), rand(T, m)) if T <: Complex @test testf(dot, rand(T, m), rand(T, m)) From a7904b53ae60b12344fd43e7766e1d4b989da6ea Mon Sep 17 00:00:00 2001 From: Tim Besard Date: Wed, 8 Jan 2025 09:29:53 +0100 Subject: [PATCH 6/6] Move into subdirectories. --- test/libraries/{cublas_extensions.jl => cublas/extensions.jl} | 0 test/libraries/{cublas_level1.jl => cublas/level1.jl} | 0 test/libraries/{cublas_level2.jl => cublas/level2.jl} | 0 test/libraries/{cublas_level3.jl => cublas/level3.jl} | 0 4 files changed, 0 insertions(+), 0 deletions(-) rename test/libraries/{cublas_extensions.jl => cublas/extensions.jl} (100%) rename test/libraries/{cublas_level1.jl => cublas/level1.jl} (100%) rename test/libraries/{cublas_level2.jl => cublas/level2.jl} (100%) rename test/libraries/{cublas_level3.jl => cublas/level3.jl} (100%) diff --git a/test/libraries/cublas_extensions.jl b/test/libraries/cublas/extensions.jl similarity index 100% rename from test/libraries/cublas_extensions.jl rename to test/libraries/cublas/extensions.jl diff --git a/test/libraries/cublas_level1.jl b/test/libraries/cublas/level1.jl similarity index 100% rename from test/libraries/cublas_level1.jl rename to test/libraries/cublas/level1.jl diff --git a/test/libraries/cublas_level2.jl b/test/libraries/cublas/level2.jl similarity index 100% rename from test/libraries/cublas_level2.jl rename to test/libraries/cublas/level2.jl diff --git a/test/libraries/cublas_level3.jl b/test/libraries/cublas/level3.jl similarity index 100% rename from test/libraries/cublas_level3.jl rename to test/libraries/cublas/level3.jl