@@ -252,33 +252,38 @@ unsigned OptTable::findNearest(StringRef Option, std::string &NearestString,
252
252
unsigned MinimumLength) const {
253
253
assert (!Option.empty ());
254
254
255
- // Consider each [option prefix + option name] pair as a candidate, finding
256
- // the closest match.
255
+ // Consider each option as a candidate, finding the closest match.
257
256
unsigned BestDistance = UINT_MAX;
258
257
for (const Info &CandidateInfo :
259
258
ArrayRef<Info>(OptionInfos).drop_front (FirstSearchableIndex)) {
260
259
StringRef CandidateName = CandidateInfo.Name ;
261
260
262
- // We can eliminate some option prefix/name pairs as candidates right away:
263
- // * Ignore option candidates with empty names, such as "--", or names
264
- // that do not meet the minimum length.
261
+ // Ignore option candidates with empty names, such as "--", or names
262
+ // that do not meet the minimum length.
265
263
if (CandidateName.empty () || CandidateName.size () < MinimumLength)
266
264
continue ;
267
265
268
- // * If FlagsToInclude were specified, ignore options that don't include
269
- // those flags.
266
+ // If FlagsToInclude were specified, ignore options that don't include
267
+ // those flags.
270
268
if (FlagsToInclude && !(CandidateInfo.Flags & FlagsToInclude))
271
269
continue ;
272
- // * Ignore options that contain the FlagsToExclude.
270
+ // Ignore options that contain the FlagsToExclude.
273
271
if (CandidateInfo.Flags & FlagsToExclude)
274
272
continue ;
275
273
276
- // * Ignore positional argument option candidates (which do not
277
- // have prefixes).
274
+ // Ignore positional argument option candidates (which do not
275
+ // have prefixes).
278
276
if (!CandidateInfo.Prefixes )
279
277
continue ;
278
+ // Find the most appropriate prefix. For example, if a user asks for
279
+ // "--helm", suggest "--help" over "-help".
280
+ StringRef Prefix = CandidateInfo.Prefixes [0 ];
281
+ for (int P = 1 ; CandidateInfo.Prefixes [P]; P++) {
282
+ if (Option.startswith (CandidateInfo.Prefixes [P]))
283
+ Prefix = CandidateInfo.Prefixes [P];
284
+ }
280
285
281
- // Now check if the candidate ends with a character commonly used when
286
+ // Check if the candidate ends with a character commonly used when
282
287
// delimiting an option from its value, such as '=' or ':'. If it does,
283
288
// attempt to split the given option based on that delimiter.
284
289
std::string Delimiter = " " ;
@@ -292,19 +297,14 @@ unsigned OptTable::findNearest(StringRef Option, std::string &NearestString,
292
297
else
293
298
std::tie (LHS, RHS) = Option.split (Last);
294
299
295
- // Consider each possible prefix for each candidate to find the most
296
- // appropriate one. For example, if a user asks for "--helm", suggest
297
- // "--help" over "-help".
298
- for (int P = 0 ; const char *const CandidatePrefix = CandidateInfo.Prefixes [P]; P++) {
299
- std::string NormalizedName = (LHS + Delimiter).str ();
300
- StringRef Candidate = (CandidatePrefix + CandidateName).str ();
301
- unsigned Distance =
302
- Candidate.edit_distance (NormalizedName, /* AllowReplacements=*/ true ,
303
- /* MaxEditDistance=*/ BestDistance);
304
- if (Distance < BestDistance) {
305
- BestDistance = Distance;
306
- NearestString = (Candidate + RHS).str ();
307
- }
300
+ std::string NormalizedName =
301
+ (LHS.drop_front (Prefix.size ()) + Delimiter).str ();
302
+ unsigned Distance =
303
+ CandidateName.edit_distance (NormalizedName, /* AllowReplacements=*/ true ,
304
+ /* MaxEditDistance=*/ BestDistance);
305
+ if (Distance < BestDistance) {
306
+ BestDistance = Distance;
307
+ NearestString = (Prefix + CandidateName + RHS).str ();
308
308
}
309
309
}
310
310
return BestDistance;
0 commit comments