24#include <boost/test/unit_test.hpp>
65 BOOST_REQUIRE_EQUAL( parsedRules.size(), 1 );
66 auto parsed = std::dynamic_pointer_cast<DRC_RE_VIA_STYLE_CONSTRAINT_DATA>( parsedRules[0] );
71 BOOST_CHECK_CLOSE( parsed->GetMinViaDiameter(), original.
GetMinViaDiameter(), 0.0001 );
72 BOOST_CHECK_CLOSE( parsed->GetMaxViaDiameter(), original.
GetMaxViaDiameter(), 0.0001 );
73 BOOST_CHECK_CLOSE( parsed->GetMinViaHoleSize(), original.
GetMinViaHoleSize(), 0.0001 );
74 BOOST_CHECK_CLOSE( parsed->GetMaxViaHoleSize(), original.
GetMaxViaHoleSize(), 0.0001 );
92 BOOST_REQUIRE_EQUAL( parsedRules.size(), 1 );
93 auto parsed = std::dynamic_pointer_cast<DRC_RE_ROUTING_WIDTH_CONSTRAINT_DATA>( parsedRules[0] );
98 BOOST_CHECK_CLOSE( parsed->GetOptWidth(), original.
GetOptWidth(), 0.0001 );
99 BOOST_CHECK_CLOSE( parsed->GetWidthTolerance(), original.
GetWidthTolerance(), 0.0001 );
104 std::vector<std::shared_ptr<DRC_RE_BASE_CONSTRAINT_DATA>> rules;
106 auto rule1 = std::make_shared<DRC_RE_VIA_STYLE_CONSTRAINT_DATA>( 0, 0,
"ViaRule", 0.5, 0.8, 0.2, 0.4 );
107 rule1->SetConstraintCode(
"via_style" );
108 rule1->SetRuleCondition(
"A.NetClass == 'Power'" );
109 rules.push_back( rule1 );
111 auto rule2 = std::make_shared<DRC_RE_ROUTING_WIDTH_CONSTRAINT_DATA>( 0, 0,
"TrackRule", 0.3, 0.1 );
112 rule2->SetConstraintCode(
"track_width" );
113 rule2->SetRuleCondition(
"A.NetClass == 'Signal'" );
114 rules.push_back( rule2 );
116 wxString filename =
"test_rules.dru";
122 wxFFile file( filename );
125 file.ReadAll( &content );
128 BOOST_CHECK( content.Contains(
"(version 1)" ) );
129 BOOST_CHECK( content.Contains(
"(rule ViaRule" ) );
130 BOOST_CHECK( content.Contains(
"(constraint via_diameter" ) );
131 BOOST_CHECK( content.Contains(
"(constraint hole_size" ) );
132 BOOST_CHECK( content.Contains(
"(rule TrackRule" ) );
133 BOOST_CHECK( content.Contains(
"(constraint track_width" ) );
136 wxRemoveFile( filename );
142 std::vector<RULE_TREE_NODE> ruleTreeNodeDatas;
150 auto data = std::make_shared<DRC_RE_BASE_CONSTRAINT_DATA>( 1, 0,
"TestRule" );
151 data->SetConstraintCode(
"clearance" );
154 ruleTreeNodeDatas.push_back( node );
157 std::vector<std::shared_ptr<DRC_RE_BASE_CONSTRAINT_DATA>> rules;
161 if( n.m_nodeType !=
RULE )
164 auto d = std::dynamic_pointer_cast<DRC_RE_BASE_CONSTRAINT_DATA>( n.m_nodeData );
167 rules.push_back( d );
175 auto rule = std::make_shared<DRC_RE_NUMERIC_INPUT_CONSTRAINT_DATA>( 0, 0, 0.2,
"ClearanceRule" );
176 rule->SetConstraintCode(
"clearance" );
182 wxString ruleText = rule->GenerateRule( ctx );
184 BOOST_CHECK( ruleText.Contains(
"(constraint clearance (min 0.2mm))" ) );
187 auto rule2 = std::make_shared<DRC_RE_NUMERIC_INPUT_CONSTRAINT_DATA>( 0, 0, 10,
"ViaCountRule" );
188 rule2->SetConstraintCode(
"via_count" );
189 ctx.
ruleName = rule2->GetRuleName();
192 ruleText = rule2->GenerateRule( ctx );
193 BOOST_CHECK( ruleText.Contains(
"(constraint via_count (max 10))" ) );
196 auto rule3 = std::make_shared<DRC_RE_NUMERIC_INPUT_CONSTRAINT_DATA>( 0, 0, 45,
"AngleRule" );
197 rule3->SetConstraintCode(
"track_angle" );
198 ctx.
ruleName = rule3->GetRuleName();
201 ruleText = rule3->GenerateRule( ctx );
202 BOOST_CHECK( ruleText.Contains(
"(constraint track_angle (min 45deg))" ) );
207 auto rule = std::make_shared<DRC_RE_BOOL_INPUT_CONSTRAINT_DATA>( 0, 0,
true,
"DisallowTestRule" );
208 rule->SetConstraintCode(
"disallow" );
214 wxString ruleText = rule->GenerateRule( ctx );
216 BOOST_CHECK( ruleText.Contains(
"(constraint disallow)" ) );
219 auto rule2 = std::make_shared<DRC_RE_BOOL_INPUT_CONSTRAINT_DATA>( 0, 0,
false,
"NoDisallowTestRule" );
220 rule2->SetConstraintCode(
"disallow" );
221 ctx.
ruleName = rule2->GetRuleName();
224 ruleText = rule2->GenerateRule( ctx );
225 BOOST_CHECK( ruleText.IsEmpty() );
232 "(rule \"Clearance Rule\"\n"
233 "\t(constraint clearance (min 0.2mm))\n"
234 "\t(condition \"A.NetClass == 'Power'\"))\n"
235 "(rule \"Via Style Rule\"\n"
236 "\t(constraint via_diameter (min 0.5mm))\n"
237 "\t(constraint hole_size (min 0.3mm))\n"
238 "\t(condition \"A.NetClass == 'Power'\"))\n";
242 BOOST_REQUIRE_EQUAL( parsedRules.size(), 2 );
245 auto clearanceRule = parsedRules[0];
250 auto numericData = std::dynamic_pointer_cast<DRC_RE_NUMERIC_INPUT_CONSTRAINT_DATA>( clearanceRule );
252 BOOST_CHECK_CLOSE( numericData->GetNumericInputValue(), 0.2, 0.0001 );
255 auto viaStyleRule = parsedRules[1];
259 auto viaData = std::dynamic_pointer_cast<DRC_RE_VIA_STYLE_CONSTRAINT_DATA>( viaStyleRule );
261 BOOST_CHECK_CLOSE( viaData->GetMinViaDiameter(), 0.5, 0.0001 );
262 BOOST_CHECK_CLOSE( viaData->GetMinViaHoleSize(), 0.3, 0.0001 );
268 bool parserCalled =
false;
270 [&](
const std::shared_ptr<DRC_RULE>& aRule ) -> std::shared_ptr<DRC_RE_BASE_CONSTRAINT_DATA>
275 auto data = std::make_shared<DRC_RE_CUSTOM_RULE_CONSTRAINT_DATA>( 0, 0, aRule->m_Name );
276 data->SetConstraintCode(
"custom_clearance" );
282 wxString ruleText =
"(version 1) (rule \"Test Rule\" (constraint clearance (min 1.0mm)) (condition \"A.Type == 'Pad'\"))";
285 BOOST_CHECK( parserCalled );
286 BOOST_REQUIRE_EQUAL( parsedRules.size(), 1 );
296 BOOST_CHECK(
result.isValid );
297 BOOST_CHECK(
result.errors.empty() );
307 BOOST_CHECK( !
result.isValid );
308 BOOST_CHECK( !
result.errors.empty() );
311 bool foundMinMaxError =
false;
312 for(
const auto& error :
result.errors )
314 if( error.find(
"Minimum Via Diameter cannot be greater than Maximum Via Diameter" ) != std::string::npos )
315 foundMinMaxError =
true;
317 BOOST_CHECK( foundMinMaxError );
327 BOOST_CHECK( !
result.isValid );
328 BOOST_CHECK( !
result.errors.empty() );
331 bool foundNegativeError =
false;
332 for(
const auto& error :
result.errors )
334 if( error.find(
"must be greater than 0" ) != std::string::npos )
335 foundNegativeError =
true;
337 BOOST_CHECK( foundNegativeError );
344 [](
const std::shared_ptr<DRC_RULE>& aRule ) -> std::shared_ptr<DRC_RE_BASE_CONSTRAINT_DATA> {
345 if( aRule->m_Name ==
"Test" )
346 return std::make_shared<DRC_RE_CUSTOM_RULE_CONSTRAINT_DATA>( 0, 0, aRule->m_Name +
"_1" );
352 [](
const std::shared_ptr<DRC_RULE>& aRule ) -> std::shared_ptr<DRC_RE_BASE_CONSTRAINT_DATA> {
353 if( aRule->m_Name ==
"Test" )
354 return std::make_shared<DRC_RE_CUSTOM_RULE_CONSTRAINT_DATA>( 0, 0, aRule->m_Name +
"_2" );
358 wxString ruleText =
"(version 1) (rule \"Test\" (constraint clearance (min 1.0mm)))";
361 BOOST_REQUIRE_EQUAL( parsedRules.size(), 1 );
372 BOOST_CHECK(
result.isValid );
373 BOOST_CHECK(
result.errors.empty() );
383 BOOST_CHECK( !result1.
isValid );
384 BOOST_CHECK( !result1.
errors.empty() );
386 bool foundToleranceError =
false;
388 for(
const auto& error : result1.
errors )
390 if( error.find(
"Tolerance is too large" ) != std::string::npos )
391 foundToleranceError =
true;
394 BOOST_CHECK( foundToleranceError );
401 BOOST_CHECK( !result2.
isValid );
403 bool foundNegativeError =
false;
405 for(
const auto& error : result2.
errors )
407 if( error.find(
"Tolerance must be greater than or equal to 0" ) != std::string::npos )
408 foundNegativeError =
true;
411 BOOST_CHECK( foundNegativeError );
422 BOOST_CHECK(
result.isValid );
423 BOOST_CHECK(
result.errors.empty() );
433 BOOST_CHECK( !result1.
isValid );
434 BOOST_CHECK( !result1.
errors.empty() );
436 bool foundWidthError =
false;
437 for(
const auto& error : result1.
errors )
439 if( error.find(
"Width Tolerance must be less than Optimum Width" ) != std::string::npos )
440 foundWidthError =
true;
442 BOOST_CHECK( foundWidthError );
449 BOOST_CHECK( !result2.
isValid );
451 bool foundNegativeError =
false;
452 for(
const auto& error : result2.
errors )
454 if( error.find(
"must be greater than or equal to 0" ) != std::string::npos )
455 foundNegativeError =
true;
457 BOOST_CHECK( foundNegativeError );
467 BOOST_CHECK( result1.
isValid );
468 BOOST_CHECK( result1.
errors.empty() );
475 BOOST_CHECK( result2.
isValid );
476 BOOST_CHECK( result2.
errors.empty() );
483 BOOST_CHECK( result3.
isValid );
484 BOOST_CHECK( result3.
errors.empty() );
494 BOOST_CHECK( !
result.isValid );
495 BOOST_CHECK( !
result.errors.empty() );
497 bool foundLayerError =
false;
498 for(
const auto& error :
result.errors )
500 if( error.find(
"At least one layer must be selected" ) != std::string::npos )
501 foundLayerError =
true;
503 BOOST_CHECK( foundLayerError );
512 BOOST_CHECK(
result.isValid );
513 BOOST_CHECK(
result.errors.empty() );
522 BOOST_CHECK( !
result.isValid );
523 BOOST_CHECK( !
result.errors.empty() );
526 bool foundError =
false;
527 for(
const auto& error :
result.errors )
529 if( error.find(
"must be greater than 0" ) != std::string::npos )
532 BOOST_CHECK( foundError );
541 BOOST_CHECK(
result.isValid );
542 BOOST_CHECK(
result.errors.empty() );
552 BOOST_CHECK( !
result.isValid );
553 BOOST_CHECK( !
result.errors.empty() );
555 bool foundError =
false;
556 for(
const auto& error :
result.errors )
558 if( error.find(
"Width Tolerance must be less than Optimum Width" ) != std::string::npos )
561 BOOST_CHECK( foundError );
571 BOOST_CHECK( result1.
isValid );
572 BOOST_CHECK( result1.
errors.empty() );
579 BOOST_CHECK( result2.
isValid );
580 BOOST_CHECK( result2.
errors.empty() );
587 BOOST_CHECK( result3.
isValid );
588 BOOST_CHECK( result3.
errors.empty() );
595 BOOST_CHECK( result4.
isValid );
596 BOOST_CHECK( result4.
errors.empty() );
606 BOOST_CHECK( !
result.isValid );
607 BOOST_CHECK( !
result.errors.empty() );
609 bool foundOrientationError =
false;
610 for(
const auto& error :
result.errors )
612 if( error.find(
"At least one orientation must be selected" ) != std::string::npos )
613 foundOrientationError =
true;
615 BOOST_CHECK( foundOrientationError );
625 BOOST_CHECK( resultTrue.
isValid );
626 BOOST_CHECK( resultTrue.
errors.empty() );
633 BOOST_CHECK( resultFalse.
isValid );
634 BOOST_CHECK( resultFalse.
errors.empty() );
644 BOOST_CHECK(
result.isValid );
645 BOOST_CHECK(
result.errors.empty() );
655 BOOST_CHECK( !result1.
isValid );
656 BOOST_CHECK( !result1.
errors.empty() );
658 bool foundHeightError =
false;
659 for(
const auto& error : result1.
errors )
661 if( error.find(
"Minimum Text Height must be greater than 0" ) != std::string::npos )
662 foundHeightError =
true;
664 BOOST_CHECK( foundHeightError );
671 BOOST_CHECK( !result2.
isValid );
673 bool foundThicknessError =
false;
674 for(
const auto& error : result2.
errors )
676 if( error.find(
"Minimum Text Thickness must be greater than 0" ) != std::string::npos )
677 foundThicknessError =
true;
679 BOOST_CHECK( foundThicknessError );
686 BOOST_CHECK( !result3.
isValid );
693 data.
SetRuleText(
"(constraint clearance (min 0.2mm))" );
697 BOOST_CHECK(
result.isValid );
698 BOOST_CHECK(
result.errors.empty() );
708 BOOST_CHECK( !
result.isValid );
709 BOOST_CHECK( !
result.errors.empty() );
712 bool foundEmptyError =
false;
713 for(
const auto& error :
result.errors )
715 if( error.find(
"Rule text cannot be empty" ) != std::string::npos )
716 foundEmptyError =
true;
718 BOOST_CHECK( foundEmptyError );
744 std::vector<DRC_PANEL_MATCH> matches = matcher.
MatchRule( rule );
746 BOOST_REQUIRE_EQUAL( matches.size(), 1 );
756 DRC_RULE rule(
"ViaDiameterOnlyRule" );
763 std::vector<DRC_PANEL_MATCH> matches = matcher.
MatchRule( rule );
765 BOOST_REQUIRE_EQUAL( matches.size(), 1 );
789 std::vector<DRC_PANEL_MATCH> matches = matcher.
MatchRule( rule );
791 BOOST_REQUIRE_EQUAL( matches.size(), 2 );
821 std::vector<DRC_PANEL_MATCH> matches = matcher.
MatchRule( rule );
823 BOOST_REQUIRE_EQUAL( matches.size(), 1 );
832 DRC_RULE rule(
"DiffPairWithUncoupledRule" );
847 std::vector<DRC_PANEL_MATCH> matches = matcher.
MatchRule( rule );
849 BOOST_REQUIRE_EQUAL( matches.size(), 1 );
860 DRC_RULE rule(
"TrackWidthOnlyRule" );
869 std::vector<DRC_PANEL_MATCH> matches = matcher.
MatchRule( rule );
871 BOOST_REQUIRE_EQUAL( matches.size(), 1 );
890 std::vector<DRC_PANEL_MATCH> matches = matcher.
MatchRule( rule );
892 BOOST_REQUIRE_EQUAL( matches.size(), 1 );
909 std::vector<DRC_PANEL_MATCH> matches = matcher.
MatchRule( rule );
911 BOOST_REQUIRE_EQUAL( matches.size(), 1 );
919 DRC_RULE rule(
"MatchedLengthRule" );
932 std::vector<DRC_PANEL_MATCH> matches = matcher.
MatchRule( rule );
934 BOOST_REQUIRE_EQUAL( matches.size(), 1 );
937 BOOST_CHECK( matches[0].claimedConstraints.count(
SKEW_CONSTRAINT ) > 0 );
946 std::vector<DRC_PANEL_MATCH> matches = matcher.
MatchRule( rule );
991 "(rule \"Via Test\"\n"
992 " (constraint via_diameter (min 0.5mm) (opt 0.6mm) (max 0.8mm))\n"
993 " (constraint hole_size (min 0.2mm) (opt 0.3mm) (max 0.4mm)))";
996 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( ruleText );
998 BOOST_REQUIRE_EQUAL( entries.size(), 1 );
1002 auto viaData = std::dynamic_pointer_cast<DRC_RE_VIA_STYLE_CONSTRAINT_DATA>( entries[0].constraintData );
1005 BOOST_CHECK_CLOSE( viaData->GetMinViaDiameter(), 0.5, 0.0001 );
1006 BOOST_CHECK_CLOSE( viaData->GetMaxViaDiameter(), 0.8, 0.0001 );
1007 BOOST_CHECK_CLOSE( viaData->GetMinViaHoleSize(), 0.2, 0.0001 );
1008 BOOST_CHECK_CLOSE( viaData->GetMaxViaHoleSize(), 0.4, 0.0001 );
1016 "(rule \"Track Test\"\n"
1017 " (constraint track_width (min 0.2mm) (opt 0.25mm) (max 0.3mm)))";
1020 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( ruleText );
1022 BOOST_REQUIRE_EQUAL( entries.size(), 1 );
1026 auto trackData = std::dynamic_pointer_cast<DRC_RE_ROUTING_WIDTH_CONSTRAINT_DATA>( entries[0].constraintData );
1029 BOOST_CHECK_CLOSE( trackData->GetOptWidth(), 0.25, 0.0001 );
1030 BOOST_CHECK_CLOSE( trackData->GetWidthTolerance(), 0.05, 0.0001 );
1038 "(rule \"Diff Pair Test\"\n"
1039 " (constraint track_width (min 0.2mm) (opt 0.25mm) (max 0.3mm))\n"
1040 " (constraint diff_pair_gap (min 0.1mm) (opt 0.15mm) (max 0.2mm))\n"
1041 " (constraint diff_pair_uncoupled (max 5mm)))";
1044 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( ruleText );
1046 BOOST_REQUIRE_EQUAL( entries.size(), 1 );
1049 auto dpData = std::dynamic_pointer_cast<DRC_RE_ROUTING_DIFF_PAIR_CONSTRAINT_DATA>( entries[0].constraintData );
1052 BOOST_CHECK_CLOSE( dpData->GetOptWidth(), 0.25, 0.0001 );
1053 BOOST_CHECK_CLOSE( dpData->GetWidthTolerance(), 0.05, 0.0001 );
1054 BOOST_CHECK_CLOSE( dpData->GetOptGap(), 0.15, 0.0001 );
1055 BOOST_CHECK_CLOSE( dpData->GetGapTolerance(), 0.05, 0.0001 );
1056 BOOST_CHECK_CLOSE( dpData->GetMaxUncoupledLength(), 5.0, 0.0001 );
1064 "(rule \"Split Test\"\n"
1065 " (constraint via_diameter (min 0.5mm))\n"
1066 " (constraint hole_size (min 0.2mm))\n"
1067 " (constraint clearance (min 0.15mm)))";
1070 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( ruleText );
1072 BOOST_REQUIRE_EQUAL( entries.size(), 2 );
1082 auto numericData = std::dynamic_pointer_cast<DRC_RE_NUMERIC_INPUT_CONSTRAINT_DATA>( entries[1].constraintData );
1084 BOOST_CHECK_CLOSE( numericData->GetNumericInputValue(), 0.15, 0.0001 );
1094 "(rule \"clearance_under_fpga\"\n"
1095 " (constraint clearance (min 0.1mm))\n"
1096 " (constraint hole_size (min 0.2mm))\n"
1097 " (constraint via_diameter (min 0.4mm))\n"
1098 " (condition \"A.intersectsArea('underFPGA') || A.intersectsArea('underDDR')\"))";
1101 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( ruleText );
1104 BOOST_REQUIRE_EQUAL( entries.size(), 2 );
1109 BOOST_CHECK_EQUAL( entries[0].condition,
"A.intersectsArea('underFPGA') || A.intersectsArea('underDDR')" );
1114 BOOST_CHECK_EQUAL( entries[1].condition,
"A.intersectsArea('underFPGA') || A.intersectsArea('underDDR')" );
1117 auto viaData = std::dynamic_pointer_cast<DRC_RE_VIA_STYLE_CONSTRAINT_DATA>( entries[0].constraintData );
1119 BOOST_CHECK_CLOSE( viaData->GetMinViaDiameter(), 0.4, 0.0001 );
1120 BOOST_CHECK_CLOSE( viaData->GetMinViaHoleSize(), 0.2, 0.0001 );
1123 auto clearanceData = std::dynamic_pointer_cast<DRC_RE_NUMERIC_INPUT_CONSTRAINT_DATA>( entries[1].constraintData );
1125 BOOST_CHECK_CLOSE( clearanceData->GetNumericInputValue(), 0.1, 0.0001 );
1133 "(rule \"Text Test\"\n"
1134 " (constraint text_height (min 1mm))\n"
1135 " (constraint text_thickness (min 0.15mm)))";
1138 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( ruleText );
1140 BOOST_REQUIRE_EQUAL( entries.size(), 1 );
1143 auto textData = std::dynamic_pointer_cast<DRC_RE_MINIMUM_TEXT_HEIGHT_THICKNESS_CONSTRAINT_DATA>( entries[0].constraintData );
1146 BOOST_CHECK_CLOSE( textData->GetMinTextHeight(), 1.0, 0.0001 );
1147 BOOST_CHECK_CLOSE( textData->GetMinTextThickness(), 0.15, 0.0001 );
1155 "(rule \"Length Test\"\n"
1156 " (constraint length (min 10mm) (opt 30mm) (max 50mm)))";
1159 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( ruleText );
1161 BOOST_REQUIRE_EQUAL( entries.size(), 1 );
1165 auto lengthData = dynamic_pointer_cast<DRC_RE_ABSOLUTE_LENGTH_TWO_CONSTRAINT_DATA>( entries[0].constraintData );
1167 BOOST_CHECK_CLOSE( lengthData->GetMinimumLength(), 10.0, 0.0001 );
1168 BOOST_CHECK_CLOSE( lengthData->GetOptimumLength(), 30.0, 0.0001 );
1169 BOOST_CHECK_CLOSE( lengthData->GetMaximumLength(), 50.0, 0.0001 );
1177 "(rule \"DP Length Match\"\n"
1178 " (constraint length (min 10mm) (opt 30mm) (max 50mm))\n"
1179 " (constraint skew (max 1mm)))";
1182 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( ruleText );
1184 BOOST_REQUIRE_EQUAL( entries.size(), 1 );
1189 std::dynamic_pointer_cast<DRC_RE_MATCHED_LENGTH_DIFF_PAIR_CONSTRAINT_DATA>( entries[0].constraintData );
1192 BOOST_CHECK_CLOSE( matchedData->GetMinimumLength(), 10.0, 0.0001 );
1193 BOOST_CHECK_CLOSE( matchedData->GetOptimumLength(), 30.0, 0.0001 );
1194 BOOST_CHECK_CLOSE( matchedData->GetMaximumLength(), 50.0, 0.0001 );
1195 BOOST_CHECK_CLOSE( matchedData->GetMaxSkew(), 1.0, 0.0001 );
1203 "(rule \"DP Match RT\"\n"
1204 " (constraint length (min 10mm) (opt 30mm) (max 50mm))\n"
1205 " (constraint skew (max 2.5mm)))";
1208 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( ruleText );
1210 BOOST_REQUIRE_EQUAL( entries.size(), 1 );
1211 entries[0].wasEdited =
true;
1216 BOOST_CHECK( savedText.Contains(
"length" ) );
1217 BOOST_CHECK( savedText.Contains(
"skew" ) );
1218 BOOST_CHECK( savedText.Contains(
"2.5mm" ) );
1220 std::vector<DRC_RE_LOADED_PANEL_ENTRY> reloaded = loader.
LoadFromString( savedText );
1222 BOOST_REQUIRE_EQUAL( reloaded.size(), 1 );
1226 std::dynamic_pointer_cast<DRC_RE_MATCHED_LENGTH_DIFF_PAIR_CONSTRAINT_DATA>( reloaded[0].constraintData );
1229 BOOST_CHECK_CLOSE( reloadedData->GetMaxSkew(), 2.5, 0.0001 );
1230 BOOST_CHECK_CLOSE( reloadedData->GetMinimumLength(), 10.0, 0.0001 );
1231 BOOST_CHECK_CLOSE( reloadedData->GetOptimumLength(), 30.0, 0.0001 );
1232 BOOST_CHECK_CLOSE( reloadedData->GetMaximumLength(), 50.0, 0.0001 );
1238 wxString ruleText =
"(version 1)\n"
1239 "(rule \"DP Skew Within\"\n"
1240 " (constraint length (min 10mm) (opt 30mm) (max 50mm))\n"
1241 " (constraint skew (max 1mm) (within_diff_pairs)))";
1244 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( ruleText );
1246 BOOST_REQUIRE_EQUAL( entries.size(), 1 );
1250 std::dynamic_pointer_cast<DRC_RE_MATCHED_LENGTH_DIFF_PAIR_CONSTRAINT_DATA>( entries[0].constraintData );
1253 BOOST_CHECK_CLOSE( matchedData->GetMaxSkew(), 1.0, 0.0001 );
1254 BOOST_CHECK( matchedData->GetWithinDiffPairs() );
1257 entries[0].wasEdited =
true;
1262 BOOST_CHECK( savedText.Contains(
"within_diff_pairs" ) );
1264 std::vector<DRC_RE_LOADED_PANEL_ENTRY> reloaded = loader.
LoadFromString( savedText );
1266 BOOST_REQUIRE_EQUAL( reloaded.size(), 1 );
1269 std::dynamic_pointer_cast<DRC_RE_MATCHED_LENGTH_DIFF_PAIR_CONSTRAINT_DATA>( reloaded[0].constraintData );
1272 BOOST_CHECK_CLOSE( reloadedData->GetMaxSkew(), 1.0, 0.0001 );
1273 BOOST_CHECK( reloadedData->GetWithinDiffPairs() );
1279 wxString ruleText =
"(version 1)\n"
1280 "(rule \"DP Skew Bus\"\n"
1281 " (constraint length (min 10mm) (opt 30mm) (max 50mm))\n"
1282 " (constraint skew (max 2mm)))";
1285 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( ruleText );
1287 BOOST_REQUIRE_EQUAL( entries.size(), 1 );
1290 std::dynamic_pointer_cast<DRC_RE_MATCHED_LENGTH_DIFF_PAIR_CONSTRAINT_DATA>( entries[0].constraintData );
1293 BOOST_CHECK( !matchedData->GetWithinDiffPairs() );
1296 entries[0].wasEdited =
true;
1301 BOOST_CHECK( !savedText.Contains(
"within_diff_pairs" ) );
1309 "(rule \"Conditional Test\"\n"
1310 " (condition \"A.NetClass == 'Power'\")\n"
1311 " (constraint clearance (min 0.3mm)))";
1314 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( ruleText );
1316 BOOST_REQUIRE_EQUAL( entries.size(), 1 );
1319 BOOST_CHECK_EQUAL( entries[0].constraintData->GetRuleCondition(),
"A.NetClass == 'Power'" );
1327 "(rule \"Rule A\"\n"
1328 " (constraint clearance (min 0.2mm)))\n"
1329 "(rule \"Rule B\"\n"
1330 " (constraint track_width (min 0.15mm)))\n"
1331 "(rule \"Rule C\"\n"
1332 " (constraint via_diameter (min 0.5mm))\n"
1333 " (constraint hole_size (min 0.3mm)))";
1336 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( ruleText );
1338 BOOST_REQUIRE_EQUAL( entries.size(), 3 );
1355 "(rule \"Empty Rule\")";
1358 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( ruleText );
1360 BOOST_REQUIRE_EQUAL( entries.size(), 1 );
1368 wxString ruleText =
"not valid rule text at all";
1371 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( ruleText );
1381 "(rule \"Edge Clearance\"\n"
1382 " (constraint edge_clearance (min 0.5mm)))\n"
1383 "(rule \"Hole Clearance\"\n"
1384 " (constraint hole_clearance (min 0.3mm)))\n"
1385 "(rule \"Courtyard\"\n"
1386 " (constraint courtyard_clearance (min 0.25mm)))";
1389 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( ruleText );
1391 BOOST_REQUIRE_EQUAL( entries.size(), 3 );
1411 auto numericData = std::make_shared<DRC_RE_NUMERIC_INPUT_CONSTRAINT_DATA>();
1412 numericData->SetRuleName(
"TestClearance" );
1413 numericData->SetConstraintCode(
"clearance" );
1414 numericData->SetNumericInputValue( 0.2 );
1418 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = { entry };
1421 BOOST_CHECK(
result.Contains(
"(version 1)" ) );
1422 BOOST_CHECK(
result.Contains(
"TestClearance" ) );
1423 BOOST_CHECK(
result.Contains(
"clearance" ) );
1432 entry.
originalRuleText =
"(rule \"Preserved Rule\"\n\t(constraint clearance (min 0.5mm)))";
1435 auto numericData = std::make_shared<DRC_RE_NUMERIC_INPUT_CONSTRAINT_DATA>();
1436 numericData->SetRuleName(
"Preserved Rule" );
1437 numericData->SetConstraintCode(
"clearance" );
1438 numericData->SetNumericInputValue( 0.5 );
1442 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = { entry };
1454 entry.
originalRuleText =
"(rule \"EditedRule\"\n\t(constraint clearance (min 0.5mm)))";
1457 auto numericData = std::make_shared<DRC_RE_NUMERIC_INPUT_CONSTRAINT_DATA>();
1458 numericData->SetRuleName(
"EditedRule" );
1459 numericData->SetConstraintCode(
"clearance" );
1460 numericData->SetNumericInputValue( 0.3 );
1464 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = { entry };
1468 BOOST_CHECK( !
result.Contains(
"0.5mm" ) );
1469 BOOST_CHECK(
result.Contains(
"EditedRule" ) );
1479 auto viaData = std::make_shared<DRC_RE_VIA_STYLE_CONSTRAINT_DATA>();
1480 viaData->SetRuleName(
"ViaTest" );
1481 viaData->SetMinViaDiameter( 0.5 );
1482 viaData->SetMaxViaDiameter( 0.8 );
1483 viaData->SetMinViaHoleSize( 0.2 );
1484 viaData->SetMaxViaHoleSize( 0.4 );
1489 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = { entry };
1492 BOOST_CHECK(
result.Contains(
"ViaTest" ) );
1493 BOOST_CHECK(
result.Contains(
"via_diameter" ) );
1494 BOOST_CHECK(
result.Contains(
"hole_size" ) );
1500 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries;
1505 auto data1 = std::make_shared<DRC_RE_NUMERIC_INPUT_CONSTRAINT_DATA>();
1506 data1->SetRuleName(
"RuleA" );
1507 data1->SetConstraintCode(
"clearance" );
1508 data1->SetNumericInputValue( 0.2 );
1511 entries.push_back( entry1 );
1516 auto data2 = std::make_shared<DRC_RE_ROUTING_WIDTH_CONSTRAINT_DATA>();
1517 data2->SetRuleName(
"RuleB" );
1518 data2->SetOptWidth( 0.2 );
1519 data2->SetWidthTolerance( 0.1 );
1522 entries.push_back( entry2 );
1528 BOOST_CHECK(
result.Contains(
"RuleA" ) );
1529 BOOST_CHECK(
result.Contains(
"RuleB" ) );
1532 size_t posA =
result.Find(
"RuleA" );
1533 size_t posB =
result.Find(
"RuleB" );
1534 BOOST_CHECK( posA < posB );
1542 entry.
ruleName =
"ConditionalRule";
1543 entry.
condition =
"A.NetClass == 'Power'";
1545 auto numericData = std::make_shared<DRC_RE_NUMERIC_INPUT_CONSTRAINT_DATA>();
1546 numericData->SetRuleName(
"ConditionalRule" );
1547 numericData->SetConstraintCode(
"clearance" );
1548 numericData->SetRuleCondition(
"A.NetClass == 'Power'" );
1549 numericData->SetNumericInputValue( 0.3 );
1554 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = { entry };
1557 BOOST_CHECK(
result.Contains(
"ConditionalRule" ) );
1558 BOOST_CHECK(
result.Contains(
"condition" ) );
1559 BOOST_CHECK(
result.Contains(
"Power" ) );
1566 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries;
1569 BOOST_CHECK(
result.Contains(
"(version 1)" ) );
1584 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = { entry };
1588 BOOST_CHECK(
result.Contains(
"(version 1)" ) );
1589 BOOST_CHECK( !
result.Contains(
"Null Data Rule" ) );
1595 wxString originalText =
1597 "(rule \"RoundTripTest\"\n"
1598 " (constraint clearance (min 0.25mm)))";
1601 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( originalText );
1603 BOOST_REQUIRE_EQUAL( entries.size(), 1 );
1607 entries[0].wasEdited =
true;
1614 std::vector<DRC_RE_LOADED_PANEL_ENTRY> reloadedEntries = loader.
LoadFromString( savedText );
1616 BOOST_REQUIRE_EQUAL( reloadedEntries.size(), 1 );
1626 const wxString originalName = wxS(
"Annular width: LEDs" );
1633 auto numericData = std::make_shared<DRC_RE_NUMERIC_INPUT_CONSTRAINT_DATA>();
1634 numericData->SetRuleName( originalName );
1635 numericData->SetConstraintCode(
"annular_width" );
1636 numericData->SetNumericInputValue( 0.085 );
1640 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = { entry };
1644 BOOST_CHECK(
result.Contains( wxS(
"\"Annular width: LEDs\"" ) ) );
1645 BOOST_CHECK( !
result.Contains( wxS(
"Annular_width__LEDs" ) ) );
1651 BOOST_REQUIRE_EQUAL( reloaded.size(), 1 );
1661 entry.
ruleName = wxS(
"My_Bare_Rule-1.0" );
1664 auto numericData = std::make_shared<DRC_RE_NUMERIC_INPUT_CONSTRAINT_DATA>();
1665 numericData->SetRuleName( entry.
ruleName );
1666 numericData->SetConstraintCode(
"clearance" );
1667 numericData->SetNumericInputValue( 0.2 );
1671 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = { entry };
1674 BOOST_CHECK(
result.Contains( wxS(
"(rule My_Bare_Rule-1.0\n" ) ) );
1675 BOOST_CHECK( !
result.Contains( wxS(
"\"My_Bare_Rule-1.0\"" ) ) );
1685 entry.
ruleName = wxS(
"3.3V_Power" );
1688 auto numericData = std::make_shared<DRC_RE_NUMERIC_INPUT_CONSTRAINT_DATA>();
1689 numericData->SetRuleName( entry.
ruleName );
1690 numericData->SetConstraintCode(
"clearance" );
1691 numericData->SetNumericInputValue( 0.2 );
1695 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = { entry };
1698 BOOST_CHECK(
result.Contains( wxS(
"\"3.3V_Power\"" ) ) );
1702 BOOST_REQUIRE_EQUAL( reloaded.size(), 1 );
1714 auto dpData = std::make_shared<DRC_RE_ROUTING_DIFF_PAIR_CONSTRAINT_DATA>();
1715 dpData->SetRuleName(
"DiffPairTest" );
1716 dpData->SetOptWidth( 0.25 );
1717 dpData->SetWidthTolerance( 0.05 );
1718 dpData->SetOptGap( 0.15 );
1719 dpData->SetGapTolerance( 0.05 );
1720 dpData->SetMaxUncoupledLength( 5.0 );
1725 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = { entry };
1728 BOOST_CHECK(
result.Contains(
"DiffPairTest" ) );
1729 BOOST_CHECK(
result.Contains(
"track_width" ) );
1730 BOOST_CHECK(
result.Contains(
"diff_pair_gap" ) );
1743 "(rule \"PTH to Track Clearance\"\n"
1744 " (constraint hole_clearance (min 1.0mm))\n"
1745 " (condition \"A.Type == 'Pad' && A.Pad_Type == 'Through-hole' && B.Type =='Track'\"))\n"
1746 "(rule \"Max Drill Hole Size Mechanical\"\n"
1747 " (constraint hole (max 2.0mm))\n"
1748 " (condition \"A.Type == 'Pad'\"))";
1751 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( ruleText );
1753 BOOST_REQUIRE_GE( entries.size(), 2 );
1758 BOOST_CHECK( !entries[0].condition.IsEmpty() );
1770 "(rule board_edge\n"
1771 " (constraint edge_clearance (min 1mm))\n"
1772 " (condition \"A.memberOf('board_edge')\")\n"
1773 " (severity ignore))";
1776 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( ruleText );
1778 BOOST_REQUIRE_EQUAL( entries.size(), 1 );
1789 "(rule high_current_netclass\n"
1790 " (constraint connection_width (min 0.16mm))\n"
1791 " (condition \"A.NetClass == 'High_current'\"))\n"
1792 "(rule high_current_area\n"
1793 " (constraint connection_width (min 0.16mm))\n"
1794 " (condition \"A.insideArea('high_current')\"))";
1797 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( ruleText );
1799 BOOST_REQUIRE_EQUAL( entries.size(), 2 );
1806 BOOST_CHECK( !entries[0].condition.IsEmpty() );
1807 BOOST_CHECK( !entries[1].condition.IsEmpty() );
1814 wxString originalText =
1816 "(rule ClearanceRule\n"
1817 " (constraint clearance (min 0.2mm))\n"
1818 " (condition \"A.NetClass == 'Power'\"))\n"
1819 "(rule TrackWidthRule\n"
1820 " (constraint track_width (min 0.15mm) (opt 0.2mm) (max 0.3mm)))";
1823 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( originalText );
1825 BOOST_REQUIRE_EQUAL( entries.size(), 2 );
1828 for(
auto& entry : entries )
1829 entry.wasEdited =
true;
1836 std::vector<DRC_RE_LOADED_PANEL_ENTRY> reloadedEntries = loader.
LoadFromString( savedText );
1838 BOOST_REQUIRE_EQUAL( reloadedEntries.size(), 2 );
1852 wxString originalText =
1854 "(rule \"Via Style\"\n"
1855 " (constraint via_diameter (min 0.5mm) (opt 0.6mm) (max 0.8mm))\n"
1856 " (constraint hole_size (min 0.2mm) (opt 0.3mm) (max 0.4mm)))";
1859 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( originalText );
1861 BOOST_REQUIRE_EQUAL( entries.size(), 1 );
1864 auto viaData = std::dynamic_pointer_cast<DRC_RE_VIA_STYLE_CONSTRAINT_DATA>( entries[0].constraintData );
1867 double originalMinDia = viaData->GetMinViaDiameter();
1868 double originalMinHole = viaData->GetMinViaHoleSize();
1871 entries[0].wasEdited =
true;
1876 std::vector<DRC_RE_LOADED_PANEL_ENTRY> reloadedEntries = loader.
LoadFromString( savedText );
1878 BOOST_REQUIRE_EQUAL( reloadedEntries.size(), 1 );
1881 auto reloadedViaData =
1882 std::dynamic_pointer_cast<DRC_RE_VIA_STYLE_CONSTRAINT_DATA>( reloadedEntries[0].constraintData );
1885 BOOST_CHECK_CLOSE( reloadedViaData->GetMinViaDiameter(), originalMinDia, 0.0001 );
1886 BOOST_CHECK_CLOSE( reloadedViaData->GetMinViaHoleSize(), originalMinHole, 0.0001 );
1894 "(rule \"Rule A\"\n"
1895 " (constraint clearance (min 0.2mm)))\n"
1896 "(rule \"Rule B\"\n"
1897 " (constraint via_diameter (min 0.5mm))\n"
1898 " (constraint hole_size (min 0.2mm))\n"
1899 " (constraint track_width (min 0.15mm)))\n"
1900 "(rule \"Rule C\"\n"
1901 " (constraint edge_clearance (min 0.3mm)))";
1904 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( ruleText );
1907 BOOST_REQUIRE_GE( entries.size(), 4 );
1910 int ruleACount = 0, ruleBCount = 0, ruleCCount = 0;
1912 for(
const auto& entry : entries )
1914 if( entry.ruleName ==
"Rule A" )
1916 else if( entry.ruleName ==
"Rule B" )
1918 else if( entry.ruleName ==
"Rule C" )
1923 BOOST_CHECK_GE( ruleBCount, 2 );
1927 int lastAIndex = -1, firstBIndex = entries.size(), lastBIndex = -1, firstCIndex = entries.size();
1929 for(
size_t i = 0; i < entries.size(); i++ )
1931 if( entries[i].ruleName ==
"Rule A" )
1933 else if( entries[i].ruleName ==
"Rule B" )
1935 if( (
int)i < firstBIndex )
1940 else if( entries[i].ruleName ==
"Rule C" )
1942 if( (
int)i < firstCIndex )
1947 BOOST_CHECK( lastAIndex < firstBIndex );
1948 BOOST_CHECK( lastBIndex < firstCIndex );
1960 wxString expr =
"A.NetName == 'VCC'";
1963 BOOST_CHECK( !expr.IsEmpty() );
1964 BOOST_CHECK( expr.Contains(
"NetName" ) );
1965 BOOST_CHECK( expr.Contains(
"VCC" ) );
1971 wxString expr =
"A.NetName == 'VCC' && B.NetClass == 'Power'";
1974 BOOST_CHECK( expr.Contains(
"&&" ) );
1975 BOOST_CHECK( expr.Contains(
"NetName" ) );
1976 BOOST_CHECK( expr.Contains(
"NetClass" ) );
1982 wxString expr =
"A.intersectsArea('underFPGA') || A.intersectsArea('underDDR')";
1985 BOOST_CHECK( expr.Contains(
"||" ) );
1986 BOOST_CHECK( expr.Contains(
"intersectsArea" ) );
1992 wxString expr =
"A.NetClass == 'Power' && !A.intersectsArea('NoRouting')";
1995 BOOST_CHECK( expr.Contains(
"&&" ) );
1996 BOOST_CHECK( expr.Contains(
"!" ) );
2002 wxString expr =
"A.intersectsArea('underFPGA') || A.intersectsArea('underDDR')";
2005 BOOST_CHECK( expr.Contains(
"||" ) );
2006 BOOST_CHECK( expr.Contains(
"underFPGA" ) );
2007 BOOST_CHECK( expr.Contains(
"underDDR" ) );
2013 while( ( pos = expr.find(
"||", pos ) ) != wxString::npos )
2036 for(
KICAD_T type : excludedTypes )
2043 BOOST_CHECK(
true );
2047 BOOST_FAIL(
"Excluded type not handled in filter" );
2052 for(
KICAD_T type : includedTypes )
2059 BOOST_FAIL(
"Included type incorrectly matched by filter" );
2063 BOOST_CHECK(
true );
2076 entry.
condition =
"A.NetClass == 'Power'";
2080 auto data = std::make_shared<DRC_RE_SILK_TO_SOLDERMASK_CLEARANCE_CONSTRAINT_DATA>();
2081 data->SetRuleName(
"SilkMaskTest" );
2082 data->SetConstraintCode(
"silk_clearance" );
2083 data->SetRuleCondition(
"A.NetClass == 'Power'" );
2084 data->SetNumericInputValue( 0.15 );
2088 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = { entry };
2092 BOOST_CHECK(
result.Contains(
"L == 'F.Mask'" ) );
2093 BOOST_CHECK(
result.Contains(
"Power" ) );
2094 BOOST_CHECK( !
result.Contains(
"(layer" ) );
2099 while( ( pos =
result.find(
"(condition", pos ) ) != wxString::npos )
2117 auto data = std::make_shared<DRC_RE_SILK_TO_SOLDERMASK_CLEARANCE_CONSTRAINT_DATA>();
2118 data->SetRuleName(
"SilkMaskSimple" );
2119 data->SetConstraintCode(
"silk_clearance" );
2120 data->SetNumericInputValue( 0.1 );
2124 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = { entry };
2127 BOOST_CHECK(
result.Contains(
"L == 'B.Mask'" ) );
2128 BOOST_CHECK(
result.Contains(
"(condition" ) );
2129 BOOST_CHECK( !
result.Contains(
"(layer" ) );
2141 auto data = std::make_shared<DRC_RE_SILK_TO_SILK_CLEARANCE_CONSTRAINT_DATA>();
2142 data->SetRuleName(
"SilkSilkBoth" );
2143 data->SetConstraintCode(
"silk_clearance" );
2144 data->SetNumericInputValue( 0.2 );
2148 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = { entry };
2151 BOOST_CHECK(
result.Contains(
"L == 'F.SilkS' || L == 'B.SilkS'" ) );
2152 BOOST_CHECK(
result.Contains(
"(condition" ) );
2153 BOOST_CHECK( !
result.Contains(
"(layer" ) );
2165 auto data = std::make_shared<DRC_RE_SILK_TO_SILK_CLEARANCE_CONSTRAINT_DATA>();
2166 data->SetRuleName(
"SilkSilkFront" );
2167 data->SetConstraintCode(
"silk_clearance" );
2168 data->SetNumericInputValue( 0.2 );
2172 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = { entry };
2175 BOOST_CHECK(
result.Contains(
"L == 'F.SilkS'" ) );
2176 BOOST_CHECK( !
result.Contains(
"B.SilkS" ) );
2177 BOOST_CHECK(
result.Contains(
"(condition" ) );
2178 BOOST_CHECK( !
result.Contains(
"(layer" ) );
2187 entry.
condition =
"A.NetClass == 'Signal'";
2191 auto data = std::make_shared<DRC_RE_SILK_TO_SILK_CLEARANCE_CONSTRAINT_DATA>();
2192 data->SetRuleName(
"SilkSilkCond" );
2193 data->SetConstraintCode(
"silk_clearance" );
2194 data->SetRuleCondition(
"A.NetClass == 'Signal'" );
2195 data->SetNumericInputValue( 0.2 );
2199 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = { entry };
2202 BOOST_CHECK(
result.Contains(
"L == 'F.SilkS'" ) );
2203 BOOST_CHECK(
result.Contains(
"Signal" ) );
2208 while( ( pos =
result.find(
"(condition", pos ) ) != wxString::npos )
2221 BOOST_CHECK( result1.
isValid );
2222 BOOST_CHECK( result1.
errors.empty() );
2227 BOOST_CHECK( result2.
isValid );
2228 BOOST_CHECK( result2.
errors.empty() );
2236 BOOST_CHECK( !
result.isValid );
2237 BOOST_CHECK( !
result.errors.empty() );
2239 bool foundError =
false;
2240 for(
const auto& error :
result.errors )
2242 if( error.find(
"At least one via type must be selected" ) != std::string::npos )
2245 BOOST_CHECK( foundError );
2257 BOOST_REQUIRE_EQUAL( clauses.size(), 1 );
2258 BOOST_CHECK( clauses[0].Contains(
"disallow via" ) );
2259 BOOST_CHECK( !clauses[0].Contains(
"through_via" ) );
2264 BOOST_REQUIRE_EQUAL( clauses.size(), 1 );
2265 BOOST_CHECK( clauses[0].Contains(
"micro_via" ) );
2266 BOOST_CHECK( !clauses[0].Contains(
"through_via" ) );
2267 BOOST_CHECK( !clauses[0].Contains(
"blind_via" ) );
2268 BOOST_CHECK( !clauses[0].Contains(
"buried_via" ) );
2273 BOOST_REQUIRE_EQUAL( clauses.size(), 1 );
2274 BOOST_CHECK( clauses[0].Contains(
"through_via" ) );
2275 BOOST_CHECK( clauses[0].Contains(
"blind_via" ) );
2276 BOOST_CHECK( !clauses[0].Contains(
"micro_via" ) );
2277 BOOST_CHECK( !clauses[0].Contains(
"buried_via" ) );
2282 BOOST_CHECK( clauses.empty() );
2288 wxString originalText =
"(version 1)\n"
2289 "(rule \"ViasUnderSmdTest\"\n"
2290 " (constraint disallow micro_via)\n"
2291 " (condition \"A.Pad_Type == 'SMD' && A.insideArea('BGA*')\"))";
2294 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( originalText );
2296 BOOST_REQUIRE_EQUAL( entries.size(), 1 );
2299 auto viaData = std::dynamic_pointer_cast<DRC_RE_VIAS_UNDER_SMD_CONSTRAINT_DATA>( entries[0].constraintData );
2303 BOOST_CHECK( !viaData->GetDisallowThroughVias() );
2304 BOOST_CHECK( viaData->GetDisallowMicroVias() );
2305 BOOST_CHECK( !viaData->GetDisallowBlindVias() );
2306 BOOST_CHECK( !viaData->GetDisallowBuriedVias() );
2309 entries[0].wasEdited =
true;
2314 BOOST_CHECK( savedText.Contains(
"micro_via" ) );
2315 BOOST_CHECK( !savedText.Contains(
"through_via" ) );
2316 BOOST_CHECK( !savedText.Contains(
"blind_via" ) );
2317 BOOST_CHECK( !savedText.Contains(
"buried_via" ) );
2320 std::vector<DRC_RE_LOADED_PANEL_ENTRY> reloadedEntries = loader.
LoadFromString( savedText );
2321 BOOST_REQUIRE_EQUAL( reloadedEntries.size(), 1 );
2324 std::dynamic_pointer_cast<DRC_RE_VIAS_UNDER_SMD_CONSTRAINT_DATA>( reloadedEntries[0].constraintData );
2327 BOOST_CHECK( !reloadedData->GetDisallowThroughVias() );
2328 BOOST_CHECK( reloadedData->GetDisallowMicroVias() );
2329 BOOST_CHECK( !reloadedData->GetDisallowBlindVias() );
2330 BOOST_CHECK( !reloadedData->GetDisallowBuriedVias() );
2333 wxString blanketText =
"(version 1)\n"
2334 "(rule \"BlanketVia\"\n"
2335 " (constraint disallow via)\n"
2336 " (condition \"A.Pad_Type == 'SMD' && A.insideArea('BGA*')\"))";
2339 BOOST_REQUIRE_EQUAL( entries.size(), 1 );
2341 auto blanketData = std::dynamic_pointer_cast<DRC_RE_VIAS_UNDER_SMD_CONSTRAINT_DATA>( entries[0].constraintData );
2344 BOOST_CHECK( blanketData->GetDisallowThroughVias() );
2345 BOOST_CHECK( blanketData->GetDisallowMicroVias() );
2346 BOOST_CHECK( blanketData->GetDisallowBlindVias() );
2347 BOOST_CHECK( blanketData->GetDisallowBuriedVias() );
2350 wxString multiText =
"(version 1)\n"
2351 "(rule \"TwoVias\"\n"
2352 " (constraint disallow through_via blind_via)\n"
2353 " (condition \"A.Pad_Type == 'SMD' && A.insideArea('BGA*')\"))";
2356 BOOST_REQUIRE_EQUAL( entries.size(), 1 );
2358 auto multiData = std::dynamic_pointer_cast<DRC_RE_VIAS_UNDER_SMD_CONSTRAINT_DATA>( entries[0].constraintData );
2361 BOOST_CHECK( multiData->GetDisallowThroughVias() );
2362 BOOST_CHECK( !multiData->GetDisallowMicroVias() );
2363 BOOST_CHECK( multiData->GetDisallowBlindVias() );
2364 BOOST_CHECK( !multiData->GetDisallowBuriedVias() );
2366 entries[0].wasEdited =
true;
2369 BOOST_CHECK( savedText.Contains(
"through_via" ) );
2370 BOOST_CHECK( savedText.Contains(
"blind_via" ) );
2371 BOOST_CHECK( !savedText.Contains(
"micro_via" ) );
2372 BOOST_CHECK( !savedText.Contains(
"buried_via" ) );
2378 wxString ruleText =
"(version 1)\n"
2379 "(rule \"Angled\"\n"
2380 " (constraint assertion \"A.Orientation == 46 deg\")\n"
2384 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( ruleText );
2386 BOOST_REQUIRE_EQUAL( entries.size(), 1 );
2389 auto customData = std::dynamic_pointer_cast<DRC_RE_CUSTOM_RULE_CONSTRAINT_DATA>( entries[0].constraintData );
2391 BOOST_CHECK( customData->GetRuleText().Contains( wxS(
"46 deg" ) ) );
2392 BOOST_CHECK( !customData->GetRuleText().Contains( wxS(
"Angled" ) ) );
2395 wxString ruleTextUnquoted =
"(version 1)\n"
2397 " (constraint assertion \"A.Orientation == 46 deg\")\n"
2402 BOOST_REQUIRE_EQUAL( entries.size(), 1 );
2405 auto customDataUnquoted =
2406 std::dynamic_pointer_cast<DRC_RE_CUSTOM_RULE_CONSTRAINT_DATA>( entries[0].constraintData );
2408 BOOST_CHECK( customDataUnquoted->GetRuleText().Contains( wxS(
"46 deg" ) ) );
2409 BOOST_CHECK( !customDataUnquoted->GetRuleText().Contains( wxS(
"Angled" ) ) );
2414 wxString ruleText =
"(version 1)\n"
2415 "(rule \"Standard\"\n"
2416 " (constraint assertion \"A.Orientation == 0 deg || A.Orientation == 90 deg\")\n"
2420 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( ruleText );
2422 BOOST_REQUIRE_EQUAL( entries.size(), 1 );
2426 std::dynamic_pointer_cast<DRC_RE_ALLOWED_ORIENTATION_CONSTRAINT_DATA>( entries[0].constraintData );
2428 BOOST_CHECK( orientData->GetIsZeroDegreesAllowed() );
2429 BOOST_CHECK( orientData->GetIsNinetyDegreesAllowed() );
2430 BOOST_CHECK( !orientData->GetIsOneEightyDegreesAllowed() );
2431 BOOST_CHECK( !orientData->GetIsAllDegreesAllowed() );
2434 wxString singleLine =
"(version 1) (rule \"SingleLine\" (constraint assertion \"A.Orientation == 0 deg || "
2435 "A.Orientation == 90 deg\"))";
2439 BOOST_REQUIRE_EQUAL( entries.size(), 1 );
2442 auto singleLineData =
2443 std::dynamic_pointer_cast<DRC_RE_ALLOWED_ORIENTATION_CONSTRAINT_DATA>( entries[0].constraintData );
2445 BOOST_CHECK( singleLineData->GetIsZeroDegreesAllowed() );
2446 BOOST_CHECK( singleLineData->GetIsNinetyDegreesAllowed() );
2447 BOOST_CHECK( !singleLineData->GetIsOneEightyDegreesAllowed() );
2448 BOOST_CHECK( !singleLineData->GetIsAllDegreesAllowed() );
2453 wxString ruleText =
"(version 1)\n"
2454 "(rule \"AllAngles\"\n"
2455 " (constraint assertion \"A.Orientation == 0 deg || A.Orientation == 90 deg || "
2456 "A.Orientation == 180 deg || A.Orientation == 270 deg\")\n"
2460 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( ruleText );
2462 BOOST_REQUIRE_EQUAL( entries.size(), 1 );
2466 std::dynamic_pointer_cast<DRC_RE_ALLOWED_ORIENTATION_CONSTRAINT_DATA>( entries[0].constraintData );
2468 BOOST_CHECK( orientData->GetIsAllDegreesAllowed() );
2469 BOOST_CHECK( orientData->GetIsZeroDegreesAllowed() );
2470 BOOST_CHECK( orientData->GetIsNinetyDegreesAllowed() );
2471 BOOST_CHECK( orientData->GetIsOneEightyDegreesAllowed() );
2472 BOOST_CHECK( orientData->GetIsTwoSeventyDegreesAllowed() );
2477 wxString ruleText =
"(version 1)\n"
2478 "(rule \"InnerLayers\"\n"
2479 " (constraint assertion \"A.Layer == 'F.Cu' || A.Layer == 'In1.Cu'\")\n"
2483 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( ruleText );
2485 BOOST_REQUIRE_EQUAL( entries.size(), 1 );
2488 auto customData = std::dynamic_pointer_cast<DRC_RE_CUSTOM_RULE_CONSTRAINT_DATA>( entries[0].constraintData );
2490 BOOST_CHECK( customData->GetRuleText().Contains( wxS(
"In1.Cu" ) ) );
2495 wxString ruleText =
"(version 1)\n"
2496 "(rule \"StandardLayers\"\n"
2497 " (constraint assertion \"A.Layer == 'F.Cu' || A.Layer == 'B.Cu'\")\n"
2501 std::vector<DRC_RE_LOADED_PANEL_ENTRY> entries = loader.
LoadFromString( ruleText );
2503 BOOST_REQUIRE_EQUAL( entries.size(), 1 );
2506 auto layerData = std::dynamic_pointer_cast<DRC_RE_PERMITTED_LAYERS_CONSTRAINT_DATA>( entries[0].constraintData );
2508 BOOST_CHECK( layerData->GetTopLayerEnabled() );
2509 BOOST_CHECK( layerData->GetBottomLayerEnabled() );
MINOPTMAX< int > & Value()
Maps DRC rule constraints to appropriate editor panels.
DRC_RULE_EDITOR_CONSTRAINT_NAME GetPanelForConstraint(DRC_CONSTRAINT_T aConstraintType)
Get the panel type for a single constraint.
bool CanPanelLoad(DRC_RULE_EDITOR_CONSTRAINT_NAME aPanel, const std::set< DRC_CONSTRAINT_T > &aConstraints)
Check if a specific panel type can load a set of constraints.
std::vector< DRC_PANEL_MATCH > MatchRule(const DRC_RULE &aRule)
Match a DRC rule to one or more panel types.
VALIDATION_RESULT Validate() const override
Validates the constraint data.
VALIDATION_RESULT Validate() const override
Validates the constraint data.
wxString GetConstraintCode() const
void SetConstraintCode(const wxString &aCode)
wxString GetRuleCondition()
void SetRuleCondition(const wxString &aRuleCondition)
Simple constraint data object used by custom rules.
void SetRuleText(const wxString &aText)
VALIDATION_RESULT Validate() const override
Validates the constraint data.
VALIDATION_RESULT Validate() const override
Validates the constraint data.
VALIDATION_RESULT Validate() const override
Validates the constraint data.
VALIDATION_RESULT Validate() const override
Validates the constraint data.
double GetWidthTolerance()
wxString GenerateRule(const RULE_GENERATION_CONTEXT &aContext) override
VALIDATION_RESULT Validate() const override
Validates the constraint data.
VALIDATION_RESULT Validate() const override
Validates the constraint data.
std::vector< wxString > GetConstraintClauses(const RULE_GENERATION_CONTEXT &aContext) const override
Returns just the constraint clauses without the rule wrapper.
double GetMaxViaDiameter()
double GetMinViaHoleSize()
double GetMinViaDiameter()
double GetMaxViaHoleSize()
VALIDATION_RESULT Validate() const override
Validates the constraint data.
wxString GenerateRule(const RULE_GENERATION_CONTEXT &aContext) override
static bool SaveRules(const wxString &aFilename, const std::vector< std::shared_ptr< DRC_RE_BASE_CONSTRAINT_DATA > > &aRules, const BOARD *aBoard)
static void RegisterRuleConverter(RuleConverter aConverter)
static std::vector< std::shared_ptr< DRC_RE_BASE_CONSTRAINT_DATA > > ParseRules(const wxString &aRules)
Loads DRC rules from .kicad_dru files and converts them to panel entries.
std::vector< DRC_RE_LOADED_PANEL_ENTRY > LoadFromString(const wxString &aRulesText)
Load rules from a text string.
Saves DRC panel entries back to .kicad_dru files.
wxString GenerateRulesText(const std::vector< DRC_RE_LOADED_PANEL_ENTRY > &aEntries, const BOARD *aBoard=nullptr)
Generate rule text from panel entries.
void AddConstraint(DRC_CONSTRAINT &aConstraint)
LSET is a set of PCB_LAYER_IDs.
wxString GetRuleName()
Get the name of the rule.
@ VIA_DIAMETER_CONSTRAINT
@ DIFF_PAIR_GAP_CONSTRAINT
@ SILK_CLEARANCE_CONSTRAINT
@ EDGE_CLEARANCE_CONSTRAINT
@ TEXT_THICKNESS_CONSTRAINT
@ MAX_UNCOUPLED_CONSTRAINT
@ HOLE_CLEARANCE_CONSTRAINT
@ HOLE_TO_HOLE_CONSTRAINT
@ SILK_TO_SOLDERMASK_CLEARANCE
@ MINIMUM_CONNECTION_WIDTH
@ MINIMUM_TEXT_HEIGHT_AND_THICKNESS
@ COPPER_TO_HOLE_CLEARANCE
@ MATCHED_LENGTH_DIFF_PAIR
@ COPPER_TO_EDGE_CLEARANCE
Represents a rule loaded from a .kicad_dru file and mapped to a panel.
wxString originalRuleText
std::shared_ptr< DRC_RE_BASE_CONSTRAINT_DATA > constraintData
DRC_RULE_EDITOR_CONSTRAINT_NAME panelType
wxString conditionExpression
Structure representing a node in a rule tree, collection of this used for building the rule tree.
std::shared_ptr< RULE_EDITOR_DATA_BASE > m_nodeData
Result of a validation operation.
std::vector< wxString > errors
BOOST_AUTO_TEST_CASE(HorizontalAlignment)
BOOST_AUTO_TEST_SUITE(CadstarPartParser)
BOOST_AUTO_TEST_CASE(RoundTripViaStyle)
BOOST_REQUIRE(intersection.has_value()==c.ExpectedIntersection.has_value())
BOOST_AUTO_TEST_SUITE_END()
wxString result
Test unit parsing edge cases and error handling.
BOOST_CHECK_EQUAL(result, "25.4")
KICAD_T
The set of class identification values stored in EDA_ITEM::m_structType.
@ PCB_SHAPE_T
class PCB_SHAPE, a segment not on copper layers
@ PCB_GENERATOR_T
class PCB_GENERATOR, generator on a layer
@ PCB_VIA_T
class PCB_VIA, a via (like a track segment on a copper layer)
@ PCB_GROUP_T
class PCB_GROUP, a set of BOARD_ITEMs
@ PCB_ZONE_T
class ZONE, a copper pour area
@ PCB_TEXT_T
class PCB_TEXT, text on a layer
@ PCB_FOOTPRINT_T
class FOOTPRINT, a footprint
@ PCB_PAD_T
class PAD, a pad in a footprint
@ PCB_NETINFO_T
class NETINFO_ITEM, a description of a net
@ PCB_TRACE_T
class PCB_TRACK, a track segment (segment on a copper layer)