diff --git a/gerbview/am_primitive.cpp b/gerbview/am_primitive.cpp
index 37afe673da..80135abfcb 100644
--- a/gerbview/am_primitive.cpp
+++ b/gerbview/am_primitive.cpp
@@ -92,6 +92,7 @@ void AM_PRIMITIVE::ConvertBasicShapeToPolygon( APERTURE_MACRO* aApertMacro,
                                                SHAPE_POLY_SET& aShapeBuffer )
 {
     // Draw the primitive shape for flashed items.
+    // Note: rotation of primitives inside a macro must be always done around the macro origin.
     // Create a static buffer to avoid a lot of memory reallocation.
     static std::vector<VECTOR2I> polybuffer;
     polybuffer.clear();
@@ -218,15 +219,17 @@ void AM_PRIMITIVE::ConvertBasicShapeToPolygon( APERTURE_MACRO* aApertMacro,
         for( int ii = 0; ii < 4; ii++ )
         {
             polybuffer = subshape_poly;
-            EDA_ANGLE sub_rotation = rotation + ANGLE_90 * ii;
+            EDA_ANGLE sub_rotation = ANGLE_90 * ii;
 
             for( unsigned jj = 0; jj < polybuffer.size(); jj++ )
                 RotatePoint( polybuffer[jj], -sub_rotation );
 
-            // Move to center position given by the tool:
+            // Move to center position given by the tool, and rotate the full shape around
+            // the center position (origin of the macro):
             for( unsigned jj = 0; jj < polybuffer.size(); jj++ )
             {
                 polybuffer[jj] += center;
+                RotatePoint( polybuffer[jj], -rotation );
             }
 
             aShapeBuffer.NewOutline();
@@ -264,6 +267,8 @@ void AM_PRIMITIVE::ConvertBasicShapeToPolygon( APERTURE_MACRO* aApertMacro,
         VECTOR2I center( mapPt( m_Params[0].GetValueFromMacro( aApertMacro ),
                          m_Params[1].GetValueFromMacro( aApertMacro ), m_GerbMetric ) );
 
+        EDA_ANGLE rotation( m_Params[8].GetValueFromMacro( aApertMacro ), DEGREES_T );
+
         // adjust outerDiam by this on each nested circle
         int diamAdjust = ( gap + penThickness ) * 2;
 
@@ -272,17 +277,21 @@ void AM_PRIMITIVE::ConvertBasicShapeToPolygon( APERTURE_MACRO* aApertMacro,
             if( outerDiam <= 0 )
                 break;
 
+            // calculate the rotated position of the center:
+            VECTOR2I circle_center = center;
+            RotatePoint( circle_center, -rotation );
+
             // Note: outerDiam is the outer diameter of the ring.
             // the ring graphic diameter is (outerDiam - penThickness)
             if( outerDiam <= penThickness )
             {   // No room to draw a ring (no room for the hole):
                 // draw a circle instead (with no hole), with the right diameter
-                TransformCircleToPolygon( aShapeBuffer, center, outerDiam / 2, arc_to_seg_error,
+                TransformCircleToPolygon( aShapeBuffer, circle_center, outerDiam / 2, arc_to_seg_error,
                                           ERROR_INSIDE );
             }
             else
             {
-                TransformRingToPolygon( aShapeBuffer, center, ( outerDiam - penThickness ) / 2,
+                TransformRingToPolygon( aShapeBuffer, circle_center, ( outerDiam - penThickness ) / 2,
                                         penThickness, arc_to_seg_error, ERROR_INSIDE );
             }
         }
@@ -290,13 +299,11 @@ void AM_PRIMITIVE::ConvertBasicShapeToPolygon( APERTURE_MACRO* aApertMacro,
         // Draw the cross:
         ConvertShapeToPolygon( aApertMacro, polybuffer );
 
-        EDA_ANGLE rotation( m_Params[8].GetValueFromMacro( aApertMacro ), DEGREES_T );
-
         for( unsigned ii = 0; ii < polybuffer.size(); ii++ )
         {
             // move crossair shape to center and rotate shape:
-            RotatePoint( polybuffer[ii], -rotation );
             polybuffer[ii] += center;
+            RotatePoint( polybuffer[ii], -rotation );
         }
 
         break;
@@ -375,13 +382,13 @@ void AM_PRIMITIVE::ConvertBasicShapeToPolygon( APERTURE_MACRO* aApertMacro,
         // Creates the shape:
         ConvertShapeToPolygon( aApertMacro, polybuffer );
 
-        // rotate polygon
+        // move and rotate polygonal shape
         EDA_ANGLE rotation( m_Params[5].GetValueFromMacro( aApertMacro ), DEGREES_T );
 
         for( unsigned ii = 0; ii < polybuffer.size(); ii++ )
         {
-            RotatePoint( polybuffer[ii], -rotation );
             polybuffer[ii] += curPos;
+            RotatePoint( polybuffer[ii], -rotation );
         }
 
         break;
diff --git a/gerbview/gerber_file_image.h b/gerbview/gerber_file_image.h
index 2fdd60a2fa..91cdb8758a 100644
--- a/gerbview/gerber_file_image.h
+++ b/gerbview/gerber_file_image.h
@@ -99,6 +99,10 @@ public:
                                         // and the actual coordinates calculation must handle this
 };
 
+// size of a single line of text from a gerber file.
+// warning: some files can have *very long* lines, so the buffer must be large.
+#define GERBER_BUFZ 1000000
+
 /**
  * Hold the image data and parameters for one gerber file and layer parameters.
  *
@@ -469,6 +473,9 @@ public:
     VECTOR2I           m_DisplayOffset;
     EDA_ANGLE          m_DisplayRotation;
 
+    // A large buffer to store one line
+    static char m_LineBuffer[GERBER_BUFZ+1];
+
 private:
     wxArrayString      m_messagesList;         // A list of messages created when reading a file
 
diff --git a/gerbview/gerber_test_files/aperture_macro_with_prim_rotation_test.gbr b/gerbview/gerber_test_files/aperture_macro_with_prim_rotation_test.gbr
new file mode 100644
index 0000000000..638f857787
--- /dev/null
+++ b/gerbview/gerber_test_files/aperture_macro_with_prim_rotation_test.gbr
@@ -0,0 +1,125 @@
+G04 ================================================================================================
+Copyright (C) 2023 Tony Luken <tonyluken62+gerberfileparser.gmail.com>
+
+This file is part of GerberFileParser.
+
+GerberFileParser is free software: you can redistribute it and/or modify it under the terms of
+the GNU General Public License as published by the Free Software Foundation, either version 3 of
+the License, or (at your option) any later version.
+
+GerberFileParser is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; 
+without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License along with GerberFileParser. If
+not, see <http://www.gnu.org/licenses/>.
+===================================================================================================*
+
+G04 Demonstrate rotation of all macro primitives*
+
+G04 Set coordinate format and units*
+%FSLAX26Y26*%
+%MOMM*%
+
+G04 Demo Circle Primitive*
+%AMSomeCircles*
+0 1, Exposure, Diameter, Center X, Center Y[, Rotation]*
+1,1,4,8,-5,0*
+1,1,4,8,-5,120*
+1,1,4,8,-5,240*
+%
+
+G04 Demo Polygon Primitive*
+%AMSomePolygons*
+0 5, Exposure, # vertices, Center X, Center Y, Diameter, Rotation*
+5,1,5,7,8,7.5,0*
+5,1,5,7,8,7.5,120*
+5,1,5,7,8,7.5,240*
+%
+
+G04 Demo Thermal Primitive*
+%AMSomeThermals*
+0 7, Center X, Center Y, Outer diameter, Inner diameter, Gap, Rotation*
+7,-15,-5,3,2,0.5,0*
+7,-15,-5,3,2,0.5,120*
+7,-15,-5,3,2,0.5,-120*
+%
+
+G04 Demo Outline Primitive*
+%AMSomeOutlines*
+0 4, Exposure, # vertices, Start X, Start Y, Subsequent points..., Rotation*
+4,1,3,15,2,15,5,19,2,15,2,0*
+4,1,3,15,2,15,5,19,2,15,2,120*
+4,1,3,15,2,15,5,19,2,15,2,240*
+%
+
+G04 Demo deprecated Lower Left Line Primitive*
+%AMSomeLowerLeftLines*
+0 22, Exposure, Width, Height, Lower Left X, Lower Left Y, Rotation*
+22,1,20,3,-10,-25,0*
+22,1,20,3,-10,-25,120*
+22,1,20,3,-10,-25,240*
+%
+
+G04 Demo Center Line Primitive*
+%AMSomeCenterLines*
+0 21, Exposure, width, height, center X, center Y, rotation*
+21,1,13,1.5,2,20,0*
+21,1,13,1.5,2,20,120*
+21,1,13,1.5,2,20,-120*
+%
+
+G04 Demo Vector Line Primitive including deprecated version (2)*
+%AMSomeVectorLines*
+0 20 (or 2), Exposure , Width, Start X, Start Y, End X, End Y, Rotation*
+20,1,1.5,-2,2,-6,7,0*
+20,1,1.5,-2,2,-6,7,120*
+2,1,1.5,-2,2,-6,7,240*
+%
+
+G04 Demo deprecated Moire Primitive*
+%AMSomeMoires*
+0 6, Center X, Center Y, Outer Diameter, Ring Thickness, Ring Gap, Max #Rings, 
+Crosshair Thickness, Crosshair Length, Rotation*
+6,3,-17,5,0.5,0.5,2,0.1,6,0*
+6,3,-17,5,0.5,0.5,2,0.1,6,120*
+6,3,-17,5,0.5,0.5,2,0.1,6,-120*
+%
+
+%ADD10SomeCircles*%
+%ADD11SomePolygons*%
+%ADD12SomeThermals*%
+%ADD13SomeOutlines*%
+%ADD14SomeLowerLeftLines*%
+%ADD15SomeCenterLines*%
+%ADD16SomeVectorLines*%
+%ADD17SomeMoires*%
+
+G01*
+%LPD*%
+
+D10*
+X0Y0D03*
+
+D11*
+X0Y0D03*
+
+D12*
+X0Y0D03*
+
+D13*
+X0Y0D03*
+
+D14*
+X0Y0D03*
+
+D15*
+X0Y0D03*
+
+D16*
+X0Y0D03*
+
+D17*
+X0Y0D03*
+
+M02*
\ No newline at end of file
diff --git a/gerbview/readgerb.cpp b/gerbview/readgerb.cpp
index c3f6869c7c..982e261bde 100644
--- a/gerbview/readgerb.cpp
+++ b/gerbview/readgerb.cpp
@@ -2,7 +2,7 @@
  * This program source code file is part of KiCad, a free EDA CAD application.
  *
  * Copyright (C) 2007-2016 Jean-Pierre Charras  jp.charras at wanadoo.fr
- * Copyright (C) 1992-2016 KiCad Developers, see AUTHORS.txt for contributors.
+ * Copyright (C) 1992-2023 KiCad Developers, see AUTHORS.txt for contributors.
  *
  * This program is free software; you can redistribute it and/or
  * modify it under the terms of the GNU General Public License
@@ -224,11 +224,9 @@ bool GERBER_FILE_IMAGE::TestFileIsRS274( const wxString& aFullFileName )
     return false;
 }
 
-// size of a single line of text from a gerber file.
-// warning: some files can have *very long* lines, so the buffer must be large.
-#define GERBER_BUFZ 1000000
+
 // A large buffer to store one line
-static char lineBuffer[GERBER_BUFZ+1];
+char GERBER_FILE_IMAGE::m_LineBuffer[GERBER_BUFZ+1];
 
 bool GERBER_FILE_IMAGE::LoadGerberFile( const wxString& aFullFileName )
 {
@@ -253,11 +251,11 @@ bool GERBER_FILE_IMAGE::LoadGerberFile( const wxString& aFullFileName )
 
     while( true )
     {
-        if( fgets( lineBuffer, GERBER_BUFZ, m_Current_File ) == nullptr )
+        if( fgets( m_LineBuffer, GERBER_BUFZ, m_Current_File ) == nullptr )
             break;
 
         m_LineNum++;
-        text = StrPurge( lineBuffer );
+        text = StrPurge( m_LineBuffer );
 
         while( text && *text )
         {
@@ -314,7 +312,7 @@ bool GERBER_FILE_IMAGE::LoadGerberFile( const wxString& aFullFileName )
                 if( m_CommandState != ENTER_RS274X_CMD )
                 {
                     m_CommandState = ENTER_RS274X_CMD;
-                    ReadRS274XCommand( lineBuffer, GERBER_BUFZ, text );
+                    ReadRS274XCommand( m_LineBuffer, GERBER_BUFZ, text );
                 }
                 else        //Error
                 {
diff --git a/gerbview/rs274d.cpp b/gerbview/rs274d.cpp
index f472a0831b..c8659728ad 100644
--- a/gerbview/rs274d.cpp
+++ b/gerbview/rs274d.cpp
@@ -1,8 +1,3 @@
-/**
- * @file rs274d.cpp
- * @brief functions to read the rs274d commands from a rs274d/rs274x file
- */
-
 /*
  * This program source code file is part of KiCad, a free EDA CAD application.
  *
@@ -26,6 +21,11 @@
  * 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA
  */
 
+/**
+ * @brief functions to read the rs274d commands from a rs274d/rs274x file
+ */
+
+
 #include <gerbview.h>
 #include <gerbview_frame.h>
 #include <trigo.h>
@@ -464,8 +464,7 @@ bool GERBER_FILE_IMAGE::Execute_G_Command( char*& text, int G_command )
             ExecuteRS274XCommand( code_command, nullptr, 0, cptr );
         }
 
-        while( *text && (*text != '*') )
-            text++;
+        GetEndOfBlock( m_LineBuffer, GERBER_BUFZ, text, m_Current_File );
 
         break;
 
diff --git a/gerbview/rs274x.cpp b/gerbview/rs274x.cpp
index b6c4ada0d5..a975c92560 100644
--- a/gerbview/rs274x.cpp
+++ b/gerbview/rs274x.cpp
@@ -2,7 +2,7 @@
  * This program source code file is part of KiCad, a free EDA CAD application.
  *
  * Copyright (C) 2007-2018 Jean-Pierre Charras  jp.charras at wanadoo.fr
- * Copyright (C) 1992-2018 KiCad Developers, see AUTHORS.txt for contributors.
+ * Copyright (C) 1992-2023 KiCad Developers, see AUTHORS.txt for contributors.
  *
  * This program is free software; you can redistribute it and/or
  * modify it under the terms of the GNU General Public License
@@ -22,9 +22,6 @@
  * 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA
  */
 
-/**
- * @file rs274x.cpp
- */
 
 #include <base_units.h>
 #include <math/util.h>      // for KiROUND
@@ -1049,8 +1046,7 @@ bool GERBER_FILE_IMAGE::ReadApertureMacro( char *aBuff, unsigned int aBuffSize,
             is_comment = true;
 
             // Skip comment
-            while( *aText && ( *aText != '*' ) )
-                aText++;
+            GetEndOfBlock( m_LineBuffer, GERBER_BUFZ, aText, m_Current_File );
 
             break;