From 692949fd0af71993208589bd32f72d8a4be27473 Mon Sep 17 00:00:00 2001 From: jean-pierre charras Date: Sat, 11 Nov 2023 18:17:20 +0100 Subject: [PATCH] Gerbview: fix some issues related to comments and primitives in macros: - comments on more than one line were not fully skipped, creating read errors - in macros, some primitives (not all) were incorrectly rotated: they must be rotated around the aperture macro origin, not the primitive origin. - a new test file with rotated primitives in macros is added. From master branch Fixes #16049 https://gitlab.com/kicad/code/kicad/-/issues/16049 --- gerbview/am_primitive.cpp | 25 ++++++++++++++++--------- gerbview/gerber_file_image.h | 7 +++++++ gerbview/readgerb.cpp | 14 ++++++-------- gerbview/rs274d.cpp | 13 ++++++------- gerbview/rs274x.cpp | 8 ++------ 5 files changed, 37 insertions(+), 30 deletions(-) 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 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 4fc7e90f54..047ba90fe4 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. * @@ -455,6 +459,9 @@ public: GERBER_LAYER m_GBRLayerParams; // hold params for the current gerber layer GERBER_DRAW_ITEMS m_drawings; // linked list of Gerber Items to draw + // 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/readgerb.cpp b/gerbview/readgerb.cpp index 84cd3c8a8f..10ad735e8e 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 @@ -223,11 +223,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 ) { @@ -252,11 +250,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 ) { @@ -313,7 +311,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 f53b62dbcf..bf90832803 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 #include #include @@ -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 e7a1336e80..e9d97bcc91 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 #include // 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;