1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719 |
- //===-- 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/Optional.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/Triple.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/Host.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 <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() : ActiveSubCommand(nullptr) {
- registerSubCommand(&*TopLevelSubCommand);
- registerSubCommand(&*AllSubCommands);
- }
- 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 == &*AllSubCommands) {
- for (auto *Sub : RegisteredSubCommands) {
- if (SC == Sub)
- continue;
- addLiteralOption(Opt, Sub, Name);
- }
- }
- }
- void addLiteralOption(Option &Opt, StringRef Name) {
- if (Opt.Subs.empty())
- addLiteralOption(Opt, &*TopLevelSubCommand, 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 == &*AllSubCommands) {
- 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, &*TopLevelSubCommand);
- } 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, &*TopLevelSubCommand);
- 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, &*TopLevelSubCommand);
- 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 != &*AllSubCommands) {
- for (auto &E : AllSubCommands->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();
- TopLevelSubCommand->reset();
- AllSubCommands->reset();
- registerSubCommand(&*TopLevelSubCommand);
- registerSubCommand(&*AllSubCommands);
- DefaultOptions.clear();
- }
- private:
- SubCommand *ActiveSubCommand;
- 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;
- 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 != &*AllSubCommands);
- 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 &*TopLevelSubCommand;
- for (auto *S : RegisteredSubCommands) {
- if (S == &*AllSubCommands)
- continue;
- if (S->getName().empty())
- continue;
- if (StringRef(S->getName()) == StringRef(Name))
- return S;
- }
- return &*TopLevelSubCommand;
- }
- /// 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.
- static bool isWindowsSpecialChar(char C) {
- return isWhitespaceOrNull(C) || 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) {
- SmallString<128> Token;
- // 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;
- 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();
- } else if (Src[I] == '\"') {
- Token += NormalChars;
- State = QUOTED;
- } else if (Src[I] == '\\') {
- 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')
- MarkEOL();
- State = INIT;
- } else if (Src[I] == '\"') {
- State = QUOTED;
- } else if (Src[I] == '\\') {
- 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] == '\\') {
- I = parseBackslash(Src, I, Token);
- } else {
- Token.push_back(Src[I]);
- }
- break;
- }
- }
- if (State == UNQUOTED)
- 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);
- }
- 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);
- }
- 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.
- static llvm::Error ExpandResponseFile(StringRef FName, StringSaver &Saver,
- TokenizerCallback Tokenizer,
- SmallVectorImpl<const char *> &NewArgv,
- bool MarkEOLs, bool RelativeNames,
- bool ExpandBasePath,
- llvm::vfs::FileSystem &FS) {
- assert(sys::path::is_absolute(FName));
- llvm::ErrorOr<std::unique_ptr<MemoryBuffer>> MemBufOrErr =
- FS.getBufferForFile(FName);
- if (!MemBufOrErr)
- return llvm::errorCodeToError(MemBufOrErr.getError());
- 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);
- if (!RelativeNames)
- return Error::success();
- llvm::StringRef BasePath = llvm::sys::path::parent_path(FName);
- // If names of nested response files should be resolved relative to including
- // file, replace the included response file names with their full paths
- // obtained by required resolution.
- for (auto &Arg : NewArgv) {
- if (!Arg)
- continue;
- // Substitute <CFGDIR> with the file's base path.
- if (ExpandBasePath)
- ExpandBasePaths(BasePath, Saver, Arg);
- // Skip non-rsp file arguments.
- if (Arg[0] != '@')
- continue;
- StringRef FileName(Arg + 1);
- // Skip if non-relative.
- if (!llvm::sys::path::is_relative(FileName))
- continue;
- SmallString<128> ResponseFile;
- ResponseFile.push_back('@');
- 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.
- bool cl::ExpandResponseFiles(StringSaver &Saver, TokenizerCallback Tokenizer,
- SmallVectorImpl<const char *> &Argv, bool MarkEOLs,
- bool RelativeNames, bool ExpandBasePath,
- llvm::Optional<llvm::StringRef> CurrentDir,
- llvm::vfs::FileSystem &FS) {
- bool AllExpanded = true;
- 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)
- llvm::sys::fs::current_path(CurrDir);
- else
- CurrDir = *CurrentDir;
- llvm::sys::path::append(CurrDir, FName);
- FName = CurrDir.c_str();
- }
- auto IsEquivalent = [FName, &FS](const ResponseFileRecord &RFile) {
- llvm::ErrorOr<llvm::vfs::Status> LHS = FS.status(FName);
- if (!LHS) {
- // TODO: The error should be propagated up the stack.
- llvm::consumeError(llvm::errorCodeToError(LHS.getError()));
- return false;
- }
- llvm::ErrorOr<llvm::vfs::Status> RHS = FS.status(RFile.File);
- if (!RHS) {
- // TODO: The error should be propagated up the stack.
- llvm::consumeError(llvm::errorCodeToError(RHS.getError()));
- return false;
- }
- return LHS->equivalent(*RHS);
- };
- // Check for recursive response files.
- if (any_of(drop_begin(FileStack), IsEquivalent)) {
- // This file is recursive, so we leave it in the argument stream and
- // move on.
- AllExpanded = false;
- ++I;
- continue;
- }
- // 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 (llvm::Error Err =
- ExpandResponseFile(FName, Saver, Tokenizer, ExpandedArgv, MarkEOLs,
- RelativeNames, ExpandBasePath, FS)) {
- // We couldn't read this file, so we leave it in the argument stream and
- // move on.
- // TODO: The error should be propagated up the stack.
- llvm::consumeError(std::move(Err));
- AllExpanded = false;
- ++I;
- continue;
- }
- 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 AllExpanded;
- }
- bool cl::ExpandResponseFiles(StringSaver &Saver, TokenizerCallback Tokenizer,
- SmallVectorImpl<const char *> &Argv, bool MarkEOLs,
- bool RelativeNames, bool ExpandBasePath,
- llvm::Optional<StringRef> CurrentDir) {
- return ExpandResponseFiles(Saver, std::move(Tokenizer), Argv, MarkEOLs,
- RelativeNames, ExpandBasePath,
- std::move(CurrentDir), *vfs::getRealFileSystem());
- }
- bool cl::expandResponseFiles(int Argc, const char *const *Argv,
- const char *EnvVar, StringSaver &Saver,
- SmallVectorImpl<const char *> &NewArgv) {
- auto Tokenize = Triple(sys::getProcessTriple()).isOSWindows()
- ? cl::TokenizeWindowsCommandLine
- : cl::TokenizeGNUCommandLine;
- // The environment variable specifies initial options.
- if (EnvVar)
- if (llvm::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);
- return ExpandResponseFiles(Saver, Tokenize, NewArgv);
- }
- bool cl::readConfigFile(StringRef CfgFile, StringSaver &Saver,
- SmallVectorImpl<const char *> &Argv) {
- SmallString<128> AbsPath;
- if (sys::path::is_relative(CfgFile)) {
- llvm::sys::fs::current_path(AbsPath);
- llvm::sys::path::append(AbsPath, CfgFile);
- CfgFile = AbsPath.str();
- }
- if (llvm::Error Err = ExpandResponseFile(
- CfgFile, Saver, cl::tokenizeConfigFile, Argv,
- /*MarkEOLs=*/false, /*RelativeNames=*/true, /*ExpandBasePath=*/true,
- *llvm::vfs::getRealFileSystem())) {
- // TODO: The error should be propagated up the stack.
- llvm::consumeError(std::move(Err));
- return false;
- }
- return ExpandResponseFiles(Saver, cl::tokenizeConfigFile, Argv,
- /*MarkEOLs=*/false, /*RelativeNames=*/true,
- /*ExpandBasePath=*/true, llvm::None);
- }
- 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 (llvm::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!");
- // Expand response files.
- SmallVector<const char *, 20> newArgv(argv, argv + argc);
- BumpPtrAllocator A;
- StringSaver Saver(A);
- ExpandResponseFiles(Saver,
- Triple(sys::getProcessTriple()).isOSWindows() ?
- cl::TokenizeWindowsCommandLine : cl::TokenizeGNUCommandLine,
- newArgv);
- 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])));
- ProgramOverview = Overview;
- bool IgnoreErrors = Errs;
- if (!Errs)
- Errs = &errs();
- bool ErrorParsing = false;
- // 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 = &*TopLevelSubCommand;
- 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 != &*TopLevelSubCommand)
- 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
- LLVM_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;
- }
- LLVM_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
- switch (getNumOccurrencesFlag()) {
- case Optional:
- if (NumOccurrences > 1)
- return error("may only occur zero or one times!", ArgName);
- break;
- case Required:
- if (NumOccurrences > 1)
- return error("must occur exactly one time!", ArgName);
- LLVM_FALLTHROUGH;
- case OneOrMore:
- case ZeroOrMore:
- case ConsumeAfter:
- break;
- }
- 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() << "=<" << 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() {}
- // 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 == &*TopLevelSubCommand) {
- 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 == &*TopLevelSubCommand && !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(find(SortedCategories, Cat) != SortedCategories.end() &&
- "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() {
- raw_ostream &OS = outs();
- #ifdef PACKAGE_VENDOR
- OS << PACKAGE_VENDOR << " ";
- #else
- OS << "LLVM (http://llvm.org/):\n ";
- #endif
- OS << PACKAGE_NAME << " version " << PACKAGE_VERSION;
- #ifdef LLVM_VERSION_INFO
- OS << " " << LLVM_VERSION_INFO;
- #endif
- OS << "\n ";
- #if LLVM_IS_DEBUG_BUILD
- OS << "DEBUG build";
- #else
- OS << "Optimized build";
- #endif
- #ifndef NDEBUG
- OS << " with assertions";
- #endif
- #if LLVM_VERSION_PRINTER_SHOW_HOST_TARGET_INFO
- std::string CPU = std::string(sys::getHostCPUName());
- if (CPU == "generic")
- CPU = "(unknown)";
- OS << ".\n"
- << " Default target: " << sys::getDefaultTargetTriple() << '\n'
- << " Host CPU: " << CPU;
- #endif
- OS << '\n';
- }
- 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(*AllSubCommands)};
- 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(*AllSubCommands)};
- // 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(*AllSubCommands)};
- 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(*AllSubCommands)};
- 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(*AllSubCommands)};
- 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(*AllSubCommands)};
- 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();
- // Iterate over any registered extra printers and call them to add further
- // information.
- if (!CommonOptions->ExtraVersionPrinters.empty()) {
- outs() << '\n';
- for (const auto &I : CommonOptions->ExtraVersionPrinters)
- I(outs());
- }
- 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();
- }
- 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());
- }
|