| 1 | //===- OptTable.cpp - Option Table Implementation -------------------------===// | 
|---|
| 2 | // | 
|---|
| 3 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. | 
|---|
| 4 | // See https://llvm.org/LICENSE.txt for license information. | 
|---|
| 5 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception | 
|---|
| 6 | // | 
|---|
| 7 | //===----------------------------------------------------------------------===// | 
|---|
| 8 |  | 
|---|
| 9 | #include "llvm/Option/OptTable.h" | 
|---|
| 10 | #include "llvm/ADT/STLExtras.h" | 
|---|
| 11 | #include "llvm/ADT/StringRef.h" | 
|---|
| 12 | #include "llvm/Option/Arg.h" | 
|---|
| 13 | #include "llvm/Option/ArgList.h" | 
|---|
| 14 | #include "llvm/Option/OptSpecifier.h" | 
|---|
| 15 | #include "llvm/Option/Option.h" | 
|---|
| 16 | #include "llvm/Support/CommandLine.h" // for expandResponseFiles | 
|---|
| 17 | #include "llvm/Support/Compiler.h" | 
|---|
| 18 | #include "llvm/Support/ErrorHandling.h" | 
|---|
| 19 | #include "llvm/Support/OptionStrCmp.h" | 
|---|
| 20 | #include "llvm/Support/raw_ostream.h" | 
|---|
| 21 | #include <algorithm> | 
|---|
| 22 | #include <cassert> | 
|---|
| 23 | #include <cctype> | 
|---|
| 24 | #include <cstring> | 
|---|
| 25 | #include <map> | 
|---|
| 26 | #include <set> | 
|---|
| 27 | #include <string> | 
|---|
| 28 | #include <utility> | 
|---|
| 29 | #include <vector> | 
|---|
| 30 |  | 
|---|
| 31 | using namespace llvm; | 
|---|
| 32 | using namespace llvm::opt; | 
|---|
| 33 |  | 
|---|
| 34 | namespace { | 
|---|
| 35 | struct OptNameLess { | 
|---|
| 36 | const StringTable *StrTable; | 
|---|
| 37 | ArrayRef<StringTable::Offset> PrefixesTable; | 
|---|
| 38 |  | 
|---|
| 39 | explicit OptNameLess(const StringTable &StrTable, | 
|---|
| 40 | ArrayRef<StringTable::Offset> PrefixesTable) | 
|---|
| 41 | : StrTable(&StrTable), PrefixesTable(PrefixesTable) {} | 
|---|
| 42 |  | 
|---|
| 43 | #ifndef NDEBUG | 
|---|
| 44 | inline bool operator()(const OptTable::Info &A, | 
|---|
| 45 | const OptTable::Info &B) const { | 
|---|
| 46 | if (&A == &B) | 
|---|
| 47 | return false; | 
|---|
| 48 |  | 
|---|
| 49 | if (int Cmp = StrCmpOptionName(A.getName(*StrTable, PrefixesTable), | 
|---|
| 50 | B.getName(*StrTable, PrefixesTable))) | 
|---|
| 51 | return Cmp < 0; | 
|---|
| 52 |  | 
|---|
| 53 | SmallVector<StringRef, 8> APrefixes, BPrefixes; | 
|---|
| 54 | A.appendPrefixes(*StrTable, PrefixesTable, APrefixes); | 
|---|
| 55 | B.appendPrefixes(*StrTable, PrefixesTable, BPrefixes); | 
|---|
| 56 |  | 
|---|
| 57 | if (int Cmp = StrCmpOptionPrefixes(APrefixes, BPrefixes)) | 
|---|
| 58 | return Cmp < 0; | 
|---|
| 59 |  | 
|---|
| 60 | // Names are the same, check that classes are in order; exactly one | 
|---|
| 61 | // should be joined, and it should succeed the other. | 
|---|
| 62 | assert( | 
|---|
| 63 | ((A.Kind == Option::JoinedClass) ^ (B.Kind == Option::JoinedClass)) && | 
|---|
| 64 | "Unexpected classes for options with same name."); | 
|---|
| 65 | return B.Kind == Option::JoinedClass; | 
|---|
| 66 | } | 
|---|
| 67 | #endif | 
|---|
| 68 |  | 
|---|
| 69 | // Support lower_bound between info and an option name. | 
|---|
| 70 | inline bool operator()(const OptTable::Info &I, StringRef Name) const { | 
|---|
| 71 | // Do not fallback to case sensitive comparison. | 
|---|
| 72 | return StrCmpOptionName(A: I.getName(StrTable: *StrTable, PrefixesTable), B: Name, FallbackCaseSensitive: false) < | 
|---|
| 73 | 0; | 
|---|
| 74 | } | 
|---|
| 75 | }; | 
|---|
| 76 | } // namespace | 
|---|
| 77 |  | 
|---|
| 78 | OptSpecifier::OptSpecifier(const Option *Opt) : ID(Opt->getID()) {} | 
|---|
| 79 |  | 
|---|
| 80 | OptTable::OptTable(const StringTable &StrTable, | 
|---|
| 81 | ArrayRef<StringTable::Offset> PrefixesTable, | 
|---|
| 82 | ArrayRef<Info> OptionInfos, bool IgnoreCase) | 
|---|
| 83 | : StrTable(&StrTable), PrefixesTable(PrefixesTable), | 
|---|
| 84 | OptionInfos(OptionInfos), IgnoreCase(IgnoreCase) { | 
|---|
| 85 | // Explicitly zero initialize the error to work around a bug in array | 
|---|
| 86 | // value-initialization on MinGW with gcc 4.3.5. | 
|---|
| 87 |  | 
|---|
| 88 | // Find start of normal options. | 
|---|
| 89 | for (unsigned i = 0, e = getNumOptions(); i != e; ++i) { | 
|---|
| 90 | unsigned Kind = getInfo(Opt: i + 1).Kind; | 
|---|
| 91 | if (Kind == Option::InputClass) { | 
|---|
| 92 | assert(!InputOptionID && "Cannot have multiple input options!"); | 
|---|
| 93 | InputOptionID = getInfo(Opt: i + 1).ID; | 
|---|
| 94 | } else if (Kind == Option::UnknownClass) { | 
|---|
| 95 | assert(!UnknownOptionID && "Cannot have multiple unknown options!"); | 
|---|
| 96 | UnknownOptionID = getInfo(Opt: i + 1).ID; | 
|---|
| 97 | } else if (Kind != Option::GroupClass) { | 
|---|
| 98 | FirstSearchableIndex = i; | 
|---|
| 99 | break; | 
|---|
| 100 | } | 
|---|
| 101 | } | 
|---|
| 102 | assert(FirstSearchableIndex != 0 && "No searchable options?"); | 
|---|
| 103 |  | 
|---|
| 104 | #ifndef NDEBUG | 
|---|
| 105 | // Check that everything after the first searchable option is a | 
|---|
| 106 | // regular option class. | 
|---|
| 107 | for (unsigned i = FirstSearchableIndex, e = getNumOptions(); i != e; ++i) { | 
|---|
| 108 | Option::OptionClass Kind = (Option::OptionClass) getInfo(i + 1).Kind; | 
|---|
| 109 | assert((Kind != Option::InputClass && Kind != Option::UnknownClass && | 
|---|
| 110 | Kind != Option::GroupClass) && | 
|---|
| 111 | "Special options should be defined first!"); | 
|---|
| 112 | } | 
|---|
| 113 |  | 
|---|
| 114 | // Check that options are in order. | 
|---|
| 115 | for (unsigned i = FirstSearchableIndex + 1, e = getNumOptions(); i != e; ++i){ | 
|---|
| 116 | if (!(OptNameLess(StrTable, PrefixesTable)(getInfo(i), getInfo(i + 1)))) { | 
|---|
| 117 | getOption(i).dump(); | 
|---|
| 118 | getOption(i + 1).dump(); | 
|---|
| 119 | llvm_unreachable( "Options are not in order!"); | 
|---|
| 120 | } | 
|---|
| 121 | } | 
|---|
| 122 | #endif | 
|---|
| 123 | } | 
|---|
| 124 |  | 
|---|
| 125 | void OptTable::buildPrefixChars() { | 
|---|
| 126 | assert(PrefixChars.empty() && "rebuilding a non-empty prefix char"); | 
|---|
| 127 |  | 
|---|
| 128 | // Build prefix chars. | 
|---|
| 129 | for (StringRef Prefix : PrefixesUnion) { | 
|---|
| 130 | for (char C : Prefix) | 
|---|
| 131 | if (!is_contained(Range&: PrefixChars, Element: C)) | 
|---|
| 132 | PrefixChars.push_back(Elt: C); | 
|---|
| 133 | } | 
|---|
| 134 | } | 
|---|
| 135 |  | 
|---|
| 136 | OptTable::~OptTable() = default; | 
|---|
| 137 |  | 
|---|
| 138 | const Option OptTable::getOption(OptSpecifier Opt) const { | 
|---|
| 139 | unsigned id = Opt.getID(); | 
|---|
| 140 | if (id == 0) | 
|---|
| 141 | return Option(nullptr, nullptr); | 
|---|
| 142 | assert((unsigned) (id - 1) < getNumOptions() && "Invalid ID."); | 
|---|
| 143 | return Option(&getInfo(Opt: id), this); | 
|---|
| 144 | } | 
|---|
| 145 |  | 
|---|
| 146 | static bool isInput(const ArrayRef<StringRef> &Prefixes, StringRef Arg) { | 
|---|
| 147 | if (Arg == "-") | 
|---|
| 148 | return true; | 
|---|
| 149 | for (const StringRef &Prefix : Prefixes) | 
|---|
| 150 | if (Arg.starts_with(Prefix)) | 
|---|
| 151 | return false; | 
|---|
| 152 | return true; | 
|---|
| 153 | } | 
|---|
| 154 |  | 
|---|
| 155 | /// \returns Matched size. 0 means no match. | 
|---|
| 156 | static unsigned matchOption(const StringTable &StrTable, | 
|---|
| 157 | ArrayRef<StringTable::Offset> PrefixesTable, | 
|---|
| 158 | const OptTable::Info *I, StringRef Str, | 
|---|
| 159 | bool IgnoreCase) { | 
|---|
| 160 | StringRef Name = I->getName(StrTable, PrefixesTable); | 
|---|
| 161 | for (auto PrefixOffset : I->getPrefixOffsets(PrefixesTable)) { | 
|---|
| 162 | StringRef Prefix = StrTable[PrefixOffset]; | 
|---|
| 163 | if (Str.starts_with(Prefix)) { | 
|---|
| 164 | StringRef Rest = Str.substr(Start: Prefix.size()); | 
|---|
| 165 | bool Matched = IgnoreCase ? Rest.starts_with_insensitive(Prefix: Name) | 
|---|
| 166 | : Rest.starts_with(Prefix: Name); | 
|---|
| 167 | if (Matched) | 
|---|
| 168 | return Prefix.size() + Name.size(); | 
|---|
| 169 | } | 
|---|
| 170 | } | 
|---|
| 171 | return 0; | 
|---|
| 172 | } | 
|---|
| 173 |  | 
|---|
| 174 | // Returns true if one of the Prefixes + In.Names matches Option | 
|---|
| 175 | static bool optionMatches(const StringTable &StrTable, | 
|---|
| 176 | ArrayRef<StringTable::Offset> PrefixesTable, | 
|---|
| 177 | const OptTable::Info &In, StringRef Option) { | 
|---|
| 178 | StringRef Name = In.getName(StrTable, PrefixesTable); | 
|---|
| 179 | if (Option.consume_back(Suffix: Name)) | 
|---|
| 180 | for (auto PrefixOffset : In.getPrefixOffsets(PrefixesTable)) | 
|---|
| 181 | if (Option == StrTable[PrefixOffset]) | 
|---|
| 182 | return true; | 
|---|
| 183 | return false; | 
|---|
| 184 | } | 
|---|
| 185 |  | 
|---|
| 186 | // This function is for flag value completion. | 
|---|
| 187 | // Eg. When "-stdlib=" and "l" was passed to this function, it will return | 
|---|
| 188 | // appropiriate values for stdlib, which starts with l. | 
|---|
| 189 | std::vector<std::string> | 
|---|
| 190 | OptTable::suggestValueCompletions(StringRef Option, StringRef Arg) const { | 
|---|
| 191 | // Search all options and return possible values. | 
|---|
| 192 | for (size_t I = FirstSearchableIndex, E = OptionInfos.size(); I < E; I++) { | 
|---|
| 193 | const Info &In = OptionInfos[I]; | 
|---|
| 194 | if (!In.Values || !optionMatches(StrTable: *StrTable, PrefixesTable, In, Option)) | 
|---|
| 195 | continue; | 
|---|
| 196 |  | 
|---|
| 197 | SmallVector<StringRef, 8> Candidates; | 
|---|
| 198 | StringRef(In.Values).split(A&: Candidates, Separator: ",", MaxSplit: -1, KeepEmpty: false); | 
|---|
| 199 |  | 
|---|
| 200 | std::vector<std::string> Result; | 
|---|
| 201 | for (StringRef Val : Candidates) | 
|---|
| 202 | if (Val.starts_with(Prefix: Arg) && Arg != Val) | 
|---|
| 203 | Result.push_back(x: std::string(Val)); | 
|---|
| 204 | return Result; | 
|---|
| 205 | } | 
|---|
| 206 | return {}; | 
|---|
| 207 | } | 
|---|
| 208 |  | 
|---|
| 209 | std::vector<std::string> | 
|---|
| 210 | OptTable::findByPrefix(StringRef Cur, Visibility VisibilityMask, | 
|---|
| 211 | unsigned int DisableFlags) const { | 
|---|
| 212 | std::vector<std::string> Ret; | 
|---|
| 213 | for (size_t I = FirstSearchableIndex, E = OptionInfos.size(); I < E; I++) { | 
|---|
| 214 | const Info &In = OptionInfos[I]; | 
|---|
| 215 | if (In.hasNoPrefix() || (!In.HelpText && !In.GroupID)) | 
|---|
| 216 | continue; | 
|---|
| 217 | if (!(In.Visibility & VisibilityMask)) | 
|---|
| 218 | continue; | 
|---|
| 219 | if (In.Flags & DisableFlags) | 
|---|
| 220 | continue; | 
|---|
| 221 |  | 
|---|
| 222 | StringRef Name = In.getName(StrTable: *StrTable, PrefixesTable); | 
|---|
| 223 | for (auto PrefixOffset : In.getPrefixOffsets(PrefixesTable)) { | 
|---|
| 224 | StringRef Prefix = (*StrTable)[PrefixOffset]; | 
|---|
| 225 | std::string S = (Twine(Prefix) + Name + "\t").str(); | 
|---|
| 226 | if (In.HelpText) | 
|---|
| 227 | S += In.HelpText; | 
|---|
| 228 | if (StringRef(S).starts_with(Prefix: Cur) && S != std::string(Cur) + "\t") | 
|---|
| 229 | Ret.push_back(x: S); | 
|---|
| 230 | } | 
|---|
| 231 | } | 
|---|
| 232 | return Ret; | 
|---|
| 233 | } | 
|---|
| 234 |  | 
|---|
| 235 | unsigned OptTable::findNearest(StringRef Option, std::string &NearestString, | 
|---|
| 236 | Visibility VisibilityMask, | 
|---|
| 237 | unsigned MinimumLength, | 
|---|
| 238 | unsigned MaximumDistance) const { | 
|---|
| 239 | return internalFindNearest( | 
|---|
| 240 | Option, NearestString, MinimumLength, MaximumDistance, | 
|---|
| 241 | ExcludeOption: [VisibilityMask](const Info &CandidateInfo) { | 
|---|
| 242 | return (CandidateInfo.Visibility & VisibilityMask) == 0; | 
|---|
| 243 | }); | 
|---|
| 244 | } | 
|---|
| 245 |  | 
|---|
| 246 | unsigned OptTable::findNearest(StringRef Option, std::string &NearestString, | 
|---|
| 247 | unsigned FlagsToInclude, unsigned FlagsToExclude, | 
|---|
| 248 | unsigned MinimumLength, | 
|---|
| 249 | unsigned MaximumDistance) const { | 
|---|
| 250 | return internalFindNearest( | 
|---|
| 251 | Option, NearestString, MinimumLength, MaximumDistance, | 
|---|
| 252 | ExcludeOption: [FlagsToInclude, FlagsToExclude](const Info &CandidateInfo) { | 
|---|
| 253 | if (FlagsToInclude && !(CandidateInfo.Flags & FlagsToInclude)) | 
|---|
| 254 | return true; | 
|---|
| 255 | if (CandidateInfo.Flags & FlagsToExclude) | 
|---|
| 256 | return true; | 
|---|
| 257 | return false; | 
|---|
| 258 | }); | 
|---|
| 259 | } | 
|---|
| 260 |  | 
|---|
| 261 | unsigned OptTable::internalFindNearest( | 
|---|
| 262 | StringRef Option, std::string &NearestString, unsigned MinimumLength, | 
|---|
| 263 | unsigned MaximumDistance, | 
|---|
| 264 | std::function<bool(const Info &)> ExcludeOption) const { | 
|---|
| 265 | assert(!Option.empty()); | 
|---|
| 266 |  | 
|---|
| 267 | // Consider each [option prefix + option name] pair as a candidate, finding | 
|---|
| 268 | // the closest match. | 
|---|
| 269 | unsigned BestDistance = | 
|---|
| 270 | MaximumDistance == UINT_MAX ? UINT_MAX : MaximumDistance + 1; | 
|---|
| 271 | SmallString<16> Candidate; | 
|---|
| 272 | SmallString<16> NormalizedName; | 
|---|
| 273 |  | 
|---|
| 274 | for (const Info &CandidateInfo : | 
|---|
| 275 | ArrayRef<Info>(OptionInfos).drop_front(N: FirstSearchableIndex)) { | 
|---|
| 276 | StringRef CandidateName = CandidateInfo.getName(StrTable: *StrTable, PrefixesTable); | 
|---|
| 277 |  | 
|---|
| 278 | // We can eliminate some option prefix/name pairs as candidates right away: | 
|---|
| 279 | // * Ignore option candidates with empty names, such as "--", or names | 
|---|
| 280 | //   that do not meet the minimum length. | 
|---|
| 281 | if (CandidateName.size() < MinimumLength) | 
|---|
| 282 | continue; | 
|---|
| 283 |  | 
|---|
| 284 | // Ignore options that are excluded via masks | 
|---|
| 285 | if (ExcludeOption(CandidateInfo)) | 
|---|
| 286 | continue; | 
|---|
| 287 |  | 
|---|
| 288 | // * Ignore positional argument option candidates (which do not | 
|---|
| 289 | //   have prefixes). | 
|---|
| 290 | if (CandidateInfo.hasNoPrefix()) | 
|---|
| 291 | continue; | 
|---|
| 292 |  | 
|---|
| 293 | // Now check if the candidate ends with a character commonly used when | 
|---|
| 294 | // delimiting an option from its value, such as '=' or ':'. If it does, | 
|---|
| 295 | // attempt to split the given option based on that delimiter. | 
|---|
| 296 | char Last = CandidateName.back(); | 
|---|
| 297 | bool CandidateHasDelimiter = Last == '=' || Last == ':'; | 
|---|
| 298 | StringRef RHS; | 
|---|
| 299 | if (CandidateHasDelimiter) { | 
|---|
| 300 | std::tie(args&: NormalizedName, args&: RHS) = Option.split(Separator: Last); | 
|---|
| 301 | if (Option.find(C: Last) == NormalizedName.size()) | 
|---|
| 302 | NormalizedName += Last; | 
|---|
| 303 | } else | 
|---|
| 304 | NormalizedName = Option; | 
|---|
| 305 |  | 
|---|
| 306 | // Consider each possible prefix for each candidate to find the most | 
|---|
| 307 | // appropriate one. For example, if a user asks for "--helm", suggest | 
|---|
| 308 | // "--help" over "-help". | 
|---|
| 309 | for (auto CandidatePrefixOffset : | 
|---|
| 310 | CandidateInfo.getPrefixOffsets(PrefixesTable)) { | 
|---|
| 311 | StringRef CandidatePrefix = (*StrTable)[CandidatePrefixOffset]; | 
|---|
| 312 | // If Candidate and NormalizedName have more than 'BestDistance' | 
|---|
| 313 | // characters of difference, no need to compute the edit distance, it's | 
|---|
| 314 | // going to be greater than BestDistance. Don't bother computing Candidate | 
|---|
| 315 | // at all. | 
|---|
| 316 | size_t CandidateSize = CandidatePrefix.size() + CandidateName.size(), | 
|---|
| 317 | NormalizedSize = NormalizedName.size(); | 
|---|
| 318 | size_t AbsDiff = CandidateSize > NormalizedSize | 
|---|
| 319 | ? CandidateSize - NormalizedSize | 
|---|
| 320 | : NormalizedSize - CandidateSize; | 
|---|
| 321 | if (AbsDiff > BestDistance) { | 
|---|
| 322 | continue; | 
|---|
| 323 | } | 
|---|
| 324 | Candidate = CandidatePrefix; | 
|---|
| 325 | Candidate += CandidateName; | 
|---|
| 326 | unsigned Distance = StringRef(Candidate).edit_distance( | 
|---|
| 327 | Other: NormalizedName, /*AllowReplacements=*/true, | 
|---|
| 328 | /*MaxEditDistance=*/BestDistance); | 
|---|
| 329 | if (RHS.empty() && CandidateHasDelimiter) { | 
|---|
| 330 | // The Candidate ends with a = or : delimiter, but the option passed in | 
|---|
| 331 | // didn't contain the delimiter (or doesn't have anything after it). | 
|---|
| 332 | // In that case, penalize the correction: `-nodefaultlibs` is more | 
|---|
| 333 | // likely to be a spello for `-nodefaultlib` than `-nodefaultlib:` even | 
|---|
| 334 | // though both have an unmodified editing distance of 1, since the | 
|---|
| 335 | // latter would need an argument. | 
|---|
| 336 | ++Distance; | 
|---|
| 337 | } | 
|---|
| 338 | if (Distance < BestDistance) { | 
|---|
| 339 | BestDistance = Distance; | 
|---|
| 340 | NearestString = (Candidate + RHS).str(); | 
|---|
| 341 | } | 
|---|
| 342 | } | 
|---|
| 343 | } | 
|---|
| 344 | return BestDistance; | 
|---|
| 345 | } | 
|---|
| 346 |  | 
|---|
| 347 | // Parse a single argument, return the new argument, and update Index. If | 
|---|
| 348 | // GroupedShortOptions is true, -a matches "-abc" and the argument in Args will | 
|---|
| 349 | // be updated to "-bc". This overload does not support VisibilityMask or case | 
|---|
| 350 | // insensitive options. | 
|---|
| 351 | std::unique_ptr<Arg> OptTable::parseOneArgGrouped(InputArgList &Args, | 
|---|
| 352 | unsigned &Index) const { | 
|---|
| 353 | // Anything that doesn't start with PrefixesUnion is an input, as is '-' | 
|---|
| 354 | // itself. | 
|---|
| 355 | const char *CStr = Args.getArgString(Index); | 
|---|
| 356 | StringRef Str(CStr); | 
|---|
| 357 | if (isInput(Prefixes: PrefixesUnion, Arg: Str)) | 
|---|
| 358 | return std::make_unique<Arg>(args: getOption(Opt: InputOptionID), args&: Str, args: Index++, args&: CStr); | 
|---|
| 359 |  | 
|---|
| 360 | const Info *End = OptionInfos.data() + OptionInfos.size(); | 
|---|
| 361 | StringRef Name = Str.ltrim(Chars: PrefixChars); | 
|---|
| 362 | const Info *Start = | 
|---|
| 363 | std::lower_bound(first: OptionInfos.data() + FirstSearchableIndex, last: End, val: Name, | 
|---|
| 364 | comp: OptNameLess(*StrTable, PrefixesTable)); | 
|---|
| 365 | const Info *Fallback = nullptr; | 
|---|
| 366 | unsigned Prev = Index; | 
|---|
| 367 |  | 
|---|
| 368 | // Search for the option which matches Str. | 
|---|
| 369 | for (; Start != End; ++Start) { | 
|---|
| 370 | unsigned ArgSize = | 
|---|
| 371 | matchOption(StrTable: *StrTable, PrefixesTable, I: Start, Str, IgnoreCase); | 
|---|
| 372 | if (!ArgSize) | 
|---|
| 373 | continue; | 
|---|
| 374 |  | 
|---|
| 375 | Option Opt(Start, this); | 
|---|
| 376 | if (std::unique_ptr<Arg> A = | 
|---|
| 377 | Opt.accept(Args, CurArg: StringRef(Args.getArgString(Index), ArgSize), | 
|---|
| 378 | /*GroupedShortOption=*/false, Index)) | 
|---|
| 379 | return A; | 
|---|
| 380 |  | 
|---|
| 381 | // If Opt is a Flag of length 2 (e.g. "-a"), we know it is a prefix of | 
|---|
| 382 | // the current argument (e.g. "-abc"). Match it as a fallback if no longer | 
|---|
| 383 | // option (e.g. "-ab") exists. | 
|---|
| 384 | if (ArgSize == 2 && Opt.getKind() == Option::FlagClass) | 
|---|
| 385 | Fallback = Start; | 
|---|
| 386 |  | 
|---|
| 387 | // Otherwise, see if the argument is missing. | 
|---|
| 388 | if (Prev != Index) | 
|---|
| 389 | return nullptr; | 
|---|
| 390 | } | 
|---|
| 391 | if (Fallback) { | 
|---|
| 392 | Option Opt(Fallback, this); | 
|---|
| 393 | // Check that the last option isn't a flag wrongly given an argument. | 
|---|
| 394 | if (Str[2] == '=') | 
|---|
| 395 | return std::make_unique<Arg>(args: getOption(Opt: UnknownOptionID), args&: Str, args: Index++, | 
|---|
| 396 | args&: CStr); | 
|---|
| 397 |  | 
|---|
| 398 | if (std::unique_ptr<Arg> A = Opt.accept( | 
|---|
| 399 | Args, CurArg: Str.substr(Start: 0, N: 2), /*GroupedShortOption=*/true, Index)) { | 
|---|
| 400 | Args.replaceArgString(Index, S: Twine('-') + Str.substr(Start: 2)); | 
|---|
| 401 | return A; | 
|---|
| 402 | } | 
|---|
| 403 | } | 
|---|
| 404 |  | 
|---|
| 405 | // In the case of an incorrect short option extract the character and move to | 
|---|
| 406 | // the next one. | 
|---|
| 407 | if (Str[1] != '-') { | 
|---|
| 408 | CStr = Args.MakeArgString(Str: Str.substr(Start: 0, N: 2)); | 
|---|
| 409 | Args.replaceArgString(Index, S: Twine('-') + Str.substr(Start: 2)); | 
|---|
| 410 | return std::make_unique<Arg>(args: getOption(Opt: UnknownOptionID), args&: CStr, args&: Index, args&: CStr); | 
|---|
| 411 | } | 
|---|
| 412 |  | 
|---|
| 413 | return std::make_unique<Arg>(args: getOption(Opt: UnknownOptionID), args&: Str, args: Index++, args&: CStr); | 
|---|
| 414 | } | 
|---|
| 415 |  | 
|---|
| 416 | std::unique_ptr<Arg> OptTable::ParseOneArg(const ArgList &Args, unsigned &Index, | 
|---|
| 417 | Visibility VisibilityMask) const { | 
|---|
| 418 | return internalParseOneArg(Args, Index, ExcludeOption: [VisibilityMask](const Option &Opt) { | 
|---|
| 419 | return !Opt.hasVisibilityFlag(Val: VisibilityMask); | 
|---|
| 420 | }); | 
|---|
| 421 | } | 
|---|
| 422 |  | 
|---|
| 423 | std::unique_ptr<Arg> OptTable::ParseOneArg(const ArgList &Args, unsigned &Index, | 
|---|
| 424 | unsigned FlagsToInclude, | 
|---|
| 425 | unsigned FlagsToExclude) const { | 
|---|
| 426 | return internalParseOneArg( | 
|---|
| 427 | Args, Index, ExcludeOption: [FlagsToInclude, FlagsToExclude](const Option &Opt) { | 
|---|
| 428 | if (FlagsToInclude && !Opt.hasFlag(Val: FlagsToInclude)) | 
|---|
| 429 | return true; | 
|---|
| 430 | if (Opt.hasFlag(Val: FlagsToExclude)) | 
|---|
| 431 | return true; | 
|---|
| 432 | return false; | 
|---|
| 433 | }); | 
|---|
| 434 | } | 
|---|
| 435 |  | 
|---|
| 436 | std::unique_ptr<Arg> OptTable::internalParseOneArg( | 
|---|
| 437 | const ArgList &Args, unsigned &Index, | 
|---|
| 438 | std::function<bool(const Option &)> ExcludeOption) const { | 
|---|
| 439 | unsigned Prev = Index; | 
|---|
| 440 | StringRef Str = Args.getArgString(Index); | 
|---|
| 441 |  | 
|---|
| 442 | // Anything that doesn't start with PrefixesUnion is an input, as is '-' | 
|---|
| 443 | // itself. | 
|---|
| 444 | if (isInput(Prefixes: PrefixesUnion, Arg: Str)) | 
|---|
| 445 | return std::make_unique<Arg>(args: getOption(Opt: InputOptionID), args&: Str, args: Index++, | 
|---|
| 446 | args: Str.data()); | 
|---|
| 447 |  | 
|---|
| 448 | const Info *Start = OptionInfos.data() + FirstSearchableIndex; | 
|---|
| 449 | const Info *End = OptionInfos.data() + OptionInfos.size(); | 
|---|
| 450 | StringRef Name = Str.ltrim(Chars: PrefixChars); | 
|---|
| 451 |  | 
|---|
| 452 | // Search for the first next option which could be a prefix. | 
|---|
| 453 | Start = | 
|---|
| 454 | std::lower_bound(first: Start, last: End, val: Name, comp: OptNameLess(*StrTable, PrefixesTable)); | 
|---|
| 455 |  | 
|---|
| 456 | // Options are stored in sorted order, with '\0' at the end of the | 
|---|
| 457 | // alphabet. Since the only options which can accept a string must | 
|---|
| 458 | // prefix it, we iteratively search for the next option which could | 
|---|
| 459 | // be a prefix. | 
|---|
| 460 | // | 
|---|
| 461 | // FIXME: This is searching much more than necessary, but I am | 
|---|
| 462 | // blanking on the simplest way to make it fast. We can solve this | 
|---|
| 463 | // problem when we move to TableGen. | 
|---|
| 464 | for (; Start != End; ++Start) { | 
|---|
| 465 | unsigned ArgSize = 0; | 
|---|
| 466 | // Scan for first option which is a proper prefix. | 
|---|
| 467 | for (; Start != End; ++Start) | 
|---|
| 468 | if ((ArgSize = | 
|---|
| 469 | matchOption(StrTable: *StrTable, PrefixesTable, I: Start, Str, IgnoreCase))) | 
|---|
| 470 | break; | 
|---|
| 471 | if (Start == End) | 
|---|
| 472 | break; | 
|---|
| 473 |  | 
|---|
| 474 | Option Opt(Start, this); | 
|---|
| 475 |  | 
|---|
| 476 | if (ExcludeOption(Opt)) | 
|---|
| 477 | continue; | 
|---|
| 478 |  | 
|---|
| 479 | // See if this option matches. | 
|---|
| 480 | if (std::unique_ptr<Arg> A = | 
|---|
| 481 | Opt.accept(Args, CurArg: StringRef(Args.getArgString(Index), ArgSize), | 
|---|
| 482 | /*GroupedShortOption=*/false, Index)) | 
|---|
| 483 | return A; | 
|---|
| 484 |  | 
|---|
| 485 | // Otherwise, see if this argument was missing values. | 
|---|
| 486 | if (Prev != Index) | 
|---|
| 487 | return nullptr; | 
|---|
| 488 | } | 
|---|
| 489 |  | 
|---|
| 490 | // If we failed to find an option and this arg started with /, then it's | 
|---|
| 491 | // probably an input path. | 
|---|
| 492 | if (Str[0] == '/') | 
|---|
| 493 | return std::make_unique<Arg>(args: getOption(Opt: InputOptionID), args&: Str, args: Index++, | 
|---|
| 494 | args: Str.data()); | 
|---|
| 495 |  | 
|---|
| 496 | return std::make_unique<Arg>(args: getOption(Opt: UnknownOptionID), args&: Str, args: Index++, | 
|---|
| 497 | args: Str.data()); | 
|---|
| 498 | } | 
|---|
| 499 |  | 
|---|
| 500 | InputArgList OptTable::ParseArgs(ArrayRef<const char *> Args, | 
|---|
| 501 | unsigned &MissingArgIndex, | 
|---|
| 502 | unsigned &MissingArgCount, | 
|---|
| 503 | Visibility VisibilityMask) const { | 
|---|
| 504 | return internalParseArgs( | 
|---|
| 505 | Args, MissingArgIndex, MissingArgCount, | 
|---|
| 506 | ExcludeOption: [VisibilityMask](const Option &Opt) { | 
|---|
| 507 | return !Opt.hasVisibilityFlag(Val: VisibilityMask); | 
|---|
| 508 | }); | 
|---|
| 509 | } | 
|---|
| 510 |  | 
|---|
| 511 | InputArgList OptTable::ParseArgs(ArrayRef<const char *> Args, | 
|---|
| 512 | unsigned &MissingArgIndex, | 
|---|
| 513 | unsigned &MissingArgCount, | 
|---|
| 514 | unsigned FlagsToInclude, | 
|---|
| 515 | unsigned FlagsToExclude) const { | 
|---|
| 516 | return internalParseArgs( | 
|---|
| 517 | Args, MissingArgIndex, MissingArgCount, | 
|---|
| 518 | ExcludeOption: [FlagsToInclude, FlagsToExclude](const Option &Opt) { | 
|---|
| 519 | if (FlagsToInclude && !Opt.hasFlag(Val: FlagsToInclude)) | 
|---|
| 520 | return true; | 
|---|
| 521 | if (Opt.hasFlag(Val: FlagsToExclude)) | 
|---|
| 522 | return true; | 
|---|
| 523 | return false; | 
|---|
| 524 | }); | 
|---|
| 525 | } | 
|---|
| 526 |  | 
|---|
| 527 | InputArgList OptTable::internalParseArgs( | 
|---|
| 528 | ArrayRef<const char *> ArgArr, unsigned &MissingArgIndex, | 
|---|
| 529 | unsigned &MissingArgCount, | 
|---|
| 530 | std::function<bool(const Option &)> ExcludeOption) const { | 
|---|
| 531 | InputArgList Args(ArgArr.begin(), ArgArr.end()); | 
|---|
| 532 |  | 
|---|
| 533 | // FIXME: Handle '@' args (or at least error on them). | 
|---|
| 534 |  | 
|---|
| 535 | MissingArgIndex = MissingArgCount = 0; | 
|---|
| 536 | unsigned Index = 0, End = ArgArr.size(); | 
|---|
| 537 | while (Index < End) { | 
|---|
| 538 | // Ingore nullptrs, they are response file's EOL markers | 
|---|
| 539 | if (Args.getArgString(Index) == nullptr) { | 
|---|
| 540 | ++Index; | 
|---|
| 541 | continue; | 
|---|
| 542 | } | 
|---|
| 543 | // Ignore empty arguments (other things may still take them as arguments). | 
|---|
| 544 | StringRef Str = Args.getArgString(Index); | 
|---|
| 545 | if (Str == "") { | 
|---|
| 546 | ++Index; | 
|---|
| 547 | continue; | 
|---|
| 548 | } | 
|---|
| 549 |  | 
|---|
| 550 | // In DashDashParsing mode, the first "--" stops option scanning and treats | 
|---|
| 551 | // all subsequent arguments as positional. | 
|---|
| 552 | if (DashDashParsing && Str == "--") { | 
|---|
| 553 | while (++Index < End) { | 
|---|
| 554 | Args.append(A: new Arg(getOption(Opt: InputOptionID), Str, Index, | 
|---|
| 555 | Args.getArgString(Index))); | 
|---|
| 556 | } | 
|---|
| 557 | break; | 
|---|
| 558 | } | 
|---|
| 559 |  | 
|---|
| 560 | unsigned Prev = Index; | 
|---|
| 561 | std::unique_ptr<Arg> A = GroupedShortOptions | 
|---|
| 562 | ? parseOneArgGrouped(Args, Index) | 
|---|
| 563 | : internalParseOneArg(Args, Index, ExcludeOption); | 
|---|
| 564 | assert((Index > Prev || GroupedShortOptions) && | 
|---|
| 565 | "Parser failed to consume argument."); | 
|---|
| 566 |  | 
|---|
| 567 | // Check for missing argument error. | 
|---|
| 568 | if (!A) { | 
|---|
| 569 | assert(Index >= End && "Unexpected parser error."); | 
|---|
| 570 | assert(Index - Prev - 1 && "No missing arguments!"); | 
|---|
| 571 | MissingArgIndex = Prev; | 
|---|
| 572 | MissingArgCount = Index - Prev - 1; | 
|---|
| 573 | break; | 
|---|
| 574 | } | 
|---|
| 575 |  | 
|---|
| 576 | Args.append(A: A.release()); | 
|---|
| 577 | } | 
|---|
| 578 |  | 
|---|
| 579 | return Args; | 
|---|
| 580 | } | 
|---|
| 581 |  | 
|---|
| 582 | InputArgList OptTable::parseArgs(int Argc, char *const *Argv, | 
|---|
| 583 | OptSpecifier Unknown, StringSaver &Saver, | 
|---|
| 584 | std::function<void(StringRef)> ErrorFn) const { | 
|---|
| 585 | SmallVector<const char *, 0> NewArgv; | 
|---|
| 586 | // The environment variable specifies initial options which can be overridden | 
|---|
| 587 | // by commnad line options. | 
|---|
| 588 | cl::expandResponseFiles(Argc, Argv, EnvVar, Saver, NewArgv); | 
|---|
| 589 |  | 
|---|
| 590 | unsigned MAI, MAC; | 
|---|
| 591 | opt::InputArgList Args = ParseArgs(Args: ArrayRef(NewArgv), MissingArgIndex&: MAI, MissingArgCount&: MAC); | 
|---|
| 592 | if (MAC) | 
|---|
| 593 | ErrorFn((Twine(Args.getArgString(Index: MAI)) + ": missing argument").str()); | 
|---|
| 594 |  | 
|---|
| 595 | // For each unknwon option, call ErrorFn with a formatted error message. The | 
|---|
| 596 | // message includes a suggested alternative option spelling if available. | 
|---|
| 597 | std::string Nearest; | 
|---|
| 598 | for (const opt::Arg *A : Args.filtered(Ids: Unknown)) { | 
|---|
| 599 | std::string Spelling = A->getAsString(Args); | 
|---|
| 600 | if (findNearest(Option: Spelling, NearestString&: Nearest) > 1) | 
|---|
| 601 | ErrorFn( "unknown argument '"+ Spelling + "'"); | 
|---|
| 602 | else | 
|---|
| 603 | ErrorFn( "unknown argument '"+ Spelling + "', did you mean '"+ Nearest + | 
|---|
| 604 | "'?"); | 
|---|
| 605 | } | 
|---|
| 606 | return Args; | 
|---|
| 607 | } | 
|---|
| 608 |  | 
|---|
| 609 | static std::string getOptionHelpName(const OptTable &Opts, OptSpecifier Id) { | 
|---|
| 610 | const Option O = Opts.getOption(Opt: Id); | 
|---|
| 611 | std::string Name = O.getPrefixedName().str(); | 
|---|
| 612 |  | 
|---|
| 613 | // Add metavar, if used. | 
|---|
| 614 | switch (O.getKind()) { | 
|---|
| 615 | case Option::GroupClass: case Option::InputClass: case Option::UnknownClass: | 
|---|
| 616 | llvm_unreachable( "Invalid option with help text."); | 
|---|
| 617 |  | 
|---|
| 618 | case Option::MultiArgClass: | 
|---|
| 619 | if (const char *MetaVarName = Opts.getOptionMetaVar(id: Id)) { | 
|---|
| 620 | // For MultiArgs, metavar is full list of all argument names. | 
|---|
| 621 | Name += ' '; | 
|---|
| 622 | Name += MetaVarName; | 
|---|
| 623 | } | 
|---|
| 624 | else { | 
|---|
| 625 | // For MultiArgs<N>, if metavar not supplied, print <value> N times. | 
|---|
| 626 | for (unsigned i=0, e=O.getNumArgs(); i< e; ++i) { | 
|---|
| 627 | Name += " <value>"; | 
|---|
| 628 | } | 
|---|
| 629 | } | 
|---|
| 630 | break; | 
|---|
| 631 |  | 
|---|
| 632 | case Option::FlagClass: | 
|---|
| 633 | break; | 
|---|
| 634 |  | 
|---|
| 635 | case Option::ValuesClass: | 
|---|
| 636 | break; | 
|---|
| 637 |  | 
|---|
| 638 | case Option::SeparateClass: case Option::JoinedOrSeparateClass: | 
|---|
| 639 | case Option::RemainingArgsClass: case Option::RemainingArgsJoinedClass: | 
|---|
| 640 | Name += ' '; | 
|---|
| 641 | [[fallthrough]]; | 
|---|
| 642 | case Option::JoinedClass: case Option::CommaJoinedClass: | 
|---|
| 643 | case Option::JoinedAndSeparateClass: | 
|---|
| 644 | if (const char *MetaVarName = Opts.getOptionMetaVar(id: Id)) | 
|---|
| 645 | Name += MetaVarName; | 
|---|
| 646 | else | 
|---|
| 647 | Name += "<value>"; | 
|---|
| 648 | break; | 
|---|
| 649 | } | 
|---|
| 650 |  | 
|---|
| 651 | return Name; | 
|---|
| 652 | } | 
|---|
| 653 |  | 
|---|
| 654 | namespace { | 
|---|
| 655 | struct OptionInfo { | 
|---|
| 656 | std::string Name; | 
|---|
| 657 | StringRef HelpText; | 
|---|
| 658 | }; | 
|---|
| 659 | } // namespace | 
|---|
| 660 |  | 
|---|
| 661 | static void PrintHelpOptionList(raw_ostream &OS, StringRef Title, | 
|---|
| 662 | std::vector<OptionInfo> &OptionHelp) { | 
|---|
| 663 | OS << Title << ":\n"; | 
|---|
| 664 |  | 
|---|
| 665 | // Find the maximum option length. | 
|---|
| 666 | unsigned OptionFieldWidth = 0; | 
|---|
| 667 | for (const OptionInfo &Opt : OptionHelp) { | 
|---|
| 668 | // Limit the amount of padding we are willing to give up for alignment. | 
|---|
| 669 | unsigned Length = Opt.Name.size(); | 
|---|
| 670 | if (Length <= 23) | 
|---|
| 671 | OptionFieldWidth = std::max(a: OptionFieldWidth, b: Length); | 
|---|
| 672 | } | 
|---|
| 673 |  | 
|---|
| 674 | const unsigned InitialPad = 2; | 
|---|
| 675 | for (const OptionInfo &Opt : OptionHelp) { | 
|---|
| 676 | const std::string &Option = Opt.Name; | 
|---|
| 677 | int Pad = OptionFieldWidth + InitialPad; | 
|---|
| 678 | int FirstLinePad = OptionFieldWidth - int(Option.size()); | 
|---|
| 679 | OS.indent(NumSpaces: InitialPad) << Option; | 
|---|
| 680 |  | 
|---|
| 681 | // Break on long option names. | 
|---|
| 682 | if (FirstLinePad < 0) { | 
|---|
| 683 | OS << "\n"; | 
|---|
| 684 | FirstLinePad = OptionFieldWidth + InitialPad; | 
|---|
| 685 | Pad = FirstLinePad; | 
|---|
| 686 | } | 
|---|
| 687 |  | 
|---|
| 688 | SmallVector<StringRef> Lines; | 
|---|
| 689 | Opt.HelpText.split(A&: Lines, Separator: '\n'); | 
|---|
| 690 | assert(Lines.size() && "Expected at least the first line in the help text"); | 
|---|
| 691 | auto *LinesIt = Lines.begin(); | 
|---|
| 692 | OS.indent(NumSpaces: FirstLinePad + 1) << *LinesIt << '\n'; | 
|---|
| 693 | while (Lines.end() != ++LinesIt) | 
|---|
| 694 | OS.indent(NumSpaces: Pad + 1) << *LinesIt << '\n'; | 
|---|
| 695 | } | 
|---|
| 696 | } | 
|---|
| 697 |  | 
|---|
| 698 | static const char *getOptionHelpGroup(const OptTable &Opts, OptSpecifier Id) { | 
|---|
| 699 | unsigned GroupID = Opts.getOptionGroupID(id: Id); | 
|---|
| 700 |  | 
|---|
| 701 | // If not in a group, return the default help group. | 
|---|
| 702 | if (!GroupID) | 
|---|
| 703 | return "OPTIONS"; | 
|---|
| 704 |  | 
|---|
| 705 | // Abuse the help text of the option groups to store the "help group" | 
|---|
| 706 | // name. | 
|---|
| 707 | // | 
|---|
| 708 | // FIXME: Split out option groups. | 
|---|
| 709 | if (const char *GroupHelp = Opts.getOptionHelpText(id: GroupID)) | 
|---|
| 710 | return GroupHelp; | 
|---|
| 711 |  | 
|---|
| 712 | // Otherwise keep looking. | 
|---|
| 713 | return getOptionHelpGroup(Opts, Id: GroupID); | 
|---|
| 714 | } | 
|---|
| 715 |  | 
|---|
| 716 | void OptTable::printHelp(raw_ostream &OS, const char *Usage, const char *Title, | 
|---|
| 717 | bool ShowHidden, bool ShowAllAliases, | 
|---|
| 718 | Visibility VisibilityMask) const { | 
|---|
| 719 | return internalPrintHelp( | 
|---|
| 720 | OS, Usage, Title, ShowHidden, ShowAllAliases, | 
|---|
| 721 | ExcludeOption: [VisibilityMask](const Info &CandidateInfo) -> bool { | 
|---|
| 722 | return (CandidateInfo.Visibility & VisibilityMask) == 0; | 
|---|
| 723 | }, | 
|---|
| 724 | VisibilityMask); | 
|---|
| 725 | } | 
|---|
| 726 |  | 
|---|
| 727 | void OptTable::printHelp(raw_ostream &OS, const char *Usage, const char *Title, | 
|---|
| 728 | unsigned FlagsToInclude, unsigned FlagsToExclude, | 
|---|
| 729 | bool ShowAllAliases) const { | 
|---|
| 730 | bool ShowHidden = !(FlagsToExclude & HelpHidden); | 
|---|
| 731 | FlagsToExclude &= ~HelpHidden; | 
|---|
| 732 | return internalPrintHelp( | 
|---|
| 733 | OS, Usage, Title, ShowHidden, ShowAllAliases, | 
|---|
| 734 | ExcludeOption: [FlagsToInclude, FlagsToExclude](const Info &CandidateInfo) { | 
|---|
| 735 | if (FlagsToInclude && !(CandidateInfo.Flags & FlagsToInclude)) | 
|---|
| 736 | return true; | 
|---|
| 737 | if (CandidateInfo.Flags & FlagsToExclude) | 
|---|
| 738 | return true; | 
|---|
| 739 | return false; | 
|---|
| 740 | }, | 
|---|
| 741 | VisibilityMask: Visibility(0)); | 
|---|
| 742 | } | 
|---|
| 743 |  | 
|---|
| 744 | void OptTable::internalPrintHelp( | 
|---|
| 745 | raw_ostream &OS, const char *Usage, const char *Title, bool ShowHidden, | 
|---|
| 746 | bool ShowAllAliases, std::function<bool(const Info &)> ExcludeOption, | 
|---|
| 747 | Visibility VisibilityMask) const { | 
|---|
| 748 | OS << "OVERVIEW: "<< Title << "\n\n"; | 
|---|
| 749 | OS << "USAGE: "<< Usage << "\n\n"; | 
|---|
| 750 |  | 
|---|
| 751 | // Render help text into a map of group-name to a list of (option, help) | 
|---|
| 752 | // pairs. | 
|---|
| 753 | std::map<std::string, std::vector<OptionInfo>> GroupedOptionHelp; | 
|---|
| 754 |  | 
|---|
| 755 | for (unsigned Id = 1, e = getNumOptions() + 1; Id != e; ++Id) { | 
|---|
| 756 | // FIXME: Split out option groups. | 
|---|
| 757 | if (getOptionKind(id: Id) == Option::GroupClass) | 
|---|
| 758 | continue; | 
|---|
| 759 |  | 
|---|
| 760 | const Info &CandidateInfo = getInfo(Opt: Id); | 
|---|
| 761 | if (!ShowHidden && (CandidateInfo.Flags & opt::HelpHidden)) | 
|---|
| 762 | continue; | 
|---|
| 763 |  | 
|---|
| 764 | if (ExcludeOption(CandidateInfo)) | 
|---|
| 765 | continue; | 
|---|
| 766 |  | 
|---|
| 767 | // If an alias doesn't have a help text, show a help text for the aliased | 
|---|
| 768 | // option instead. | 
|---|
| 769 | const char *HelpText = getOptionHelpText(id: Id, VisibilityMask); | 
|---|
| 770 | if (!HelpText && ShowAllAliases) { | 
|---|
| 771 | const Option Alias = getOption(Opt: Id).getAlias(); | 
|---|
| 772 | if (Alias.isValid()) | 
|---|
| 773 | HelpText = getOptionHelpText(id: Alias.getID(), VisibilityMask); | 
|---|
| 774 | } | 
|---|
| 775 |  | 
|---|
| 776 | if (HelpText && (strlen(s: HelpText) != 0)) { | 
|---|
| 777 | const char *HelpGroup = getOptionHelpGroup(Opts: *this, Id); | 
|---|
| 778 | const std::string &OptName = getOptionHelpName(Opts: *this, Id); | 
|---|
| 779 | GroupedOptionHelp[HelpGroup].push_back(x: {.Name: OptName, .HelpText: HelpText}); | 
|---|
| 780 | } | 
|---|
| 781 | } | 
|---|
| 782 |  | 
|---|
| 783 | for (auto& OptionGroup : GroupedOptionHelp) { | 
|---|
| 784 | if (OptionGroup.first != GroupedOptionHelp.begin()->first) | 
|---|
| 785 | OS << "\n"; | 
|---|
| 786 | PrintHelpOptionList(OS, Title: OptionGroup.first, OptionHelp&: OptionGroup.second); | 
|---|
| 787 | } | 
|---|
| 788 |  | 
|---|
| 789 | OS.flush(); | 
|---|
| 790 | } | 
|---|
| 791 |  | 
|---|
| 792 | GenericOptTable::GenericOptTable(const StringTable &StrTable, | 
|---|
| 793 | ArrayRef<StringTable::Offset> PrefixesTable, | 
|---|
| 794 | ArrayRef<Info> OptionInfos, bool IgnoreCase) | 
|---|
| 795 | : OptTable(StrTable, PrefixesTable, OptionInfos, IgnoreCase) { | 
|---|
| 796 |  | 
|---|
| 797 | std::set<StringRef> TmpPrefixesUnion; | 
|---|
| 798 | for (auto const &Info : OptionInfos.drop_front(N: FirstSearchableIndex)) | 
|---|
| 799 | for (auto PrefixOffset : Info.getPrefixOffsets(PrefixesTable)) | 
|---|
| 800 | TmpPrefixesUnion.insert(x: StrTable[PrefixOffset]); | 
|---|
| 801 | PrefixesUnion.append(in_start: TmpPrefixesUnion.begin(), in_end: TmpPrefixesUnion.end()); | 
|---|
| 802 | buildPrefixChars(); | 
|---|
| 803 | } | 
|---|
| 804 |  | 
|---|