aboutsummaryrefslogtreecommitdiffstats
path: root/src
diff options
context:
space:
mode:
authorTristan Gingold <tgingold@free.fr>2022-06-05 09:09:26 +0200
committerTristan Gingold <tgingold@free.fr>2022-06-05 09:09:26 +0200
commit315c1969f53125ce2e396285209aac085f84d7e6 (patch)
treea21c2368a23a888ebff8020591b2a10018f3def8 /src
parent70396b5ccab87e0d4d329b39602090c5db9ce0be (diff)
downloadghdl-315c1969f53125ce2e396285209aac085f84d7e6.tar.gz
ghdl-315c1969f53125ce2e396285209aac085f84d7e6.tar.bz2
ghdl-315c1969f53125ce2e396285209aac085f84d7e6.zip
synth-vhdl_eval: handle find_leftmost and find_rightmost
Diffstat (limited to 'src')
-rw-r--r--src/synth/synth-ieee-numeric_std.adb39
-rw-r--r--src/synth/synth-ieee-numeric_std.ads3
-rw-r--r--src/synth/synth-vhdl_eval.adb13
3 files changed, 55 insertions, 0 deletions
diff --git a/src/synth/synth-ieee-numeric_std.adb b/src/synth/synth-ieee-numeric_std.adb
index ad41757fc..0e1b4be0e 100644
--- a/src/synth/synth-ieee-numeric_std.adb
+++ b/src/synth/synth-ieee-numeric_std.adb
@@ -1495,4 +1495,43 @@ package body Synth.Ieee.Numeric_Std is
return Res;
end Minmax;
+ function Offset_To_Index (Off : Int32; Typ : Type_Acc) return Int32 is
+ begin
+ case Typ.Abound.Dir is
+ when Dir_To =>
+ return Typ.Abound.Left + Off;
+ when Dir_Downto =>
+ return Typ.Abound.Left - Off;
+ end case;
+ end Offset_To_Index;
+
+ function Find_Rightmost (Arg : Memtyp; Val : Memtyp) return Int32
+ is
+ Len : constant Uns32 := Arg.Typ.Abound.Len;
+ Y : Std_Ulogic;
+ begin
+ Y := Read_Std_Logic (Val.Mem, 0);
+
+ for I in reverse 1 .. Len loop
+ if Match_Eq_Table (Read_Std_Logic (Arg.Mem, I - 1), Y) = '1' then
+ return Offset_To_Index (Int32 (I - 1), Arg.Typ);
+ end if;
+ end loop;
+ return -1;
+ end Find_Rightmost;
+
+ function Find_Leftmost (Arg : Memtyp; Val : Memtyp) return Int32
+ is
+ Len : constant Uns32 := Arg.Typ.Abound.Len;
+ Y : Std_Ulogic;
+ begin
+ Y := Read_Std_Logic (Val.Mem, 0);
+
+ for I in 1 .. Len loop
+ if Match_Eq_Table (Read_Std_Logic (Arg.Mem, I - 1), Y) = '1' then
+ return Offset_To_Index (Int32 (I - 1), Arg.Typ);
+ end if;
+ end loop;
+ return -1;
+ end Find_Leftmost;
end Synth.Ieee.Numeric_Std;
diff --git a/src/synth/synth-ieee-numeric_std.ads b/src/synth/synth-ieee-numeric_std.ads
index 22616986e..4c5888463 100644
--- a/src/synth/synth-ieee-numeric_std.ads
+++ b/src/synth/synth-ieee-numeric_std.ads
@@ -130,4 +130,7 @@ package Synth.Ieee.Numeric_Std is
function Minmax (L, R : Memtyp; Is_Signed : Boolean; Is_Max : Boolean)
return Memtyp;
+ -- Find_Rightmost/Find_Leftmost
+ function Find_Rightmost (Arg : Memtyp; Val : Memtyp) return Int32;
+ function Find_Leftmost (Arg : Memtyp; Val : Memtyp) return Int32;
end Synth.Ieee.Numeric_Std;
diff --git a/src/synth/synth-vhdl_eval.adb b/src/synth/synth-vhdl_eval.adb
index 6b4788556..f69fe6192 100644
--- a/src/synth/synth-vhdl_eval.adb
+++ b/src/synth/synth-vhdl_eval.adb
@@ -2082,6 +2082,19 @@ package body Synth.Vhdl_Eval is
return Minmax (Get_Memtyp (Param1), Get_Memtyp (Param2),
True, False);
+ when Iir_Predefined_Ieee_Numeric_Std_Find_Rightmost_Uns
+ | Iir_Predefined_Ieee_Numeric_Std_Find_Rightmost_Sgn =>
+ return Create_Memory_Discrete
+ (Int64 (Find_Rightmost (Get_Memtyp (Param1),
+ Get_Memtyp (Param2))),
+ Res_Typ);
+ when Iir_Predefined_Ieee_Numeric_Std_Find_Leftmost_Uns
+ | Iir_Predefined_Ieee_Numeric_Std_Find_Leftmost_Sgn =>
+ return Create_Memory_Discrete
+ (Int64 (Find_Leftmost (Get_Memtyp (Param1),
+ Get_Memtyp (Param2))),
+ Res_Typ);
+
when Iir_Predefined_Ieee_Math_Real_Log2 =>
declare
function Log2 (Arg : Fp64) return Fp64;