From 8dd77bd77723f7021f4f995a37bc7064ecbf1a9f Mon Sep 17 00:00:00 2001 From: Jeff Bezanson Date: Fri, 22 Sep 2017 15:12:26 -0400 Subject: [PATCH] fix #15489, don't check conversions during mixed signedness arithmetic also skip checks for integer bitwise ops --- base/essentials.jl | 2 +- base/int.jl | 9 +++++++++ base/promotion.jl | 4 ---- test/int.jl | 8 ++++++++ 4 files changed, 18 insertions(+), 5 deletions(-) diff --git a/base/essentials.jl b/base/essentials.jl index d762cfdd17bec..0fe46630a3aaf 100644 --- a/base/essentials.jl +++ b/base/essentials.jl @@ -533,7 +533,7 @@ end endof(v::SimpleVector) = length(v) start(v::SimpleVector) = 1 next(v::SimpleVector,i) = (v[i],i+1) -done(v::SimpleVector,i) = (i > length(v)) +done(v::SimpleVector,i) = (length(v) < i) isempty(v::SimpleVector) = (length(v) == 0) indices(v::SimpleVector) = (OneTo(length(v)),) linearindices(v::SimpleVector) = indices(v, 1) diff --git a/base/int.jl b/base/int.jl index 4d3f3c1522aa1..6d063368f7140 100644 --- a/base/int.jl +++ b/base/int.jl @@ -517,6 +517,7 @@ if module_name(@__MODULE__) === :Base end rem(x::T, ::Type{T}) where {T<:Integer} = x +rem(x::Integer, T::Type{<:Integer}) = convert(T, x) # `x % T` falls back to `convert` rem(x::Integer, ::Type{Bool}) = ((x & 1) != 0) mod(x::Integer, ::Type{T}) where {T<:Integer} = rem(x, T) @@ -767,3 +768,11 @@ else rem(x::Int128, y::Int128) = checked_srem_int(x, y) rem(x::UInt128, y::UInt128) = checked_urem_int(x, y) end + +# issue #15489: since integer ops are unchecked, they shouldn't check promotion +for op in (:+, :-, :*, :&, :|, :xor) + @eval function $op(a::Integer, b::Integer) + T = promote_typeof(a, b) + return $op(a % T, b % T) + end +end diff --git a/base/promotion.jl b/base/promotion.jl index 928f8a38808f4..3402cb836ab22 100644 --- a/base/promotion.jl +++ b/base/promotion.jl @@ -310,10 +310,6 @@ julia> A^3 fma(x::Number, y::Number, z::Number) = fma(promote(x,y,z)...) muladd(x::Number, y::Number, z::Number) = muladd(promote(x,y,z)...) -(&)(x::Integer, y::Integer) = (&)(promote(x,y)...) -(|)(x::Integer, y::Integer) = (|)(promote(x,y)...) -xor(x::Integer, y::Integer) = xor(promote(x,y)...) - ==(x::Number, y::Number) = (==)(promote(x,y)...) <( x::Real, y::Real) = (< )(promote(x,y)...) <=(x::Real, y::Real) = (<=)(promote(x,y)...) diff --git a/test/int.jl b/test/int.jl index f8e8432c7ac2d..9ad8b5ff5710c 100644 --- a/test/int.jl +++ b/test/int.jl @@ -223,6 +223,14 @@ for T in [Base.BitInteger_types..., BigInt], @test typeof(rand(U(0):U(127)) % T) === T end +# issue #15489 +@test 0x00007ffea27edaa0 + (-40) === (-40) + 0x00007ffea27edaa0 === 0x00007ffea27eda78 +@test UInt64(1) * Int64(-1) === typemax(UInt64) +@test UInt(1) - (-1) == 2 +@test UInt64(15) & -4 === UInt64(12) +@test UInt64(15) | -4 === typemax(UInt64) +@test UInt64(15) ⊻ -4 === 0xfffffffffffffff3 + @testset "left shift with Vector{Int} on BigInt-scalar #13832" begin x = BigInt(1) .<< [1:70;]