83 *aErrorText = wxEmptyString;
86 [&](
const wxString& aMessage ) -> wxString
89 *aErrorText = aMessage;
94 auto escapeQuotedRuleText =
95 [](
const wxString& aText ) -> wxString
97 wxString escaped = aText;
98 escaped.Replace(
"\\",
"\\\\" );
99 escaped.Replace(
"\"",
"\\\"" );
103 auto hasConstraintOption =
104 [](
const kiapi::board::CustomRuleConstraint& aConstraint,
105 const kiapi::board::CustomRuleConstraintOption aOption ) ->
bool
107 for(
int optionValue : aConstraint.options() )
109 if(
static_cast<kiapi::board::CustomRuleConstraintOption
>( optionValue ) == aOption )
116 auto getConstraintToken =
117 [](
const kiapi::board::CustomRuleConstraintType aType,
118 bool* aAddWithinDiffPairs ) -> std::optional<wxString>
120 if( aAddWithinDiffPairs )
121 *aAddWithinDiffPairs =
false;
125 case kiapi::board::CRCT_CLEARANCE:
return wxS(
"clearance" );
126 case kiapi::board::CRCT_CREEPAGE:
return wxS(
"creepage" );
127 case kiapi::board::CRCT_HOLE_CLEARANCE:
return wxS(
"hole_clearance" );
128 case kiapi::board::CRCT_HOLE_TO_HOLE:
return wxS(
"hole_to_hole" );
129 case kiapi::board::CRCT_EDGE_CLEARANCE:
return wxS(
"edge_clearance" );
130 case kiapi::board::CRCT_HOLE_SIZE:
return wxS(
"hole_size" );
131 case kiapi::board::CRCT_COURTYARD_CLEARANCE:
return wxS(
"courtyard_clearance" );
132 case kiapi::board::CRCT_SILK_CLEARANCE:
return wxS(
"silk_clearance" );
133 case kiapi::board::CRCT_TEXT_HEIGHT:
return wxS(
"text_height" );
134 case kiapi::board::CRCT_TEXT_THICKNESS:
return wxS(
"text_thickness" );
135 case kiapi::board::CRCT_TRACK_WIDTH:
return wxS(
"track_width" );
136 case kiapi::board::CRCT_TRACK_SEGMENT_LENGTH:
return wxS(
"track_segment_length" );
137 case kiapi::board::CRCT_ANNULAR_WIDTH:
return wxS(
"annular_width" );
138 case kiapi::board::CRCT_ZONE_CONNECTION:
return wxS(
"zone_connection" );
139 case kiapi::board::CRCT_THERMAL_RELIEF_GAP:
return wxS(
"thermal_relief_gap" );
140 case kiapi::board::CRCT_THERMAL_SPOKE_WIDTH:
return wxS(
"thermal_spoke_width" );
141 case kiapi::board::CRCT_MIN_RESOLVED_SPOKES:
return wxS(
"min_resolved_spokes" );
142 case kiapi::board::CRCT_SOLDER_MASK_EXPANSION:
return wxS(
"solder_mask_expansion" );
143 case kiapi::board::CRCT_SOLDER_PASTE_ABS_MARGIN:
return wxS(
"solder_paste_abs_margin" );
144 case kiapi::board::CRCT_SOLDER_PASTE_REL_MARGIN:
return wxS(
"solder_paste_rel_margin" );
145 case kiapi::board::CRCT_DISALLOW:
return wxS(
"disallow" );
146 case kiapi::board::CRCT_VIA_DIAMETER:
return wxS(
"via_diameter" );
147 case kiapi::board::CRCT_LENGTH:
return wxS(
"length" );
148 case kiapi::board::CRCT_SKEW:
return wxS(
"skew" );
149 case kiapi::board::CRCT_DIFF_PAIR_GAP:
return wxS(
"diff_pair_gap" );
150 case kiapi::board::CRCT_MAX_UNCOUPLED:
return wxS(
"diff_pair_uncoupled" );
151 case kiapi::board::CRCT_DIFF_PAIR_INTRA_SKEW:
153 if( aAddWithinDiffPairs )
154 *aAddWithinDiffPairs =
true;
156 return wxS(
"skew" );
158 case kiapi::board::CRCT_VIA_COUNT:
return wxS(
"via_count" );
159 case kiapi::board::CRCT_PHYSICAL_CLEARANCE:
return wxS(
"physical_clearance" );
160 case kiapi::board::CRCT_PHYSICAL_HOLE_CLEARANCE:
return wxS(
"physical_hole_clearance" );
161 case kiapi::board::CRCT_ASSERTION:
return wxS(
"assertion" );
162 case kiapi::board::CRCT_CONNECTION_WIDTH:
return wxS(
"connection_width" );
163 case kiapi::board::CRCT_TRACK_ANGLE:
return wxS(
"track_angle" );
164 case kiapi::board::CRCT_VIA_DANGLING:
return wxS(
"via_dangling" );
165 case kiapi::board::CRCT_BRIDGED_MASK:
return wxS(
"bridged_mask" );
166 case kiapi::board::CRCT_SOLDER_MASK_SLIVER:
return wxS(
"solder_mask_sliver" );
168 case kiapi::board::CRCT_UNKNOWN:
174 auto formatNumericValueForConstraint =
175 [&](
const kiapi::board::CustomRuleConstraint& aConstraint,
176 const int aValue ) -> wxString
178 if( aConstraint.type() == kiapi::board::CRCT_TRACK_ANGLE )
179 return wxString::Format(
"%ddeg", aValue );
181 switch( aConstraint.type() )
183 case kiapi::board::CRCT_VIA_COUNT:
184 case kiapi::board::CRCT_MIN_RESOLVED_SPOKES:
185 case kiapi::board::CRCT_VIA_DANGLING:
186 case kiapi::board::CRCT_BRIDGED_MASK:
187 return wxString::Format(
"%d", aValue );
193 if( hasConstraintOption( aConstraint, kiapi::board::CRCO_TIME_DOMAIN ) )
194 return wxString::Format(
"%dps", aValue );
197 return formatted +
"mm";
201 [](
const kiapi::common::types::RuleSeverity aSeverity ) -> wxString
205 case kiapi::common::types::RuleSeverity::RS_WARNING:
return wxS(
"warning" );
206 case kiapi::common::types::RuleSeverity::RS_ERROR:
return wxS(
"error" );
207 case kiapi::common::types::RuleSeverity::RS_EXCLUSION:
return wxS(
"exclusion" );
208 case kiapi::common::types::RuleSeverity::RS_IGNORE:
return wxS(
"ignore" );
211 return wxEmptyString;
215 if( aRule.name().empty() )
216 return fail( wxS(
"Rules must have a name" ) );
219 ruleText <<
"(rule \"" << escapeQuotedRuleText( wxString::FromUTF8( aRule.name() ) ) <<
"\"\n";
221 if( aRule.has_comments() && !aRule.comments().empty() )
223 wxArrayString commentLines = wxSplit( wxString::FromUTF8( aRule.comments() ),
'\n',
'\0' );
225 for(
const wxString& line : commentLines )
226 ruleText <<
"\t# " << line <<
"\n";
229 if( !aRule.condition().empty() )
231 ruleText <<
"\t(condition \""
232 << escapeQuotedRuleText( wxString::FromUTF8( aRule.condition() ) )
236 if( aRule.layer_condition_case() == kiapi::board::CustomRule::LayerConditionCase::kLayerMode )
238 switch( aRule.layer_mode() )
240 case kiapi::board::CRLM_OUTER:
241 ruleText <<
"\t(layer outer)\n";
244 case kiapi::board::CRLM_INNER:
245 ruleText <<
"\t(layer inner)\n";
248 case kiapi::board::CRLM_UNKNOWN:
253 else if( aRule.layer_condition_case() == kiapi::board::CustomRule::LayerConditionCase::kSingleLayer )
259 return fail( wxS(
"Invalid single_layer value in custom rule" ) );
264 for(
const kiapi::board::CustomRuleConstraint& constraint : aRule.constraints() )
266 bool addWithinDiffPairs =
false;
267 std::optional<wxString> token = getConstraintToken( constraint.type(), &addWithinDiffPairs );
270 return fail( wxS(
"Unsupported custom rule constraint type" ) );
272 wxString
text = wxS(
"\t(constraint " ) + *token;
274 if( constraint.has_name() && !constraint.name().empty() )
276 text += wxS(
" (name \"" )
277 + escapeQuotedRuleText( wxString::FromUTF8( constraint.name() ) )
281 if( constraint.value_case() == kiapi::board::CustomRuleConstraint::ValueCase::kDisallow )
283 for(
int disallowTypeValue : constraint.disallow().types() )
285 kiapi::board::CustomRuleDisallowType disallowType =
286 static_cast<kiapi::board::CustomRuleDisallowType
>( disallowTypeValue );
288 switch( disallowType )
290 case kiapi::board::CRDT_THROUGH_VIAS:
text += wxS(
" through_via" );
break;
291 case kiapi::board::CRDT_MICRO_VIAS:
text += wxS(
" micro_via" );
break;
292 case kiapi::board::CRDT_BLIND_VIAS:
text += wxS(
" blind_via" );
break;
293 case kiapi::board::CRDT_BURIED_VIAS:
text += wxS(
" buried_via" );
break;
294 case kiapi::board::CRDT_TRACKS:
text += wxS(
" track" );
break;
295 case kiapi::board::CRDT_PADS:
text += wxS(
" pad" );
break;
296 case kiapi::board::CRDT_ZONES:
text += wxS(
" zone" );
break;
297 case kiapi::board::CRDT_TEXTS:
text += wxS(
" text" );
break;
298 case kiapi::board::CRDT_GRAPHICS:
text += wxS(
" graphic" );
break;
299 case kiapi::board::CRDT_HOLES:
text += wxS(
" hole" );
break;
300 case kiapi::board::CRDT_FOOTPRINTS:
text += wxS(
" footprint" );
break;
302 case kiapi::board::CRDT_UNKNOWN:
308 else if( constraint.value_case() == kiapi::board::CustomRuleConstraint::ValueCase::kZoneConnection )
310 switch( constraint.zone_connection() )
312 case kiapi::board::types::ZCS_FULL:
313 text += wxS(
" solid" );
316 case kiapi::board::types::ZCS_THERMAL:
317 text += wxS(
" thermal_reliefs" );
320 case kiapi::board::types::ZCS_NONE:
321 text += wxS(
" none" );
325 return fail( wxS(
"Unsupported zone connection style" ) );
328 else if( constraint.value_case() == kiapi::board::CustomRuleConstraint::ValueCase::kAssertionExpression )
331 + escapeQuotedRuleText( wxString::FromUTF8( constraint.assertion_expression() ) )
334 else if( constraint.value_case() == kiapi::board::CustomRuleConstraint::ValueCase::kNumeric )
336 const kiapi::common::types::MinOptMax& numeric = constraint.numeric();
338 if( numeric.has_min() )
339 text += wxS(
" (min " )
340 + formatNumericValueForConstraint( constraint, numeric.min() )
343 if( numeric.has_opt() )
344 text += wxS(
" (opt " )
345 + formatNumericValueForConstraint( constraint, numeric.opt() )
348 if( numeric.has_max() )
349 text += wxS(
" (max " )
350 + formatNumericValueForConstraint( constraint, numeric.max() )
354 if( addWithinDiffPairs
355 || hasConstraintOption( constraint, kiapi::board::CRCO_SKEW_WITHIN_DIFF_PAIRS ) )
357 text += wxS(
" (within_diff_pairs)" );
361 ruleText <<
text <<
"\n";
364 wxString severity = severityToken( aRule.severity() );
366 if( !severity.IsEmpty() )
367 ruleText <<
"\t(severity " << severity <<
")\n";