aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorTristan Gingold <tgingold@free.fr>2017-04-16 11:31:41 +0200
committerTristan Gingold <tgingold@free.fr>2017-04-19 20:48:23 +0200
commit918d0411347bdb2d8e5027a58803a072613e02b2 (patch)
treec078d225731eb96d0c9f5ecb634ea75370208b11
parent7d17b527ea6bad8ef1e3f5450c0d7899c60febee (diff)
downloadghdl-918d0411347bdb2d8e5027a58803a072613e02b2.tar.gz
ghdl-918d0411347bdb2d8e5027a58803a072613e02b2.tar.bz2
ghdl-918d0411347bdb2d8e5027a58803a072613e02b2.zip
Use grt.fcvt for 'image.
-rw-r--r--src/grt/grt-fcvt.adb176
-rw-r--r--src/grt/grt-fcvt.ads26
-rw-r--r--src/grt/grt-vstrings.adb90
-rw-r--r--src/vhdl/evaluation.adb79
4 files changed, 171 insertions, 200 deletions
diff --git a/src/grt/grt-fcvt.adb b/src/grt/grt-fcvt.adb
index 2e3cd0d23..0757886a2 100644
--- a/src/grt/grt-fcvt.adb
+++ b/src/grt/grt-fcvt.adb
@@ -574,6 +574,26 @@ package body Grt.Fcvt is
Len := Len + 1;
end Append;
+ procedure Insert (Str : in out String;
+ Len : in out Natural;
+ Pos : Positive;
+ C : Character)
+ is
+ Prev_C, C1 : Character;
+ begin
+ if Pos > Str'Length then
+ return;
+ end if;
+
+ C1 := C;
+ for I in Pos .. Positive'Min (Str'Length, Len) loop
+ Prev_C := Str (I);
+ Str (I) := C1;
+ C1 := Prev_C;
+ end loop;
+ Append (Str, Len, C1);
+ end Insert;
+
procedure Append_Digit (Str : in out String;
Len : in out Natural;
D : Natural) is
@@ -764,19 +784,11 @@ package body Grt.Fcvt is
end Dragon4;
procedure Output_Nan_Inf (Str : out String;
- Len : out Natural;
- Is_Inf : Boolean;
- S : Boolean) is
+ Len : in out Natural;
+ Is_Inf : Boolean) is
begin
- Len := 0;
-
if Is_Inf then
-- Infinite
- if S then
- Append (Str, Len, '-');
- else
- Append (Str, Len, '+');
- end if;
Append (Str, Len, 'i');
Append (Str, Len, 'n');
Append (Str, Len, 'f');
@@ -789,6 +801,9 @@ package body Grt.Fcvt is
procedure To_String (Str : out String;
Len : out Natural;
+ Is_Num : out Boolean;
+ Is_Neg : out Boolean;
+ Exp : out Integer;
V : IEEE_Float_64)
is
pragma Assert (Str'First = 1);
@@ -800,16 +815,19 @@ package body Grt.Fcvt is
E : constant Integer := Integer (Shift_Right (V_Bits, 52) and 16#7_ff#);
Ctxt : Fcvt_Context;
- First : Natural;
begin
+ Is_Neg := S;
+ Len := 0;
+
-- Handle NaN & Inf
if E = 2047 then
- Output_Nan_Inf (Str, Len, M = 0, S);
+ Output_Nan_Inf (Str, Len, M = 0);
+ Is_Num := False;
return;
end if;
-- Normal or denormal float.
- Len := 0;
+ Is_Num := True;
Ctxt.F.N := 2;
Ctxt.F.V (1) := Unsigned_32 (M and 16#ffff_ffff#);
@@ -821,11 +839,6 @@ package body Grt.Fcvt is
-- Bignum digits may be 0.
Bignum_Normalize (Ctxt.F);
- -- Display sign.
- if S then
- Append (Str, Len, '-');
- end if;
-
Ctxt.Is_Emin := True;
Ctxt.Is_Pow2 := False; -- Not needed.
@@ -845,11 +858,6 @@ package body Grt.Fcvt is
-- Implicit leading 1.
Ctxt.F.V (2) := Ctxt.F.V (2) or 16#10_00_00#;
- -- Display sign.
- if S then
- Append (Str, Len, '-');
- end if;
-
Ctxt.Is_Emin := False;
Ctxt.Is_Pow2 := M = 0;
Ctxt.Log2v := E - 1023;
@@ -857,8 +865,6 @@ package body Grt.Fcvt is
pragma Assert (Bignum_Is_Valid (Ctxt.F));
- First := Len;
-
-- At this point, the number is represented as:
-- F * 2**K
if Ctxt.F.N = 0 then
@@ -869,23 +875,39 @@ package body Grt.Fcvt is
Dragon4 (Str, Len, Ctxt);
end if;
+ Exp := Ctxt.K;
+ end To_String;
+
+ procedure To_String (Str : out String;
+ Len : out Natural;
+ V : IEEE_Float_64)
+ is
+ Is_Num : Boolean;
+ Is_Neg : Boolean;
+ Exp : Integer;
+ First : Positive;
+ begin
+ To_String (Str, Len, Is_Num, Is_Neg, Exp, V);
+
+ -- Handle sign.
+ if Is_Neg then
+ First := 2;
+ Insert (Str, Len, 1, '-');
+ else
+ First := 1;
+ end if;
+
+ if not Is_Num then
+ return;
+ end if;
+
+ -- At this point STR contains the minus sign (if any) and digits.
+ -- The value is 0.NNNN * 10**K
+
-- Formatting.
-- Insert the dot.
- declare
- C, Prev_C : Character;
- begin
- if Len > First + 1 then
- C := '.';
- for I in First + 2 .. Len loop
- Prev_C := Str (I);
- Str (I) := C;
- C := Prev_C;
- end loop;
- Len := Len + 1;
- Str (Len) := C;
- end if;
- Ctxt.K := Ctxt.K - 1;
- end;
+ Insert (Str, Len, First + 1, '.');
+ Exp := Exp - 1;
Append (Str, Len, 'e');
declare
@@ -893,7 +915,7 @@ package body Grt.Fcvt is
T : Integer;
Den : Natural;
begin
- K := Ctxt.K;
+ K := Exp;
if K < 0 then
Append (Str, Len, '-');
K := -K;
@@ -1087,4 +1109,76 @@ package body Grt.Fcvt is
-- F * 10**EXP
return To_Float_64 (Neg, F, 10, Exp);
end From_String;
+
+ procedure Format_Image
+ (Str : out String; Last : out Natural; N : IEEE_Float_64)
+ is
+ P : Natural;
+ S : String (1 .. 20);
+ Len : Natural;
+ Is_Num : Boolean;
+ Is_Neg : Boolean;
+ Exp : Integer;
+ begin
+ To_String (S, Len, Is_Num, Is_Neg, Exp, N);
+
+ -- The sign.
+ P := Str'First;
+ if Is_Neg then
+ Str (P) := '-';
+ P := P + 1;
+ end if;
+
+ -- Non numbers
+ if not Is_Num then
+ Str (P .. P + Len - 1) := S (1 .. Len);
+ Last := P + Len - 1;
+ return;
+ end if;
+
+ -- Mantissa N.NNNNN
+ Str (P) := S (1);
+ Str (P + 1) := '.';
+ Exp := Exp - 1;
+ if Len = 1 then
+ Str (P + 2) := '0';
+ P := P + 3;
+ else
+ Str (P + 2 .. P + 2 + Len - 2) := S (2 .. Len);
+ P := P + 2 + Len - 2 + 1;
+ end if;
+
+ -- Exponent
+ if Exp /= 0 then
+ -- LRM93 14.3
+ -- if the exponent is present, the `e' is written as a lower case
+ -- character.
+ Str (P) := 'e';
+ P := P + 1;
+
+ if Exp < 0 then
+ Str (P) := '-';
+ P := P + 1;
+ Exp := -Exp;
+ end if;
+ declare
+ B : Boolean;
+ D : Natural;
+ begin
+ B := False;
+ for I in 0 .. 4 loop
+ D := (Exp / 10000) mod 10;
+ if D /= 0 or B or I = 4 then
+ Str (P) := Character'Val (48 + D);
+ P := P + 1;
+ B := True;
+ end if;
+ Exp := (Exp - D * 10000) * 10;
+ end loop;
+ end;
+ end if;
+
+ Last := P - 1;
+ end Format_Image;
+
end Grt.Fcvt;
diff --git a/src/grt/grt-fcvt.ads b/src/grt/grt-fcvt.ads
index e1e272850..2ec42fcfa 100644
--- a/src/grt/grt-fcvt.ads
+++ b/src/grt/grt-fcvt.ads
@@ -30,6 +30,32 @@
with Interfaces; use Interfaces;
package Grt.Fcvt is
+ pragma Preelaborate;
+
+ -- Convert (without formatting) number V.
+ -- The result is LEN characters stored to STR. If STR is too short, then
+ -- the output is truncated but LEN contains the number of characters that
+ -- would be needed. The procedure assumes STR'First = 1.
+ -- The digits are the digits after the point. The output has to be read as
+ -- 0.NNNNN * 10**EXP
+ -- IS_NUM is true if V is a number and false for NaN or Infinite.
+ -- IS_NEG is true if the number if negative.
+ procedure To_String (Str : out String;
+ Len : out Natural;
+ Is_Num : out Boolean;
+ Is_Neg : out Boolean;
+ Exp : out Integer;
+ V : IEEE_Float_64);
+
+ -- Formatting.
+
+ -- For 'Image
+ -- STR must have at least 24 characters:
+ -- Sign (1) + digit (1) + dot (1) + digits (15) + 'e' (1) + sign (1)
+ -- + exp_digits (4) -> 24.
+ procedure Format_Image
+ (Str : out String; Last : out Natural; N : IEEE_Float_64);
+
-- Convert V to 10-based number stored (in ASCII) in STR/LEN [using at most
-- NDIGITS digits.]
-- LEN is the number of characters needed (so it may be greater than
diff --git a/src/grt/grt-vstrings.adb b/src/grt/grt-vstrings.adb
index d5d14c856..544626f46 100644
--- a/src/grt/grt-vstrings.adb
+++ b/src/grt/grt-vstrings.adb
@@ -22,10 +22,11 @@
-- covered by the GNU General Public License. This exception does not
-- however invalidate any other reasons why the executable file might be
-- covered by the GNU Public License.
-with System.Storage_Elements; -- Work around GNAT bug.
-pragma Unreferenced (System.Storage_Elements);
+
+with Interfaces;
with Grt.Errors; use Grt.Errors;
with Grt.C; use Grt.C;
+with Grt.Fcvt;
package body Grt.Vstrings is
procedure Free (Fs : Fat_String_Acc);
@@ -262,90 +263,9 @@ package body Grt.Vstrings is
procedure To_String (Str : out String; First : out Natural; N : Ghdl_I64)
renames To_String_I64;
- procedure To_String (Str : out String; Last : out Natural; N : Ghdl_F64)
- is
- function Trunc (V : Ghdl_F64) return Ghdl_F64;
- pragma Import (C, Trunc);
-
- P : Natural := Str'First;
- V : Ghdl_F64;
- Vmax : Ghdl_F64;
- Vd : Ghdl_F64;
- Exp : Integer;
- D : Integer;
- B : Boolean;
+ procedure To_String (Str : out String; Last : out Natural; N : Ghdl_F64) is
begin
- -- Handle sign.
- if N < 0.0 then
- Str (P) := '-';
- P := P + 1;
- V := -N;
- else
- V := N;
- end if;
-
- -- Compute the mantissa.
- -- and normalize V in [0 .. 10.0[
- -- FIXME: should do a dichotomy.
- if V = 0.0 then
- Exp := 0;
- elsif V < 1.0 then
- Exp := 0;
- loop
- exit when V >= 1.0;
- Exp := Exp - 1;
- V := V * 10.0;
- end loop;
- else
- Exp := 0;
- loop
- exit when V < 10.0;
- Exp := Exp + 1;
- V := V / 10.0;
- end loop;
- end if;
-
- Vmax := 10.0 ** (1 - 15);
- for I in 0 .. 15 loop
- -- Vd := Ghdl_F64'Truncation (V);
- Vd := Trunc (V);
- Str (P) := Character'Val (48 + Integer (Vd));
- P := P + 1;
- V := (V - Vd) * 10.0;
-
- if I = 0 then
- Str (P) := '.';
- P := P + 1;
- end if;
- exit when I > 0 and V < Vmax;
- Vmax := Vmax * 10.0;
- end loop;
-
- if Exp /= 0 then
- -- LRM93 14.3
- -- if the exponent is present, the `e' is written as a lower case
- -- character.
- Str (P) := 'e';
- P := P + 1;
-
- if Exp < 0 then
- Str (P) := '-';
- P := P + 1;
- Exp := -Exp;
- end if;
- B := False;
- for I in 0 .. 4 loop
- D := (Exp / 10000) mod 10;
- if D /= 0 or B or I = 4 then
- Str (P) := Character'Val (48 + D);
- P := P + 1;
- B := True;
- end if;
- Exp := (Exp - D * 10000) * 10;
- end loop;
- end if;
-
- Last := P - 1;
+ Grt.Fcvt.Format_Image (Str, Last, Interfaces.IEEE_Float_64 (N));
end To_String;
procedure To_String (Str : out String_Real_Digits;
diff --git a/src/vhdl/evaluation.adb b/src/vhdl/evaluation.adb
index 85c99209f..214deb5ca 100644
--- a/src/vhdl/evaluation.adb
+++ b/src/vhdl/evaluation.adb
@@ -16,6 +16,7 @@
-- Software Foundation, 59 Temple Place - Suite 330, Boston, MA
-- 02111-1307, USA.
with Ada.Unchecked_Deallocation;
+with Interfaces;
with Scanner;
with Errorout; use Errorout;
with Name_Table; use Name_Table;
@@ -25,6 +26,7 @@ with Std_Package; use Std_Package;
with Flags; use Flags;
with Std_Names;
with Ada.Characters.Handling;
+with Grt.Fcvt;
package body Evaluation is
-- If FORCE is true, always return a literal.
@@ -1763,87 +1765,16 @@ package body Evaluation is
use Str_Table;
Id : String8_Id;
- -- Sign (1) + digit (1) + dot (1) + digits (15) + exp (1) + sign (1)
+ -- Sign (1) + digit (1) + dot (1) + digits (15) + 'e' (1) + sign (1)
-- + exp_digits (4) -> 24.
Str : String (1 .. 25);
P : Natural;
- V : Iir_Fp64;
- Vd : Iir_Fp64;
- Exp : Integer;
- D : Integer;
- B : Boolean;
Res : Iir;
begin
- -- Handle sign.
- if Val < 0.0 then
- Str (1) := '-';
- P := 1;
- V := -Val;
- else
- P := 0;
- V := Val;
- end if;
-
- -- Compute the mantissa.
- -- FIXME: should do a dichotomy.
- if V = 0.0 then
- Exp := 0;
- elsif V < 1.0 then
- Exp := -1;
- while V * (10.0 ** (-Exp)) < 1.0 loop
- Exp := Exp - 1;
- end loop;
- else
- Exp := 0;
- while V / (10.0 ** Exp) >= 10.0 loop
- Exp := Exp + 1;
- end loop;
- end if;
-
- -- Normalize VAL: in [0; 10[
- if Exp >= 0 then
- V := V / (10.0 ** Exp);
- else
- V := V * 10.0 ** (-Exp);
- end if;
+ P := Str'First;
- for I in 0 .. 15 loop
- Vd := Iir_Fp64'Truncation (V);
- P := P + 1;
- Str (P) := Character'Val (48 + Integer (Vd));
- V := (V - Vd) * 10.0;
-
- if I = 0 then
- P := P + 1;
- Str (P) := '.';
- end if;
- exit when I > 0 and V < 10.0 ** (I + 1 - 15);
- end loop;
-
- if Exp /= 0 then
- -- LRM93 14.3
- -- if the exponent is present, the `e' is written as a lower case
- -- character.
- P := P + 1;
- Str (P) := 'e';
-
- if Exp < 0 then
- P := P + 1;
- Str (P) := '-';
- Exp := -Exp;
- end if;
- B := False;
- for I in 0 .. 4 loop
- D := (Exp / 10000) mod 10;
- if D /= 0 or B or I = 4 then
- P := P + 1;
- Str (P) := Character'Val (48 + D);
- B := True;
- end if;
- Exp := (Exp - D * 10000) * 10;
- end loop;
- end if;
+ Grt.Fcvt.Format_Image (Str, P, Interfaces.IEEE_Float_64 (Val));
Id := Create_String8;
for I in 1 .. P loop