Skip to content

Commit 0d29712

Browse files
authored
Merge pull request #44096 from JuliaLang/jn/43551
implement promote for BitArray
2 parents 5deb503 + 5cac5b9 commit 0d29712

File tree

6 files changed

+47
-20
lines changed

6 files changed

+47
-20
lines changed

base/indices.jl

+13-7
Original file line numberDiff line numberDiff line change
@@ -351,6 +351,8 @@ struct Slice{T<:AbstractUnitRange} <: AbstractUnitRange{Int}
351351
indices::T
352352
end
353353
Slice(S::Slice) = S
354+
Slice{T}(S::Slice) where {T<:AbstractUnitRange} = Slice{T}(T(S.indices))
355+
354356
axes(S::Slice) = (IdentityUnitRange(S.indices),)
355357
axes1(S::Slice) = IdentityUnitRange(S.indices)
356358
axes(S::Slice{<:OneTo}) = (S.indices,)
@@ -366,7 +368,6 @@ getindex(S::Slice, i::StepRange{<:Integer}) = (@inline; @boundscheck checkbounds
366368
show(io::IO, r::Slice) = print(io, "Base.Slice(", r.indices, ")")
367369
iterate(S::Slice, s...) = iterate(S.indices, s...)
368370

369-
370371
"""
371372
IdentityUnitRange(range::AbstractUnitRange)
372373
@@ -378,6 +379,8 @@ struct IdentityUnitRange{T<:AbstractUnitRange} <: AbstractUnitRange{Int}
378379
indices::T
379380
end
380381
IdentityUnitRange(S::IdentityUnitRange) = S
382+
IdentityUnitRange{T}(S::IdentityUnitRange) where {T<:AbstractUnitRange} = IdentityUnitRange{T}(T(S.indices))
383+
381384
# IdentityUnitRanges are offset and thus have offset axes, so they are their own axes
382385
axes(S::IdentityUnitRange) = (S,)
383386
axes1(S::IdentityUnitRange) = S
@@ -448,6 +451,8 @@ julia> linear[1,2]
448451
struct LinearIndices{N,R<:NTuple{N,AbstractUnitRange{Int}}} <: AbstractArray{Int,N}
449452
indices::R
450453
end
454+
convert(::Type{LinearIndices{N,R}}, inds::LinearIndices{N}) where {N,R<:NTuple{N,AbstractUnitRange{Int}}} =
455+
LinearIndices{N,R}(convert(R, inds.indices))
451456

452457
LinearIndices(::Tuple{}) = LinearIndices{0,typeof(())}(())
453458
LinearIndices(inds::NTuple{N,AbstractUnitRange{<:Integer}}) where {N} =
@@ -459,16 +464,17 @@ LinearIndices(A::Union{AbstractArray,SimpleVector}) = LinearIndices(axes(A))
459464
_convert2ind(i::Integer) = Base.OneTo(i)
460465
_convert2ind(ind::AbstractUnitRange) = first(ind):last(ind)
461466

462-
promote_rule(::Type{LinearIndices{N,R1}}, ::Type{LinearIndices{N,R2}}) where {N,R1,R2} =
463-
LinearIndices{N,indices_promote_type(R1,R2)}
464-
465467
function indices_promote_type(::Type{Tuple{R1,Vararg{R1,N}}}, ::Type{Tuple{R2,Vararg{R2,N}}}) where {R1,R2,N}
466468
R = promote_type(R1, R2)
467-
Tuple{R,Vararg{R,N}}
469+
return Tuple{R, Vararg{R, N}}
468470
end
469471

470-
convert(::Type{LinearIndices{N,R}}, inds::LinearIndices{N}) where {N,R} =
471-
LinearIndices(convert(R, inds.indices))
472+
promote_rule(::Type{LinearIndices{N,R1}}, ::Type{LinearIndices{N,R2}}) where {N,R1,R2} =
473+
LinearIndices{N,indices_promote_type(R1,R2)}
474+
promote_rule(a::Type{Slice{T1}}, b::Type{Slice{T2}}) where {T1,T2} =
475+
el_same(promote_type(T1, T2), a, b)
476+
promote_rule(a::Type{IdentityUnitRange{T1}}, b::Type{IdentityUnitRange{T2}}) where {T1,T2} =
477+
el_same(promote_type(T1, T2), a, b)
472478

473479
# AbstractArray implementation
474480
IndexStyle(::Type{<:LinearIndices}) = IndexLinear()

base/promotion.jl

+2-2
Original file line numberDiff line numberDiff line change
@@ -303,9 +303,9 @@ it for new types as appropriate.
303303
"""
304304
function promote_rule end
305305

306-
promote_rule(::Type{<:Any}, ::Type{<:Any}) = Bottom
306+
promote_rule(::Type, ::Type) = Bottom
307307

308-
promote_result(::Type{<:Any},::Type{<:Any},::Type{T},::Type{S}) where {T,S} = (@inline; promote_type(T,S))
308+
promote_result(::Type,::Type,::Type{T},::Type{S}) where {T,S} = (@inline; promote_type(T,S))
309309
# If no promote_rule is defined, both directions give Bottom. In that
310310
# case use typejoin on the original types instead.
311311
promote_result(::Type{T},::Type{S},::Type{Bottom},::Type{Bottom}) where {T,S} = (@inline; typejoin(T, S))

base/range.jl

+6-2
Original file line numberDiff line numberDiff line change
@@ -1263,13 +1263,17 @@ function -(r::LinRange)
12631263
LinRange{typeof(start)}(start, -r.stop, length(r))
12641264
end
12651265

1266-
12671266
# promote eltype if at least one container wouldn't change, otherwise join container types.
1268-
el_same(::Type{T}, a::Type{<:AbstractArray{T,n}}, b::Type{<:AbstractArray{T,n}}) where {T,n} = a
1267+
el_same(::Type{T}, a::Type{<:AbstractArray{T,n}}, b::Type{<:AbstractArray{T,n}}) where {T,n} = a # we assume a === b
12691268
el_same(::Type{T}, a::Type{<:AbstractArray{T,n}}, b::Type{<:AbstractArray{S,n}}) where {T,S,n} = a
12701269
el_same(::Type{T}, a::Type{<:AbstractArray{S,n}}, b::Type{<:AbstractArray{T,n}}) where {T,S,n} = b
12711270
el_same(::Type, a, b) = promote_typejoin(a, b)
12721271

1272+
promote_result(::Type{<:AbstractArray}, ::Type{<:AbstractArray}, ::Type{T}, ::Type{S}) where {T,S} = (@inline; promote_type(T,S))
1273+
promote_result(::Type{T}, ::Type{S}, ::Type{Bottom}, ::Type{Bottom}) where {T<:AbstractArray,S<:AbstractArray} = (@inline; promote_typejoin(T,S))
1274+
# If no promote_rule is defined, both directions give Bottom. In that case use typejoin on the eltypes instead and give Array as the container.
1275+
promote_result(::Type{<:AbstractArray{T,n}}, ::Type{<:AbstractArray{S,n}}, ::Type{Bottom}, ::Type{Bottom}) where {T,S,n} = (@inline; Array{promote_type(T,S),n})
1276+
12731277
promote_rule(a::Type{UnitRange{T1}}, b::Type{UnitRange{T2}}) where {T1,T2} =
12741278
el_same(promote_type(T1, T2), a, b)
12751279
UnitRange{T}(r::UnitRange{T}) where {T<:Real} = r

base/reducedim.jl

+2-2
Original file line numberDiff line numberDiff line change
@@ -44,7 +44,7 @@ function reduced_indices0(inds::Indices{N}, d::Int) where N
4444
end
4545

4646
function reduced_indices(inds::Indices{N}, region) where N
47-
rinds = [inds...]
47+
rinds = collect(inds)
4848
for i in region
4949
isa(i, Integer) || throw(ArgumentError("reduced dimension(s) must be integers"))
5050
d = Int(i)
@@ -58,7 +58,7 @@ function reduced_indices(inds::Indices{N}, region) where N
5858
end
5959

6060
function reduced_indices0(inds::Indices{N}, region) where N
61-
rinds = [inds...]
61+
rinds = collect(inds)
6262
for i in region
6363
isa(i, Integer) || throw(ArgumentError("reduced dimension(s) must be integers"))
6464
d = Int(i)

test/bitarray.jl

+15-1
Original file line numberDiff line numberDiff line change
@@ -98,6 +98,20 @@ end
9898

9999
timesofar("conversions")
100100

101+
@testset "Promotions for size $sz" for (sz, T) in allsizes
102+
@test isequal(promote(falses(sz...), zeros(sz...)),
103+
(zeros(sz...), zeros(sz...)))
104+
@test isequal(promote(trues(sz...), ones(sz...)),
105+
(ones(sz...), ones(sz...)))
106+
ae = falses(1, sz...)
107+
ex = (@test_throws ErrorException promote(ae, ones(sz...))).value
108+
@test startswith(ex.msg, "promotion of types Bit")
109+
ex = (@test_throws ErrorException promote(ae, falses(sz...))).value
110+
@test startswith(ex.msg, "promotion of types Bit")
111+
end
112+
113+
timesofar("promotions")
114+
101115
@testset "utility functions" begin
102116
b1 = bitrand(v1)
103117
@test isequal(fill!(b1, true), trues(size(b1)))
@@ -1767,4 +1781,4 @@ end
17671781
@test all(bitarray[rangeout, rangein] .== true)
17681782
@test all(bitarray[rangein, rangeout] .== true)
17691783
end
1770-
end
1784+
end

test/broadcast.jl

+9-6
Original file line numberDiff line numberDiff line change
@@ -691,16 +691,19 @@ end
691691
@test a == [1 1; 2 2; 3 3]
692692
end
693693

694-
@testset "scalar .=" begin
695-
A = [[1,2,3],4:5,6]
694+
@testset "scalar .= and promotion" begin
695+
A = [[1, 2, 3], 4:5, 6]
696+
@test A isa Vector{Any}
696697
A[1] .= 0
697-
@test A[1] == [0,0,0]
698+
@test A[1] == [0, 0, 0]
698699
@test_throws Base.CanonicalIndexError A[2] .= 0
699700
@test_throws MethodError A[3] .= 0
700-
A = [[1,2,3],4:5]
701+
A = [[1, 2, 3], 4:5]
702+
@test A isa Vector{Vector{Int}}
701703
A[1] .= 0
702-
@test A[1] == [0,0,0]
703-
@test_throws Base.CanonicalIndexError A[2] .= 0
704+
A[2] .= 0
705+
@test A[1] == [0, 0, 0]
706+
@test A[2] == [0, 0]
704707
end
705708

706709
# Issue #22180

0 commit comments

Comments
 (0)