blob: d9e7112769037e07f4a9fb8fb3168343a8eb6d08 [file] [log] [blame]
Alexander Kornienkobef51cd2014-05-19 16:39:08 +00001//===--- NamespaceCommentCheck.cpp - clang-tidy ---------------------------===//
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 "NamespaceCommentCheck.h"
11#include "clang/AST/ASTContext.h"
12#include "clang/ASTMatchers/ASTMatchers.h"
13#include "clang/Lex/Lexer.h"
Alexander Kornienko6e0cbc82014-09-12 08:53:36 +000014#include "llvm/ADT/StringExtras.h"
Alexander Kornienkobef51cd2014-05-19 16:39:08 +000015
16using namespace clang::ast_matchers;
17
18namespace clang {
19namespace tidy {
Alexander Kornienko33fc3db2014-09-22 10:41:39 +000020namespace readability {
Alexander Kornienkobef51cd2014-05-19 16:39:08 +000021
Alexander Kornienko6e0cbc82014-09-12 08:53:36 +000022NamespaceCommentCheck::NamespaceCommentCheck(StringRef Name,
23 ClangTidyContext *Context)
24 : ClangTidyCheck(Name, Context),
25 NamespaceCommentPattern("^/[/*] *(end (of )?)? *(anonymous|unnamed)? *"
Alexander Kornienko27f126b2014-10-16 15:11:54 +000026 "namespace( +([a-zA-Z0-9_]+))?\\.? *(\\*/)?$",
Alexander Kornienkobef51cd2014-05-19 16:39:08 +000027 llvm::Regex::IgnoreCase),
Alexander Kornienko6e0cbc82014-09-12 08:53:36 +000028 ShortNamespaceLines(Options.get("ShortNamespaceLines", 1u)),
Alexander Kornienko1efc4252014-10-16 11:27:57 +000029 SpacesBeforeComments(Options.get("SpacesBeforeComments", 1u)) {}
Alexander Kornienko6e0cbc82014-09-12 08:53:36 +000030
31void NamespaceCommentCheck::storeOptions(ClangTidyOptions::OptionMap &Opts) {
32 Options.store(Opts, "ShortNamespaceLines", ShortNamespaceLines);
33 Options.store(Opts, "SpacesBeforeComments", SpacesBeforeComments);
34}
Alexander Kornienkobef51cd2014-05-19 16:39:08 +000035
36void NamespaceCommentCheck::registerMatchers(MatchFinder *Finder) {
Aaron Ballman1f1b0672015-09-02 16:05:21 +000037 // Only register the matchers for C++; the functionality currently does not
38 // provide any benefit to other languages, despite being benign.
39 if (getLangOpts().CPlusPlus)
40 Finder->addMatcher(namespaceDecl().bind("namespace"), this);
Alexander Kornienkobef51cd2014-05-19 16:39:08 +000041}
42
Benjamin Kramere7103712015-03-23 12:49:15 +000043static bool locationsInSameFile(const SourceManager &Sources,
44 SourceLocation Loc1, SourceLocation Loc2) {
Alexander Kornienkobef51cd2014-05-19 16:39:08 +000045 return Loc1.isFileID() && Loc2.isFileID() &&
46 Sources.getFileID(Loc1) == Sources.getFileID(Loc2);
47}
48
Benjamin Kramere7103712015-03-23 12:49:15 +000049static std::string getNamespaceComment(const NamespaceDecl *ND,
50 bool InsertLineBreak) {
Alexander Kornienkobef51cd2014-05-19 16:39:08 +000051 std::string Fix = "// namespace";
52 if (!ND->isAnonymousNamespace())
Alexander Kornienko33fc3db2014-09-22 10:41:39 +000053 Fix.append(" ").append(ND->getNameAsString());
Alexander Kornienkobef51cd2014-05-19 16:39:08 +000054 if (InsertLineBreak)
55 Fix.append("\n");
56 return Fix;
57}
58
59void NamespaceCommentCheck::check(const MatchFinder::MatchResult &Result) {
Piotr Padlewski08124b12016-12-14 15:29:23 +000060 const auto *ND = Result.Nodes.getNodeAs<NamespaceDecl>("namespace");
Alexander Kornienkobef51cd2014-05-19 16:39:08 +000061 const SourceManager &Sources = *Result.SourceManager;
62
63 if (!locationsInSameFile(Sources, ND->getLocStart(), ND->getRBraceLoc()))
64 return;
65
66 // Don't require closing comments for namespaces spanning less than certain
67 // number of lines.
68 unsigned StartLine = Sources.getSpellingLineNumber(ND->getLocStart());
69 unsigned EndLine = Sources.getSpellingLineNumber(ND->getRBraceLoc());
70 if (EndLine - StartLine + 1 <= ShortNamespaceLines)
71 return;
72
73 // Find next token after the namespace closing brace.
74 SourceLocation AfterRBrace = ND->getRBraceLoc().getLocWithOffset(1);
75 SourceLocation Loc = AfterRBrace;
76 Token Tok;
77 // Skip whitespace until we find the next token.
Gabor Horvathafad84c2016-09-24 02:13:45 +000078 while (Lexer::getRawToken(Loc, Tok, Sources, getLangOpts()) ||
Alexander Kornienko2a538302015-12-16 15:44:42 +000079 Tok.is(tok::semi)) {
Alexander Kornienkobef51cd2014-05-19 16:39:08 +000080 Loc = Loc.getLocWithOffset(1);
81 }
82 if (!locationsInSameFile(Sources, ND->getRBraceLoc(), Loc))
83 return;
84
85 bool NextTokenIsOnSameLine = Sources.getSpellingLineNumber(Loc) == EndLine;
Alexander Kornienko16ff5c72014-05-19 17:46:28 +000086 // If we insert a line comment before the token in the same line, we need
87 // to insert a line break.
Alexander Kornienkobef51cd2014-05-19 16:39:08 +000088 bool NeedLineBreak = NextTokenIsOnSameLine && Tok.isNot(tok::eof);
89
Alexander Kornienkoda4ebb22015-03-05 14:56:11 +000090 SourceRange OldCommentRange(AfterRBrace, AfterRBrace);
Daniel Jasper08201e32015-03-05 23:17:32 +000091 std::string Message = "%0 not terminated with a closing comment";
Alexander Kornienkoda4ebb22015-03-05 14:56:11 +000092
Alexander Kornienkobef51cd2014-05-19 16:39:08 +000093 // Try to find existing namespace closing comment on the same line.
94 if (Tok.is(tok::comment) && NextTokenIsOnSameLine) {
95 StringRef Comment(Sources.getCharacterData(Loc), Tok.getLength());
Alexander Kornienko27f126b2014-10-16 15:11:54 +000096 SmallVector<StringRef, 7> Groups;
Alexander Kornienkobef51cd2014-05-19 16:39:08 +000097 if (NamespaceCommentPattern.match(Comment, &Groups)) {
Alexander Kornienko27f126b2014-10-16 15:11:54 +000098 StringRef NamespaceNameInComment = Groups.size() > 5 ? Groups[5] : "";
99 StringRef Anonymous = Groups.size() > 3 ? Groups[3] : "";
Alexander Kornienkobef51cd2014-05-19 16:39:08 +0000100
101 // Check if the namespace in the comment is the same.
102 if ((ND->isAnonymousNamespace() && NamespaceNameInComment.empty()) ||
Alexander Kornienko27f126b2014-10-16 15:11:54 +0000103 (ND->getNameAsString() == NamespaceNameInComment &&
104 Anonymous.empty())) {
Alexander Kornienkobef51cd2014-05-19 16:39:08 +0000105 // FIXME: Maybe we need a strict mode, where we always fix namespace
106 // comments with different format.
107 return;
108 }
109
110 // Otherwise we need to fix the comment.
111 NeedLineBreak = Comment.startswith("/*");
Alexander Kornienkoda4ebb22015-03-05 14:56:11 +0000112 OldCommentRange =
113 SourceRange(AfterRBrace, Loc.getLocWithOffset(Tok.getLength()));
114 Message =
115 (llvm::Twine(
116 "%0 ends with a comment that refers to a wrong namespace '") +
Mandeep Singh Grang7c7ea7d2016-11-08 07:50:19 +0000117 NamespaceNameInComment + "'")
118 .str();
Alexander Kornienkoda4ebb22015-03-05 14:56:11 +0000119 } else if (Comment.startswith("//")) {
120 // Assume that this is an unrecognized form of a namespace closing line
121 // comment. Replace it.
Alexander Kornienkobef51cd2014-05-19 16:39:08 +0000122 NeedLineBreak = false;
Alexander Kornienkoda4ebb22015-03-05 14:56:11 +0000123 OldCommentRange =
124 SourceRange(AfterRBrace, Loc.getLocWithOffset(Tok.getLength()));
125 Message = "%0 ends with an unrecognized comment";
126 }
127 // If it's a block comment, just move it to the next line, as it can be
128 // multi-line or there may be other tokens behind it.
Alexander Kornienkobef51cd2014-05-19 16:39:08 +0000129 }
130
Alexander Kornienkob23eb5e2014-11-17 17:32:32 +0000131 std::string NamespaceName =
132 ND->isAnonymousNamespace()
133 ? "anonymous namespace"
134 : ("namespace '" + ND->getNameAsString() + "'");
135
Alexander Kornienkoda4ebb22015-03-05 14:56:11 +0000136 diag(AfterRBrace, Message)
Alexander Kornienko66d7e302015-04-08 12:54:57 +0000137 << NamespaceName << FixItHint::CreateReplacement(
138 CharSourceRange::getCharRange(OldCommentRange),
139 std::string(SpacesBeforeComments, ' ') +
Alexander Kornienkoda4ebb22015-03-05 14:56:11 +0000140 getNamespaceComment(ND, NeedLineBreak));
Alexander Kornienkob23eb5e2014-11-17 17:32:32 +0000141 diag(ND->getLocation(), "%0 starts here", DiagnosticIDs::Note)
142 << NamespaceName;
Alexander Kornienkobef51cd2014-05-19 16:39:08 +0000143}
144
Alexander Kornienko33fc3db2014-09-22 10:41:39 +0000145} // namespace readability
Alexander Kornienkobef51cd2014-05-19 16:39:08 +0000146} // namespace tidy
147} // namespace clang