clang-tools 19.0.0git
InlineFunctionDeclCheck.cpp
Go to the documentation of this file.
1//===-- InlineFunctionDeclCheck.cpp ---------------------------------------===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8
10#include "../utils/FileExtensionsUtils.h"
11#include "../utils/LexerUtils.h"
12#include "clang/AST/ASTContext.h"
13#include "clang/ASTMatchers/ASTMatchFinder.h"
14
15#include "llvm/ADT/StringSet.h"
16
17using namespace clang::ast_matchers;
18
19namespace clang::tidy::llvm_libc {
20
21namespace {
22
23const TemplateParameterList *
24getLastTemplateParameterList(const FunctionDecl *FuncDecl) {
25 const TemplateParameterList *ReturnList =
26 FuncDecl->getDescribedTemplateParams();
27
28 if (!ReturnList) {
29 const unsigned NumberOfTemplateParameterLists =
30 FuncDecl->getNumTemplateParameterLists();
31
32 if (NumberOfTemplateParameterLists > 0)
33 ReturnList = FuncDecl->getTemplateParameterList(
34 NumberOfTemplateParameterLists - 1);
35 }
36
37 return ReturnList;
38}
39
40} // namespace
41
43 ClangTidyContext *Context)
44 : ClangTidyCheck(Name, Context),
45 HeaderFileExtensions(Context->getHeaderFileExtensions()) {}
46
48 // Ignore functions that have been deleted.
49 Finder->addMatcher(decl(functionDecl(unless(isDeleted()))).bind("func_decl"),
50 this);
51}
52
53void InlineFunctionDeclCheck::check(const MatchFinder::MatchResult &Result) {
54 const auto *FuncDecl = Result.Nodes.getNodeAs<FunctionDecl>("func_decl");
55
56 // Consider only explicitly or implicitly inline functions.
57 if (FuncDecl == nullptr || !FuncDecl->isInlined())
58 return;
59
60 SourceLocation SrcBegin = FuncDecl->getBeginLoc();
61
62 // If we have a template parameter list, we need to skip that because the
63 // LIBC_INLINE macro must be placed after that.
64 if (const TemplateParameterList *TemplateParams =
65 getLastTemplateParameterList(FuncDecl)) {
66 SrcBegin = TemplateParams->getRAngleLoc();
67 std::optional<Token> NextToken =
69 SrcBegin, *Result.SourceManager, Result.Context->getLangOpts());
70 if (NextToken)
71 SrcBegin = NextToken->getLocation();
72 }
73
74 // Consider functions only in header files.
75 if (!utils::isSpellingLocInHeaderFile(SrcBegin, *Result.SourceManager,
76 HeaderFileExtensions))
77 return;
78
79 // Ignore lambda functions as they are internal and implicit.
80 if (const auto *MethodDecl = dyn_cast<CXXMethodDecl>(FuncDecl))
81 if (MethodDecl->getParent()->isLambda())
82 return;
83
84 // Check if decl starts with LIBC_INLINE
85 auto Loc = FullSourceLoc(Result.SourceManager->getFileLoc(SrcBegin),
86 *Result.SourceManager);
87 llvm::StringRef SrcText = Loc.getBufferData().drop_front(Loc.getFileOffset());
88 if (SrcText.starts_with("LIBC_INLINE"))
89 return;
90
91 diag(SrcBegin, "%0 must be tagged with the LIBC_INLINE macro; the macro "
92 "should be placed at the beginning of the declaration")
93 << FuncDecl << FixItHint::CreateInsertion(Loc, "LIBC_INLINE ");
94}
95
96} // namespace clang::tidy::llvm_libc
llvm::SmallString< 256U > Name
SourceLocation Loc
Base class for all clang-tidy checks.
DiagnosticBuilder diag(SourceLocation Loc, StringRef Description, DiagnosticIDs::Level Level=DiagnosticIDs::Warning)
Add a diagnostic with the check's name.
Every ClangTidyCheck reports errors through a DiagnosticsEngine provided by this context.
void registerMatchers(ast_matchers::MatchFinder *Finder) override
Override this to register AST matchers with Finder.
void check(const ast_matchers::MatchFinder::MatchResult &Result) override
ClangTidyChecks that register ASTMatchers should do the actual work in here.
InlineFunctionDeclCheck(StringRef Name, ClangTidyContext *Context)
std::optional< Token > findNextTokenSkippingComments(SourceLocation Start, const SourceManager &SM, const LangOptions &LangOpts)
Definition: LexerUtils.cpp:111
bool isSpellingLocInHeaderFile(SourceLocation Loc, SourceManager &SM, const FileExtensionsSet &HeaderFileExtensions)
Checks whether spelling location of Loc is in header file.