45 top_layer = *layer_it;
47 bottom_layer = *layer_it;
62 const int start = aForward ? 0 : aLine.
PointCount() - 1;
63 const int delta = aForward ? 1 : -1;
75 for(
int vertex = start +
delta; aForward ? vertex < aLine.
PointCount() : vertex >= 0; vertex +=
delta )
79 bool containsA = shape->Contains( seg.
A );
80 bool containsB = shape->Contains( seg.
B );
82 if( containsA && containsB )
92 if( shape->Collide( seg, 0,
nullptr, &loc ) )
99 if( !aForward && clip < start )
100 aLine.
Remove( clip + 1, start );
101 else if( clip > start )
102 aLine.
Remove( start, clip - 1 );
111 const PAD* aStartPad,
const PAD* aEndPad,
112 const bool aWithLayerLengths )
const
117 std::vector<LENGTH_CALCULATION_ITEM*> pads;
118 std::vector<LENGTH_CALCULATION_ITEM*> lines;
119 std::vector<LENGTH_CALCULATION_ITEM*> vias;
122 std::map<VECTOR2I, std::unordered_set<LENGTH_CALCULATION_ITEM*>> linesPositionMap;
125 std::map<VECTOR2I, std::unordered_set<LENGTH_CALCULATION_ITEM*>> padsPositionMap;
129 if( item.Type() == LENGTH_CALCULATION_ITEM::TYPE::PAD )
131 pads.emplace_back( &item );
132 padsPositionMap[item.GetPad()->GetPosition()].insert( &item );
134 else if( item.Type() == LENGTH_CALCULATION_ITEM::TYPE::VIA )
136 vias.emplace_back( &item );
138 else if( item.Type() == LENGTH_CALCULATION_ITEM::TYPE::LINE )
140 lines.emplace_back( &item );
141 linesPositionMap[item.GetLine().CPoint( 0 )].insert( &item );
142 linesPositionMap[item.GetLine().CLastPoint()].insert( &item );
158 if( aWithLayerLengths )
159 details.
LayerLengths = std::make_unique<std::map<PCB_LAYER_ID, int64_t>>();
177 || item.Type() == LENGTH_CALCULATION_ITEM::TYPE::UNKNOWN )
182 if( item.Type() == LENGTH_CALCULATION_ITEM::TYPE::LINE )
184 const int64_t length = item.GetLine().Length();
189 ( *details.
LayerLengths )[item.GetStartLayer()] += length;
191 else if( item.Type() == LENGTH_CALCULATION_ITEM::TYPE::VIA && useHeight )
193 const auto [layerStart, layerEnd] = item.GetLayers();
197 else if( item.Type() == LENGTH_CALCULATION_ITEM::TYPE::PAD )
211 if( aPad && aItem.
Type() == LENGTH_CALCULATION_ITEM::TYPE::LINE )
216 if( !padLayers.
Contains( startBottomLayer ) )
230 const PAD* aEndPad )
const
256 std::vector<LENGTH_CALCULATION_ITEM*>& aLines,
257 std::map<
VECTOR2I, std::unordered_set<LENGTH_CALCULATION_ITEM*>>& aLinesPositionMap )
260 std::vector<LENGTH_CALCULATION_ITEM*> pads;
264 aLinesPositionMap[line->GetLine().CPoint( 0 )].erase( line );
265 aLinesPositionMap[line->GetLine().CLastPoint()].erase( line );
269 auto tryMerge = [&removeFromPositionMap, &aLinesPositionMap](
const MERGE_POINT aMergePoint,
274 const auto startItr = aLinesPositionMap.find( aMergePos );
276 if( startItr == aLinesPositionMap.end() )
279 std::unordered_set<LENGTH_CALCULATION_ITEM*>& startItems = startItr->second;
281 if( startItems.size() != 1 )
291 if( aPrimaryItem->GetStartLayer() != lineToMerge->
GetStartLayer() )
297 removeFromPositionMap( lineToMerge );
309 removeFromPositionMap( primaryItem );
315 bool mergeComplete =
false;
317 while( !mergeComplete )
319 bool startMerged =
false;
320 bool endMerged =
false;
328 mergeComplete = !startMerged && !endMerged;
341 for(
auto itr = aSecondary.
CPoints().begin() + 1; itr != aSecondary.
CPoints().end(); ++itr )
342 aPrimary.
Insert( 0, *itr );
348 for(
auto itr = aSecondary.
CPoints().rbegin() + 1; itr != aSecondary.
CPoints().rend(); ++itr )
349 aPrimary.
Insert( 0, *itr );
356 for(
auto itr = aSecondary.
CPoints().begin() + 1; itr != aSecondary.
CPoints().end(); ++itr )
363 for(
auto itr = aSecondary.
CPoints().rbegin() + 1; itr != aSecondary.
CPoints().rend(); ++itr )
371 const std::vector<LENGTH_CALCULATION_ITEM*>& aLines )
375 PAD*
pad = padItem->GetPad();
383 const PCB_LAYER_ID pcbLayer = lineItem->GetStartLayer();
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 )
399 auto lineItr = aLinesPositionMap.find(
via->GetVia()->GetPosition() );
401 if( lineItr == aLinesPositionMap.end() )
404 std::unordered_set<LENGTH_CALCULATION_ITEM*>& connectedLines = lineItr->second;
406 if( connectedLines.empty() )
409 via->SetLayers(
via->GetVia()->GetLayer(),
via->GetVia()->GetLayer() );
411 else if( connectedLines.size() == 1 )
414 bool isViaInPad =
false;
415 const PCB_LAYER_ID lineLayer = ( *connectedLines.begin() )->GetStartLayer();
417 auto padItr = aPadsPositionMap.find(
via->GetVia()->GetPosition() );
419 if( padItr != aPadsPositionMap.end() )
422 const std::unordered_set<LENGTH_CALCULATION_ITEM*>& pads = padItr->second;
424 if( pads.size() == 1 )
440 via->SetLayers( lineLayer, lineLayer );
450 layers.
set( lineItem->GetStartLayer() );
466 via->SetLayers( firstLayer, firstLayer );
468 via->SetLayers( firstLayer, lastLayer );
485 if( shape->Contains( aLine.
CPoint( 0 ) ) )
487 else if( shape->Contains( aLine.
CPoint( -1 ) ) )
523 std::vector<VECTOR2I> points{ track->GetStart(), track->GetEnd() };
533 else if(
PAD*
pad =
dynamic_cast<PAD*
>( aBoardItem ) )
538 LSET& layers =
pad->Padstack().LayerSet();
550 item.
SetLayers( firstLayer, secondLayer );
BASE_SET & set(size_t pos)
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.
int GetCopperLayerCount() const
BOARD_DESIGN_SETTINGS & GetDesignSettings() const
std::shared_ptr< CONNECTIVITY_DATA > GetConnectivity() const
Return a list of missing connections between components/tracks.
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.
LSET is a set of PCB_LAYER_IDs.
copper_layers_iterator copper_layers_end() const
LSEQ CuStack() const
Return a sequence of copper layers in starting from the front/top and extending to the back/bottom.
copper_layers_iterator copper_layers_begin() const
bool Contains(PCB_LAYER_ID aLayer) const
See if the layer set contains a PCB layer.
const LSET & LayerSet() const
bool FlashLayer(int aLayer, bool aOnlyCheckIfPermitted=false) const
Check to see whether the pad should be flashed on the specific layer.
VECTOR2I GetPosition() const override
const PADSTACK & Padstack() const
const std::shared_ptr< SHAPE_POLY_SET > & GetEffectivePolygon(PCB_LAYER_ID aLayer, ERROR_LOC aErrorLoc=ERROR_INSIDE) const
const VECTOR2I & GetMid() const
const VECTOR2I & GetStart() const
const VECTOR2I & GetEnd() const
virtual int GetWidth() const
PCB_LAYER_ID BottomLayer() const
PCB_LAYER_ID TopLayer() const
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:
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.
@ PCB_VIA_T
class PCB_VIA, a via (like a track segment on a copper layer)
@ PCB_PAD_T
class PAD, a pad in a footprint
@ PCB_ARC_T
class PCB_ARC, an arc track segment on a copper layer
@ PCB_TRACE_T
class PCB_TRACK, a track segment (segment on a copper layer)