diff options
-rw-r--r-- | disp_tree.adb | 4 | ||||
-rw-r--r-- | iirs.adb | 18 | ||||
-rw-r--r-- | iirs.ads | 13 | ||||
-rw-r--r-- | sem_expr.adb | 4 | ||||
-rw-r--r-- | translate/translation.adb | 163 |
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 => @@ -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 @@ -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; |