1dc9fb65cSRiver Riddle //===- CompilationDatabase.cpp - LSP Compilation Database -----------------===//
2dc9fb65cSRiver Riddle //
3dc9fb65cSRiver Riddle // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4dc9fb65cSRiver Riddle // See https://llvm.org/LICENSE.txt for license information.
5dc9fb65cSRiver Riddle // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6dc9fb65cSRiver Riddle //
7dc9fb65cSRiver Riddle //===----------------------------------------------------------------------===//
8dc9fb65cSRiver Riddle
9*305d7185SRiver Riddle #include "mlir/Tools/lsp-server-support/CompilationDatabase.h"
10dc9fb65cSRiver Riddle #include "mlir/Support/FileUtilities.h"
11*305d7185SRiver Riddle #include "mlir/Tools/lsp-server-support/Logging.h"
12*305d7185SRiver Riddle #include "mlir/Tools/lsp-server-support/Protocol.h"
13dc9fb65cSRiver Riddle #include "llvm/ADT/SetVector.h"
14dc9fb65cSRiver Riddle #include "llvm/ADT/StringRef.h"
15dc9fb65cSRiver Riddle #include "llvm/Support/YAMLTraits.h"
16dc9fb65cSRiver Riddle
17dc9fb65cSRiver Riddle using namespace mlir;
18dc9fb65cSRiver Riddle using namespace mlir::lsp;
19dc9fb65cSRiver Riddle
20dc9fb65cSRiver Riddle //===----------------------------------------------------------------------===//
21dc9fb65cSRiver Riddle // YamlFileInfo
22dc9fb65cSRiver Riddle //===----------------------------------------------------------------------===//
23dc9fb65cSRiver Riddle
24dc9fb65cSRiver Riddle namespace {
25dc9fb65cSRiver Riddle struct YamlFileInfo {
26dc9fb65cSRiver Riddle /// The absolute path to the file.
27dc9fb65cSRiver Riddle std::string filename;
28dc9fb65cSRiver Riddle /// The include directories available for the file.
29dc9fb65cSRiver Riddle std::vector<std::string> includeDirs;
30dc9fb65cSRiver Riddle };
31dc9fb65cSRiver Riddle } // namespace
32dc9fb65cSRiver Riddle
33dc9fb65cSRiver Riddle //===----------------------------------------------------------------------===//
34dc9fb65cSRiver Riddle // CompilationDatabase
35dc9fb65cSRiver Riddle //===----------------------------------------------------------------------===//
36dc9fb65cSRiver Riddle
37dc9fb65cSRiver Riddle LLVM_YAML_IS_DOCUMENT_LIST_VECTOR(YamlFileInfo)
38dc9fb65cSRiver Riddle
39dc9fb65cSRiver Riddle namespace llvm {
40dc9fb65cSRiver Riddle namespace yaml {
41dc9fb65cSRiver Riddle template <>
42dc9fb65cSRiver Riddle struct MappingTraits<YamlFileInfo> {
mappingllvm::yaml::MappingTraits43dc9fb65cSRiver Riddle static void mapping(IO &io, YamlFileInfo &info) {
44dc9fb65cSRiver Riddle // Parse the filename and normalize it to the form we will expect from
45dc9fb65cSRiver Riddle // incoming URIs.
46dc9fb65cSRiver Riddle io.mapRequired("filepath", info.filename);
47dc9fb65cSRiver Riddle
48dc9fb65cSRiver Riddle // Normalize the filename to avoid incompatability with incoming URIs.
49dc9fb65cSRiver Riddle if (Expected<lsp::URIForFile> uri =
50dc9fb65cSRiver Riddle lsp::URIForFile::fromFile(info.filename))
51dc9fb65cSRiver Riddle info.filename = uri->file().str();
52dc9fb65cSRiver Riddle
53dc9fb65cSRiver Riddle // Parse the includes from the yaml stream. These are in the form of a
54dc9fb65cSRiver Riddle // semi-colon delimited list.
55dc9fb65cSRiver Riddle std::string combinedIncludes;
56dc9fb65cSRiver Riddle io.mapRequired("includes", combinedIncludes);
57dc9fb65cSRiver Riddle for (StringRef include : llvm::split(combinedIncludes, ";")) {
58dc9fb65cSRiver Riddle if (!include.empty())
59dc9fb65cSRiver Riddle info.includeDirs.push_back(include.str());
60dc9fb65cSRiver Riddle }
61dc9fb65cSRiver Riddle }
62dc9fb65cSRiver Riddle };
63dc9fb65cSRiver Riddle } // end namespace yaml
64dc9fb65cSRiver Riddle } // end namespace llvm
65dc9fb65cSRiver Riddle
CompilationDatabase(ArrayRef<std::string> databases)66dc9fb65cSRiver Riddle CompilationDatabase::CompilationDatabase(ArrayRef<std::string> databases) {
67dc9fb65cSRiver Riddle for (StringRef filename : databases)
68dc9fb65cSRiver Riddle loadDatabase(filename);
69dc9fb65cSRiver Riddle }
70dc9fb65cSRiver Riddle
71dc9fb65cSRiver Riddle const CompilationDatabase::FileInfo &
getFileInfo(StringRef filename) const72dc9fb65cSRiver Riddle CompilationDatabase::getFileInfo(StringRef filename) const {
73dc9fb65cSRiver Riddle auto it = files.find(filename);
74dc9fb65cSRiver Riddle return it == files.end() ? defaultFileInfo : it->second;
75dc9fb65cSRiver Riddle }
76dc9fb65cSRiver Riddle
loadDatabase(StringRef filename)77dc9fb65cSRiver Riddle void CompilationDatabase::loadDatabase(StringRef filename) {
78dc9fb65cSRiver Riddle if (filename.empty())
79dc9fb65cSRiver Riddle return;
80dc9fb65cSRiver Riddle
81dc9fb65cSRiver Riddle // Set up the input file.
82dc9fb65cSRiver Riddle std::string errorMessage;
83dc9fb65cSRiver Riddle std::unique_ptr<llvm::MemoryBuffer> inputFile =
84dc9fb65cSRiver Riddle openInputFile(filename, &errorMessage);
85dc9fb65cSRiver Riddle if (!inputFile) {
86dc9fb65cSRiver Riddle Logger::error("Failed to open compilation database: {0}", errorMessage);
87dc9fb65cSRiver Riddle return;
88dc9fb65cSRiver Riddle }
89dc9fb65cSRiver Riddle llvm::yaml::Input yaml(inputFile->getBuffer());
90dc9fb65cSRiver Riddle
91dc9fb65cSRiver Riddle // Parse the yaml description and add any new files to the database.
92dc9fb65cSRiver Riddle std::vector<YamlFileInfo> parsedFiles;
93dc9fb65cSRiver Riddle yaml >> parsedFiles;
94dc9fb65cSRiver Riddle
95dc9fb65cSRiver Riddle SetVector<StringRef> knownIncludes;
96dc9fb65cSRiver Riddle for (auto &file : parsedFiles) {
97dc9fb65cSRiver Riddle auto it = files.try_emplace(file.filename, std::move(file.includeDirs));
98dc9fb65cSRiver Riddle
99dc9fb65cSRiver Riddle // If we encounter a duplicate file, log a warning and ignore it.
100dc9fb65cSRiver Riddle if (!it.second) {
101dc9fb65cSRiver Riddle Logger::info("Duplicate file in compilation database: {0}",
102dc9fb65cSRiver Riddle file.filename);
103dc9fb65cSRiver Riddle continue;
104dc9fb65cSRiver Riddle }
105dc9fb65cSRiver Riddle
106dc9fb65cSRiver Riddle // Track the includes for the file.
107dc9fb65cSRiver Riddle for (StringRef include : it.first->second.includeDirs)
108dc9fb65cSRiver Riddle knownIncludes.insert(include);
109dc9fb65cSRiver Riddle }
110dc9fb65cSRiver Riddle
111dc9fb65cSRiver Riddle // Add all of the known includes to the default file info. We don't know any
112dc9fb65cSRiver Riddle // information about how to treat these files, but these may be project files
113dc9fb65cSRiver Riddle // that we just don't yet have information for. In these cases, providing some
114dc9fb65cSRiver Riddle // heuristic information provides a better user experience, and generally
115dc9fb65cSRiver Riddle // shouldn't lead to any negative side effects.
116dc9fb65cSRiver Riddle for (StringRef include : knownIncludes)
117dc9fb65cSRiver Riddle defaultFileInfo.includeDirs.push_back(include.str());
118dc9fb65cSRiver Riddle }
119