aboutsummaryrefslogtreecommitdiffstats
path: root/src/synth/elab-vhdl_expr.adb
diff options
context:
space:
mode:
authorTristan Gingold <tgingold@free.fr>2023-01-06 21:05:13 +0100
committerTristan Gingold <tgingold@free.fr>2023-01-06 21:05:13 +0100
commit8c8d9a8bf91d22d4cb7c350a016e0cdcfdae2ef9 (patch)
tree69fe1c4e5f9d5a16da49fa6dbc813379f97d2823 /src/synth/elab-vhdl_expr.adb
parenta0cc0a8059b97339c158a87937461676fcb87dae (diff)
downloadghdl-8c8d9a8bf91d22d4cb7c350a016e0cdcfdae2ef9.tar.gz
ghdl-8c8d9a8bf91d22d4cb7c350a016e0cdcfdae2ef9.tar.bz2
ghdl-8c8d9a8bf91d22d4cb7c350a016e0cdcfdae2ef9.zip
synth: handle value attribute for physical types
Diffstat (limited to 'src/synth/elab-vhdl_expr.adb')
-rw-r--r--src/synth/elab-vhdl_expr.adb73
1 files changed, 73 insertions, 0 deletions
diff --git a/src/synth/elab-vhdl_expr.adb b/src/synth/elab-vhdl_expr.adb
index 1119f6ae9..c32601ef1 100644
--- a/src/synth/elab-vhdl_expr.adb
+++ b/src/synth/elab-vhdl_expr.adb
@@ -38,7 +38,9 @@ with Synth.Errors; use Synth.Errors;
with Grt.Types;
with Grt.Vhdl_Types;
+with Grt.Strings;
with Grt.To_Strings;
+with Grt.Values;
with Grt.Vstrings;
package body Elab.Vhdl_Expr is
@@ -99,6 +101,24 @@ package body Elab.Vhdl_Expr is
return Synth_Subtype_Conversion (null, Vt, Dtype, Bounds, Loc);
end Exec_Subtype_Conversion;
+ -- Return True iff ID = S, case insensitive.
+ function Match_Id (Id : Name_Id; S : String) return Boolean is
+ begin
+ if Name_Table.Get_Name_Length (Id) /= S'Length then
+ return False;
+ end if;
+ declare
+ Img : constant String (S'Range) := Name_Table.Image (Id);
+ begin
+ for I in Img'Range loop
+ if Grt.Strings.To_Lower (S (I)) /= Img (I) then
+ return False;
+ end if;
+ end loop;
+ return True;
+ end;
+ end Match_Id;
+
function Exec_Value_Attribute (Syn_Inst : Synth_Instance_Acc; Attr : Node)
return Valtyp
is
@@ -163,6 +183,59 @@ package body Elab.Vhdl_Expr is
return No_Valtyp;
end if;
end;
+ when Iir_Kind_Physical_Type_Definition =>
+ declare
+ use Grt.Types;
+ use Grt.Vhdl_Types;
+ Value1 : String renames Value (First .. Last);
+ F : constant Ghdl_Index_Type := Ghdl_Index_Type (First);
+ S : constant Std_String_Basep :=
+ To_Std_String_Basep (Value1'Address);
+ Len : constant Ghdl_Index_Type :=
+ Ghdl_Index_Type (Last - First + 1);
+ Is_Real : Boolean;
+ Lit_Pos : Ghdl_Index_Type;
+ Lit_End : Ghdl_Index_Type;
+ Unit_Pos : Ghdl_Index_Type;
+ Unit_F, Unit_L : Positive;
+ Mult : Int64;
+ Unit : Iir;
+ Unit_Id : Name_Id;
+ Val_F : Grt.To_Strings.Value_F64_Result;
+ begin
+ Grt.Values.Ghdl_Value_Physical_Split
+ (S, Len, Is_Real, Lit_Pos, Lit_End, Unit_Pos);
+ Unit_F := Positive (Unit_Pos + 1);
+
+ -- Find unit.
+ Unit_L := Unit_F;
+ for I in Unit_F .. Last loop
+ exit when Grt.Strings.Is_Whitespace (Value1 (I));
+ Unit_L := I;
+ end loop;
+
+ Unit := Get_Primary_Unit (Btype);
+ while Unit /= Null_Iir loop
+ Unit_Id := Get_Identifier (Unit);
+ exit when Match_Id (Unit_Id, Value1 (Unit_F .. Unit_L));
+ Unit := Get_Chain (Unit);
+ end loop;
+
+ if Unit = Null_Iir then
+ Error_Msg_Synth (Syn_Inst, Attr, "incorrect unit name");
+ return No_Valtyp;
+ end if;
+ Mult := Get_Value (Get_Physical_Literal (Unit));
+
+ if Is_Real then
+ Val_F := Grt.To_Strings.Value_F64 (S, Lit_End, F);
+ Val := Int64 (Val_F.Val * Ghdl_F64 (Mult));
+ else
+ Val := Int64 (Grt.Values.Value_I64 (S, Lit_End, F))
+ * Mult;
+ end if;
+ end;
+
when others =>
Error_Msg_Elab (+Attr, "unhandled type for 'value");
return No_Valtyp;