85 *aErrorText = wxEmptyString;
88 [&](
const wxString& aMessage ) -> wxString
91 *aErrorText = aMessage;
96 auto escapeQuotedRuleText =
97 [](
const wxString& aText ) -> wxString
99 wxString escaped = aText;
100 escaped.Replace(
"\\",
"\\\\" );
101 escaped.Replace(
"\"",
"\\\"" );
105 auto hasConstraintOption =
106 [](
const kiapi::board::CustomRuleConstraint& aConstraint,
107 const kiapi::board::CustomRuleConstraintOption aOption ) ->
bool
109 for(
int optionValue : aConstraint.options() )
111 if(
static_cast<kiapi::board::CustomRuleConstraintOption
>( optionValue ) == aOption )
118 auto getConstraintToken =
119 [](
const kiapi::board::CustomRuleConstraintType aType,
120 bool* aAddWithinDiffPairs ) -> std::optional<wxString>
122 if( aAddWithinDiffPairs )
123 *aAddWithinDiffPairs =
false;
127 case kiapi::board::CRCT_CLEARANCE:
return wxS(
"clearance" );
128 case kiapi::board::CRCT_CREEPAGE:
return wxS(
"creepage" );
129 case kiapi::board::CRCT_HOLE_CLEARANCE:
return wxS(
"hole_clearance" );
130 case kiapi::board::CRCT_HOLE_TO_HOLE:
return wxS(
"hole_to_hole" );
131 case kiapi::board::CRCT_EDGE_CLEARANCE:
return wxS(
"edge_clearance" );
132 case kiapi::board::CRCT_HOLE_SIZE:
return wxS(
"hole_size" );
133 case kiapi::board::CRCT_COURTYARD_CLEARANCE:
return wxS(
"courtyard_clearance" );
134 case kiapi::board::CRCT_SILK_CLEARANCE:
return wxS(
"silk_clearance" );
135 case kiapi::board::CRCT_TEXT_HEIGHT:
return wxS(
"text_height" );
136 case kiapi::board::CRCT_TEXT_THICKNESS:
return wxS(
"text_thickness" );
137 case kiapi::board::CRCT_TRACK_WIDTH:
return wxS(
"track_width" );
138 case kiapi::board::CRCT_TRACK_SEGMENT_LENGTH:
return wxS(
"track_segment_length" );
139 case kiapi::board::CRCT_ANNULAR_WIDTH:
return wxS(
"annular_width" );
140 case kiapi::board::CRCT_ZONE_CONNECTION:
return wxS(
"zone_connection" );
141 case kiapi::board::CRCT_THERMAL_RELIEF_GAP:
return wxS(
"thermal_relief_gap" );
142 case kiapi::board::CRCT_THERMAL_SPOKE_WIDTH:
return wxS(
"thermal_spoke_width" );
143 case kiapi::board::CRCT_MIN_RESOLVED_SPOKES:
return wxS(
"min_resolved_spokes" );
144 case kiapi::board::CRCT_SOLDER_MASK_EXPANSION:
return wxS(
"solder_mask_expansion" );
145 case kiapi::board::CRCT_SOLDER_PASTE_ABS_MARGIN:
return wxS(
"solder_paste_abs_margin" );
146 case kiapi::board::CRCT_SOLDER_PASTE_REL_MARGIN:
return wxS(
"solder_paste_rel_margin" );
147 case kiapi::board::CRCT_DISALLOW:
return wxS(
"disallow" );
148 case kiapi::board::CRCT_VIA_DIAMETER:
return wxS(
"via_diameter" );
149 case kiapi::board::CRCT_LENGTH:
return wxS(
"length" );
150 case kiapi::board::CRCT_SKEW:
return wxS(
"skew" );
151 case kiapi::board::CRCT_DIFF_PAIR_GAP:
return wxS(
"diff_pair_gap" );
152 case kiapi::board::CRCT_MAX_UNCOUPLED:
return wxS(
"diff_pair_uncoupled" );
153 case kiapi::board::CRCT_DIFF_PAIR_INTRA_SKEW:
155 if( aAddWithinDiffPairs )
156 *aAddWithinDiffPairs =
true;
158 return wxS(
"skew" );
160 case kiapi::board::CRCT_VIA_COUNT:
return wxS(
"via_count" );
161 case kiapi::board::CRCT_PHYSICAL_CLEARANCE:
return wxS(
"physical_clearance" );
162 case kiapi::board::CRCT_PHYSICAL_HOLE_CLEARANCE:
return wxS(
"physical_hole_clearance" );
163 case kiapi::board::CRCT_ASSERTION:
return wxS(
"assertion" );
164 case kiapi::board::CRCT_CONNECTION_WIDTH:
return wxS(
"connection_width" );
165 case kiapi::board::CRCT_TRACK_ANGLE:
return wxS(
"track_angle" );
166 case kiapi::board::CRCT_VIA_DANGLING:
return wxS(
"via_dangling" );
167 case kiapi::board::CRCT_BRIDGED_MASK:
return wxS(
"bridged_mask" );
168 case kiapi::board::CRCT_SOLDER_MASK_SLIVER:
return wxS(
"solder_mask_sliver" );
170 case kiapi::board::CRCT_UNKNOWN:
176 auto formatNumericValueForConstraint =
177 [&](
const kiapi::board::CustomRuleConstraint& aConstraint,
178 const int aValue ) -> wxString
180 if( aConstraint.type() == kiapi::board::CRCT_TRACK_ANGLE )
181 return wxString::Format(
"%ddeg", aValue );
183 switch( aConstraint.type() )
185 case kiapi::board::CRCT_VIA_COUNT:
186 case kiapi::board::CRCT_MIN_RESOLVED_SPOKES:
187 case kiapi::board::CRCT_VIA_DANGLING:
188 case kiapi::board::CRCT_BRIDGED_MASK:
189 return wxString::Format(
"%d", aValue );
195 if( hasConstraintOption( aConstraint, kiapi::board::CRCO_TIME_DOMAIN ) )
196 return wxString::Format(
"%dps", aValue );
199 return formatted +
"mm";
203 [](
const kiapi::common::types::RuleSeverity aSeverity ) -> wxString
207 case kiapi::common::types::RuleSeverity::RS_WARNING:
return wxS(
"warning" );
208 case kiapi::common::types::RuleSeverity::RS_ERROR:
return wxS(
"error" );
209 case kiapi::common::types::RuleSeverity::RS_EXCLUSION:
return wxS(
"exclusion" );
210 case kiapi::common::types::RuleSeverity::RS_IGNORE:
return wxS(
"ignore" );
213 return wxEmptyString;
217 if( aRule.name().empty() )
218 return fail( wxS(
"Rules must have a name" ) );
221 ruleText <<
"(rule \"" << escapeQuotedRuleText( wxString::FromUTF8( aRule.name() ) ) <<
"\"\n";
223 if( aRule.has_comments() && !aRule.comments().empty() )
225 wxArrayString commentLines = wxSplit( wxString::FromUTF8( aRule.comments() ),
'\n',
'\0' );
227 for(
const wxString& line : commentLines )
228 ruleText <<
"\t# " << line <<
"\n";
231 if( !aRule.condition().empty() )
233 ruleText <<
"\t(condition \""
234 << escapeQuotedRuleText( wxString::FromUTF8( aRule.condition() ) )
238 if( aRule.has_layer_mode() )
240 switch( aRule.layer_mode() )
242 case kiapi::board::CRLM_OUTER:
243 ruleText <<
"\t(layer outer)\n";
246 case kiapi::board::CRLM_INNER:
247 ruleText <<
"\t(layer inner)\n";
250 case kiapi::board::CRLM_UNKNOWN:
255 else if( aRule.has_single_layer() )
261 return fail( wxS(
"Invalid single_layer value in custom rule" ) );
266 for(
const kiapi::board::CustomRuleConstraint& constraint : aRule.constraints() )
268 bool addWithinDiffPairs =
false;
269 std::optional<wxString> token = getConstraintToken( constraint.type(), &addWithinDiffPairs );
272 return fail( wxS(
"Unsupported custom rule constraint type" ) );
274 wxString
text = wxS(
"\t(constraint " ) + *token;
276 if( constraint.has_name() && !constraint.name().empty() )
278 text += wxS(
" (name \"" )
279 + escapeQuotedRuleText( wxString::FromUTF8( constraint.name() ) )
283 if( constraint.has_disallow() )
285 for(
int disallowTypeValue : constraint.disallow().types() )
287 kiapi::board::CustomRuleDisallowType disallowType =
288 static_cast<kiapi::board::CustomRuleDisallowType
>( disallowTypeValue );
290 switch( disallowType )
292 case kiapi::board::CRDT_THROUGH_VIAS:
text += wxS(
" through_via" );
break;
293 case kiapi::board::CRDT_MICRO_VIAS:
text += wxS(
" micro_via" );
break;
294 case kiapi::board::CRDT_BLIND_VIAS:
text += wxS(
" blind_via" );
break;
295 case kiapi::board::CRDT_BURIED_VIAS:
text += wxS(
" buried_via" );
break;
296 case kiapi::board::CRDT_TRACKS:
text += wxS(
" track" );
break;
297 case kiapi::board::CRDT_PADS:
text += wxS(
" pad" );
break;
298 case kiapi::board::CRDT_ZONES:
text += wxS(
" zone" );
break;
299 case kiapi::board::CRDT_TEXTS:
text += wxS(
" text" );
break;
300 case kiapi::board::CRDT_GRAPHICS:
text += wxS(
" graphic" );
break;
301 case kiapi::board::CRDT_HOLES:
text += wxS(
" hole" );
break;
302 case kiapi::board::CRDT_FOOTPRINTS:
text += wxS(
" footprint" );
break;
304 case kiapi::board::CRDT_UNKNOWN:
310 else if( constraint.has_zone_connection() )
312 switch( constraint.zone_connection() )
314 case kiapi::board::types::ZCS_FULL:
315 text += wxS(
" solid" );
318 case kiapi::board::types::ZCS_THERMAL:
319 text += wxS(
" thermal_reliefs" );
322 case kiapi::board::types::ZCS_NONE:
323 text += wxS(
" none" );
327 return fail( wxS(
"Unsupported zone connection style" ) );
330 else if( constraint.has_assertion_expression() )
333 + escapeQuotedRuleText( wxString::FromUTF8( constraint.assertion_expression() ) )
336 else if( constraint.has_numeric() )
338 const kiapi::common::types::MinOptMax& numeric = constraint.numeric();
340 if( numeric.has_min() )
341 text += wxS(
" (min " )
342 + formatNumericValueForConstraint( constraint, numeric.min() )
345 if( numeric.has_opt() )
346 text += wxS(
" (opt " )
347 + formatNumericValueForConstraint( constraint, numeric.opt() )
350 if( numeric.has_max() )
351 text += wxS(
" (max " )
352 + formatNumericValueForConstraint( constraint, numeric.max() )
356 if( addWithinDiffPairs
357 || hasConstraintOption( constraint, kiapi::board::CRCO_SKEW_WITHIN_DIFF_PAIRS ) )
359 text += wxS(
" (within_diff_pairs)" );
363 ruleText <<
text <<
"\n";
366 wxString severity = severityToken( aRule.severity() );
368 if( !severity.IsEmpty() )
369 ruleText <<
"\t(severity " << severity <<
")\n";