blob: 35ecbf3d5f4daac9712d8cbc6f74d162f67c109d [file] [log] [blame]
Eric Liuc5105f92018-02-16 14:15:55 +00001//===--- Headers.cpp - Include headers ---------------------------*- C++-*-===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
10#include "Headers.h"
11#include "Compiler.h"
12#include "Logger.h"
Eric Liu155f5a42018-05-14 12:19:16 +000013#include "SourceCode.h"
Eric Liuc5105f92018-02-16 14:15:55 +000014#include "clang/Frontend/CompilerInstance.h"
15#include "clang/Frontend/CompilerInvocation.h"
16#include "clang/Frontend/FrontendActions.h"
17#include "clang/Lex/HeaderSearch.h"
Eric Liu709bde82018-02-19 18:48:44 +000018#include "llvm/Support/Path.h"
Eric Liuc5105f92018-02-16 14:15:55 +000019
20namespace clang {
21namespace clangd {
22namespace {
23
24class RecordHeaders : public PPCallbacks {
25public:
Eric Liu155f5a42018-05-14 12:19:16 +000026 RecordHeaders(const SourceManager &SM,
27 std::function<void(Inclusion)> Callback)
28 : SM(SM), Callback(std::move(Callback)) {}
Eric Liuc5105f92018-02-16 14:15:55 +000029
Eric Liu155f5a42018-05-14 12:19:16 +000030 // Record existing #includes - both written and resolved paths. Only #includes
31 // in the main file are collected.
32 void InclusionDirective(SourceLocation HashLoc, const Token & /*IncludeTok*/,
Eric Liu6c8e8582018-02-26 08:32:13 +000033 llvm::StringRef FileName, bool IsAngled,
Eric Liu155f5a42018-05-14 12:19:16 +000034 CharSourceRange FilenameRange, const FileEntry *File,
35 llvm::StringRef /*SearchPath*/,
Eric Liuc5105f92018-02-16 14:15:55 +000036 llvm::StringRef /*RelativePath*/,
Julie Hockett546943f2018-05-10 19:13:14 +000037 const Module * /*Imported*/,
38 SrcMgr::CharacteristicKind /*FileType*/) override {
Eric Liu155f5a42018-05-14 12:19:16 +000039 // Only inclusion directives in the main file make sense. The user cannot
40 // select directives not in the main file.
41 if (HashLoc.isInvalid() || !SM.isInMainFile(HashLoc))
42 return;
43 std::string Written =
44 (IsAngled ? "<" + FileName + ">" : "\"" + FileName + "\"").str();
45 std::string Resolved = (!File || File->tryGetRealPathName().empty())
46 ? ""
47 : File->tryGetRealPathName();
48 Callback({halfOpenToRange(SM, FilenameRange), Written, Resolved});
Eric Liuc5105f92018-02-16 14:15:55 +000049 }
50
51private:
Eric Liu155f5a42018-05-14 12:19:16 +000052 const SourceManager &SM;
53 std::function<void(Inclusion)> Callback;
Eric Liuc5105f92018-02-16 14:15:55 +000054};
55
56} // namespace
57
Eric Liu6c8e8582018-02-26 08:32:13 +000058bool isLiteralInclude(llvm::StringRef Include) {
59 return Include.startswith("<") || Include.startswith("\"");
60}
61
62bool HeaderFile::valid() const {
63 return (Verbatim && isLiteralInclude(File)) ||
64 (!Verbatim && llvm::sys::path::is_absolute(File));
65}
66
Eric Liu155f5a42018-05-14 12:19:16 +000067std::unique_ptr<PPCallbacks>
68collectInclusionsInMainFileCallback(const SourceManager &SM,
69 std::function<void(Inclusion)> Callback) {
70 return llvm::make_unique<RecordHeaders>(SM, std::move(Callback));
71}
72
Eric Liuc5105f92018-02-16 14:15:55 +000073/// FIXME(ioeric): we might not want to insert an absolute include path if the
74/// path is not shortened.
Eric Liu8f3678d2018-06-15 13:34:18 +000075bool IncludeInserter::shouldInsertInclude(
76 const HeaderFile &DeclaringHeader, const HeaderFile &InsertedHeader) const {
Eric Liu6c8e8582018-02-26 08:32:13 +000077 assert(DeclaringHeader.valid() && InsertedHeader.valid());
Eric Liu8f3678d2018-06-15 13:34:18 +000078 if (FileName == DeclaringHeader.File || FileName == InsertedHeader.File)
79 return false;
Eric Liu155f5a42018-05-14 12:19:16 +000080 llvm::StringSet<> IncludedHeaders;
81 for (const auto &Inc : Inclusions) {
82 IncludedHeaders.insert(Inc.Written);
83 if (!Inc.Resolved.empty())
84 IncludedHeaders.insert(Inc.Resolved);
85 }
Eric Liu6c8e8582018-02-26 08:32:13 +000086 auto Included = [&](llvm::StringRef Header) {
Eric Liu155f5a42018-05-14 12:19:16 +000087 return IncludedHeaders.find(Header) != IncludedHeaders.end();
Eric Liu6c8e8582018-02-26 08:32:13 +000088 };
Eric Liu8f3678d2018-06-15 13:34:18 +000089 return !Included(DeclaringHeader.File) && !Included(InsertedHeader.File);
90}
Eric Liuc5105f92018-02-16 14:15:55 +000091
Eric Liu8f3678d2018-06-15 13:34:18 +000092std::string
93IncludeInserter::calculateIncludePath(const HeaderFile &DeclaringHeader,
94 const HeaderFile &InsertedHeader) const {
95 assert(DeclaringHeader.valid() && InsertedHeader.valid());
Eric Liu6c8e8582018-02-26 08:32:13 +000096 if (InsertedHeader.Verbatim)
97 return InsertedHeader.File;
Eric Liu8f3678d2018-06-15 13:34:18 +000098 bool IsSystem = false;
Eric Liuc5105f92018-02-16 14:15:55 +000099 std::string Suggested = HeaderSearchInfo.suggestPathToFileForDiagnostics(
Eric Liu63f419a2018-05-15 15:29:32 +0000100 InsertedHeader.File, BuildDir, &IsSystem);
Eric Liuc5105f92018-02-16 14:15:55 +0000101 if (IsSystem)
102 Suggested = "<" + Suggested + ">";
103 else
104 Suggested = "\"" + Suggested + "\"";
Eric Liuc5105f92018-02-16 14:15:55 +0000105 return Suggested;
106}
107
Eric Liu8f3678d2018-06-15 13:34:18 +0000108Optional<TextEdit> IncludeInserter::insert(StringRef VerbatimHeader) const {
109 Optional<TextEdit> Edit = None;
110 if (auto Insertion = Inserter.insert(VerbatimHeader.trim("\"<>"),
111 VerbatimHeader.startswith("<")))
112 Edit = replacementToEdit(Code, *Insertion);
113 return Edit;
Eric Liu63f419a2018-05-15 15:29:32 +0000114}
115
Eric Liuc5105f92018-02-16 14:15:55 +0000116} // namespace clangd
117} // namespace clang