summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--disp_tree.adb4
-rw-r--r--iirs.adb18
-rw-r--r--iirs.ads13
-rw-r--r--sem_expr.adb4
-rw-r--r--translate/translation.adb163
5 files changed, 106 insertions, 96 deletions
diff --git a/disp_tree.adb b/disp_tree.adb
index 0656aa9..8c3ef62 100644
--- a/disp_tree.adb
+++ b/disp_tree.adb
@@ -1737,8 +1737,8 @@ package body Disp_Tree is
Disp_Tree (Get_Aggr_Low_Limit (Tree), Ntab, False);
Header ("aggr_high_limit:");
Disp_Tree (Get_Aggr_High_Limit (Tree), Ntab, False);
- Header ("aggr_max_length:" &
- Iir_Int32'Image (Get_Aggr_Max_Length (Tree)));
+ Header ("aggr_min_length:" &
+ Iir_Int32'Image (Get_Aggr_Min_Length (Tree)));
Header ("sub_aggregate_info:");
Disp_Tree (Get_Sub_Aggregate_Info (Tree), Ntab);
when Iir_Kind_Operator_Symbol =>
diff --git a/iirs.adb b/iirs.adb
index 707a09f..34af7f9 100644
--- a/iirs.adb
+++ b/iirs.adb
@@ -6399,29 +6399,29 @@ package body Iirs is
Set_Flag3 (Target, Val);
end Set_Aggr_Dynamic_Flag;
- procedure Check_Kind_For_Aggr_Max_Length (Target : Iir) is
+ procedure Check_Kind_For_Aggr_Min_Length (Target : Iir) is
begin
case Get_Kind (Target) is
when Iir_Kind_Aggregate_Info =>
null;
when others =>
- Failed ("Aggr_Max_Length", Target);
+ Failed ("Aggr_Min_Length", Target);
end case;
- end Check_Kind_For_Aggr_Max_Length;
+ end Check_Kind_For_Aggr_Min_Length;
- function Get_Aggr_Max_Length (Info : Iir_Aggregate_Info) return Iir_Int32
+ function Get_Aggr_Min_Length (Info : Iir_Aggregate_Info) return Iir_Int32
is
begin
- Check_Kind_For_Aggr_Max_Length (Info);
+ Check_Kind_For_Aggr_Min_Length (Info);
return Iir_To_Iir_Int32 (Get_Field4 (Info));
- end Get_Aggr_Max_Length;
+ end Get_Aggr_Min_Length;
- procedure Set_Aggr_Max_Length (Info : Iir_Aggregate_Info; Nbr : Iir_Int32)
+ procedure Set_Aggr_Min_Length (Info : Iir_Aggregate_Info; Nbr : Iir_Int32)
is
begin
- Check_Kind_For_Aggr_Max_Length (Info);
+ Check_Kind_For_Aggr_Min_Length (Info);
Set_Field4 (Info, Iir_Int32_To_Iir (Nbr));
- end Set_Aggr_Max_Length;
+ end Set_Aggr_Min_Length;
procedure Check_Kind_For_Aggr_Low_Limit (Target : Iir) is
begin
diff --git a/iirs.ads b/iirs.ads
index 9e940b5..ab46a39 100644
--- a/iirs.ads
+++ b/iirs.ads
@@ -684,6 +684,7 @@ package Iirs is
--
-- Get/Set_Identifier (Field3)
--
+ -- The corresponding package declaration.
-- Get/Set_Package (Field4)
-- Iir_Kind_Library_Declaration (Medium)
@@ -2384,7 +2385,7 @@ package Iirs is
-- Get/Set_Sub_Aggregate_Info (Field1)
--
-- For array aggregate only:
- -- If TRUE, the aggregate bounds are not locally static.
+ -- If TRUE, the choices are not locally static.
-- This flag is only valid when the array aggregate is constrained, ie
-- has no 'others' choice.
-- Get/Set_Aggr_Dynamic_Flag (Flag3)
@@ -2405,8 +2406,8 @@ package Iirs is
--
-- Get/Set_Aggr_High_Limit (Field3)
--
- -- The maximum number of elements, if any.
- -- Get/Set_Aggr_Max_Length (Field4)
+ -- The minimum number of elements, if any. This is a minimax.
+ -- Get/Set_Aggr_Min_Length (Field4)
--
-- True if the choice list has an 'others' choice.
-- Get/Set_Aggr_Others_Flag (Flag2)
@@ -5248,13 +5249,13 @@ package Iirs is
function Get_Aggr_Dynamic_Flag (Target : Iir) return Boolean;
procedure Set_Aggr_Dynamic_Flag (Target : Iir; Val : Boolean);
- -- Get/Set the maximum number of elements for the lowest dimension of
+ -- Get/Set the minimum number of elements for the lowest dimension of
-- the aggregate or for the current dimension of a sub-aggregate.
-- The real number of elements may be greater than this number if there
-- is an 'other' choice.
-- Field: Field4 (uc)
- function Get_Aggr_Max_Length (Info : Iir_Aggregate_Info) return Iir_Int32;
- procedure Set_Aggr_Max_Length (Info : Iir_Aggregate_Info; Nbr : Iir_Int32);
+ function Get_Aggr_Min_Length (Info : Iir_Aggregate_Info) return Iir_Int32;
+ procedure Set_Aggr_Min_Length (Info : Iir_Aggregate_Info; Nbr : Iir_Int32);
-- Highest index choice, if any.
-- Field: Field2
diff --git a/sem_expr.adb b/sem_expr.adb
index aec8a83..0814355 100644
--- a/sem_expr.adb
+++ b/sem_expr.adb
@@ -3335,8 +3335,6 @@ package body Sem_Expr is
Set_Index_Constraint_Flag (A_Subtype, True);
Set_Constraint_State (A_Subtype, Fully_Constrained);
Set_Type (Aggr, A_Subtype);
- else
- Set_Type (Aggr, Base_Type);
end if;
Prev_Info := Null_Iir;
@@ -3355,7 +3353,7 @@ package body Sem_Expr is
Set_Aggr_Named_Flag (Info, Infos (I).Has_Named);
Set_Aggr_Low_Limit (Info, Infos (I).Low);
Set_Aggr_High_Limit (Info, Infos (I).High);
- Set_Aggr_Max_Length (Info, Iir_Int32 (Infos (I).Min_Length));
+ Set_Aggr_Min_Length (Info, Iir_Int32 (Infos (I).Min_Length));
Set_Aggr_Others_Flag (Info, Infos (I).Has_Others);
end loop;
return Aggr;
diff --git a/translate/translation.adb b/translate/translation.adb
index 0d9e8bf..808cd3b 100644
--- a/translate/translation.adb
+++ b/translate/translation.adb
@@ -14028,31 +14028,29 @@ package body Translation is
is
use Name_Table;
- Lit_Type : Iir;
+ Lit_Type : constant Iir := Get_Type (Str);
+ Type_Info : constant Type_Info_Acc := Get_Info (Lit_Type);
Index_Type : Iir;
Bound_Aggr : O_Record_Aggr_List;
Index_Aggr : O_Record_Aggr_List;
Res_Aggr : O_Record_Aggr_List;
Res : O_Cnode;
- Type_Info : Type_Info_Acc;
Index_Type_Info : Type_Info_Acc;
Len : Int32;
Val : Var_Acc;
Bound : Var_Acc;
R : O_Enode;
begin
- Lit_Type := Get_Type (Str);
- Type_Info := Get_Info (Lit_Type);
-
-- Create the string value.
Len := Get_String_Length (Str);
Val := Create_String_Literal_Var (Str);
+ Index_Type :=
+ Get_First_Element (Get_Index_Subtype_List (Lit_Type));
+ Index_Type_Info := Get_Info (Index_Type);
+
if Type_Info.Type_Mode = Type_Mode_Fat_Array then
-- Create the string bound.
- Index_Type :=
- Get_First_Element (Get_Index_Subtype_List (Lit_Type));
- Index_Type_Info := Get_Info (Index_Type);
Start_Record_Aggr (Bound_Aggr, Type_Info.T.Bounds_Type);
Start_Record_Aggr (Index_Aggr, Index_Type_Info.T.Range_Type);
New_Record_Aggr_El
@@ -14093,7 +14091,15 @@ package body Translation is
(Create_Uniq_Identifier, Type_Info.Ortho_Type (Mode_Value),
O_Storage_Private, Res);
elsif Type_Info.Type_Mode = Type_Mode_Array then
- null;
+ -- Type of string literal isn't statically known; check the
+ -- length.
+ Chap6.Check_Bound_Error
+ (New_Compare_Op
+ (ON_Neq,
+ New_Lit (New_Index_Lit (Unsigned_64 (Len))),
+ Chap3.Get_Array_Type_Length (Lit_Type),
+ Ghdl_Bool_Type),
+ Str, 1);
else
raise Internal_Error;
end if;
@@ -15826,35 +15832,39 @@ package body Translation is
Var_Index : O_Dnode;
Targ : Mnode;
- Range_Ptr : Mnode;
Rinfo : Type_Info_Acc;
Bt : Iir;
- function Check_Value
- (Lval : Iir; Lop : ON_Op_Kind; Rval : Iir; Rop : ON_Op_Kind)
- return O_Enode
+ -- Generate code for: (LVAL lop RNG.left) or (RVAL rop RNG.right)
+ function Check_Value (Lval : Iir;
+ Lop : ON_Op_Kind;
+ Rval : Iir;
+ Rop : ON_Op_Kind;
+ Rng : Mnode)
+ return O_Enode
is
L, R : O_Enode;
begin
L := New_Compare_Op
(Lop,
New_Lit (Translate_Static_Expression (Lval, Bt)),
- M2E (Chap3.Range_To_Left (Range_Ptr)),
+ M2E (Chap3.Range_To_Left (Rng)),
Ghdl_Bool_Type);
R := New_Compare_Op
(Rop,
New_Lit (Translate_Static_Expression (Rval, Bt)),
- M2E (Chap3.Range_To_Right (Range_Ptr)),
+ M2E (Chap3.Range_To_Right (Rng)),
Ghdl_Bool_Type);
return New_Dyadic_Op (ON_Or, L, R);
end Check_Value;
+ Range_Ptr : Mnode;
Index_List : Iir_List;
Targ_Index_List : Iir_List;
Subtarg_Type : Iir;
Subaggr_Type : Iir;
L, H : Iir;
- Max : Iir_Int32;
+ Min : Iir_Int32;
Has_Others : Boolean;
Aggr_Info : Iir_Aggregate_Info;
@@ -15882,56 +15892,53 @@ package body Translation is
Bt := Get_Base_Type (Subaggr_Type);
Rinfo := Get_Info (Bt);
- if Get_Type_Staticness (Subaggr_Type) /= Locally then
- -- Aggregate has dynamic bounds.
- if Subaggr_Type /= Subtarg_Type then
- -- And it is not the same as the target.
- -- Must be checked.
-
- Open_Temp;
- declare
- A_Range : O_Dnode;
- Rng_Ptr : O_Dnode;
- begin
- -- Evaluate the range.
- Chap3.Translate_Anonymous_Type_Definition
- (Subaggr_Type, True);
-
- A_Range := Create_Temp (Rinfo.T.Range_Type);
- Rng_Ptr := Create_Temp_Ptr
- (Rinfo.T.Range_Ptr_Type, New_Obj (A_Range));
- Chap7.Translate_Range_Ptr
- (Rng_Ptr,
- Get_Range_Constraint (Subaggr_Type),
- Subaggr_Type);
-
- -- Check range length VS target length.
- Chap6.Check_Bound_Error
- (New_Compare_Op
- (ON_Neq,
- M2E (Chap3.Range_To_Length
- (Dv2M (A_Range,
- Rinfo,
- Mode_Value,
- Rinfo.T.Range_Type,
- Rinfo.T.Range_Ptr_Type))),
- M2E (Chap3.Range_To_Length
- (Chap3.Bounds_To_Range
- (Bounds, Target_Type, I + 1))),
- Ghdl_Bool_Type),
- Aggr, I);
- end;
- Close_Temp;
- end if;
- else
+ if Get_Aggr_Dynamic_Flag (Aggr_Info) then
+ -- Dynamic range, must evaluate it.
+ Open_Temp;
+ declare
+ A_Range : O_Dnode;
+ Rng_Ptr : O_Dnode;
+ begin
+ -- Evaluate the range.
+ Chap3.Translate_Anonymous_Type_Definition
+ (Subaggr_Type, True);
+
+ A_Range := Create_Temp (Rinfo.T.Range_Type);
+ Rng_Ptr := Create_Temp_Ptr
+ (Rinfo.T.Range_Ptr_Type, New_Obj (A_Range));
+ Chap7.Translate_Range_Ptr
+ (Rng_Ptr,
+ Get_Range_Constraint (Subaggr_Type),
+ Subaggr_Type);
+
+ -- Check range length VS target length.
+ Chap6.Check_Bound_Error
+ (New_Compare_Op
+ (ON_Neq,
+ M2E (Chap3.Range_To_Length
+ (Dv2M (A_Range,
+ Rinfo,
+ Mode_Value,
+ Rinfo.T.Range_Type,
+ Rinfo.T.Range_Ptr_Type))),
+ M2E (Chap3.Range_To_Length
+ (Chap3.Bounds_To_Range
+ (Bounds, Target_Type, I + 1))),
+ Ghdl_Bool_Type),
+ Aggr, I);
+ end;
+ Close_Temp;
+ elsif Get_Type_Staticness (Subaggr_Type) /= Locally
+ or else Subaggr_Type /= Subtarg_Type
+ then
-- Note: if the aggregate has no others, then the bounds
-- must be the same, otherwise, aggregate bounds must be
-- inside type bounds.
Has_Others := Get_Aggr_Others_Flag (Aggr_Info);
- Max := Get_Aggr_Max_Length (Aggr_Info);
+ Min := Get_Aggr_Min_Length (Aggr_Info);
L := Get_Aggr_Low_Limit (Aggr_Info);
- if Max > 0 or L /= Null_Iir then
+ if Min > 0 or L /= Null_Iir then
Open_Temp;
-- Pointer to the range.
@@ -15941,6 +15948,9 @@ package body Translation is
H := Get_Aggr_High_Limit (Aggr_Info);
if L /= Null_Iir then
+ -- Check the index range of the aggregrate is equal
+ -- (or within in presence of 'others') the index range
+ -- of the target.
Start_If_Stmt
(If_Blk,
New_Compare_Op (ON_Eq,
@@ -15948,26 +15958,30 @@ package body Translation is
New_Lit (Ghdl_Dir_To_Node),
Ghdl_Bool_Type));
if Has_Others then
- E := Check_Value (L, ON_Lt, H, ON_Gt);
+ E := Check_Value (L, ON_Lt, H, ON_Gt, Range_Ptr);
else
- E := Check_Value (L, ON_Neq, H, ON_Neq);
+ E := Check_Value (L, ON_Neq, H, ON_Neq, Range_Ptr);
end if;
New_Assign_Stmt (New_Obj (Var_Err), E);
New_Else_Stmt (If_Blk);
if Has_Others then
- E := Check_Value (H, ON_Gt, L, ON_Lt);
+ E := Check_Value (H, ON_Gt, L, ON_Lt, Range_Ptr);
else
- E := Check_Value (H, ON_Neq, L, ON_Neq);
+ E := Check_Value (H, ON_Neq, L, ON_Neq, Range_Ptr);
end if;
New_Assign_Stmt (New_Obj (Var_Err), E);
Finish_If_Stmt (If_Blk);
- -- If L and H are greather than the maximum length,
- -- then there is no need to check with max.
- if Iir_Int32 (Eval_Pos (H) - Eval_Pos (L) + 1) >= Max then
- Max := 0;
+ -- If L and H are greather than the minimum length,
+ -- then there is no need to check with min.
+ if Iir_Int32 (Eval_Pos (H) - Eval_Pos (L) + 1) >= Min then
+ Min := 0;
end if;
end if;
- if Max > 0 then
+
+ if Min > 0 then
+ -- Check the number of elements is equal (or less in
+ -- presence of 'others') than the length of the index
+ -- range of the target.
if Has_Others then
Op := ON_Lt;
else
@@ -15977,7 +15991,7 @@ package body Translation is
(Op,
M2E (Chap3.Range_To_Length (Range_Ptr)),
New_Lit (New_Unsigned_Literal (Ghdl_Index_Type,
- Unsigned_64 (Max))),
+ Unsigned_64 (Min))),
Ghdl_Bool_Type);
if L /= Null_Iir then
E := New_Dyadic_Op (ON_Or, E, New_Obj_Value (Var_Err));
@@ -19274,14 +19288,11 @@ package body Translation is
procedure Translate_Variable_Assignment_Statement
(Stmt : Iir_Variable_Assignment_Statement)
is
- Target : Iir;
- Targ_Type : Iir;
- Expr : Iir;
+ Target : constant Iir := Get_Target (Stmt);
+ Targ_Type : constant Iir := Get_Type (Target);
+ Expr : constant Iir := Get_Expression (Stmt);
Targ_Node : Mnode;
begin
- Target := Get_Target (Stmt);
- Targ_Type := Get_Type (Target);
- Expr := Get_Expression (Stmt);
if Get_Kind (Target) = Iir_Kind_Aggregate then
declare
E : O_Enode;