Skip to content

Commit

Permalink
Fix for length(::StepRange{T}) where T isa Union
Browse files Browse the repository at this point in the history
And define `firstindex` accordingly.

Co-Authored-By: Jameson Nash <vtjnash+github@gmail.com>
  • Loading branch information
N5N3 and vtjnash committed May 12, 2022
1 parent f8fd837 commit 987a9b0
Show file tree
Hide file tree
Showing 2 changed files with 31 additions and 25 deletions.
46 changes: 22 additions & 24 deletions base/range.jl
Original file line number Diff line number Diff line change
Expand Up @@ -689,10 +689,6 @@ axes(r::AbstractRange) = (oneto(length(r)),)

# Needed to ensure `has_offset_axes` can constant-fold.
has_offset_axes(::StepRange) = false
let baseints = Union{Int8,UInt8,Int16,UInt16,Int32,UInt32,Int64,UInt64,Int128,UInt128}
global firstindex
firstindex(::StepRange{T,<:baseints}) where {T<:baseints} = sizeof(T) < sizeof(Int) ? 1 : one(T)
end

# n.b. checked_length for these is defined iff checked_add and checked_sub are
# defined between the relevant types
Expand Down Expand Up @@ -755,64 +751,66 @@ length(r::OneTo) = Integer(r.stop - zero(r.stop))
length(r::StepRangeLen) = r.len
length(r::LinRange) = r.len

let bigints = Union{Int, UInt, Int64, UInt64, Int128, UInt128}
global length, checked_length
let bigints = Union{Int, UInt, Int64, UInt64, Int128, UInt128},
smallints = (Int === Int64 ?
Union{Int8, UInt8, Int16, UInt16, Int32, UInt32} :
Union{Int8, UInt8, Int16, UInt16}),
bitints = Union{bigints, smallints}
global length, checked_length, firstindex
# compile optimization for which promote_type(T, Int) == T
length(r::OneTo{T}) where {T<:bigints} = r.stop
# slightly more accurate length and checked_length in extreme cases
# (near typemax) for types with known `unsigned` functions
function length(r::OrdinalRange{T}) where T<:bigints
s = step(r)
isempty(r) && return zero(T)
diff = last(r) - first(r)
isempty(r) && return zero(diff)
# if |s| > 1, diff might have overflowed, but unsigned(diff)÷s should
# therefore still be valid (if the result is representable at all)
# n.b. !(s isa T)
if s isa Unsigned || -1 <= s <= 1 || s == -s
a = div(diff, s) % T
a = div(diff, s) % typeof(diff)
elseif s < 0
a = div(unsigned(-diff), -s) % T
a = div(unsigned(-diff), -s) % typeof(diff)
else
a = div(unsigned(diff), s) % T
a = div(unsigned(diff), s) % typeof(diff)
end
return a + oneunit(T)
return a + oneunit(a)
end
function checked_length(r::OrdinalRange{T}) where T<:bigints
s = step(r)
isempty(r) && return zero(T)
stop, start = last(r), first(r)
ET = promote_type(typeof(stop), typeof(start))
isempty(r) && return zero(ET)
# n.b. !(s isa T)
if s > 1
diff = stop - start
a = convert(T, div(unsigned(diff), s))
a = convert(ET, div(unsigned(diff), s))
elseif s < -1
diff = start - stop
a = convert(T, div(unsigned(diff), -s))
a = convert(ET, div(unsigned(diff), -s))
elseif s > 0
a = div(checked_sub(stop, start), s)
a = convert(ET, div(checked_sub(stop, start), s))
else
a = div(checked_sub(start, stop), -s)
a = convert(ET, div(checked_sub(start, stop), -s))
end
return checked_add(convert(T, a), oneunit(T))
return checked_add(a, oneunit(a))
end
end
firstindex(r::StepRange{<:bigints,<:bitints}) = one(last(r)-first(r))

# some special cases to favor default Int type
let smallints = (Int === Int64 ?
Union{Int8, UInt8, Int16, UInt16, Int32, UInt32} :
Union{Int8, UInt8, Int16, UInt16})
global length, checked_length
# n.b. !(step isa T)
# some special cases to favor default Int type
function length(r::OrdinalRange{<:smallints})
s = step(r)
isempty(r) && return 0
# n.b. !(step isa T)
return Int(div(Int(last(r)) - Int(first(r)), s)) + 1
end
length(r::AbstractUnitRange{<:smallints}) = Int(last(r)) - Int(first(r)) + 1
length(r::OneTo{<:smallints}) = Int(r.stop)
checked_length(r::OrdinalRange{<:smallints}) = length(r)
checked_length(r::AbstractUnitRange{<:smallints}) = length(r)
checked_length(r::OneTo{<:smallints}) = length(r)
firstindex(::StepRange{<:smallints,<:bitints}) = 1
end

first(r::OrdinalRange{T}) where {T} = convert(T, r.start)
Expand Down
10 changes: 9 additions & 1 deletion test/ranges.jl
Original file line number Diff line number Diff line change
Expand Up @@ -2037,6 +2037,12 @@ end
@test typeof(length(r1)) == typeof(checked_length(r1)) ==
typeof(length(r2)) == typeof(checked_length(r2))
end
SR = StepRange{Union{Int64,Int128},Int}
test_length(r, l) = length(r) === checked_length(r) === l
@test test_length(SR(Int64(1), 1, Int128(1)), Int128(1))
@test test_length(SR(Int64(1), 1, Int128(0)), Int128(0))
@test test_length(SR(Int64(1), 1, Int64(1)), Int64(1))
@test test_length(SR(Int64(1), 1, Int64(0)), Int64(0))
end

@testset "LinRange eltype for element types that wrap integers" begin
Expand Down Expand Up @@ -2351,10 +2357,12 @@ end

@test length(range(1, length=typemax(Int128))) === typemax(Int128)

@testset "firstindex(::StepRange{T,T})" begin
@testset "firstindex(::StepRange{<:Base.BitInteger})" begin
test_firstindex(x) = firstindex(x) === first(Base.axes1(x))
for T in Base.BitInteger_types, S in Base.BitInteger_types
@test test_firstindex(StepRange{T,S}(1, 1, 1))
@test test_firstindex(StepRange{T,S}(1, 1, 0))
end
@test test_firstindex(StepRange{Union{Int64,Int128},Int}(Int64(1), 1, Int128(1)))
@test test_firstindex(StepRange{Union{Int64,Int128},Int}(Int64(1), 1, Int128(0)))
end

0 comments on commit 987a9b0

Please sign in to comment.