diff options
-rw-r--r-- | src/vhdl/elocations.adb | 26 | ||||
-rw-r--r-- | src/vhdl/elocations.ads | 30 | ||||
-rw-r--r-- | src/vhdl/elocations_meta.adb | 35 | ||||
-rw-r--r-- | src/vhdl/elocations_meta.ads | 2 | ||||
-rw-r--r-- | src/vhdl/parse.adb | 187 |
5 files changed, 231 insertions, 49 deletions
diff --git a/src/vhdl/elocations.adb b/src/vhdl/elocations.adb index a73ce1ca4..94b647c4b 100644 --- a/src/vhdl/elocations.adb +++ b/src/vhdl/elocations.adb @@ -225,7 +225,6 @@ package body Elocations is | Iir_Kind_Subtype_Definition | Iir_Kind_Scalar_Nature_Definition | Iir_Kind_Overload_List - | Iir_Kind_Anonymous_Type_Declaration | Iir_Kind_Nature_Declaration | Iir_Kind_Subnature_Declaration | Iir_Kind_Unit_Declaration @@ -302,12 +301,9 @@ package body Elocations is | Iir_Kind_Concurrent_Procedure_Call_Statement | Iir_Kind_Psl_Assert_Statement | Iir_Kind_Psl_Cover_Statement - | Iir_Kind_If_Generate_Statement | Iir_Kind_Case_Generate_Statement - | Iir_Kind_For_Generate_Statement | Iir_Kind_Psl_Default_Clock | Iir_Kind_Simple_Simultaneous_Statement - | Iir_Kind_If_Generate_Else_Clause | Iir_Kind_Simple_Signal_Assignment_Statement | Iir_Kind_Conditional_Signal_Assignment_Statement | Iir_Kind_Selected_Waveform_Assignment_Statement @@ -318,7 +314,6 @@ package body Elocations is | Iir_Kind_Variable_Assignment_Statement | Iir_Kind_Conditional_Variable_Assignment_Statement | Iir_Kind_Return_Statement - | Iir_Kind_While_Loop_Statement | Iir_Kind_Next_Statement | Iir_Kind_Exit_Statement | Iir_Kind_Procedure_Call_Statement @@ -379,6 +374,7 @@ package body Elocations is | Iir_Kind_Attribute_Name => return Format_None; when Iir_Kind_Library_Clause + | Iir_Kind_Anonymous_Type_Declaration | Iir_Kind_Attribute_Declaration | Iir_Kind_Group_Template_Declaration | Iir_Kind_Group_Declaration @@ -419,9 +415,13 @@ package body Elocations is | Iir_Kind_Sensitized_Process_Statement | Iir_Kind_Process_Statement | Iir_Kind_Block_Statement + | Iir_Kind_If_Generate_Statement + | Iir_Kind_For_Generate_Statement | Iir_Kind_Component_Instantiation_Statement | Iir_Kind_Generate_Statement_Body + | Iir_Kind_If_Generate_Else_Clause | Iir_Kind_For_Loop_Statement + | Iir_Kind_While_Loop_Statement | Iir_Kind_If_Statement | Iir_Kind_Elsif => return Format_L3; @@ -529,6 +529,22 @@ package body Elocations is Set_Field3 (N, Loc); end Set_Loop_Location; + function Get_Generate_Location (N : Iir) return Location_Type is + begin + pragma Assert (N /= Null_Iir); + pragma Assert (Has_Generate_Location (Get_Kind (N)), + "no field Generate_Location"); + return Get_Field3 (N); + end Get_Generate_Location; + + procedure Set_Generate_Location (N : Iir; Loc : Location_Type) is + begin + pragma Assert (N /= Null_Iir); + pragma Assert (Has_Generate_Location (Get_Kind (N)), + "no field Generate_Location"); + Set_Field3 (N, Loc); + end Set_Generate_Location; + function Get_Generic_Location (N : Iir) return Location_Type is begin pragma Assert (N /= Null_Iir); diff --git a/src/vhdl/elocations.ads b/src/vhdl/elocations.ads index 6c671794f..a6b728fa4 100644 --- a/src/vhdl/elocations.ads +++ b/src/vhdl/elocations.ads @@ -189,7 +189,9 @@ package Elocations is -- -- Get/Set_Start_Location (Field1) - -- Iir_Kind_Anonymous_Type_Declaration (None) + -- Iir_Kind_Anonymous_Type_Declaration (L1) + -- + -- Get/Set_Start_Location (Field1) -- Iir_Kind_Type_Declaration (L2) -- @@ -411,11 +413,22 @@ package Elocations is -- -- Get/Set_Begin_Location (Field3) - -- Iir_Kind_For_Generate_Statement (None) - - -- Iir_Kind_If_Generate_Else_Clause (None) + -- Iir_Kind_For_Generate_Statement (L3) + -- + -- Get/Set_Start_Location (Field1) + -- + -- Get/Set_End_Location (Field2) + -- + -- Get/Set_Generate_Location (Field3) - -- Iir_Kind_If_Generate_Statement (None) + -- Iir_Kind_If_Generate_Else_Clause (L3) + -- Iir_Kind_If_Generate_Statement (L3) + -- + -- Get/Set_Start_Location (Field1) + -- + -- Get/Set_End_Location (Field2) + -- + -- Get/Set_Generate_Location (Field3) -- Iir_Kind_Case_Generate_Statement (None) @@ -437,6 +450,7 @@ package Elocations is -- Get/Set_Then_Location (Field3) -- Iir_Kind_For_Loop_Statement (L3) + -- Iir_Kind_While_Loop_Statement (L3) -- -- Get/Set_Start_Location (Field1) -- @@ -444,8 +458,6 @@ package Elocations is -- -- Get/Set_Loop_Location (Field3) - -- Iir_Kind_While_Loop_Statement (None) - -- Iir_Kind_Exit_Statement (None) -- Iir_Kind_Next_Statement (None) @@ -629,6 +641,10 @@ package Elocations is function Get_Loop_Location (N : Iir) return Location_Type; procedure Set_Loop_Location (N : Iir; Loc : Location_Type); + -- Field: Field3 + function Get_Generate_Location (N : Iir) return Location_Type; + procedure Set_Generate_Location (N : Iir; Loc : Location_Type); + -- Field: Field4 function Get_Generic_Location (N : Iir) return Location_Type; procedure Set_Generic_Location (N : Iir; Loc : Location_Type); diff --git a/src/vhdl/elocations_meta.adb b/src/vhdl/elocations_meta.adb index b76e251fd..433edd32e 100644 --- a/src/vhdl/elocations_meta.adb +++ b/src/vhdl/elocations_meta.adb @@ -34,6 +34,8 @@ package body Elocations_Meta is return "then_location"; when Field_Loop_Location => return "loop_location"; + when Field_Generate_Location => + return "generate_location"; when Field_Generic_Location => return "generic_location"; when Field_Port_Location => @@ -73,6 +75,8 @@ package body Elocations_Meta is return Get_Then_Location (N); when Field_Loop_Location => return Get_Loop_Location (N); + when Field_Generate_Location => + return Get_Generate_Location (N); when Field_Generic_Location => return Get_Generic_Location (N); when Field_Port_Location => @@ -103,6 +107,8 @@ package body Elocations_Meta is Set_Then_Location (N, V); when Field_Loop_Location => Set_Loop_Location (N, V); + when Field_Generate_Location => + Set_Generate_Location (N, V); when Field_Generic_Location => Set_Generic_Location (N, V); when Field_Port_Location => @@ -124,6 +130,7 @@ package body Elocations_Meta is | Iir_Kind_Record_Type_Definition | Iir_Kind_Protected_Type_Body | Iir_Kind_Type_Declaration + | Iir_Kind_Anonymous_Type_Declaration | Iir_Kind_Subtype_Declaration | Iir_Kind_Package_Declaration | Iir_Kind_Package_Instantiation_Declaration @@ -158,8 +165,12 @@ package body Elocations_Meta is | Iir_Kind_Concurrent_Simple_Signal_Assignment | Iir_Kind_Concurrent_Conditional_Signal_Assignment | Iir_Kind_Concurrent_Selected_Signal_Assignment + | Iir_Kind_If_Generate_Statement + | Iir_Kind_For_Generate_Statement | Iir_Kind_Generate_Statement_Body + | Iir_Kind_If_Generate_Else_Clause | Iir_Kind_For_Loop_Statement + | Iir_Kind_While_Loop_Statement | Iir_Kind_If_Statement | Iir_Kind_Elsif => return True; @@ -187,8 +198,12 @@ package body Elocations_Meta is | Iir_Kind_Sensitized_Process_Statement | Iir_Kind_Process_Statement | Iir_Kind_Block_Statement + | Iir_Kind_If_Generate_Statement + | Iir_Kind_For_Generate_Statement | Iir_Kind_Generate_Statement_Body + | Iir_Kind_If_Generate_Else_Clause | Iir_Kind_For_Loop_Statement + | Iir_Kind_While_Loop_Statement | Iir_Kind_Case_Statement | Iir_Kind_If_Statement | Iir_Kind_Elsif => @@ -239,9 +254,27 @@ package body Elocations_Meta is function Has_Loop_Location (K : Iir_Kind) return Boolean is begin - return K = Iir_Kind_For_Loop_Statement; + case K is + when Iir_Kind_For_Loop_Statement + | Iir_Kind_While_Loop_Statement => + return True; + when others => + return False; + end case; end Has_Loop_Location; + function Has_Generate_Location (K : Iir_Kind) return Boolean is + begin + case K is + when Iir_Kind_If_Generate_Statement + | Iir_Kind_For_Generate_Statement + | Iir_Kind_If_Generate_Else_Clause => + return True; + when others => + return False; + end case; + end Has_Generate_Location; + function Has_Generic_Location (K : Iir_Kind) return Boolean is begin case K is diff --git a/src/vhdl/elocations_meta.ads b/src/vhdl/elocations_meta.ads index 4bde59559..34d461a77 100644 --- a/src/vhdl/elocations_meta.ads +++ b/src/vhdl/elocations_meta.ads @@ -29,6 +29,7 @@ package Elocations_Meta is Field_Begin_Location, Field_Then_Location, Field_Loop_Location, + Field_Generate_Location, Field_Generic_Location, Field_Port_Location, Field_Generic_Map_Location, @@ -52,6 +53,7 @@ package Elocations_Meta is function Has_Begin_Location (K : Iir_Kind) return Boolean; function Has_Then_Location (K : Iir_Kind) return Boolean; function Has_Loop_Location (K : Iir_Kind) return Boolean; + function Has_Generate_Location (K : Iir_Kind) return Boolean; function Has_Generic_Location (K : Iir_Kind) return Boolean; function Has_Port_Location (K : Iir_Kind) return Boolean; function Has_Generic_Map_Location (K : Iir_Kind) return Boolean; diff --git a/src/vhdl/parse.adb b/src/vhdl/parse.adb index a046c10aa..b842bd008 100644 --- a/src/vhdl/parse.adb +++ b/src/vhdl/parse.adb @@ -2353,9 +2353,11 @@ package body Parse is Loc : Location_Type; Ident : Name_Id; Decl : Iir; + Start_Loc : Location_Type; begin -- The current token must be type. pragma Assert (Current_Token = Tok_Type); + Start_Loc := Get_Token_Location; -- Get the identifier Scan_Expect (Tok_Identifier, @@ -2363,7 +2365,7 @@ package body Parse is Loc := Get_Token_Location; Ident := Current_Identifier; - -- Skip identifier + -- Skip identifier. Scan; if Current_Token = Tok_Semi_Colon then @@ -2372,6 +2374,12 @@ package body Parse is Decl := Create_Iir (Iir_Kind_Type_Declaration); Set_Identifier (Decl, Ident); Set_Location (Decl, Loc); + + if Flag_Elocations then + Create_Elocations (Decl); + Set_Start_Location (Decl, Start_Loc); + end if; + return Decl; end if; @@ -2491,6 +2499,12 @@ package body Parse is -- ';' is expected after end of type declaration Expect (Tok_Semi_Colon); Invalidate_Current_Token; + + if Flag_Elocations then + Create_Elocations (Decl); + Set_Start_Location (Decl, Start_Loc); + end if; + return Decl; end Parse_Type_Declaration; @@ -2766,10 +2780,14 @@ package body Parse is is Decl: Iir_Subtype_Declaration; Def: Iir; + Start_Loc : Location_Type; begin Decl := Create_Iir (Iir_Kind_Subtype_Declaration); + Start_Loc := Get_Token_Location; + -- Eat 'subtype'. Scan_Expect (Tok_Identifier); + Set_Identifier (Decl, Current_Identifier); Set_Parent (Decl, Parent); Set_Location (Decl); @@ -2784,6 +2802,12 @@ package body Parse is Set_Subtype_Indication (Decl, Def); Expect (Tok_Semi_Colon); + + if Flag_Elocations then + Create_Elocations (Decl); + Set_Start_Location (Decl, Start_Loc); + end if; + return Decl; end Parse_Subtype_Declaration; @@ -3243,10 +3267,12 @@ package body Parse is Kind: Iir_Kind; Shared : Boolean; Has_Mode : Boolean; + Start_Loc : Location_Type; begin Sub_Chain_Init (First, Last); - -- object keyword was just scanned. + -- Object keyword was just scanned. + Start_Loc := Get_Token_Location; case Current_Token is when Tok_Signal => Kind := Iir_Kind_Signal_Declaration; @@ -3266,19 +3292,28 @@ package body Parse is end case; loop - -- object or "," was just scanned. + -- Object or "," was just scanned. Object := Create_Iir (Kind); if Kind = Iir_Kind_Variable_Declaration then Set_Shared_Flag (Object, Shared); end if; + + -- Eat class or ','. Scan_Expect (Tok_Identifier); Set_Identifier (Object, Current_Identifier); Set_Location (Object); Set_Parent (Object, Parent); + if Flag_Elocations then + Create_Elocations (Object); + Set_Start_Location (Object, Start_Loc); + end if; + Sub_Chain_Append (First, Last, Object); + -- Eat identifier. Scan; + exit when Current_Token = Tok_Colon; if Current_Token /= Tok_Comma then case Current_Token is @@ -6409,6 +6444,60 @@ package body Parse is return Stmt; end Parse_For_Loop_Statement; + -- precond: WHILE or LOOP + -- postcond: ';' + -- + -- [ 8.9 ] + -- loop_statement ::= + -- [ LOOP_label : ] + -- [ iteration_scheme ] LOOP + -- sequence_of_statements + -- END LOOP [ LOOP_label ] ; + function Parse_While_Loop_Statement (Label : Name_Id) return Iir + is + Stmt : Iir; + Start_Loc, Loop_Loc, End_Loc : Location_Type; + begin + Stmt := Create_Iir (Iir_Kind_While_Loop_Statement); + Start_Loc := Get_Token_Location; + Set_Location (Stmt, Start_Loc); + Set_Label (Stmt, Label); + if Current_Token = Tok_While then + -- Skip 'while'. + Scan; + + Set_Condition (Stmt, Parse_Expression); + Expect (Tok_Loop); + end if; + + -- Skip 'loop'. + Loop_Loc := Get_Token_Location; + Scan; + + Set_Sequential_Statement_Chain + (Stmt, Parse_Sequential_Statements (Stmt)); + + End_Loc := Get_Token_Location; + Expect (Tok_End); + + -- Skip 'end'. + Scan_Expect (Tok_Loop); + + -- Skip 'loop'. + Scan; + + Check_End_Name (Stmt); + + if Flag_Elocations then + Create_Elocations (Stmt); + Set_Start_Location (Stmt, Start_Loc); + Set_Loop_Location (Stmt, Loop_Loc); + Set_End_Location (Stmt, End_Loc); + end if; + + return Stmt; + end Parse_While_Loop_Statement; + -- precond: next token -- postcond: next token -- @@ -6547,21 +6636,9 @@ package body Parse is when Tok_While | Tok_Loop => - Stmt := Create_Iir (Iir_Kind_While_Loop_Statement); - Set_Location (Stmt); - Set_Label (Stmt, Label); - if Current_Token = Tok_While then - Scan; - Set_Condition (Stmt, Parse_Expression); - Expect (Tok_Loop); - end if; - Scan; - Set_Sequential_Statement_Chain - (Stmt, Parse_Sequential_Statements (Stmt)); - Expect (Tok_End); - Scan_Expect (Tok_Loop); - Scan; - Check_End_Name (Stmt); + Stmt := Parse_While_Loop_Statement (Label); + Set_Location (Stmt, Loc); + -- A loop statement can have a label, even in vhdl87. Label := Null_Identifier; @@ -6655,10 +6732,10 @@ package body Parse is Kind : Iir_Kind; Subprg: Iir; Subprg_Body : Iir; - Begin_Loc : Location_Type; - End_Loc : Location_Type; + Start_Loc, Begin_Loc, End_Loc : Location_Type; begin -- Create the node. + Start_Loc := Get_Token_Location; case Current_Token is when Tok_Procedure => Kind := Iir_Kind_Procedure_Declaration; @@ -6708,6 +6785,11 @@ package body Parse is Parse_Subprogram_Parameters_And_Return (Subprg, Kind = Iir_Kind_Function_Declaration); + if Flag_Elocations then + Create_Elocations (Subprg); + Set_Start_Location (Subprg, Start_Loc); + end if; + if Current_Token = Tok_Semi_Colon then return Subprg; end if; @@ -7313,8 +7395,10 @@ package body Parse is -- { concurrent_statement } -- Note there is no END. This part is followed by: -- END GENERATE [ /generate/_label ] ; - function Parse_Generate_Statement_Body (Parent : Iir; Label : Name_Id) - return Iir + procedure Parse_Generate_Statement_Body (Parent : Iir; + Label : Name_Id; + Bod : out Iir; + End_Loc : out Location_Type) is function Is_Early_End return Boolean is begin @@ -7333,14 +7417,13 @@ package body Parse is end case; return False; end Is_Early_End; - - Bod : Iir; begin Bod := Create_Iir (Iir_Kind_Generate_Statement_Body); Set_Location (Bod); Set_Parent (Bod, Parent); Set_Alternative_Label (Bod, Label); Set_Has_Label (Bod, Label /= Null_Identifier); + End_Loc := No_Location; -- Check for a block declarative item. case Current_Token is @@ -7399,12 +7482,12 @@ package body Parse is -- Return now if no 'end' (and not expected). if Is_Early_End then - return Bod; + return; end if; - Expect (Tok_End); - -- Skip 'end' + End_Loc := Get_Token_Location; + Expect (Tok_End); Scan; if Vhdl_Std >= Vhdl_08 and then Current_Token /= Tok_Generate then @@ -7415,16 +7498,15 @@ package body Parse is -- Return now if no 'end' (and not expected). if Is_Early_End then - return Bod; + return; end if; Expect (Tok_End); + End_Loc := Get_Token_Location; -- Skip 'end' Scan; end if; - - return Bod; end Parse_Generate_Statement_Body; -- precond : FOR @@ -7455,6 +7537,8 @@ package body Parse is return Iir is Res : Iir; + Bod : Iir; + Start_Loc, Generate_Loc, End_Loc : Location_Type; begin if Label = Null_Identifier then Error_Msg_Parse ("a generate statement must have a label"); @@ -7462,6 +7546,7 @@ package body Parse is Res := Create_Iir (Iir_Kind_For_Generate_Statement); Set_Location (Res, Loc); Set_Label (Res, Label); + Start_Loc := Get_Token_Location; -- Skip 'for' Scan; @@ -7470,10 +7555,11 @@ package body Parse is -- Skip 'generate' Expect (Tok_Generate); + Generate_Loc := Get_Token_Location; Scan; - Set_Generate_Statement_Body - (Res, Parse_Generate_Statement_Body (Res, Null_Identifier)); + Parse_Generate_Statement_Body (Res, Null_Identifier, Bod, End_Loc); + Set_Generate_Statement_Body (Res, Bod); Expect (Tok_Generate); Set_End_Has_Reserved_Id (Res, True); @@ -7486,6 +7572,14 @@ package body Parse is -- the generate label. Check_End_Name (Res); Expect (Tok_Semi_Colon); + + if Flag_Elocations then + Create_Elocations (Res); + Set_Start_Location (Res, Start_Loc); + Set_Generate_Location (Res, Generate_Loc); + Set_End_Location (Res, End_Loc); + end if; + return Res; end Parse_For_Generate_Statement; @@ -7525,6 +7619,7 @@ package body Parse is Clause : Iir; Bod : Iir; Last : Iir; + Start_Loc, Generate_Loc, End_Loc : Location_Type; begin if Label = Null_Identifier then Error_Msg_Parse ("a generate statement must have a label"); @@ -7532,6 +7627,7 @@ package body Parse is Res := Create_Iir (Iir_Kind_If_Generate_Statement); Set_Location (Res, Loc); Set_Label (Res, Label); + Start_Loc := Get_Token_Location; -- Skip 'if'. Scan; @@ -7567,10 +7663,11 @@ package body Parse is Set_Condition (Clause, Cond); -- Skip 'generate' + Generate_Loc := Get_Token_Location; Expect (Tok_Generate); Scan; - Bod := Parse_Generate_Statement_Body (Res, Alt_Label); + Parse_Generate_Statement_Body (Res, Alt_Label, Bod, End_Loc); if Alt_Label /= Null_Identifier then -- Set location on the label, for xrefs. @@ -7585,11 +7682,19 @@ package body Parse is end if; Last := Clause; + if Flag_Elocations then + Create_Elocations (Clause); + Set_Start_Location (Clause, Start_Loc); + Set_Generate_Location (Clause, Generate_Loc); + Set_End_Location (Clause, End_Loc); + end if; + exit when Current_Token /= Tok_Elsif; -- Create new alternative. Clause := Create_Iir (Iir_Kind_If_Generate_Statement); Set_Location (Clause, Loc); + Start_Loc := Get_Token_Location; -- Skip 'elsif' Scan; @@ -7601,7 +7706,8 @@ package body Parse is end if; Clause := Create_Iir (Iir_Kind_If_Generate_Else_Clause); - Set_Location (Clause); + Start_Loc := Get_Token_Location; + Set_Location (Clause, Start_Loc); -- Skip 'else' Scan; @@ -7622,10 +7728,11 @@ package body Parse is end if; -- Skip 'generate' + Generate_Loc := Get_Token_Location; Expect (Tok_Generate); Scan; - Bod := Parse_Generate_Statement_Body (Res, Alt_Label); + Parse_Generate_Statement_Body (Res, Alt_Label, Bod, End_Loc); if Alt_Label /= Null_Identifier then -- Set location on the label, for xrefs. Set_Location (Bod, Alt_Loc); @@ -7634,6 +7741,13 @@ package body Parse is Set_Generate_Statement_Body (Clause, Bod); Set_Generate_Else_Clause (Last, Clause); + + if Flag_Elocations then + Create_Elocations (Clause); + Set_Start_Location (Clause, Start_Loc); + Set_Generate_Location (Clause, Generate_Loc); + Set_End_Location (Clause, End_Loc); + end if; end if; Expect (Tok_Generate); @@ -7664,6 +7778,7 @@ package body Parse is Bod : Iir; Assoc : Iir; Expr : Iir; + End_Loc : Location_Type; begin Loc := Get_Token_Location; @@ -7709,7 +7824,7 @@ package body Parse is Expect (Tok_Double_Arrow); Scan; - Bod := Parse_Generate_Statement_Body (Parent, Alt_Label); + Parse_Generate_Statement_Body (Parent, Alt_Label, Bod, End_Loc); Set_Associated_Block (Assoc, Bod); if Alt_Label /= Null_Identifier then -- Set location on the label, for xrefs. |