aboutsummaryrefslogtreecommitdiffstats
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/vhdl/evaluation.adb137
-rw-r--r--src/vhdl/iirs_utils.adb9
-rw-r--r--src/vhdl/iirs_utils.ads4
-rw-r--r--src/vhdl/sem_expr.adb7
4 files changed, 152 insertions, 5 deletions
diff --git a/src/vhdl/evaluation.adb b/src/vhdl/evaluation.adb
index 4c1386a0f..e17246abf 100644
--- a/src/vhdl/evaluation.adb
+++ b/src/vhdl/evaluation.adb
@@ -2418,9 +2418,8 @@ package body Evaluation is
when Iir_Kind_Attribute_Name =>
-- An attribute name designates an attribute value.
declare
- Attr_Val : constant Iir := Get_Named_Entity (Expr);
Attr_Expr : constant Iir :=
- Get_Expression (Get_Attribute_Specification (Attr_Val));
+ Get_Attribute_Name_Expression (Expr);
Val : Iir;
begin
Val := Eval_Static_Expr (Attr_Expr);
@@ -2701,11 +2700,141 @@ package body Evaluation is
end if;
end Eval_Expr;
+ -- Subroutine of Can_Eval_Composite_Value. Return True iff EXPR is
+ -- considered as a small composite.
+ function Is_Small_Composite_Value (Expr : Iir) return Boolean
+ is
+ Expr_Type : constant Iir := Get_Type (Expr);
+ Indexes : Iir_List;
+ Len : Iir_Int64;
+ begin
+ -- Consider only arrays. Records are never composite.
+ if Get_Kind (Expr_Type) /= Iir_Kind_Array_Subtype_Definition then
+ return False;
+ end if;
+
+ -- Element must be scalar.
+ if Get_Kind (Get_Element_Subtype (Expr_Type)) not in
+ Iir_Kinds_Scalar_Type_Definition
+ then
+ return False;
+ end if;
+
+ Indexes := Get_Index_Subtype_List (Expr_Type);
+
+ -- Multi-dimensional arrays aren't considered as small.
+ if Get_Nbr_Elements (Indexes) /= 1 then
+ return False;
+ end if;
+
+ Len := Eval_Discrete_Type_Length (Get_Nth_Element (Indexes, 0));
+ return Len <= 128;
+ end Is_Small_Composite_Value;
+
+ function Can_Eval_Composite_Value (Expr : Iir; Top : Boolean := False)
+ return Boolean;
+
+ -- Return True if EXPR should be evaluated.
+ function Can_Eval_Value (Expr : Iir; Top : Boolean) return Boolean is
+ begin
+ -- Always evaluate scalar values.
+ if Get_Kind (Get_Type (Expr)) in Iir_Kinds_Scalar_Type_Definition then
+ return True;
+ end if;
+ return Can_Eval_Composite_Value (Expr, Top);
+ end Can_Eval_Value;
+
+ -- For composite values.
+ -- Evluating a composite value is a trade-off: it can simplify the
+ -- generated code if the value is small enough, or it can be a bad idea if
+ -- the value is very large. It is very easy to create large static
+ -- composite values (like: bit_vector'(1 to 10**4 => '0'))
+ function Can_Eval_Composite_Value (Expr : Iir; Top : Boolean := False)
+ return Boolean
+ is
+ -- We are only considering static values.
+ pragma Assert (Get_Expr_Staticness (Expr) = Locally);
+
+ -- We are only considering composite types.
+ pragma Assert (Get_Kind (Get_Type (Expr))
+ not in Iir_Kinds_Scalar_Type_Definition);
+ begin
+ case Get_Kind (Expr) is
+ when Iir_Kind_Type_Conversion
+ | Iir_Kind_Qualified_Expression =>
+ return Can_Eval_Composite_Value (Get_Expression (Expr));
+ when Iir_Kinds_Denoting_Name =>
+ return Can_Eval_Composite_Value (Get_Named_Entity (Expr), Top);
+ when Iir_Kind_Constant_Declaration =>
+ -- Pass through names only for small values.
+ if Top or else not Is_Small_Composite_Value (Expr) then
+ return False;
+ else
+ return Can_Eval_Composite_Value (Get_Default_Value (Expr));
+ end if;
+ when Iir_Kind_Attribute_Name =>
+ if Top or else not Is_Small_Composite_Value (Expr) then
+ return False;
+ else
+ return Can_Eval_Composite_Value
+ (Get_Attribute_Name_Expression (Expr));
+ end if;
+ when Iir_Kinds_Dyadic_Operator =>
+ -- Concatenation can increase the size.
+ -- Others (rol, ror...) don't.
+ return Can_Eval_Value (Get_Left (Expr), False)
+ and then Can_Eval_Value (Get_Right (Expr), False);
+ when Iir_Kinds_Monadic_Operator =>
+ -- For not.
+ return Can_Eval_Composite_Value (Get_Operand (Expr));
+ when Iir_Kind_Aggregate =>
+ return Is_Small_Composite_Value (Expr);
+ when Iir_Kinds_Literal
+ | Iir_Kind_Enumeration_Literal
+ | Iir_Kind_Simple_Aggregate
+ | Iir_Kind_Image_Attribute
+ | Iir_Kind_Simple_Name_Attribute =>
+ return True;
+ when Iir_Kind_Overflow_Literal =>
+ return True;
+ when Iir_Kind_Function_Call =>
+ -- Either using post-fixed notation or implicit functions like
+ -- to_string.
+ -- Cannot be a user function (won't be locally static).
+ declare
+ Assoc : Iir;
+ Assoc_Expr : Iir;
+ begin
+ Assoc := Get_Parameter_Association_Chain (Expr);
+ while Is_Valid (Assoc) loop
+ case Iir_Kinds_Association_Element (Get_Kind (Assoc)) is
+ when Iir_Kind_Association_Element_By_Expression =>
+ Assoc_Expr := Get_Actual (Assoc);
+ if not Can_Eval_Value (Assoc_Expr, False) then
+ return False;
+ end if;
+ when Iir_Kind_Association_Element_Open =>
+ null;
+ when Iir_Kind_Association_Element_By_Individual =>
+ return False;
+ end case;
+ Assoc := Get_Chain (Assoc);
+ end loop;
+ return True;
+ end;
+
+ when others =>
+ -- Be safe, don't crash on unhandled expression.
+ -- Error_Kind ("can_eval_composite_value", Expr);
+ return False;
+ end case;
+ end Can_Eval_Composite_Value;
+
function Eval_Expr_If_Static (Expr : Iir) return Iir is
begin
if Expr /= Null_Iir and then Get_Expr_Staticness (Expr) = Locally then
- -- Evaluate only scalar expressions.
- if Get_Kind (Get_Type (Expr)) in Iir_Kinds_Scalar_Type_Definition then
+ -- Evaluate only when there is a positive effect.
+ if Can_Eval_Value (Expr, True) then
return Eval_Expr_Keep_Orig (Expr, False);
else
return Expr;
diff --git a/src/vhdl/iirs_utils.adb b/src/vhdl/iirs_utils.adb
index 4664f8dfa..5d407a3e6 100644
--- a/src/vhdl/iirs_utils.adb
+++ b/src/vhdl/iirs_utils.adb
@@ -1278,6 +1278,15 @@ package body Iirs_Utils is
end case;
end Is_Entity_Instantiation;
+ function Get_Attribute_Name_Expression (Name : Iir) return Iir
+ is
+ Attr_Val : constant Iir := Get_Named_Entity (Name);
+ Attr_Spec : constant Iir := Get_Attribute_Specification (Attr_Val);
+ Attr_Expr : constant Iir := Get_Expression (Attr_Spec);
+ begin
+ return Attr_Expr;
+ end Get_Attribute_Name_Expression;
+
function Get_String_Type_Bound_Type (Sub_Type : Iir) return Iir is
begin
if Get_Kind (Sub_Type) /= Iir_Kind_Array_Subtype_Definition then
diff --git a/src/vhdl/iirs_utils.ads b/src/vhdl/iirs_utils.ads
index dace93c3c..39d56a8ff 100644
--- a/src/vhdl/iirs_utils.ads
+++ b/src/vhdl/iirs_utils.ads
@@ -267,6 +267,10 @@ package Iirs_Utils is
(Inst : Iir_Component_Instantiation_Statement)
return Boolean;
+ -- Get the expression of the attribute specification corresponding to the
+ -- attribute name NAME. Meaningful only for static values.
+ function Get_Attribute_Name_Expression (Name : Iir) return Iir;
+
-- Return the bound type of a string type, ie the type of the (first)
-- dimension of a one-dimensional array type.
function Get_String_Type_Bound_Type (Sub_Type : Iir) return Iir;
diff --git a/src/vhdl/sem_expr.adb b/src/vhdl/sem_expr.adb
index 63cd7de6c..6b7624358 100644
--- a/src/vhdl/sem_expr.adb
+++ b/src/vhdl/sem_expr.adb
@@ -3338,7 +3338,12 @@ package body Sem_Expr is
end if;
-- Analyze aggregate elements.
- Expr_Staticness := Locally;
+ if Constrained then
+ Expr_Staticness := Get_Type_Staticness (Index_Type);
+ else
+ Expr_Staticness := Locally;
+ end if;
+
if Dim = Get_Nbr_Elements (Index_List) then
-- A type has been found for AGGR, analyze AGGR as if it was
-- an aggregate with a subtype (and not a string).