From 23eedafcf6d933addf7411d866f9660596a61f6a Mon Sep 17 00:00:00 2001 From: Jameson Nash Date: Wed, 2 Jun 2021 16:13:46 -0400 Subject: [PATCH] cat: remove incorrect inbounds/inline annotations There are no boundschecks so these lead to problems such as #41047. --- base/abstractarray.jl | 51 ++++++++++++----------- stdlib/LinearAlgebra/src/LinearAlgebra.jl | 2 +- test/abstractarray.jl | 2 + 3 files changed, 29 insertions(+), 26 deletions(-) diff --git a/base/abstractarray.jl b/base/abstractarray.jl index 92db812b249ad3..6cb1c8782f9975 100644 --- a/base/abstractarray.jl +++ b/base/abstractarray.jl @@ -1529,10 +1529,10 @@ vcat(X::T...) where {T<:Number} = T[ X[i] for i=1:length(X) ] hcat(X::T...) where {T} = T[ X[j] for i=1:1, j=1:length(X) ] hcat(X::T...) where {T<:Number} = T[ X[j] for i=1:1, j=1:length(X) ] -vcat(X::Number...) = hvcat_fill(Vector{promote_typeof(X...)}(undef, length(X)), X) -hcat(X::Number...) = hvcat_fill(Matrix{promote_typeof(X...)}(undef, 1,length(X)), X) -typed_vcat(::Type{T}, X::Number...) where {T} = hvcat_fill(Vector{T}(undef, length(X)), X) -typed_hcat(::Type{T}, X::Number...) where {T} = hvcat_fill(Matrix{T}(undef, 1,length(X)), X) +vcat(X::Number...) = hvcat_fill!(Vector{promote_typeof(X...)}(undef, length(X)), X) +hcat(X::Number...) = hvcat_fill!(Matrix{promote_typeof(X...)}(undef, 1,length(X)), X) +typed_vcat(::Type{T}, X::Number...) where {T} = hvcat_fill!(Vector{T}(undef, length(X)), X) +typed_hcat(::Type{T}, X::Number...) where {T} = hvcat_fill!(Matrix{T}(undef, 1,length(X)), X) vcat(V::AbstractVector...) = typed_vcat(promote_eltype(V...), V...) vcat(V::AbstractVector{T}...) where {T} = typed_vcat(T, V...) @@ -1993,9 +1993,10 @@ function hvcat(rows::Tuple{Vararg{Int}}, xs::T...) where T<:Number a end -function hvcat_fill(a::Array, xs::Tuple) +function hvcat_fill!(a::Array, xs::Tuple) k = 1 nr, nc = size(a,1), size(a,2) + @boundscheck checkbounds(a, 1:nr*nc) for i=1:nr @inbounds for j=1:nc a[i,j] = xs[k] @@ -2020,7 +2021,7 @@ function typed_hvcat(::Type{T}, rows::Tuple{Vararg{Int}}, xs::Number...) where T if nr*nc != len throw(ArgumentError("argument count $(len) does not match specified shape $((nr,nc))")) end - hvcat_fill(Matrix{T}(undef, nr, nc), xs) + hvcat_fill!(Matrix{T}(undef, nr, nc), xs) end function typed_hvcat(::Type{T}, rows::Tuple{Vararg{Int}}, as...) where T @@ -2136,9 +2137,9 @@ function hvncat_fill!(A::Array, row_first::Bool, xs::Tuple) if row_first nr, nc = size(A, 1), size(A, 2) nrc = nr * nc - @inbounds na = prod(size(A)[3:end]) + na = prod(size(A)[3:end]) k = 1 - @inbounds for d ∈ 1:na + for d ∈ 1:na dd = nrc * (d - 1) for i ∈ 1:nr Ai = dd + i @@ -2150,7 +2151,7 @@ function hvncat_fill!(A::Array, row_first::Bool, xs::Tuple) end end else - @inbounds for k ∈ eachindex(xs) + for k ∈ eachindex(xs) A[k] = xs[k] end end @@ -2162,7 +2163,7 @@ _typed_hvncat(T::Type, ::Val{N}, xs::Number...) where N = _typed_hvncat(T, (ntup function _typed_hvncat(::Type{T}, ::Val{N}, as::AbstractArray...) where {T, N} # optimization for arrays that can be concatenated by copying them linearly into the destination # conditions: the elements must all have 1- or 0-length dimensions above N - @inbounds for a ∈ as + for a ∈ as ndims(a) <= N || all(x -> size(a, x) == 1, (N + 1):ndims(a)) || return _typed_hvncat(T, (ntuple(x -> 1, N - 1)..., length(as)), false, as...) end @@ -2170,16 +2171,16 @@ function _typed_hvncat(::Type{T}, ::Val{N}, as::AbstractArray...) where {T, N} nd = max(N, ndims(as[1])) Ndim = 0 - @inbounds for i ∈ 1:lastindex(as) + for i ∈ 1:lastindex(as) Ndim += cat_size(as[i], N) for d ∈ 1:N - 1 cat_size(as[1], d) == cat_size(as[i], d) || throw(ArgumentError("mismatched size along axis $d in element $i")) end end - @inbounds A = Array{T, nd}(undef, ntuple(d -> cat_size(as[1], d), N - 1)..., Ndim, ntuple(x -> 1, nd - N)...) + A = Array{T, nd}(undef, ntuple(d -> cat_size(as[1], d), N - 1)..., Ndim, ntuple(x -> 1, nd - N)...) k = 1 - @inbounds for a ∈ as + for a ∈ as for i ∈ eachindex(a) A[k] = a[i] k += 1 @@ -2196,7 +2197,7 @@ function _typed_hvncat(::Type{T}, ::Val{N}, as...) where {T, N} # into the destination nd = N Ndim = 0 - @inbounds for a ∈ as + for a ∈ as if a isa AbstractArray cat_size(a, N) == length(a) || throw(ArgumentError("all dimensions of elements other than $N must be of length 1")) @@ -2205,10 +2206,10 @@ function _typed_hvncat(::Type{T}, ::Val{N}, as...) where {T, N} Ndim += cat_size(a, N) end - @inbounds A = Array{T, nd}(undef, ntuple(x -> 1, N - 1)..., Ndim, ntuple(x -> 1, nd - N)...) + A = Array{T, nd}(undef, ntuple(x -> 1, N - 1)..., Ndim, ntuple(x -> 1, nd - N)...) k = 1 - @inbounds for a ∈ as + for a ∈ as if a isa AbstractArray lena = length(a) copyto!(A, k, a, 1, lena) @@ -2226,17 +2227,17 @@ function _typed_hvncat(::Type{T}, dims::Tuple{Vararg{Int, N}}, row_first::Bool, d2 = row_first ? 1 : 2 # discover dimensions - @inbounds nd = max(N, cat_ndims(as[1])) + nd = max(N, cat_ndims(as[1])) outdims = zeros(Int, nd) # discover number of rows or columns - @inbounds for i ∈ 1:dims[d1] + for i ∈ 1:dims[d1] outdims[d1] += cat_size(as[i], d1) end currentdims = zeros(Int, nd) blockcount = 0 - @inbounds for i ∈ eachindex(as) + for i ∈ eachindex(as) currentdims[d1] += cat_size(as[i], d1) if currentdims[d1] == outdims[d1] currentdims[d1] = 0 @@ -2287,18 +2288,18 @@ function _typed_hvncat(::Type{T}, shape::Tuple{Vararg{Tuple, N}}, row_first::Boo d1 = row_first ? 2 : 1 d2 = row_first ? 1 : 2 shape = collect(shape) # saves allocations later - @inbounds shapelength = shape[end][1] + shapelength = shape[end][1] lengthas = length(as) shapelength == lengthas || throw(ArgumentError("number of elements does not match shape; expected $(shapelength), got $lengthas)")) # discover dimensions - @inbounds nd = max(N, cat_ndims(as[1])) + nd = max(N, cat_ndims(as[1])) outdims = zeros(Int, nd) currentdims = zeros(Int, nd) blockcounts = zeros(Int, nd) shapepos = ones(Int, nd) - @inbounds for i ∈ eachindex(as) + for i ∈ eachindex(as) wasstartblock = false for d ∈ 1:N ad = (d < 3 && row_first) ? (d == 1 ? 2 : 1) : d @@ -2328,7 +2329,7 @@ function _typed_hvncat(::Type{T}, shape::Tuple{Vararg{Tuple, N}}, row_first::Boo end if row_first - @inbounds outdims[1], outdims[2] = outdims[2], outdims[1] + outdims[1], outdims[2] = outdims[2], outdims[1] end # @assert all(==(0), currentdims) @@ -2340,11 +2341,11 @@ function _typed_hvncat(::Type{T}, shape::Tuple{Vararg{Tuple, N}}, row_first::Boo return A end -@inline function hvncat_fill!(A::Array{T, N}, scratch1::Vector{Int}, scratch2::Vector{Int}, d1::Int, d2::Int, as::Tuple{Vararg}) where {T, N} +function hvncat_fill!(A::Array{T, N}, scratch1::Vector{Int}, scratch2::Vector{Int}, d1::Int, d2::Int, as::Tuple{Vararg}) where {T, N} outdims = size(A) offsets = scratch1 inneroffsets = scratch2 - @inbounds for a ∈ as + for a ∈ as if isa(a, AbstractArray) for ai ∈ a Ai = hvncat_calcindex(offsets, inneroffsets, outdims, N) diff --git a/stdlib/LinearAlgebra/src/LinearAlgebra.jl b/stdlib/LinearAlgebra/src/LinearAlgebra.jl index 07bb9548073612..8dfed4b5c89b96 100644 --- a/stdlib/LinearAlgebra/src/LinearAlgebra.jl +++ b/stdlib/LinearAlgebra/src/LinearAlgebra.jl @@ -15,7 +15,7 @@ import Base: USE_BLAS64, abs, acos, acosh, acot, acoth, acsc, acsch, adjoint, as oneunit, parent, power_by_squaring, print_matrix, promote_rule, real, round, sec, sech, setindex!, show, similar, sin, sincos, sinh, size, sqrt, strides, stride, tan, tanh, transpose, trunc, typed_hcat, vec -using Base: hvcat_fill, IndexLinear, promote_op, promote_typeof, +using Base: IndexLinear, promote_op, promote_typeof, @propagate_inbounds, @pure, reduce, typed_vcat, require_one_based_indexing using Base.Broadcast: Broadcasted, broadcasted import Libdl diff --git a/test/abstractarray.jl b/test/abstractarray.jl index 05ab7147e1f7e3..65f9f4efa2cd3e 100644 --- a/test/abstractarray.jl +++ b/test/abstractarray.jl @@ -1388,6 +1388,8 @@ end for v = (1, "test") @test [v v;;; fill(v, 1, 2)] == fill(v, 1, 2, 2) end + + @test_throws BoundsError hvncat(((1, 2), (3,)), false, zeros(Int, 0, 0, 0), 7, 8) end @testset "keepat!" begin