1*480093f4SDimitry Andric //===- ExpandResponseFileCompilationDataBase.cpp --------------------------===// 2*480093f4SDimitry Andric // 3*480093f4SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4*480093f4SDimitry Andric // See https://llvm.org/LICENSE.txt for license information. 5*480093f4SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6*480093f4SDimitry Andric // 7*480093f4SDimitry Andric //===----------------------------------------------------------------------===// 8*480093f4SDimitry Andric 9*480093f4SDimitry Andric #include "clang/Tooling/CompilationDatabase.h" 10*480093f4SDimitry Andric #include "llvm/ADT/StringRef.h" 11*480093f4SDimitry Andric #include "llvm/ADT/Triple.h" 12*480093f4SDimitry Andric #include "llvm/Support/CommandLine.h" 13*480093f4SDimitry Andric #include "llvm/Support/ConvertUTF.h" 14*480093f4SDimitry Andric #include "llvm/Support/ErrorOr.h" 15*480093f4SDimitry Andric #include "llvm/Support/MemoryBuffer.h" 16*480093f4SDimitry Andric #include "llvm/Support/Path.h" 17*480093f4SDimitry Andric #include "llvm/Support/StringSaver.h" 18*480093f4SDimitry Andric 19*480093f4SDimitry Andric namespace clang { 20*480093f4SDimitry Andric namespace tooling { 21*480093f4SDimitry Andric namespace { 22*480093f4SDimitry Andric 23*480093f4SDimitry Andric class ExpandResponseFilesDatabase : public CompilationDatabase { 24*480093f4SDimitry Andric public: 25*480093f4SDimitry Andric ExpandResponseFilesDatabase( 26*480093f4SDimitry Andric std::unique_ptr<CompilationDatabase> Base, 27*480093f4SDimitry Andric llvm::cl::TokenizerCallback Tokenizer, 28*480093f4SDimitry Andric llvm::IntrusiveRefCntPtr<llvm::vfs::FileSystem> FS) 29*480093f4SDimitry Andric : Base(std::move(Base)), Tokenizer(Tokenizer), FS(std::move(FS)) { 30*480093f4SDimitry Andric assert(this->Base != nullptr); 31*480093f4SDimitry Andric assert(this->Tokenizer != nullptr); 32*480093f4SDimitry Andric assert(this->FS != nullptr); 33*480093f4SDimitry Andric } 34*480093f4SDimitry Andric 35*480093f4SDimitry Andric std::vector<std::string> getAllFiles() const override { 36*480093f4SDimitry Andric return Base->getAllFiles(); 37*480093f4SDimitry Andric } 38*480093f4SDimitry Andric 39*480093f4SDimitry Andric std::vector<CompileCommand> 40*480093f4SDimitry Andric getCompileCommands(StringRef FilePath) const override { 41*480093f4SDimitry Andric return expand(Base->getCompileCommands(FilePath)); 42*480093f4SDimitry Andric } 43*480093f4SDimitry Andric 44*480093f4SDimitry Andric std::vector<CompileCommand> getAllCompileCommands() const override { 45*480093f4SDimitry Andric return expand(Base->getAllCompileCommands()); 46*480093f4SDimitry Andric } 47*480093f4SDimitry Andric 48*480093f4SDimitry Andric private: 49*480093f4SDimitry Andric std::vector<CompileCommand> expand(std::vector<CompileCommand> Cmds) const { 50*480093f4SDimitry Andric for (auto &Cmd : Cmds) { 51*480093f4SDimitry Andric bool SeenRSPFile = false; 52*480093f4SDimitry Andric llvm::SmallVector<const char *, 20> Argv; 53*480093f4SDimitry Andric Argv.reserve(Cmd.CommandLine.size()); 54*480093f4SDimitry Andric for (auto &Arg : Cmd.CommandLine) { 55*480093f4SDimitry Andric Argv.push_back(Arg.c_str()); 56*480093f4SDimitry Andric SeenRSPFile |= Arg.front() == '@'; 57*480093f4SDimitry Andric } 58*480093f4SDimitry Andric if (!SeenRSPFile) 59*480093f4SDimitry Andric continue; 60*480093f4SDimitry Andric llvm::BumpPtrAllocator Alloc; 61*480093f4SDimitry Andric llvm::StringSaver Saver(Alloc); 62*480093f4SDimitry Andric llvm::cl::ExpandResponseFiles(Saver, Tokenizer, Argv, false, false, *FS, 63*480093f4SDimitry Andric llvm::StringRef(Cmd.Directory)); 64*480093f4SDimitry Andric // Don't assign directly, Argv aliases CommandLine. 65*480093f4SDimitry Andric std::vector<std::string> ExpandedArgv(Argv.begin(), Argv.end()); 66*480093f4SDimitry Andric Cmd.CommandLine = std::move(ExpandedArgv); 67*480093f4SDimitry Andric } 68*480093f4SDimitry Andric return Cmds; 69*480093f4SDimitry Andric } 70*480093f4SDimitry Andric 71*480093f4SDimitry Andric private: 72*480093f4SDimitry Andric std::unique_ptr<CompilationDatabase> Base; 73*480093f4SDimitry Andric llvm::cl::TokenizerCallback Tokenizer; 74*480093f4SDimitry Andric llvm::IntrusiveRefCntPtr<llvm::vfs::FileSystem> FS; 75*480093f4SDimitry Andric }; 76*480093f4SDimitry Andric 77*480093f4SDimitry Andric } // namespace 78*480093f4SDimitry Andric 79*480093f4SDimitry Andric std::unique_ptr<CompilationDatabase> 80*480093f4SDimitry Andric expandResponseFiles(std::unique_ptr<CompilationDatabase> Base, 81*480093f4SDimitry Andric llvm::IntrusiveRefCntPtr<llvm::vfs::FileSystem> FS) { 82*480093f4SDimitry Andric auto Tokenizer = llvm::Triple(llvm::sys::getProcessTriple()).isOSWindows() 83*480093f4SDimitry Andric ? llvm::cl::TokenizeWindowsCommandLine 84*480093f4SDimitry Andric : llvm::cl::TokenizeGNUCommandLine; 85*480093f4SDimitry Andric return std::make_unique<ExpandResponseFilesDatabase>( 86*480093f4SDimitry Andric std::move(Base), Tokenizer, std::move(FS)); 87*480093f4SDimitry Andric } 88*480093f4SDimitry Andric 89*480093f4SDimitry Andric } // namespace tooling 90*480093f4SDimitry Andric } // namespace clang 91