Skip to content

Commit

Permalink
Fix sparse array setindex(::Int, ::Vector) (#43678)
Browse files Browse the repository at this point in the history
(cherry picked from commit dc61f29)
  • Loading branch information
dkarrasch authored and KristofferC committed Mar 15, 2022
1 parent 9efdba4 commit 74165cf
Show file tree
Hide file tree
Showing 2 changed files with 26 additions and 17 deletions.
28 changes: 11 additions & 17 deletions stdlib/SparseArrays/src/sparsematrix.jl
Original file line number Diff line number Diff line change
Expand Up @@ -2787,6 +2787,7 @@ end

# Nonscalar A[I,J] = B: Convert B to a SparseMatrixCSC of the appropriate shape first
_to_same_csc(::AbstractSparseMatrixCSC{Tv, Ti}, V::AbstractMatrix, I...) where {Tv,Ti} = convert(SparseMatrixCSC{Tv,Ti}, V)
_to_same_csc(::AbstractSparseMatrixCSC{Tv, Ti}, V::AbstractMatrix, i::Integer, J) where {Tv,Ti} = convert(SparseMatrixCSC{Tv,Ti}, reshape(V, (1, length(J))))
_to_same_csc(::AbstractSparseMatrixCSC{Tv, Ti}, V::AbstractVector, I...) where {Tv,Ti} = convert(SparseMatrixCSC{Tv,Ti}, reshape(V, map(length, I)))

setindex!(A::AbstractSparseMatrixCSC{Tv}, B::AbstractVecOrMat, I::Integer, J::Integer) where {Tv} = _setindex_scalar!(A, B, I, J)
Expand All @@ -2795,12 +2796,20 @@ function setindex!(A::AbstractSparseMatrixCSC{Tv,Ti}, V::AbstractVecOrMat, Ix::U
require_one_based_indexing(A, V, Ix, Jx)
(I, J) = Base.ensure_indexable(to_indices(A, (Ix, Jx)))
checkbounds(A, I, J)
Base.setindex_shape_check(V, length(I), length(J))
nJ = length(J)
Base.setindex_shape_check(V, length(I), nJ)
B = _to_same_csc(A, V, I, J)

m, n = size(A)
if (!isempty(I) && (I[1] < 1 || I[end] > m)) || (!isempty(J) && (J[1] < 1 || J[end] > n))
throw(BoundsError(A, (I, J)))
end
if isempty(I) || isempty(J)
return A
end

issortedI = issorted(I)
issortedJ = issorted(J)

if !issortedI && !issortedJ
pI = sortperm(I); @inbounds I = I[pI]
pJ = sortperm(J); @inbounds J = J[pJ]
Expand All @@ -2813,20 +2822,6 @@ function setindex!(A::AbstractSparseMatrixCSC{Tv,Ti}, V::AbstractVecOrMat, Ix::U
B = B[:, pJ]
end

m, n = size(A)
mB, nB = size(B)

if (!isempty(I) && (I[1] < 1 || I[end] > m)) || (!isempty(J) && (J[1] < 1 || J[end] > n))
throw(BoundsError(A, (I, J)))
end

if isempty(I) || isempty(J)
return A
end

nI = length(I)
nJ = length(J)

colptrA = getcolptr(A); rowvalA = rowvals(A); nzvalA = nonzeros(A)
colptrB = getcolptr(B); rowvalB = rowvals(B); nzvalB = nonzeros(B)

Expand All @@ -2840,7 +2835,6 @@ function setindex!(A::AbstractSparseMatrixCSC{Tv,Ti}, V::AbstractVecOrMat, Ix::U
resize!(nzvalA, nnzS)

colB = 1
asgn_col = J[colB]

I_asgn = falses(m)
fill!(view(I_asgn, I), true)
Expand Down
15 changes: 15 additions & 0 deletions test/arrayops.jl
Original file line number Diff line number Diff line change
Expand Up @@ -2876,3 +2876,18 @@ end
@test [fill(1); fill(2, (2,1,1))] == reshape([1; 2; 2], (3, 1, 1))
@test_throws DimensionMismatch [fill(1); rand(2, 2, 2)]
end

@testset "Allow assignment of singleton array to sparse array #43644" begin
K = spzeros(3,3)
b = zeros(3,3)
b[3,:] = [1,2,3]
K[3,1:3] += [1.0 2.0 3.0]'
@test K == b
K[3:3,1:3] += zeros(1, 3)
@test K == b
K[3,1:3] += zeros(3)
@test K == b
K[3,:] += zeros(3,1)
@test K == b
@test_throws DimensionMismatch K[3,1:2] += [1.0 2.0 3.0]'
end

0 comments on commit 74165cf

Please sign in to comment.