12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807 |
- //===-- CommandLine.cpp - Command line parser implementation --------------===//
- //
- // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
- // See https://llvm.org/LICENSE.txt for license information.
- // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- //
- //===----------------------------------------------------------------------===//
- //
- // This class implements a command line argument processor that is useful when
- // creating a tool. It provides a simple, minimalistic interface that is easily
- // extensible and supports nonlocal (library) command line options.
- //
- // Note that rather than trying to figure out what this code does, you could try
- // reading the library documentation located in docs/CommandLine.html
- //
- //===----------------------------------------------------------------------===//
- #include "llvm/Support/CommandLine.h"
- #include "DebugOptions.h"
- #include "llvm-c/Support.h"
- #include "llvm/ADT/ArrayRef.h"
- #include "llvm/ADT/STLFunctionalExtras.h"
- #include "llvm/ADT/SmallPtrSet.h"
- #include "llvm/ADT/SmallString.h"
- #include "llvm/ADT/StringExtras.h"
- #include "llvm/ADT/StringMap.h"
- #include "llvm/ADT/StringRef.h"
- #include "llvm/ADT/Twine.h"
- #include "llvm/Config/config.h"
- #include "llvm/Support/ConvertUTF.h"
- #include "llvm/Support/Debug.h"
- #include "llvm/Support/Error.h"
- #include "llvm/Support/ErrorHandling.h"
- #include "llvm/Support/FileSystem.h"
- #include "llvm/Support/ManagedStatic.h"
- #include "llvm/Support/MemoryBuffer.h"
- #include "llvm/Support/Path.h"
- #include "llvm/Support/Process.h"
- #include "llvm/Support/StringSaver.h"
- #include "llvm/Support/VirtualFileSystem.h"
- #include "llvm/Support/raw_ostream.h"
- #include <cstdlib>
- #include <optional>
- #include <string>
- using namespace llvm;
- using namespace cl;
- #define DEBUG_TYPE "commandline"
- //===----------------------------------------------------------------------===//
- // Template instantiations and anchors.
- //
- namespace llvm {
- namespace cl {
- template class basic_parser<bool>;
- template class basic_parser<boolOrDefault>;
- template class basic_parser<int>;
- template class basic_parser<long>;
- template class basic_parser<long long>;
- template class basic_parser<unsigned>;
- template class basic_parser<unsigned long>;
- template class basic_parser<unsigned long long>;
- template class basic_parser<double>;
- template class basic_parser<float>;
- template class basic_parser<std::string>;
- template class basic_parser<char>;
- template class opt<unsigned>;
- template class opt<int>;
- template class opt<std::string>;
- template class opt<char>;
- template class opt<bool>;
- } // namespace cl
- } // namespace llvm
- // Pin the vtables to this file.
- void GenericOptionValue::anchor() {}
- void OptionValue<boolOrDefault>::anchor() {}
- void OptionValue<std::string>::anchor() {}
- void Option::anchor() {}
- void basic_parser_impl::anchor() {}
- void parser<bool>::anchor() {}
- void parser<boolOrDefault>::anchor() {}
- void parser<int>::anchor() {}
- void parser<long>::anchor() {}
- void parser<long long>::anchor() {}
- void parser<unsigned>::anchor() {}
- void parser<unsigned long>::anchor() {}
- void parser<unsigned long long>::anchor() {}
- void parser<double>::anchor() {}
- void parser<float>::anchor() {}
- void parser<std::string>::anchor() {}
- void parser<char>::anchor() {}
- //===----------------------------------------------------------------------===//
- const static size_t DefaultPad = 2;
- static StringRef ArgPrefix = "-";
- static StringRef ArgPrefixLong = "--";
- static StringRef ArgHelpPrefix = " - ";
- static size_t argPlusPrefixesSize(StringRef ArgName, size_t Pad = DefaultPad) {
- size_t Len = ArgName.size();
- if (Len == 1)
- return Len + Pad + ArgPrefix.size() + ArgHelpPrefix.size();
- return Len + Pad + ArgPrefixLong.size() + ArgHelpPrefix.size();
- }
- static SmallString<8> argPrefix(StringRef ArgName, size_t Pad = DefaultPad) {
- SmallString<8> Prefix;
- for (size_t I = 0; I < Pad; ++I) {
- Prefix.push_back(' ');
- }
- Prefix.append(ArgName.size() > 1 ? ArgPrefixLong : ArgPrefix);
- return Prefix;
- }
- // Option predicates...
- static inline bool isGrouping(const Option *O) {
- return O->getMiscFlags() & cl::Grouping;
- }
- static inline bool isPrefixedOrGrouping(const Option *O) {
- return isGrouping(O) || O->getFormattingFlag() == cl::Prefix ||
- O->getFormattingFlag() == cl::AlwaysPrefix;
- }
- namespace {
- class PrintArg {
- StringRef ArgName;
- size_t Pad;
- public:
- PrintArg(StringRef ArgName, size_t Pad = DefaultPad) : ArgName(ArgName), Pad(Pad) {}
- friend raw_ostream &operator<<(raw_ostream &OS, const PrintArg &);
- };
- raw_ostream &operator<<(raw_ostream &OS, const PrintArg& Arg) {
- OS << argPrefix(Arg.ArgName, Arg.Pad) << Arg.ArgName;
- return OS;
- }
- class CommandLineParser {
- public:
- // Globals for name and overview of program. Program name is not a string to
- // avoid static ctor/dtor issues.
- std::string ProgramName;
- StringRef ProgramOverview;
- // This collects additional help to be printed.
- std::vector<StringRef> MoreHelp;
- // This collects Options added with the cl::DefaultOption flag. Since they can
- // be overridden, they are not added to the appropriate SubCommands until
- // ParseCommandLineOptions actually runs.
- SmallVector<Option*, 4> DefaultOptions;
- // This collects the different option categories that have been registered.
- SmallPtrSet<OptionCategory *, 16> RegisteredOptionCategories;
- // This collects the different subcommands that have been registered.
- SmallPtrSet<SubCommand *, 4> RegisteredSubCommands;
- CommandLineParser() {
- registerSubCommand(&SubCommand::getTopLevel());
- registerSubCommand(&SubCommand::getAll());
- }
- void ResetAllOptionOccurrences();
- bool ParseCommandLineOptions(int argc, const char *const *argv,
- StringRef Overview, raw_ostream *Errs = nullptr,
- bool LongOptionsUseDoubleDash = false);
- void addLiteralOption(Option &Opt, SubCommand *SC, StringRef Name) {
- if (Opt.hasArgStr())
- return;
- if (!SC->OptionsMap.insert(std::make_pair(Name, &Opt)).second) {
- errs() << ProgramName << ": CommandLine Error: Option '" << Name
- << "' registered more than once!\n";
- report_fatal_error("inconsistency in registered CommandLine options");
- }
- // If we're adding this to all sub-commands, add it to the ones that have
- // already been registered.
- if (SC == &SubCommand::getAll()) {
- for (auto *Sub : RegisteredSubCommands) {
- if (SC == Sub)
- continue;
- addLiteralOption(Opt, Sub, Name);
- }
- }
- }
- void addLiteralOption(Option &Opt, StringRef Name) {
- if (Opt.Subs.empty())
- addLiteralOption(Opt, &SubCommand::getTopLevel(), Name);
- else {
- for (auto *SC : Opt.Subs)
- addLiteralOption(Opt, SC, Name);
- }
- }
- void addOption(Option *O, SubCommand *SC) {
- bool HadErrors = false;
- if (O->hasArgStr()) {
- // If it's a DefaultOption, check to make sure it isn't already there.
- if (O->isDefaultOption() &&
- SC->OptionsMap.find(O->ArgStr) != SC->OptionsMap.end())
- return;
- // Add argument to the argument map!
- if (!SC->OptionsMap.insert(std::make_pair(O->ArgStr, O)).second) {
- errs() << ProgramName << ": CommandLine Error: Option '" << O->ArgStr
- << "' registered more than once!\n";
- HadErrors = true;
- }
- }
- // Remember information about positional options.
- if (O->getFormattingFlag() == cl::Positional)
- SC->PositionalOpts.push_back(O);
- else if (O->getMiscFlags() & cl::Sink) // Remember sink options
- SC->SinkOpts.push_back(O);
- else if (O->getNumOccurrencesFlag() == cl::ConsumeAfter) {
- if (SC->ConsumeAfterOpt) {
- O->error("Cannot specify more than one option with cl::ConsumeAfter!");
- HadErrors = true;
- }
- SC->ConsumeAfterOpt = O;
- }
- // Fail hard if there were errors. These are strictly unrecoverable and
- // indicate serious issues such as conflicting option names or an
- // incorrectly
- // linked LLVM distribution.
- if (HadErrors)
- report_fatal_error("inconsistency in registered CommandLine options");
- // If we're adding this to all sub-commands, add it to the ones that have
- // already been registered.
- if (SC == &SubCommand::getAll()) {
- for (auto *Sub : RegisteredSubCommands) {
- if (SC == Sub)
- continue;
- addOption(O, Sub);
- }
- }
- }
- void addOption(Option *O, bool ProcessDefaultOption = false) {
- if (!ProcessDefaultOption && O->isDefaultOption()) {
- DefaultOptions.push_back(O);
- return;
- }
- if (O->Subs.empty()) {
- addOption(O, &SubCommand::getTopLevel());
- } else {
- for (auto *SC : O->Subs)
- addOption(O, SC);
- }
- }
- void removeOption(Option *O, SubCommand *SC) {
- SmallVector<StringRef, 16> OptionNames;
- O->getExtraOptionNames(OptionNames);
- if (O->hasArgStr())
- OptionNames.push_back(O->ArgStr);
- SubCommand &Sub = *SC;
- auto End = Sub.OptionsMap.end();
- for (auto Name : OptionNames) {
- auto I = Sub.OptionsMap.find(Name);
- if (I != End && I->getValue() == O)
- Sub.OptionsMap.erase(I);
- }
- if (O->getFormattingFlag() == cl::Positional)
- for (auto *Opt = Sub.PositionalOpts.begin();
- Opt != Sub.PositionalOpts.end(); ++Opt) {
- if (*Opt == O) {
- Sub.PositionalOpts.erase(Opt);
- break;
- }
- }
- else if (O->getMiscFlags() & cl::Sink)
- for (auto *Opt = Sub.SinkOpts.begin(); Opt != Sub.SinkOpts.end(); ++Opt) {
- if (*Opt == O) {
- Sub.SinkOpts.erase(Opt);
- break;
- }
- }
- else if (O == Sub.ConsumeAfterOpt)
- Sub.ConsumeAfterOpt = nullptr;
- }
- void removeOption(Option *O) {
- if (O->Subs.empty())
- removeOption(O, &SubCommand::getTopLevel());
- else {
- if (O->isInAllSubCommands()) {
- for (auto *SC : RegisteredSubCommands)
- removeOption(O, SC);
- } else {
- for (auto *SC : O->Subs)
- removeOption(O, SC);
- }
- }
- }
- bool hasOptions(const SubCommand &Sub) const {
- return (!Sub.OptionsMap.empty() || !Sub.PositionalOpts.empty() ||
- nullptr != Sub.ConsumeAfterOpt);
- }
- bool hasOptions() const {
- for (const auto *S : RegisteredSubCommands) {
- if (hasOptions(*S))
- return true;
- }
- return false;
- }
- SubCommand *getActiveSubCommand() { return ActiveSubCommand; }
- void updateArgStr(Option *O, StringRef NewName, SubCommand *SC) {
- SubCommand &Sub = *SC;
- if (!Sub.OptionsMap.insert(std::make_pair(NewName, O)).second) {
- errs() << ProgramName << ": CommandLine Error: Option '" << O->ArgStr
- << "' registered more than once!\n";
- report_fatal_error("inconsistency in registered CommandLine options");
- }
- Sub.OptionsMap.erase(O->ArgStr);
- }
- void updateArgStr(Option *O, StringRef NewName) {
- if (O->Subs.empty())
- updateArgStr(O, NewName, &SubCommand::getTopLevel());
- else {
- if (O->isInAllSubCommands()) {
- for (auto *SC : RegisteredSubCommands)
- updateArgStr(O, NewName, SC);
- } else {
- for (auto *SC : O->Subs)
- updateArgStr(O, NewName, SC);
- }
- }
- }
- void printOptionValues();
- void registerCategory(OptionCategory *cat) {
- assert(count_if(RegisteredOptionCategories,
- [cat](const OptionCategory *Category) {
- return cat->getName() == Category->getName();
- }) == 0 &&
- "Duplicate option categories");
- RegisteredOptionCategories.insert(cat);
- }
- void registerSubCommand(SubCommand *sub) {
- assert(count_if(RegisteredSubCommands,
- [sub](const SubCommand *Sub) {
- return (!sub->getName().empty()) &&
- (Sub->getName() == sub->getName());
- }) == 0 &&
- "Duplicate subcommands");
- RegisteredSubCommands.insert(sub);
- // For all options that have been registered for all subcommands, add the
- // option to this subcommand now.
- if (sub != &SubCommand::getAll()) {
- for (auto &E : SubCommand::getAll().OptionsMap) {
- Option *O = E.second;
- if ((O->isPositional() || O->isSink() || O->isConsumeAfter()) ||
- O->hasArgStr())
- addOption(O, sub);
- else
- addLiteralOption(*O, sub, E.first());
- }
- }
- }
- void unregisterSubCommand(SubCommand *sub) {
- RegisteredSubCommands.erase(sub);
- }
- iterator_range<typename SmallPtrSet<SubCommand *, 4>::iterator>
- getRegisteredSubcommands() {
- return make_range(RegisteredSubCommands.begin(),
- RegisteredSubCommands.end());
- }
- void reset() {
- ActiveSubCommand = nullptr;
- ProgramName.clear();
- ProgramOverview = StringRef();
- MoreHelp.clear();
- RegisteredOptionCategories.clear();
- ResetAllOptionOccurrences();
- RegisteredSubCommands.clear();
- SubCommand::getTopLevel().reset();
- SubCommand::getAll().reset();
- registerSubCommand(&SubCommand::getTopLevel());
- registerSubCommand(&SubCommand::getAll());
- DefaultOptions.clear();
- }
- private:
- SubCommand *ActiveSubCommand = nullptr;
- Option *LookupOption(SubCommand &Sub, StringRef &Arg, StringRef &Value);
- Option *LookupLongOption(SubCommand &Sub, StringRef &Arg, StringRef &Value,
- bool LongOptionsUseDoubleDash, bool HaveDoubleDash) {
- Option *Opt = LookupOption(Sub, Arg, Value);
- if (Opt && LongOptionsUseDoubleDash && !HaveDoubleDash && !isGrouping(Opt))
- return nullptr;
- return Opt;
- }
- SubCommand *LookupSubCommand(StringRef Name);
- };
- } // namespace
- static ManagedStatic<CommandLineParser> GlobalParser;
- void cl::AddLiteralOption(Option &O, StringRef Name) {
- GlobalParser->addLiteralOption(O, Name);
- }
- extrahelp::extrahelp(StringRef Help) : morehelp(Help) {
- GlobalParser->MoreHelp.push_back(Help);
- }
- void Option::addArgument() {
- GlobalParser->addOption(this);
- FullyInitialized = true;
- }
- void Option::removeArgument() { GlobalParser->removeOption(this); }
- void Option::setArgStr(StringRef S) {
- if (FullyInitialized)
- GlobalParser->updateArgStr(this, S);
- assert((S.empty() || S[0] != '-') && "Option can't start with '-");
- ArgStr = S;
- if (ArgStr.size() == 1)
- setMiscFlag(Grouping);
- }
- void Option::addCategory(OptionCategory &C) {
- assert(!Categories.empty() && "Categories cannot be empty.");
- // Maintain backward compatibility by replacing the default GeneralCategory
- // if it's still set. Otherwise, just add the new one. The GeneralCategory
- // must be explicitly added if you want multiple categories that include it.
- if (&C != &getGeneralCategory() && Categories[0] == &getGeneralCategory())
- Categories[0] = &C;
- else if (!is_contained(Categories, &C))
- Categories.push_back(&C);
- }
- void Option::reset() {
- NumOccurrences = 0;
- setDefault();
- if (isDefaultOption())
- removeArgument();
- }
- void OptionCategory::registerCategory() {
- GlobalParser->registerCategory(this);
- }
- // A special subcommand representing no subcommand. It is particularly important
- // that this ManagedStatic uses constant initailization and not dynamic
- // initialization because it is referenced from cl::opt constructors, which run
- // dynamically in an arbitrary order.
- LLVM_REQUIRE_CONSTANT_INITIALIZATION
- ManagedStatic<SubCommand> llvm::cl::TopLevelSubCommand;
- // A special subcommand that can be used to put an option into all subcommands.
- ManagedStatic<SubCommand> llvm::cl::AllSubCommands;
- SubCommand &SubCommand::getTopLevel() { return *TopLevelSubCommand; }
- SubCommand &SubCommand::getAll() { return *AllSubCommands; }
- void SubCommand::registerSubCommand() {
- GlobalParser->registerSubCommand(this);
- }
- void SubCommand::unregisterSubCommand() {
- GlobalParser->unregisterSubCommand(this);
- }
- void SubCommand::reset() {
- PositionalOpts.clear();
- SinkOpts.clear();
- OptionsMap.clear();
- ConsumeAfterOpt = nullptr;
- }
- SubCommand::operator bool() const {
- return (GlobalParser->getActiveSubCommand() == this);
- }
- //===----------------------------------------------------------------------===//
- // Basic, shared command line option processing machinery.
- //
- /// LookupOption - Lookup the option specified by the specified option on the
- /// command line. If there is a value specified (after an equal sign) return
- /// that as well. This assumes that leading dashes have already been stripped.
- Option *CommandLineParser::LookupOption(SubCommand &Sub, StringRef &Arg,
- StringRef &Value) {
- // Reject all dashes.
- if (Arg.empty())
- return nullptr;
- assert(&Sub != &SubCommand::getAll());
- size_t EqualPos = Arg.find('=');
- // If we have an equals sign, remember the value.
- if (EqualPos == StringRef::npos) {
- // Look up the option.
- return Sub.OptionsMap.lookup(Arg);
- }
- // If the argument before the = is a valid option name and the option allows
- // non-prefix form (ie is not AlwaysPrefix), we match. If not, signal match
- // failure by returning nullptr.
- auto I = Sub.OptionsMap.find(Arg.substr(0, EqualPos));
- if (I == Sub.OptionsMap.end())
- return nullptr;
- auto *O = I->second;
- if (O->getFormattingFlag() == cl::AlwaysPrefix)
- return nullptr;
- Value = Arg.substr(EqualPos + 1);
- Arg = Arg.substr(0, EqualPos);
- return I->second;
- }
- SubCommand *CommandLineParser::LookupSubCommand(StringRef Name) {
- if (Name.empty())
- return &SubCommand::getTopLevel();
- for (auto *S : RegisteredSubCommands) {
- if (S == &SubCommand::getAll())
- continue;
- if (S->getName().empty())
- continue;
- if (StringRef(S->getName()) == StringRef(Name))
- return S;
- }
- return &SubCommand::getTopLevel();
- }
- /// LookupNearestOption - Lookup the closest match to the option specified by
- /// the specified option on the command line. If there is a value specified
- /// (after an equal sign) return that as well. This assumes that leading dashes
- /// have already been stripped.
- static Option *LookupNearestOption(StringRef Arg,
- const StringMap<Option *> &OptionsMap,
- std::string &NearestString) {
- // Reject all dashes.
- if (Arg.empty())
- return nullptr;
- // Split on any equal sign.
- std::pair<StringRef, StringRef> SplitArg = Arg.split('=');
- StringRef &LHS = SplitArg.first; // LHS == Arg when no '=' is present.
- StringRef &RHS = SplitArg.second;
- // Find the closest match.
- Option *Best = nullptr;
- unsigned BestDistance = 0;
- for (StringMap<Option *>::const_iterator it = OptionsMap.begin(),
- ie = OptionsMap.end();
- it != ie; ++it) {
- Option *O = it->second;
- // Do not suggest really hidden options (not shown in any help).
- if (O->getOptionHiddenFlag() == ReallyHidden)
- continue;
- SmallVector<StringRef, 16> OptionNames;
- O->getExtraOptionNames(OptionNames);
- if (O->hasArgStr())
- OptionNames.push_back(O->ArgStr);
- bool PermitValue = O->getValueExpectedFlag() != cl::ValueDisallowed;
- StringRef Flag = PermitValue ? LHS : Arg;
- for (const auto &Name : OptionNames) {
- unsigned Distance = StringRef(Name).edit_distance(
- Flag, /*AllowReplacements=*/true, /*MaxEditDistance=*/BestDistance);
- if (!Best || Distance < BestDistance) {
- Best = O;
- BestDistance = Distance;
- if (RHS.empty() || !PermitValue)
- NearestString = std::string(Name);
- else
- NearestString = (Twine(Name) + "=" + RHS).str();
- }
- }
- }
- return Best;
- }
- /// CommaSeparateAndAddOccurrence - A wrapper around Handler->addOccurrence()
- /// that does special handling of cl::CommaSeparated options.
- static bool CommaSeparateAndAddOccurrence(Option *Handler, unsigned pos,
- StringRef ArgName, StringRef Value,
- bool MultiArg = false) {
- // Check to see if this option accepts a comma separated list of values. If
- // it does, we have to split up the value into multiple values.
- if (Handler->getMiscFlags() & CommaSeparated) {
- StringRef Val(Value);
- StringRef::size_type Pos = Val.find(',');
- while (Pos != StringRef::npos) {
- // Process the portion before the comma.
- if (Handler->addOccurrence(pos, ArgName, Val.substr(0, Pos), MultiArg))
- return true;
- // Erase the portion before the comma, AND the comma.
- Val = Val.substr(Pos + 1);
- // Check for another comma.
- Pos = Val.find(',');
- }
- Value = Val;
- }
- return Handler->addOccurrence(pos, ArgName, Value, MultiArg);
- }
- /// ProvideOption - For Value, this differentiates between an empty value ("")
- /// and a null value (StringRef()). The later is accepted for arguments that
- /// don't allow a value (-foo) the former is rejected (-foo=).
- static inline bool ProvideOption(Option *Handler, StringRef ArgName,
- StringRef Value, int argc,
- const char *const *argv, int &i) {
- // Is this a multi-argument option?
- unsigned NumAdditionalVals = Handler->getNumAdditionalVals();
- // Enforce value requirements
- switch (Handler->getValueExpectedFlag()) {
- case ValueRequired:
- if (!Value.data()) { // No value specified?
- // If no other argument or the option only supports prefix form, we
- // cannot look at the next argument.
- if (i + 1 >= argc || Handler->getFormattingFlag() == cl::AlwaysPrefix)
- return Handler->error("requires a value!");
- // Steal the next argument, like for '-o filename'
- assert(argv && "null check");
- Value = StringRef(argv[++i]);
- }
- break;
- case ValueDisallowed:
- if (NumAdditionalVals > 0)
- return Handler->error("multi-valued option specified"
- " with ValueDisallowed modifier!");
- if (Value.data())
- return Handler->error("does not allow a value! '" + Twine(Value) +
- "' specified.");
- break;
- case ValueOptional:
- break;
- }
- // If this isn't a multi-arg option, just run the handler.
- if (NumAdditionalVals == 0)
- return CommaSeparateAndAddOccurrence(Handler, i, ArgName, Value);
- // If it is, run the handle several times.
- bool MultiArg = false;
- if (Value.data()) {
- if (CommaSeparateAndAddOccurrence(Handler, i, ArgName, Value, MultiArg))
- return true;
- --NumAdditionalVals;
- MultiArg = true;
- }
- while (NumAdditionalVals > 0) {
- if (i + 1 >= argc)
- return Handler->error("not enough values!");
- assert(argv && "null check");
- Value = StringRef(argv[++i]);
- if (CommaSeparateAndAddOccurrence(Handler, i, ArgName, Value, MultiArg))
- return true;
- MultiArg = true;
- --NumAdditionalVals;
- }
- return false;
- }
- bool llvm::cl::ProvidePositionalOption(Option *Handler, StringRef Arg, int i) {
- int Dummy = i;
- return ProvideOption(Handler, Handler->ArgStr, Arg, 0, nullptr, Dummy);
- }
- // getOptionPred - Check to see if there are any options that satisfy the
- // specified predicate with names that are the prefixes in Name. This is
- // checked by progressively stripping characters off of the name, checking to
- // see if there options that satisfy the predicate. If we find one, return it,
- // otherwise return null.
- //
- static Option *getOptionPred(StringRef Name, size_t &Length,
- bool (*Pred)(const Option *),
- const StringMap<Option *> &OptionsMap) {
- StringMap<Option *>::const_iterator OMI = OptionsMap.find(Name);
- if (OMI != OptionsMap.end() && !Pred(OMI->getValue()))
- OMI = OptionsMap.end();
- // Loop while we haven't found an option and Name still has at least two
- // characters in it (so that the next iteration will not be the empty
- // string.
- while (OMI == OptionsMap.end() && Name.size() > 1) {
- Name = Name.substr(0, Name.size() - 1); // Chop off the last character.
- OMI = OptionsMap.find(Name);
- if (OMI != OptionsMap.end() && !Pred(OMI->getValue()))
- OMI = OptionsMap.end();
- }
- if (OMI != OptionsMap.end() && Pred(OMI->second)) {
- Length = Name.size();
- return OMI->second; // Found one!
- }
- return nullptr; // No option found!
- }
- /// HandlePrefixedOrGroupedOption - The specified argument string (which started
- /// with at least one '-') does not fully match an available option. Check to
- /// see if this is a prefix or grouped option. If so, split arg into output an
- /// Arg/Value pair and return the Option to parse it with.
- static Option *
- HandlePrefixedOrGroupedOption(StringRef &Arg, StringRef &Value,
- bool &ErrorParsing,
- const StringMap<Option *> &OptionsMap) {
- if (Arg.size() == 1)
- return nullptr;
- // Do the lookup!
- size_t Length = 0;
- Option *PGOpt = getOptionPred(Arg, Length, isPrefixedOrGrouping, OptionsMap);
- if (!PGOpt)
- return nullptr;
- do {
- StringRef MaybeValue =
- (Length < Arg.size()) ? Arg.substr(Length) : StringRef();
- Arg = Arg.substr(0, Length);
- assert(OptionsMap.count(Arg) && OptionsMap.find(Arg)->second == PGOpt);
- // cl::Prefix options do not preserve '=' when used separately.
- // The behavior for them with grouped options should be the same.
- if (MaybeValue.empty() || PGOpt->getFormattingFlag() == cl::AlwaysPrefix ||
- (PGOpt->getFormattingFlag() == cl::Prefix && MaybeValue[0] != '=')) {
- Value = MaybeValue;
- return PGOpt;
- }
- if (MaybeValue[0] == '=') {
- Value = MaybeValue.substr(1);
- return PGOpt;
- }
- // This must be a grouped option.
- assert(isGrouping(PGOpt) && "Broken getOptionPred!");
- // Grouping options inside a group can't have values.
- if (PGOpt->getValueExpectedFlag() == cl::ValueRequired) {
- ErrorParsing |= PGOpt->error("may not occur within a group!");
- return nullptr;
- }
- // Because the value for the option is not required, we don't need to pass
- // argc/argv in.
- int Dummy = 0;
- ErrorParsing |= ProvideOption(PGOpt, Arg, StringRef(), 0, nullptr, Dummy);
- // Get the next grouping option.
- Arg = MaybeValue;
- PGOpt = getOptionPred(Arg, Length, isGrouping, OptionsMap);
- } while (PGOpt);
- // We could not find a grouping option in the remainder of Arg.
- return nullptr;
- }
- static bool RequiresValue(const Option *O) {
- return O->getNumOccurrencesFlag() == cl::Required ||
- O->getNumOccurrencesFlag() == cl::OneOrMore;
- }
- static bool EatsUnboundedNumberOfValues(const Option *O) {
- return O->getNumOccurrencesFlag() == cl::ZeroOrMore ||
- O->getNumOccurrencesFlag() == cl::OneOrMore;
- }
- static bool isWhitespace(char C) {
- return C == ' ' || C == '\t' || C == '\r' || C == '\n';
- }
- static bool isWhitespaceOrNull(char C) {
- return isWhitespace(C) || C == '\0';
- }
- static bool isQuote(char C) { return C == '\"' || C == '\''; }
- void cl::TokenizeGNUCommandLine(StringRef Src, StringSaver &Saver,
- SmallVectorImpl<const char *> &NewArgv,
- bool MarkEOLs) {
- SmallString<128> Token;
- for (size_t I = 0, E = Src.size(); I != E; ++I) {
- // Consume runs of whitespace.
- if (Token.empty()) {
- while (I != E && isWhitespace(Src[I])) {
- // Mark the end of lines in response files.
- if (MarkEOLs && Src[I] == '\n')
- NewArgv.push_back(nullptr);
- ++I;
- }
- if (I == E)
- break;
- }
- char C = Src[I];
- // Backslash escapes the next character.
- if (I + 1 < E && C == '\\') {
- ++I; // Skip the escape.
- Token.push_back(Src[I]);
- continue;
- }
- // Consume a quoted string.
- if (isQuote(C)) {
- ++I;
- while (I != E && Src[I] != C) {
- // Backslash escapes the next character.
- if (Src[I] == '\\' && I + 1 != E)
- ++I;
- Token.push_back(Src[I]);
- ++I;
- }
- if (I == E)
- break;
- continue;
- }
- // End the token if this is whitespace.
- if (isWhitespace(C)) {
- if (!Token.empty())
- NewArgv.push_back(Saver.save(Token.str()).data());
- // Mark the end of lines in response files.
- if (MarkEOLs && C == '\n')
- NewArgv.push_back(nullptr);
- Token.clear();
- continue;
- }
- // This is a normal character. Append it.
- Token.push_back(C);
- }
- // Append the last token after hitting EOF with no whitespace.
- if (!Token.empty())
- NewArgv.push_back(Saver.save(Token.str()).data());
- }
- /// Backslashes are interpreted in a rather complicated way in the Windows-style
- /// command line, because backslashes are used both to separate path and to
- /// escape double quote. This method consumes runs of backslashes as well as the
- /// following double quote if it's escaped.
- ///
- /// * If an even number of backslashes is followed by a double quote, one
- /// backslash is output for every pair of backslashes, and the last double
- /// quote remains unconsumed. The double quote will later be interpreted as
- /// the start or end of a quoted string in the main loop outside of this
- /// function.
- ///
- /// * If an odd number of backslashes is followed by a double quote, one
- /// backslash is output for every pair of backslashes, and a double quote is
- /// output for the last pair of backslash-double quote. The double quote is
- /// consumed in this case.
- ///
- /// * Otherwise, backslashes are interpreted literally.
- static size_t parseBackslash(StringRef Src, size_t I, SmallString<128> &Token) {
- size_t E = Src.size();
- int BackslashCount = 0;
- // Skip the backslashes.
- do {
- ++I;
- ++BackslashCount;
- } while (I != E && Src[I] == '\\');
- bool FollowedByDoubleQuote = (I != E && Src[I] == '"');
- if (FollowedByDoubleQuote) {
- Token.append(BackslashCount / 2, '\\');
- if (BackslashCount % 2 == 0)
- return I - 1;
- Token.push_back('"');
- return I;
- }
- Token.append(BackslashCount, '\\');
- return I - 1;
- }
- // Windows treats whitespace, double quotes, and backslashes specially, except
- // when parsing the first token of a full command line, in which case
- // backslashes are not special.
- static bool isWindowsSpecialChar(char C) {
- return isWhitespaceOrNull(C) || C == '\\' || C == '\"';
- }
- static bool isWindowsSpecialCharInCommandName(char C) {
- return isWhitespaceOrNull(C) || C == '\"';
- }
- // Windows tokenization implementation. The implementation is designed to be
- // inlined and specialized for the two user entry points.
- static inline void tokenizeWindowsCommandLineImpl(
- StringRef Src, StringSaver &Saver, function_ref<void(StringRef)> AddToken,
- bool AlwaysCopy, function_ref<void()> MarkEOL, bool InitialCommandName) {
- SmallString<128> Token;
- // Sometimes, this function will be handling a full command line including an
- // executable pathname at the start. In that situation, the initial pathname
- // needs different handling from the following arguments, because when
- // CreateProcess or cmd.exe scans the pathname, it doesn't treat \ as
- // escaping the quote character, whereas when libc scans the rest of the
- // command line, it does.
- bool CommandName = InitialCommandName;
- // Try to do as much work inside the state machine as possible.
- enum { INIT, UNQUOTED, QUOTED } State = INIT;
- for (size_t I = 0, E = Src.size(); I < E; ++I) {
- switch (State) {
- case INIT: {
- assert(Token.empty() && "token should be empty in initial state");
- // Eat whitespace before a token.
- while (I < E && isWhitespaceOrNull(Src[I])) {
- if (Src[I] == '\n')
- MarkEOL();
- ++I;
- }
- // Stop if this was trailing whitespace.
- if (I >= E)
- break;
- size_t Start = I;
- if (CommandName) {
- while (I < E && !isWindowsSpecialCharInCommandName(Src[I]))
- ++I;
- } else {
- while (I < E && !isWindowsSpecialChar(Src[I]))
- ++I;
- }
- StringRef NormalChars = Src.slice(Start, I);
- if (I >= E || isWhitespaceOrNull(Src[I])) {
- // No special characters: slice out the substring and start the next
- // token. Copy the string if the caller asks us to.
- AddToken(AlwaysCopy ? Saver.save(NormalChars) : NormalChars);
- if (I < E && Src[I] == '\n') {
- MarkEOL();
- CommandName = InitialCommandName;
- } else {
- CommandName = false;
- }
- } else if (Src[I] == '\"') {
- Token += NormalChars;
- State = QUOTED;
- } else if (Src[I] == '\\') {
- assert(!CommandName && "or else we'd have treated it as a normal char");
- Token += NormalChars;
- I = parseBackslash(Src, I, Token);
- State = UNQUOTED;
- } else {
- llvm_unreachable("unexpected special character");
- }
- break;
- }
- case UNQUOTED:
- if (isWhitespaceOrNull(Src[I])) {
- // Whitespace means the end of the token. If we are in this state, the
- // token must have contained a special character, so we must copy the
- // token.
- AddToken(Saver.save(Token.str()));
- Token.clear();
- if (Src[I] == '\n') {
- CommandName = InitialCommandName;
- MarkEOL();
- } else {
- CommandName = false;
- }
- State = INIT;
- } else if (Src[I] == '\"') {
- State = QUOTED;
- } else if (Src[I] == '\\' && !CommandName) {
- I = parseBackslash(Src, I, Token);
- } else {
- Token.push_back(Src[I]);
- }
- break;
- case QUOTED:
- if (Src[I] == '\"') {
- if (I < (E - 1) && Src[I + 1] == '"') {
- // Consecutive double-quotes inside a quoted string implies one
- // double-quote.
- Token.push_back('"');
- ++I;
- } else {
- // Otherwise, end the quoted portion and return to the unquoted state.
- State = UNQUOTED;
- }
- } else if (Src[I] == '\\' && !CommandName) {
- I = parseBackslash(Src, I, Token);
- } else {
- Token.push_back(Src[I]);
- }
- break;
- }
- }
- if (State != INIT)
- AddToken(Saver.save(Token.str()));
- }
- void cl::TokenizeWindowsCommandLine(StringRef Src, StringSaver &Saver,
- SmallVectorImpl<const char *> &NewArgv,
- bool MarkEOLs) {
- auto AddToken = [&](StringRef Tok) { NewArgv.push_back(Tok.data()); };
- auto OnEOL = [&]() {
- if (MarkEOLs)
- NewArgv.push_back(nullptr);
- };
- tokenizeWindowsCommandLineImpl(Src, Saver, AddToken,
- /*AlwaysCopy=*/true, OnEOL, false);
- }
- void cl::TokenizeWindowsCommandLineNoCopy(StringRef Src, StringSaver &Saver,
- SmallVectorImpl<StringRef> &NewArgv) {
- auto AddToken = [&](StringRef Tok) { NewArgv.push_back(Tok); };
- auto OnEOL = []() {};
- tokenizeWindowsCommandLineImpl(Src, Saver, AddToken, /*AlwaysCopy=*/false,
- OnEOL, false);
- }
- void cl::TokenizeWindowsCommandLineFull(StringRef Src, StringSaver &Saver,
- SmallVectorImpl<const char *> &NewArgv,
- bool MarkEOLs) {
- auto AddToken = [&](StringRef Tok) { NewArgv.push_back(Tok.data()); };
- auto OnEOL = [&]() {
- if (MarkEOLs)
- NewArgv.push_back(nullptr);
- };
- tokenizeWindowsCommandLineImpl(Src, Saver, AddToken,
- /*AlwaysCopy=*/true, OnEOL, true);
- }
- void cl::tokenizeConfigFile(StringRef Source, StringSaver &Saver,
- SmallVectorImpl<const char *> &NewArgv,
- bool MarkEOLs) {
- for (const char *Cur = Source.begin(); Cur != Source.end();) {
- SmallString<128> Line;
- // Check for comment line.
- if (isWhitespace(*Cur)) {
- while (Cur != Source.end() && isWhitespace(*Cur))
- ++Cur;
- continue;
- }
- if (*Cur == '#') {
- while (Cur != Source.end() && *Cur != '\n')
- ++Cur;
- continue;
- }
- // Find end of the current line.
- const char *Start = Cur;
- for (const char *End = Source.end(); Cur != End; ++Cur) {
- if (*Cur == '\\') {
- if (Cur + 1 != End) {
- ++Cur;
- if (*Cur == '\n' ||
- (*Cur == '\r' && (Cur + 1 != End) && Cur[1] == '\n')) {
- Line.append(Start, Cur - 1);
- if (*Cur == '\r')
- ++Cur;
- Start = Cur + 1;
- }
- }
- } else if (*Cur == '\n')
- break;
- }
- // Tokenize line.
- Line.append(Start, Cur);
- cl::TokenizeGNUCommandLine(Line, Saver, NewArgv, MarkEOLs);
- }
- }
- // It is called byte order marker but the UTF-8 BOM is actually not affected
- // by the host system's endianness.
- static bool hasUTF8ByteOrderMark(ArrayRef<char> S) {
- return (S.size() >= 3 && S[0] == '\xef' && S[1] == '\xbb' && S[2] == '\xbf');
- }
- // Substitute <CFGDIR> with the file's base path.
- static void ExpandBasePaths(StringRef BasePath, StringSaver &Saver,
- const char *&Arg) {
- assert(sys::path::is_absolute(BasePath));
- constexpr StringLiteral Token("<CFGDIR>");
- const StringRef ArgString(Arg);
- SmallString<128> ResponseFile;
- StringRef::size_type StartPos = 0;
- for (StringRef::size_type TokenPos = ArgString.find(Token);
- TokenPos != StringRef::npos;
- TokenPos = ArgString.find(Token, StartPos)) {
- // Token may appear more than once per arg (e.g. comma-separated linker
- // args). Support by using path-append on any subsequent appearances.
- const StringRef LHS = ArgString.substr(StartPos, TokenPos - StartPos);
- if (ResponseFile.empty())
- ResponseFile = LHS;
- else
- llvm::sys::path::append(ResponseFile, LHS);
- ResponseFile.append(BasePath);
- StartPos = TokenPos + Token.size();
- }
- if (!ResponseFile.empty()) {
- // Path-append the remaining arg substring if at least one token appeared.
- const StringRef Remaining = ArgString.substr(StartPos);
- if (!Remaining.empty())
- llvm::sys::path::append(ResponseFile, Remaining);
- Arg = Saver.save(ResponseFile.str()).data();
- }
- }
- // FName must be an absolute path.
- Error ExpansionContext::expandResponseFile(
- StringRef FName, SmallVectorImpl<const char *> &NewArgv) {
- assert(sys::path::is_absolute(FName));
- llvm::ErrorOr<std::unique_ptr<MemoryBuffer>> MemBufOrErr =
- FS->getBufferForFile(FName);
- if (!MemBufOrErr) {
- std::error_code EC = MemBufOrErr.getError();
- return llvm::createStringError(EC, Twine("cannot not open file '") + FName +
- "': " + EC.message());
- }
- MemoryBuffer &MemBuf = *MemBufOrErr.get();
- StringRef Str(MemBuf.getBufferStart(), MemBuf.getBufferSize());
- // If we have a UTF-16 byte order mark, convert to UTF-8 for parsing.
- ArrayRef<char> BufRef(MemBuf.getBufferStart(), MemBuf.getBufferEnd());
- std::string UTF8Buf;
- if (hasUTF16ByteOrderMark(BufRef)) {
- if (!convertUTF16ToUTF8String(BufRef, UTF8Buf))
- return llvm::createStringError(std::errc::illegal_byte_sequence,
- "Could not convert UTF16 to UTF8");
- Str = StringRef(UTF8Buf);
- }
- // If we see UTF-8 BOM sequence at the beginning of a file, we shall remove
- // these bytes before parsing.
- // Reference: http://en.wikipedia.org/wiki/UTF-8#Byte_order_mark
- else if (hasUTF8ByteOrderMark(BufRef))
- Str = StringRef(BufRef.data() + 3, BufRef.size() - 3);
- // Tokenize the contents into NewArgv.
- Tokenizer(Str, Saver, NewArgv, MarkEOLs);
- // Expanded file content may require additional transformations, like using
- // absolute paths instead of relative in '@file' constructs or expanding
- // macros.
- if (!RelativeNames && !InConfigFile)
- return Error::success();
- StringRef BasePath = llvm::sys::path::parent_path(FName);
- for (const char *&Arg : NewArgv) {
- if (!Arg)
- continue;
- // Substitute <CFGDIR> with the file's base path.
- if (InConfigFile)
- ExpandBasePaths(BasePath, Saver, Arg);
- // Discover the case, when argument should be transformed into '@file' and
- // evaluate 'file' for it.
- StringRef ArgStr(Arg);
- StringRef FileName;
- bool ConfigInclusion = false;
- if (ArgStr.consume_front("@")) {
- FileName = ArgStr;
- if (!llvm::sys::path::is_relative(FileName))
- continue;
- } else if (ArgStr.consume_front("--config=")) {
- FileName = ArgStr;
- ConfigInclusion = true;
- } else {
- continue;
- }
- // Update expansion construct.
- SmallString<128> ResponseFile;
- ResponseFile.push_back('@');
- if (ConfigInclusion && !llvm::sys::path::has_parent_path(FileName)) {
- SmallString<128> FilePath;
- if (!findConfigFile(FileName, FilePath))
- return createStringError(
- std::make_error_code(std::errc::no_such_file_or_directory),
- "cannot not find configuration file: " + FileName);
- ResponseFile.append(FilePath);
- } else {
- ResponseFile.append(BasePath);
- llvm::sys::path::append(ResponseFile, FileName);
- }
- Arg = Saver.save(ResponseFile.str()).data();
- }
- return Error::success();
- }
- /// Expand response files on a command line recursively using the given
- /// StringSaver and tokenization strategy.
- Error ExpansionContext::expandResponseFiles(
- SmallVectorImpl<const char *> &Argv) {
- struct ResponseFileRecord {
- std::string File;
- size_t End;
- };
- // To detect recursive response files, we maintain a stack of files and the
- // position of the last argument in the file. This position is updated
- // dynamically as we recursively expand files.
- SmallVector<ResponseFileRecord, 3> FileStack;
- // Push a dummy entry that represents the initial command line, removing
- // the need to check for an empty list.
- FileStack.push_back({"", Argv.size()});
- // Don't cache Argv.size() because it can change.
- for (unsigned I = 0; I != Argv.size();) {
- while (I == FileStack.back().End) {
- // Passing the end of a file's argument list, so we can remove it from the
- // stack.
- FileStack.pop_back();
- }
- const char *Arg = Argv[I];
- // Check if it is an EOL marker
- if (Arg == nullptr) {
- ++I;
- continue;
- }
- if (Arg[0] != '@') {
- ++I;
- continue;
- }
- const char *FName = Arg + 1;
- // Note that CurrentDir is only used for top-level rsp files, the rest will
- // always have an absolute path deduced from the containing file.
- SmallString<128> CurrDir;
- if (llvm::sys::path::is_relative(FName)) {
- if (CurrentDir.empty()) {
- if (auto CWD = FS->getCurrentWorkingDirectory()) {
- CurrDir = *CWD;
- } else {
- return createStringError(
- CWD.getError(), Twine("cannot get absolute path for: ") + FName);
- }
- } else {
- CurrDir = CurrentDir;
- }
- llvm::sys::path::append(CurrDir, FName);
- FName = CurrDir.c_str();
- }
- ErrorOr<llvm::vfs::Status> Res = FS->status(FName);
- if (!Res || !Res->exists()) {
- std::error_code EC = Res.getError();
- if (!InConfigFile) {
- // If the specified file does not exist, leave '@file' unexpanded, as
- // libiberty does.
- if (!EC || EC == llvm::errc::no_such_file_or_directory) {
- ++I;
- continue;
- }
- }
- if (!EC)
- EC = llvm::errc::no_such_file_or_directory;
- return createStringError(EC, Twine("cannot not open file '") + FName +
- "': " + EC.message());
- }
- const llvm::vfs::Status &FileStatus = Res.get();
- auto IsEquivalent =
- [FileStatus, this](const ResponseFileRecord &RFile) -> ErrorOr<bool> {
- ErrorOr<llvm::vfs::Status> RHS = FS->status(RFile.File);
- if (!RHS)
- return RHS.getError();
- return FileStatus.equivalent(*RHS);
- };
- // Check for recursive response files.
- for (const auto &F : drop_begin(FileStack)) {
- if (ErrorOr<bool> R = IsEquivalent(F)) {
- if (R.get())
- return createStringError(
- R.getError(), Twine("recursive expansion of: '") + F.File + "'");
- } else {
- return createStringError(R.getError(),
- Twine("cannot open file: ") + F.File);
- }
- }
- // Replace this response file argument with the tokenization of its
- // contents. Nested response files are expanded in subsequent iterations.
- SmallVector<const char *, 0> ExpandedArgv;
- if (Error Err = expandResponseFile(FName, ExpandedArgv))
- return Err;
- for (ResponseFileRecord &Record : FileStack) {
- // Increase the end of all active records by the number of newly expanded
- // arguments, minus the response file itself.
- Record.End += ExpandedArgv.size() - 1;
- }
- FileStack.push_back({FName, I + ExpandedArgv.size()});
- Argv.erase(Argv.begin() + I);
- Argv.insert(Argv.begin() + I, ExpandedArgv.begin(), ExpandedArgv.end());
- }
- // If successful, the top of the file stack will mark the end of the Argv
- // stream. A failure here indicates a bug in the stack popping logic above.
- // Note that FileStack may have more than one element at this point because we
- // don't have a chance to pop the stack when encountering recursive files at
- // the end of the stream, so seeing that doesn't indicate a bug.
- assert(FileStack.size() > 0 && Argv.size() == FileStack.back().End);
- return Error::success();
- }
- bool cl::expandResponseFiles(int Argc, const char *const *Argv,
- const char *EnvVar, StringSaver &Saver,
- SmallVectorImpl<const char *> &NewArgv) {
- #ifdef _WIN32
- auto Tokenize = cl::TokenizeWindowsCommandLine;
- #else
- auto Tokenize = cl::TokenizeGNUCommandLine;
- #endif
- // The environment variable specifies initial options.
- if (EnvVar)
- if (std::optional<std::string> EnvValue = sys::Process::GetEnv(EnvVar))
- Tokenize(*EnvValue, Saver, NewArgv, /*MarkEOLs=*/false);
- // Command line options can override the environment variable.
- NewArgv.append(Argv + 1, Argv + Argc);
- ExpansionContext ECtx(Saver.getAllocator(), Tokenize);
- if (Error Err = ECtx.expandResponseFiles(NewArgv)) {
- errs() << toString(std::move(Err)) << '\n';
- return false;
- }
- return true;
- }
- bool cl::ExpandResponseFiles(StringSaver &Saver, TokenizerCallback Tokenizer,
- SmallVectorImpl<const char *> &Argv) {
- ExpansionContext ECtx(Saver.getAllocator(), Tokenizer);
- if (Error Err = ECtx.expandResponseFiles(Argv)) {
- errs() << toString(std::move(Err)) << '\n';
- return false;
- }
- return true;
- }
- ExpansionContext::ExpansionContext(BumpPtrAllocator &A, TokenizerCallback T)
- : Saver(A), Tokenizer(T), FS(vfs::getRealFileSystem().get()) {}
- bool ExpansionContext::findConfigFile(StringRef FileName,
- SmallVectorImpl<char> &FilePath) {
- SmallString<128> CfgFilePath;
- const auto FileExists = [this](SmallString<128> Path) -> bool {
- auto Status = FS->status(Path);
- return Status &&
- Status->getType() == llvm::sys::fs::file_type::regular_file;
- };
- // If file name contains directory separator, treat it as a path to
- // configuration file.
- if (llvm::sys::path::has_parent_path(FileName)) {
- CfgFilePath = FileName;
- if (llvm::sys::path::is_relative(FileName) && FS->makeAbsolute(CfgFilePath))
- return false;
- if (!FileExists(CfgFilePath))
- return false;
- FilePath.assign(CfgFilePath.begin(), CfgFilePath.end());
- return true;
- }
- // Look for the file in search directories.
- for (const StringRef &Dir : SearchDirs) {
- if (Dir.empty())
- continue;
- CfgFilePath.assign(Dir);
- llvm::sys::path::append(CfgFilePath, FileName);
- llvm::sys::path::native(CfgFilePath);
- if (FileExists(CfgFilePath)) {
- FilePath.assign(CfgFilePath.begin(), CfgFilePath.end());
- return true;
- }
- }
- return false;
- }
- Error ExpansionContext::readConfigFile(StringRef CfgFile,
- SmallVectorImpl<const char *> &Argv) {
- SmallString<128> AbsPath;
- if (sys::path::is_relative(CfgFile)) {
- AbsPath.assign(CfgFile);
- if (std::error_code EC = FS->makeAbsolute(AbsPath))
- return make_error<StringError>(
- EC, Twine("cannot get absolute path for " + CfgFile));
- CfgFile = AbsPath.str();
- }
- InConfigFile = true;
- RelativeNames = true;
- if (Error Err = expandResponseFile(CfgFile, Argv))
- return Err;
- return expandResponseFiles(Argv);
- }
- static void initCommonOptions();
- bool cl::ParseCommandLineOptions(int argc, const char *const *argv,
- StringRef Overview, raw_ostream *Errs,
- const char *EnvVar,
- bool LongOptionsUseDoubleDash) {
- initCommonOptions();
- SmallVector<const char *, 20> NewArgv;
- BumpPtrAllocator A;
- StringSaver Saver(A);
- NewArgv.push_back(argv[0]);
- // Parse options from environment variable.
- if (EnvVar) {
- if (std::optional<std::string> EnvValue =
- sys::Process::GetEnv(StringRef(EnvVar)))
- TokenizeGNUCommandLine(*EnvValue, Saver, NewArgv);
- }
- // Append options from command line.
- for (int I = 1; I < argc; ++I)
- NewArgv.push_back(argv[I]);
- int NewArgc = static_cast<int>(NewArgv.size());
- // Parse all options.
- return GlobalParser->ParseCommandLineOptions(NewArgc, &NewArgv[0], Overview,
- Errs, LongOptionsUseDoubleDash);
- }
- /// Reset all options at least once, so that we can parse different options.
- void CommandLineParser::ResetAllOptionOccurrences() {
- // Reset all option values to look like they have never been seen before.
- // Options might be reset twice (they can be reference in both OptionsMap
- // and one of the other members), but that does not harm.
- for (auto *SC : RegisteredSubCommands) {
- for (auto &O : SC->OptionsMap)
- O.second->reset();
- for (Option *O : SC->PositionalOpts)
- O->reset();
- for (Option *O : SC->SinkOpts)
- O->reset();
- if (SC->ConsumeAfterOpt)
- SC->ConsumeAfterOpt->reset();
- }
- }
- bool CommandLineParser::ParseCommandLineOptions(int argc,
- const char *const *argv,
- StringRef Overview,
- raw_ostream *Errs,
- bool LongOptionsUseDoubleDash) {
- assert(hasOptions() && "No options specified!");
- ProgramOverview = Overview;
- bool IgnoreErrors = Errs;
- if (!Errs)
- Errs = &errs();
- bool ErrorParsing = false;
- // Expand response files.
- SmallVector<const char *, 20> newArgv(argv, argv + argc);
- BumpPtrAllocator A;
- #ifdef _WIN32
- auto Tokenize = cl::TokenizeWindowsCommandLine;
- #else
- auto Tokenize = cl::TokenizeGNUCommandLine;
- #endif
- ExpansionContext ECtx(A, Tokenize);
- if (Error Err = ECtx.expandResponseFiles(newArgv)) {
- *Errs << toString(std::move(Err)) << '\n';
- return false;
- }
- argv = &newArgv[0];
- argc = static_cast<int>(newArgv.size());
- // Copy the program name into ProgName, making sure not to overflow it.
- ProgramName = std::string(sys::path::filename(StringRef(argv[0])));
- // Check out the positional arguments to collect information about them.
- unsigned NumPositionalRequired = 0;
- // Determine whether or not there are an unlimited number of positionals
- bool HasUnlimitedPositionals = false;
- int FirstArg = 1;
- SubCommand *ChosenSubCommand = &SubCommand::getTopLevel();
- if (argc >= 2 && argv[FirstArg][0] != '-') {
- // If the first argument specifies a valid subcommand, start processing
- // options from the second argument.
- ChosenSubCommand = LookupSubCommand(StringRef(argv[FirstArg]));
- if (ChosenSubCommand != &SubCommand::getTopLevel())
- FirstArg = 2;
- }
- GlobalParser->ActiveSubCommand = ChosenSubCommand;
- assert(ChosenSubCommand);
- auto &ConsumeAfterOpt = ChosenSubCommand->ConsumeAfterOpt;
- auto &PositionalOpts = ChosenSubCommand->PositionalOpts;
- auto &SinkOpts = ChosenSubCommand->SinkOpts;
- auto &OptionsMap = ChosenSubCommand->OptionsMap;
- for (auto *O: DefaultOptions) {
- addOption(O, true);
- }
- if (ConsumeAfterOpt) {
- assert(PositionalOpts.size() > 0 &&
- "Cannot specify cl::ConsumeAfter without a positional argument!");
- }
- if (!PositionalOpts.empty()) {
- // Calculate how many positional values are _required_.
- bool UnboundedFound = false;
- for (size_t i = 0, e = PositionalOpts.size(); i != e; ++i) {
- Option *Opt = PositionalOpts[i];
- if (RequiresValue(Opt))
- ++NumPositionalRequired;
- else if (ConsumeAfterOpt) {
- // ConsumeAfter cannot be combined with "optional" positional options
- // unless there is only one positional argument...
- if (PositionalOpts.size() > 1) {
- if (!IgnoreErrors)
- Opt->error("error - this positional option will never be matched, "
- "because it does not Require a value, and a "
- "cl::ConsumeAfter option is active!");
- ErrorParsing = true;
- }
- } else if (UnboundedFound && !Opt->hasArgStr()) {
- // This option does not "require" a value... Make sure this option is
- // not specified after an option that eats all extra arguments, or this
- // one will never get any!
- //
- if (!IgnoreErrors)
- Opt->error("error - option can never match, because "
- "another positional argument will match an "
- "unbounded number of values, and this option"
- " does not require a value!");
- *Errs << ProgramName << ": CommandLine Error: Option '" << Opt->ArgStr
- << "' is all messed up!\n";
- *Errs << PositionalOpts.size();
- ErrorParsing = true;
- }
- UnboundedFound |= EatsUnboundedNumberOfValues(Opt);
- }
- HasUnlimitedPositionals = UnboundedFound || ConsumeAfterOpt;
- }
- // PositionalVals - A vector of "positional" arguments we accumulate into
- // the process at the end.
- //
- SmallVector<std::pair<StringRef, unsigned>, 4> PositionalVals;
- // If the program has named positional arguments, and the name has been run
- // across, keep track of which positional argument was named. Otherwise put
- // the positional args into the PositionalVals list...
- Option *ActivePositionalArg = nullptr;
- // Loop over all of the arguments... processing them.
- bool DashDashFound = false; // Have we read '--'?
- for (int i = FirstArg; i < argc; ++i) {
- Option *Handler = nullptr;
- Option *NearestHandler = nullptr;
- std::string NearestHandlerString;
- StringRef Value;
- StringRef ArgName = "";
- bool HaveDoubleDash = false;
- // Check to see if this is a positional argument. This argument is
- // considered to be positional if it doesn't start with '-', if it is "-"
- // itself, or if we have seen "--" already.
- //
- if (argv[i][0] != '-' || argv[i][1] == 0 || DashDashFound) {
- // Positional argument!
- if (ActivePositionalArg) {
- ProvidePositionalOption(ActivePositionalArg, StringRef(argv[i]), i);
- continue; // We are done!
- }
- if (!PositionalOpts.empty()) {
- PositionalVals.push_back(std::make_pair(StringRef(argv[i]), i));
- // All of the positional arguments have been fulfulled, give the rest to
- // the consume after option... if it's specified...
- //
- if (PositionalVals.size() >= NumPositionalRequired && ConsumeAfterOpt) {
- for (++i; i < argc; ++i)
- PositionalVals.push_back(std::make_pair(StringRef(argv[i]), i));
- break; // Handle outside of the argument processing loop...
- }
- // Delay processing positional arguments until the end...
- continue;
- }
- } else if (argv[i][0] == '-' && argv[i][1] == '-' && argv[i][2] == 0 &&
- !DashDashFound) {
- DashDashFound = true; // This is the mythical "--"?
- continue; // Don't try to process it as an argument itself.
- } else if (ActivePositionalArg &&
- (ActivePositionalArg->getMiscFlags() & PositionalEatsArgs)) {
- // If there is a positional argument eating options, check to see if this
- // option is another positional argument. If so, treat it as an argument,
- // otherwise feed it to the eating positional.
- ArgName = StringRef(argv[i] + 1);
- // Eat second dash.
- if (!ArgName.empty() && ArgName[0] == '-') {
- HaveDoubleDash = true;
- ArgName = ArgName.substr(1);
- }
- Handler = LookupLongOption(*ChosenSubCommand, ArgName, Value,
- LongOptionsUseDoubleDash, HaveDoubleDash);
- if (!Handler || Handler->getFormattingFlag() != cl::Positional) {
- ProvidePositionalOption(ActivePositionalArg, StringRef(argv[i]), i);
- continue; // We are done!
- }
- } else { // We start with a '-', must be an argument.
- ArgName = StringRef(argv[i] + 1);
- // Eat second dash.
- if (!ArgName.empty() && ArgName[0] == '-') {
- HaveDoubleDash = true;
- ArgName = ArgName.substr(1);
- }
- Handler = LookupLongOption(*ChosenSubCommand, ArgName, Value,
- LongOptionsUseDoubleDash, HaveDoubleDash);
- // Check to see if this "option" is really a prefixed or grouped argument.
- if (!Handler && !(LongOptionsUseDoubleDash && HaveDoubleDash))
- Handler = HandlePrefixedOrGroupedOption(ArgName, Value, ErrorParsing,
- OptionsMap);
- // Otherwise, look for the closest available option to report to the user
- // in the upcoming error.
- if (!Handler && SinkOpts.empty())
- NearestHandler =
- LookupNearestOption(ArgName, OptionsMap, NearestHandlerString);
- }
- if (!Handler) {
- if (SinkOpts.empty()) {
- *Errs << ProgramName << ": Unknown command line argument '" << argv[i]
- << "'. Try: '" << argv[0] << " --help'\n";
- if (NearestHandler) {
- // If we know a near match, report it as well.
- *Errs << ProgramName << ": Did you mean '"
- << PrintArg(NearestHandlerString, 0) << "'?\n";
- }
- ErrorParsing = true;
- } else {
- for (Option *SinkOpt : SinkOpts)
- SinkOpt->addOccurrence(i, "", StringRef(argv[i]));
- }
- continue;
- }
- // If this is a named positional argument, just remember that it is the
- // active one...
- if (Handler->getFormattingFlag() == cl::Positional) {
- if ((Handler->getMiscFlags() & PositionalEatsArgs) && !Value.empty()) {
- Handler->error("This argument does not take a value.\n"
- "\tInstead, it consumes any positional arguments until "
- "the next recognized option.", *Errs);
- ErrorParsing = true;
- }
- ActivePositionalArg = Handler;
- }
- else
- ErrorParsing |= ProvideOption(Handler, ArgName, Value, argc, argv, i);
- }
- // Check and handle positional arguments now...
- if (NumPositionalRequired > PositionalVals.size()) {
- *Errs << ProgramName
- << ": Not enough positional command line arguments specified!\n"
- << "Must specify at least " << NumPositionalRequired
- << " positional argument" << (NumPositionalRequired > 1 ? "s" : "")
- << ": See: " << argv[0] << " --help\n";
- ErrorParsing = true;
- } else if (!HasUnlimitedPositionals &&
- PositionalVals.size() > PositionalOpts.size()) {
- *Errs << ProgramName << ": Too many positional arguments specified!\n"
- << "Can specify at most " << PositionalOpts.size()
- << " positional arguments: See: " << argv[0] << " --help\n";
- ErrorParsing = true;
- } else if (!ConsumeAfterOpt) {
- // Positional args have already been handled if ConsumeAfter is specified.
- unsigned ValNo = 0, NumVals = static_cast<unsigned>(PositionalVals.size());
- for (size_t i = 0, e = PositionalOpts.size(); i != e; ++i) {
- if (RequiresValue(PositionalOpts[i])) {
- ProvidePositionalOption(PositionalOpts[i], PositionalVals[ValNo].first,
- PositionalVals[ValNo].second);
- ValNo++;
- --NumPositionalRequired; // We fulfilled our duty...
- }
- // If we _can_ give this option more arguments, do so now, as long as we
- // do not give it values that others need. 'Done' controls whether the
- // option even _WANTS_ any more.
- //
- bool Done = PositionalOpts[i]->getNumOccurrencesFlag() == cl::Required;
- while (NumVals - ValNo > NumPositionalRequired && !Done) {
- switch (PositionalOpts[i]->getNumOccurrencesFlag()) {
- case cl::Optional:
- Done = true; // Optional arguments want _at most_ one value
- [[fallthrough]];
- case cl::ZeroOrMore: // Zero or more will take all they can get...
- case cl::OneOrMore: // One or more will take all they can get...
- ProvidePositionalOption(PositionalOpts[i],
- PositionalVals[ValNo].first,
- PositionalVals[ValNo].second);
- ValNo++;
- break;
- default:
- llvm_unreachable("Internal error, unexpected NumOccurrences flag in "
- "positional argument processing!");
- }
- }
- }
- } else {
- assert(ConsumeAfterOpt && NumPositionalRequired <= PositionalVals.size());
- unsigned ValNo = 0;
- for (size_t J = 0, E = PositionalOpts.size(); J != E; ++J)
- if (RequiresValue(PositionalOpts[J])) {
- ErrorParsing |= ProvidePositionalOption(PositionalOpts[J],
- PositionalVals[ValNo].first,
- PositionalVals[ValNo].second);
- ValNo++;
- }
- // Handle the case where there is just one positional option, and it's
- // optional. In this case, we want to give JUST THE FIRST option to the
- // positional option and keep the rest for the consume after. The above
- // loop would have assigned no values to positional options in this case.
- //
- if (PositionalOpts.size() == 1 && ValNo == 0 && !PositionalVals.empty()) {
- ErrorParsing |= ProvidePositionalOption(PositionalOpts[0],
- PositionalVals[ValNo].first,
- PositionalVals[ValNo].second);
- ValNo++;
- }
- // Handle over all of the rest of the arguments to the
- // cl::ConsumeAfter command line option...
- for (; ValNo != PositionalVals.size(); ++ValNo)
- ErrorParsing |=
- ProvidePositionalOption(ConsumeAfterOpt, PositionalVals[ValNo].first,
- PositionalVals[ValNo].second);
- }
- // Loop over args and make sure all required args are specified!
- for (const auto &Opt : OptionsMap) {
- switch (Opt.second->getNumOccurrencesFlag()) {
- case Required:
- case OneOrMore:
- if (Opt.second->getNumOccurrences() == 0) {
- Opt.second->error("must be specified at least once!");
- ErrorParsing = true;
- }
- [[fallthrough]];
- default:
- break;
- }
- }
- // Now that we know if -debug is specified, we can use it.
- // Note that if ReadResponseFiles == true, this must be done before the
- // memory allocated for the expanded command line is free()d below.
- LLVM_DEBUG(dbgs() << "Args: ";
- for (int i = 0; i < argc; ++i) dbgs() << argv[i] << ' ';
- dbgs() << '\n';);
- // Free all of the memory allocated to the map. Command line options may only
- // be processed once!
- MoreHelp.clear();
- // If we had an error processing our arguments, don't let the program execute
- if (ErrorParsing) {
- if (!IgnoreErrors)
- exit(1);
- return false;
- }
- return true;
- }
- //===----------------------------------------------------------------------===//
- // Option Base class implementation
- //
- bool Option::error(const Twine &Message, StringRef ArgName, raw_ostream &Errs) {
- if (!ArgName.data())
- ArgName = ArgStr;
- if (ArgName.empty())
- Errs << HelpStr; // Be nice for positional arguments
- else
- Errs << GlobalParser->ProgramName << ": for the " << PrintArg(ArgName, 0);
- Errs << " option: " << Message << "\n";
- return true;
- }
- bool Option::addOccurrence(unsigned pos, StringRef ArgName, StringRef Value,
- bool MultiArg) {
- if (!MultiArg)
- NumOccurrences++; // Increment the number of times we have been seen
- return handleOccurrence(pos, ArgName, Value);
- }
- // getValueStr - Get the value description string, using "DefaultMsg" if nothing
- // has been specified yet.
- //
- static StringRef getValueStr(const Option &O, StringRef DefaultMsg) {
- if (O.ValueStr.empty())
- return DefaultMsg;
- return O.ValueStr;
- }
- //===----------------------------------------------------------------------===//
- // cl::alias class implementation
- //
- // Return the width of the option tag for printing...
- size_t alias::getOptionWidth() const {
- return argPlusPrefixesSize(ArgStr);
- }
- void Option::printHelpStr(StringRef HelpStr, size_t Indent,
- size_t FirstLineIndentedBy) {
- assert(Indent >= FirstLineIndentedBy);
- std::pair<StringRef, StringRef> Split = HelpStr.split('\n');
- outs().indent(Indent - FirstLineIndentedBy)
- << ArgHelpPrefix << Split.first << "\n";
- while (!Split.second.empty()) {
- Split = Split.second.split('\n');
- outs().indent(Indent) << Split.first << "\n";
- }
- }
- void Option::printEnumValHelpStr(StringRef HelpStr, size_t BaseIndent,
- size_t FirstLineIndentedBy) {
- const StringRef ValHelpPrefix = " ";
- assert(BaseIndent >= FirstLineIndentedBy);
- std::pair<StringRef, StringRef> Split = HelpStr.split('\n');
- outs().indent(BaseIndent - FirstLineIndentedBy)
- << ArgHelpPrefix << ValHelpPrefix << Split.first << "\n";
- while (!Split.second.empty()) {
- Split = Split.second.split('\n');
- outs().indent(BaseIndent + ValHelpPrefix.size()) << Split.first << "\n";
- }
- }
- // Print out the option for the alias.
- void alias::printOptionInfo(size_t GlobalWidth) const {
- outs() << PrintArg(ArgStr);
- printHelpStr(HelpStr, GlobalWidth, argPlusPrefixesSize(ArgStr));
- }
- //===----------------------------------------------------------------------===//
- // Parser Implementation code...
- //
- // basic_parser implementation
- //
- // Return the width of the option tag for printing...
- size_t basic_parser_impl::getOptionWidth(const Option &O) const {
- size_t Len = argPlusPrefixesSize(O.ArgStr);
- auto ValName = getValueName();
- if (!ValName.empty()) {
- size_t FormattingLen = 3;
- if (O.getMiscFlags() & PositionalEatsArgs)
- FormattingLen = 6;
- Len += getValueStr(O, ValName).size() + FormattingLen;
- }
- return Len;
- }
- // printOptionInfo - Print out information about this option. The
- // to-be-maintained width is specified.
- //
- void basic_parser_impl::printOptionInfo(const Option &O,
- size_t GlobalWidth) const {
- outs() << PrintArg(O.ArgStr);
- auto ValName = getValueName();
- if (!ValName.empty()) {
- if (O.getMiscFlags() & PositionalEatsArgs) {
- outs() << " <" << getValueStr(O, ValName) << ">...";
- } else if (O.getValueExpectedFlag() == ValueOptional)
- outs() << "[=<" << getValueStr(O, ValName) << ">]";
- else {
- outs() << (O.ArgStr.size() == 1 ? " <" : "=<") << getValueStr(O, ValName)
- << '>';
- }
- }
- Option::printHelpStr(O.HelpStr, GlobalWidth, getOptionWidth(O));
- }
- void basic_parser_impl::printOptionName(const Option &O,
- size_t GlobalWidth) const {
- outs() << PrintArg(O.ArgStr);
- outs().indent(GlobalWidth - O.ArgStr.size());
- }
- // parser<bool> implementation
- //
- bool parser<bool>::parse(Option &O, StringRef ArgName, StringRef Arg,
- bool &Value) {
- if (Arg == "" || Arg == "true" || Arg == "TRUE" || Arg == "True" ||
- Arg == "1") {
- Value = true;
- return false;
- }
- if (Arg == "false" || Arg == "FALSE" || Arg == "False" || Arg == "0") {
- Value = false;
- return false;
- }
- return O.error("'" + Arg +
- "' is invalid value for boolean argument! Try 0 or 1");
- }
- // parser<boolOrDefault> implementation
- //
- bool parser<boolOrDefault>::parse(Option &O, StringRef ArgName, StringRef Arg,
- boolOrDefault &Value) {
- if (Arg == "" || Arg == "true" || Arg == "TRUE" || Arg == "True" ||
- Arg == "1") {
- Value = BOU_TRUE;
- return false;
- }
- if (Arg == "false" || Arg == "FALSE" || Arg == "False" || Arg == "0") {
- Value = BOU_FALSE;
- return false;
- }
- return O.error("'" + Arg +
- "' is invalid value for boolean argument! Try 0 or 1");
- }
- // parser<int> implementation
- //
- bool parser<int>::parse(Option &O, StringRef ArgName, StringRef Arg,
- int &Value) {
- if (Arg.getAsInteger(0, Value))
- return O.error("'" + Arg + "' value invalid for integer argument!");
- return false;
- }
- // parser<long> implementation
- //
- bool parser<long>::parse(Option &O, StringRef ArgName, StringRef Arg,
- long &Value) {
- if (Arg.getAsInteger(0, Value))
- return O.error("'" + Arg + "' value invalid for long argument!");
- return false;
- }
- // parser<long long> implementation
- //
- bool parser<long long>::parse(Option &O, StringRef ArgName, StringRef Arg,
- long long &Value) {
- if (Arg.getAsInteger(0, Value))
- return O.error("'" + Arg + "' value invalid for llong argument!");
- return false;
- }
- // parser<unsigned> implementation
- //
- bool parser<unsigned>::parse(Option &O, StringRef ArgName, StringRef Arg,
- unsigned &Value) {
- if (Arg.getAsInteger(0, Value))
- return O.error("'" + Arg + "' value invalid for uint argument!");
- return false;
- }
- // parser<unsigned long> implementation
- //
- bool parser<unsigned long>::parse(Option &O, StringRef ArgName, StringRef Arg,
- unsigned long &Value) {
- if (Arg.getAsInteger(0, Value))
- return O.error("'" + Arg + "' value invalid for ulong argument!");
- return false;
- }
- // parser<unsigned long long> implementation
- //
- bool parser<unsigned long long>::parse(Option &O, StringRef ArgName,
- StringRef Arg,
- unsigned long long &Value) {
- if (Arg.getAsInteger(0, Value))
- return O.error("'" + Arg + "' value invalid for ullong argument!");
- return false;
- }
- // parser<double>/parser<float> implementation
- //
- static bool parseDouble(Option &O, StringRef Arg, double &Value) {
- if (to_float(Arg, Value))
- return false;
- return O.error("'" + Arg + "' value invalid for floating point argument!");
- }
- bool parser<double>::parse(Option &O, StringRef ArgName, StringRef Arg,
- double &Val) {
- return parseDouble(O, Arg, Val);
- }
- bool parser<float>::parse(Option &O, StringRef ArgName, StringRef Arg,
- float &Val) {
- double dVal;
- if (parseDouble(O, Arg, dVal))
- return true;
- Val = (float)dVal;
- return false;
- }
- // generic_parser_base implementation
- //
- // findOption - Return the option number corresponding to the specified
- // argument string. If the option is not found, getNumOptions() is returned.
- //
- unsigned generic_parser_base::findOption(StringRef Name) {
- unsigned e = getNumOptions();
- for (unsigned i = 0; i != e; ++i) {
- if (getOption(i) == Name)
- return i;
- }
- return e;
- }
- static StringRef EqValue = "=<value>";
- static StringRef EmptyOption = "<empty>";
- static StringRef OptionPrefix = " =";
- static size_t getOptionPrefixesSize() {
- return OptionPrefix.size() + ArgHelpPrefix.size();
- }
- static bool shouldPrintOption(StringRef Name, StringRef Description,
- const Option &O) {
- return O.getValueExpectedFlag() != ValueOptional || !Name.empty() ||
- !Description.empty();
- }
- // Return the width of the option tag for printing...
- size_t generic_parser_base::getOptionWidth(const Option &O) const {
- if (O.hasArgStr()) {
- size_t Size =
- argPlusPrefixesSize(O.ArgStr) + EqValue.size();
- for (unsigned i = 0, e = getNumOptions(); i != e; ++i) {
- StringRef Name = getOption(i);
- if (!shouldPrintOption(Name, getDescription(i), O))
- continue;
- size_t NameSize = Name.empty() ? EmptyOption.size() : Name.size();
- Size = std::max(Size, NameSize + getOptionPrefixesSize());
- }
- return Size;
- } else {
- size_t BaseSize = 0;
- for (unsigned i = 0, e = getNumOptions(); i != e; ++i)
- BaseSize = std::max(BaseSize, getOption(i).size() + 8);
- return BaseSize;
- }
- }
- // printOptionInfo - Print out information about this option. The
- // to-be-maintained width is specified.
- //
- void generic_parser_base::printOptionInfo(const Option &O,
- size_t GlobalWidth) const {
- if (O.hasArgStr()) {
- // When the value is optional, first print a line just describing the
- // option without values.
- if (O.getValueExpectedFlag() == ValueOptional) {
- for (unsigned i = 0, e = getNumOptions(); i != e; ++i) {
- if (getOption(i).empty()) {
- outs() << PrintArg(O.ArgStr);
- Option::printHelpStr(O.HelpStr, GlobalWidth,
- argPlusPrefixesSize(O.ArgStr));
- break;
- }
- }
- }
- outs() << PrintArg(O.ArgStr) << EqValue;
- Option::printHelpStr(O.HelpStr, GlobalWidth,
- EqValue.size() +
- argPlusPrefixesSize(O.ArgStr));
- for (unsigned i = 0, e = getNumOptions(); i != e; ++i) {
- StringRef OptionName = getOption(i);
- StringRef Description = getDescription(i);
- if (!shouldPrintOption(OptionName, Description, O))
- continue;
- size_t FirstLineIndent = OptionName.size() + getOptionPrefixesSize();
- outs() << OptionPrefix << OptionName;
- if (OptionName.empty()) {
- outs() << EmptyOption;
- assert(FirstLineIndent >= EmptyOption.size());
- FirstLineIndent += EmptyOption.size();
- }
- if (!Description.empty())
- Option::printEnumValHelpStr(Description, GlobalWidth, FirstLineIndent);
- else
- outs() << '\n';
- }
- } else {
- if (!O.HelpStr.empty())
- outs() << " " << O.HelpStr << '\n';
- for (unsigned i = 0, e = getNumOptions(); i != e; ++i) {
- StringRef Option = getOption(i);
- outs() << " " << PrintArg(Option);
- Option::printHelpStr(getDescription(i), GlobalWidth, Option.size() + 8);
- }
- }
- }
- static const size_t MaxOptWidth = 8; // arbitrary spacing for printOptionDiff
- // printGenericOptionDiff - Print the value of this option and it's default.
- //
- // "Generic" options have each value mapped to a name.
- void generic_parser_base::printGenericOptionDiff(
- const Option &O, const GenericOptionValue &Value,
- const GenericOptionValue &Default, size_t GlobalWidth) const {
- outs() << " " << PrintArg(O.ArgStr);
- outs().indent(GlobalWidth - O.ArgStr.size());
- unsigned NumOpts = getNumOptions();
- for (unsigned i = 0; i != NumOpts; ++i) {
- if (Value.compare(getOptionValue(i)))
- continue;
- outs() << "= " << getOption(i);
- size_t L = getOption(i).size();
- size_t NumSpaces = MaxOptWidth > L ? MaxOptWidth - L : 0;
- outs().indent(NumSpaces) << " (default: ";
- for (unsigned j = 0; j != NumOpts; ++j) {
- if (Default.compare(getOptionValue(j)))
- continue;
- outs() << getOption(j);
- break;
- }
- outs() << ")\n";
- return;
- }
- outs() << "= *unknown option value*\n";
- }
- // printOptionDiff - Specializations for printing basic value types.
- //
- #define PRINT_OPT_DIFF(T) \
- void parser<T>::printOptionDiff(const Option &O, T V, OptionValue<T> D, \
- size_t GlobalWidth) const { \
- printOptionName(O, GlobalWidth); \
- std::string Str; \
- { \
- raw_string_ostream SS(Str); \
- SS << V; \
- } \
- outs() << "= " << Str; \
- size_t NumSpaces = \
- MaxOptWidth > Str.size() ? MaxOptWidth - Str.size() : 0; \
- outs().indent(NumSpaces) << " (default: "; \
- if (D.hasValue()) \
- outs() << D.getValue(); \
- else \
- outs() << "*no default*"; \
- outs() << ")\n"; \
- }
- PRINT_OPT_DIFF(bool)
- PRINT_OPT_DIFF(boolOrDefault)
- PRINT_OPT_DIFF(int)
- PRINT_OPT_DIFF(long)
- PRINT_OPT_DIFF(long long)
- PRINT_OPT_DIFF(unsigned)
- PRINT_OPT_DIFF(unsigned long)
- PRINT_OPT_DIFF(unsigned long long)
- PRINT_OPT_DIFF(double)
- PRINT_OPT_DIFF(float)
- PRINT_OPT_DIFF(char)
- void parser<std::string>::printOptionDiff(const Option &O, StringRef V,
- const OptionValue<std::string> &D,
- size_t GlobalWidth) const {
- printOptionName(O, GlobalWidth);
- outs() << "= " << V;
- size_t NumSpaces = MaxOptWidth > V.size() ? MaxOptWidth - V.size() : 0;
- outs().indent(NumSpaces) << " (default: ";
- if (D.hasValue())
- outs() << D.getValue();
- else
- outs() << "*no default*";
- outs() << ")\n";
- }
- // Print a placeholder for options that don't yet support printOptionDiff().
- void basic_parser_impl::printOptionNoValue(const Option &O,
- size_t GlobalWidth) const {
- printOptionName(O, GlobalWidth);
- outs() << "= *cannot print option value*\n";
- }
- //===----------------------------------------------------------------------===//
- // -help and -help-hidden option implementation
- //
- static int OptNameCompare(const std::pair<const char *, Option *> *LHS,
- const std::pair<const char *, Option *> *RHS) {
- return strcmp(LHS->first, RHS->first);
- }
- static int SubNameCompare(const std::pair<const char *, SubCommand *> *LHS,
- const std::pair<const char *, SubCommand *> *RHS) {
- return strcmp(LHS->first, RHS->first);
- }
- // Copy Options into a vector so we can sort them as we like.
- static void sortOpts(StringMap<Option *> &OptMap,
- SmallVectorImpl<std::pair<const char *, Option *>> &Opts,
- bool ShowHidden) {
- SmallPtrSet<Option *, 32> OptionSet; // Duplicate option detection.
- for (StringMap<Option *>::iterator I = OptMap.begin(), E = OptMap.end();
- I != E; ++I) {
- // Ignore really-hidden options.
- if (I->second->getOptionHiddenFlag() == ReallyHidden)
- continue;
- // Unless showhidden is set, ignore hidden flags.
- if (I->second->getOptionHiddenFlag() == Hidden && !ShowHidden)
- continue;
- // If we've already seen this option, don't add it to the list again.
- if (!OptionSet.insert(I->second).second)
- continue;
- Opts.push_back(
- std::pair<const char *, Option *>(I->getKey().data(), I->second));
- }
- // Sort the options list alphabetically.
- array_pod_sort(Opts.begin(), Opts.end(), OptNameCompare);
- }
- static void
- sortSubCommands(const SmallPtrSetImpl<SubCommand *> &SubMap,
- SmallVectorImpl<std::pair<const char *, SubCommand *>> &Subs) {
- for (auto *S : SubMap) {
- if (S->getName().empty())
- continue;
- Subs.push_back(std::make_pair(S->getName().data(), S));
- }
- array_pod_sort(Subs.begin(), Subs.end(), SubNameCompare);
- }
- namespace {
- class HelpPrinter {
- protected:
- const bool ShowHidden;
- typedef SmallVector<std::pair<const char *, Option *>, 128>
- StrOptionPairVector;
- typedef SmallVector<std::pair<const char *, SubCommand *>, 128>
- StrSubCommandPairVector;
- // Print the options. Opts is assumed to be alphabetically sorted.
- virtual void printOptions(StrOptionPairVector &Opts, size_t MaxArgLen) {
- for (size_t i = 0, e = Opts.size(); i != e; ++i)
- Opts[i].second->printOptionInfo(MaxArgLen);
- }
- void printSubCommands(StrSubCommandPairVector &Subs, size_t MaxSubLen) {
- for (const auto &S : Subs) {
- outs() << " " << S.first;
- if (!S.second->getDescription().empty()) {
- outs().indent(MaxSubLen - strlen(S.first));
- outs() << " - " << S.second->getDescription();
- }
- outs() << "\n";
- }
- }
- public:
- explicit HelpPrinter(bool showHidden) : ShowHidden(showHidden) {}
- virtual ~HelpPrinter() = default;
- // Invoke the printer.
- void operator=(bool Value) {
- if (!Value)
- return;
- printHelp();
- // Halt the program since help information was printed
- exit(0);
- }
- void printHelp() {
- SubCommand *Sub = GlobalParser->getActiveSubCommand();
- auto &OptionsMap = Sub->OptionsMap;
- auto &PositionalOpts = Sub->PositionalOpts;
- auto &ConsumeAfterOpt = Sub->ConsumeAfterOpt;
- StrOptionPairVector Opts;
- sortOpts(OptionsMap, Opts, ShowHidden);
- StrSubCommandPairVector Subs;
- sortSubCommands(GlobalParser->RegisteredSubCommands, Subs);
- if (!GlobalParser->ProgramOverview.empty())
- outs() << "OVERVIEW: " << GlobalParser->ProgramOverview << "\n";
- if (Sub == &SubCommand::getTopLevel()) {
- outs() << "USAGE: " << GlobalParser->ProgramName;
- if (Subs.size() > 2)
- outs() << " [subcommand]";
- outs() << " [options]";
- } else {
- if (!Sub->getDescription().empty()) {
- outs() << "SUBCOMMAND '" << Sub->getName()
- << "': " << Sub->getDescription() << "\n\n";
- }
- outs() << "USAGE: " << GlobalParser->ProgramName << " " << Sub->getName()
- << " [options]";
- }
- for (auto *Opt : PositionalOpts) {
- if (Opt->hasArgStr())
- outs() << " --" << Opt->ArgStr;
- outs() << " " << Opt->HelpStr;
- }
- // Print the consume after option info if it exists...
- if (ConsumeAfterOpt)
- outs() << " " << ConsumeAfterOpt->HelpStr;
- if (Sub == &SubCommand::getTopLevel() && !Subs.empty()) {
- // Compute the maximum subcommand length...
- size_t MaxSubLen = 0;
- for (size_t i = 0, e = Subs.size(); i != e; ++i)
- MaxSubLen = std::max(MaxSubLen, strlen(Subs[i].first));
- outs() << "\n\n";
- outs() << "SUBCOMMANDS:\n\n";
- printSubCommands(Subs, MaxSubLen);
- outs() << "\n";
- outs() << " Type \"" << GlobalParser->ProgramName
- << " <subcommand> --help\" to get more help on a specific "
- "subcommand";
- }
- outs() << "\n\n";
- // Compute the maximum argument length...
- size_t MaxArgLen = 0;
- for (size_t i = 0, e = Opts.size(); i != e; ++i)
- MaxArgLen = std::max(MaxArgLen, Opts[i].second->getOptionWidth());
- outs() << "OPTIONS:\n";
- printOptions(Opts, MaxArgLen);
- // Print any extra help the user has declared.
- for (const auto &I : GlobalParser->MoreHelp)
- outs() << I;
- GlobalParser->MoreHelp.clear();
- }
- };
- class CategorizedHelpPrinter : public HelpPrinter {
- public:
- explicit CategorizedHelpPrinter(bool showHidden) : HelpPrinter(showHidden) {}
- // Helper function for printOptions().
- // It shall return a negative value if A's name should be lexicographically
- // ordered before B's name. It returns a value greater than zero if B's name
- // should be ordered before A's name, and it returns 0 otherwise.
- static int OptionCategoryCompare(OptionCategory *const *A,
- OptionCategory *const *B) {
- return (*A)->getName().compare((*B)->getName());
- }
- // Make sure we inherit our base class's operator=()
- using HelpPrinter::operator=;
- protected:
- void printOptions(StrOptionPairVector &Opts, size_t MaxArgLen) override {
- std::vector<OptionCategory *> SortedCategories;
- DenseMap<OptionCategory *, std::vector<Option *>> CategorizedOptions;
- // Collect registered option categories into vector in preparation for
- // sorting.
- for (OptionCategory *Category : GlobalParser->RegisteredOptionCategories)
- SortedCategories.push_back(Category);
- // Sort the different option categories alphabetically.
- assert(SortedCategories.size() > 0 && "No option categories registered!");
- array_pod_sort(SortedCategories.begin(), SortedCategories.end(),
- OptionCategoryCompare);
- // Walk through pre-sorted options and assign into categories.
- // Because the options are already alphabetically sorted the
- // options within categories will also be alphabetically sorted.
- for (size_t I = 0, E = Opts.size(); I != E; ++I) {
- Option *Opt = Opts[I].second;
- for (auto &Cat : Opt->Categories) {
- assert(llvm::is_contained(SortedCategories, Cat) &&
- "Option has an unregistered category");
- CategorizedOptions[Cat].push_back(Opt);
- }
- }
- // Now do printing.
- for (OptionCategory *Category : SortedCategories) {
- // Hide empty categories for --help, but show for --help-hidden.
- const auto &CategoryOptions = CategorizedOptions[Category];
- bool IsEmptyCategory = CategoryOptions.empty();
- if (!ShowHidden && IsEmptyCategory)
- continue;
- // Print category information.
- outs() << "\n";
- outs() << Category->getName() << ":\n";
- // Check if description is set.
- if (!Category->getDescription().empty())
- outs() << Category->getDescription() << "\n\n";
- else
- outs() << "\n";
- // When using --help-hidden explicitly state if the category has no
- // options associated with it.
- if (IsEmptyCategory) {
- outs() << " This option category has no options.\n";
- continue;
- }
- // Loop over the options in the category and print.
- for (const Option *Opt : CategoryOptions)
- Opt->printOptionInfo(MaxArgLen);
- }
- }
- };
- // This wraps the Uncategorizing and Categorizing printers and decides
- // at run time which should be invoked.
- class HelpPrinterWrapper {
- private:
- HelpPrinter &UncategorizedPrinter;
- CategorizedHelpPrinter &CategorizedPrinter;
- public:
- explicit HelpPrinterWrapper(HelpPrinter &UncategorizedPrinter,
- CategorizedHelpPrinter &CategorizedPrinter)
- : UncategorizedPrinter(UncategorizedPrinter),
- CategorizedPrinter(CategorizedPrinter) {}
- // Invoke the printer.
- void operator=(bool Value);
- };
- } // End anonymous namespace
- #if defined(__GNUC__)
- // GCC and GCC-compatible compilers define __OPTIMIZE__ when optimizations are
- // enabled.
- # if defined(__OPTIMIZE__)
- # define LLVM_IS_DEBUG_BUILD 0
- # else
- # define LLVM_IS_DEBUG_BUILD 1
- # endif
- #elif defined(_MSC_VER)
- // MSVC doesn't have a predefined macro indicating if optimizations are enabled.
- // Use _DEBUG instead. This macro actually corresponds to the choice between
- // debug and release CRTs, but it is a reasonable proxy.
- # if defined(_DEBUG)
- # define LLVM_IS_DEBUG_BUILD 1
- # else
- # define LLVM_IS_DEBUG_BUILD 0
- # endif
- #else
- // Otherwise, for an unknown compiler, assume this is an optimized build.
- # define LLVM_IS_DEBUG_BUILD 0
- #endif
- namespace {
- class VersionPrinter {
- public:
- void print(std::vector<VersionPrinterTy> ExtraPrinters = {}) {
- raw_ostream &OS = outs();
- #ifdef PACKAGE_VENDOR
- OS << PACKAGE_VENDOR << " ";
- #else
- OS << "LLVM (http://llvm.org/):\n ";
- #endif
- OS << PACKAGE_NAME << " version " << PACKAGE_VERSION << "\n ";
- #if LLVM_IS_DEBUG_BUILD
- OS << "DEBUG build";
- #else
- OS << "Optimized build";
- #endif
- #ifndef NDEBUG
- OS << " with assertions";
- #endif
- OS << ".\n";
- // Iterate over any registered extra printers and call them to add further
- // information.
- if (!ExtraPrinters.empty()) {
- for (const auto &I : ExtraPrinters)
- I(outs());
- }
- }
- void operator=(bool OptionWasSpecified);
- };
- struct CommandLineCommonOptions {
- // Declare the four HelpPrinter instances that are used to print out help, or
- // help-hidden as an uncategorized list or in categories.
- HelpPrinter UncategorizedNormalPrinter{false};
- HelpPrinter UncategorizedHiddenPrinter{true};
- CategorizedHelpPrinter CategorizedNormalPrinter{false};
- CategorizedHelpPrinter CategorizedHiddenPrinter{true};
- // Declare HelpPrinter wrappers that will decide whether or not to invoke
- // a categorizing help printer
- HelpPrinterWrapper WrappedNormalPrinter{UncategorizedNormalPrinter,
- CategorizedNormalPrinter};
- HelpPrinterWrapper WrappedHiddenPrinter{UncategorizedHiddenPrinter,
- CategorizedHiddenPrinter};
- // Define a category for generic options that all tools should have.
- cl::OptionCategory GenericCategory{"Generic Options"};
- // Define uncategorized help printers.
- // --help-list is hidden by default because if Option categories are being
- // used then --help behaves the same as --help-list.
- cl::opt<HelpPrinter, true, parser<bool>> HLOp{
- "help-list",
- cl::desc(
- "Display list of available options (--help-list-hidden for more)"),
- cl::location(UncategorizedNormalPrinter),
- cl::Hidden,
- cl::ValueDisallowed,
- cl::cat(GenericCategory),
- cl::sub(SubCommand::getAll())};
- cl::opt<HelpPrinter, true, parser<bool>> HLHOp{
- "help-list-hidden",
- cl::desc("Display list of all available options"),
- cl::location(UncategorizedHiddenPrinter),
- cl::Hidden,
- cl::ValueDisallowed,
- cl::cat(GenericCategory),
- cl::sub(SubCommand::getAll())};
- // Define uncategorized/categorized help printers. These printers change their
- // behaviour at runtime depending on whether one or more Option categories
- // have been declared.
- cl::opt<HelpPrinterWrapper, true, parser<bool>> HOp{
- "help",
- cl::desc("Display available options (--help-hidden for more)"),
- cl::location(WrappedNormalPrinter),
- cl::ValueDisallowed,
- cl::cat(GenericCategory),
- cl::sub(SubCommand::getAll())};
- cl::alias HOpA{"h", cl::desc("Alias for --help"), cl::aliasopt(HOp),
- cl::DefaultOption};
- cl::opt<HelpPrinterWrapper, true, parser<bool>> HHOp{
- "help-hidden",
- cl::desc("Display all available options"),
- cl::location(WrappedHiddenPrinter),
- cl::Hidden,
- cl::ValueDisallowed,
- cl::cat(GenericCategory),
- cl::sub(SubCommand::getAll())};
- cl::opt<bool> PrintOptions{
- "print-options",
- cl::desc("Print non-default options after command line parsing"),
- cl::Hidden,
- cl::init(false),
- cl::cat(GenericCategory),
- cl::sub(SubCommand::getAll())};
- cl::opt<bool> PrintAllOptions{
- "print-all-options",
- cl::desc("Print all option values after command line parsing"),
- cl::Hidden,
- cl::init(false),
- cl::cat(GenericCategory),
- cl::sub(SubCommand::getAll())};
- VersionPrinterTy OverrideVersionPrinter = nullptr;
- std::vector<VersionPrinterTy> ExtraVersionPrinters;
- // Define the --version option that prints out the LLVM version for the tool
- VersionPrinter VersionPrinterInstance;
- cl::opt<VersionPrinter, true, parser<bool>> VersOp{
- "version", cl::desc("Display the version of this program"),
- cl::location(VersionPrinterInstance), cl::ValueDisallowed,
- cl::cat(GenericCategory)};
- };
- } // End anonymous namespace
- // Lazy-initialized global instance of options controlling the command-line
- // parser and general handling.
- static ManagedStatic<CommandLineCommonOptions> CommonOptions;
- static void initCommonOptions() {
- *CommonOptions;
- initDebugCounterOptions();
- initGraphWriterOptions();
- initSignalsOptions();
- initStatisticOptions();
- initTimerOptions();
- initTypeSizeOptions();
- initWithColorOptions();
- initDebugOptions();
- initRandomSeedOptions();
- }
- OptionCategory &cl::getGeneralCategory() {
- // Initialise the general option category.
- static OptionCategory GeneralCategory{"General options"};
- return GeneralCategory;
- }
- void VersionPrinter::operator=(bool OptionWasSpecified) {
- if (!OptionWasSpecified)
- return;
- if (CommonOptions->OverrideVersionPrinter != nullptr) {
- CommonOptions->OverrideVersionPrinter(outs());
- exit(0);
- }
- print(CommonOptions->ExtraVersionPrinters);
- exit(0);
- }
- void HelpPrinterWrapper::operator=(bool Value) {
- if (!Value)
- return;
- // Decide which printer to invoke. If more than one option category is
- // registered then it is useful to show the categorized help instead of
- // uncategorized help.
- if (GlobalParser->RegisteredOptionCategories.size() > 1) {
- // unhide --help-list option so user can have uncategorized output if they
- // want it.
- CommonOptions->HLOp.setHiddenFlag(NotHidden);
- CategorizedPrinter = true; // Invoke categorized printer
- } else
- UncategorizedPrinter = true; // Invoke uncategorized printer
- }
- // Print the value of each option.
- void cl::PrintOptionValues() { GlobalParser->printOptionValues(); }
- void CommandLineParser::printOptionValues() {
- if (!CommonOptions->PrintOptions && !CommonOptions->PrintAllOptions)
- return;
- SmallVector<std::pair<const char *, Option *>, 128> Opts;
- sortOpts(ActiveSubCommand->OptionsMap, Opts, /*ShowHidden*/ true);
- // Compute the maximum argument length...
- size_t MaxArgLen = 0;
- for (size_t i = 0, e = Opts.size(); i != e; ++i)
- MaxArgLen = std::max(MaxArgLen, Opts[i].second->getOptionWidth());
- for (size_t i = 0, e = Opts.size(); i != e; ++i)
- Opts[i].second->printOptionValue(MaxArgLen, CommonOptions->PrintAllOptions);
- }
- // Utility function for printing the help message.
- void cl::PrintHelpMessage(bool Hidden, bool Categorized) {
- if (!Hidden && !Categorized)
- CommonOptions->UncategorizedNormalPrinter.printHelp();
- else if (!Hidden && Categorized)
- CommonOptions->CategorizedNormalPrinter.printHelp();
- else if (Hidden && !Categorized)
- CommonOptions->UncategorizedHiddenPrinter.printHelp();
- else
- CommonOptions->CategorizedHiddenPrinter.printHelp();
- }
- /// Utility function for printing version number.
- void cl::PrintVersionMessage() {
- CommonOptions->VersionPrinterInstance.print(CommonOptions->ExtraVersionPrinters);
- }
- void cl::SetVersionPrinter(VersionPrinterTy func) {
- CommonOptions->OverrideVersionPrinter = func;
- }
- void cl::AddExtraVersionPrinter(VersionPrinterTy func) {
- CommonOptions->ExtraVersionPrinters.push_back(func);
- }
- StringMap<Option *> &cl::getRegisteredOptions(SubCommand &Sub) {
- initCommonOptions();
- auto &Subs = GlobalParser->RegisteredSubCommands;
- (void)Subs;
- assert(is_contained(Subs, &Sub));
- return Sub.OptionsMap;
- }
- iterator_range<typename SmallPtrSet<SubCommand *, 4>::iterator>
- cl::getRegisteredSubcommands() {
- return GlobalParser->getRegisteredSubcommands();
- }
- void cl::HideUnrelatedOptions(cl::OptionCategory &Category, SubCommand &Sub) {
- initCommonOptions();
- for (auto &I : Sub.OptionsMap) {
- bool Unrelated = true;
- for (auto &Cat : I.second->Categories) {
- if (Cat == &Category || Cat == &CommonOptions->GenericCategory)
- Unrelated = false;
- }
- if (Unrelated)
- I.second->setHiddenFlag(cl::ReallyHidden);
- }
- }
- void cl::HideUnrelatedOptions(ArrayRef<const cl::OptionCategory *> Categories,
- SubCommand &Sub) {
- initCommonOptions();
- for (auto &I : Sub.OptionsMap) {
- bool Unrelated = true;
- for (auto &Cat : I.second->Categories) {
- if (is_contained(Categories, Cat) ||
- Cat == &CommonOptions->GenericCategory)
- Unrelated = false;
- }
- if (Unrelated)
- I.second->setHiddenFlag(cl::ReallyHidden);
- }
- }
- void cl::ResetCommandLineParser() { GlobalParser->reset(); }
- void cl::ResetAllOptionOccurrences() {
- GlobalParser->ResetAllOptionOccurrences();
- }
- void LLVMParseCommandLineOptions(int argc, const char *const *argv,
- const char *Overview) {
- llvm::cl::ParseCommandLineOptions(argc, argv, StringRef(Overview),
- &llvm::nulls());
- }
|