llvm-project/clang/lib/Tooling/ExpandResponseFilesCompilationDatabase.cpp
Kadir Cetinkaya 0f12f9096e
Revert "[llvm][Support] Take in CurrentDirectory as a parameter in ExpandResponseFiles"
This reverts commit 75656005dbc8866e1888932a68a830b0df403560.
2019-12-04 15:58:01 +01:00

93 lines
3.1 KiB
C++

//===- ExpandResponseFileCompilationDataBase.cpp --------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
#include "clang/Tooling/CompilationDatabase.h"
#include "llvm/ADT/Triple.h"
#include "llvm/Support/CommandLine.h"
#include "llvm/Support/ConvertUTF.h"
#include "llvm/Support/ErrorOr.h"
#include "llvm/Support/MemoryBuffer.h"
#include "llvm/Support/Path.h"
#include "llvm/Support/StringSaver.h"
namespace clang {
namespace tooling {
namespace {
class ExpandResponseFilesDatabase : public CompilationDatabase {
public:
ExpandResponseFilesDatabase(
std::unique_ptr<CompilationDatabase> Base,
llvm::cl::TokenizerCallback Tokenizer,
llvm::IntrusiveRefCntPtr<llvm::vfs::FileSystem> FS)
: Base(std::move(Base)), Tokenizer(Tokenizer), FS(std::move(FS)) {
assert(this->Base != nullptr);
assert(this->Tokenizer != nullptr);
assert(this->FS != nullptr);
}
std::vector<std::string> getAllFiles() const override {
return Base->getAllFiles();
}
std::vector<CompileCommand>
getCompileCommands(StringRef FilePath) const override {
return expand(Base->getCompileCommands(FilePath));
}
std::vector<CompileCommand> getAllCompileCommands() const override {
return expand(Base->getAllCompileCommands());
}
private:
std::vector<CompileCommand> expand(std::vector<CompileCommand> Cmds) const {
for (auto &Cmd : Cmds) {
// FIXME: we should rather propagate the current directory into
// ExpandResponseFiles as well in addition to FS.
if (std::error_code EC = FS->setCurrentWorkingDirectory(Cmd.Directory)) {
llvm::consumeError(llvm::errorCodeToError(EC));
continue;
}
bool SeenRSPFile = false;
llvm::SmallVector<const char *, 20> Argv;
Argv.reserve(Cmd.CommandLine.size());
for (auto &Arg : Cmd.CommandLine) {
Argv.push_back(Arg.c_str());
SeenRSPFile |= Arg.front() == '@';
}
if (!SeenRSPFile)
continue;
llvm::BumpPtrAllocator Alloc;
llvm::StringSaver Saver(Alloc);
llvm::cl::ExpandResponseFiles(Saver, Tokenizer, Argv, false, false, *FS);
Cmd.CommandLine.assign(Argv.begin(), Argv.end());
}
return Cmds;
}
private:
std::unique_ptr<CompilationDatabase> Base;
llvm::cl::TokenizerCallback Tokenizer;
llvm::IntrusiveRefCntPtr<llvm::vfs::FileSystem> FS;
};
} // namespace
std::unique_ptr<CompilationDatabase>
expandResponseFiles(std::unique_ptr<CompilationDatabase> Base,
llvm::IntrusiveRefCntPtr<llvm::vfs::FileSystem> FS) {
auto Tokenizer = llvm::Triple(llvm::sys::getProcessTriple()).isOSWindows()
? llvm::cl::TokenizeWindowsCommandLine
: llvm::cl::TokenizeGNUCommandLine;
return std::make_unique<ExpandResponseFilesDatabase>(
std::move(Base), Tokenizer, std::move(FS));
}
} // namespace tooling
} // namespace clang