diff options
Diffstat (limited to 'src')
| -rw-r--r-- | src/grt/grt-values.adb | 71 | ||||
| -rw-r--r-- | src/grt/grt-values.ads | 36 | ||||
| -rw-r--r-- | src/vhdl/vhdl-evaluation.adb | 23 | 
3 files changed, 97 insertions, 33 deletions
| diff --git a/src/grt/grt-values.adb b/src/grt/grt-values.adb index 26d15753d..404fec43c 100644 --- a/src/grt/grt-values.adb +++ b/src/grt/grt-values.adb @@ -135,9 +135,9 @@ package body Grt.Values is     end Ghdl_Value_E32;     --  Convert S (INIT_POS .. LEN) to a signed integer. -   function Value_I64 -     (S : Std_String_Basep; Len : Ghdl_Index_Type; Init_Pos : Ghdl_Index_Type) -     return Ghdl_I64 +   function Value_I64 (S : Std_String_Basep; +                       Len : Ghdl_Index_Type; +                       Init_Pos : Ghdl_Index_Type) return Value_I64_Result     is        Pos : Ghdl_Index_Type := Init_Pos;        C : Character; @@ -147,6 +147,7 @@ package body Grt.Values is        Is_Neg : Boolean;     begin        C := S (Pos); +      Val := 0;        --  LRM02 14.1 Predefined attributes        --  Restrictions: It is an error is the parameter is not a valid string @@ -160,14 +161,13 @@ package body Grt.Values is        Is_Neg := False;        if C = '+' or C = '-' then           if Pos = Len then -            Error_E ("'value: missing digit after +/-"); +            return (Status => Value_Err_No_Digit, Pos => Pos);           end if;           Pos := Pos + 1;           Is_Neg := C = '-';           C := S (Pos);        end if; -      Val := 0;        loop           if C in '0' .. '9' then              Val := Val * 10 - (Character'Pos (C) - Character'Pos ('0')); @@ -175,13 +175,13 @@ package body Grt.Values is              exit when Pos >= Len;              C := S (Pos);           else -            Error_E ("'value: decimal digit expected"); +            return (Status => Value_Err_No_Digit, Pos => Pos);           end if;           case C is              when '_' =>                 Pos := Pos + 1;                 if Pos >= Len then -                  Error_E ("'value: trailing underscore"); +                  return (Status => Value_Err_Underscore, Pos => Pos);                 end if;                 C := S (Pos);              when '#' @@ -203,7 +203,7 @@ package body Grt.Values is           if not Is_Neg then              Val := -Val;           end if; -         return Val; +         return (Status => Value_Ok, Val => Val);        end if;        if C = '#' or C = ':' then @@ -212,10 +212,10 @@ package body Grt.Values is           Sep := C;           Pos := Pos + 1;           if Base < 2 or Base > 16 then -            Error_E ("'value: bad base"); +            return (Status => Value_Err_Bad_Base, Pos => Pos);           end if;           if Pos >= Len then -            Error_E ("'value: missing based integer"); +            return (Status => Value_Err_No_Digit, Pos => Pos);           end if;           C := S (Pos);           loop @@ -227,27 +227,27 @@ package body Grt.Values is                 when 'A' .. 'F' =>                    D := Character'Pos (C) - Character'Pos ('A') + 10;                 when others => -                  Error_E ("'value: digit expected"); +                  return (Status => Value_Err_Bad_Digit, Pos => Pos);              end case;              if D >= Base then -               Error_E ("'value: digit >= base"); +               return (Status => Value_Err_Bad_Digit, Pos => Pos);              end if;              Val := Val * Base - D;              Pos := Pos + 1;              if Pos >= Len then -               Error_E ("'value: missing end sign number"); +               return (Status => Value_Err_Bad_End_Sign, Pos => Pos);              end if;              C := S (Pos);              if C = '#' or C = ':' then                 if C /= Sep then -                  Error_E ("'value: sign number mismatch"); +                  return (Status => Value_Err_Bad_End_Sign, Pos => Pos);                 end if;                 Pos := Pos + 1;                 exit;              elsif C = '_' then                 Pos := Pos + 1;                 if Pos >= Len then -                  Error_E ("'value: no character after underscore"); +                  return (Status => Value_Err_Underscore, Pos => Pos);                 end if;                 C := S (Pos);              end if; @@ -260,17 +260,17 @@ package body Grt.Values is        if C = 'e' or C = 'E' then           Pos := Pos + 1;           if Pos >= Len then -            Error_E ("'value: no character after exponent"); +            return (Status => Value_Err_No_Digit, Pos => Pos);           end if;           C := S (Pos);           if C = '+' then              Pos := Pos + 1;              if Pos >= Len then -               Error_E ("'value: no character after sign"); +               return (Status => Value_Err_No_Digit, Pos => Pos);              end if;              C := S (Pos);           elsif C = '-' then -            Error_E ("'value: negativ exponent not allowed"); +            return (Status => Value_Err_Bad_Exponent, Pos => Pos);           end if;           Exp := 0;           loop @@ -280,13 +280,13 @@ package body Grt.Values is                 exit when Pos >= Len;                 C := S (Pos);              else -               Error_E ("'value: decimal digit expected"); +               return (Status => Value_Err_Bad_Digit, Pos => Pos);              end if;              case C is                 when '_' =>                    Pos := Pos + 1;                    if Pos >= Len then -                     Error_E ("'value: trailing underscore"); +                     return (Status => Value_Err_Underscore, Pos => Pos);                    end if;                    C := S (Pos);                 when ' ' @@ -308,14 +308,41 @@ package body Grt.Values is        end if;        if Pos /= Len then -         Error_E ("'value: trailing characters after blank"); +         return (Status => Value_Err_Trailing_Chars, Pos => Pos);        end if;        if not Is_Neg then           Val := -Val;        end if; -      return Val; +      return (Status => Value_Ok, Val => Val); +   end Value_I64; + +   function Value_I64 +     (S : Std_String_Basep; Len : Ghdl_Index_Type; Init_Pos : Ghdl_Index_Type) +     return Ghdl_I64 +   is +      Res : Value_I64_Result; +   begin +      Res := Value_I64 (S, Len, Init_Pos); +      case Res.Status is +         when Value_Ok => +            return Res.Val; +         when Value_Err_No_Digit => +            Error_E ("'value: missing digit"); +         when Value_Err_Underscore => +            Error_E ("'value: incorrect underscore"); +         when Value_Err_Bad_Base => +            Error_E ("'value: bad base"); +         when Value_Err_Bad_Digit => +            Error_E ("'value: digit expected"); +         when Value_Err_Bad_End_Sign => +            Error_E ("'value: incorrect or missing sign number"); +         when Value_Err_Bad_Exponent => +            Error_E ("'value: negativ exponent not allowed"); +         when Value_Err_Trailing_Chars => +            Error_E ("'value: trailing characters after blank"); +      end case;     end Value_I64;     function Ghdl_Value_I64 (Str : Std_String_Ptr) return Ghdl_I64 diff --git a/src/grt/grt-values.ads b/src/grt/grt-values.ads index 1269d194f..c88e1dfc3 100644 --- a/src/grt/grt-values.ads +++ b/src/grt/grt-values.ads @@ -50,15 +50,41 @@ package Grt.Values is     function Ghdl_Value_P32 (Str : Std_String_Ptr; Rti : Ghdl_Rti_Access)                             return Ghdl_I32; -   --  Convert S (INIT_POS .. LEN) to a signed integer. -   function Value_I64 -     (S : Std_String_Basep; Len : Ghdl_Index_Type; Init_Pos : Ghdl_Index_Type) -     return Ghdl_I64; -     --  Return the value of STR for enumerated type RTI.     function Value_Enum       (S : Std_String_Basep; Len : Ghdl_Index_Type; Rti : Ghdl_Rti_Access)       return Ghdl_Index_Type; + +   type Value_Status is +     ( +      Value_Ok, +      Value_Err_No_Digit,  -- After sign, at start, after exponent... +      Value_Err_Bad_Digit, +      Value_Err_Underscore, +      Value_Err_Bad_Base, +      Value_Err_Bad_End_Sign,  --  Missing or mismatch +      Value_Err_Bad_Exponent, +      Value_Err_Trailing_Chars +     ); + +   type Value_I64_Result (Status : Value_Status := Value_Ok) is record +      case Status is +         when Value_Ok => +            Val : Ghdl_I64; +         when others => +            Pos : Ghdl_Index_Type; +      end case; +   end record; + +   --  Convert S (INIT_POS .. LEN) to a signed integer. +   function Value_I64 (S : Std_String_Basep; +                       Len : Ghdl_Index_Type; +                       Init_Pos : Ghdl_Index_Type) return Value_I64_Result; + +   --  Likewise but report any error. +   function Value_I64 +     (S : Std_String_Basep; Len : Ghdl_Index_Type; Init_Pos : Ghdl_Index_Type) +     return Ghdl_I64;  private     pragma Export (Ada, Ghdl_Value_B1, "__ghdl_value_b1");     pragma Export (C, Ghdl_Value_E8, "__ghdl_value_e8"); diff --git a/src/vhdl/vhdl-evaluation.adb b/src/vhdl/vhdl-evaluation.adb index 36ca4c1af..57bc6f915 100644 --- a/src/vhdl/vhdl-evaluation.adb +++ b/src/vhdl/vhdl-evaluation.adb @@ -33,6 +33,7 @@ with Vhdl.Ieee.Std_Logic_1164;  with Grt.Types;  with Grt.Fcvt;  with Grt.To_Strings; +with Grt.Values;  package body Vhdl.Evaluation is     --  If FORCE is true, always return a literal. @@ -2774,7 +2775,22 @@ package body Vhdl.Evaluation is        begin           case Get_Kind (Base_Type) is              when Iir_Kind_Integer_Type_Definition => -               return Build_Discrete (Int64'Value (Value1), Orig); +               declare +                  use Grt.Values; +                  use Grt.Types; +                  Res : Value_I64_Result; +               begin +                  Res := Value_I64 (To_Std_String_Basep (Value1'Address), +                                    Value1'Length, 0); +                  if Res.Status = Value_Ok then +                     return Build_Discrete (Int64 (Res.Val), Orig); +                  else +                     Warning_Msg_Sem +                       (Warnid_Runtime_Error, +Get_Parameter (Orig), +                        "incorrect parameter for value attribute"); +                     return Build_Overflow (Orig); +                  end if; +               end;              when Iir_Kind_Enumeration_Type_Definition =>                 return Build_Enumeration_Value (Value1, Base_Type, Orig);              when Iir_Kind_Floating_Type_Definition => @@ -2784,11 +2800,6 @@ package body Vhdl.Evaluation is              when others =>                 Error_Kind ("eval_value_attribute", Base_Type);           end case; -      exception -         when Constraint_Error => -            Warning_Msg_Sem (Warnid_Runtime_Error, +Orig, -                             "incorrect parameter for value attribute"); -            return Build_Overflow (Orig);        end;     end Eval_Value_Attribute; | 
