forked from OSchip/llvm-project
[clang-cl] Use the Windows response file tokenizer
We were still using the Unix response file tokenizer for all driver modes. This was difficult to get right in the beginning because there is a circular dependency. The Driver class also can't officially determine its mode until it can see all possible --driver-mode= flags, and those flags could come from the response file. Now we use the Windows parsing algorithm if the program name looks like clang-cl, or if the --driver-mode=cl flag is present on the main command line. Fixes PR23709. Reviewers: hans Differential Revision: http://reviews.llvm.org/D11229 llvm-svn: 242346
This commit is contained in:
parent
3f5ed1566e
commit
e2d03448ba
|
@ -98,7 +98,9 @@ void Command::writeResponseFile(raw_ostream &OS) const {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// In regular response files, we send all arguments to the response file
|
// In regular response files, we send all arguments to the response file.
|
||||||
|
// Wrapping all arguments in double quotes ensures that both Unix tools and
|
||||||
|
// Windows tools understand the response file.
|
||||||
for (const char *Arg : Arguments) {
|
for (const char *Arg : Arguments) {
|
||||||
OS << '"';
|
OS << '"';
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,13 @@
|
||||||
|
// Don't attempt slash switches on msys bash.
|
||||||
|
// REQUIRES: shell-preserves-root
|
||||||
|
|
||||||
|
// Test that we use the Windows tokenizer for clang-cl response files. The
|
||||||
|
// trailing backslash before the space should be interpreted as a literal
|
||||||
|
// backslash. PR23709
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// RUN: echo '/I%S\Inputs\cl-response-file\ /DFOO=2' > %t.rsp
|
||||||
|
// RUN: %clang_cl /c -### @%t.rsp -- %s 2>&1 | FileCheck %s
|
||||||
|
|
||||||
|
// CHECK: "-D" "FOO=2" "-I" "{{.*}}\\Inputs\\cl-response-file\\"
|
|
@ -231,8 +231,18 @@ static const DriverSuffix *FindDriverSuffix(StringRef ProgName) {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ParseProgName(SmallVectorImpl<const char *> &ArgVector,
|
/// Normalize the program name from argv[0] by stripping the file extension if
|
||||||
std::set<std::string> &SavedStrings) {
|
/// present and lower-casing the string on Windows.
|
||||||
|
static std::string normalizeProgramName(const char *Argv0) {
|
||||||
|
std::string ProgName = llvm::sys::path::stem(Argv0);
|
||||||
|
#ifdef LLVM_ON_WIN32
|
||||||
|
// Transform to lowercase for case insensitive file systems.
|
||||||
|
std::transform(ProgName.begin(), ProgName.end(), ProgName.begin(), ::tolower);
|
||||||
|
#endif
|
||||||
|
return ProgName;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const DriverSuffix *parseDriverSuffix(StringRef ProgName) {
|
||||||
// Try to infer frontend type and default target from the program name by
|
// Try to infer frontend type and default target from the program name by
|
||||||
// comparing it against DriverSuffixes in order.
|
// comparing it against DriverSuffixes in order.
|
||||||
|
|
||||||
|
@ -240,54 +250,53 @@ static void ParseProgName(SmallVectorImpl<const char *> &ArgVector,
|
||||||
// E.g. "x86_64-linux-clang" as interpreted as suffix "clang" with target
|
// E.g. "x86_64-linux-clang" as interpreted as suffix "clang" with target
|
||||||
// prefix "x86_64-linux". If such a target prefix is found, is gets added via
|
// prefix "x86_64-linux". If such a target prefix is found, is gets added via
|
||||||
// -target as implicit first argument.
|
// -target as implicit first argument.
|
||||||
|
const DriverSuffix *DS = FindDriverSuffix(ProgName);
|
||||||
std::string ProgName =llvm::sys::path::stem(ArgVector[0]);
|
|
||||||
#ifdef LLVM_ON_WIN32
|
|
||||||
// Transform to lowercase for case insensitive file systems.
|
|
||||||
ProgName = StringRef(ProgName).lower();
|
|
||||||
#endif
|
|
||||||
|
|
||||||
StringRef ProgNameRef = ProgName;
|
|
||||||
const DriverSuffix *DS = FindDriverSuffix(ProgNameRef);
|
|
||||||
|
|
||||||
if (!DS) {
|
if (!DS) {
|
||||||
// Try again after stripping any trailing version number:
|
// Try again after stripping any trailing version number:
|
||||||
// clang++3.5 -> clang++
|
// clang++3.5 -> clang++
|
||||||
ProgNameRef = ProgNameRef.rtrim("0123456789.");
|
ProgName = ProgName.rtrim("0123456789.");
|
||||||
DS = FindDriverSuffix(ProgNameRef);
|
DS = FindDriverSuffix(ProgName);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!DS) {
|
if (!DS) {
|
||||||
// Try again after stripping trailing -component.
|
// Try again after stripping trailing -component.
|
||||||
// clang++-tot -> clang++
|
// clang++-tot -> clang++
|
||||||
ProgNameRef = ProgNameRef.slice(0, ProgNameRef.rfind('-'));
|
ProgName = ProgName.slice(0, ProgName.rfind('-'));
|
||||||
DS = FindDriverSuffix(ProgNameRef);
|
DS = FindDriverSuffix(ProgName);
|
||||||
|
}
|
||||||
|
return DS;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void insertArgsFromProgramName(StringRef ProgName,
|
||||||
|
const DriverSuffix *DS,
|
||||||
|
SmallVectorImpl<const char *> &ArgVector,
|
||||||
|
std::set<std::string> &SavedStrings) {
|
||||||
|
if (!DS)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (const char *Flag = DS->ModeFlag) {
|
||||||
|
// Add Flag to the arguments.
|
||||||
|
auto it = ArgVector.begin();
|
||||||
|
if (it != ArgVector.end())
|
||||||
|
++it;
|
||||||
|
ArgVector.insert(it, Flag);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (DS) {
|
StringRef::size_type LastComponent = ProgName.rfind(
|
||||||
if (const char *Flag = DS->ModeFlag) {
|
'-', ProgName.size() - strlen(DS->Suffix));
|
||||||
// Add Flag to the arguments.
|
if (LastComponent == StringRef::npos)
|
||||||
auto it = ArgVector.begin();
|
return;
|
||||||
if (it != ArgVector.end())
|
|
||||||
++it;
|
|
||||||
ArgVector.insert(it, Flag);
|
|
||||||
}
|
|
||||||
|
|
||||||
StringRef::size_type LastComponent = ProgNameRef.rfind(
|
// Infer target from the prefix.
|
||||||
'-', ProgNameRef.size() - strlen(DS->Suffix));
|
StringRef Prefix = ProgName.slice(0, LastComponent);
|
||||||
if (LastComponent == StringRef::npos)
|
std::string IgnoredError;
|
||||||
return;
|
if (llvm::TargetRegistry::lookupTarget(Prefix, IgnoredError)) {
|
||||||
|
auto it = ArgVector.begin();
|
||||||
// Infer target from the prefix.
|
if (it != ArgVector.end())
|
||||||
StringRef Prefix = ProgNameRef.slice(0, LastComponent);
|
++it;
|
||||||
std::string IgnoredError;
|
const char *arr[] = { "-target", GetStableCStr(SavedStrings, Prefix) };
|
||||||
if (llvm::TargetRegistry::lookupTarget(Prefix, IgnoredError)) {
|
ArgVector.insert(it, std::begin(arr), std::end(arr));
|
||||||
auto it = ArgVector.begin();
|
|
||||||
if (it != ArgVector.end())
|
|
||||||
++it;
|
|
||||||
const char *arr[] = { "-target", GetStableCStr(SavedStrings, Prefix) };
|
|
||||||
ArgVector.insert(it, std::begin(arr), std::end(arr));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -380,16 +389,33 @@ int main(int argc_, const char **argv_) {
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::string ProgName = normalizeProgramName(argv[0]);
|
||||||
|
const DriverSuffix *DS = parseDriverSuffix(ProgName);
|
||||||
|
|
||||||
llvm::BumpPtrAllocator A;
|
llvm::BumpPtrAllocator A;
|
||||||
llvm::BumpPtrStringSaver Saver(A);
|
llvm::BumpPtrStringSaver Saver(A);
|
||||||
|
|
||||||
|
// Parse response files using the GNU syntax, unless we're in CL mode. There
|
||||||
|
// are two ways to put clang in CL compatibility mode: argv[0] is either
|
||||||
|
// clang-cl or cl, or --driver-mode=cl is on the command line. The normal
|
||||||
|
// command line parsing can't happen until after response file parsing, so we
|
||||||
|
// have to manually search for a --driver-mode=cl argument the hard way.
|
||||||
|
// Finally, our -cc1 tools don't care which tokenization mode we use because
|
||||||
|
// response files written by clang will tokenize the same way in either mode.
|
||||||
|
llvm::cl::TokenizerCallback Tokenizer = &llvm::cl::TokenizeGNUCommandLine;
|
||||||
|
if ((DS && DS->ModeFlag && strcmp(DS->ModeFlag, "--driver-mode=cl") == 0) ||
|
||||||
|
std::find_if(argv.begin(), argv.end(), [](const char *F) {
|
||||||
|
return F && strcmp(F, "--driver-mode=cl") == 0;
|
||||||
|
}) != argv.end()) {
|
||||||
|
Tokenizer = &llvm::cl::TokenizeWindowsCommandLine;
|
||||||
|
}
|
||||||
|
|
||||||
// Determines whether we want nullptr markers in argv to indicate response
|
// Determines whether we want nullptr markers in argv to indicate response
|
||||||
// files end-of-lines. We only use this for the /LINK driver argument.
|
// files end-of-lines. We only use this for the /LINK driver argument.
|
||||||
bool MarkEOLs = true;
|
bool MarkEOLs = true;
|
||||||
if (argv.size() > 1 && StringRef(argv[1]).startswith("-cc1"))
|
if (argv.size() > 1 && StringRef(argv[1]).startswith("-cc1"))
|
||||||
MarkEOLs = false;
|
MarkEOLs = false;
|
||||||
llvm::cl::ExpandResponseFiles(Saver, llvm::cl::TokenizeGNUCommandLine, argv,
|
llvm::cl::ExpandResponseFiles(Saver, Tokenizer, argv, MarkEOLs);
|
||||||
MarkEOLs);
|
|
||||||
|
|
||||||
// Handle -cc1 integrated tools, even if -cc1 was expanded from a response
|
// Handle -cc1 integrated tools, even if -cc1 was expanded from a response
|
||||||
// file.
|
// file.
|
||||||
|
@ -450,7 +476,7 @@ int main(int argc_, const char **argv_) {
|
||||||
SetInstallDir(argv, TheDriver);
|
SetInstallDir(argv, TheDriver);
|
||||||
|
|
||||||
llvm::InitializeAllTargets();
|
llvm::InitializeAllTargets();
|
||||||
ParseProgName(argv, SavedStrings);
|
insertArgsFromProgramName(ProgName, DS, argv, SavedStrings);
|
||||||
|
|
||||||
SetBackdoorDriverOutputsFromEnvVars(TheDriver);
|
SetBackdoorDriverOutputsFromEnvVars(TheDriver);
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue