diff options
| author | Tristan Gingold <tgingold@free.fr> | 2021-07-28 22:20:40 +0200 | 
|---|---|---|
| committer | Tristan Gingold <tgingold@free.fr> | 2021-07-28 22:22:10 +0200 | 
| commit | d86f8929f7d49f02c304d3e95d29eae760657f9a (patch) | |
| tree | 9e77941b4bec053f92f24727df1ad2602a314909 | |
| parent | f9036aba4d5d18cf2cf4893589e8a41ea9b064b7 (diff) | |
| download | ghdl-d86f8929f7d49f02c304d3e95d29eae760657f9a.tar.gz ghdl-d86f8929f7d49f02c304d3e95d29eae760657f9a.tar.bz2 ghdl-d86f8929f7d49f02c304d3e95d29eae760657f9a.zip | |
vhdl-sem_expr: analyze choices before expressions in array aggregate.
Avoid considering expression to be possibly of the type of the aggregate
if the choice is an expression.
| -rw-r--r-- | src/vhdl/vhdl-sem_expr.adb | 61 | 
1 files changed, 39 insertions, 22 deletions
| diff --git a/src/vhdl/vhdl-sem_expr.adb b/src/vhdl/vhdl-sem_expr.adb index e8ba7dea6..286397410 100644 --- a/src/vhdl/vhdl-sem_expr.adb +++ b/src/vhdl/vhdl-sem_expr.adb @@ -3407,7 +3407,14 @@ package body Vhdl.Sem_Expr is              --  it is ambiguous.  But there is no point in using aggregates              --  to specify a range of choices.              --  FIXME: fix LRM ? + +            --  LRM08 9.3.3.3 Array aggregates +            --  If the type of the expression of an element association is the +            --  type of the aggregate, then either the element association +            --  shall be positional or the choice shall be a discrete range.              if Elements_Types = Null_Iir +              or else not Kind_In (El, Iir_Kind_Choice_By_None, +                                   Iir_Kind_Choice_By_Range)                or else Get_Kind (El_Expr) = Iir_Kind_Aggregate              then                 Expr := Sem_Expression (El_Expr, Element_Type); @@ -3566,6 +3573,27 @@ package body Vhdl.Sem_Expr is        Info : Array_Aggr_Info renames Infos (Dim);     begin +      if Get_Kind (Aggr) = Iir_Kind_Aggregate then +         Assoc_Chain := Get_Association_Choices_Chain (Aggr); +         Sem_Choices_Range (Assoc_Chain, Index_Type, Low, High, +                            Get_Location (Aggr), not Constrained, False); +         Set_Association_Choices_Chain (Aggr, Assoc_Chain); + +         --  Update infos. +         if Low /= Null_Iir +           and then (Info.Low = Null_Iir +                       or else Eval_Pos (Low) < Eval_Pos (Info.Low)) +         then +            Info.Low := Low; +         end if; +         if High /= Null_Iir +           and then (Info.High = Null_Iir +                       or else Eval_Pos (High) > Eval_Pos (Info.High)) +         then +            Info.High := High; +         end if; +      end if; +        --  Analyze aggregate elements.        if Constrained then           Expr_Staticness := Get_Type_Staticness (Index_Type); @@ -3637,25 +3665,6 @@ package body Vhdl.Sem_Expr is        Len_Staticness := Locally;        case Get_Kind (Aggr) is           when Iir_Kind_Aggregate => -            Assoc_Chain := Get_Association_Choices_Chain (Aggr); -            Sem_Choices_Range (Assoc_Chain, Index_Type, Low, High, -                               Get_Location (Aggr), not Constrained, False); -            Set_Association_Choices_Chain (Aggr, Assoc_Chain); - -            --  Update infos. -            if Low /= Null_Iir -              and then (Info.Low = Null_Iir -                        or else Eval_Pos (Low) < Eval_Pos (Info.Low)) -            then -               Info.Low := Low; -            end if; -            if High /= Null_Iir -              and then (Info.High = Null_Iir -                        or else Eval_Pos (High) > Eval_Pos (Info.High)) -            then -               Info.High := High; -            end if; -              --  Determine if the aggregate is positionnal or named;              --    and compute choice staticness.              Is_Positional := Unknown; @@ -3669,9 +3678,8 @@ package body Vhdl.Sem_Expr is                    when Iir_Kind_Choice_By_Range                      | Iir_Kind_Choice_By_Expression =>                       Is_Positional := False; -                     Choice_Staticness := -                       Nodes.Min (Choice_Staticness, -                                  Get_Choice_Staticness (Choice)); +                     Choice_Staticness := Min (Choice_Staticness, +                                               Get_Choice_Staticness (Choice));                       --  FIXME: not true for range.                       Len := Len + 1;                    when Iir_Kind_Choice_By_None => @@ -3777,6 +3785,15 @@ package body Vhdl.Sem_Expr is              then                 Info.Index_Subtype := Create_Range_Subtype_By_Length                   (Index_Type, Int64 (Len), Get_Location (Aggr)); + +               --  In vhdl08 and later, the number of elements may also depend +               --  from associated expressions. +               if Vhdl_Std >= Vhdl_08 +                 and then Get_Index_Constraint_Flag (A_Type) +                 and then Eval_Discrete_Type_Length (Index_Type) /= Int64 (Len) +               then +                  Error_Msg_Sem (+Aggr, "incorrect number of elements"); +               end if;              end if;           else              --  Create an index subtype. | 
