xref: /freebsd/contrib/llvm-project/clang/lib/Tooling/ExpandResponseFilesCompilationDatabase.cpp (revision 480093f4440d54b30b3025afeac24b48f2ba7a2e)
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