aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorDaniel Dunbar <daniel@zuster.org>2009-03-18 09:29:36 +0000
committerDaniel Dunbar <daniel@zuster.org>2009-03-18 09:29:36 +0000
commit18a7f33af877fca8f72cfde00d52f4aef600547f (patch)
tree6184ed0779cc1e1c6e286323d82bc26082626c1c
parentb488c1dac8e53206f07103d794a62a3f5012c0f4 (diff)
Driver: Add argument translation utilities to ArgList.
- Support things like telling which -ffoo -fno-foo option won, and forwarding all arguments matching a certain set of options to the tool. git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@67189 91177308-0d34-0410-b5e6-96231b3b80d8
-rw-r--r--include/clang/Driver/ArgList.h25
-rw-r--r--lib/Driver/ArgList.cpp65
2 files changed, 90 insertions, 0 deletions
diff --git a/include/clang/Driver/ArgList.h b/include/clang/Driver/ArgList.h
index 9e6996bd08..67b08aa943 100644
--- a/include/clang/Driver/ArgList.h
+++ b/include/clang/Driver/ArgList.h
@@ -113,6 +113,31 @@ namespace driver {
Arg *MakeJoinedArg(const Option *Opt, const char *Value) const;
/// @}
+ /// @name Translation Utilities
+ /// @{
+
+ /// hasFlag - Given an option \arg Pos and its negative form \arg
+ /// Neg, return true if the option is present, false if the
+ /// negation is present, and \arg Default if neither option is
+ /// given. If both the option and its negation are present, the
+ /// last one wins.
+ bool hasFlag(options::ID Pos, options::ID Neg, bool Default) const;
+
+ /// AddLastArg - Render only the last argument match \arg Id0, if
+ /// present.
+ void AddLastArg(ArgStringList &Output, options::ID Id0) const;
+
+ /// AddAllArgs - Render all arguments matching the given ids.
+ void AddAllArgs(ArgStringList &Output, options::ID Id0) const;
+ void AddAllArgs(ArgStringList &Output, options::ID Id0, options::ID Id1) const;
+ void AddAllArgs(ArgStringList &Output, options::ID Id0, options::ID Id1,
+ options::ID Id2) const;
+
+ /// AddAllArgValues - Render the argument values of all arguments
+ /// matching the given ids.
+ void AddAllArgValues(ArgStringList &Output, options::ID Id0) const;
+
+ /// @}
};
} // end namespace driver
} // end namespace clang
diff --git a/lib/Driver/ArgList.cpp b/lib/Driver/ArgList.cpp
index 3c67f8f1e9..ee201f28bd 100644
--- a/lib/Driver/ArgList.cpp
+++ b/lib/Driver/ArgList.cpp
@@ -58,6 +58,16 @@ Arg *ArgList::getLastArg(options::ID Id0, options::ID Id1, bool Claim) const {
return Res;
}
+bool ArgList::hasFlag(options::ID Pos, options::ID Neg, bool Default) const {
+ Arg *PosA = getLastArg(Pos);
+ Arg *NegA = getLastArg(Pos);
+ if (PosA && NegA)
+ return NegA->getIndex() < PosA->getIndex();
+ if (PosA) return true;
+ if (NegA) return false;
+ return Default;
+}
+
unsigned ArgList::MakeIndex(const char *String0) const {
unsigned Index = ArgStrings.size();
@@ -97,3 +107,58 @@ Arg *ArgList::MakeJoinedArg(const Option *Opt, const char *Value) const {
Joined += Value;
return new JoinedArg(Opt, MakeIndex(Joined.c_str()));
}
+
+void ArgList::AddLastArg(ArgStringList &Output, options::ID Id) const {
+ if (Arg *A = getLastArg(Id)) {
+ A->claim();
+ A->render(*this, Output);
+ }
+}
+
+void ArgList::AddAllArgs(ArgStringList &Output, options::ID Id0) const {
+ // FIXME: Make fast.
+ for (const_iterator it = begin(), ie = end(); it != ie; ++it) {
+ const Arg *A = *it;
+ if (A->getOption().matches(Id0)) {
+ A->claim();
+ A->render(*this, Output);
+ }
+ }
+}
+
+void ArgList::AddAllArgs(ArgStringList &Output, options::ID Id0,
+ options::ID Id1) const {
+ // FIXME: Make fast.
+ for (const_iterator it = begin(), ie = end(); it != ie; ++it) {
+ const Arg *A = *it;
+ if (A->getOption().matches(Id0) || A->getOption().matches(Id1)) {
+ A->claim();
+ A->render(*this, Output);
+ }
+ }
+}
+
+void ArgList::AddAllArgs(ArgStringList &Output, options::ID Id0,
+ options::ID Id1, options::ID Id2) const {
+ // FIXME: Make fast.
+ for (const_iterator it = begin(), ie = end(); it != ie; ++it) {
+ const Arg *A = *it;
+ if (A->getOption().matches(Id0) || A->getOption().matches(Id1) ||
+ A->getOption().matches(Id2)) {
+ A->claim();
+ A->render(*this, Output);
+ }
+ }
+}
+
+void ArgList::AddAllArgValues(ArgStringList &Output, options::ID Id0) const {
+ // FIXME: Make fast.
+ for (const_iterator it = begin(), ie = end(); it != ie; ++it) {
+ const Arg *A = *it;
+ if (A->getOption().matches(Id0)) {
+ A->claim();
+ for (unsigned i = 0, e = A->getNumValues(); i != e; ++i)
+ Output.push_back(A->getValue(*this, i));
+ }
+ }
+}