mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-06-17 04:24:00 +00:00
[Support] Promote cl::StringSaver to a separate utility
This class is generally useful. In breaking it out, the primary change is that it has been made non-virtual. It seems like being abstract led to there being 3 different (2 in llvm + 1 in clang) concrete implementations which disagreed about the ownership of the saved strings (see the manual call to free() in the unittest StrDupSaver; yes this is different from the CommandLine.cpp StrDupSaver which owns the stored strings; which is different from Clang's StringSetSaver which just holds a reference to a std::set<std::string> which owns the strings). I've identified 2 other places in the codebase that are open-coding this pattern: memcpy(Alloc.Allocate<char>(strlen(S)+1), S, strlen(S)+1) I'll be switching them over. They are * llvm::sys::Process::GetArgumentVector * The StringAllocator member of YAMLIO's Input class This also will allow simplifying Clang's driver.cpp quite a bit. Let me know if there are any other places that could benefit from StringSaver. I'm also thinking of adding a saveStringRef member for getting a stable StringRef. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@215784 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
@ -76,7 +76,6 @@ void parser<double>::anchor() {}
|
||||
void parser<float>::anchor() {}
|
||||
void parser<std::string>::anchor() {}
|
||||
void parser<char>::anchor() {}
|
||||
void StringSaver::anchor() {}
|
||||
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
@ -509,7 +508,7 @@ void cl::TokenizeGNUCommandLine(StringRef Src, StringSaver &Saver,
|
||||
// End the token if this is whitespace.
|
||||
if (isWhitespace(Src[I])) {
|
||||
if (!Token.empty())
|
||||
NewArgv.push_back(Saver.SaveString(Token.c_str()));
|
||||
NewArgv.push_back(Saver.saveCStr(Token.c_str()));
|
||||
Token.clear();
|
||||
continue;
|
||||
}
|
||||
@ -520,7 +519,7 @@ void cl::TokenizeGNUCommandLine(StringRef Src, StringSaver &Saver,
|
||||
|
||||
// Append the last token after hitting EOF with no whitespace.
|
||||
if (!Token.empty())
|
||||
NewArgv.push_back(Saver.SaveString(Token.c_str()));
|
||||
NewArgv.push_back(Saver.saveCStr(Token.c_str()));
|
||||
}
|
||||
|
||||
/// Backslashes are interpreted in a rather complicated way in the Windows-style
|
||||
@ -593,7 +592,7 @@ void cl::TokenizeWindowsCommandLine(StringRef Src, StringSaver &Saver,
|
||||
if (State == UNQUOTED) {
|
||||
// Whitespace means the end of the token.
|
||||
if (isWhitespace(Src[I])) {
|
||||
NewArgv.push_back(Saver.SaveString(Token.c_str()));
|
||||
NewArgv.push_back(Saver.saveCStr(Token.c_str()));
|
||||
Token.clear();
|
||||
State = INIT;
|
||||
continue;
|
||||
@ -625,7 +624,7 @@ void cl::TokenizeWindowsCommandLine(StringRef Src, StringSaver &Saver,
|
||||
}
|
||||
// Append the last token after hitting EOF with no whitespace.
|
||||
if (!Token.empty())
|
||||
NewArgv.push_back(Saver.SaveString(Token.c_str()));
|
||||
NewArgv.push_back(Saver.saveCStr(Token.c_str()));
|
||||
}
|
||||
|
||||
static bool ExpandResponseFile(const char *FName, StringSaver &Saver,
|
||||
@ -691,25 +690,6 @@ bool cl::ExpandResponseFiles(StringSaver &Saver, TokenizerCallback Tokenizer,
|
||||
return AllExpanded;
|
||||
}
|
||||
|
||||
namespace {
|
||||
class StrDupSaver : public StringSaver {
|
||||
std::vector<char*> Dups;
|
||||
public:
|
||||
~StrDupSaver() {
|
||||
for (std::vector<char *>::iterator I = Dups.begin(), E = Dups.end();
|
||||
I != E; ++I) {
|
||||
char *Dup = *I;
|
||||
free(Dup);
|
||||
}
|
||||
}
|
||||
const char *SaveString(const char *Str) override {
|
||||
char *Dup = strdup(Str);
|
||||
Dups.push_back(Dup);
|
||||
return Dup;
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
/// ParseEnvironmentOptions - An alternative entry point to the
|
||||
/// CommandLine library, which allows you to read the program's name
|
||||
/// from the caller (as PROGNAME) and its command-line arguments from
|
||||
@ -729,8 +709,8 @@ void cl::ParseEnvironmentOptions(const char *progName, const char *envVar,
|
||||
// Get program's "name", which we wouldn't know without the caller
|
||||
// telling us.
|
||||
SmallVector<const char *, 20> newArgv;
|
||||
StrDupSaver Saver;
|
||||
newArgv.push_back(Saver.SaveString(progName));
|
||||
StringSaver Saver;
|
||||
newArgv.push_back(Saver.saveCStr(progName));
|
||||
|
||||
// Parse the value of the environment variable into a "command line"
|
||||
// and hand it off to ParseCommandLineOptions().
|
||||
@ -754,7 +734,7 @@ void cl::ParseCommandLineOptions(int argc, const char * const *argv,
|
||||
SmallVector<const char *, 20> newArgv;
|
||||
for (int i = 0; i != argc; ++i)
|
||||
newArgv.push_back(argv[i]);
|
||||
StrDupSaver Saver;
|
||||
StringSaver Saver;
|
||||
ExpandResponseFiles(Saver, TokenizeGNUCommandLine, newArgv);
|
||||
argv = &newArgv[0];
|
||||
argc = static_cast<int>(newArgv.size());
|
||||
|
Reference in New Issue
Block a user