KiCad PCB EDA Suite
Loading...
Searching...
No Matches
from_to_cache.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.
5 *
6 * This program is free software: you can redistribute it and/or modify it
7 * under the terms of the GNU General Public License as published by the
8 * Free Software Foundation, either version 3 of the License, or (at your
9 * option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful, but
12 * WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License along
17 * with this program. If not, see <http://www.gnu.org/licenses/>.
18 */
19
20#include <cstdio>
21#include <memory>
22#include <reporter.h>
23#include <board.h>
24#include <string_utils.h>
25
26#include <pcbexpr_evaluator.h>
27
30
32
34{
35 m_ftEndpoints.clear();
36
37 for( FOOTPRINT* footprint : m_board->Footprints() )
38 {
39 for( PAD* pad : footprint->Pads() )
40 {
41 FT_ENDPOINT ent;
42 ent.name = footprint->GetReference() + wxT( "-" ) + pad->GetNumber();
43 ent.parent = pad;
44 m_ftEndpoints.push_back( ent );
45 ent.name = footprint->GetReference();
46 ent.parent = pad;
47 m_ftEndpoints.push_back( ent );
48 }
49 }
50}
51
52
54 PS_OK = 0,
56 PS_NO_PATH = -2
57};
58
59
60static bool isVertexVisited( CN_ITEM* v, const std::vector<CN_ITEM*>& path )
61{
62 for( CN_ITEM* u : path )
63 {
64 if ( u == v )
65 return true;
66 }
67
68 return false;
69}
70
71
72static PATH_STATUS uniquePathBetweenNodes( CN_ITEM* u, CN_ITEM* v, std::vector<CN_ITEM*>& outPath )
73{
74 using Path = std::vector<CN_ITEM*>;
75 std::deque<Path> Q;
76
77 Path pInit;
78 bool pathFound = false;
79 pInit.push_back( u );
80 Q.push_back( pInit );
81
82 while( Q.size() )
83 {
84 Path path = Q.front();
85 Q.pop_front();
86 CN_ITEM* last = path.back();
87
88 if( last == v )
89 {
90 outPath = path;
91
92 if( pathFound )
93 return PS_MULTIPLE_PATHS;
94
95 pathFound = true;
96 }
97
98 for( CN_ITEM* ci : last->ConnectedItems() )
99 {
100 bool vertexVisited = isVertexVisited( ci, path );
101
102 for( std::vector<CN_ITEM*>& p : Q )
103 if( isVertexVisited( ci, p ) )
104 {
105 vertexVisited = true;
106 break;
107 }
108
109 if( !vertexVisited )
110 {
111 Path newpath( path );
112 newpath.push_back( ci );
113 Q.push_back( newpath );
114 }
115 }
116 }
117
118 return pathFound ? PS_OK : PS_NO_PATH;
119};
120
121
122int FROM_TO_CACHE::cacheFromToPaths( const wxString& aFrom, const wxString& aTo )
123{
124 std::vector<FT_PATH> paths;
125 std::shared_ptr<CONNECTIVITY_DATA> connectivity = m_board->GetConnectivity();
126 std::shared_ptr<CN_CONNECTIVITY_ALGO> cnAlgo = connectivity->GetConnectivityAlgo();
127
128 for( FT_ENDPOINT& endpoint : m_ftEndpoints )
129 {
130 if( WildCompareString( aFrom, endpoint.name, false ) )
131 {
132 FT_PATH p;
133 p.net = endpoint.parent->GetNetCode();
134 p.from = endpoint.parent;
135 p.to = nullptr;
136 paths.push_back(p);
137 }
138 }
139
140 for( FT_PATH& path : paths )
141 {
142 int count = 0;
143
144 wxString fromName = path.from->GetParentFootprint()->GetReference()
145 + wxT( "-" ) + path.from->GetNumber();
146
147 auto padCandidates = connectivity->GetConnectedItems( path.from, EXCLUDE_ZONES );
148 PAD* toPad = nullptr;
149
150 for( BOARD_CONNECTED_ITEM* pitem : padCandidates )
151 {
152 if( pitem == path.from )
153 continue;
154
155 if( pitem->Type() != PCB_PAD_T )
156 continue;
157
158 const PAD *pad = static_cast<const PAD*>( pitem );
159
160 wxString toName = pad->GetParentFootprint()->GetReference()
161 + wxT( "-" ) + pad->GetNumber();
162
163 for( const FT_ENDPOINT& endpoint : m_ftEndpoints )
164 {
165 if( pad == endpoint.parent )
166 {
167 if( WildCompareString( aTo, endpoint.name, false ) )
168 {
169 count++;
170 toPad = endpoint.parent;
171
172 path.to = toPad;
173 path.fromName = fromName;
174 path.toName = toName;
175 path.fromWildcard = aFrom;
176 path.toWildcard = aTo;
177
178 if( count >= 2 )
179 {
180 // fixme: report this somewhere?
181 path.to = nullptr;
182 }
183 }
184 }
185 }
186 }
187 }
188
189 int newPaths = 0;
190
191 for( FT_PATH& path : paths )
192 {
193 if( !path.from || !path.to )
194 continue;
195
196 CN_ITEM* cnFrom = cnAlgo->ItemEntry( path.from ).GetItems().front();
197 CN_ITEM* cnTo = cnAlgo->ItemEntry( path.to ).GetItems().front();
198 std::vector<CN_ITEM*> upath;
199
200 auto result = uniquePathBetweenNodes( cnFrom, cnTo, upath );
201
202 if( result == PS_OK )
203 path.isUnique = true;
204 else
205 path.isUnique = false;
206
207 if( result == PS_NO_PATH )
208 continue;
209
210 for( const auto item : upath )
211 {
212 path.pathItems.insert( item->Parent() );
213 }
214
215 m_ftPaths.push_back(path);
216 newPaths++;
217 }
218
219 // reportAux( _("Cached %d paths\n"), newPaths );
220
221 return newPaths;
222}
223
224bool FROM_TO_CACHE::IsOnFromToPath( BOARD_CONNECTED_ITEM* aItem, const wxString& aFrom, const wxString& aTo )
225{
226 int nFromTosFound = 0;
227
228 if( !m_board )
229 return false;
230
231 for( int attempt = 0; attempt < 2; attempt++ )
232 {
233 // item already belongs to path
234 for( FT_PATH& ftPath : m_ftPaths )
235 {
236 if( aFrom == ftPath.fromWildcard && aTo == ftPath.toWildcard )
237 {
238 nFromTosFound++;
239
240 if( ftPath.pathItems.count( aItem ) )
241 return true;
242 }
243 }
244
245 if( !nFromTosFound )
246 cacheFromToPaths( aFrom, aTo );
247 else
248 return false;
249 }
250
251 return false;
252}
253
254
256{
257 m_board = aBoard;
259 m_ftPaths.clear();
260}
261
262
263FROM_TO_CACHE::FT_PATH* FROM_TO_CACHE::QueryFromToPath( const std::set<BOARD_CONNECTED_ITEM*>& aItems )
264{
265 for( FT_PATH& ftPath : m_ftPaths )
266 {
267 if ( ftPath.pathItems == aItems )
268 return &ftPath;
269 }
270
271 return nullptr;
272}
A base class derived from BOARD_ITEM for items that can be connected and have a net,...
Information pertinent to a Pcbnew printed circuit board.
Definition: board.h:296
const FOOTPRINTS & Footprints() const
Definition: board.h:337
std::shared_ptr< CONNECTIVITY_DATA > GetConnectivity() const
Return a list of missing connections between components/tracks.
Definition: board.h:495
CN_ITEM represents a BOARD_CONNETED_ITEM in the connectivity system (ie: a pad, track/arc/via,...
const std::vector< CN_ITEM * > & ConnectedItems() const
int cacheFromToPaths(const wxString &aFrom, const wxString &aTo)
std::vector< FT_PATH > m_ftPaths
Definition: from_to_cache.h:69
FT_PATH * QueryFromToPath(const std::set< BOARD_CONNECTED_ITEM * > &aItems)
bool IsOnFromToPath(BOARD_CONNECTED_ITEM *aItem, const wxString &aFrom, const wxString &aTo)
std::vector< FT_ENDPOINT > m_ftEndpoints
Definition: from_to_cache.h:68
BOARD * m_board
Definition: from_to_cache.h:71
void buildEndpointList()
void Rebuild(BOARD *aBoard)
Definition: pad.h:54
#define EXCLUDE_ZONES
PATH_STATUS
@ PS_OK
@ PS_MULTIPLE_PATHS
@ PS_NO_PATH
static bool isVertexVisited(CN_ITEM *v, const std::vector< CN_ITEM * > &path)
static PATH_STATUS uniquePathBetweenNodes(CN_ITEM *u, CN_ITEM *v, std::vector< CN_ITEM * > &outPath)
bool WildCompareString(const wxString &pattern, const wxString &string_to_tst, bool case_sensitive)
Compare a string against wild card (* and ?) pattern using the usual rules.
@ PCB_PAD_T
class PAD, a pad in a footprint
Definition: typeinfo.h:87