Skip to content

Commit e396c03

Browse files
committed
A few more #26670 fixes
Fixes BigFloat digit rounding (JuliaIO/Formatting.jl#56). I've also tweaked the definitions to make it easier to extend to new number formats.
1 parent 89d2397 commit e396c03

File tree

5 files changed

+43
-39
lines changed

5 files changed

+43
-39
lines changed

base/float.jl

+17-17
Original file line numberDiff line numberDiff line change
@@ -348,26 +348,26 @@ trunc(::Type{Integer}, x::Float64) = trunc(Int,x)
348348
trunc(::Type{T}, x::Float16) where {T<:Integer} = trunc(T, Float32(x))
349349

350350
# fallbacks
351-
floor(::Type{T}, x::AbstractFloat) where {T<:Integer} = trunc(T,_round(x, RoundDown))
351+
floor(::Type{T}, x::AbstractFloat) where {T<:Integer} = trunc(T,round(x, RoundDown))
352352
floor(::Type{T}, x::Float16) where {T<:Integer} = floor(T, Float32(x))
353-
ceil(::Type{T}, x::AbstractFloat) where {T<:Integer} = trunc(T,_round(x, RoundUp))
353+
ceil(::Type{T}, x::AbstractFloat) where {T<:Integer} = trunc(T,round(x, RoundUp))
354354
ceil(::Type{T}, x::Float16) where {T<:Integer} = ceil(T, Float32(x))
355-
round(::Type{T}, x::AbstractFloat) where {T<:Integer} = trunc(T,_round(x, RoundNearest))
355+
round(::Type{T}, x::AbstractFloat) where {T<:Integer} = trunc(T,round(x, RoundNearest))
356356
round(::Type{T}, x::Float16) where {T<:Integer} = round(T, Float32(x))
357357

358-
_round(x::Float64, r::RoundingMode{:ToZero}) = trunc_llvm(x)
359-
_round(x::Float32, r::RoundingMode{:ToZero}) = trunc_llvm(x)
360-
_round(x::Float64, r::RoundingMode{:Down}) = floor_llvm(x)
361-
_round(x::Float32, r::RoundingMode{:Down}) = floor_llvm(x)
362-
_round(x::Float64, r::RoundingMode{:Up}) = ceil_llvm(x)
363-
_round(x::Float32, r::RoundingMode{:Up}) = ceil_llvm(x)
364-
_round(x::Float64, r::RoundingMode{:Nearest}) = rint_llvm(x)
365-
_round(x::Float32, r::RoundingMode{:Nearest}) = rint_llvm(x)
358+
round(x::Float64, r::RoundingMode{:ToZero}) = trunc_llvm(x)
359+
round(x::Float32, r::RoundingMode{:ToZero}) = trunc_llvm(x)
360+
round(x::Float64, r::RoundingMode{:Down}) = floor_llvm(x)
361+
round(x::Float32, r::RoundingMode{:Down}) = floor_llvm(x)
362+
round(x::Float64, r::RoundingMode{:Up}) = ceil_llvm(x)
363+
round(x::Float32, r::RoundingMode{:Up}) = ceil_llvm(x)
364+
round(x::Float64, r::RoundingMode{:Nearest}) = rint_llvm(x)
365+
round(x::Float32, r::RoundingMode{:Nearest}) = rint_llvm(x)
366366

367-
_round(x::Float16, r::RoundingMode{:ToZero}) = Float16(_round(Float32(x), r))
368-
_round(x::Float16, r::RoundingMode{:Down}) = Float16(_round(Float32(x), r))
369-
_round(x::Float16, r::RoundingMode{:Up}) = Float16(_round(Float32(x), r))
370-
_round(x::Float16, r::RoundingMode{:Nearest}) = Float16(_round(Float32(x), r))
367+
round(x::Float16, r::RoundingMode{:ToZero}) = Float16(round(Float32(x), r))
368+
round(x::Float16, r::RoundingMode{:Down}) = Float16(round(Float32(x), r))
369+
round(x::Float16, r::RoundingMode{:Up}) = Float16(round(Float32(x), r))
370+
round(x::Float16, r::RoundingMode{:Nearest}) = Float16(round(Float32(x), r))
371371

372372
## floating point promotions ##
373373
promote_rule(::Type{Float32}, ::Type{Float16}) = Float32
@@ -660,7 +660,7 @@ for Ti in (Int8, Int16, Int32, Int64, Int128, UInt8, UInt16, UInt32, UInt64, UIn
660660
end
661661
end
662662
function (::Type{$Ti})(x::$Tf)
663-
if ($(Tf(typemin(Ti))) <= x <= $(Tf(typemax(Ti)))) && (_round(x, RoundToZero) == x)
663+
if ($(Tf(typemin(Ti))) <= x <= $(Tf(typemax(Ti)))) && (round(x, RoundToZero) == x)
664664
return unsafe_trunc($Ti,x)
665665
else
666666
throw(InexactError($(Expr(:quote,Ti.name.name)), $Ti, x))
@@ -681,7 +681,7 @@ for Ti in (Int8, Int16, Int32, Int64, Int128, UInt8, UInt16, UInt32, UInt64, UIn
681681
end
682682
end
683683
function (::Type{$Ti})(x::$Tf)
684-
if ($(Tf(typemin(Ti))) <= x < $(Tf(typemax(Ti)))) && (_round(x, RoundToZero) == x)
684+
if ($(Tf(typemin(Ti))) <= x < $(Tf(typemax(Ti)))) && (round(x, RoundToZero) == x)
685685
return unsafe_trunc($Ti,x)
686686
else
687687
throw(InexactError($(Expr(:quote,Ti.name.name)), $Ti, x))

base/floatfuncs.jl

+12-8
Original file line numberDiff line numberDiff line change
@@ -114,13 +114,14 @@ julia> round(357.913; sigdigits=4, base=2)
114114
115115
# Extensions
116116
117-
To extend `round` to new numeric types, it is typically sufficient to define `Base._round(x::NewType, ::RoundingMode)`.
117+
To extend `round` to new numeric types, it is typically sufficient to define `Base.round(x::NewType, r::RoundingMode)`.
118118
"""
119119
round(T::Type, x)
120120

121121
round(::Type{T}, x::AbstractFloat, r::RoundingMode{:ToZero}) where {T<:Integer} = trunc(T, x)
122-
round(::Type{T}, x::AbstractFloat, r::RoundingMode) where {T<:Integer} = trunc(T, _round(x,r))
122+
round(::Type{T}, x::AbstractFloat, r::RoundingMode) where {T<:Integer} = trunc(T, round(x,r))
123123

124+
# NOTE: this relies on the current keyword dispatch behaviour (#9498).
124125
function round(x::Real, r::RoundingMode=RoundNearest;
125126
digits::Union{Nothing,Integer}=nothing, sigdigits::Union{Nothing,Integer}=nothing, base=10)
126127
isfinite(x) || return x
@@ -130,12 +131,15 @@ trunc(x::Real; kwargs...) = round(x, RoundToZero; kwargs...)
130131
floor(x::Real; kwargs...) = round(x, RoundDown; kwargs...)
131132
ceil(x::Real; kwargs...) = round(x, RoundUp; kwargs...)
132133

133-
_round(x, r::RoundingMode, digits::Nothing, sigdigits::Nothing, base) = _round(x, r)
134-
_round(x::Integer, r::RoundingMode) = x
134+
# avoid recursive calls
135+
round(x::Real, r::RoundingMode) = throw(MethodError(round, (x,r)))
136+
round(x::Integer, r::RoundingMode) = x
137+
138+
_round(x, r::RoundingMode, digits::Nothing, sigdigits::Nothing, base) = round(x, r)
135139

136140
# round x to multiples of 1/invstep
137141
function _round_invstep(x, invstep, r::RoundingMode)
138-
y = _round(x * invstep, r) / invstep
142+
y = round(x * invstep, r) / invstep
139143
if !isfinite(y)
140144
return x
141145
end
@@ -145,7 +149,7 @@ end
145149
# round x to multiples of step
146150
function _round_step(x, step, r::RoundingMode)
147151
# TODO: use div with rounding mode
148-
y = _round(x / step, r) * step
152+
y = round(x / step, r) * step
149153
if !isfinite(y)
150154
if x > 0
151155
return (r == RoundUp ? oftype(x, Inf) : zero(x))
@@ -191,12 +195,12 @@ _round(x, r::RoundingMode, digits::Integer, sigdigits::Integer, base) =
191195
throw(ArgumentError("`round` cannot use both `digits` and `sigdigits` arguments."))
192196

193197
# C-style round
194-
function _round(x::AbstractFloat, ::RoundingMode{:NearestTiesAway})
198+
function round(x::AbstractFloat, ::RoundingMode{:NearestTiesAway})
195199
y = trunc(x)
196200
ifelse(x==y,y,trunc(2*x-y))
197201
end
198202
# Java-style round
199-
function _round(x::AbstractFloat, ::RoundingMode{:NearestTiesUp})
203+
function round(x::AbstractFloat, ::RoundingMode{:NearestTiesUp})
200204
y = floor(x)
201205
ifelse(x==y,y,copysign(floor(2*x-y),x))
202206
end

base/irrationals.jl

+1-1
Original file line numberDiff line numberDiff line change
@@ -125,7 +125,7 @@ for op in Symbol[:+, :-, :*, :/, :^]
125125
end
126126
*(x::Bool, y::AbstractIrrational) = ifelse(x, Float64(y), 0.0)
127127

128-
_round(x::Irrational, r::RoundingMode) = _round(float(x), r)
128+
round(x::Irrational, r::RoundingMode) = round(float(x), r)
129129

130130
macro irrational(sym, val, def)
131131
esym = esc(sym)

base/mpfr.jl

+6-13
Original file line numberDiff line numberDiff line change
@@ -827,27 +827,20 @@ function isinteger(x::BigFloat)
827827
return ccall((:mpfr_integer_p, :libmpfr), Int32, (Ref{BigFloat},), x) != 0
828828
end
829829

830-
for f in (:ceil, :floor, :trunc)
830+
for (f,R) in ((:roundeven, :Nearest),
831+
(:ceil, :Up),
832+
(:floor, :Down),
833+
(:trunc, :ToZero),
834+
(:round, :NearestTiesAway))
831835
@eval begin
832-
function ($f)(x::BigFloat)
836+
function round(x::BigFloat, ::RoundingMode{$(QuoteNode(R))})
833837
z = BigFloat()
834838
ccall(($(string(:mpfr_,f)), :libmpfr), Int32, (Ref{BigFloat}, Ref{BigFloat}), z, x)
835839
return z
836840
end
837841
end
838842
end
839843

840-
function round(x::BigFloat)
841-
z = BigFloat()
842-
ccall((:mpfr_rint, :libmpfr), Int32, (Ref{BigFloat}, Ref{BigFloat}, Cint), z, x, ROUNDING_MODE[])
843-
return z
844-
end
845-
function round(x::BigFloat,::RoundingMode{:NearestTiesAway})
846-
z = BigFloat()
847-
ccall((:mpfr_round, :libmpfr), Int32, (Ref{BigFloat}, Ref{BigFloat}), z, x)
848-
return z
849-
end
850-
851844
function isinf(x::BigFloat)
852845
return ccall((:mpfr_inf_p, :libmpfr), Int32, (Ref{BigFloat},), x) != 0
853846
end

test/rounding.jl

+7
Original file line numberDiff line numberDiff line change
@@ -272,6 +272,13 @@ end
272272
@test round(pi, sigdigits=1) 3.
273273
@test round(pi, sigdigits=3) 3.14
274274
@test round(pi, sigdigits=4, base=2) 3.25
275+
@test round(big(pi)) big"3."
276+
@test round(big(pi), digits=0) big"3."
277+
@test round(big(pi), digits=1) big"3.1"
278+
@test round(big(pi), digits=3, base=2) big"3.125"
279+
@test round(big(pi), sigdigits=1) big"3."
280+
@test round(big(pi), sigdigits=3) big"3.14"
281+
@test round(big(pi), sigdigits=4, base=2) big"3.25"
275282
@test round(10*pi, digits=-1) 30.
276283
@test round(.1, digits=0) == 0.
277284
@test round(-.1, digits=0) == -0.

0 commit comments

Comments
 (0)