From b8f3420d1eb77075f66c83bced17bb2067f1c01f Mon Sep 17 00:00:00 2001 From: Hans Wennborg Date: Fri, 2 Aug 2013 21:20:27 +0000 Subject: [PATCH] Option parsing: recognize the special -- token Everything that comes after -- should be treated as a filename. This enables passing in filenames that would otherwise be conflated with command-line options. This is especially important for clang-cl which supports options starting with /, which are easily conflatable with Unix-style path names. Differential Revision: http://llvm-reviews.chandlerc.com/D1274 llvm-svn: 187675 --- llvm/lib/Option/OptTable.cpp | 17 ++++++++++++++++- llvm/unittests/Option/OptionParsingTest.cpp | 13 +++++++++++++ 2 files changed, 29 insertions(+), 1 deletion(-) diff --git a/llvm/lib/Option/OptTable.cpp b/llvm/lib/Option/OptTable.cpp index 11439d3..98e63bc 100644 --- a/llvm/lib/Option/OptTable.cpp +++ b/llvm/lib/Option/OptTable.cpp @@ -253,11 +253,26 @@ InputArgList *OptTable::ParseArgs(const char *const *ArgBegin, unsigned Index = 0, End = ArgEnd - ArgBegin; while (Index < End) { // Ignore empty arguments (other things may still take them as arguments). - if (Args->getArgString(Index)[0] == '\0') { + StringRef Str = Args->getArgString(Index); + if (Str == "") { ++Index; continue; } + if (Str == "--") { + // Everything after -- is a filename. + ++Index; + + assert(TheInputOptionID != 0 && "Invalid input option ID."); + while (Index < End) { + Args->append(new Arg(getOption(TheInputOptionID), + Args->getArgString(Index), Index, + Args->getArgString(Index))); + ++Index; + } + break; + } + unsigned Prev = Index; Arg *A = ParseOneArg(*Args, Index, FlagsToInclude, FlagsToExclude); assert(Index > Prev && "Parser failed to consume argument."); diff --git a/llvm/unittests/Option/OptionParsingTest.cpp b/llvm/unittests/Option/OptionParsingTest.cpp index 2a5a5a9..5a76d65 100644 --- a/llvm/unittests/Option/OptionParsingTest.cpp +++ b/llvm/unittests/Option/OptionParsingTest.cpp @@ -156,3 +156,16 @@ TEST(Option, AliasArgs) { EXPECT_EQ(AL->getAllArgValues(OPT_B)[0], "foo"); EXPECT_EQ(AL->getAllArgValues(OPT_B)[1], "bar"); } + +TEST(Option, DashDash) { + TestOptTable T; + unsigned MAI, MAC; + + const char *MyArgs[] = { "-A", "--", "-B", "--" }; + OwningPtr AL(T.ParseArgs(MyArgs, array_endof(MyArgs), MAI, MAC)); + EXPECT_TRUE(AL->hasArg(OPT_A)); + EXPECT_FALSE(AL->hasArg(OPT_B)); + EXPECT_EQ(AL->getAllArgValues(OPT_INPUT).size(), 2U); + EXPECT_EQ(AL->getAllArgValues(OPT_INPUT)[0], "-B"); + EXPECT_EQ(AL->getAllArgValues(OPT_INPUT)[1], "--"); +} -- 2.7.4