diff --git a/src/LazyAlgebra.jl b/src/LazyAlgebra.jl index ee27de6..5969a13 100644 --- a/src/LazyAlgebra.jl +++ b/src/LazyAlgebra.jl @@ -47,6 +47,7 @@ export Scalar, SelfAdjoint, SelfAdjointType, + SimpleFiniteDifferences, SingularSystem, SparseOperator, SymmetricRankOneOperator, @@ -161,6 +162,8 @@ include("blas.jl") include("coder.jl") include("mappings.jl") include("sparse.jl") +include("finitedifferences.jl") +import .FiniteDifferences: SimpleFiniteDifferences include("fft.jl") import .FFT: FFTOperator include("conjgrad.jl") diff --git a/src/finitedifferences.jl b/src/finitedifferences.jl new file mode 100644 index 0000000..a0c9e27 --- /dev/null +++ b/src/finitedifferences.jl @@ -0,0 +1,244 @@ +# +# finitedifferences.jl - +# +# Implement rules for basic operations involving mappings. +# +#------------------------------------------------------------------------------- +# +# This file is part of LazyAlgebra (https://github.com/emmt/LazyAlgebra.jl) +# released under the MIT "Expat" license. +# +# Copyright (c) 2017-2018 Éric Thiébaut. +# + +module FiniteDifferences + +export + SimpleFiniteDifferences + + +using Compat +using ..Coder +using ...LazyAlgebra +import ...LazyAlgebra: vcreate, apply!, HalfHessian +using ...LazyAlgebra: fastrange, @callable + +# Define operator D which implements simple finite differences. Make it +# callable. +struct SimpleFiniteDifferences <: LinearMapping end +@callable SimpleFiniteDifferences + +# Extend the vcreate() and apply!() methods for these operators. The apply!() +# method does all the checking and, then, calls a private method specialized +# for the considered dimensionality. + +function vcreate(::Type{Direct}, ::SimpleFiniteDifferences, + x::AbstractArray{T,N}) where {T<:Real,N} + return Array{T}(undef, (N, size(x)...)) +end + +function vcreate(::Type{Adjoint}, ::SimpleFiniteDifferences, + x::AbstractArray{T,N}) where {T<:Real,N} + N ≥ 2 || + throw(DimensionMismatch("argument must have at least 2 dimensions")) + dims = size(x) + dims[1] == N - 1 || + throw(DimensionMismatch("first dimension should be $(N-1)")) + return Array{T}(undef, dims[2:end]) +end + +# FIXME: The following is not absolutely needed because HalfHessian is +# automatically an Endomorphism. +for P in (Direct, Adjoint) + @eval function vcreate(::Type{$P}, + ::HalfHessian{SimpleFiniteDifferences}, + x::AbstractArray{T,N}) where {T<:Real,N} + return Array{T}(undef, size(x)) + end +end + +function apply!(α::Real, ::Type{<:Direct}, ::SimpleFiniteDifferences, + x::AbstractArray{Tx,Nx}, β::Real, + y::AbstractArray{Ty,Ny}) where {Tx<:Real,Nx,Ty<:Real,Ny} + Ny == Nx + 1 || + throw(DimensionMismatch("incompatible number of dimensions")) + ydims = size(y) + ydims[1]== Nx || + throw(DimensionMismatch("first dimension of destination must be $Nx")) + ydims[2:end] == size(x) || + throw(DimensionMismatch("dimensions 2:end of destination must be $(size(x))")) + if α == 0 + vscale!(y, β) + else + T = promote_type(Tx, Ty) + _apply_D!(convert(T, α), x, convert(T, β), y) + end + return y +end + +function apply!(α::Real, ::Type{<:Adjoint}, ::SimpleFiniteDifferences, + x::AbstractArray{Tx,Nx}, β::Real, + y::AbstractArray{Ty,Ny}) where {Tx<:Real,Nx,Ty<:Real,Ny} + Ny == Nx - 1 || + throw(DimensionMismatch("incompatible number of dimensions")) + xdims = size(x) + xdims[1]== Ny || + throw(DimensionMismatch("first dimension of source must be $Ny")) + xdims[2:end] == size(y) || + throw(DimensionMismatch("dimensions 2:end of source must be $(size(y))")) + vscale!(y, β) + if α != 0 + T = promote_type(Tx, Ty) + _apply_Dt!(convert(T, α), x, convert(T, β), y) + end + return y +end + +function apply!(α::Real, ::Type{<:Union{Direct,Adjoint}}, + ::HalfHessian{SimpleFiniteDifferences}, + x::AbstractArray{Tx,Nx}, β::Real, + y::AbstractArray{Ty,Ny}) where {Tx<:Real,Nx,Ty<:Real,Ny} + Ny == Nx || + throw(DimensionMismatch("incompatible number of dimensions")) + size(x) == size(y) || + throw(DimensionMismatch("source and destination must have the same dimensions")) + vscale!(y, β) + if α != 0 + T = promote_type(Tx, Ty) + _apply_DtD!(convert(T, α), x, convert(T, β), y) + end + return y +end + +offset(::Type{CartesianIndex{N}}, d::Integer, s::Integer=1) where {N} = + CartesianIndex{N}(ntuple(i -> (i == d ? s : 0), N)) + +@generated function _apply_D!(a::T, x::AbstractArray{<:Real,N}, + b::T, y::AbstractArray{<:Real,Np1} + ) where {N,Np1,T<:Real} + # We know that a ≠ 0. + @assert Np1 == N + 1 + D = generate_symbols("d", N) + I = generate_symbols("i", N) + S = generate_symbols("s", N) + common = ( + [:( $(I[d]) = min(i + $(S[d]), imax) ) for d in 1:N]..., + :( xi = x[i] ), + [:( $(D[d]) = x[$(I[d])] - xi ) for d in 1:N]... + ) + return encode( + :( inds = fastrange(size(x)) ), + :( imax = last(inds) ), + [:( $(S[d]) = $(offset(CartesianIndex{N}, d)) ) for d in 1:N]..., + :inbounds, + ( + :if, :(b == 0), + ( + :simd_for, :(i in inds), + ( + common..., + [:( y[$d,i] = a*$(D[d]) ) for d in 1:N]... + ) + ), + :elseif, :(b == 1), + ( + :simd_for, :(i in inds), + ( + common..., + [:( y[$d,i] += a*$(D[d]) ) for d in 1:N]... + ) + ), + :else, + ( + :simd_for, :(i in inds), + ( + common..., + [:( y[$d,i] = a*$(D[d]) + b*y[$d,i] ) for d in 1:N]... + ) + ) + ) + ) +end + +@generated function _apply_Dt!(a::T, x::AbstractArray{<:Real,Np1}, + b::T, y::AbstractArray{<:Real,N} + ) where {N,Np1,T<:Real} + # We know that a ≠ 0 and that y has been pre-multiplied by b. + @assert Np1 == N + 1 + D = generate_symbols("d", N) + I = generate_symbols("i", N) + S = generate_symbols("s", N) + common = [:( $(I[d]) = min(i + $(S[d]), imax) ) for d in 1:N] + return encode( + :( inds = fastrange(size(y)) ), + :( imax = last(inds) ), + [:( $(S[d]) = $(offset(CartesianIndex{N}, d)) ) for d in 1:N]..., + :inbounds, + ( + :if, :(a == 1), + ( + :simd_for, :(i in inds), + ( + common..., + [:( $(D[d]) = x[$d,i] ) for d in 1:N]..., + :( y[i] -= $(encode_sum_of_terms(D)) ), + [:( y[$(I[d])] += $(D[d]) ) for d in 1:N]... + ), + ), + :else, + ( + :simd_for, :(i in inds), + ( + common..., + [:( $(D[d]) = a*x[$d,i] ) for d in 1:N]..., + :( y[i] -= $(encode_sum_of_terms(D)) ), + [:( y[$(I[d])] += $(D[d]) ) for d in 1:N]... + ), + ) + ) + ) +end + +@generated function _apply_DtD!(a::T, x::AbstractArray{<:Real,N}, + b::T, y::AbstractArray{<:Real,N} + ) where {N,T<:Real} + # We know that a ≠ 0 and that y has been pre-multiplied by b. + D = generate_symbols("d", N) + I = generate_symbols("i", N) + S = generate_symbols("s", N) + common = ( + [:( $(I[d]) = min(i + $(S[d]), imax) ) for d in 1:N]..., + :( xi = x[i] )) + return encode( + :( inds = fastrange(size(x)) ), + :( imax = last(inds) ), + [:( $(S[d]) = $(offset(CartesianIndex{N}, d)) ) for d in 1:N]..., + :inbounds, + ( + :if, :(a == 1), + ( + :simd_for, :(i in inds), + ( + common..., + [:( $(D[d]) = x[$(I[d])] - xi ) for d in 1:N]..., + :( y[i] -= $(encode_sum_of_terms(D)) ), + [:( y[$(I[d])] += $(D[d]) ) for d in 1:N]... + ), + ), + :else, + ( + :simd_for, :(i in inds), + ( + common..., + [:( $(D[d]) = a*(x[$(I[d])] - xi) ) for d in 1:N]..., + :( y[i] -= $(encode_sum_of_terms(D)) ), + [:( y[$(I[d])] += $(D[d]) ) for d in 1:N]... + ), + ) + ) + ) +end + + + +end # module diff --git a/test/mappings.jl b/test/mappings.jl index f02b9a0..2dc70e5 100644 --- a/test/mappings.jl +++ b/test/mappings.jl @@ -3,6 +3,13 @@ # # Tests for basic mappings. # +#------------------------------------------------------------------------------- +# +# This file is part of LazyAlgebra (https://github.com/emmt/LazyAlgebra.jl) +# released under the MIT "Expat" license. +# +# Copyright (c) 2017-2018 Éric Thiébaut. +# #isdefined(:LazyAlgebra) || include("../src/LazyAlgebra.jl") @@ -37,283 +44,320 @@ end floats = (Float32, Float64) complexes = (ComplexF32, ComplexF64) - @testset "Identity" begin - @test I === LazyAlgebra.I - @test I' === I - @test inv(I) === I - @test I*I === I - @test I\I === I - @test I/I === I - @test I+I === 2I - @test I+2I === 3I - @test I+I === 2I - @test 2I+3I === 5I - @test 3I + (I + 2I) === 6I - @test inv(3I) === (1/3)*I - @test I - I === 0I - @test I + I - 2I === 0I - @test 2I - (I + I) === 0I - @test SelfAdjointType(I) <: SelfAdjoint - @test MorphismType(I) <: Endomorphism - @test DiagonalType(I) <: DiagonalMapping - for P in operations - @test InPlaceType(P, I) <: InPlace - end - for T in floats - atol, rtol = zero(T), sqrt(eps(T)) - x = randn(T, dims) - y = randn(T, dims) - @test pointer(I*x) == pointer(x) - for P in operations - @test pointer(apply(P,I,x)) == pointer(I*x) - z = vcreate(P, I, x) - for α in alphas, β in betas - vcopy!(z, y) - @test apply!(α, P, I, x, β, z) ≈ α*x + β*y atol=atol rtol=rtol norm=vnorm2 - end - end - end - end - - @testset "UniformScaling" begin - # Check + operator. - @test I + UniformScaling(1) === 2I - @test I + UniformScaling(2) === 3I - @test UniformScaling(1) + I === 2I - @test UniformScaling(2) + I === 3I - # Check - operator. - @test I - UniformScaling(1) === 0I - @test I - UniformScaling(2) === -I - @test UniformScaling(1) - I === 0I - @test UniformScaling(2) - I === I - # Check * operator. - @test I*UniformScaling(1) === I - @test I*UniformScaling(2) === 2I - @test UniformScaling(1)*I === I - @test UniformScaling(2)*I === 2I - # Check \circ operator. - @test I∘UniformScaling(1) === I - @test I∘UniformScaling(2) === 2I - @test UniformScaling(1)∘I === I - @test UniformScaling(2)∘I === 2I - # \cdot is specific. - @test I⋅UniformScaling(1) === I - @test I⋅UniformScaling(2) === 2I - @test UniformScaling(1)⋅I === I - @test UniformScaling(2)⋅I === 2I - # Check / operator. - @test I/UniformScaling(1) === I - @test I/UniformScaling(2) === (1/2)*I - @test UniformScaling(1)/I === I - @test UniformScaling(2)/I === 2I - # Check \ operator. - @test I\UniformScaling(1) === I - @test I\UniformScaling(2) === 2I - @test UniformScaling(1)\I === I - @test UniformScaling(2)\I === (1/2)*I - end - - @testset "Rank 1 operators ($T)" for T in floats - w = randn(T, dims) - x = randn(T, dims) - y = randn(T, dims) - A = RankOneOperator(w, w) - B = RankOneOperator(w, y) - C = SymmetricRankOneOperator(w) - atol, rtol = zero(T), sqrt(eps(T)) - @test LinearType(A) <: Linear - @test LinearType(C) <: Linear - @test MorphismType(C) <: Endomorphism - for P in operations - @test InPlaceType(P, C) <: InPlace - end - @test A*I === A - @test I*A === A - @test A*x ≈ sum(w.*x)*w atol=atol rtol=rtol norm=vnorm2 - @test A'*x ≈ sum(w.*x)*w atol=atol rtol=rtol norm=vnorm2 - @test B*x ≈ sum(y.*x)*w atol=atol rtol=rtol norm=vnorm2 - @test B'*x ≈ sum(w.*x)*y atol=atol rtol=rtol norm=vnorm2 - @test C*x ≈ sum(w.*x)*w atol=atol rtol=rtol norm=vnorm2 - @test C'*x ≈ sum(w.*x)*w atol=atol rtol=rtol norm=vnorm2 - for α in alphas, - β in betas - for P in (Direct, Adjoint) - @test apply!(α, P, C, x, β, vcopy(y)) ≈ - T(α*vdot(w,x))*w + T(β)*y atol=atol rtol=rtol norm=vnorm2 - end - end - end - - @testset "Uniform scaling ($T)" for T in floats - x = randn(T, dims) - y = randn(T, dims) - γ = sqrt(2) - U = γ*I - atol, rtol = zero(T), sqrt(eps(T)) - @test U*x ≈ γ*x atol=atol rtol=rtol norm=vnorm2 - @test U'*x ≈ γ*x atol=atol rtol=rtol norm=vnorm2 - @test U\x ≈ (1/γ)*x atol=atol rtol=rtol norm=vnorm2 - @test U'\x ≈ (1/γ)*x atol=atol rtol=rtol norm=vnorm2 - for α in alphas, - β in betas - for P in (Direct, Adjoint) - @test apply!(α, P, U, x, β, vcopy(y)) ≈ - T(α*γ)*x + T(β)*y atol=atol rtol=rtol norm=vnorm2 - end - for P in (Inverse, InverseAdjoint) - @test apply!(α, P, U, x, β, vcopy(y)) ≈ - T(α/γ)*x + T(β)*y atol=atol rtol=rtol norm=vnorm2 - end - end - end + #@testset "Identity" begin + # @test I === LazyAlgebra.I + # @test I' === I + # @test inv(I) === I + # @test I*I === I + # @test I\I === I + # @test I/I === I + # @test I+I === 2I + # @test I+2I === 3I + # @test I+I === 2I + # @test 2I+3I === 5I + # @test 3I + (I + 2I) === 6I + # @test inv(3I) === (1/3)*I + # @test I - I === 0I + # @test I + I - 2I === 0I + # @test 2I - (I + I) === 0I + # @test SelfAdjointType(I) <: SelfAdjoint + # @test MorphismType(I) <: Endomorphism + # @test DiagonalType(I) <: DiagonalMapping + # for P in operations + # @test InPlaceType(P, I) <: InPlace + # end + # for T in floats + # atol, rtol = zero(T), sqrt(eps(T)) + # x = randn(T, dims) + # y = randn(T, dims) + # @test pointer(I*x) == pointer(x) + # for P in operations + # @test pointer(apply(P,I,x)) == pointer(I*x) + # z = vcreate(P, I, x) + # for α in alphas, β in betas + # vcopy!(z, y) + # @test apply!(α, P, I, x, β, z) ≈ α*x + β*y atol=atol rtol=rtol norm=vnorm2 + # end + # end + # end + #end + # + #@testset "UniformScaling" begin + # # Check + operator. + # @test I + UniformScaling(1) === 2I + # @test I + UniformScaling(2) === 3I + # @test UniformScaling(1) + I === 2I + # @test UniformScaling(2) + I === 3I + # # Check - operator. + # @test I - UniformScaling(1) === 0I + # @test I - UniformScaling(2) === -I + # @test UniformScaling(1) - I === 0I + # @test UniformScaling(2) - I === I + # # Check * operator. + # @test I*UniformScaling(1) === I + # @test I*UniformScaling(2) === 2I + # @test UniformScaling(1)*I === I + # @test UniformScaling(2)*I === 2I + # # Check \circ operator. + # @test I∘UniformScaling(1) === I + # @test I∘UniformScaling(2) === 2I + # @test UniformScaling(1)∘I === I + # @test UniformScaling(2)∘I === 2I + # # \cdot is specific. + # @test I⋅UniformScaling(1) === I + # @test I⋅UniformScaling(2) === 2I + # @test UniformScaling(1)⋅I === I + # @test UniformScaling(2)⋅I === 2I + # # Check / operator. + # @test I/UniformScaling(1) === I + # @test I/UniformScaling(2) === (1/2)*I + # @test UniformScaling(1)/I === I + # @test UniformScaling(2)/I === 2I + # # Check \ operator. + # @test I\UniformScaling(1) === I + # @test I\UniformScaling(2) === 2I + # @test UniformScaling(1)\I === I + # @test UniformScaling(2)\I === (1/2)*I + #end + # + #@testset "Rank 1 operators ($T)" for T in floats + # w = randn(T, dims) + # x = randn(T, dims) + # y = randn(T, dims) + # A = RankOneOperator(w, w) + # B = RankOneOperator(w, y) + # C = SymmetricRankOneOperator(w) + # atol, rtol = zero(T), sqrt(eps(T)) + # @test LinearType(A) <: Linear + # @test LinearType(C) <: Linear + # @test MorphismType(C) <: Endomorphism + # for P in operations + # @test InPlaceType(P, C) <: InPlace + # end + # @test A*I === A + # @test I*A === A + # @test A*x ≈ sum(w.*x)*w atol=atol rtol=rtol norm=vnorm2 + # @test A'*x ≈ sum(w.*x)*w atol=atol rtol=rtol norm=vnorm2 + # @test B*x ≈ sum(y.*x)*w atol=atol rtol=rtol norm=vnorm2 + # @test B'*x ≈ sum(w.*x)*y atol=atol rtol=rtol norm=vnorm2 + # @test C*x ≈ sum(w.*x)*w atol=atol rtol=rtol norm=vnorm2 + # @test C'*x ≈ sum(w.*x)*w atol=atol rtol=rtol norm=vnorm2 + # for α in alphas, + # β in betas + # for P in (Direct, Adjoint) + # @test apply!(α, P, C, x, β, vcopy(y)) ≈ + # T(α*vdot(w,x))*w + T(β)*y atol=atol rtol=rtol norm=vnorm2 + # end + # end + #end + # + #@testset "Uniform scaling ($T)" for T in floats + # x = randn(T, dims) + # y = randn(T, dims) + # γ = sqrt(2) + # U = γ*I + # atol, rtol = zero(T), sqrt(eps(T)) + # @test U*x ≈ γ*x atol=atol rtol=rtol norm=vnorm2 + # @test U'*x ≈ γ*x atol=atol rtol=rtol norm=vnorm2 + # @test U\x ≈ (1/γ)*x atol=atol rtol=rtol norm=vnorm2 + # @test U'\x ≈ (1/γ)*x atol=atol rtol=rtol norm=vnorm2 + # for α in alphas, + # β in betas + # for P in (Direct, Adjoint) + # @test apply!(α, P, U, x, β, vcopy(y)) ≈ + # T(α*γ)*x + T(β)*y atol=atol rtol=rtol norm=vnorm2 + # end + # for P in (Inverse, InverseAdjoint) + # @test apply!(α, P, U, x, β, vcopy(y)) ≈ + # T(α/γ)*x + T(β)*y atol=atol rtol=rtol norm=vnorm2 + # end + # end + #end + # + #@testset "Non-uniform scaling ($T)" for T in floats + # w = randn(T, dims) + # for i in eachindex(w) + # while w[i] == 0 + # w[i] = randn(T) + # end + # end + # x = randn(T, dims) + # y = randn(T, dims) + # z = vcreate(y) + # S = NonuniformScalingOperator(w) + # atol, rtol = zero(T), sqrt(eps(T)) + # @test S*x ≈ w.*x atol=atol rtol=rtol norm=vnorm2 + # @test S'*x ≈ w.*x atol=atol rtol=rtol norm=vnorm2 + # @test S\x ≈ x./w atol=atol rtol=rtol norm=vnorm2 + # @test S'\x ≈ x./w atol=atol rtol=rtol norm=vnorm2 + # for α in alphas, + # β in betas + # for P in (Direct, Adjoint) + # @test apply!(α, P, S, x, β, vcopy(y)) ≈ + # T(α)*w.*x + T(β)*y atol=atol rtol=rtol norm=vnorm2 + # end + # for P in (Inverse, InverseAdjoint) + # @test apply!(α, P, S, x, β, vcopy(y)) ≈ + # T(α)*x./w + T(β)*y atol=atol rtol=rtol norm=vnorm2 + # end + # end + #end + # + #@testset "Non-uniform scaling (Complex{$T})" for T in floats + # w = complex.(randn(T, dims), randn(T, dims)) + # for i in eachindex(w) + # while w[i] == 0 + # w[i] = complex(randn(T), randn(T)) + # end + # end + # x = complex.(randn(T, dims), randn(T, dims)) + # y = complex.(randn(T, dims), randn(T, dims)) + # wx = w.*x + # qx = x./w + # z = vcreate(y) + # S = NonuniformScalingOperator(w) + # atol, rtol = zero(T), sqrt(eps(T)) + # @test S*x ≈ w.*x atol=atol rtol=rtol norm=vnorm2 + # @test S'*x ≈ conj.(w).*x atol=atol rtol=rtol norm=vnorm2 + # @test S\x ≈ x./w atol=atol rtol=rtol norm=vnorm2 + # @test S'\x ≈ x./conj.(w) atol=atol rtol=rtol norm=vnorm2 + # for α in alphas, + # β in betas + # @test apply!(α, Direct, S, x, β, vcopy(y)) ≈ + # T(α)*w.*x + T(β)*y atol=atol rtol=rtol norm=vnorm2 + # @test apply!(α, Adjoint, S, x, β, vcopy(y)) ≈ + # T(α)*conj.(w).*x + T(β)*y atol=atol rtol=rtol norm=vnorm2 + # @test apply!(α, Inverse, S, x, β, vcopy(y)) ≈ + # T(α)*x./w + T(β)*y atol=atol rtol=rtol norm=vnorm2 + # @test apply!(α, InverseAdjoint, S, x, β, vcopy(y)) ≈ + # T(α)*x./conj.(w) + T(β)*y atol=atol rtol=rtol norm=vnorm2 + # end + #end + # + #rows, cols = (2,3,4), (5,6) + #nrows, ncols = prod(rows), prod(cols) + #@testset "Generalized matrices ($T)" for T in floats + # A = randn(T, rows..., cols...) + # x = randn(T, cols) + # y = randn(T, rows) + # G = GeneralMatrix(A) + # atol, rtol = zero(T), sqrt(eps(T)) + # mA = reshape(A, nrows, ncols) + # vx = reshape(x, ncols) + # vy = reshape(y, nrows) + # Gx = G*x + # Gty = G'*y + # @test Gx ≈ reshape(mA*vx, rows) atol=atol rtol=rtol norm=vnorm2 + # @test Gty ≈ reshape(mA'*vy, cols) atol=atol rtol=rtol norm=vnorm2 + # for α in alphas, + # β in betas + # @test apply!(α, Direct, G, x, β, vcopy(y)) ≈ + # T(α)*Gx + T(β)*y atol=atol rtol=rtol norm=vnorm2 + # @test apply!(α, Adjoint, G, y, β, vcopy(x)) ≈ + # T(α)*Gty + T(β)*x atol=atol rtol=rtol norm=vnorm2 + # end + #end + # + #@static if VERSION ≥ v"0.7" + # rows, cols = (2,3,4), (5,6) + # nrows, ncols = prod(rows), prod(cols) + # @testset "Sparse matrices ($T)" for T in floats + # A = randn(T, rows..., cols...) + # A[rand(T, size(A)) .≤ 0.7] .= 0 # 70% of zeros + # x = randn(T, cols) + # y = randn(T, rows) + # G = GeneralMatrix(A) + # S = SparseOperator(A, length(rows)) + # atol, rtol = zero(T), sqrt(eps(T)) + # mA = reshape(A, nrows, ncols) + # vx = reshape(x, ncols) + # vy = reshape(y, nrows) + # Sx = S*x + # Sty = S'*y + # @test Sx ≈ G*x atol=atol rtol=rtol norm=vnorm2 + # @test Sty ≈ G'*y atol=atol rtol=rtol norm=vnorm2 + # @test Sx ≈ reshape(mA*vx, rows) atol=atol rtol=rtol norm=vnorm2 + # @test Sty ≈ reshape(mA'*vy, cols) atol=atol rtol=rtol norm=vnorm2 + # for α in alphas, + # β in betas + # @test apply!(α, Direct, S, x, β, vcopy(y)) ≈ + # T(α)*Sx + T(β)*y atol=atol rtol=rtol norm=vnorm2 + # @test apply!(α, Adjoint, S, y, β, vcopy(x)) ≈ + # T(α)*Sty + T(β)*x atol=atol rtol=rtol norm=vnorm2 + # end + # + # end + #else + # @warn "Sparse matrices tests broken for Julia ≤ 0.6" + #end - @testset "Non-uniform scaling ($T)" for T in floats - w = randn(T, dims) - for i in eachindex(w) - while w[i] == 0 - w[i] = randn(T) - end - end - x = randn(T, dims) - y = randn(T, dims) - z = vcreate(y) - S = NonuniformScalingOperator(w) - atol, rtol = zero(T), sqrt(eps(T)) - @test S*x ≈ w.*x atol=atol rtol=rtol norm=vnorm2 - @test S'*x ≈ w.*x atol=atol rtol=rtol norm=vnorm2 - @test S\x ≈ x./w atol=atol rtol=rtol norm=vnorm2 - @test S'\x ≈ x./w atol=atol rtol=rtol norm=vnorm2 - for α in alphas, - β in betas - for P in (Direct, Adjoint) - @test apply!(α, P, S, x, β, vcopy(y)) ≈ - T(α)*w.*x + T(β)*y atol=atol rtol=rtol norm=vnorm2 - end - for P in (Inverse, InverseAdjoint) - @test apply!(α, P, S, x, β, vcopy(y)) ≈ - T(α)*x./w + T(β)*y atol=atol rtol=rtol norm=vnorm2 - end - end - end - - @testset "Non-uniform scaling (Complex{$T})" for T in floats - w = complex.(randn(T, dims), randn(T, dims)) - for i in eachindex(w) - while w[i] == 0 - w[i] = complex(randn(T), randn(T)) + sizes = ((50,), (8, 9), (4,5,6)) + @testset "Finite differences ($T)" for T in floats + D = SimpleFiniteDifferences() + for dims in sizes + x = randn(T, dims) + y = randn(T, ndims(x), size(x)...) + y0 = Array{T}(undef, size(y)) + y1 = Array{T}(undef, size(y)) + if ndims(x) == 1 + y0[1,1:end-1] = x[2:end] - x[1:end-1] + y0[1,end] = 0 + elseif ndims(x) == 2 + y0[1,1:end-1,:] = x[2:end,:] - x[1:end-1,:] + y0[1,end,:] .= 0 + y0[2,:,1:end-1] = x[:,2:end] - x[:,1:end-1] + y0[2,:,end] .= 0 + elseif ndims(x) == 3 + y0[1,1:end-1,:,:] = x[2:end,:,:] - x[1:end-1,:,:] + y0[1,end,:,:] .= 0 + y0[2,:,1:end-1,:] = x[:,2:end,:] - x[:,1:end-1,:] + y0[2,:,end,:] .= 0 + y0[3,:,:,1:end-1] = x[:,:,2:end] - x[:,:,1:end-1] + y0[3,:,:,end] .= 0 end - end - x = complex.(randn(T, dims), randn(T, dims)) - y = complex.(randn(T, dims), randn(T, dims)) - wx = w.*x - qx = x./w - z = vcreate(y) - S = NonuniformScalingOperator(w) - atol, rtol = zero(T), sqrt(eps(T)) - @test S*x ≈ w.*x atol=atol rtol=rtol norm=vnorm2 - @test S'*x ≈ conj.(w).*x atol=atol rtol=rtol norm=vnorm2 - @test S\x ≈ x./w atol=atol rtol=rtol norm=vnorm2 - @test S'\x ≈ x./conj.(w) atol=atol rtol=rtol norm=vnorm2 - for α in alphas, - β in betas - @test apply!(α, Direct, S, x, β, vcopy(y)) ≈ - T(α)*w.*x + T(β)*y atol=atol rtol=rtol norm=vnorm2 - @test apply!(α, Adjoint, S, x, β, vcopy(y)) ≈ - T(α)*conj.(w).*x + T(β)*y atol=atol rtol=rtol norm=vnorm2 - @test apply!(α, Inverse, S, x, β, vcopy(y)) ≈ - T(α)*x./w + T(β)*y atol=atol rtol=rtol norm=vnorm2 - @test apply!(α, InverseAdjoint, S, x, β, vcopy(y)) ≈ - T(α)*x./conj.(w) + T(β)*y atol=atol rtol=rtol norm=vnorm2 - end - end - - rows, cols = (2,3,4), (5,6) - nrows, ncols = prod(rows), prod(cols) - @testset "Generalized matrices ($T)" for T in floats - A = randn(T, rows..., cols...) - x = randn(T, cols) - y = randn(T, rows) - G = GeneralMatrix(A) - atol, rtol = zero(T), sqrt(eps(T)) - mA = reshape(A, nrows, ncols) - vx = reshape(x, ncols) - vy = reshape(y, nrows) - Gx = G*x - Gty = G'*y - @test Gx ≈ reshape(mA*vx, rows) atol=atol rtol=rtol norm=vnorm2 - @test Gty ≈ reshape(mA'*vy, cols) atol=atol rtol=rtol norm=vnorm2 - for α in alphas, - β in betas - @test apply!(α, Direct, G, x, β, vcopy(y)) ≈ - T(α)*Gx + T(β)*y atol=atol rtol=rtol norm=vnorm2 - @test apply!(α, Adjoint, G, y, β, vcopy(x)) ≈ - T(α)*Gty + T(β)*x atol=atol rtol=rtol norm=vnorm2 - end - end - - @static if VERSION ≥ v"0.7" - rows, cols = (2,3,4), (5,6) - nrows, ncols = prod(rows), prod(cols) - @testset "Sparse matrices ($T)" for T in floats - A = randn(T, rows..., cols...) - A[rand(T, size(A)) .≤ 0.7] .= 0 # 70% of zeros - x = randn(T, cols) - y = randn(T, rows) - G = GeneralMatrix(A) - S = SparseOperator(A, length(rows)) - atol, rtol = zero(T), sqrt(eps(T)) - mA = reshape(A, nrows, ncols) - vx = reshape(x, ncols) - vy = reshape(y, nrows) - Sx = S*x - Sty = S'*y - @test Sx ≈ G*x atol=atol rtol=rtol norm=vnorm2 - @test Sty ≈ G'*y atol=atol rtol=rtol norm=vnorm2 - @test Sx ≈ reshape(mA*vx, rows) atol=atol rtol=rtol norm=vnorm2 - @test Sty ≈ reshape(mA'*vy, cols) atol=atol rtol=rtol norm=vnorm2 + atol, rtol = zero(T), eps(T)*2 + Dx = D*x + # Here there should be no differences. + @test Dx ≈ y0 atol=atol rtol=rtol norm=vnorm2 + @test vnorm2(Dx - y0) == 0 for α in alphas, β in betas - @test apply!(α, Direct, S, x, β, vcopy(y)) ≈ - T(α)*Sx + T(β)*y atol=atol rtol=rtol norm=vnorm2 - @test apply!(α, Adjoint, S, y, β, vcopy(x)) ≈ - T(α)*Sty + T(β)*x atol=atol rtol=rtol norm=vnorm2 + @test apply!(α, Direct, D, x, β, vcopy(y)) ≈ + T(α)*Dx + T(β)*y atol=atol rtol=rtol norm=vnorm2 end - end - else - @warn "Sparse matrices tests broken for Julia ≤ 0.6" end - @testset "FFT ($T)" for T in floats - for dims in ((45,), (20,), (33,12), (30,20), (4,5,6)) - for cmplx in (false, true) - if cmplx - x = randn(T, dims) + 1im*randn(T, dims) - else - x = randn(T, dims) - end - F = FFTOperator(x) - if cmplx - y = randn(T, dims) + 1im*randn(T, dims) - else - y = randn(T, output_size(F)) + 1im*randn(T, output_size(F)) - end - ϵ = eps(T) - atol, rtol = zero(T), eps(T) - z = (cmplx ? fft(x) : rfft(x)) - w = (cmplx ? ifft(y) : irfft(y, dims[1])) - @test F*x ≈ z atol=0 rtol=ϵ norm=vnorm2 - @test F\y ≈ w atol=0 rtol=ϵ norm=vnorm2 - for α in alphas, - β in betas - @test apply!(α, Direct, F, x, β, vcopy(y)) ≈ - T(α)*z + T(β)*y atol=0 rtol=ϵ - @test apply!(α, Inverse, F, y, β, vcopy(x)) ≈ - T(α)*w + T(β)*x atol=0 rtol=ϵ - end - end - end - end + #@testset "FFT ($T)" for T in floats + # for dims in ((45,), (20,), (33,12), (30,20), (4,5,6)) + # for cmplx in (false, true) + # if cmplx + # x = randn(T, dims) + 1im*randn(T, dims) + # else + # x = randn(T, dims) + # end + # F = FFTOperator(x) + # if cmplx + # y = randn(T, dims) + 1im*randn(T, dims) + # else + # y = randn(T, output_size(F)) + 1im*randn(T, output_size(F)) + # end + # ϵ = eps(T) + # atol, rtol = zero(T), eps(T) + # z = (cmplx ? fft(x) : rfft(x)) + # w = (cmplx ? ifft(y) : irfft(y, dims[1])) + # @test F*x ≈ z atol=0 rtol=ϵ norm=vnorm2 + # @test F\y ≈ w atol=0 rtol=ϵ norm=vnorm2 + # for α in alphas, + # β in betas + # @test apply!(α, Direct, F, x, β, vcopy(y)) ≈ + # T(α)*z + T(β)*y atol=0 rtol=ϵ + # @test apply!(α, Inverse, F, y, β, vcopy(x)) ≈ + # T(α)*w + T(β)*x atol=0 rtol=ϵ + # end + # end + # end + #end end end