mirror of
https://github.com/nim-lang/Nim.git
synced 2026-04-19 22:10:33 +00:00
fix negative nan (#16628)
This commit is contained in:
@@ -2485,7 +2485,10 @@ proc gen(p: PProc, n: PNode, r: var TCompRes) =
|
||||
let f = n.floatVal
|
||||
case classify(f)
|
||||
of fcNan:
|
||||
r.res = rope"NaN"
|
||||
if signbit(f):
|
||||
r.res = rope"-NaN"
|
||||
else:
|
||||
r.res = rope"NaN"
|
||||
of fcNegZero:
|
||||
r.res = rope"-0.0"
|
||||
of fcZero:
|
||||
|
||||
@@ -8,7 +8,7 @@
|
||||
#
|
||||
|
||||
## Serialization utilities for the compiler.
|
||||
import strutils, math
|
||||
import std/[strutils, math]
|
||||
|
||||
# bcc on windows doesn't have C99 functions
|
||||
when defined(windows) and defined(bcc):
|
||||
@@ -33,10 +33,19 @@ when defined(windows) and defined(bcc):
|
||||
|
||||
proc c_snprintf(s: cstring; n:uint; frmt: cstring): cint {.importc: "snprintf", header: "<stdio.h>", nodecl, varargs.}
|
||||
|
||||
|
||||
when not declared(signbit):
|
||||
proc c_signbit(x: SomeFloat): cint {.importc: "signbit", header: "<math.h>".}
|
||||
proc signbit*(x: SomeFloat): bool {.inline.} =
|
||||
result = c_signbit(x) != 0
|
||||
|
||||
proc toStrMaxPrecision*(f: BiggestFloat, literalPostfix = ""): string =
|
||||
case classify(f)
|
||||
of fcNan:
|
||||
result = "NAN"
|
||||
if signbit(f):
|
||||
result = "-NAN"
|
||||
else:
|
||||
result = "NAN"
|
||||
of fcNegZero:
|
||||
result = "-0.0" & literalPostfix
|
||||
of fcZero:
|
||||
|
||||
@@ -319,6 +319,18 @@ template main =
|
||||
doAssert not Inf.isNaN
|
||||
doAssert isNaN(Inf - Inf)
|
||||
|
||||
block: # signbit
|
||||
let x1 = NaN
|
||||
let x2 = -NaN
|
||||
let x3 = -x1
|
||||
|
||||
doAssert isNaN(x1)
|
||||
doAssert isNaN(x2)
|
||||
doAssert isNaN(x3)
|
||||
doAssert not signbit(x1)
|
||||
doAssert signbit(x2)
|
||||
doAssert signbit(x3)
|
||||
|
||||
block: # copySign
|
||||
doAssert copySign(10.0, -1.0) == -10.0
|
||||
doAssert copySign(-10.0, -1.0) == -10.0
|
||||
@@ -385,8 +397,8 @@ template main =
|
||||
discard
|
||||
else:
|
||||
when not defined(js):
|
||||
doAssert copySign(-1.0, -NaN) == 1.0
|
||||
doAssert copySign(10.0, -NaN) == 10.0
|
||||
doAssert copySign(-1.0, -NaN) == -1.0
|
||||
doAssert copySign(10.0, -NaN) == -10.0
|
||||
doAssert copySign(1.0, copySign(NaN, -1.0)) == -1.0 # fails in VM
|
||||
|
||||
block:
|
||||
|
||||
Reference in New Issue
Block a user