KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
length_calculation.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 The KiCad Developers, see AUTHORS.txt for contributors.
5 *
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License
8 * as published by the Free Software Foundation; either version 2
9 * of the License, or (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, you may find one here:
18 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
19 * or you may search the http://www.gnu.org website for the version 2 license,
20 * or you may write to the Free Software Foundation, Inc.,
21 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
22 */
23
24#include <length_calculation.h>
25
26#include <board.h>
29
30
32{
33 static std::initializer_list<KICAD_T> traceAndPadTypes = { PCB_TRACE_T, PCB_ARC_T, PCB_PAD_T };
34
35 PCB_LAYER_ID top_layer = UNDEFINED_LAYER;
36 PCB_LAYER_ID bottom_layer = UNDEFINED_LAYER;
37
38 const LSET layers = aBoard->GetDesignSettings().GetEnabledLayers();
39
40 for( auto layer_it = layers.copper_layers_begin(); layer_it != layers.copper_layers_end(); ++layer_it )
41 {
42 if( aBoard->GetConnectivity()->IsConnectedOnLayer( m_via, *layer_it, traceAndPadTypes ) )
43 {
44 if( top_layer == UNDEFINED_LAYER )
45 top_layer = *layer_it;
46 else
47 bottom_layer = *layer_it;
48 }
49 }
50
51 if( top_layer == UNDEFINED_LAYER )
52 top_layer = m_via->TopLayer();
53 if( bottom_layer == UNDEFINED_LAYER )
54 bottom_layer = m_via->BottomLayer();
55
56 SetLayers( bottom_layer, top_layer );
57}
58
59
60void LENGTH_CALCULATION::clipLineToPad( SHAPE_LINE_CHAIN& aLine, const PAD* aPad, PCB_LAYER_ID aLayer, bool aForward )
61{
62 const int start = aForward ? 0 : aLine.PointCount() - 1;
63 const int delta = aForward ? 1 : -1;
64
65 // Note: we don't apply the clip-to-pad optimization if an arc ends in a pad
66 // Room for future improvement.
67 if( aLine.IsPtOnArc( start ) )
68 return;
69
70 const auto& shape = aPad->GetEffectivePolygon( aLayer, ERROR_INSIDE );
71
72 // Skip the "first" (or last) vertex, we already know it's contained in the pad
73 int clip = start;
74
75 for( int vertex = start + delta; aForward ? vertex < aLine.PointCount() : vertex >= 0; vertex += delta )
76 {
77 SEG seg( aLine.GetPoint( vertex ), aLine.GetPoint( vertex - delta ) );
78
79 bool containsA = shape->Contains( seg.A );
80 bool containsB = shape->Contains( seg.B );
81
82 if( containsA && containsB )
83 {
84 // Whole segment is inside: clip out this segment
85 clip = vertex;
86 }
87 else if( containsB )
88 {
89 // Only one point inside: Find the intersection
90 VECTOR2I loc;
91
92 if( shape->Collide( seg, 0, nullptr, &loc ) )
93 {
94 aLine.Replace( vertex - delta, vertex - delta, loc );
95 }
96 }
97 }
98
99 if( !aForward && clip < start )
100 aLine.Remove( clip + 1, start );
101 else if( clip > start )
102 aLine.Remove( start, clip - 1 );
103
104 // Now connect the dots
105 aLine.Insert( aForward ? 0 : aLine.PointCount(), aPad->GetPosition() );
106}
107
108
109LENGTH_DETAILS LENGTH_CALCULATION::CalculateLengthDetails( std::vector<LENGTH_CALCULATION_ITEM>& aItems,
110 const PATH_OPTIMISATIONS aOptimisations,
111 const PAD* aStartPad, const PAD* aEndPad,
112 const bool aWithLayerLengths ) const
113{
114 // If this set of items has not been optimised, optimise for shortest electrical path
115 if( aOptimisations.OptimiseViaLayers || aOptimisations.MergeTracks || aOptimisations.MergeTracks )
116 {
117 std::vector<LENGTH_CALCULATION_ITEM*> pads;
118 std::vector<LENGTH_CALCULATION_ITEM*> lines;
119 std::vector<LENGTH_CALCULATION_ITEM*> vias;
120
121 // Map of line endpoints to line objects
122 std::map<VECTOR2I, std::unordered_set<LENGTH_CALCULATION_ITEM*>> linesPositionMap;
123
124 // Map of pad positions to pad objects
125 std::map<VECTOR2I, std::unordered_set<LENGTH_CALCULATION_ITEM*>> padsPositionMap;
126
127 for( LENGTH_CALCULATION_ITEM& item : aItems )
128 {
129 if( item.Type() == LENGTH_CALCULATION_ITEM::TYPE::PAD )
130 {
131 pads.emplace_back( &item );
132 padsPositionMap[item.GetPad()->GetPosition()].insert( &item );
133 }
134 else if( item.Type() == LENGTH_CALCULATION_ITEM::TYPE::VIA )
135 {
136 vias.emplace_back( &item );
137 }
138 else if( item.Type() == LENGTH_CALCULATION_ITEM::TYPE::LINE )
139 {
140 lines.emplace_back( &item );
141 linesPositionMap[item.GetLine().CPoint( 0 )].insert( &item );
142 linesPositionMap[item.GetLine().CLastPoint()].insert( &item );
143 }
144 }
145
146 if( aOptimisations.OptimiseViaLayers )
147 optimiseViaLayers( vias, lines, linesPositionMap, padsPositionMap );
148
149 if( aOptimisations.MergeTracks )
150 mergeLines( lines, linesPositionMap );
151
152 if( aOptimisations.OptimiseTracesInPads )
153 optimiseTracesInPads( pads, lines );
154 }
155
156 LENGTH_DETAILS details;
157
158 if( aWithLayerLengths )
159 details.LayerLengths = std::make_unique<std::map<PCB_LAYER_ID, int64_t>>();
160
161 const bool useHeight = m_board->GetDesignSettings().m_UseHeightForLengthCalcs;
162
163 // If this is a contiguous set of items, check if we have an inferred fanout via at either end. Note that this
164 // condition only arises as a result of how PNS assembles tuning paths - for DRC / net inspector calculations these
165 // fanout vias will be present in the object set and therefore do not need to be inferred
166 if( aOptimisations.InferViaInPad && useHeight )
167 {
168 inferViaInPad( aStartPad, aItems.front(), details );
169 inferViaInPad( aEndPad, aItems.back(), details );
170 }
171
172 // Add stats for each item
173 for( const LENGTH_CALCULATION_ITEM& item : aItems )
174 {
175 // Don't include merged items
177 || item.Type() == LENGTH_CALCULATION_ITEM::TYPE::UNKNOWN )
178 {
179 continue;
180 }
181
182 if( item.Type() == LENGTH_CALCULATION_ITEM::TYPE::LINE )
183 {
184 const int64_t length = item.GetLine().Length();
185
186 details.TrackLength += length;
187
188 if( details.LayerLengths )
189 ( *details.LayerLengths )[item.GetStartLayer()] += length;
190 }
191 else if( item.Type() == LENGTH_CALCULATION_ITEM::TYPE::VIA && useHeight )
192 {
193 const auto [layerStart, layerEnd] = item.GetLayers();
194 details.ViaLength += stackupHeight( layerStart, layerEnd );
195 details.NumVias += 1;
196 }
197 else if( item.Type() == LENGTH_CALCULATION_ITEM::TYPE::PAD )
198 {
199 details.PadToDieLength += item.GetPad()->GetPadToDieLength();
200 details.NumPads += 1;
201 }
202 }
203
204 return details;
205}
206
207
209 LENGTH_DETAILS& aDetails ) const
210{
211 if( aPad && aItem.Type() == LENGTH_CALCULATION_ITEM::TYPE::LINE )
212 {
213 const PCB_LAYER_ID startBottomLayer = aItem.GetStartLayer();
214 const LSET padLayers = aPad->Padstack().LayerSet();
215
216 if( !padLayers.Contains( startBottomLayer ) )
217 {
218 // This must be either F_Cu or B_Cu
219 const PCB_LAYER_ID padLayer = padLayers.Contains( F_Cu ) ? F_Cu : B_Cu;
220
221 aDetails.NumVias += 1;
222 aDetails.ViaLength += stackupHeight( startBottomLayer, padLayer );
223 }
224 }
225}
226
227
228int64_t LENGTH_CALCULATION::CalculateLength( std::vector<LENGTH_CALCULATION_ITEM>& aItems,
229 const PATH_OPTIMISATIONS aOptimisations, const PAD* aStartPad,
230 const PAD* aEndPad ) const
231{
232 return CalculateLengthDetails( aItems, aOptimisations, aStartPad, aEndPad ).TotalLength();
233}
234
235
236int LENGTH_CALCULATION::stackupHeight( const PCB_LAYER_ID aFirstLayer, const PCB_LAYER_ID aSecondLayer ) const
237{
239 return 0;
240
242 {
244 return stackup.GetLayerDistance( aFirstLayer, aSecondLayer );
245 }
246 else
247 {
248 BOARD_STACKUP stackup;
250 return stackup.GetLayerDistance( aFirstLayer, aSecondLayer );
251 }
252}
253
254
256 std::vector<LENGTH_CALCULATION_ITEM*>& aLines,
257 std::map<VECTOR2I, std::unordered_set<LENGTH_CALCULATION_ITEM*>>& aLinesPositionMap )
258{
259 // Vector of pads, and an associated flag to indicate whether they have been visited by the clustering algorithm
260 std::vector<LENGTH_CALCULATION_ITEM*> pads;
261
262 auto removeFromPositionMap = [&aLinesPositionMap]( LENGTH_CALCULATION_ITEM* line )
263 {
264 aLinesPositionMap[line->GetLine().CPoint( 0 )].erase( line );
265 aLinesPositionMap[line->GetLine().CLastPoint()].erase( line );
266 };
267
268 // Attempts to merge unmerged lines in to aPrimaryLine
269 auto tryMerge = [&removeFromPositionMap, &aLinesPositionMap]( const MERGE_POINT aMergePoint,
270 const VECTOR2I& aMergePos,
271 const LENGTH_CALCULATION_ITEM* aPrimaryItem,
272 SHAPE_LINE_CHAIN& aPrimaryLine, bool* aDidMerge )
273 {
274 const auto startItr = aLinesPositionMap.find( aMergePos );
275
276 if( startItr == aLinesPositionMap.end() )
277 return;
278
279 std::unordered_set<LENGTH_CALCULATION_ITEM*>& startItems = startItr->second;
280
281 if( startItems.size() != 1 )
282 return;
283
284 LENGTH_CALCULATION_ITEM* lineToMerge = *startItems.begin();
285
286 // Don't merge if line is an arc
287 if( !lineToMerge->GetLine().CArcs().empty() )
288 return;
289
290 // Don't merge if lines are on different layers
291 if( aPrimaryItem->GetStartLayer() != lineToMerge->GetStartLayer() )
292 return;
293
294 // Merge the lines
296 mergeShapeLineChains( aPrimaryLine, lineToMerge->GetLine(), aMergePoint );
297 removeFromPositionMap( lineToMerge );
298 *aDidMerge = true;
299 };
300
301 // Cluster all lines in to contiguous entities
302 for( LENGTH_CALCULATION_ITEM* primaryItem : aLines )
303 {
304 // Don't start with an already merged line
305 if( primaryItem->GetMergeStatus() != LENGTH_CALCULATION_ITEM::MERGE_STATUS::UNMERGED )
306 continue;
307
308 // Remove starting line from the position map
309 removeFromPositionMap( primaryItem );
310
311 SHAPE_LINE_CHAIN& primaryLine = primaryItem->GetLine();
312
313 // Merge all endpoints
314 primaryItem->SetMergeStatus( LENGTH_CALCULATION_ITEM::MERGE_STATUS::MERGED_IN_USE );
315 bool mergeComplete = false;
316
317 while( !mergeComplete )
318 {
319 bool startMerged = false;
320 bool endMerged = false;
321
322 VECTOR2I startPos = primaryLine.CPoint( 0 );
323 VECTOR2I endPos = primaryLine.CLastPoint();
324
325 tryMerge( MERGE_POINT::START, startPos, primaryItem, primaryLine, &startMerged );
326 tryMerge( MERGE_POINT::END, endPos, primaryItem, primaryLine, &endMerged );
327
328 mergeComplete = !startMerged && !endMerged;
329 }
330 }
331}
332
333
335 const MERGE_POINT aMergePoint )
336{
337 if( aMergePoint == MERGE_POINT::START )
338 {
339 if( aSecondary.GetPoint( 0 ) == aPrimary.GetPoint( 0 ) )
340 {
341 for( auto itr = aSecondary.CPoints().begin() + 1; itr != aSecondary.CPoints().end(); ++itr )
342 aPrimary.Insert( 0, *itr );
343 }
344 else
345 {
346 wxASSERT( aSecondary.CLastPoint() == aPrimary.GetPoint( 0 ) );
347
348 for( auto itr = aSecondary.CPoints().rbegin() + 1; itr != aSecondary.CPoints().rend(); ++itr )
349 aPrimary.Insert( 0, *itr );
350 }
351 }
352 else
353 {
354 if( aSecondary.GetPoint( 0 ) == aPrimary.CLastPoint() )
355 {
356 for( auto itr = aSecondary.CPoints().begin() + 1; itr != aSecondary.CPoints().end(); ++itr )
357 aPrimary.Append( *itr );
358 }
359 else
360 {
361 wxASSERT( aSecondary.CLastPoint() == aPrimary.CLastPoint() );
362
363 for( auto itr = aSecondary.CPoints().rbegin() + 1; itr != aSecondary.CPoints().rend(); ++itr )
364 aPrimary.Append( *itr );
365 }
366 }
367}
368
369
370void LENGTH_CALCULATION::optimiseTracesInPads( const std::vector<LENGTH_CALCULATION_ITEM*>& aPads,
371 const std::vector<LENGTH_CALCULATION_ITEM*>& aLines )
372{
373 for( LENGTH_CALCULATION_ITEM* padItem : aPads )
374 {
375 PAD* pad = padItem->GetPad();
376
377 for( LENGTH_CALCULATION_ITEM* lineItem : aLines )
378 {
379 // Ignore merged lines
380 if( lineItem->GetMergeStatus() != LENGTH_CALCULATION_ITEM::MERGE_STATUS::MERGED_IN_USE )
381 continue;
382
383 const PCB_LAYER_ID pcbLayer = lineItem->GetStartLayer();
384 SHAPE_LINE_CHAIN& line = lineItem->GetLine();
385
386 OptimiseTraceInPad( line, pad, pcbLayer );
387 }
388 }
389}
390
391
393 const std::vector<LENGTH_CALCULATION_ITEM*>& aVias, std::vector<LENGTH_CALCULATION_ITEM*>& aLines,
394 std::map<VECTOR2I, std::unordered_set<LENGTH_CALCULATION_ITEM*>>& aLinesPositionMap,
395 const std::map<VECTOR2I, std::unordered_set<LENGTH_CALCULATION_ITEM*>>& aPadsPositionMap )
396{
397 for( LENGTH_CALCULATION_ITEM* via : aVias )
398 {
399 auto lineItr = aLinesPositionMap.find( via->GetVia()->GetPosition() );
400
401 if( lineItr == aLinesPositionMap.end() )
402 continue;
403
404 std::unordered_set<LENGTH_CALCULATION_ITEM*>& connectedLines = lineItr->second;
405
406 if( connectedLines.empty() )
407 {
408 // No connected lines - this via is floating. Set both layers to the same
409 via->SetLayers( via->GetVia()->GetLayer(), via->GetVia()->GetLayer() );
410 }
411 else if( connectedLines.size() == 1 )
412 {
413 // This is either a via stub, or a via-in-pad
414 bool isViaInPad = false;
415 const PCB_LAYER_ID lineLayer = ( *connectedLines.begin() )->GetStartLayer();
416
417 auto padItr = aPadsPositionMap.find( via->GetVia()->GetPosition() );
418
419 if( padItr != aPadsPositionMap.end() )
420 {
421 // This could be a via-in-pad - check for overlapping pads which are not on the line layer
422 const std::unordered_set<LENGTH_CALCULATION_ITEM*>& pads = padItr->second;
423
424 if( pads.size() == 1 )
425 {
426 const LENGTH_CALCULATION_ITEM* padItem = *pads.begin();
427
428 if( !padItem->GetPad()->Padstack().LayerSet().Contains( lineLayer ) )
429 {
430 // This is probably a via-in-pad
431 isViaInPad = true;
432 via->SetLayers( lineLayer, padItem->GetStartLayer() );
433 }
434 }
435 }
436
437 if( !isViaInPad )
438 {
439 // This is a via stub - make its electrical length 0
440 via->SetLayers( lineLayer, lineLayer );
441 }
442 }
443 else
444 {
445 // This via has more than one track ending at it. Calculate the connected layer span (which may be shorter
446 // than the overall via span)
447 LSET layers;
448
449 for( const LENGTH_CALCULATION_ITEM* lineItem : connectedLines )
450 layers.set( lineItem->GetStartLayer() );
451
452 LSEQ cuStack = layers.CuStack();
453
454 PCB_LAYER_ID firstLayer = UNDEFINED_LAYER;
455 PCB_LAYER_ID lastLayer = UNDEFINED_LAYER;
456
457 for( PCB_LAYER_ID layer : cuStack )
458 {
459 if( firstLayer == UNDEFINED_LAYER )
460 firstLayer = layer;
461 else
462 lastLayer = layer;
463 }
464
465 if( lastLayer == UNDEFINED_LAYER )
466 via->SetLayers( firstLayer, firstLayer );
467 else
468 via->SetLayers( firstLayer, lastLayer );
469 }
470 }
471}
472
473
475{
476 // Only consider lines which terminate in the pad
477 if( aLine.CPoint( 0 ) != aPad->GetPosition() && aLine.CLastPoint() != aPad->GetPosition() )
478 return;
479
480 if( !aPad->FlashLayer( aPcbLayer ) )
481 return;
482
483 const auto& shape = aPad->GetEffectivePolygon( aPcbLayer, ERROR_INSIDE );
484
485 if( shape->Contains( aLine.CPoint( 0 ) ) )
486 clipLineToPad( aLine, aPad, aPcbLayer, true );
487 else if( shape->Contains( aLine.CPoint( -1 ) ) )
488 clipLineToPad( aLine, aPad, aPcbLayer, false );
489}
490
491
493{
494 if( PCB_TRACK* track = dynamic_cast<PCB_TRACK*>( aBoardItem ) )
495 {
496 if( track->Type() == PCB_VIA_T )
497 {
498 PCB_VIA* via = static_cast<PCB_VIA*>( track );
499
501 item.SetVia( via );
503
504 return item;
505 }
506
507 if( track->Type() == PCB_ARC_T )
508 {
509 PCB_ARC* arcParent = static_cast<PCB_ARC*>( track );
510 SHAPE_ARC shapeArc( arcParent->GetStart(), arcParent->GetMid(), arcParent->GetEnd(),
511 arcParent->GetWidth() );
512 SHAPE_LINE_CHAIN chainArc( shapeArc );
513
515 item.SetLine( chainArc );
516 item.SetLayers( track->GetLayer() );
517
518 return item;
519 }
520
521 if( track->Type() == PCB_TRACE_T )
522 {
523 std::vector<VECTOR2I> points{ track->GetStart(), track->GetEnd() };
524 SHAPE_LINE_CHAIN shape( points );
525
527 item.SetLine( shape );
528 item.SetLayers( track->GetLayer() );
529
530 return item;
531 }
532 }
533 else if( PAD* pad = dynamic_cast<PAD*>( aBoardItem ) )
534 {
536 item.SetPad( pad );
537
538 LSET& layers = pad->Padstack().LayerSet();
539 PCB_LAYER_ID firstLayer = UNDEFINED_LAYER;
540 PCB_LAYER_ID secondLayer = UNDEFINED_LAYER;
541
542 for( auto itr = layers.copper_layers_begin(); itr != layers.copper_layers_end(); ++itr )
543 {
544 if( firstLayer == UNDEFINED_LAYER )
545 firstLayer = *itr;
546 else
547 secondLayer = *itr;
548 }
549
550 item.SetLayers( firstLayer, secondLayer );
551
552 return item;
553 }
554
555 return {};
556}
@ ERROR_INSIDE
Definition: approximation.h:34
BASE_SET & set(size_t pos)
Definition: base_set.h:116
A base class derived from BOARD_ITEM for items that can be connected and have a net,...
LSET GetEnabledLayers() const
Return a bit-mask of all the layers that are enabled.
BOARD_STACKUP & GetStackupDescriptor()
bool m_UseHeightForLengthCalcs
Enable inclusion of stackup height in track length measurements and length tuning.
Manage layers needed to make a physical board.
void BuildDefaultStackupList(const BOARD_DESIGN_SETTINGS *aSettings, int aActiveCopperLayersCount=0)
Create a default stackup, according to the current BOARD_DESIGN_SETTINGS settings.
int GetLayerDistance(PCB_LAYER_ID aFirstLayer, PCB_LAYER_ID aSecondLayer) const
Calculate the distance (height) between the two given copper layers.
Information pertinent to a Pcbnew printed circuit board.
Definition: board.h:297
int GetCopperLayerCount() const
Definition: board.cpp:781
BOARD_DESIGN_SETTINGS & GetDesignSettings() const
Definition: board.cpp:946
std::shared_ptr< CONNECTIVITY_DATA > GetConnectivity() const
Return a list of missing connections between components/tracks.
Definition: board.h:495
Lightweight class which holds a pad, via, or a routed trace outline.
void CalculateViaLayers(const BOARD *aBoard)
Calculates active via payers for a proxied VIA object.
void SetPad(PAD *aPad)
Sets the parent PAD associated with this item.
SHAPE_LINE_CHAIN & GetLine() const
Gets the SHAPE_LINE_CHAIN associated with this item.
void SetLayers(const PCB_LAYER_ID aStart, const PCB_LAYER_ID aEnd=PCB_LAYER_ID::UNDEFINED_LAYER)
Sets the first and last layers associated with this item.
void SetMergeStatus(const MERGE_STATUS aStatus)
Sets the MERGE_STATUS of this item.
PCB_VIA * m_via
A proxied PVIAAD object. Set to nullptr if not proxying a VIA.
TYPE Type() const
Gets the routing item type.
void SetLine(const SHAPE_LINE_CHAIN &aLine)
Sets the source SHAPE_LINE_CHAIN of this item.
PCB_LAYER_ID GetStartLayer() const
Gets the start board layer for the proxied item.
void SetVia(PCB_VIA *aVia)
Sets the VIA associated with this item.
PAD * GetPad() const
Gets the parent PAD associated with this item.
static void optimiseViaLayers(const std::vector< LENGTH_CALCULATION_ITEM * > &aVias, std::vector< LENGTH_CALCULATION_ITEM * > &aLines, std::map< VECTOR2I, std::unordered_set< LENGTH_CALCULATION_ITEM * > > &aLinesPositionMap, const std::map< VECTOR2I, std::unordered_set< LENGTH_CALCULATION_ITEM * > > &aPadsPositionMap)
Optimises the via layers.
static void optimiseTracesInPads(const std::vector< LENGTH_CALCULATION_ITEM * > &aPads, const std::vector< LENGTH_CALCULATION_ITEM * > &aLines)
Optimises the given set of items to minimise the electrical path length.
int stackupHeight(PCB_LAYER_ID aFirstLayer, PCB_LAYER_ID aSecondLayer) const
Returns the stackup distance between the two given layers.
static void OptimiseTraceInPad(SHAPE_LINE_CHAIN &aLine, const PAD *aPad, PCB_LAYER_ID aPcbLayer)
Optimises the given trace / line to minimise the electrical path length within the given pad.
static void mergeLines(std::vector< LENGTH_CALCULATION_ITEM * > &aLines, std::map< VECTOR2I, std::unordered_set< LENGTH_CALCULATION_ITEM * > > &aLinesPositionMap)
Merges any lines (traces) that are contiguous, on one layer, and with no junctions.
LENGTH_DETAILS CalculateLengthDetails(std::vector< LENGTH_CALCULATION_ITEM > &aItems, PATH_OPTIMISATIONS aOptimisations, const PAD *aStartPad=nullptr, const PAD *aEndPad=nullptr, bool aWithLayerLengths=false) const
Calculates the electrical length of the given items.
BOARD * m_board
The parent board for all items.
void inferViaInPad(const PAD *aPad, const LENGTH_CALCULATION_ITEM &aItem, LENGTH_DETAILS &aDetails) const
Infers if there is a via in the given pad.
int64_t CalculateLength(std::vector< LENGTH_CALCULATION_ITEM > &aItems, PATH_OPTIMISATIONS aOptimisations, const PAD *aStartPad=nullptr, const PAD *aEndPad=nullptr) const
Calculates the electrical length of the given items.
LENGTH_CALCULATION_ITEM GetLengthCalculationItem(BOARD_CONNECTED_ITEM *aBoardItem) const
Return a LENGTH_CALCULATION_ITEM constructed from the given BOARD_CONNECTED_ITEM.
static void clipLineToPad(SHAPE_LINE_CHAIN &aLine, const PAD *aPad, PCB_LAYER_ID aLayer, bool aForward=true)
Clips the given line to the minimal direct electrical length within the pad.
MERGE_POINT
Enum to describe whether track merging is attempted from the start or end of a track segment.
static void mergeShapeLineChains(SHAPE_LINE_CHAIN &aPrimary, const SHAPE_LINE_CHAIN &aSecondary, MERGE_POINT aMergePoint)
Merges two SHAPE_LINE_CHAINs where there is a shared endpoing.
LSEQ is a sequence (and therefore also a set) of PCB_LAYER_IDs.
Definition: lseq.h:47
LSET is a set of PCB_LAYER_IDs.
Definition: lset.h:37
copper_layers_iterator copper_layers_end() const
Definition: lset.cpp:891
LSEQ CuStack() const
Return a sequence of copper layers in starting from the front/top and extending to the back/bottom.
Definition: lset.cpp:247
copper_layers_iterator copper_layers_begin() const
Definition: lset.cpp:885
bool Contains(PCB_LAYER_ID aLayer) const
See if the layer set contains a PCB layer.
Definition: lset.h:63
const LSET & LayerSet() const
Definition: padstack.h:280
Definition: pad.h:54
bool FlashLayer(int aLayer, bool aOnlyCheckIfPermitted=false) const
Check to see whether the pad should be flashed on the specific layer.
Definition: pad.cpp:355
VECTOR2I GetPosition() const override
Definition: pad.h:208
const PADSTACK & Padstack() const
Definition: pad.h:321
const std::shared_ptr< SHAPE_POLY_SET > & GetEffectivePolygon(PCB_LAYER_ID aLayer, ERROR_LOC aErrorLoc=ERROR_INSIDE) const
Definition: pad.cpp:513
const VECTOR2I & GetMid() const
Definition: pcb_track.h:337
const VECTOR2I & GetStart() const
Definition: pcb_track.h:152
const VECTOR2I & GetEnd() const
Definition: pcb_track.h:149
virtual int GetWidth() const
Definition: pcb_track.h:146
PCB_LAYER_ID BottomLayer() const
Definition: pcb_track.cpp:1349
PCB_LAYER_ID TopLayer() const
Definition: pcb_track.cpp:1343
Definition: seg.h:42
VECTOR2I A
Definition: seg.h:49
VECTOR2I B
Definition: seg.h:50
Represent a polyline containing arcs as well as line segments: A chain of connected line and/or arc s...
bool IsPtOnArc(size_t aPtIndex) const
virtual const VECTOR2I GetPoint(int aIndex) const override
int PointCount() const
Return the number of points (vertices) in this line chain.
void Replace(int aStartIndex, int aEndIndex, const VECTOR2I &aP)
Replace points with indices in range [start_index, end_index] with a single point aP.
const std::vector< SHAPE_ARC > & CArcs() const
void Append(int aX, int aY, bool aAllowDuplication=false)
Append a new point at the end of the line chain.
const VECTOR2I & CPoint(int aIndex) const
Return a reference to a given point in the line chain.
const VECTOR2I & CLastPoint() const
Return the last point in the line chain.
void Remove(int aStartIndex, int aEndIndex)
Remove the range of points [start_index, end_index] from the line chain.
void Insert(size_t aVertex, const VECTOR2I &aP)
const std::vector< VECTOR2I > & CPoints() const
a few functions useful in geometry calculations.
PCB_LAYER_ID
A quick note on layer IDs:
Definition: layer_ids.h:60
@ B_Cu
Definition: layer_ids.h:65
@ UNDEFINED_LAYER
Definition: layer_ids.h:61
@ F_Cu
Definition: layer_ids.h:64
Holds length measurement result details and statistics.
std::unique_ptr< std::map< PCB_LAYER_ID, int64_t > > LayerLengths
int64_t TotalLength() const
Struct to control which optimisations the length calculation code runs on the given path objects.
bool InferViaInPad
Determines if there is a via-in-pad present on the board but not in the item set.
bool OptimiseViaLayers
Optimise via layers for height calculations, ensuring only the distance between routed segments is co...
bool MergeTracks
Merges all contiguous (end-to-end, same layer) tracks.
bool OptimiseTracesInPads
Optimises the electrical length of tracks within pads.
constexpr int delta
@ PCB_VIA_T
class PCB_VIA, a via (like a track segment on a copper layer)
Definition: typeinfo.h:97
@ PCB_PAD_T
class PAD, a pad in a footprint
Definition: typeinfo.h:87
@ PCB_ARC_T
class PCB_ARC, an arc track segment on a copper layer
Definition: typeinfo.h:98
@ PCB_TRACE_T
class PCB_TRACK, a track segment (segment on a copper layer)
Definition: typeinfo.h:96