cling/lib/Interpreter/InvocationOptions.cpp
Raphael Isemann d9475011ca Extended InvocationOptions with C++ modules flags.
This makes it easier for cling to check if -fmodules and/or
-fmodule-name is passed so that we can act on these flags in the
CIFactory.
2017-09-14 13:29:20 +02:00

221 lines
7.7 KiB
C++

//--------------------------------------------------------------------*- C++ -*-
// CLING - the C++ LLVM-based InterpreterG :)
// author: Axel Naumann <axel@cern.ch>
//
// This file is dual-licensed: you can choose to license it under the University
// of Illinois Open Source License or the GNU Lesser General Public License. See
// LICENSE.TXT for details.
//------------------------------------------------------------------------------
#include "cling/Interpreter/InvocationOptions.h"
#include "cling/Interpreter/ClingOptions.h"
#include "cling/Utils/Output.h"
#include "clang/Basic/LangOptions.h"
#include "clang/Driver/Options.h"
#include "llvm/Option/Arg.h"
#include "llvm/Option/ArgList.h"
#include "llvm/Option/Option.h"
#include "llvm/Option/OptTable.h"
#include <memory>
using namespace clang;
using namespace clang::driver;
using namespace llvm;
using namespace llvm::opt;
using namespace cling;
using namespace cling::driver::clingoptions;
namespace {
// MSVC C++ backend currently does not support -nostdinc++. Translate it to
// -nostdinc so users scripts are insulated from mundane implementation details.
#if defined(LLVM_ON_WIN32) && !defined(_LIBCPP_VERSION)
#define CLING_TRANSLATE_NOSTDINCxx
// Likely to be string-pooled, but make sure it's valid after func exit.
static const char kNoStdInc[] = "-nostdinc";
#endif
#define PREFIX(NAME, VALUE) const char *const NAME[] = VALUE;
#define OPTION(PREFIX, NAME, ID, KIND, GROUP, ALIAS, ALIASARGS, FLAGS, PARAM, \
HELPTEXT, METAVAR)
#include "cling/Interpreter/ClingOptions.inc"
#undef OPTION
#undef PREFIX
static const OptTable::Info ClingInfoTable[] = {
#define PREFIX(NAME, VALUE)
#define OPTION(PREFIX, NAME, ID, KIND, GROUP, ALIAS, ALIASARGS, FLAGS, PARAM, \
HELPTEXT, METAVAR) \
{ PREFIX, NAME, HELPTEXT, METAVAR, OPT_##ID, Option::KIND##Class, PARAM, \
FLAGS, OPT_##GROUP, OPT_##ALIAS, ALIASARGS },
#include "cling/Interpreter/ClingOptions.inc"
#undef OPTION
#undef PREFIX
};
class ClingOptTable : public OptTable {
public:
ClingOptTable()
: OptTable(ClingInfoTable) {}
};
static OptTable* CreateClingOptTable() {
return new ClingOptTable();
}
static void ParseStartupOpts(cling::InvocationOptions& Opts,
InputArgList& Args) {
Opts.ErrorOut = Args.hasArg(OPT__errorout);
Opts.NoLogo = Args.hasArg(OPT__nologo);
Opts.ShowVersion = Args.hasArg(OPT_version);
Opts.Help = Args.hasArg(OPT_help);
Opts.NoRuntime = Args.hasArg(OPT_noruntime);
if (Arg* MetaStringArg = Args.getLastArg(OPT__metastr, OPT__metastr_EQ)) {
Opts.MetaString = MetaStringArg->getValue();
if (Opts.MetaString.empty()) {
cling::errs() << "ERROR: meta string must be non-empty! Defaulting to '.'.\n";
Opts.MetaString = ".";
}
}
}
static void Extend(std::vector<std::string>& A, std::vector<std::string> B) {
A.reserve(A.size()+B.size());
for (std::string& Val: B)
A.push_back(std::move(Val));
}
static void ParseLinkerOpts(cling::InvocationOptions& Opts,
InputArgList& Args /* , Diags */) {
Extend(Opts.LibsToLoad, Args.getAllArgValues(OPT_l));
Extend(Opts.LibSearchPath, Args.getAllArgValues(OPT_L));
}
}
CompilerOptions::CompilerOptions(int argc, const char* const* argv)
: Language(false), ResourceDir(false), SysRoot(false), NoBuiltinInc(false),
NoCXXInc(false), StdVersion(false), StdLib(false), HasOutput(false),
Verbose(false), CxxModules(false) {
if (argc && argv) {
// Preserve what's already in Remaining, the user might want to push args
// to clang while still using main's argc, argv
// insert should/usually does call reserve, but its not part of the standard
Remaining.reserve(Remaining.size() + argc);
Remaining.insert(Remaining.end(), argv, argv+argc);
Parse(argc, argv);
}
}
void CompilerOptions::Parse(int argc, const char* const argv[],
std::vector<std::string>* Inputs) {
unsigned MissingArgIndex, MissingArgCount;
std::unique_ptr<OptTable> OptsC1(createDriverOptTable());
ArrayRef<const char *> ArgStrings(argv+1, argv + argc);
InputArgList Args(OptsC1->ParseArgs(ArgStrings, MissingArgIndex,
MissingArgCount, 0,
options::NoDriverOption | options::CLOption));
for (const Arg* arg : Args) {
switch (arg->getOption().getID()) {
// case options::OPT_d_Flag:
case options::OPT_E:
case options::OPT_o: HasOutput = true; break;
case options::OPT_x: Language = true; break;
case options::OPT_resource_dir: ResourceDir = true; break;
case options::OPT_isysroot: SysRoot = true; break;
case options::OPT_std_EQ: StdVersion = true; break;
case options::OPT_stdlib_EQ: StdLib = true; break;
// case options::OPT_nostdlib:
case options::OPT_nobuiltininc: NoBuiltinInc = true; break;
// case options::OPT_nostdinc:
case options::OPT_nostdincxx: NoCXXInc = true; break;
case options::OPT_v: Verbose = true; break;
case options::OPT_fmodules: CxxModules = true; break;
case options::OPT_fmodule_name_EQ: LLVM_FALLTHROUGH;
case options::OPT_fmodule_name: ModuleName = arg->getValue(); break;
case options::OPT_fmodules_cache_path: CachePath = arg->getValue(); break;
default:
if (Inputs && arg->getOption().getKind() == Option::InputClass)
Inputs->push_back(arg->getValue());
break;
}
}
}
bool CompilerOptions::DefaultLanguage(const LangOptions* LangOpts) const {
// When StdVersion is set (-std=c++11, -std=gnu++11, etc.) then definitely
// don't setup the defaults, as they may interfere with what the user is doing
if (StdVersion)
return false;
// Also don't set up the defaults when language is explicitly set; unless
// the language was set to generate a PCH, in which case definitely do.
if (Language)
return HasOutput || (LangOpts && LangOpts->CompilingPCH);
return true;
}
InvocationOptions::InvocationOptions(int argc, const char* const* argv) :
MetaString("."), ErrorOut(false), NoLogo(false), ShowVersion(false),
Help(false), NoRuntime(false) {
ArrayRef<const char *> ArgStrings(argv, argv + argc);
unsigned MissingArgIndex, MissingArgCount;
std::unique_ptr<OptTable> Opts(CreateClingOptTable());
InputArgList Args(Opts->ParseArgs(ArgStrings, MissingArgIndex,
MissingArgCount, 0,
options::NoDriverOption | options::CLOption));
// Forward unknown arguments.
for (const Arg* arg : Args) {
switch (arg->getOption().getKind()) {
case Option::FlagClass:
// pass -v to clang as well
if (arg->getOption().getID() != OPT_v)
break;
case Option::UnknownClass:
case Option::InputClass:
// prune "-" we need to control where it appears when invoking clang
if (!arg->getSpelling().equals("-")) {
if (const char* Arg = argv[arg->getIndex()]) {
#ifdef CLING_TRANSLATE_NOSTDINCxx
if (!::strcmp(Arg, "-nostdinc++"))
Arg = kNoStdInc;
#endif
CompilerOpts.Remaining.push_back(Arg);
}
}
default:
break;
}
}
// Get Input list and any compiler specific flags we're interested in
CompilerOpts.Parse(argc, argv, &Inputs);
ParseStartupOpts(*this, Args);
ParseLinkerOpts(*this, Args);
}
void InvocationOptions::PrintHelp() {
std::unique_ptr<OptTable> Opts(CreateClingOptTable());
Opts->PrintHelp(cling::outs(), "cling",
"cling: LLVM/clang C++ Interpreter: http://cern.ch/cling");
cling::outs() << "\n\n";
std::unique_ptr<OptTable> OptsC1(createDriverOptTable());
OptsC1->PrintHelp(cling::outs(), "clang -cc1",
"LLVM 'Clang' Compiler: http://clang.llvm.org");
}