/base/grisu.jl

https://github.com/cloudchamber/julia · Lisp · 156 lines · 145 code · 11 blank · 0 comment · 31 complexity · d85fc709737405d817849c79331d29eb MD5 · raw file

  1. _jl_libgrisu = dlopen("libgrisu")
  2. const _jl_neg = Array(Bool,1)
  3. const _jl_digits = Array(Uint8,309+17)
  4. const _jl_buflen = int32(length(_jl_digits)+1)
  5. const _jl_length = Array(Int32,1)
  6. const _jl_point = Array(Int32,1)
  7. macro grisu_ccall(value, mode, ndigits)
  8. quote
  9. ccall(dlsym(_jl_libgrisu, :grisu), Void,
  10. (Float64, Int32, Int32, Ptr{Uint8}, Int32,
  11. Ptr{Bool}, Ptr{Int32}, Ptr{Int32}),
  12. $value, $mode, $ndigits,
  13. _jl_digits, _jl_buflen, _jl_neg, _jl_length, _jl_point)
  14. end
  15. end
  16. const GRISU_SHORTEST = int32(0) # shortest exact representation for doubles
  17. const GRISU_SHORTEST_SINGLE = int32(1) # shortest exact representation for singles
  18. const GRISU_FIXED = int32(2) # fixed number of trailing decimal points
  19. const GRISU_PRECISION = int32(3) # fixed precision regardless of magnitude
  20. # wrapper for the core grisu function, primarily for debugging
  21. global grisu, grisu_fix, grisu_sig
  22. function grisu(x::Float64, mode::Integer, ndigits::Integer)
  23. if !isfinite(x); error("non-finite value: $x"); end
  24. if ndigits < 0; error("negative digits requested"); end
  25. @grisu_ccall x mode ndigits
  26. _jl_neg[1], _jl_digits[1:_jl_length[1]], _jl_point[1]
  27. end
  28. grisu(x::Float64) = grisu(x, GRISU_SHORTEST, int32(0))
  29. grisu(x::Float32) = grisu(float64(x), GRISU_SHORTEST_SINGLE, int32(0))
  30. grisu(x::Real) = grisu(float(x))
  31. function grisu_fix(x::Real, n::Integer)
  32. if n > 17; n = 17; end
  33. grisu(float64(x), GRISU_FIXED, int32(n))
  34. end
  35. function grisu_sig(x::Real, n::Integer)
  36. if n > 309; n = 309; end
  37. grisu(float64(x), GRISU_PRECISION, int32(n))
  38. end
  39. function _show(io, x::Float, mode::Int32, n::Int)
  40. if isnan(x); return write(io, "NaN"); end
  41. if isinf(x); return write(io, x < 0 ? "-Inf" : "Inf"); end
  42. @grisu_ccall x mode n
  43. pdigits = pointer(_jl_digits)
  44. neg = _jl_neg[1]
  45. len = _jl_length[1]
  46. pt = _jl_point[1]
  47. if mode == GRISU_PRECISION
  48. while len > 1 && _jl_digits[len] == '0'
  49. len -= 1
  50. end
  51. end
  52. if neg
  53. write(io, '-')
  54. end
  55. if pt <= -4 || pt > 6 # .00001 to 100000.
  56. # => #.#######e###
  57. write(io, pdigits, 1)
  58. write(io, '.')
  59. if len > 1
  60. write(io, pdigits+1, len-1)
  61. else
  62. write(io, '0')
  63. end
  64. write(io, 'e')
  65. write(io, dec(pt-1))
  66. elseif pt <= 0
  67. # => 0.00########
  68. write(io, "0.")
  69. while pt < 0
  70. write(io, '0')
  71. pt += 1
  72. end
  73. write(io, pdigits, len)
  74. elseif pt >= len
  75. # => ########00.0
  76. write(io, pdigits, len)
  77. while pt > len
  78. write(io, '0')
  79. len += 1
  80. end
  81. write(io, ".0")
  82. else # => ####.####
  83. write(io, pdigits, pt)
  84. write(io, '.')
  85. write(io, pdigits+pt, len-pt)
  86. end
  87. nothing
  88. end
  89. show(io, x::Float64) = _show(io, x, GRISU_SHORTEST, 0)
  90. show(io, x::Float32) = _show(io, x, GRISU_SHORTEST_SINGLE, 0)
  91. showcompact(io, x::Float) = _show(io, x, GRISU_PRECISION, 6)
  92. # normal:
  93. # 0 < pt < len ####.#### len+1
  94. # pt <= 0 .000######## len-pt+1
  95. # len <= pt (dot) ########000. pt+1
  96. # len <= pt (no dot) ########000 pt
  97. # exponential:
  98. # pt <= 0 ########e-### len+k+2
  99. # 0 < pt ########e### len+k+1
  100. function _jl_print_shortest(io, x::Float, dot::Bool, mode::Int32)
  101. if isnan(x); return write(io, "NaN"); end
  102. if isinf(x); return write(io, x < 0 ? "-Inf" : "Inf"); end
  103. @grisu_ccall x mode 0
  104. pdigits = pointer(_jl_digits)
  105. neg = _jl_neg[1]
  106. len = _jl_length[1]
  107. pt = _jl_point[1]
  108. if neg
  109. write(io, '-')
  110. end
  111. e = pt-len
  112. k = -9<=e<=9 ? 1 : 2
  113. if -pt > k+1 || e+dot > k+1
  114. # => ########e###
  115. write(io, pdigits+0, len)
  116. write(io, 'e')
  117. write(io, dec(e))
  118. elseif pt <= 0
  119. # => .000########
  120. write(io, '.')
  121. while pt < 0
  122. write(io, '0')
  123. pt += 1
  124. end
  125. write(io, pdigits+0, len)
  126. elseif e >= dot
  127. # => ########000.
  128. write(io, pdigits+0, len)
  129. while e > 0
  130. write(io, '0')
  131. e -= 1
  132. end
  133. if dot
  134. write(io, '.')
  135. end
  136. else # => ####.####
  137. write(io, pdigits+0, pt)
  138. write(io, '.')
  139. write(io, pdigits+pt, len-pt)
  140. end
  141. nothing
  142. end
  143. print_shortest(io, x::Float64, dot::Bool) = _jl_print_shortest(io, x, dot, GRISU_SHORTEST)
  144. print_shortest(io, x::Float32, dot::Bool) = _jl_print_shortest(io, x, dot, GRISU_SHORTEST_SINGLE)
  145. print_shortest(io, x::Union(Float,Integer)) = print_shortest(io, float(x), false)