aboutsummaryrefslogtreecommitdiffstats
path: root/src/vhdl
diff options
context:
space:
mode:
authorTristan Gingold <tgingold@free.fr>2022-07-07 22:33:42 +0200
committerTristan Gingold <tgingold@free.fr>2022-07-07 22:33:42 +0200
commit1586fadd37445cb5595867f3ed73c428dc2b7f68 (patch)
treec7f6cd58ff7a733402176627fb964b66a8d40be8 /src/vhdl
parentfe420f2cd66a7769abdbdc804bb228b0105edb37 (diff)
downloadghdl-1586fadd37445cb5595867f3ed73c428dc2b7f68.tar.gz
ghdl-1586fadd37445cb5595867f3ed73c428dc2b7f68.tar.bz2
ghdl-1586fadd37445cb5595867f3ed73c428dc2b7f68.zip
vhdl-evaluation: explicitly compute integer_exp to handle overflow.
Tentatively fix crash on mingw32
Diffstat (limited to 'src/vhdl')
-rw-r--r--src/vhdl/vhdl-evaluation.adb33
1 files changed, 31 insertions, 2 deletions
diff --git a/src/vhdl/vhdl-evaluation.adb b/src/vhdl/vhdl-evaluation.adb
index 653cbdb31..935f6e8ed 100644
--- a/src/vhdl/vhdl-evaluation.adb
+++ b/src/vhdl/vhdl-evaluation.adb
@@ -2167,8 +2167,37 @@ package body Vhdl.Evaluation is
return Build_Overflow (Orig);
end if;
when Iir_Predefined_Integer_Exp =>
- return Build_Integer_Check
- (Get_Value (Left) ** Integer (Get_Value (Right)), Orig);
+ declare
+ Exp : Int64;
+ Val : Int64;
+ Res : Int64;
+ begin
+ Val := Get_Value (Left);
+ -- LRM08 9.2.8 Misellaneous operators
+ -- Exponentiation with a negative exponent is only allowed for
+ -- a list operand of a floating-point type.
+ Exp := Get_Value (Right);
+ if Exp < 0 then
+ raise Constraint_Error;
+ end if;
+ -- LRM08 9.2.8 Misellaneous operators
+ -- Exponentiation with an integer exponent is equivalent to
+ -- repeated multiplication of the left operand by itself for
+ -- a number of times indicated by the absolute value of the
+ -- exponent and from left to right; [...]
+ -- GHDL: use the standard power-of-2 approach. This is not
+ -- strictly equivalent however.
+ Res := 1;
+ loop
+ if Exp mod 2 = 1 then
+ Res := Res * Val;
+ end if;
+ Exp := Exp / 2;
+ exit when Exp = 0;
+ Val := Val * Val;
+ end loop;
+ return Build_Integer_Check (Res, Orig);
+ end;
when Iir_Predefined_Integer_Equality =>
return Build_Boolean (Get_Value (Left) = Get_Value (Right));