87 *aErrorText = wxEmptyString;
90 [&](
const wxString& aMessage ) -> wxString
93 *aErrorText = aMessage;
98 auto escapeQuotedRuleText =
99 [](
const wxString& aText ) -> wxString
101 wxString escaped = aText;
102 escaped.Replace(
"\\",
"\\\\" );
103 escaped.Replace(
"\"",
"\\\"" );
107 auto hasConstraintOption =
108 [](
const kiapi::board::CustomRuleConstraint& aConstraint,
109 const kiapi::board::CustomRuleConstraintOption aOption ) ->
bool
111 for(
int optionValue : aConstraint.options() )
113 if(
static_cast<kiapi::board::CustomRuleConstraintOption
>( optionValue ) == aOption )
120 auto getConstraintToken =
121 [](
const kiapi::board::CustomRuleConstraintType aType,
122 bool* aAddWithinDiffPairs ) -> std::optional<wxString>
124 if( aAddWithinDiffPairs )
125 *aAddWithinDiffPairs =
false;
129 case kiapi::board::CRCT_CLEARANCE:
return wxS(
"clearance" );
130 case kiapi::board::CRCT_CREEPAGE:
return wxS(
"creepage" );
131 case kiapi::board::CRCT_HOLE_CLEARANCE:
return wxS(
"hole_clearance" );
132 case kiapi::board::CRCT_HOLE_TO_HOLE:
return wxS(
"hole_to_hole" );
133 case kiapi::board::CRCT_EDGE_CLEARANCE:
return wxS(
"edge_clearance" );
134 case kiapi::board::CRCT_HOLE_SIZE:
return wxS(
"hole_size" );
135 case kiapi::board::CRCT_COURTYARD_CLEARANCE:
return wxS(
"courtyard_clearance" );
136 case kiapi::board::CRCT_SILK_CLEARANCE:
return wxS(
"silk_clearance" );
137 case kiapi::board::CRCT_TEXT_HEIGHT:
return wxS(
"text_height" );
138 case kiapi::board::CRCT_TEXT_THICKNESS:
return wxS(
"text_thickness" );
139 case kiapi::board::CRCT_TRACK_WIDTH:
return wxS(
"track_width" );
140 case kiapi::board::CRCT_TRACK_SEGMENT_LENGTH:
return wxS(
"track_segment_length" );
141 case kiapi::board::CRCT_ANNULAR_WIDTH:
return wxS(
"annular_width" );
142 case kiapi::board::CRCT_ZONE_CONNECTION:
return wxS(
"zone_connection" );
143 case kiapi::board::CRCT_THERMAL_RELIEF_GAP:
return wxS(
"thermal_relief_gap" );
144 case kiapi::board::CRCT_THERMAL_SPOKE_WIDTH:
return wxS(
"thermal_spoke_width" );
145 case kiapi::board::CRCT_MIN_RESOLVED_SPOKES:
return wxS(
"min_resolved_spokes" );
146 case kiapi::board::CRCT_SOLDER_MASK_EXPANSION:
return wxS(
"solder_mask_expansion" );
147 case kiapi::board::CRCT_SOLDER_PASTE_ABS_MARGIN:
return wxS(
"solder_paste_abs_margin" );
148 case kiapi::board::CRCT_SOLDER_PASTE_REL_MARGIN:
return wxS(
"solder_paste_rel_margin" );
149 case kiapi::board::CRCT_DISALLOW:
return wxS(
"disallow" );
150 case kiapi::board::CRCT_VIA_DIAMETER:
return wxS(
"via_diameter" );
151 case kiapi::board::CRCT_LENGTH:
return wxS(
"length" );
152 case kiapi::board::CRCT_SKEW:
return wxS(
"skew" );
153 case kiapi::board::CRCT_DIFF_PAIR_GAP:
return wxS(
"diff_pair_gap" );
154 case kiapi::board::CRCT_MAX_UNCOUPLED:
return wxS(
"diff_pair_uncoupled" );
155 case kiapi::board::CRCT_DIFF_PAIR_INTRA_SKEW:
157 if( aAddWithinDiffPairs )
158 *aAddWithinDiffPairs =
true;
160 return wxS(
"skew" );
162 case kiapi::board::CRCT_VIA_COUNT:
return wxS(
"via_count" );
163 case kiapi::board::CRCT_PHYSICAL_CLEARANCE:
return wxS(
"physical_clearance" );
164 case kiapi::board::CRCT_PHYSICAL_HOLE_CLEARANCE:
return wxS(
"physical_hole_clearance" );
165 case kiapi::board::CRCT_ASSERTION:
return wxS(
"assertion" );
166 case kiapi::board::CRCT_CONNECTION_WIDTH:
return wxS(
"connection_width" );
167 case kiapi::board::CRCT_TRACK_ANGLE:
return wxS(
"track_angle" );
168 case kiapi::board::CRCT_VIA_DANGLING:
return wxS(
"via_dangling" );
169 case kiapi::board::CRCT_BRIDGED_MASK:
return wxS(
"bridged_mask" );
170 case kiapi::board::CRCT_SOLDER_MASK_SLIVER:
return wxS(
"solder_mask_sliver" );
172 case kiapi::board::CRCT_UNKNOWN:
178 auto formatNumericValueForConstraint =
179 [&](
const kiapi::board::CustomRuleConstraint& aConstraint,
180 const int aValue ) -> wxString
182 if( aConstraint.type() == kiapi::board::CRCT_TRACK_ANGLE )
183 return wxString::Format(
"%ddeg", aValue );
185 switch( aConstraint.type() )
187 case kiapi::board::CRCT_VIA_COUNT:
188 case kiapi::board::CRCT_MIN_RESOLVED_SPOKES:
189 case kiapi::board::CRCT_VIA_DANGLING:
190 case kiapi::board::CRCT_BRIDGED_MASK:
191 return wxString::Format(
"%d", aValue );
197 if( hasConstraintOption( aConstraint, kiapi::board::CRCO_TIME_DOMAIN ) )
198 return wxString::Format(
"%dps", aValue );
201 return formatted +
"mm";
205 [](
const kiapi::common::types::RuleSeverity aSeverity ) -> wxString
209 case kiapi::common::types::RuleSeverity::RS_WARNING:
return wxS(
"warning" );
210 case kiapi::common::types::RuleSeverity::RS_ERROR:
return wxS(
"error" );
211 case kiapi::common::types::RuleSeverity::RS_EXCLUSION:
return wxS(
"exclusion" );
212 case kiapi::common::types::RuleSeverity::RS_IGNORE:
return wxS(
"ignore" );
215 return wxEmptyString;
219 if( aRule.name().empty() )
220 return fail( wxS(
"Rules must have a name" ) );
223 ruleText <<
"(rule \"" << escapeQuotedRuleText( wxString::FromUTF8( aRule.name() ) ) <<
"\"\n";
225 if( aRule.has_comments() && !aRule.comments().empty() )
227 wxArrayString commentLines = wxSplit( wxString::FromUTF8( aRule.comments() ),
'\n',
'\0' );
229 for(
const wxString& line : commentLines )
230 ruleText <<
"\t# " << line <<
"\n";
233 if( !aRule.condition().empty() )
235 ruleText <<
"\t(condition \""
236 << escapeQuotedRuleText( wxString::FromUTF8( aRule.condition() ) )
240 if( aRule.layer_condition_case() == kiapi::board::CustomRule::LayerConditionCase::kLayerMode )
242 switch( aRule.layer_mode() )
244 case kiapi::board::CRLM_OUTER:
245 ruleText <<
"\t(layer outer)\n";
248 case kiapi::board::CRLM_INNER:
249 ruleText <<
"\t(layer inner)\n";
252 case kiapi::board::CRLM_UNKNOWN:
257 else if( aRule.layer_condition_case() == kiapi::board::CustomRule::LayerConditionCase::kSingleLayer )
263 return fail( wxS(
"Invalid single_layer value in custom rule" ) );
268 for(
const kiapi::board::CustomRuleConstraint& constraint : aRule.constraints() )
270 bool addWithinDiffPairs =
false;
271 std::optional<wxString> token = getConstraintToken( constraint.type(), &addWithinDiffPairs );
274 return fail( wxS(
"Unsupported custom rule constraint type" ) );
276 wxString
text = wxS(
"\t(constraint " ) + *token;
278 if( constraint.has_name() && !constraint.name().empty() )
280 text += wxS(
" (name \"" )
281 + escapeQuotedRuleText( wxString::FromUTF8( constraint.name() ) )
285 if( constraint.value_case() == kiapi::board::CustomRuleConstraint::ValueCase::kDisallow )
287 for(
int disallowTypeValue : constraint.disallow().types() )
289 kiapi::board::CustomRuleDisallowType disallowType =
290 static_cast<kiapi::board::CustomRuleDisallowType
>( disallowTypeValue );
292 switch( disallowType )
294 case kiapi::board::CRDT_THROUGH_VIAS:
text += wxS(
" through_via" );
break;
295 case kiapi::board::CRDT_MICRO_VIAS:
text += wxS(
" micro_via" );
break;
296 case kiapi::board::CRDT_BLIND_VIAS:
text += wxS(
" blind_via" );
break;
297 case kiapi::board::CRDT_BURIED_VIAS:
text += wxS(
" buried_via" );
break;
298 case kiapi::board::CRDT_TRACKS:
text += wxS(
" track" );
break;
299 case kiapi::board::CRDT_PADS:
text += wxS(
" pad" );
break;
300 case kiapi::board::CRDT_ZONES:
text += wxS(
" zone" );
break;
301 case kiapi::board::CRDT_TEXTS:
text += wxS(
" text" );
break;
302 case kiapi::board::CRDT_GRAPHICS:
text += wxS(
" graphic" );
break;
303 case kiapi::board::CRDT_HOLES:
text += wxS(
" hole" );
break;
304 case kiapi::board::CRDT_FOOTPRINTS:
text += wxS(
" footprint" );
break;
306 case kiapi::board::CRDT_UNKNOWN:
312 else if( constraint.value_case() == kiapi::board::CustomRuleConstraint::ValueCase::kZoneConnection )
314 switch( constraint.zone_connection() )
316 case kiapi::board::types::ZCS_FULL:
317 text += wxS(
" solid" );
320 case kiapi::board::types::ZCS_THERMAL:
321 text += wxS(
" thermal_reliefs" );
324 case kiapi::board::types::ZCS_NONE:
325 text += wxS(
" none" );
329 return fail( wxS(
"Unsupported zone connection style" ) );
332 else if( constraint.value_case() == kiapi::board::CustomRuleConstraint::ValueCase::kAssertionExpression )
335 + escapeQuotedRuleText( wxString::FromUTF8( constraint.assertion_expression() ) )
338 else if( constraint.value_case() == kiapi::board::CustomRuleConstraint::ValueCase::kNumeric )
340 const kiapi::common::types::MinOptMax& numeric = constraint.numeric();
342 if( numeric.has_min() )
343 text += wxS(
" (min " )
344 + formatNumericValueForConstraint( constraint, numeric.min() )
347 if( numeric.has_opt() )
348 text += wxS(
" (opt " )
349 + formatNumericValueForConstraint( constraint, numeric.opt() )
352 if( numeric.has_max() )
353 text += wxS(
" (max " )
354 + formatNumericValueForConstraint( constraint, numeric.max() )
358 if( addWithinDiffPairs
359 || hasConstraintOption( constraint, kiapi::board::CRCO_SKEW_WITHIN_DIFF_PAIRS ) )
361 text += wxS(
" (within_diff_pairs)" );
365 ruleText <<
text <<
"\n";
368 wxString severity = severityToken( aRule.severity() );
370 if( !severity.IsEmpty() )
371 ruleText <<
"\t(severity " << severity <<
")\n";