Gerbview: avoid a crash with malformed Gerber files.
Fix also incorrect handling of %LN (Load Name) deprecated command
This commit is contained in:
parent
302f234264
commit
cffe1b51e2
|
@ -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;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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
|
||||||
|
|
Loading…
Reference in New Issue