KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
drc_rule_parser.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, 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 <board.h>
26#include <zones.h>
27#include <drc/drc_rule_parser.h>
29#include <drc_rules_lexer.h>
30#include <pcbexpr_evaluator.h>
31#include <reporter.h>
33
34using namespace DRCRULE_T;
35
36
37DRC_RULES_PARSER::DRC_RULES_PARSER( const wxString& aSource, const wxString& aSourceDescr ) :
38 DRC_RULES_LEXER( aSource.ToStdString(), aSourceDescr ),
39 m_requiredVersion( 0 ),
40 m_tooRecent( false ),
41 m_reporter( nullptr )
42{
43}
44
45
46DRC_RULES_PARSER::DRC_RULES_PARSER( FILE* aFile, const wxString& aFilename ) :
47 DRC_RULES_LEXER( aFile, aFilename ),
48 m_requiredVersion( 0 ),
49 m_tooRecent( false ),
50 m_reporter( nullptr )
51{
52}
53
54
55void DRC_RULES_PARSER::reportError( const wxString& aMessage )
56{
57 wxString rest;
58 wxString first = aMessage.BeforeFirst( '|', &rest );
59
60 if( m_reporter )
61 {
62 wxString msg = wxString::Format( _( "ERROR: <a href='%d:%d'>%s</a>%s" ), CurLineNumber(),
63 CurOffset(), first, rest );
64
66 }
67 else
68 {
69 wxString msg = wxString::Format( _( "ERROR: %s%s" ), first, rest );
70
71 THROW_PARSE_ERROR( msg, CurSource(), CurLine(), CurLineNumber(), CurOffset() );
72 }
73}
74
75
76void DRC_RULES_PARSER::reportDeprecation( const wxString& oldToken, const wxString newToken )
77{
78 if( m_reporter )
79 {
80 wxString msg = wxString::Format( _( "The '%s' keyword has been deprecated. "
81 "Please use '%s' instead." ),
82 oldToken,
83 newToken);
84
86 }
87}
88
89
91{
92 int depth = 1;
93
94 for( T token = NextTok(); token != T_EOF; token = NextTok() )
95 {
96 if( token == T_LEFT )
97 depth++;
98
99 if( token == T_RIGHT )
100 {
101 if( --depth == 0 )
102 break;
103 }
104 }
105}
106
107
108void DRC_RULES_PARSER::Parse( std::vector<std::shared_ptr<DRC_RULE>>& aRules, REPORTER* aReporter )
109{
110 bool haveVersion = false;
111 wxString msg;
112
113 m_reporter = aReporter;
114
115 for( T token = NextTok(); token != T_EOF; token = NextTok() )
116 {
117 if( token != T_LEFT )
118 reportError( _( "Missing '('." ) );
119
120 token = NextTok();
121
122 if( !haveVersion && token != T_version )
123 {
124 reportError( _( "Missing version statement." ) );
125 haveVersion = true; // don't keep on reporting it
126 }
127
128 switch( token )
129 {
130 case T_version:
131 haveVersion = true;
132 token = NextTok();
133
134 if( (int) token == DSN_RIGHT )
135 {
136 reportError( _( "Missing version number." ) );
137 break;
138 }
139
140 if( (int) token == DSN_NUMBER )
141 {
142 m_requiredVersion = (int)strtol( CurText(), nullptr, 10 );
144 token = NextTok();
145 }
146 else
147 {
148 msg.Printf( _( "Unrecognized item '%s'.| Expected version number." ),
149 FromUTF8() );
150 reportError( msg );
151 }
152
153 if( (int) token != DSN_RIGHT )
154 {
155 msg.Printf( _( "Unrecognized item '%s'." ),
156 FromUTF8() );
157 reportError( msg );
158 parseUnknown();
159 }
160
161 break;
162
163 case T_rule:
164 aRules.emplace_back( parseDRC_RULE() );
165 break;
166
167 case T_EOF:
168 reportError( _( "Incomplete statement." ) );
169 break;
170
171 default:
172 msg.Printf( _( "Unrecognized item '%s'.| Expected %s." ), FromUTF8(),
173 wxT( "rule or version" ) );
174 reportError( msg );
175 parseUnknown();
176 }
177 }
178
179 if( m_reporter && !m_reporter->HasMessage() )
180 m_reporter->Report( _( "No errors found." ), RPT_SEVERITY_INFO );
181
182 m_reporter = nullptr;
183}
184
185
187 std::vector<std::shared_ptr<COMPONENT_CLASS_ASSIGNMENT_RULE>>& aRules, REPORTER* aReporter )
188{
189 bool haveVersion = false;
190 wxString msg;
191
192 m_reporter = aReporter;
193
194 for( T token = NextTok(); token != T_EOF; token = NextTok() )
195 {
196 if( token != T_LEFT )
197 reportError( _( "Missing '('." ) );
198
199 token = NextTok();
200
201 if( !haveVersion && token != T_version )
202 {
203 reportError( _( "Missing version statement." ) );
204 haveVersion = true; // don't keep on reporting it
205 }
206
207 switch( token )
208 {
209 case T_version:
210 haveVersion = true;
211 token = NextTok();
212
213 if( (int) token == DSN_RIGHT )
214 {
215 reportError( _( "Missing version number." ) );
216 break;
217 }
218
219 if( (int) token == DSN_NUMBER )
220 {
221 m_requiredVersion = (int) strtol( CurText(), nullptr, 10 );
223 token = NextTok();
224 }
225 else
226 {
227 msg.Printf( _( "Unrecognized item '%s'.| Expected version number." ), FromUTF8() );
228 reportError( msg );
229 }
230
231 if( (int) token != DSN_RIGHT )
232 {
233 msg.Printf( _( "Unrecognized item '%s'." ), FromUTF8() );
234 reportError( msg );
235 parseUnknown();
236 }
237
238 break;
239
240 case T_assign_component_class:
241 aRules.emplace_back( parseComponentClassAssignment() );
242 break;
243
244 case T_EOF: reportError( _( "Incomplete statement." ) ); break;
245
246 default:
247 msg.Printf( _( "Unrecognized item '%s'.| Expected %s." ), FromUTF8(),
248 wxT( "assign_component_class or version" ) );
249 reportError( msg );
250 parseUnknown();
251 }
252 }
253
254 if( m_reporter && !m_reporter->HasMessage() )
255 m_reporter->Report( _( "No errors found." ), RPT_SEVERITY_INFO );
256
257 m_reporter = nullptr;
258}
259
260
261std::shared_ptr<DRC_RULE> DRC_RULES_PARSER::parseDRC_RULE()
262{
263 std::shared_ptr<DRC_RULE> rule = std::make_shared<DRC_RULE>();
264
265 T token = NextTok();
266 wxString msg;
267
268 if( !IsSymbol( token ) )
269 reportError( _( "Missing rule name." ) );
270
271 rule->m_Name = FromUTF8();
272
273 for( token = NextTok(); token != T_RIGHT && token != T_EOF; token = NextTok() )
274 {
275 if( token != T_LEFT )
276 reportError( _( "Missing '('." ) );
277
278 token = NextTok();
279
280 switch( token )
281 {
282 case T_constraint:
283 parseConstraint( rule.get() );
284 break;
285
286 case T_condition:
287 token = NextTok();
288
289 if( (int) token == DSN_RIGHT )
290 {
291 reportError( _( "Missing condition expression." ) );
292 break;
293 }
294
295 if( IsSymbol( token ) )
296 {
297 rule->m_Condition = new DRC_RULE_CONDITION( FromUTF8() );
298 rule->m_Condition->Compile( m_reporter, CurLineNumber(), CurOffset() );
299 }
300 else
301 {
302 msg.Printf( _( "Unrecognized item '%s'.| Expected quoted expression." ),
303 FromUTF8() );
304 reportError( msg );
305 }
306
307 if( (int) NextTok() != DSN_RIGHT )
308 {
309 reportError( wxString::Format( _( "Unrecognized item '%s'." ), FromUTF8() ) );
310 parseUnknown();
311 }
312
313 break;
314
315 case T_layer:
316 if( rule->m_LayerCondition != LSET::AllLayersMask() )
317 reportError( _( "'layer' keyword already present." ) );
318
319 rule->m_LayerCondition = parseLayer( &rule->m_LayerSource );
320 break;
321
322 case T_severity:
323 rule->m_Severity = parseSeverity();
324 break;
325
326 case T_EOF:
327 reportError( _( "Incomplete statement." ) );
328 return rule;
329
330 default:
331 msg.Printf( _( "Unrecognized item '%s'.| Expected %s." ), FromUTF8(),
332 wxT( "constraint, condition, or disallow" ) );
333 reportError( msg );
334 parseUnknown();
335 }
336 }
337
338 if( (int) CurTok() != DSN_RIGHT )
339 reportError( _( "Missing ')'." ) );
340
341 return rule;
342}
343
344
345std::shared_ptr<COMPONENT_CLASS_ASSIGNMENT_RULE> DRC_RULES_PARSER::parseComponentClassAssignment()
346{
347 std::shared_ptr<DRC_RULE_CONDITION> condition;
348
349 T token = NextTok();
350 wxString msg;
351
352 if( !IsSymbol( token ) )
353 reportError( _( "Missing component class name." ) );
354
355 wxString componentClass = FromUTF8();
356
357 for( token = NextTok(); token != T_RIGHT && token != T_EOF; token = NextTok() )
358 {
359 if( token != T_LEFT )
360 reportError( _( "Missing '('." ) );
361
362 token = NextTok();
363
364 switch( token )
365 {
366 case T_condition:
367 token = NextTok();
368
369 if( (int) token == DSN_RIGHT )
370 {
371 reportError( _( "Missing condition expression." ) );
372 break;
373 }
374
375 if( IsSymbol( token ) )
376 {
377 condition = std::make_shared<DRC_RULE_CONDITION>( FromUTF8() );
378
379 if( !condition->Compile( m_reporter, CurLineNumber(), CurOffset() ) )
380 {
381 reportError( wxString::Format( _( "Could not parse expression '%s'." ),
382 FromUTF8() ) );
383 }
384 }
385 else
386 {
387 msg.Printf( _( "Unrecognized item '%s'.| Expected quoted expression." ),
388 FromUTF8() );
389 reportError( msg );
390 }
391
392 if( (int) NextTok() != DSN_RIGHT )
393 {
394 reportError( wxString::Format( _( "Unrecognized item '%s'." ), FromUTF8() ) );
395 parseUnknown();
396 }
397
398 break;
399
400 case T_EOF: reportError( _( "Incomplete statement." ) ); return nullptr;
401
402 default:
403 msg.Printf( _( "Unrecognized item '%s'.| Expected %s." ), FromUTF8(),
404 wxT( "condition" ) );
405 reportError( msg );
406 parseUnknown();
407 }
408 }
409
410 if( (int) CurTok() != DSN_RIGHT )
411 reportError( _( "Missing ')'." ) );
412
413 return std::make_shared<COMPONENT_CLASS_ASSIGNMENT_RULE>( componentClass,
414 std::move( condition ) );
415 ;
416}
417
418
420{
422 int value;
423 wxString msg;
424
425 T token = NextTok();
426
427 if( token == T_mechanical_clearance )
428 {
429 reportDeprecation( wxT( "mechanical_clearance" ), wxT( "physical_clearance" ) );
430 token = T_physical_clearance;
431 }
432 else if( token == T_mechanical_hole_clearance )
433 {
434 reportDeprecation( wxT( "mechanical_hole_clearance" ), wxT( "physical_hole_clearance" ) );
435 token = T_physical_hole_clearance;
436 }
437 else if( token == T_hole )
438 {
439 reportDeprecation( wxT( "hole" ), wxT( "hole_size" ) );
440 token = T_hole_size;
441 }
442 else if( (int) token == DSN_RIGHT || token == T_EOF )
443 {
444 msg.Printf( _( "Missing constraint type.| Expected %s." ),
445 wxT( "assertion, clearance, hole_clearance, edge_clearance, "
446 "physical_clearance, physical_hole_clearance, courtyard_clearance, "
447 "silk_clearance, hole_size, hole_to_hole, track_width, annular_width, "
448 "via_diameter, disallow, zone_connection, thermal_relief_gap, "
449 "thermal_spoke_width, min_resolved_spokes, length, skew, via_count, "
450 "diff_pair_gap or diff_pair_uncoupled" ) );
451 reportError( msg );
452 return;
453 }
454
455 switch( token )
456 {
457 case T_assertion: c.m_Type = ASSERTION_CONSTRAINT; break;
458 case T_clearance: c.m_Type = CLEARANCE_CONSTRAINT; break;
459 case T_creepage: c.m_Type = CREEPAGE_CONSTRAINT; break;
460 case T_hole_clearance: c.m_Type = HOLE_CLEARANCE_CONSTRAINT; break;
461 case T_edge_clearance: c.m_Type = EDGE_CLEARANCE_CONSTRAINT; break;
462 case T_hole_size: c.m_Type = HOLE_SIZE_CONSTRAINT; break;
463 case T_hole_to_hole: c.m_Type = HOLE_TO_HOLE_CONSTRAINT; break;
464 case T_courtyard_clearance: c.m_Type = COURTYARD_CLEARANCE_CONSTRAINT; break;
465 case T_silk_clearance: c.m_Type = SILK_CLEARANCE_CONSTRAINT; break;
466 case T_text_height: c.m_Type = TEXT_HEIGHT_CONSTRAINT; break;
467 case T_text_thickness: c.m_Type = TEXT_THICKNESS_CONSTRAINT; break;
468 case T_track_width: c.m_Type = TRACK_WIDTH_CONSTRAINT; break;
469 case T_track_angle: c.m_Type = TRACK_ANGLE_CONSTRAINT; break;
470 case T_track_segment_length: c.m_Type = TRACK_SEGMENT_LENGTH_CONSTRAINT; break;
471 case T_connection_width: c.m_Type = CONNECTION_WIDTH_CONSTRAINT; break;
472 case T_annular_width: c.m_Type = ANNULAR_WIDTH_CONSTRAINT; break;
473 case T_via_diameter: c.m_Type = VIA_DIAMETER_CONSTRAINT; break;
474 case T_zone_connection: c.m_Type = ZONE_CONNECTION_CONSTRAINT; break;
475 case T_thermal_relief_gap: c.m_Type = THERMAL_RELIEF_GAP_CONSTRAINT; break;
476 case T_thermal_spoke_width: c.m_Type = THERMAL_SPOKE_WIDTH_CONSTRAINT; break;
477 case T_min_resolved_spokes: c.m_Type = MIN_RESOLVED_SPOKES_CONSTRAINT; break;
478 case T_disallow: c.m_Type = DISALLOW_CONSTRAINT; break;
479 case T_length: c.m_Type = LENGTH_CONSTRAINT; break;
480 case T_skew: c.m_Type = SKEW_CONSTRAINT; break;
481 case T_via_count: c.m_Type = VIA_COUNT_CONSTRAINT; break;
482 case T_diff_pair_gap: c.m_Type = DIFF_PAIR_GAP_CONSTRAINT; break;
483 case T_diff_pair_uncoupled: c.m_Type = MAX_UNCOUPLED_CONSTRAINT; break;
484 case T_physical_clearance: c.m_Type = PHYSICAL_CLEARANCE_CONSTRAINT; break;
485 case T_physical_hole_clearance: c.m_Type = PHYSICAL_HOLE_CLEARANCE_CONSTRAINT; break;
486 default:
487 msg.Printf( _( "Unrecognized item '%s'.| Expected %s." ), FromUTF8(),
488 wxT( "assertion, clearance, hole_clearance, edge_clearance, "
489 "physical_clearance, physical_hole_clearance, courtyard_clearance, "
490 "silk_clearance, hole_size, hole_to_hole, track_width, track_angle, track_segment_length, annular_width, "
491 "disallow, zone_connection, thermal_relief_gap, thermal_spoke_width, "
492 "min_resolved_spokes, length, skew, via_count, via_diameter, "
493 "diff_pair_gap or diff_pair_uncoupled" ) );
494 reportError( msg );
495 }
496
497 if( aRule->FindConstraint( c.m_Type ) )
498 {
499 msg.Printf( _( "Rule already has a '%s' constraint." ), FromUTF8() );
500 reportError( msg );
501 }
502
503 bool unitless = c.m_Type == VIA_COUNT_CONSTRAINT
506
507 if( c.m_Type == DISALLOW_CONSTRAINT )
508 {
509 for( token = NextTok(); token != T_RIGHT; token = NextTok() )
510 {
511 if( (int) token == DSN_STRING )
512 token = GetCurStrAsToken();
513
514 switch( token )
515 {
516 case T_track: c.m_DisallowFlags |= DRC_DISALLOW_TRACKS; break;
517 case T_via: c.m_DisallowFlags |= DRC_DISALLOW_VIAS; break;
518 case T_micro_via: c.m_DisallowFlags |= DRC_DISALLOW_MICRO_VIAS; break;
519 case T_buried_via: c.m_DisallowFlags |= DRC_DISALLOW_BB_VIAS; break;
520 case T_pad: c.m_DisallowFlags |= DRC_DISALLOW_PADS; break;
521 case T_zone: c.m_DisallowFlags |= DRC_DISALLOW_ZONES; break;
522 case T_text: c.m_DisallowFlags |= DRC_DISALLOW_TEXTS; break;
523 case T_graphic: c.m_DisallowFlags |= DRC_DISALLOW_GRAPHICS; break;
524 case T_hole: c.m_DisallowFlags |= DRC_DISALLOW_HOLES; break;
525 case T_footprint: c.m_DisallowFlags |= DRC_DISALLOW_FOOTPRINTS; break;
526
527 case T_EOF:
528 reportError( _( "Missing ')'." ) );
529 return;
530
531 default:
532 msg.Printf( _( "Unrecognized item '%s'.| Expected %s." ), FromUTF8(),
533 wxT( "track, via, micro_via, buried_via, pad, zone, text, graphic, "
534 "hole, or footprint." ) );
535 reportError( msg );
536 break;
537 }
538 }
539
540 if( (int) CurTok() != DSN_RIGHT )
541 reportError( _( "Missing ')'." ) );
542
543 aRule->AddConstraint( c );
544 return;
545 }
546 else if( c.m_Type == ZONE_CONNECTION_CONSTRAINT )
547 {
548 token = NextTok();
549
550 if( (int) token == DSN_STRING )
551 token = GetCurStrAsToken();
552
553 switch( token )
554 {
555 case T_solid: c.m_ZoneConnection = ZONE_CONNECTION::FULL; break;
556 case T_thermal_reliefs: c.m_ZoneConnection = ZONE_CONNECTION::THERMAL; break;
557 case T_none: c.m_ZoneConnection = ZONE_CONNECTION::NONE; break;
558
559 case T_EOF:
560 reportError( _( "Missing ')'." ) );
561 return;
562
563 default:
564 msg.Printf( _( "Unrecognized item '%s'.| Expected %s." ), FromUTF8(),
565 "solid, thermal_reliefs or none." );
566 reportError( msg );
567 break;
568 }
569
570 if( (int) NextTok() != DSN_RIGHT )
571 reportError( _( "Missing ')'." ) );
572
573 aRule->AddConstraint( c );
574 return;
575 }
577 {
578 // We don't use a min/max/opt structure here because it would give a strong implication
579 // that you could specify the optimal number of spokes. We don't want to open that door
580 // because the spoke generator is highly optimized around being able to "cheat" off of a
581 // cartesian coordinate system.
582
583 token = NextTok();
584
585 if( (int) token == DSN_NUMBER )
586 {
587 value = (int) strtol( CurText(), nullptr, 10 );
588 c.m_Value.SetMin( value );
589 }
590 else
591 {
592 reportError( _( "Expecting number." ) );
593 parseUnknown();
594 }
595
596 if( (int) NextTok() != DSN_RIGHT )
597 reportError( _( "Missing ')'." ) );
598
599 aRule->AddConstraint( c );
600 return;
601 }
602 else if( c.m_Type == ASSERTION_CONSTRAINT )
603 {
604 token = NextTok();
605
606 if( (int) token == DSN_RIGHT )
607 reportError( _( "Missing assertion expression." ) );
608
609 if( IsSymbol( token ) )
610 {
611 c.m_Test = new DRC_RULE_CONDITION( FromUTF8() );
612 c.m_Test->Compile( m_reporter, CurLineNumber(), CurOffset() );
613 }
614 else
615 {
616 msg.Printf( _( "Unrecognized item '%s'.| Expected quoted expression." ), FromUTF8() );
617 reportError( msg );
618 }
619
620 if( (int) NextTok() != DSN_RIGHT )
621 {
622 reportError( wxString::Format( _( "Unrecognized item '%s'." ), FromUTF8() ) );
623 parseUnknown();
624 }
625
626 aRule->AddConstraint( c );
627 return;
628 }
629
630 for( token = NextTok(); token != T_RIGHT && token != T_EOF; token = NextTok() )
631 {
632 if( token != T_LEFT )
633 reportError( _( "Missing '('." ) );
634
635 token = NextTok();
636
637 switch( token )
638 {
639 case T_within_diff_pairs:
640 if( c.m_Type != SKEW_CONSTRAINT )
641 {
642 reportError( _( "within_diff_pairs option invalid for constraint type." ) );
643 break;
644 }
645
647
648 if( (int) NextTok() != DSN_RIGHT )
649 {
650 reportError( wxString::Format( _( "Unrecognized item '%s'." ), FromUTF8() ) );
651 parseUnknown();
652 }
653
654 break;
655 case T_min:
656 token = NextTok();
657
658 if( (int) token == DSN_RIGHT )
659 {
660 reportError( _( "Missing min value." ) );
661 break;
662 }
663
664 parseValueWithUnits( FromUTF8(), value, unitless );
665 c.m_Value.SetMin( value );
666
667 if( (int) NextTok() != DSN_RIGHT )
668 {
669 reportError( wxString::Format( _( "Unrecognized item '%s'." ), FromUTF8() ) );
670 parseUnknown();
671 }
672
673 break;
674
675 case T_max:
676 token = NextTok();
677
678 if( (int) token == DSN_RIGHT )
679 {
680 reportError( _( "Missing max value." ) );
681 break;
682 }
683
684 parseValueWithUnits( FromUTF8(), value, unitless );
685
686 c.m_Value.SetMax( value );
687
688 if( (int) NextTok() != DSN_RIGHT )
689 {
690 reportError( wxString::Format( _( "Unrecognized item '%s'." ), FromUTF8() ) );
691 parseUnknown();
692 }
693
694 break;
695
696 case T_opt:
697 token = NextTok();
698
699 if( (int) token == DSN_RIGHT )
700 {
701 reportError( _( "Missing opt value." ) );
702 break;
703 }
704
705 parseValueWithUnits( FromUTF8(), value, unitless );
706 c.m_Value.SetOpt( value );
707
708 if( (int) NextTok() != DSN_RIGHT )
709 {
710 reportError( wxString::Format( _( "Unrecognized item '%s'." ), FromUTF8() ) );
711 parseUnknown();
712 }
713
714 break;
715
716 case T_EOF:
717 reportError( _( "Incomplete statement." ) );
718 return;
719
720 default:
721 msg.Printf( _( "Unrecognized item '%s'.| Expected %s." ),
722 FromUTF8(),
723 wxT( "min, max, or opt" ) );
724 reportError( msg );
725 parseUnknown();
726 }
727 }
728
729 if( (int) CurTok() != DSN_RIGHT )
730 reportError( _( "Missing ')'." ) );
731
732 aRule->AddConstraint( c );
733}
734
735
736void DRC_RULES_PARSER::parseValueWithUnits( const wxString& aExpr, int& aResult, bool aUnitless )
737{
738 auto errorHandler = [&]( const wxString& aMessage, int aOffset )
739 {
740 wxString rest;
741 wxString first = aMessage.BeforeFirst( '|', &rest );
742
743 if( m_reporter )
744 {
745 wxString msg = wxString::Format( _( "ERROR: <a href='%d:%d'>%s</a>%s" ),
746 CurLineNumber(), CurOffset() + aOffset, first, rest );
747
749 }
750 else
751 {
752 wxString msg = wxString::Format( _( "ERROR: %s%s" ), first, rest );
753
754 THROW_PARSE_ERROR( msg, CurSource(), CurLine(), CurLineNumber(),
755 CurOffset() + aOffset );
756 }
757 };
758
761 evaluator.SetErrorCallback( errorHandler );
762
763 evaluator.Evaluate( aExpr );
764 aResult = evaluator.Result();
765}
766
767
769{
770 LSET retVal;
771 int token = NextTok();
772
773 if( (int) token == DSN_RIGHT )
774 {
775 reportError( _( "Missing layer name or type." ) );
776 return LSET::AllCuMask();
777 }
778 else if( token == T_outer )
779 {
780 *aSource = GetTokenString( token );
781 retVal = LSET::ExternalCuMask();
782 }
783 else if( token == T_inner )
784 {
785 *aSource = GetTokenString( token );
786 retVal = LSET::InternalCuMask();
787 }
788 else
789 {
790 wxString layerName = FromUTF8();
791 wxPGChoices& layerMap = ENUM_MAP<PCB_LAYER_ID>::Instance().Choices();
792
793 for( unsigned ii = 0; ii < layerMap.GetCount(); ++ii )
794 {
795 wxPGChoiceEntry& entry = layerMap[ii];
796
797 if( entry.GetText().Matches( layerName ) )
798 {
799 *aSource = layerName;
800 retVal.set( ToLAYER_ID( entry.GetValue() ) );
801 }
802 }
803
804 if( !retVal.any() )
805 {
806 reportError( wxString::Format( _( "Unrecognized layer '%s'." ), layerName ) );
807 retVal.set( Rescue );
808 }
809 }
810
811 if( (int) NextTok() != DSN_RIGHT )
812 {
813 reportError( wxString::Format( _( "Unrecognized item '%s'." ), FromUTF8() ) );
814 parseUnknown();
815 }
816
817 return retVal;
818}
819
820
822{
824 wxString msg;
825
826 T token = NextTok();
827
828 if( (int) token == DSN_RIGHT || token == T_EOF )
829 {
830 reportError( _( "Missing severity name." ) );
832 }
833
834 switch( token )
835 {
836 case T_ignore: retVal = RPT_SEVERITY_IGNORE; break;
837 case T_warning: retVal = RPT_SEVERITY_WARNING; break;
838 case T_error: retVal = RPT_SEVERITY_ERROR; break;
839 case T_exclusion: retVal = RPT_SEVERITY_EXCLUSION; break;
840
841 default:
842 msg.Printf( _( "Unrecognized item '%s'.| Expected %s." ),
843 FromUTF8(),
844 wxT( "ignore, warning, error, or exclusion" ) );
845 reportError( msg );
846 parseUnknown();
847 }
848
849 if( (int) NextTok() != DSN_RIGHT )
850 reportError( _( "Missing ')'." ) );
851
852 return retVal;
853}
BASE_SET & set(size_t pos)
Definition: base_set.h:116
DRC_RULE_CONDITION * m_Test
Definition: drc_rule.h:193
int m_DisallowFlags
Definition: drc_rule.h:191
void SetOption(OPTIONS option)
Definition: drc_rule.h:181
ZONE_CONNECTION m_ZoneConnection
Definition: drc_rule.h:192
MINOPTMAX< int > m_Value
Definition: drc_rule.h:190
DRC_CONSTRAINT_T m_Type
Definition: drc_rule.h:189
SEVERITY parseSeverity()
std::shared_ptr< COMPONENT_CLASS_ASSIGNMENT_RULE > parseComponentClassAssignment()
void parseValueWithUnits(const wxString &aExpr, int &aResult, bool aUnitless=false)
void Parse(std::vector< std::shared_ptr< DRC_RULE > > &aRules, REPORTER *aReporter)
REPORTER * m_reporter
void ParseComponentClassAssignmentRules(std::vector< std::shared_ptr< COMPONENT_CLASS_ASSIGNMENT_RULE > > &aRules, REPORTER *aReporter)
std::shared_ptr< DRC_RULE > parseDRC_RULE()
LSET parseLayer(wxString *aSource)
void parseConstraint(DRC_RULE *aRule)
void reportError(const wxString &aMessage)
DRC_RULES_PARSER(const wxString &aSource, const wxString &aSourceDescr)
void reportDeprecation(const wxString &oldToken, const wxString newToken)
bool Compile(REPORTER *aReporter, int aSourceLine=0, int aSourceOffset=0)
void AddConstraint(DRC_CONSTRAINT &aConstraint)
Definition: drc_rule.cpp:60
std::optional< DRC_CONSTRAINT > FindConstraint(DRC_CONSTRAINT_T aType)
Definition: drc_rule.cpp:67
static ENUM_MAP< T > & Instance()
Definition: property.h:680
LSET is a set of PCB_LAYER_IDs.
Definition: lset.h:37
static LSET ExternalCuMask()
Return a mask holding the Front and Bottom layers.
Definition: lset.cpp:594
static LSET AllLayersMask()
Definition: lset.cpp:601
static LSET InternalCuMask()
Return a complete set of internal copper layers which is all Cu layers except F_Cu and B_Cu.
Definition: lset.cpp:561
static LSET AllCuMask(int aCuLayerCount=MAX_CU_LAYERS)
Return a mask holding the requested number of Cu PCB_LAYER_IDs.
Definition: lset.cpp:572
void SetMin(T v)
Definition: minoptmax.h:41
void SetOpt(T v)
Definition: minoptmax.h:43
void SetMax(T v)
Definition: minoptmax.h:42
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.
Definition: reporter.h:73
virtual bool HasMessage() const =0
Returns true if the reporter client is non-empty.
virtual REPORTER & Report(const wxString &aText, SEVERITY aSeverity=RPT_SEVERITY_UNDEFINED)
Report a string with a given severity.
Definition: reporter.h:102
@ DRC_DISALLOW_PADS
Definition: drc_rule.h:88
@ DRC_DISALLOW_VIAS
Definition: drc_rule.h:84
@ DRC_DISALLOW_TEXTS
Definition: drc_rule.h:90
@ DRC_DISALLOW_ZONES
Definition: drc_rule.h:89
@ DRC_DISALLOW_HOLES
Definition: drc_rule.h:92
@ DRC_DISALLOW_GRAPHICS
Definition: drc_rule.h:91
@ DRC_DISALLOW_FOOTPRINTS
Definition: drc_rule.h:93
@ DRC_DISALLOW_TRACKS
Definition: drc_rule.h:87
@ DRC_DISALLOW_MICRO_VIAS
Definition: drc_rule.h:85
@ DRC_DISALLOW_BB_VIAS
Definition: drc_rule.h:86
@ ANNULAR_WIDTH_CONSTRAINT
Definition: drc_rule.h:61
@ COURTYARD_CLEARANCE_CONSTRAINT
Definition: drc_rule.h:55
@ VIA_DIAMETER_CONSTRAINT
Definition: drc_rule.h:67
@ ZONE_CONNECTION_CONSTRAINT
Definition: drc_rule.h:62
@ DIFF_PAIR_GAP_CONSTRAINT
Definition: drc_rule.h:70
@ DISALLOW_CONSTRAINT
Definition: drc_rule.h:66
@ TRACK_WIDTH_CONSTRAINT
Definition: drc_rule.h:59
@ SILK_CLEARANCE_CONSTRAINT
Definition: drc_rule.h:56
@ EDGE_CLEARANCE_CONSTRAINT
Definition: drc_rule.h:53
@ MIN_RESOLVED_SPOKES_CONSTRAINT
Definition: drc_rule.h:65
@ TRACK_SEGMENT_LENGTH_CONSTRAINT
Definition: drc_rule.h:60
@ TEXT_THICKNESS_CONSTRAINT
Definition: drc_rule.h:58
@ LENGTH_CONSTRAINT
Definition: drc_rule.h:68
@ VIA_COUNT_CONSTRAINT
Definition: drc_rule.h:73
@ PHYSICAL_HOLE_CLEARANCE_CONSTRAINT
Definition: drc_rule.h:75
@ CLEARANCE_CONSTRAINT
Definition: drc_rule.h:49
@ THERMAL_SPOKE_WIDTH_CONSTRAINT
Definition: drc_rule.h:64
@ CONNECTION_WIDTH_CONSTRAINT
Definition: drc_rule.h:77
@ THERMAL_RELIEF_GAP_CONSTRAINT
Definition: drc_rule.h:63
@ MAX_UNCOUPLED_CONSTRAINT
Definition: drc_rule.h:71
@ ASSERTION_CONSTRAINT
Definition: drc_rule.h:76
@ SKEW_CONSTRAINT
Definition: drc_rule.h:69
@ HOLE_CLEARANCE_CONSTRAINT
Definition: drc_rule.h:51
@ TRACK_ANGLE_CONSTRAINT
Definition: drc_rule.h:78
@ HOLE_SIZE_CONSTRAINT
Definition: drc_rule.h:54
@ TEXT_HEIGHT_CONSTRAINT
Definition: drc_rule.h:57
@ CREEPAGE_CONSTRAINT
Definition: drc_rule.h:50
@ PHYSICAL_CLEARANCE_CONSTRAINT
Definition: drc_rule.h:74
@ HOLE_TO_HOLE_CONSTRAINT
Definition: drc_rule.h:52
#define DRC_RULE_FILE_VERSION
@ DSN_RIGHT
Definition: dsnlexer.h:68
@ DSN_NUMBER
Definition: dsnlexer.h:67
@ DSN_STRING
Definition: dsnlexer.h:70
#define _(s)
#define THROW_PARSE_ERROR(aProblem, aSource, aInputLine, aLineNumber, aByteIndex)
Definition: ki_exception.h:165
@ Rescue
Definition: layer_ids.h:121
PCB_LAYER_ID ToLAYER_ID(int aLayer)
Definition: lset.cpp:723
SEVERITY
@ RPT_SEVERITY_WARNING
@ RPT_SEVERITY_ERROR
@ RPT_SEVERITY_UNDEFINED
@ RPT_SEVERITY_EXCLUSION
@ RPT_SEVERITY_IGNORE
@ RPT_SEVERITY_INFO