KiCad PCB EDA Suite
Loading...
Searching...
No Matches
erc_settings.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) 2024 CERN
5 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
6 * @author Jon Evans <[email protected]>
7 *
8 * This program is free software: you can redistribute it and/or modify it
9 * under the terms of the GNU General Public License as published by the
10 * Free Software Foundation, either version 3 of the License, or (at your
11 * option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful, but
14 * WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License along
19 * with this program. If not, see <http://www.gnu.org/licenses/>.
20 */
21
22#include <algorithm>
23#include <erc/erc_item.h>
24#include <erc/erc_settings.h>
25#include <schematic.h>
26#include <sch_marker.h>
27#include <sch_screen.h>
29#include <settings/parameters.h>
30
31
33
34
35
36#define OK PIN_ERROR::OK
37#define ERR PIN_ERROR::PP_ERROR
38#define WAR PIN_ERROR::WARNING
39
44{
45/* I, O, Bi, 3S, Pas, NIC, UnS, PwrI, PwrO, OC, OE, NC */
46/* I */ { OK, OK, OK, OK, OK, OK, WAR, OK, OK, OK, OK, ERR },
47/* O */ { OK, ERR, OK, WAR, OK, OK, WAR, OK, ERR, ERR, ERR, ERR },
48/* Bi */ { OK, OK, OK, OK, OK, OK, WAR, OK, WAR, OK, WAR, ERR },
49/* 3S */ { OK, WAR, OK, OK, OK, OK, WAR, WAR, ERR, WAR, WAR, ERR },
50/*Pas */ { OK, OK, OK, OK, OK, OK, WAR, OK, OK, OK, OK, ERR },
51/*NIC */ { OK, OK, OK, OK, OK, OK, OK, OK, OK, OK, OK, ERR },
52/*UnS */ { WAR, WAR, WAR, WAR, WAR, OK, WAR, WAR, WAR, WAR, WAR, ERR },
53/*PwrI*/ { OK, OK, OK, WAR, OK, OK, WAR, OK, OK, OK, OK, ERR },
54/*PwrO*/ { OK, ERR, WAR, ERR, OK, OK, WAR, OK, ERR, ERR, ERR, ERR },
55/* OC */ { OK, ERR, OK, WAR, OK, OK, WAR, OK, ERR, OK, OK, ERR },
56/* OE */ { OK, ERR, WAR, WAR, OK, OK, WAR, OK, ERR, OK, OK, ERR },
57/* NC */ { ERR, ERR, ERR, ERR, ERR, ERR, ERR, ERR, ERR, ERR, ERR, ERR }
58};
59
60
72{
73/* I, O, Bi, 3S, Pas, NIC, UnS, PwrI, PwrO, OC, OE, NC */
74/* I */ { NOD, DRV, DRV, DRV, DRV, NOD, DRV, NOD, DRV, DRV, DRV, NPI },
75/* O */ { DRV, DRV, DRV, DRV, DRV, NOD, DRV, DRV, DRV, DRV, DRV, NPI },
76/* Bi */ { DRV, DRV, DRV, DRV, DRV, NOD, DRV, NOD, DRV, DRV, DRV, NPI },
77/* 3S */ { DRV, DRV, DRV, DRV, DRV, NOD, DRV, NOD, DRV, DRV, DRV, NPI },
78/*Pas */ { DRV, DRV, DRV, DRV, DRV, NOD, DRV, NOD, DRV, DRV, DRV, NPI },
79/*NIC */ { NOD, NOD, NOD, NOD, NOD, NOD, NOD, NOD, NOD, NOD, NOD, NPI },
80/*UnS */ { DRV, DRV, DRV, DRV, DRV, NOD, DRV, NOD, DRV, DRV, DRV, NPI },
81/*PwrI*/ { NOD, DRV, NOD, NOD, NOD, NOD, NOD, NOD, DRV, NOD, NOD, NPI },
82/*PwrO*/ { DRV, DRV, DRV, DRV, DRV, NOD, DRV, DRV, DRV, DRV, DRV, NPI },
83/* OC */ { DRV, DRV, DRV, DRV, DRV, NOD, DRV, NOD, DRV, DRV, DRV, NPI },
84/* OE */ { DRV, DRV, DRV, DRV, DRV, NOD, DRV, NOD, DRV, DRV, DRV, NPI },
85/* NC */ { NPI, NPI, NPI, NPI, NPI, NPI, NPI, NPI, NPI, NPI, NPI, NPI }
86};
87
88
89ERC_SETTINGS::ERC_SETTINGS( JSON_SETTINGS* aParent, const std::string& aPath ) :
90 NESTED_SETTINGS( "erc", ercSettingsSchemaVersion, aParent, aPath )
91{
93
94 for( int i = ERCE_FIRST; i <= ERCE_LAST; ++i )
96
97 // Error is the default setting so set non-error priorities here.
127
128 m_params.emplace_back( new PARAM_LAMBDA<nlohmann::json>( "rule_severities",
129 [&]() -> nlohmann::json
130 {
131 nlohmann::json ret = {};
132
133 for( const RC_ITEM& item : ERC_ITEM::GetItemsWithSeverities() )
134 {
135 wxString name = item.GetSettingsKey();
136 int code = item.GetErrorCode();
137
138 if( name.IsEmpty() || m_ERCSeverities.count( code ) == 0 )
139 continue;
140
141 ret[std::string( name.ToUTF8() )] = SeverityToString( m_ERCSeverities[code] );
142 }
143
144 return ret;
145 },
146 [&]( const nlohmann::json& aJson )
147 {
148 if( !aJson.is_object() )
149 return;
150
151 for( const RC_ITEM& item : ERC_ITEM::GetItemsWithSeverities() )
152 {
153 int code = item.GetErrorCode();
154 wxString name = item.GetSettingsKey();
155
156 std::string key( name.ToUTF8() );
157
158 if( aJson.contains( key ) )
159 m_ERCSeverities[code] = SeverityFromString( aJson[key] );
160 }
161 },
162 {} ) );
163
164 m_params.emplace_back( new PARAM_LAMBDA<nlohmann::json>( "erc_exclusions",
165 [&]() -> nlohmann::json
166 {
167 nlohmann::json js = nlohmann::json::array();
168
169 for( const wxString& entry : m_ErcExclusions )
170 js.push_back( { entry, m_ErcExclusionComments[ entry ] } );
171
172 return js;
173 },
174 [&]( const nlohmann::json& aObj )
175 {
176 m_ErcExclusions.clear();
177
178 if( !aObj.is_array() )
179 return;
180
181 for( const nlohmann::json& entry : aObj )
182 {
183 if( entry.is_array() )
184 {
185 wxString serialized = entry[0].get<wxString>();
186 m_ErcExclusions.insert( serialized );
187 m_ErcExclusionComments[ serialized ] = entry[1].get<wxString>();
188 }
189 else if( entry.is_string() )
190 {
191 m_ErcExclusions.insert( entry.get<wxString>() );
192 }
193 }
194 },
195 {} ) );
196
197 m_params.emplace_back( new PARAM_LAMBDA<nlohmann::json>( "pin_map",
198 [&]() -> nlohmann::json
199 {
200 nlohmann::json ret = nlohmann::json::array();
201
202 for( int i = 0; i < ELECTRICAL_PINTYPES_TOTAL; i++ )
203 {
204 nlohmann::json inner = nlohmann::json::array();
205
206 for( int j = 0; j < ELECTRICAL_PINTYPES_TOTAL; j++ )
207 inner.push_back( static_cast<int>( GetPinMapValue( i, j ) ) );
208
209 ret.push_back( inner );
210 }
211
212 return ret;
213 },
214 [&]( const nlohmann::json& aJson )
215 {
216 if( !aJson.is_array() || aJson.size() != ELECTRICAL_PINTYPES_TOTAL )
217 return;
218
219 for( size_t i = 0; i < ELECTRICAL_PINTYPES_TOTAL; i++ )
220 {
221 if( i > aJson.size() - 1 )
222 break;
223
224 nlohmann::json inner = aJson[i];
225
226 if( !inner.is_array() || inner.size() != ELECTRICAL_PINTYPES_TOTAL )
227 return;
228
229 for( size_t j = 0; j < ELECTRICAL_PINTYPES_TOTAL; j++ )
230 {
231 if( inner[j].is_number_integer() )
232 {
233 int val = inner[j].get<int>();
234
235 if( val >= 0 && val <= static_cast<int>( PIN_ERROR::UNCONNECTED ) )
236 SetPinMapValue( i, j, static_cast<PIN_ERROR>( val ) );
237 }
238 }
239 }
240 },
241 {} ) );
242
243 // Pin weights used for sorting. Take care, sorting is descending!
256
258}
259
260
262{
263 if( m_parent )
264 {
265 m_parent->ReleaseNestedSettings( this );
266 m_parent = nullptr;
267 }
268}
269
270
271SEVERITY ERC_SETTINGS::GetSeverity( int aErrorCode ) const
272{
273 // Special-case duplicate pin error. Multiple pins with the same number are allowed
274 // if they share the same net, but having them on different nets is always an error.
275 if( aErrorCode == ERCE_DUPLICATE_PIN_ERROR )
276 {
277 return RPT_SEVERITY_ERROR;
278 }
279 // Pin-to-pin codes carry independent ignore state but a fixed reported severity.
280 // Warning-or-error is controlled by which errorCode it is, so each slot's stored
281 // value is consulted only to determine ignore-vs-active.
282 else if( aErrorCode == ERCE_PIN_TO_PIN_ERROR )
283 {
284 auto it = m_ERCSeverities.find( ERCE_PIN_TO_PIN_ERROR );
285
286 if( it != m_ERCSeverities.end() && it->second == RPT_SEVERITY_IGNORE )
287 return RPT_SEVERITY_IGNORE;
288
289 return RPT_SEVERITY_ERROR;
290 }
291 else if( aErrorCode == ERCE_PIN_TO_PIN_WARNING )
292 {
294
295 if( it != m_ERCSeverities.end() && it->second == RPT_SEVERITY_IGNORE )
296 return RPT_SEVERITY_IGNORE;
297
299 }
300 else if( aErrorCode == ERCE_GENERIC_WARNING )
301 {
303 }
304 else if( aErrorCode == ERCE_GENERIC_ERROR )
305 {
306 return RPT_SEVERITY_ERROR;
307 }
308
309 wxCHECK_MSG( m_ERCSeverities.count( aErrorCode ), RPT_SEVERITY_IGNORE,
310 wxS( "Missing severity from map in ERC_SETTINGS!" ) );
311
312 return m_ERCSeverities.at( aErrorCode );
313}
314
315
316void ERC_SETTINGS::SetSeverity( int aErrorCode, SEVERITY aSeverity )
317{
318 m_ERCSeverities[ aErrorCode ] = aSeverity;
319}
320
321
323{
324 memcpy( m_PinMap, m_defaultPinMap, sizeof( m_PinMap ) );
325}
326
327
329{
330 bool operator()( const SCH_MARKER* item1, const SCH_MARKER* item2 ) const
331 {
332 wxCHECK( item1 && item2, false );
333
334 const VECTOR2I& p1 = item1->GetPosition();
335 const VECTOR2I& p2 = item2->GetPosition();
336
337 if( p1 == p2 )
338 return item1->SerializeToString() < item2->SerializeToString();
339
340 // VECTOR2::operator< orders by squared magnitude, which is not a strict
341 // weak ordering: mirrored points like (a, b) and (b, a) compare equal
342 // and collide in the std::set below, silently dropping one marker.
343 return p1.x < p2.x || ( p1.x == p2.x && p1.y < p2.y );
344 }
345};
346
347
348void SHEETLIST_ERC_ITEMS_PROVIDER::visitMarkers( std::function<void( SCH_MARKER* )> aVisitor ) const
349{
350 std::set<SCH_SCREEN*> seenScreens;
351
352 for( const SCH_SHEET_PATH& sheet : m_schematic->BuildUnorderedSheetList() )
353 {
354 bool firstTime = seenScreens.count( sheet.LastScreen() ) == 0;
355
356 if( firstTime )
357 seenScreens.insert( sheet.LastScreen() );
358
359 std::set<SCH_MARKER*, CompareMarkers> orderedMarkers;
360
361 for( SCH_ITEM* item : sheet.LastScreen()->Items().OfType( SCH_MARKER_T ) )
362 orderedMarkers.insert( static_cast<SCH_MARKER*>( item ) );
363
364 for( SCH_ITEM* item : orderedMarkers )
365 {
366 SCH_MARKER* marker = static_cast<SCH_MARKER*>( item );
367
368 if( marker->GetMarkerType() != MARKER_BASE::MARKER_ERC )
369 continue;
370
371 std::shared_ptr<const ERC_ITEM> ercItem =
372 std::static_pointer_cast<const ERC_ITEM>( marker->GetRCItem() );
373
374 // Only show sheet-specific markers on the owning sheet
375 if( ercItem->IsSheetSpecific() )
376 {
377 if( ercItem->GetSpecificSheetPath() != sheet )
378 continue;
379 }
380
381 // Don't show non-specific markers more than once
382 if( !firstTime && !ercItem->IsSheetSpecific() )
383 continue;
384
385 aVisitor( marker );
386 }
387 }
388}
389
390
392{
393 m_severities = aSeverities;
394
395 m_filteredMarkers.clear();
396
397 ERC_SETTINGS& settings = m_schematic->ErcSettings();
398
400 [&]( SCH_MARKER* aMarker )
401 {
402 SEVERITY markerSeverity;
403
404 if( aMarker->IsExcluded() )
405 markerSeverity = RPT_SEVERITY_EXCLUSION;
406 else
407 markerSeverity = settings.GetSeverity( aMarker->GetRCItem()->GetErrorCode() );
408
409 if( markerSeverity & m_severities )
410 m_filteredMarkers.push_back( aMarker );
411 } );
412
413 // Sort markers so that errors appear before warnings
414 std::stable_sort( m_filteredMarkers.begin(), m_filteredMarkers.end(),
415 []( const SCH_MARKER* a, const SCH_MARKER* b )
416 {
417 return a->GetSeverity() > b->GetSeverity();
418 } );
419}
420
421
426
427
429{
430 if( aSeverity < 0 )
431 return m_filteredMarkers.size();
432
433 int count = 0;
434
435 const ERC_SETTINGS& settings = m_schematic->ErcSettings();
436
438 [&]( SCH_MARKER* aMarker )
439 {
440 SEVERITY markerSeverity;
441
442 if( aMarker->IsExcluded() )
443 markerSeverity = RPT_SEVERITY_EXCLUSION;
444 else
445 markerSeverity = settings.GetSeverity( aMarker->GetRCItem()->GetErrorCode() );
446
447 if( ( markerSeverity & aSeverity ) > 0 )
448 count++;
449 } );
450
451 return count;
452}
453
454
455std::shared_ptr<ERC_ITEM> SHEETLIST_ERC_ITEMS_PROVIDER::GetERCItem( int aIndex ) const
456{
457 SCH_MARKER* marker = m_filteredMarkers[ aIndex ];
458
459 return marker ? std::static_pointer_cast<ERC_ITEM>( marker->GetRCItem() ) : nullptr;
460}
461
462
463std::shared_ptr<RC_ITEM> SHEETLIST_ERC_ITEMS_PROVIDER::GetItem( int aIndex ) const
464{
465 return GetERCItem( aIndex );
466}
467
468
469void SHEETLIST_ERC_ITEMS_PROVIDER::DeleteItem( int aIndex, bool aDeep )
470{
471 SCH_MARKER* marker = m_filteredMarkers[ aIndex ];
472 m_filteredMarkers.erase( m_filteredMarkers.begin() + aIndex );
473
474 if( aDeep )
475 {
476 SCH_SCREENS screens( m_schematic->Root() );
477 screens.DeleteMarker( marker );
478 }
479}
480
const char * name
static std::vector< std::reference_wrapper< RC_ITEM > > GetItemsWithSeverities()
Definition erc_item.h:76
Container for ERC settings.
ERC_PIN_SORTING_METRIC m_ERCSortingMetric
The type of sorting used by the ERC checker to resolve multi-pin errors.
std::map< wxString, wxString > m_ErcExclusionComments
std::map< ELECTRICAL_PINTYPE, int > m_PinTypeWeights
Weights for electrical pins used in ERC to decide which pin gets the marker in case of a multi-pin er...
static PIN_ERROR m_defaultPinMap[ELECTRICAL_PINTYPES_TOTAL][ELECTRICAL_PINTYPES_TOTAL]
Default Look up table which gives the ERC error level for a pair of connected pins.
SEVERITY GetSeverity(int aErrorCode) const
ERC_SETTINGS(JSON_SETTINGS *aParent, const std::string &aPath)
void SetSeverity(int aErrorCode, SEVERITY aSeverity)
virtual ~ERC_SETTINGS()
PIN_ERROR GetPinMapValue(int aFirstType, int aSecondType) const
std::map< int, SEVERITY > m_ERCSeverities
std::set< wxString > m_ErcExclusions
static int m_PinMinDrive[ELECTRICAL_PINTYPES_TOTAL][ELECTRICAL_PINTYPES_TOTAL]
Look up table which gives the minimal drive for a pair of connected pins on a net.
void SetPinMapValue(int aFirstType, int aSecondType, PIN_ERROR aValue)
PIN_ERROR m_PinMap[ELECTRICAL_PINTYPES_TOTAL][ELECTRICAL_PINTYPES_TOTAL]
std::vector< PARAM_BASE * > m_params
The list of parameters (owned by this object)
JSON_SETTINGS(const wxString &aFilename, SETTINGS_LOC aLocation, int aSchemaVersion)
bool IsExcluded() const
Definition marker_base.h:93
std::shared_ptr< RC_ITEM > GetRCItem() const
enum MARKER_T GetMarkerType() const
Definition marker_base.h:91
JSON_SETTINGS * m_parent
A pointer to the parent object to load and store from.
NESTED_SETTINGS(const std::string &aName, int aSchemaVersion, JSON_SETTINGS *aParent, const std::string &aPath, bool aLoadFromFile=true)
Like a normal param, but with custom getter and setter functions.
Definition parameters.h:297
A holder for a rule check item, DRC in Pcbnew or ERC in Eeschema.
Definition rc_item.h:84
int GetErrorCode() const
Definition rc_item.h:161
Base class for any item which can be embedded within the SCHEMATIC container class,...
Definition sch_item.h:168
wxString SerializeToString() const
VECTOR2I GetPosition() const override
Definition sch_marker.h:102
Container class that holds multiple SCH_SCREEN objects in a hierarchy.
Definition sch_screen.h:750
void DeleteMarker(SCH_MARKER *aMarker)
Delete a specific marker.
Handle access to a stack of flattened SCH_SHEET objects by way of a path for creating a flattened sch...
int GetCount(int aSeverity=-1) const override
void SetSeverities(int aSeverities) override
void visitMarkers(std::function< void(SCH_MARKER *)> aVisitor) const
void DeleteItem(int aIndex, bool aDeep) override
Remove (and optionally deletes) the indexed item from the list.
std::shared_ptr< ERC_ITEM > GetERCItem(int aIndex) const
std::vector< SCH_MARKER * > m_filteredMarkers
std::shared_ptr< RC_ITEM > GetItem(int aIndex) const override
Retrieve a RC_ITEM by index.
int GetSeverities() const override
const int ercSettingsSchemaVersion
#define ERR
#define WAR
#define OK
#define DRV
@ ERCE_UNSPECIFIED
@ ERCE_DRIVER_CONFLICT
Conflicting drivers (labels, etc) on a subgraph.
@ ERCE_SIMILAR_POWER
2 power pins are equal for case insensitive comparisons.
@ ERCE_UNCONNECTED_WIRE_ENDPOINT
A label is connected to more than one wire.
@ ERCE_GROUND_PIN_NOT_GROUND
A ground-labeled pin is not on a ground net while another pin is.
@ ERCE_SIMILAR_LABELS
2 labels are equal for case insensitive comparisons.
@ ERCE_STACKED_PIN_SYNTAX
Pin name resembles stacked pin notation.
@ ERCE_FIRST
@ ERCE_ENDPOINT_OFF_GRID
Pin or wire-end off grid.
@ ERCE_SAME_LOCAL_GLOBAL_LABEL
2 labels are equal for case insensitive comparisons.
@ ERCE_LAST
@ ERCE_SIMILAR_LABEL_AND_POWER
label and pin are equal for case insensitive comparisons.
@ ERCE_SAME_LOCAL_GLOBAL_POWER
Local power port and global power port have the same name.
@ ERCE_LABEL_SINGLE_PIN
A label is connected only to a single pin.
@ ERCE_BUS_ENTRY_CONFLICT
A wire connected to a bus doesn't match the bus.
@ ERCE_FOOTPRINT_LINK_ISSUES
The footprint link is invalid, or points to a missing (or inactive) footprint or library.
@ ERCE_DUPLICATE_PIN_ERROR
@ ERCE_NOCONNECT_NOT_CONNECTED
A no connect symbol is not connected to anything.
@ ERCE_FOUR_WAY_JUNCTION
A four-way junction was found.
@ ERCE_SIMULATION_MODEL
An error was found in the simulation model.
@ ERCE_LIB_SYMBOL_MISMATCH
Symbol doesn't match copy in library.
@ ERCE_GENERIC_ERROR
@ ERCE_NOCONNECT_CONNECTED
A no connect symbol is connected to more than 1 pin.
@ ERCE_PIN_TO_PIN_WARNING
@ ERCE_MISSING_INPUT_PIN
Symbol has input pins that are not placed.
@ ERCE_MISSING_UNIT
Symbol has units that are not placed on the schematic.
@ ERCE_FIELD_NAME_WHITESPACE
Field name has leading or trailing whitespace.
@ ERCE_MISSING_BIDI_PIN
Symbol has bi-directional pins that are not placed.
@ ERCE_LIB_SYMBOL_ISSUES
Symbol not found in active libraries.
@ ERCE_FOOTPRINT_FILTERS
The assigned footprint doesn't match the footprint filters.
@ ERCE_GENERIC_WARNING
@ ERCE_SINGLE_GLOBAL_LABEL
A label only exists once in the schematic.
@ ERCE_LABEL_MULTIPLE_WIRES
A label is connected to more than one wire.
@ ERCE_PIN_TO_PIN_ERROR
PIN_ERROR
The values a pin-to-pin entry in the pin matrix can take on.
#define NOD
#define NPI
Types of drive on a net (used for legacy ERC)
@ PT_INPUT
usual pin input: must be connected
Definition pin_type.h:37
@ PT_NC
not connected (must be left open)
Definition pin_type.h:50
@ PT_OUTPUT
usual output
Definition pin_type.h:38
@ PT_TRISTATE
tri state bus pin
Definition pin_type.h:40
@ PT_NIC
not internally connected (may be connected to anything)
Definition pin_type.h:44
@ PT_BIDI
input or output (like port for a microprocessor)
Definition pin_type.h:39
@ PT_OPENEMITTER
pin type open emitter
Definition pin_type.h:49
@ PT_POWER_OUT
output of a regulator: intended to be connected to power input pins
Definition pin_type.h:47
@ PT_OPENCOLLECTOR
pin type open collector
Definition pin_type.h:48
@ PT_POWER_IN
power input (GND, VCC for ICs). Must be connected to a power output.
Definition pin_type.h:46
@ PT_UNSPECIFIED
unknown electrical properties: creates always a warning when connected
Definition pin_type.h:45
@ PT_PASSIVE
pin for passive symbols: must be connected, and can be connected to any pin.
Definition pin_type.h:43
#define ELECTRICAL_PINTYPES_TOTAL
Definition pin_type.h:56
SEVERITY
@ RPT_SEVERITY_WARNING
@ RPT_SEVERITY_ERROR
@ RPT_SEVERITY_UNDEFINED
@ RPT_SEVERITY_EXCLUSION
@ RPT_SEVERITY_IGNORE
bool operator()(const SCH_MARKER *item1, const SCH_MARKER *item2) const
@ SCH_MARKER_T
Definition typeinfo.h:159
SEVERITY SeverityFromString(const wxString &aSeverity)
Definition ui_common.cpp:56
wxString SeverityToString(const SEVERITY &aSeverity)
Definition ui_common.cpp:67
VECTOR2< int32_t > VECTOR2I
Definition vector2d.h:687