KiCad PCB EDA Suite
Loading...
Searching...
No Matches
pcbexpr_evaluator.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) 2019-2022 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
25#include <cstdio>
26#include <memory>
27#include <mutex>
28#include <board.h>
29#include <lset.h>
31#include <pcbexpr_evaluator.h>
32#include <drc/drc_engine.h>
33
34/* --------------------------------------------------------------------------------------------
35 * Specialized Expression References
36 */
37
39{
40 wxASSERT( dynamic_cast<const PCBEXPR_CONTEXT*>( aCtx ) );
41
42 const PCBEXPR_CONTEXT* ctx = static_cast<const PCBEXPR_CONTEXT*>( aCtx );
43 BOARD_ITEM* item = ctx->GetItem( m_itemIndex );
44 return item;
45}
46
47
49{
50public:
52 LIBEVAL::VALUE( LayerName( aLayer ) ),
53 m_layer( aLayer )
54 {};
55
56 virtual bool EqualTo( LIBEVAL::CONTEXT* aCtx, const VALUE* b ) const override
57 {
58 // For boards with user-defined layer names there will be 2 entries for each layer
59 // in the ENUM_MAP: one for the canonical layer name and one for the user layer name.
60 // We need to check against both.
61
62 wxPGChoices& layerMap = ENUM_MAP<PCB_LAYER_ID>::Instance().Choices();
63 const wxString& layerName = b->AsString();
64 BOARD* board = static_cast<PCBEXPR_CONTEXT*>( aCtx )->GetBoard();
65
66 {
67 std::shared_lock<std::shared_mutex> readLock( board->m_CachesMutex );
68
69 auto i = board->m_LayerExpressionCache.find( layerName );
70
71 if( i != board->m_LayerExpressionCache.end() )
72 return i->second.Contains( m_layer );
73 }
74
75 LSET mask;
76
77 for( unsigned ii = 0; ii < layerMap.GetCount(); ++ii )
78 {
79 wxPGChoiceEntry& entry = layerMap[ii];
80
81 if( entry.GetText().Matches( layerName ) )
82 mask.set( ToLAYER_ID( entry.GetValue() ) );
83 }
84
85 {
86 std::unique_lock<std::shared_mutex> writeLock( board->m_CachesMutex );
87 board->m_LayerExpressionCache[ layerName ] = mask;
88 }
89
90 return mask.Contains( m_layer );
91 }
92
93protected:
95};
96
97
99{
100public:
101 PCBEXPR_PINTYPE_VALUE( const wxString& aPinTypeName ) :
102 LIBEVAL::VALUE( aPinTypeName )
103 {};
104
105 bool EqualTo( LIBEVAL::CONTEXT* aCtx, const VALUE* b ) const override
106 {
107 const wxString& thisStr = AsString();
108 const wxString& otherStr = b->AsString();
109
110 if( thisStr.IsSameAs( otherStr, false ) )
111 return true;
112
113 // Handle cases where the netlist token is different from the EEschema token
114 if( thisStr == wxT( "tri_state" ) )
115 return otherStr.IsSameAs( wxT( "Tri-state" ), false );
116
117 if( thisStr == wxT( "power_in" ) )
118 return otherStr.IsSameAs( wxT( "Power input" ), false );
119
120 if( thisStr == wxT( "power_out" ) )
121 return otherStr.IsSameAs( wxT( "Power output" ), false );
122
123 if( thisStr == wxT( "no_connect" ) )
124 return otherStr.IsSameAs( wxT( "Unconnected" ), false );
125
126 return false;
127 }
128};
129
130
132{
133public:
135 LIBEVAL::VALUE( wxEmptyString ),
136 m_item( aItem )
137 {};
138
139 const wxString& AsString() const override
140 {
141 const_cast<PCBEXPR_NETCLASS_VALUE*>( this )->Set(
144 }
145
146 bool EqualTo( LIBEVAL::CONTEXT* aCtx, const VALUE* b ) const override
147 {
148 if( const PCBEXPR_NETCLASS_VALUE* bValue = dynamic_cast<const PCBEXPR_NETCLASS_VALUE*>( b ) )
149 return *( m_item->GetEffectiveNetClass() )
150 == *( bValue->m_item->GetEffectiveNetClass() );
151 else
152 return LIBEVAL::VALUE::EqualTo( aCtx, b );
153 }
154
155 bool NotEqualTo( LIBEVAL::CONTEXT* aCtx, const LIBEVAL::VALUE* b ) const override
156 {
157 if( const PCBEXPR_NETCLASS_VALUE* bValue = dynamic_cast<const PCBEXPR_NETCLASS_VALUE*>( b ) )
158 return m_item->GetEffectiveNetClass() != bValue->m_item->GetEffectiveNetClass();
159 else
160 return LIBEVAL::VALUE::NotEqualTo( aCtx, b );
161 }
162
163protected:
165};
166
167
169{
170public:
172 LIBEVAL::VALUE( wxEmptyString ),
173 m_item( aItem )
174 {};
175
176 const wxString& AsString() const override
177 {
178 const_cast<PCBEXPR_NET_VALUE*>( this )->Set( m_item->GetNetname() );
180 }
181
182 bool EqualTo( LIBEVAL::CONTEXT* aCtx, const VALUE* b ) const override
183 {
184 if( const PCBEXPR_NET_VALUE* bValue = dynamic_cast<const PCBEXPR_NET_VALUE*>( b ) )
185 return m_item->GetNetCode() == bValue->m_item->GetNetCode();
186 else
187 return LIBEVAL::VALUE::EqualTo( aCtx, b );
188 }
189
190 bool NotEqualTo( LIBEVAL::CONTEXT* aCtx, const LIBEVAL::VALUE* b ) const override
191 {
192 if( const PCBEXPR_NET_VALUE* bValue = dynamic_cast<const PCBEXPR_NET_VALUE*>( b ) )
193 return m_item->GetNetCode() != bValue->m_item->GetNetCode();
194 else
195 return LIBEVAL::VALUE::NotEqualTo( aCtx, b );
196 }
197
198protected:
200};
201
202
204{
205 PCBEXPR_CONTEXT* context = static_cast<PCBEXPR_CONTEXT*>( aCtx );
206
207 if( m_itemIndex == 2 )
208 return new PCBEXPR_LAYER_VALUE( context->GetLayer() );
209
210 BOARD_ITEM* item = GetObject( aCtx );
211
212 if( !item )
213 return new LIBEVAL::VALUE();
214
215 auto it = m_matchingTypes.find( TYPE_HASH( *item ) );
216
217 if( it == m_matchingTypes.end() )
218 {
219 // Don't force user to type "A.Type == 'via' && A.Via_Type == 'buried'" when the
220 // simpler "A.Via_Type == 'buried'" is perfectly clear. Instead, return an undefined
221 // value when the property doesn't appear on a particular object.
222
223 return new LIBEVAL::VALUE();
224 }
225 else
226 {
228 {
229 return new LIBEVAL::VALUE( (double) item->Get<int>( it->second ) );
230 }
231 else
232 {
233 wxString str;
234
235 if( !m_isEnum )
236 {
237 str = item->Get<wxString>( it->second );
238
239 if( it->second->Name() == wxT( "Pin Type" ) )
240 return new PCBEXPR_PINTYPE_VALUE( str );
241 else
242 return new LIBEVAL::VALUE( str );
243 }
244 else
245 {
246 const wxAny& any = item->Get( it->second );
247 PCB_LAYER_ID layer;
248
249 if( it->second->Name() == wxT( "Layer" )
250 || it->second->Name() == wxT( "Layer Top" )
251 || it->second->Name() == wxT( "Layer Bottom" ) )
252 {
253 if( any.GetAs<PCB_LAYER_ID>( &layer ) )
254 return new PCBEXPR_LAYER_VALUE( layer );
255 else if( any.GetAs<wxString>( &str ) )
256 return new PCBEXPR_LAYER_VALUE( context->GetBoard()->GetLayerID( str ) );
257 }
258 else
259 {
260 if( any.GetAs<wxString>( &str ) )
261 return new LIBEVAL::VALUE( str );
262 }
263 }
264
265 return new LIBEVAL::VALUE();
266 }
267 }
268}
269
270
272{
273 BOARD_CONNECTED_ITEM* item = dynamic_cast<BOARD_CONNECTED_ITEM*>( GetObject( aCtx ) );
274
275 if( !item )
276 return new LIBEVAL::VALUE();
277
278 return new PCBEXPR_NETCLASS_VALUE( item );
279}
280
281
283{
284 BOARD_CONNECTED_ITEM* item = dynamic_cast<BOARD_CONNECTED_ITEM*>( GetObject( aCtx ) );
285
286 if( !item )
287 return new LIBEVAL::VALUE();
288
289 return new PCBEXPR_NET_VALUE( item );
290}
291
292
294{
295 BOARD_ITEM* item = GetObject( aCtx );
296
297 if( !item )
298 return new LIBEVAL::VALUE();
299
300 return new LIBEVAL::VALUE( ENUM_MAP<KICAD_T>::Instance().ToString( item->Type() ) );
301}
302
303
305{
307
308 return registry.Get( aName.Lower() );
309}
310
311
312std::unique_ptr<LIBEVAL::VAR_REF> PCBEXPR_UCODE::CreateVarRef( const wxString& aVar,
313 const wxString& aField )
314{
316 std::unique_ptr<PCBEXPR_VAR_REF> vref;
317
318 // Check for a couple of very common cases and compile them straight to "object code".
319
320 if( aField.CmpNoCase( wxT( "NetClass" ) ) == 0 )
321 {
322 if( aVar == wxT( "A" ) )
323 return std::make_unique<PCBEXPR_NETCLASS_REF>( 0 );
324 else if( aVar == wxT( "B" ) )
325 return std::make_unique<PCBEXPR_NETCLASS_REF>( 1 );
326 else
327 return nullptr;
328 }
329 else if( aField.CmpNoCase( wxT( "NetName" ) ) == 0 )
330 {
331 if( aVar == wxT( "A" ) )
332 return std::make_unique<PCBEXPR_NETNAME_REF>( 0 );
333 else if( aVar == wxT( "B" ) )
334 return std::make_unique<PCBEXPR_NETNAME_REF>( 1 );
335 else
336 return nullptr;
337 }
338 else if( aField.CmpNoCase( wxT( "Type" ) ) == 0 )
339 {
340 if( aVar == wxT( "A" ) )
341 return std::make_unique<PCBEXPR_TYPE_REF>( 0 );
342 else if( aVar == wxT( "B" ) )
343 return std::make_unique<PCBEXPR_TYPE_REF>( 1 );
344 else
345 return nullptr;
346 }
347
348 if( aVar == wxT( "A" ) || aVar == wxT( "AB" ) )
349 vref = std::make_unique<PCBEXPR_VAR_REF>( 0 );
350 else if( aVar == wxT( "B" ) )
351 vref = std::make_unique<PCBEXPR_VAR_REF>( 1 );
352 else if( aVar == wxT( "L" ) )
353 vref = std::make_unique<PCBEXPR_VAR_REF>( 2 );
354 else
355 return nullptr;
356
357 if( aField.length() == 0 ) // return reference to base object
358 {
359 return vref;
360 }
361
362 wxString field( aField );
363 field.Replace( wxT( "_" ), wxT( " " ) );
364
365 for( const PROPERTY_MANAGER::CLASS_INFO& cls : propMgr.GetAllClasses() )
366 {
367 if( propMgr.IsOfType( cls.type, TYPE_HASH( BOARD_ITEM ) ) )
368 {
369 PROPERTY_BASE* prop = propMgr.GetProperty( cls.type, field );
370
371 if( prop )
372 {
373 vref->AddAllowedClass( cls.type, prop );
374
375 if( prop->TypeHash() == TYPE_HASH( int ) )
376 {
377 vref->SetType( LIBEVAL::VT_NUMERIC );
378 }
379 else if( prop->TypeHash() == TYPE_HASH( bool ) )
380 {
381 vref->SetType( LIBEVAL::VT_NUMERIC );
382 }
383 else if( prop->TypeHash() == TYPE_HASH( wxString ) )
384 {
385 vref->SetType( LIBEVAL::VT_STRING );
386 }
387 else if ( prop->HasChoices() )
388 { // it's an enum, we treat it as string
389 vref->SetType( LIBEVAL::VT_STRING );
390 vref->SetIsEnum ( true );
391 }
392 else
393 {
394 wxFAIL_MSG( wxT( "PCBEXPR_UCODE::createVarRef: Unknown property type." ) );
395 }
396 }
397 }
398 }
399
400 if( vref->GetType() == LIBEVAL::VT_UNDEFINED )
401 vref->SetType( LIBEVAL::VT_PARSE_ERROR );
402
403 return vref;
404}
405
406
408{
409 if( m_items[0] )
410 return m_items[0]->GetBoard();
411
412 return nullptr;
413}
414
415
416/* --------------------------------------------------------------------------------------------
417 * Unit Resolvers
418 */
419
420const std::vector<wxString>& PCBEXPR_UNIT_RESOLVER::GetSupportedUnits() const
421{
422 static const std::vector<wxString> pcbUnits = { wxT( "mil" ), wxT( "mm" ), wxT( "in" ) };
423
424 return pcbUnits;
425}
426
427
429{
430 return _( "must be mm, in, or mil" );
431}
432
433
434double PCBEXPR_UNIT_RESOLVER::Convert( const wxString& aString, int unitId ) const
435{
436 double v = wxAtof( aString );
437
438 switch( unitId )
439 {
440 case 0: return EDA_UNIT_UTILS::UI::DoubleValueFromString( pcbIUScale, EDA_UNITS::MILS, aString );
441 case 1: return EDA_UNIT_UTILS::UI::DoubleValueFromString( pcbIUScale, EDA_UNITS::MILLIMETRES, aString );
442 case 2: return EDA_UNIT_UTILS::UI::DoubleValueFromString( pcbIUScale, EDA_UNITS::INCHES, aString );
443 default: return v;
444 }
445};
446
447
448const std::vector<wxString>& PCBEXPR_UNITLESS_RESOLVER::GetSupportedUnits() const
449{
450 static const std::vector<wxString> emptyUnits;
451
452 return emptyUnits;
453}
454
455
456double PCBEXPR_UNITLESS_RESOLVER::Convert( const wxString& aString, int unitId ) const
457{
458 return wxAtof( aString );
459};
460
461
463{
464 m_unitResolver.reset( aUnitResolver );
465}
466
467
468/* --------------------------------------------------------------------------------------------
469 * PCB Expression Evaluator
470 */
471
473 m_result( 0 ),
474 m_compiler( aUnitResolver ),
475 m_ucode(),
476 m_errorStatus()
477{
478}
479
480
482{
483}
484
485
486bool PCBEXPR_EVALUATOR::Evaluate( const wxString& aExpr )
487{
488 PCBEXPR_UCODE ucode;
489 PCBEXPR_CONTEXT preflightContext( NULL_CONSTRAINT, F_Cu );
490
491 if( !m_compiler.Compile( aExpr.ToUTF8().data(), &ucode, &preflightContext ) )
492 return false;
493
494 PCBEXPR_CONTEXT evaluationContext( NULL_CONSTRAINT, F_Cu );
495 LIBEVAL::VALUE* result = ucode.Run( &evaluationContext );
496
497 if( result->GetType() == LIBEVAL::VT_NUMERIC )
498 m_result = KiROUND( result->AsDouble() );
499
500 return true;
501}
502
constexpr EDA_IU_SCALE pcbIUScale
Definition: base_units.h:108
BASE_SET & set(size_t pos=std::numeric_limits< size_t >::max(), bool value=true)
Definition: base_set.h:62
A base class derived from BOARD_ITEM for items that can be connected and have a net,...
virtual NETCLASS * GetEffectiveNetClass() const
Return the NETCLASS for this item.
A base class for any item which can be embedded within the BOARD container class, and therefore insta...
Definition: board_item.h:79
virtual const BOARD * GetBoard() const
Return the BOARD in which this BOARD_ITEM resides, or NULL if none.
Definition: board_item.cpp:47
Information pertinent to a Pcbnew printed circuit board.
Definition: board.h:289
PCB_LAYER_ID GetLayerID(const wxString &aLayerName) const
Return the ID of a layer.
Definition: board.cpp:557
std::unordered_map< wxString, LSET > m_LayerExpressionCache
Definition: board.h:1285
std::shared_mutex m_CachesMutex
Definition: board.h:1279
KICAD_T Type() const
Returns the type of object.
Definition: eda_item.h:101
static ENUM_MAP< T > & Instance()
Definition: property.h:663
wxAny Get(PROPERTY_BASE *aProperty) const
Definition: inspectable.h:99
std::unique_ptr< UNIT_RESOLVER > m_unitResolver
bool Compile(const wxString &aString, UCODE *aCode, CONTEXT *aPreflightContext)
VALUE * Run(CONTEXT *ctx)
void Set(double aValue)
virtual const wxString & AsString() const
virtual bool NotEqualTo(CONTEXT *aCtx, const VALUE *b) const
virtual double AsDouble() const
VAR_TYPE_T GetType() const
virtual bool EqualTo(CONTEXT *aCtx, const VALUE *b) const
LSET is a set of PCB_LAYER_IDs.
Definition: lset.h:35
bool Contains(PCB_LAYER_ID aLayer)
See if the layer set contains a PCB layer.
Definition: lset.h:79
const wxString GetVariableSubstitutionName() const
Gets the name of this (maybe aggregate) netclass in a format for label variable substitutions.
Definition: netclass.cpp:179
LIBEVAL::FUNC_CALL_REF Get(const wxString &name)
static PCBEXPR_BUILTIN_FUNCTIONS & Instance()
PCBEXPR_COMPILER(LIBEVAL::UNIT_RESOLVER *aUnitResolver)
BOARD_ITEM * m_items[2]
BOARD * GetBoard() const
PCB_LAYER_ID GetLayer() const
BOARD_ITEM * GetItem(int index) const
PCBEXPR_EVALUATOR(LIBEVAL::UNIT_RESOLVER *aUnitResolver)
PCBEXPR_COMPILER m_compiler
bool Evaluate(const wxString &aExpr)
PCBEXPR_LAYER_VALUE(PCB_LAYER_ID aLayer)
virtual bool EqualTo(LIBEVAL::CONTEXT *aCtx, const VALUE *b) const override
LIBEVAL::VALUE * GetValue(LIBEVAL::CONTEXT *aCtx) override
bool EqualTo(LIBEVAL::CONTEXT *aCtx, const VALUE *b) const override
BOARD_CONNECTED_ITEM * m_item
PCBEXPR_NETCLASS_VALUE(BOARD_CONNECTED_ITEM *aItem)
const wxString & AsString() const override
bool NotEqualTo(LIBEVAL::CONTEXT *aCtx, const LIBEVAL::VALUE *b) const override
LIBEVAL::VALUE * GetValue(LIBEVAL::CONTEXT *aCtx) override
bool EqualTo(LIBEVAL::CONTEXT *aCtx, const VALUE *b) const override
BOARD_CONNECTED_ITEM * m_item
PCBEXPR_NET_VALUE(BOARD_CONNECTED_ITEM *aItem)
bool NotEqualTo(LIBEVAL::CONTEXT *aCtx, const LIBEVAL::VALUE *b) const override
const wxString & AsString() const override
bool EqualTo(LIBEVAL::CONTEXT *aCtx, const VALUE *b) const override
PCBEXPR_PINTYPE_VALUE(const wxString &aPinTypeName)
LIBEVAL::VALUE * GetValue(LIBEVAL::CONTEXT *aCtx) override
virtual std::unique_ptr< LIBEVAL::VAR_REF > CreateVarRef(const wxString &aVar, const wxString &aField) override
virtual LIBEVAL::FUNC_CALL_REF CreateFuncCall(const wxString &aName) override
const std::vector< wxString > & GetSupportedUnits() const override
double Convert(const wxString &aString, int unitId) const override
double Convert(const wxString &aString, int unitId) const override
wxString GetSupportedUnitsMessage() const override
const std::vector< wxString > & GetSupportedUnits() const override
std::unordered_map< TYPE_ID, PROPERTY_BASE * > m_matchingTypes
LIBEVAL::VALUE * GetValue(LIBEVAL::CONTEXT *aCtx) override
LIBEVAL::VAR_TYPE_T m_type
BOARD_ITEM * GetObject(const LIBEVAL::CONTEXT *aCtx) const
virtual size_t TypeHash() const =0
Return type-id of the property type.
virtual bool HasChoices() const
Return true if this PROPERTY has a limited set of possible values.
Definition: property.h:241
Provide class metadata.Helper macro to map type hashes to names.
Definition: property_mgr.h:85
CLASSES_INFO GetAllClasses()
static PROPERTY_MANAGER & Instance()
Definition: property_mgr.h:87
PROPERTY_BASE * GetProperty(TYPE_ID aType, const wxString &aProperty) const
Return a property for a specific type.
bool IsOfType(TYPE_ID aDerived, TYPE_ID aBase) const
Return true if aDerived is inherited from aBase.
@ NULL_CONSTRAINT
Definition: drc_rule.h:48
#define _(s)
wxString LayerName(int aLayer)
Returns the default display name for a given layer.
Definition: layer_id.cpp:30
PCB_LAYER_ID
A quick note on layer IDs:
Definition: layer_ids.h:60
@ F_Cu
Definition: layer_ids.h:64
PCB_LAYER_ID ToLAYER_ID(int aLayer)
Definition: lset.cpp:875
KICOMMON_API double DoubleValueFromString(const EDA_IU_SCALE &aIuScale, EDA_UNITS aUnits, const wxString &aTextValue, EDA_DATA_TYPE aType=EDA_DATA_TYPE::DISTANCE)
Function DoubleValueFromString converts aTextValue to a double.
Definition: eda_units.cpp:576
std::function< void(CONTEXT *, void *)> FUNC_CALL_REF
BOARD * GetBoard()
#define TYPE_HASH(x)
Definition: property.h:71
constexpr ret_type KiROUND(fp_type v, bool aQuiet=false)
Round a floating point number to an integer using "round halfway cases away from zero".
Definition: util.h:100