29#include <drc_rules_lexer.h>
36using namespace DRCRULE_T;
40 DRC_RULES_LEXER( aSource.ToStdString(), aSourceDescr ),
51 wxString first = aMessage.BeforeFirst(
'|', &rest );
55 wxString msg = wxString::Format(
_(
"ERROR: <a href='%d:%d'>%s</a>%s" ),
57 CurOffset() + aOffset,
65 wxString msg = wxString::Format(
_(
"ERROR: %s%s" ), first, rest );
67 THROW_PARSE_ERROR( msg, CurSource(), CurLine(), CurLineNumber(), CurOffset() + aOffset );
76 wxString msg = wxString::Format(
_(
"The '%s' keyword has been deprecated. "
77 "Please use '%s' instead." ),
88 size_t pos = curText.find(
"${" );
90 if( pos == std::string::npos )
93 reportError(
_(
"Unresolved text variable" ), (
int) pos );
102 for(
T token = NextTok(); token != T_EOF; token = NextTok() )
104 if( token == T_LEFT )
107 if( token == T_RIGHT )
120 if( curText.starts_with(
"${" ) )
121 msg.Printf(
_(
"Unresolved text variable." ) );
123 msg.Printf(
_(
"Unrecognized item '%s'.| Expected %s." ), FromUTF8(), expectedTokens );
135 for(
T token = NextTok(); token != T_EOF; token = NextTok() )
137 if( token == T_LEFT )
140 if( token == T_RIGHT )
146 if( !expr.IsEmpty() )
147 expr += CurSeparator();
159 bool haveVersion =
false;
164 for(
T token = NextTok(); token != T_EOF; token = NextTok() )
169 if( token != T_LEFT )
174 if( !haveVersion && token != T_version )
214 expected( wxT(
"rule or version" ) );
226 std::vector<std::shared_ptr<COMPONENT_CLASS_ASSIGNMENT_RULE>>& aRules,
REPORTER* aReporter )
228 bool haveVersion =
false;
233 for(
T token = NextTok(); token != T_EOF; token = NextTok() )
235 if( token != T_LEFT )
240 if( !haveVersion && token != T_version )
271 case T_assign_component_class:
280 expected( wxT(
"assign_component_class or version" ) );
293 std::shared_ptr<DRC_RULE> rule = std::make_shared<DRC_RULE>();
298 if( !IsSymbol( token ) )
302 rule->m_Name = FromUTF8();
304 for( token = NextTok(); token != T_RIGHT && token != T_EOF; token = NextTok() )
309 if( token != T_LEFT )
327 else if( IsSymbol( token ) )
332 if( !rule->m_Condition->Compile(
m_reporter, CurLineNumber(), CurOffset() ) )
333 reportError( wxString::Format(
_(
"Could not parse expression '%s'." ), FromUTF8() ) );
349 rule->m_LayerCondition =
parseLayer( &rule->m_LayerSource );
361 expected( wxT(
"constraint, condition, or disallow" ) );
374 std::shared_ptr<DRC_RULE_CONDITION> condition;
379 if( !IsSymbol( token ) )
383 wxString componentClass = FromUTF8();
385 for( token = NextTok(); token != T_RIGHT && token != T_EOF; token = NextTok() )
387 if( token != T_LEFT )
401 else if( IsSymbol( token ) )
404 condition = std::make_shared<DRC_RULE_CONDITION>( FromUTF8() );
406 if( !condition->Compile(
m_reporter, CurLineNumber(), CurOffset() ) )
407 reportError( wxString::Format(
_(
"Could not parse expression '%s'." ), FromUTF8() ) );
431 return std::make_shared<COMPONENT_CLASS_ASSIGNMENT_RULE>( componentClass, std::move( condition ) );
442 bool allowsTimeDomain =
false;
444 auto validateAndSetValueWithUnits =
445 [
this, &allowsTimeDomain, &
unitsType, &c](
int aValue,
const EDA_UNITS aUnits,
auto aSetter )
450 reportError(
_(
"Time based units not allowed for constraint type." ) );
455 reportError(
_(
"Mixed units for constraint values." ) );
461 if( allowsTimeDomain )
481 if( token == T_mechanical_clearance )
484 token = T_physical_clearance;
486 else if( token == T_mechanical_hole_clearance )
488 reportDeprecation( wxT(
"mechanical_hole_clearance" ), wxT(
"physical_hole_clearance" ) );
489 token = T_physical_hole_clearance;
491 else if( token == T_hole )
496 else if( (
int) token ==
DSN_RIGHT || token == T_EOF )
498 msg.Printf(
_(
"Missing constraint type.| Expected %s." ),
499 wxT(
"assertion, clearance, hole_clearance, edge_clearance, physical_clearance, "
500 "physical_hole_clearance, courtyard_clearance, silk_clearance, hole_size, "
501 "hole_to_hole, track_width, track_angle, track_segment_length, annular_width, "
502 "disallow, zone_connection, thermal_relief_gap, thermal_spoke_width, "
503 "min_resolved_spokes, solder_mask_expansion, solder_paste_abs_margin, "
504 "solder_paste_rel_margin, length, skew, via_count, via_dangling, via_diameter, "
505 "diff_pair_gap or diff_pair_uncoupled" ) );
549 expected( wxT(
"assertion, clearance, hole_clearance, edge_clearance, physical_clearance, "
550 "physical_hole_clearance, courtyard_clearance, silk_clearance, hole_size, "
551 "hole_to_hole, track_width, track_angle, track_segment_length, annular_width, "
552 "disallow, zone_connection, thermal_relief_gap, thermal_spoke_width, "
553 "min_resolved_spokes, solder_mask_expansion, solder_mask_sliver, "
554 "solder_paste_abs_margin, solder_paste_rel_margin, length, skew, via_count, "
555 "via_dangling, via_diameter, diff_pair_gap, diff_pair_uncoupled or bridged_mask" ) );
561 msg.Printf(
_(
"Rule already has a '%s' constraint." ), FromUTF8() );
575 for( token = NextTok(); token != T_RIGHT; token = NextTok() )
578 token = GetCurStrAsToken();
603 expected( wxT(
"track, via, through_via, blind_via, micro_via, buried_via, pad, zone, text, "
604 "graphic, hole, or footprint." ) );
620 token = GetCurStrAsToken();
633 expected( wxT(
"solid, thermal_reliefs or none." ) );
654 value = (int) strtol( CurText(),
nullptr, 10 );
675 if( IsSymbol( token ) )
692 for( token = NextTok(); token != T_RIGHT && token != T_EOF; token = NextTok() )
694 if( token != T_LEFT )
701 case T_within_diff_pairs:
705 reportError(
_(
"within_diff_pairs option invalid for constraint type." ) );
714 size_t offset = CurOffset() + GetTokenString( token ).length();
724 validateAndSetValueWithUnits( value, units,
725 [&c](
const int aValue )
735 size_t offset = CurOffset() + GetTokenString( token ).length();
745 validateAndSetValueWithUnits( value, units,
746 [&c](
const int aValue )
756 size_t offset = CurOffset() + GetTokenString( token ).length();
766 validateAndSetValueWithUnits( value, units,
767 [&c](
const int aValue )
780 expected( wxT(
"min, max, opt, or within_diff_pairs" ) );
795 [&](
const wxString& message,
int offset )
798 wxString first = message.BeforeFirst(
'|', &rest );
802 wxString msg = wxString::Format(
_(
"ERROR: <a href='%d:%d'>%s</a>%s" ),
812 wxString msg = wxString::Format(
_(
"ERROR: %s%s" ), first, rest );
815 CurOffset() + aOffset );
825 aResult = evaluator.
Result();
826 aUnits = evaluator.
Units();
834 int token = NextTok();
841 else if( token == T_outer )
843 *aSource = GetTokenString( token );
846 else if( token == T_inner )
848 *aSource = GetTokenString( token );
853 wxString layerName = FromUTF8();
856 for(
unsigned ii = 0; ii < layerMap.GetCount(); ++ii )
858 wxPGChoiceEntry& entry = layerMap[ii];
860 if( entry.GetText().Matches( layerName ) )
862 *aSource = layerName;
870 reportError( wxString::Format(
_(
"Unrecognized layer '%s'." ), layerName ) );
890 if( (
int) token ==
DSN_RIGHT || token == T_EOF )
904 expected( wxT(
"ignore, warning, error, or exclusion" ) );
BASE_SET & set(size_t pos)
DRC_RULE_CONDITION * m_Test
void SetOption(OPTIONS option)
ZONE_CONNECTION m_ZoneConnection
void ClearOption(OPTIONS option)
wxString parseExpression()
std::shared_ptr< COMPONENT_CLASS_ASSIGNMENT_RULE > parseComponentClassAssignment()
void reportDeprecation(const wxString &oldToken, const wxString &newToken)
void Parse(std::vector< std::shared_ptr< DRC_RULE > > &aRules, REPORTER *aReporter)
void reportError(const wxString &aMessage, int aOffset=0)
void parseValueWithUnits(int aOffset, const wxString &aExpr, int &aResult, EDA_UNITS &aUnits, bool aUnitless=false)
void ParseComponentClassAssignmentRules(std::vector< std::shared_ptr< COMPONENT_CLASS_ASSIGNMENT_RULE > > &aRules, REPORTER *aReporter)
void expected(const wxString &expectedTokens)
bool checkUnresolvedTextVariable()
std::shared_ptr< DRC_RULE > parseDRC_RULE()
LSET parseLayer(wxString *aSource)
void parseConstraint(DRC_RULE *aRule)
DRC_RULES_PARSER(const wxString &aSource, const wxString &aSourceDescr)
bool Compile(REPORTER *aReporter, int aSourceLine=0, int aSourceOffset=0)
void AddConstraint(DRC_CONSTRAINT &aConstraint)
std::optional< DRC_CONSTRAINT > FindConstraint(DRC_CONSTRAINT_T aType)
static ENUM_MAP< T > & Instance()
LSET is a set of PCB_LAYER_IDs.
static const LSET & ExternalCuMask()
Return a mask holding the Front and Bottom layers.
static const LSET & AllLayersMask()
static LSET AllCuMask()
return AllCuMask( MAX_CU_LAYERS );
static const LSET & InternalCuMask()
Return a complete set of internal copper layers which is all Cu layers except F_Cu and B_Cu.
bool Evaluate(const wxString &aExpr)
void SetErrorCallback(std::function< void(const wxString &aMessage, int aOffset)> aCallback)
A pure virtual class used to derive REPORTER objects from.
static EDA_DATA_TYPE GetTypeFromUnits(const EDA_UNITS aUnits)
Gets the inferred type from the given units.
@ DRC_DISALLOW_BURIED_VIAS
@ DRC_DISALLOW_BLIND_VIAS
@ DRC_DISALLOW_THROUGH_VIAS
@ DRC_DISALLOW_FOOTPRINTS
@ DRC_DISALLOW_MICRO_VIAS
@ ANNULAR_WIDTH_CONSTRAINT
@ BRIDGED_MASK_CONSTRAINT
@ COURTYARD_CLEARANCE_CONSTRAINT
@ VIA_DIAMETER_CONSTRAINT
@ ZONE_CONNECTION_CONSTRAINT
@ DIFF_PAIR_GAP_CONSTRAINT
@ VIA_DANGLING_CONSTRAINT
@ SOLDER_MASK_SLIVER_CONSTRAINT
@ SILK_CLEARANCE_CONSTRAINT
@ EDGE_CLEARANCE_CONSTRAINT
@ MIN_RESOLVED_SPOKES_CONSTRAINT
@ TRACK_SEGMENT_LENGTH_CONSTRAINT
@ TEXT_THICKNESS_CONSTRAINT
@ PHYSICAL_HOLE_CLEARANCE_CONSTRAINT
@ THERMAL_SPOKE_WIDTH_CONSTRAINT
@ CONNECTION_WIDTH_CONSTRAINT
@ THERMAL_RELIEF_GAP_CONSTRAINT
@ MAX_UNCOUPLED_CONSTRAINT
@ HOLE_CLEARANCE_CONSTRAINT
@ SOLDER_PASTE_ABS_MARGIN_CONSTRAINT
@ SOLDER_MASK_EXPANSION_CONSTRAINT
@ PHYSICAL_CLEARANCE_CONSTRAINT
@ SOLDER_PASTE_REL_MARGIN_CONSTRAINT
@ HOLE_TO_HOLE_CONSTRAINT
#define DRC_RULE_FILE_VERSION
EDA_DATA_TYPE
The type of unit.
#define THROW_PARSE_ERROR(aProblem, aSource, aInputLine, aLineNumber, aByteIndex)
PCB_LAYER_ID ToLAYER_ID(int aLayer)
VECTOR3I expected(15, 30, 45)
@ THERMAL
Use thermal relief for pads.
@ NONE
Pads are not covered.
@ FULL
pads are covered by copper