1480093f4SDimitry Andric //===- ExpandResponseFileCompilationDataBase.cpp --------------------------===//
2480093f4SDimitry Andric //
3480093f4SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4480093f4SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
5480093f4SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6480093f4SDimitry Andric //
7480093f4SDimitry Andric //===----------------------------------------------------------------------===//
8480093f4SDimitry Andric 
9480093f4SDimitry Andric #include "clang/Tooling/CompilationDatabase.h"
10480093f4SDimitry Andric #include "llvm/ADT/StringRef.h"
11480093f4SDimitry Andric #include "llvm/ADT/Triple.h"
12480093f4SDimitry Andric #include "llvm/Support/CommandLine.h"
13480093f4SDimitry Andric #include "llvm/Support/ConvertUTF.h"
14480093f4SDimitry Andric #include "llvm/Support/ErrorOr.h"
155ffd83dbSDimitry Andric #include "llvm/Support/Host.h"
16480093f4SDimitry Andric #include "llvm/Support/MemoryBuffer.h"
17480093f4SDimitry Andric #include "llvm/Support/Path.h"
18480093f4SDimitry Andric #include "llvm/Support/StringSaver.h"
19480093f4SDimitry Andric 
20480093f4SDimitry Andric namespace clang {
21480093f4SDimitry Andric namespace tooling {
22480093f4SDimitry Andric namespace {
23480093f4SDimitry Andric 
24480093f4SDimitry Andric class ExpandResponseFilesDatabase : public CompilationDatabase {
25480093f4SDimitry Andric public:
ExpandResponseFilesDatabase(std::unique_ptr<CompilationDatabase> Base,llvm::cl::TokenizerCallback Tokenizer,llvm::IntrusiveRefCntPtr<llvm::vfs::FileSystem> FS)26480093f4SDimitry Andric   ExpandResponseFilesDatabase(
27480093f4SDimitry Andric       std::unique_ptr<CompilationDatabase> Base,
28480093f4SDimitry Andric       llvm::cl::TokenizerCallback Tokenizer,
29480093f4SDimitry Andric       llvm::IntrusiveRefCntPtr<llvm::vfs::FileSystem> FS)
30480093f4SDimitry Andric       : Base(std::move(Base)), Tokenizer(Tokenizer), FS(std::move(FS)) {
31480093f4SDimitry Andric     assert(this->Base != nullptr);
32480093f4SDimitry Andric     assert(this->Tokenizer != nullptr);
33480093f4SDimitry Andric     assert(this->FS != nullptr);
34480093f4SDimitry Andric   }
35480093f4SDimitry Andric 
getAllFiles() const36480093f4SDimitry Andric   std::vector<std::string> getAllFiles() const override {
37480093f4SDimitry Andric     return Base->getAllFiles();
38480093f4SDimitry Andric   }
39480093f4SDimitry Andric 
40480093f4SDimitry Andric   std::vector<CompileCommand>
getCompileCommands(StringRef FilePath) const41480093f4SDimitry Andric   getCompileCommands(StringRef FilePath) const override {
42480093f4SDimitry Andric     return expand(Base->getCompileCommands(FilePath));
43480093f4SDimitry Andric   }
44480093f4SDimitry Andric 
getAllCompileCommands() const45480093f4SDimitry Andric   std::vector<CompileCommand> getAllCompileCommands() const override {
46480093f4SDimitry Andric     return expand(Base->getAllCompileCommands());
47480093f4SDimitry Andric   }
48480093f4SDimitry Andric 
49480093f4SDimitry Andric private:
expand(std::vector<CompileCommand> Cmds) const50480093f4SDimitry Andric   std::vector<CompileCommand> expand(std::vector<CompileCommand> Cmds) const {
51480093f4SDimitry Andric     for (auto &Cmd : Cmds) {
52480093f4SDimitry Andric       bool SeenRSPFile = false;
53480093f4SDimitry Andric       llvm::SmallVector<const char *, 20> Argv;
54480093f4SDimitry Andric       Argv.reserve(Cmd.CommandLine.size());
55480093f4SDimitry Andric       for (auto &Arg : Cmd.CommandLine) {
56480093f4SDimitry Andric         Argv.push_back(Arg.c_str());
57*5f7ddb14SDimitry Andric         if (!Arg.empty())
58480093f4SDimitry Andric           SeenRSPFile |= Arg.front() == '@';
59480093f4SDimitry Andric       }
60480093f4SDimitry Andric       if (!SeenRSPFile)
61480093f4SDimitry Andric         continue;
62480093f4SDimitry Andric       llvm::BumpPtrAllocator Alloc;
63480093f4SDimitry Andric       llvm::StringSaver Saver(Alloc);
64*5f7ddb14SDimitry Andric       llvm::cl::ExpandResponseFiles(Saver, Tokenizer, Argv, false, false,
65*5f7ddb14SDimitry Andric                                     llvm::StringRef(Cmd.Directory), *FS);
66480093f4SDimitry Andric       // Don't assign directly, Argv aliases CommandLine.
67480093f4SDimitry Andric       std::vector<std::string> ExpandedArgv(Argv.begin(), Argv.end());
68480093f4SDimitry Andric       Cmd.CommandLine = std::move(ExpandedArgv);
69480093f4SDimitry Andric     }
70480093f4SDimitry Andric     return Cmds;
71480093f4SDimitry Andric   }
72480093f4SDimitry Andric 
73480093f4SDimitry Andric private:
74480093f4SDimitry Andric   std::unique_ptr<CompilationDatabase> Base;
75480093f4SDimitry Andric   llvm::cl::TokenizerCallback Tokenizer;
76480093f4SDimitry Andric   llvm::IntrusiveRefCntPtr<llvm::vfs::FileSystem> FS;
77480093f4SDimitry Andric };
78480093f4SDimitry Andric 
79480093f4SDimitry Andric } // namespace
80480093f4SDimitry Andric 
81480093f4SDimitry Andric std::unique_ptr<CompilationDatabase>
expandResponseFiles(std::unique_ptr<CompilationDatabase> Base,llvm::IntrusiveRefCntPtr<llvm::vfs::FileSystem> FS)82480093f4SDimitry Andric expandResponseFiles(std::unique_ptr<CompilationDatabase> Base,
83480093f4SDimitry Andric                     llvm::IntrusiveRefCntPtr<llvm::vfs::FileSystem> FS) {
84480093f4SDimitry Andric   auto Tokenizer = llvm::Triple(llvm::sys::getProcessTriple()).isOSWindows()
85480093f4SDimitry Andric                        ? llvm::cl::TokenizeWindowsCommandLine
86480093f4SDimitry Andric                        : llvm::cl::TokenizeGNUCommandLine;
87480093f4SDimitry Andric   return std::make_unique<ExpandResponseFilesDatabase>(
88480093f4SDimitry Andric       std::move(Base), Tokenizer, std::move(FS));
89480093f4SDimitry Andric }
90480093f4SDimitry Andric 
91480093f4SDimitry Andric } // namespace tooling
92480093f4SDimitry Andric } // namespace clang
93