KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
gendrill_file_writer_base.cpp
Go to the documentation of this file.
1/*
2 * This program source code file is part of KiCad, a free EDA CAD application.
3 *
4 * Copyright (C) 2017 Jean_Pierre Charras <jp.charras at wanadoo.fr>
5 * Copyright (C) 2015 SoftPLC Corporation, Dick Hollenbeck <dick@softplc.com>
6 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
7 *
8 * This program is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU General Public License
10 * as published by the Free Software Foundation; either version 2
11 * of the License, or (at your option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, you may find one here:
20 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
21 * or you may search the http://www.gnu.org website for the version 2 license,
22 * or you may write to the Free Software Foundation, Inc.,
23 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
24 */
25
26#include <board.h>
27#include <footprint.h>
28#include <pad.h>
29#include <pcb_track.h>
30#include <collectors.h>
31#include <reporter.h>
32#include <richio.h>
33
35
36
37/* Helper function for sorting hole list.
38 * Compare function used for sorting holes type type:
39 * plated then not plated
40 * then by increasing diameter value
41 * then by attribute type (vias, pad, mechanical)
42 * then by X then Y position
43 */
44static bool cmpHoleSorting( const HOLE_INFO& a, const HOLE_INFO& b )
45{
47 return b.m_Hole_NotPlated;
48
51
52 // At this point (same diameter, same plated type), group by attribute
53 // type (via, pad, mechanical, although currently only not plated pads are mechanical)
56
57 // At this point (same diameter, same type), sort by X then Y position.
58 // This is optimal for drilling and make the file reproducible as long as holes
59 // have not changed, even if the data order has changed.
60 if( a.m_Hole_Pos.x != b.m_Hole_Pos.x )
61 return a.m_Hole_Pos.x < b.m_Hole_Pos.x;
62
63 return a.m_Hole_Pos.y < b.m_Hole_Pos.y;
64}
65
66
68 bool aGenerateNPTH_list )
69{
70 HOLE_INFO new_hole;
71
72 m_holeListBuffer.clear();
73 m_toolListBuffer.clear();
74
75 wxASSERT( aLayerPair.first < aLayerPair.second ); // fix the caller
76
77 // build hole list for vias
78 if( ! aGenerateNPTH_list ) // vias are always plated !
79 {
80 for( auto track : m_pcb->Tracks() )
81 {
82 if( track->Type() != PCB_VIA_T )
83 continue;
84
85 PCB_VIA* via = static_cast<PCB_VIA*>( track );
86 int hole_sz = via->GetDrillValue();
87
88 if( hole_sz == 0 ) // Should not occur.
89 continue;
90
91 new_hole.m_ItemParent = via;
92
93 if( aLayerPair == DRILL_LAYER_PAIR( F_Cu, B_Cu ) )
94 new_hole.m_HoleAttribute = HOLE_ATTRIBUTE::HOLE_VIA_THROUGH;
95 else
96 new_hole.m_HoleAttribute = HOLE_ATTRIBUTE::HOLE_VIA_BURIED;
97
98 new_hole.m_Tool_Reference = -1; // Flag value for Not initialized
99 new_hole.m_Hole_Orient = ANGLE_0;
100 new_hole.m_Hole_Diameter = hole_sz;
101 new_hole.m_Hole_NotPlated = false;
102 new_hole.m_Hole_Size.x = new_hole.m_Hole_Size.y = new_hole.m_Hole_Diameter;
103
104 new_hole.m_Hole_Shape = 0; // hole shape: round
105 new_hole.m_Hole_Pos = via->GetStart();
106
107 via->LayerPair( &new_hole.m_Hole_Top_Layer, &new_hole.m_Hole_Bottom_Layer );
108
109 new_hole.m_Hole_Filled = via->Padstack().IsFilled().value_or( false );
110 new_hole.m_Hole_Capped = via->Padstack().IsCapped().value_or( false );
111
112 new_hole.m_Hole_Top_Covered =
113 via->Padstack().IsCovered( new_hole.m_Hole_Top_Layer ).value_or( false );
114
115 new_hole.m_Hole_Bot_Covered =
116 via->Padstack().IsCovered( new_hole.m_Hole_Bottom_Layer ).value_or( false );
117
118 new_hole.m_Hole_Top_Plugged =
119 via->Padstack().IsPlugged( new_hole.m_Hole_Top_Layer ).value_or( false );
120
121 new_hole.m_Hole_Bot_Plugged =
122 via->Padstack().IsPlugged( new_hole.m_Hole_Bottom_Layer ).value_or( false );
123
124 new_hole.m_Hole_Top_Tented =
125 via->Padstack().IsTented( new_hole.m_Hole_Top_Layer ).value_or( false );
126
127 new_hole.m_Hole_Bot_Tented =
128 via->Padstack().IsTented( new_hole.m_Hole_Bottom_Layer ).value_or( false );
129
130 // LayerPair() returns params with m_Hole_Bottom_Layer > m_Hole_Top_Layer
131 // Remember: top layer = 0 and bottom layer = 31 for through hole vias
132 // Any captured via should be from aLayerPair.first to aLayerPair.second exactly.
133 if( new_hole.m_Hole_Top_Layer != aLayerPair.first ||
134 new_hole.m_Hole_Bottom_Layer != aLayerPair.second )
135 continue;
136
137 m_holeListBuffer.push_back( new_hole );
138 }
139 }
140
141 if( aLayerPair == DRILL_LAYER_PAIR( F_Cu, B_Cu ) )
142 {
143 // add holes for thru hole pads
144 for( FOOTPRINT* footprint : m_pcb->Footprints() )
145 {
146 for( PAD* pad : footprint->Pads() )
147 {
148 if( !m_merge_PTH_NPTH )
149 {
150 if( !aGenerateNPTH_list && pad->GetAttribute() == PAD_ATTRIB::NPTH )
151 continue;
152
153 if( aGenerateNPTH_list && pad->GetAttribute() != PAD_ATTRIB::NPTH )
154 continue;
155 }
156
157 if( pad->GetDrillSize().x == 0 )
158 continue;
159
160 new_hole.m_ItemParent = pad;
161 new_hole.m_Hole_NotPlated = (pad->GetAttribute() == PAD_ATTRIB::NPTH);
162 new_hole.m_HoleAttribute = new_hole.m_Hole_NotPlated
163 ? HOLE_ATTRIBUTE::HOLE_MECHANICAL
164 : HOLE_ATTRIBUTE::HOLE_PAD;
165 new_hole.m_Tool_Reference = -1; // Flag is: Not initialized
166 new_hole.m_Hole_Orient = pad->GetOrientation();
167 new_hole.m_Hole_Shape = 0; // hole shape: round
168 new_hole.m_Hole_Diameter = std::min( pad->GetDrillSize().x, pad->GetDrillSize().y );
169 new_hole.m_Hole_Size.x = new_hole.m_Hole_Size.y = new_hole.m_Hole_Diameter;
170
171 // Convert oblong holes that are actually circular into drill hits
172 if( pad->GetDrillShape() != PAD_DRILL_SHAPE::CIRCLE &&
173 pad->GetDrillSizeX() != pad->GetDrillSizeY() )
174 {
175 new_hole.m_Hole_Shape = 1; // oval flag set
176 }
177
178 new_hole.m_Hole_Size = pad->GetDrillSize();
179 new_hole.m_Hole_Pos = pad->GetPosition(); // hole position
180 new_hole.m_Hole_Bottom_Layer = B_Cu;
181 new_hole.m_Hole_Top_Layer = F_Cu; // pad holes are through holes
182 m_holeListBuffer.push_back( new_hole );
183 }
184 }
185 }
186
187 // Sort holes per increasing diameter value (and for each dimater, by position)
188 sort( m_holeListBuffer.begin(), m_holeListBuffer.end(), cmpHoleSorting );
189
190 // build the tool list
191 int last_hole = -1; // Set to not initialized (this is a value not used
192 // for m_holeListBuffer[ii].m_Hole_Diameter)
193 bool last_notplated_opt = false;
194 HOLE_ATTRIBUTE last_attribute = HOLE_ATTRIBUTE::HOLE_UNKNOWN;
195
196 DRILL_TOOL new_tool( 0, false );
197 unsigned jj;
198
199 for( unsigned ii = 0; ii < m_holeListBuffer.size(); ii++ )
200 {
201 if( m_holeListBuffer[ii].m_Hole_Diameter != last_hole
202 || m_holeListBuffer[ii].m_Hole_NotPlated != last_notplated_opt
204 || m_holeListBuffer[ii].m_HoleAttribute != last_attribute
205#endif
206 )
207 {
208 new_tool.m_Diameter = m_holeListBuffer[ii].m_Hole_Diameter;
209 new_tool.m_Hole_NotPlated = m_holeListBuffer[ii].m_Hole_NotPlated;
210 new_tool.m_HoleAttribute = m_holeListBuffer[ii].m_HoleAttribute;
211 m_toolListBuffer.push_back( new_tool );
212 last_hole = new_tool.m_Diameter;
213 last_notplated_opt = new_tool.m_Hole_NotPlated;
214 last_attribute = new_tool.m_HoleAttribute;
215 }
216
217 jj = m_toolListBuffer.size();
218
219 if( jj == 0 )
220 continue; // Should not occurs
221
222 m_holeListBuffer[ii].m_Tool_Reference = jj; // Tool value Initialized (value >= 1)
223
224 m_toolListBuffer.back().m_TotalCount++;
225
226 if( m_holeListBuffer[ii].m_Hole_Shape )
227 m_toolListBuffer.back().m_OvalCount++;
228 }
229}
230
231
232std::vector<DRILL_LAYER_PAIR> GENDRILL_WRITER_BASE::getUniqueLayerPairs() const
233{
234 wxASSERT( m_pcb );
235
237
238 vias.Collect( m_pcb, { PCB_VIA_T } );
239
240 std::set<DRILL_LAYER_PAIR> unique;
241 DRILL_LAYER_PAIR layer_pair;
242
243 for( int i = 0; i < vias.GetCount(); ++i )
244 {
245 PCB_VIA* v = static_cast<PCB_VIA*>( vias[i] );
246
247 v->LayerPair( &layer_pair.first, &layer_pair.second );
248
249 // only make note of blind buried.
250 // thru hole is placed unconditionally as first in fetched list.
251 if( layer_pair != DRILL_LAYER_PAIR( F_Cu, B_Cu ) )
252 unique.insert( layer_pair );
253 }
254
255 std::vector<DRILL_LAYER_PAIR> ret;
256
257 ret.emplace_back( F_Cu, B_Cu ); // always first in returned list
258
259 for( const DRILL_LAYER_PAIR& pair : unique )
260 ret.push_back( pair );
261
262 return ret;
263}
264
265
266const std::string GENDRILL_WRITER_BASE::layerName( PCB_LAYER_ID aLayer ) const
267{
268 // Generic names here.
269 switch( aLayer )
270 {
271 case F_Cu:
272 return "front";
273 case B_Cu:
274 return "back";
275 default:
276 {
277 // aLayer use even values, and the first internal layer (In1) is B_Cu + 2.
278 int ly_id = ( aLayer - B_Cu ) / 2;
279 return StrPrintf( "in%d", ly_id );
280 }
281 }
282}
283
284
286{
287 std::string ret = layerName( aPair.first );
288 ret += '-';
289 ret += layerName( aPair.second );
290
291 return ret;
292}
293
294
296 bool aMerge_PTH_NPTH ) const
297{
298 wxASSERT( m_pcb );
299
300 wxString extend;
301
302 if( aNPTH )
303 extend = wxT( "-NPTH" );
304 else if( aPair == DRILL_LAYER_PAIR( F_Cu, B_Cu ) )
305 {
306 if( !aMerge_PTH_NPTH )
307 extend = wxT( "-PTH" );
308 // if merged, extend with nothing
309 }
310 else
311 {
312 extend += '-';
313 extend += layerPairName( aPair );
314 }
315
316 wxFileName fn = m_pcb->GetFileName();
317
318 fn.SetName( fn.GetName() + extend );
319 fn.SetExt( m_drillFileExtension );
320
321 wxString ret = fn.GetFullName();
322
323 return ret;
324}
325
326
328 IPC4761_FEATURES aFeature ) const
329{
330 wxASSERT( m_pcb );
331
332 wxString extend;
333
334 switch( aFeature )
335 {
336 case IPC4761_FEATURES::FILLED:
337 extend << wxT( "-filling-" );
338 extend << layerPairName( aPair );
339 break;
340 case IPC4761_FEATURES::CAPPED:
341 extend << wxT( "-capping-" );
342 extend << layerPairName( aPair );
343 break;
344 case IPC4761_FEATURES::COVERED_BACK:
345 extend << wxT( "-covering-" );
346 extend << layerName( aPair.second );
347 break;
348 case IPC4761_FEATURES::COVERED_FRONT:
349 extend << wxT( "-covering-" );
350 extend << layerName( aPair.first );
351 break;
352 case IPC4761_FEATURES::PLUGGED_BACK:
353 extend << wxT( "-plugging-" );
354 extend << layerName( aPair.second );
355 break;
356 case IPC4761_FEATURES::PLUGGED_FRONT:
357 extend << wxT( "-plugging-" );
358 extend << layerName( aPair.first );
359 break;
360 case IPC4761_FEATURES::TENTED_BACK:
361 extend << wxT( "-tenting-" );
362 extend << layerName( aPair.second );
363 break;
364 case IPC4761_FEATURES::TENTED_FRONT:
365 extend << wxT( "-tenting-" );
366 extend << layerName( aPair.first );
367 break;
368 }
369
370 wxFileName fn = m_pcb->GetFileName();
371
372 fn.SetName( fn.GetName() + extend );
373 fn.SetExt( m_drillFileExtension );
374
375 wxString ret = fn.GetFullName();
376
377 return ret;
378}
379
380
381bool GENDRILL_WRITER_BASE::CreateMapFilesSet( const wxString& aPlotDirectory,
382 REPORTER * aReporter )
383{
384 wxFileName fn;
385 wxString msg;
386
387 std::vector<DRILL_LAYER_PAIR> hole_sets = getUniqueLayerPairs();
388
389 // append a pair representing the NPTH set of holes, for separate drill files.
390 if( !m_merge_PTH_NPTH )
391 hole_sets.emplace_back( F_Cu, B_Cu );
392
393 for( std::vector<DRILL_LAYER_PAIR>::const_iterator it = hole_sets.begin();
394 it != hole_sets.end(); ++it )
395 {
396 DRILL_LAYER_PAIR pair = *it;
397 // For separate drill files, the last layer pair is the NPTH drill file.
398 bool doing_npth = m_merge_PTH_NPTH ? false : ( it == hole_sets.end() - 1 );
399
400 buildHolesList( pair, doing_npth );
401
402 // The file is created if it has holes, or if it is the non plated drill file
403 // to be sure the NPTH file is up to date in separate files mode.
404 // Also a PTH drill file is always created, to be sure at least one plated hole drill file
405 // is created (do not create any PTH drill file can be seen as not working drill generator).
406 if( getHolesCount() > 0 || doing_npth || pair == DRILL_LAYER_PAIR( F_Cu, B_Cu ) )
407 {
409 fn.SetPath( aPlotDirectory );
410
411 fn.SetExt( wxEmptyString ); // Will be added by GenDrillMap
412 wxString fullfilename = fn.GetFullPath() + wxT( "-drl_map" );
413 fullfilename << wxT(".") << GetDefaultPlotExtension( m_mapFileFmt );
414
415 bool success = genDrillMapFile( fullfilename, m_mapFileFmt );
416
417 if( ! success )
418 {
419 if( aReporter )
420 {
421 msg.Printf( _( "Failed to create file '%s'." ), fullfilename );
422 aReporter->Report( msg, RPT_SEVERITY_ERROR );
423 }
424
425 return false;
426 }
427 else
428 {
429 if( aReporter )
430 {
431 msg.Printf( _( "Created file '%s'." ), fullfilename );
432 aReporter->Report( msg, RPT_SEVERITY_ACTION );
433 }
434 }
435 }
436 }
437
438 return true;
439}
440
441
443 DRILL_LAYER_PAIR aLayerPair, TYPE_FILE aHoleType,
444 bool aCompatNCdrill ) const
445{
446// Build a wxString containing the .FileFunction attribute for drill files.
447// %TF.FileFunction,Plated[NonPlated],layer1num,layer2num,PTH[NPTH][Blind][Buried],Drill[Route][Mixed]*%
448 wxString text;
449
450 if( aCompatNCdrill )
451 text = wxT( "; #@! " );
452 else
453 text = wxT( "%" );
454
455 text << wxT( "TF.FileFunction," );
456
457 if( aHoleType == NPTH_FILE )
458 text << wxT( "NonPlated," );
459 else if( aHoleType == MIXED_FILE ) // only for Excellon format
460 text << wxT( "MixedPlating," );
461 else
462 text << wxT( "Plated," );
463
464 int layer1 = aLayerPair.first;
465 int layer2 = aLayerPair.second;
466 // In Gerber files, layers num are 1 to copper layer count instead of F_Cu to B_Cu
467 // (0 to copper layer count-1)
468 // Note also for a n copper layers board, gerber layers num are 1 ... n
469 //
470 // Copper layers use even values, so the layer id in file is
471 // (Copper layer id) /2 + 1 if layer is not B_Cu
472 if( layer1 == F_Cu )
473 layer1 = 1;
474 else
475 layer1 = ( ( layer1 - B_Cu ) / 2 ) + 1;
476
477 if( layer2 == B_Cu )
478 layer2 = m_pcb->GetCopperLayerCount();
479 else
480 layer2 = ( ( layer2 - B_Cu ) / 2) + 1;
481
482 text << layer1 << wxT( "," ) << layer2;
483
484 // Now add PTH or NPTH or Blind or Buried attribute
485 int toplayer = 1;
486 int bottomlayer = m_pcb->GetCopperLayerCount();
487
488 if( aHoleType == NPTH_FILE )
489 text << wxT( ",NPTH" );
490 else if( aHoleType == MIXED_FILE ) // only for Excellon format
491 {
492 // write nothing
493 }
494 else if( layer1 == toplayer && layer2 == bottomlayer )
495 text << wxT( ",PTH" );
496 else if( layer1 == toplayer || layer2 == bottomlayer )
497 text << wxT( ",Blind" );
498 else
499 text << wxT( ",Buried" );
500
501 // In NC drill file, these previous parameters should be enough:
502 if( aCompatNCdrill )
503 return text;
504
505
506 // Now add Drill or Route or Mixed:
507 // file containing only round holes have Drill attribute
508 // file containing only oblong holes have Routed attribute
509 // file containing both holes have Mixed attribute
510 bool hasOblong = false;
511 bool hasDrill = false;
512
513 for( unsigned ii = 0; ii < m_holeListBuffer.size(); ii++ )
514 {
515 const HOLE_INFO& hole_descr = m_holeListBuffer[ii];
516
517 if( hole_descr.m_Hole_Shape ) // m_Hole_Shape not 0 is an oblong hole)
518 hasOblong = true;
519 else
520 hasDrill = true;
521 }
522
523 if( hasOblong && hasDrill )
524 text << wxT( ",Mixed" );
525 else if( hasDrill )
526 text << wxT( ",Drill" );
527 else if( hasOblong )
528 text << wxT( ",Rout" );
529
530 // else: empty file.
531
532 // End of .FileFunction attribute:
533 text << wxT( "*%" );
534
535 return text;
536}
int GetCopperLayerCount() const
Definition: board.cpp:781
const FOOTPRINTS & Footprints() const
Definition: board.h:338
const TRACKS & Tracks() const
Definition: board.h:336
const wxString & GetFileName() const
Definition: board.h:334
int GetCount() const
Return the number of objects in the list.
Definition: collector.h:82
HOLE_ATTRIBUTE m_HoleAttribute
std::vector< DRILL_LAYER_PAIR > getUniqueLayerPairs() const
Get unique layer pairs by examining the micro and blind_buried vias.
virtual const wxString getDrillFileName(DRILL_LAYER_PAIR aPair, bool aNPTH, bool aMerge_PTH_NPTH) const
void buildHolesList(DRILL_LAYER_PAIR aLayerPair, bool aGenerateNPTH_list)
Create the list of holes and tools for a given board.
const wxString BuildFileFunctionAttributeString(DRILL_LAYER_PAIR aLayerPair, TYPE_FILE aHoleType, bool aCompatNCdrill=false) const
virtual const wxString getProtectionFileName(DRILL_LAYER_PAIR aPair, IPC4761_FEATURES aFeature) const
std::vector< HOLE_INFO > m_holeListBuffer
bool genDrillMapFile(const wxString &aFullFileName, PLOT_FORMAT aFormat)
Plot a map of drill marks for holes.
std::vector< DRILL_TOOL > m_toolListBuffer
const std::string layerPairName(DRILL_LAYER_PAIR aPair) const
bool CreateMapFilesSet(const wxString &aPlotDirectory, REPORTER *aReporter=nullptr)
Create the full set of map files for the board, in PS, PDF ... format (use SetMapFileFormat() to sele...
const std::string layerName(PCB_LAYER_ID aLayer) const
Handle hole which must be drilled (diameter, position and layers).
PCB_LAYER_ID m_Hole_Bottom_Layer
PCB_LAYER_ID m_Hole_Top_Layer
HOLE_ATTRIBUTE m_HoleAttribute
BOARD_ITEM * m_ItemParent
Definition: pad.h:54
Collect all BOARD_ITEM objects of a given set of KICAD_T type(s).
Definition: collectors.h:511
void Collect(BOARD_ITEM *aBoard, const std::vector< KICAD_T > &aTypes)
Collect BOARD_ITEM objects using this class's Inspector method, which does the collection.
Definition: collectors.cpp:518
void LayerPair(PCB_LAYER_ID *top_layer, PCB_LAYER_ID *bottom_layer) const
Return the 2 layers used by the via (the via actually uses all layers between these 2 layers)
Definition: pcb_track.cpp:1321
A pure virtual class used to derive REPORTER objects from.
Definition: reporter.h:73
virtual REPORTER & Report(const wxString &aText, SEVERITY aSeverity=RPT_SEVERITY_UNDEFINED)
Report a string with a given severity.
Definition: reporter.h:102
wxString GetDefaultPlotExtension(PLOT_FORMAT aFormat)
Return the default plot extension for a format.
#define _(s)
static constexpr EDA_ANGLE ANGLE_0
Definition: eda_angle.h:401
static bool cmpHoleSorting(const HOLE_INFO &a, const HOLE_INFO &b)
helper classes to handle hole info for drill files generators.
#define USE_ATTRIB_FOR_HOLES
std::pair< PCB_LAYER_ID, PCB_LAYER_ID > DRILL_LAYER_PAIR
PCB_LAYER_ID
A quick note on layer IDs:
Definition: layer_ids.h:60
@ B_Cu
Definition: layer_ids.h:65
@ F_Cu
Definition: layer_ids.h:64
@ RPT_SEVERITY_ERROR
@ RPT_SEVERITY_ACTION
int StrPrintf(std::string *result, const char *format,...)
This is like sprintf() but the output is appended to a std::string instead of to a character array.
Definition: richio.cpp:70
@ PCB_VIA_T
class PCB_VIA, a via (like a track segment on a copper layer)
Definition: typeinfo.h:97