2
0
Эх сурвалжийг харах

* fix #41254: move setting up the relationship between current_structdef and its type symbol for records from parse_record_members() to record_dec() so that its also applied for non-advanced records (other code still ensures that the record type itself can't be used inappropriately)
+ added test

Sven/Sarah Barth 2 сар өмнө
parent
commit
e6114a573d

+ 37 - 33
compiler/ptype.pas

@@ -666,29 +666,6 @@ implementation
                   not(symtablestack.stack^.next^.symtable.symtabletype in [globalsymtable,staticsymtable,objectsymtable,recordsymtable]);
                   not(symtablestack.stack^.next^.symtable.symtabletype in [globalsymtable,staticsymtable,objectsymtable,recordsymtable]);
         end;
         end;
 
 
-      var
-        olddef : tdef;
-
-      procedure set_typesym;
-        begin
-          if not assigned(recsym) then
-            exit;
-          if ttypesym(recsym).typedef=current_structdef then
-            exit;
-          ttypesym(recsym).typedef:=current_structdef;
-          current_structdef.typesym:=recsym;
-        end;
-
-      procedure reset_typesym;
-        begin
-          if not assigned(recsym) then
-            exit;
-          if ttypesym(recsym).typedef<>current_structdef then
-            exit;
-          ttypesym(recsym).typedef:=olddef;
-          current_structdef.typesym:=nil;
-        end;
-
       var
       var
         pd : tprocdef;
         pd : tprocdef;
         oldparse_only: boolean;
         oldparse_only: boolean;
@@ -713,14 +690,6 @@ implementation
         if (token=_SEMICOLON) then
         if (token=_SEMICOLON) then
           Exit;
           Exit;
 
 
-        { the correct typesym<->def relationship is needed for example when
-          parsing parameters that are specializations of the record or when
-          using nested constants and such }
-        if assigned(recsym) then
-          olddef:=ttypesym(recsym).typedef
-        else
-          olddef:=nil;
-        set_typesym;
         current_structdef.symtable.currentvisibility:=vis_public;
         current_structdef.symtable.currentvisibility:=vis_public;
         fields_allowed:=true;
         fields_allowed:=true;
         is_classdef:=false;
         is_classdef:=false;
@@ -1042,11 +1011,33 @@ implementation
               consume(_ID); { Give a ident expected message, like tp7 }
               consume(_ID); { Give a ident expected message, like tp7 }
           end;
           end;
         until false;
         until false;
-        reset_typesym;
       end;
       end;
 
 
     { reads a record declaration }
     { reads a record declaration }
     function record_dec(const n:tidstring;recsym:tsym;genericdef:tstoreddef;genericlist:tfphashobjectlist):tdef;
     function record_dec(const n:tidstring;recsym:tsym;genericdef:tstoreddef;genericlist:tfphashobjectlist):tdef;
+      var
+         olddef : tdef;
+
+      procedure set_typesym;
+        begin
+          if not assigned(recsym) then
+            exit;
+          if ttypesym(recsym).typedef=current_structdef then
+            exit;
+          ttypesym(recsym).typedef:=current_structdef;
+          current_structdef.typesym:=recsym;
+        end;
+
+      procedure reset_typesym;
+        begin
+          if not assigned(recsym) then
+            exit;
+          if ttypesym(recsym).typedef<>current_structdef then
+            exit;
+          ttypesym(recsym).typedef:=olddef;
+          current_structdef.typesym:=nil;
+        end;
+
       var
       var
          old_current_structdef: tabstractrecorddef;
          old_current_structdef: tabstractrecorddef;
          old_current_genericdef,
          old_current_genericdef,
@@ -1113,7 +1104,17 @@ implementation
          maybe_insert_generic_rename_symbol(n,genericlist);
          maybe_insert_generic_rename_symbol(n,genericlist);
          { apply $RTTI directive to current object }
          { apply $RTTI directive to current object }
          current_structdef.apply_rtti_directive(current_module.rtti_directive);
          current_structdef.apply_rtti_directive(current_module.rtti_directive);
-         
+
+         { the correct typesym<->def relationship is needed for example when
+           parsing parameters that are specializations of the record, when
+           using nested constants and such or when specializing the type
+           itself as a pointer type }
+         if assigned(recsym) then
+           olddef:=ttypesym(recsym).typedef
+         else
+           olddef:=nil;
+         set_typesym;
+
          if m_advanced_records in current_settings.modeswitches then
          if m_advanced_records in current_settings.modeswitches then
            begin
            begin
              parse_record_members(recsym);
              parse_record_members(recsym);
@@ -1129,6 +1130,9 @@ implementation
                add_typedconst_init_routine(current_structdef);
                add_typedconst_init_routine(current_structdef);
              consume(_END);
              consume(_END);
             end;
             end;
+
+         reset_typesym;
+
          if (token=_ID) and (pattern='ALIGN') then
          if (token=_ID) and (pattern='ALIGN') then
            begin
            begin
              consume(_ID);
              consume(_ID);

+ 16 - 0
tests/webtbs/tw41254.pp

@@ -0,0 +1,16 @@
+{ %NORUN }
+
+program tw41254;
+
+{$mode objfpc}
+
+type
+  generic tt<T> = record
+    zz: ^specialize tt<T>; // Error: Internal error 2019112401
+  end;
+
+  ttint = specialize tt<longint>;
+
+begin
+end.
+