summaryrefslogtreecommitdiffstats
path: root/llvm
diff options
context:
space:
mode:
authorJeffrey Yasskin <jyasskin@google.com>2009-09-24 01:14:07 +0000
committerJeffrey Yasskin <jyasskin@google.com>2009-09-24 01:14:07 +0000
commita75d6bf3a85e792e424e19615d919784d8e5ab02 (patch)
tree3e886c6042c9291cbc27e92a3b12fe5313bd3a83 /llvm
parentd9120d35750e94c3bf5209fb04fa1bcd918426bb (diff)
downloadbcm5719-llvm-a75d6bf3a85e792e424e19615d919784d8e5ab02.tar.gz
bcm5719-llvm-a75d6bf3a85e792e424e19615d919784d8e5ab02.zip
Roll back r82348, which introduced an infinite loop in ParseCStringVector() that
a trivial unittest would have caught. This revision also adds the trivial unittest. llvm-svn: 82675
Diffstat (limited to 'llvm')
-rw-r--r--llvm/lib/Support/CommandLine.cpp55
-rw-r--r--llvm/unittests/Support/CommandLineTest.cpp48
2 files changed, 81 insertions, 22 deletions
diff --git a/llvm/lib/Support/CommandLine.cpp b/llvm/lib/Support/CommandLine.cpp
index 43686c07cce..204145396e4 100644
--- a/llvm/lib/Support/CommandLine.cpp
+++ b/llvm/lib/Support/CommandLine.cpp
@@ -351,31 +351,42 @@ static bool EatsUnboundedNumberOfValues(const Option *O) {
/// using strdup(), so it is the caller's responsibility to free()
/// them later.
///
-static void ParseCStringVector(std::vector<char *> &OutputVector,
- const char *Input) {
+static void ParseCStringVector(std::vector<char *> &output,
+ const char *input) {
// Characters which will be treated as token separators:
- StringRef Delims = " \v\f\t\r\n";
-
- StringRef WorkStr(Input);
- while (!WorkStr.empty()) {
- // If the first character is a delimiter, strip them off.
- if (Delims.find(WorkStr[0]) != StringRef::npos) {
- size_t Pos = WorkStr.find_first_not_of(Delims);
- if (Pos == StringRef::npos) Pos = WorkStr.size();
- WorkStr = WorkStr.substr(Pos);
- continue;
+ static const char *const delims = " \v\f\t\r\n";
+
+ std::string work(input);
+ // Skip past any delims at head of input string.
+ size_t pos = work.find_first_not_of(delims);
+ // If the string consists entirely of delims, then exit early.
+ if (pos == std::string::npos) return;
+ // Otherwise, jump forward to beginning of first word.
+ work = work.substr(pos);
+ // Find position of first delimiter.
+ pos = work.find_first_of(delims);
+
+ while (!work.empty() && pos != std::string::npos) {
+ // Everything from 0 to POS is the next word to copy.
+ output.push_back(strdup(work.substr(0,pos).c_str()));
+ // Is there another word in the string?
+ size_t nextpos = work.find_first_not_of(delims, pos + 1);
+ if (nextpos != std::string::npos) {
+ // Yes? Then remove delims from beginning ...
+ work = work.substr(work.find_first_not_of(delims, pos + 1));
+ // and find the end of the word.
+ pos = work.find_first_of(delims);
+ } else {
+ // No? (Remainder of string is delims.) End the loop.
+ work = "";
+ pos = std::string::npos;
}
-
- // Find position of first delimiter.
- size_t Pos = WorkStr.find_first_of(Delims);
- if (Pos == StringRef::npos) Pos = WorkStr.size();
-
- // Everything from 0 to Pos is the next word to copy.
- char *NewStr = (char*)malloc(Pos+1);
- memcpy(NewStr, WorkStr.data(), Pos);
- NewStr[Pos] = 0;
- OutputVector.push_back(NewStr);
}
+
+ // If `input' ended with non-delim char, then we'll get here with
+ // the last word of `input' in `work'; copy it now.
+ if (!work.empty())
+ output.push_back(strdup(work.c_str()));
}
/// ParseEnvironmentOptions - An alternative entry point to the
diff --git a/llvm/unittests/Support/CommandLineTest.cpp b/llvm/unittests/Support/CommandLineTest.cpp
new file mode 100644
index 00000000000..70d6950f8b0
--- /dev/null
+++ b/llvm/unittests/Support/CommandLineTest.cpp
@@ -0,0 +1,48 @@
+//===- llvm/unittest/Support/CommandLineTest.cpp - CommandLine tests ------===//
+//
+// The LLVM Compiler Infrastructure
+//
+// This file is distributed under the University of Illinois Open Source
+// License. See LICENSE.TXT for details.
+//
+//===----------------------------------------------------------------------===//
+
+#include "llvm/Support/CommandLine.h"
+
+#include "gtest/gtest.h"
+
+#include <string>
+#include <stdlib.h>
+
+using namespace llvm;
+
+namespace {
+
+class TempEnvVar {
+ public:
+ TempEnvVar(const char *name, const char *value)
+ : name(name) {
+ const char *old_value = getenv(name);
+ EXPECT_EQ(NULL, old_value) << old_value;
+ setenv(name, value, true);
+ }
+
+ ~TempEnvVar() {
+ unsetenv(name);
+ }
+
+ private:
+ const char *const name;
+};
+
+const char test_env_var[] = "LLVM_TEST_COMMAND_LINE_FLAGS";
+
+cl::opt<std::string> EnvironmentTestOption("env-test-opt");
+TEST(CommandLineTest, ParseEnvironment) {
+ TempEnvVar TEV(test_env_var, "-env-test-opt=hello");
+ EXPECT_EQ("", EnvironmentTestOption);
+ cl::ParseEnvironmentOptions("CommandLineTest", test_env_var);
+ EXPECT_EQ("hello", EnvironmentTestOption);
+}
+
+} // anonymous namespace
OpenPOWER on IntegriCloud