mirror of
https://github.com/ldc-developers/ldc.git
synced 2025-05-01 23:50:43 +03:00

This approach is easier and requires less code. With some template foo the BoolOrDefaultAdapter is not required. This solutions applies to previous LLVM versions, too.
191 lines
6.2 KiB
C++
191 lines
6.2 KiB
C++
//===-- gen/cl_helpers.h - Command line processing helpers ------*- C++ -*-===//
|
||
//
|
||
// LDC – the LLVM D compiler
|
||
//
|
||
// This file is distributed under the BSD-style LDC license. See the LICENSE
|
||
// file for details.
|
||
//
|
||
//===----------------------------------------------------------------------===//
|
||
//
|
||
// Helpers to augment the LLVM command line parsing library with some extra
|
||
// functionality.
|
||
//
|
||
//===----------------------------------------------------------------------===//
|
||
|
||
#ifndef LDC_GEN_CL_HELPERS_H
|
||
#define LDC_GEN_CL_HELPERS_H
|
||
|
||
#include "llvm/Support/CommandLine.h"
|
||
#include "llvm/Support/Compiler.h"
|
||
#include "gen/llvmcompat.h"
|
||
|
||
#if LDC_LLVM_VER < 306
|
||
#define LLVM_END_WITH_NULL END_WITH_NULL
|
||
#endif
|
||
|
||
template <typename TYPE> struct Array;
|
||
typedef Array<const char *> Strings;
|
||
|
||
namespace opts {
|
||
namespace cl = llvm::cl;
|
||
|
||
/// Helper class to determine values
|
||
template<class DT>
|
||
struct FlagParserDataType {};
|
||
|
||
template<>
|
||
struct FlagParserDataType<bool>{
|
||
static bool true_val() { return true; }
|
||
static bool false_val() { return false; }
|
||
};
|
||
|
||
template<>
|
||
struct FlagParserDataType<cl::boolOrDefault> {
|
||
static cl::boolOrDefault true_val() { return cl::BOU_TRUE; }
|
||
static cl::boolOrDefault false_val() { return cl::BOU_FALSE; }
|
||
};
|
||
|
||
template <class DataType>
|
||
class FlagParser : public cl::generic_parser_base {
|
||
protected:
|
||
llvm::SmallVector<std::pair<std::string, DataType>, 2> switches;
|
||
#if LDC_LLVM_VER >= 307
|
||
cl::Option& owner() const { return Owner; }
|
||
#else
|
||
cl::Option* Owner;
|
||
cl::Option& owner() const { return *Owner; }
|
||
#endif
|
||
|
||
public:
|
||
#if LDC_LLVM_VER >= 307
|
||
FlagParser(cl::Option& O) : generic_parser_base(O) {}
|
||
#else
|
||
FlagParser() : generic_parser_base(), Owner(0) {}
|
||
#endif
|
||
typedef DataType parser_data_type;
|
||
|
||
#if LDC_LLVM_VER >= 307
|
||
void initialize() {
|
||
#else
|
||
void initialize(cl::Option& O) {
|
||
Owner = &O;
|
||
#endif
|
||
std::string Name(owner().ArgStr);
|
||
switches.push_back(make_pair("enable-" + Name, FlagParserDataType<DataType>::true_val()));
|
||
switches.push_back(make_pair("disable-" + Name, FlagParserDataType<DataType>::false_val()));
|
||
// Replace <foo> with -enable-<foo> and register -disable-<foo>
|
||
#if LDC_LLVM_VER >= 307
|
||
// A literal option can only registered if the argstr is empty -
|
||
// just do this first.
|
||
owner().setArgStr("");
|
||
AddLiteralOption(Owner, strdup(switches[1].first.data()));
|
||
#endif
|
||
owner().setArgStr(switches[0].first.data());
|
||
}
|
||
|
||
enum cl::ValueExpected getValueExpectedFlagDefault() const {
|
||
return cl::ValueOptional;
|
||
}
|
||
|
||
// Implement virtual functions needed by generic_parser_base
|
||
unsigned getNumOptions() const LLVM_OVERRIDE { return 1; }
|
||
const char* getOption(unsigned N) const LLVM_OVERRIDE {
|
||
assert(N == 0);
|
||
return owner().ArgStr;
|
||
}
|
||
|
||
const char* getDescription(unsigned N) const LLVM_OVERRIDE {
|
||
assert(N == 0);
|
||
return owner().HelpStr;
|
||
}
|
||
|
||
private:
|
||
struct OptionValue : cl::OptionValueBase < DataType, false > {
|
||
OptionValue() { };
|
||
};
|
||
const OptionValue EmptyOptionValue;
|
||
|
||
public:
|
||
// getOptionValue - Return the value of option name N.
|
||
const cl::GenericOptionValue &getOptionValue(unsigned N) const LLVM_OVERRIDE {
|
||
return EmptyOptionValue;
|
||
}
|
||
|
||
// parse - Return true on error.
|
||
bool parse(cl::Option& O, llvm::StringRef ArgName, llvm::StringRef Arg, DataType& Val) {
|
||
typedef typename llvm::SmallVector<std::pair<std::string, DataType>, 2>::iterator It;
|
||
for (It I = switches.begin(), E = switches.end(); I != E; ++I) {
|
||
llvm::StringRef name = I->first;
|
||
if (name == ArgName
|
||
|| (name.size() < ArgName.size()
|
||
&& ArgName.substr(0, name.size()) == name
|
||
&& ArgName[name.size()] == '=')) {
|
||
if (!parse(owner(), Arg, Val))
|
||
{
|
||
Val = (Val == I->second) ? FlagParserDataType<DataType>::true_val() : FlagParserDataType<DataType>::false_val();
|
||
return false;
|
||
}
|
||
// Invalid option value
|
||
break;
|
||
}
|
||
}
|
||
return true;
|
||
}
|
||
|
||
void getExtraOptionNames(llvm::SmallVectorImpl<const char*>& Names) {
|
||
typedef typename llvm::SmallVector<std::pair<std::string, DataType>, 2>::iterator It;
|
||
for (It I = switches.begin() + 1, E = switches.end(); I != E; ++I) {
|
||
Names.push_back(I->first.data());
|
||
}
|
||
}
|
||
|
||
private:
|
||
static bool parse(cl::Option &O, llvm::StringRef Arg, DataType &Val) {
|
||
if (Arg == "" || Arg == "true" || Arg == "TRUE" || Arg == "True" ||
|
||
Arg == "1") {
|
||
Val = FlagParserDataType<DataType>::true_val();
|
||
return false;
|
||
}
|
||
|
||
if (Arg == "false" || Arg == "FALSE" || Arg == "False" || Arg == "0") {
|
||
Val = FlagParserDataType<DataType>::false_val();
|
||
return false;
|
||
}
|
||
return O.error("'" + Arg +
|
||
"' is invalid value for boolean argument! Try 0 or 1");
|
||
}
|
||
};
|
||
|
||
/// Helper class for options that set multiple flags
|
||
class MultiSetter {
|
||
std::vector<bool*> locations;
|
||
bool invert;
|
||
MultiSetter(bool); //not implemented, disable auto-conversion
|
||
public:
|
||
MultiSetter(bool invert, bool* p, ...) LLVM_END_WITH_NULL;
|
||
|
||
void operator=(bool val);
|
||
};
|
||
|
||
/// Helper class to fill Strings with char* when given strings
|
||
/// (Errors on empty strings)
|
||
class StringsAdapter {
|
||
const char* name;
|
||
Strings** arrp;
|
||
public:
|
||
StringsAdapter(const char* name_, Strings*& arr) {
|
||
name = name_;
|
||
arrp = &arr;
|
||
assert(name);
|
||
assert(arrp);
|
||
}
|
||
|
||
void push_back(const char* cstr);
|
||
|
||
void push_back(const std::string& str) {
|
||
push_back(str.c_str());
|
||
}
|
||
};
|
||
}
|
||
|
||
#endif
|