Gerbview: avoid a crash with malformed Gerber files.

Fix also incorrect handling of %LN (Load Name) deprecated command
This commit is contained in:
jean-pierre charras 2017-10-24 18:09:33 +02:00
parent 302f234264
commit cffe1b51e2
3 changed files with 49 additions and 22 deletions

View File

@ -127,7 +127,8 @@ double AM_PARAM::GetValue( const D_CODE* aDcode ) const
break; break;
default: default:
wxLogDebug( "AM_PARAM::GetValue(): unexpected type\n" ); wxLogDebug( "AM_PARAM::GetValue(): dcode %d prm %d/%d: unexpected type %d",
aDcode->m_Num_Dcode, ii, m_paramStack.size(), item.GetType() );
break; break;
} }
} }

View File

@ -352,19 +352,34 @@ void AM_PRIMITIVE::DrawBasicShape( GERBER_DRAW_ITEM* aParent,
*/ */
/* Generated by an aperture macro declaration like: /* Generated by an aperture macro declaration like:
* "4,1,3,0.0,0.0,0.0,0.5,0.5,0.5,0.5,0.0,-25" * "4,1,3,0.0,0.0,0.0,0.5,0.5,0.5,0.5,0.0,-25"
* type(4), exposure, corners count, corner1.x, corner.1y, ..., rotation * type(4), exposure, corners count, corner1.x, corner.1y, ..., corner1.x, corner.1y, rotation
* type is not stored in parameters list, so the first parameter is exposure * type is not stored in parameters list, so the first parameter is exposure
*/ */
int numPoints = (int) params[1].GetValue( tool ); // params[0] is the exposure and params[1] is the corners count after the first corner
rotation = params[numPoints * 2 + 4].GetValue( tool ) * 10.0; int numCorners = (int) params[1].GetValue( tool );
// the shape rotation is the last param of list, after corners
int last_prm = params.size() - 1;
rotation = params[last_prm].GetValue( tool ) * 10.0;
wxPoint pos; wxPoint pos;
// Read points. numPoints does not include the starting point, so add 1.
for( int i = 0; i<numPoints + 1; ++i ) // Read points.
// Note: numCorners is the polygon corner count, following the first corner
// * the polygon is always closed,
// * therefore the last XY coordinate is the same as the first
int prm_idx = 2; // params[2] is the first X coordinate
for( int i = 0; i <= numCorners; ++i )
{ {
int jj = i * 2 + 2; pos.x = scaletoIU( params[prm_idx].GetValue( tool ), m_GerbMetric );
pos.x = scaletoIU( params[jj].GetValue( tool ), m_GerbMetric ); prm_idx++;
pos.y = scaletoIU( params[jj + 1].GetValue( tool ), m_GerbMetric ); pos.y = scaletoIU( params[prm_idx].GetValue( tool ), m_GerbMetric );
prm_idx++;
polybuffer.push_back(pos); polybuffer.push_back(pos);
// Guard: ensure prm_idx < last_prm
// I saw malformed gerber files with numCorners = number
// of coordinates instead of number of coordinates following the first point
if( prm_idx >= last_prm )
break;
} }
// rotate polygon and move it to the actual position // rotate polygon and move it to the actual position
// shape rotation: // shape rotation:
@ -701,15 +716,20 @@ int AM_PRIMITIVE::GetShapeDim( GERBER_DRAW_ITEM* aParent )
// dim = min side of the bounding box (this is a poor criteria, but what is a good criteria b?) // dim = min side of the bounding box (this is a poor criteria, but what is a good criteria b?)
{ {
// exposure, corners count, corner1.x, corner.1y, ..., rotation // exposure, corners count, corner1.x, corner.1y, ..., rotation
// note: corners count is the count of corners following corner1
int numPoints = (int) params[1].GetValue( tool ); int numPoints = (int) params[1].GetValue( tool );
// Read points. numPoints does not include the starting point, so add 1. // Read points. numPoints does not include the starting point, so add 1.
// and calculate the bounding box; // and calculate the bounding box;
wxSize pos_min, pos_max, pos; wxSize pos_min, pos_max, pos;
for( int i = 0; i<numPoints + 1; ++i ) int prm_idx = 2; // params[2] is the first X coordinate
int last_prm = params.size() - 1;
for( int i = 0; i<= numPoints; ++i )
{ {
int jj = i * 2 + 2; pos.x = scaletoIU( params[prm_idx].GetValue( tool ), m_GerbMetric );
pos.x = scaletoIU( params[jj].GetValue( tool ), m_GerbMetric ); prm_idx++;
pos.y = scaletoIU( params[jj + 1].GetValue( tool ), m_GerbMetric ); pos.y = scaletoIU( params[prm_idx].GetValue( tool ), m_GerbMetric );
prm_idx++;
if( i == 0 ) if( i == 0 )
pos_min = pos_max = pos; pos_min = pos_max = pos;
else else
@ -725,6 +745,12 @@ int AM_PRIMITIVE::GetShapeDim( GERBER_DRAW_ITEM* aParent )
if( pos_max.y < pos.y ) if( pos_max.y < pos.y )
pos_max.y = pos.y; pos_max.y = pos.y;
} }
// Guard: ensure prm_idx < last_prm (last prm is orientation)
// I saw malformed gerber files with numCorners = number
// of coordinates instead of number of coordinates following the first point
if( prm_idx >= last_prm )
break;
} }
// calculate dim // calculate dim
wxSize size; wxSize size;

View File

@ -65,7 +65,7 @@ enum RS274X_PARAMETERS {
SCALE_FACTOR = CODE( 'S', 'F' ), // Default: A = 1.0, B = 1.0 SCALE_FACTOR = CODE( 'S', 'F' ), // Default: A = 1.0, B = 1.0
// Image parameters: // Image parameters:
// commands used only once at the beginning of the file // commands used only once at the beginning of the file, and are deprecated
IMAGE_JUSTIFY = CODE( 'I', 'J' ), // Default: no justification IMAGE_JUSTIFY = CODE( 'I', 'J' ), // Default: no justification
IMAGE_NAME = CODE( 'I', 'N' ), // Default: void IMAGE_NAME = CODE( 'I', 'N' ), // Default: void
IMAGE_OFFSET = CODE( 'I', 'O' ), // Default: A = 0, B = 0 IMAGE_OFFSET = CODE( 'I', 'O' ), // Default: A = 0, B = 0
@ -102,11 +102,12 @@ enum RS274X_PARAMETERS {
// May be used singly or may be layer specfic // May be used singly or may be layer specfic
// theses parameters are at the beginning of the file or layer // theses parameters are at the beginning of the file or layer
// and reset some layer parameters (like interpolation) // and reset some layer parameters (like interpolation)
LAYER_NAME = CODE( 'L', 'N' ), // Default: Positive
LAYER_POLARITY = CODE( 'L', 'P' ),
KNOCKOUT = CODE( 'K', 'O' ), // Default: off KNOCKOUT = CODE( 'K', 'O' ), // Default: off
STEP_AND_REPEAT = CODE( 'S', 'R' ), // Default: A = 1, B = 1 STEP_AND_REPEAT = CODE( 'S', 'R' ), // Default: A = 1, B = 1
ROTATE = CODE( 'R', 'O' ), // Default: 0 ROTATE = CODE( 'R', 'O' ), // Default: 0
LOAD_POLARITY = CODE( 'L', 'P' ), //LPC or LPD. Default: Dark (LPD)
LOAD_NAME = CODE( 'L', 'N' ), // Deprecated: equivalent to G04
}; };
@ -660,12 +661,11 @@ bool GERBER_FILE_IMAGE::ExecuteRS274XCommand( int command, char* buff, char*& te
break; break;
case LAYER_NAME: case LOAD_NAME:
m_Iterpolation = GERB_INTERPOL_LINEAR_1X; // Start a new Gerber layer // %LN is a (deprecated) equivalentto G04: a comment
GetLayerParams( ).m_LayerName.Empty(); while( *text && *text != '*' )
while( *text != '*' )
{ {
GetLayerParams( ).m_LayerName.Append( *text++ ); text++; // Skip text
} }
break; break;
@ -679,7 +679,7 @@ bool GERBER_FILE_IMAGE::ExecuteRS274XCommand( int command, char* buff, char*& te
m_ImageNegative ? "true" : "false" ); ) m_ImageNegative ? "true" : "false" ); )
break; break;
case LAYER_POLARITY: case LOAD_POLARITY:
if( *text == 'C' ) if( *text == 'C' )
GetLayerParams().m_LayerNegative = true; GetLayerParams().m_LayerNegative = true;
else else