2018-11-17 10:37:21 +08:00
|
|
|
//===--- FunctionNamingCheck.cpp - clang-tidy -----------------------------===//
|
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2018-11-17 10:37:21 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "FunctionNamingCheck.h"
|
|
|
|
#include "clang/AST/ASTContext.h"
|
|
|
|
#include "clang/ASTMatchers/ASTMatchFinder.h"
|
|
|
|
#include "llvm/Support/Regex.h"
|
|
|
|
|
|
|
|
using namespace clang::ast_matchers;
|
|
|
|
|
|
|
|
namespace clang {
|
|
|
|
namespace tidy {
|
|
|
|
namespace google {
|
|
|
|
namespace objc {
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
std::string validFunctionNameRegex(bool RequirePrefix) {
|
|
|
|
// Allow the following name patterns for all functions:
|
|
|
|
// • ABFoo (prefix + UpperCamelCase)
|
|
|
|
// • ABURL (prefix + capitalized acronym/initialism)
|
|
|
|
//
|
|
|
|
// If no prefix is required, additionally allow the following name patterns:
|
|
|
|
// • Foo (UpperCamelCase)
|
|
|
|
// • URL (capitalized acronym/initialism)
|
|
|
|
//
|
|
|
|
// The function name following the prefix can contain standard and
|
|
|
|
// non-standard capitalized character sequences including acronyms,
|
|
|
|
// initialisms, and prefixes of symbols (e.g., UIColorFromNSString). For this
|
|
|
|
// reason, the regex only verifies that the function name after the prefix
|
|
|
|
// begins with a capital letter followed by an arbitrary sequence of
|
|
|
|
// alphanumeric characters.
|
|
|
|
//
|
|
|
|
// If a prefix is required, the regex checks for a capital letter followed by
|
|
|
|
// another capital letter or number that is part of the prefix and another
|
|
|
|
// capital letter or number that begins the name following the prefix.
|
|
|
|
std::string FunctionNameMatcher =
|
|
|
|
std::string(RequirePrefix ? "[A-Z][A-Z0-9]+" : "") + "[A-Z][a-zA-Z0-9]*";
|
|
|
|
return std::string("::(") + FunctionNameMatcher + ")$";
|
|
|
|
}
|
|
|
|
|
|
|
|
/// For now we will only fix functions of static storage class with names like
|
|
|
|
/// 'functionName' or 'function_name' and convert them to 'FunctionName'. For
|
|
|
|
/// other cases the user must determine an appropriate name on their own.
|
|
|
|
FixItHint generateFixItHint(const FunctionDecl *Decl) {
|
|
|
|
// A fixit can be generated for functions of static storage class but
|
|
|
|
// otherwise the check cannot determine the appropriate function name prefix
|
|
|
|
// to use.
|
2021-01-05 06:17:45 +08:00
|
|
|
if (Decl->getStorageClass() != SC_Static)
|
2018-11-17 10:37:21 +08:00
|
|
|
return FixItHint();
|
|
|
|
|
|
|
|
StringRef Name = Decl->getName();
|
|
|
|
std::string NewName = Decl->getName().str();
|
|
|
|
|
|
|
|
size_t Index = 0;
|
|
|
|
bool AtWordBoundary = true;
|
|
|
|
while (Index < NewName.size()) {
|
2021-01-29 07:49:53 +08:00
|
|
|
char Ch = NewName[Index];
|
|
|
|
if (isalnum(Ch)) {
|
2018-11-17 10:37:21 +08:00
|
|
|
// Capitalize the first letter after every word boundary.
|
|
|
|
if (AtWordBoundary) {
|
|
|
|
NewName[Index] = toupper(NewName[Index]);
|
|
|
|
AtWordBoundary = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Advance the index after every alphanumeric character.
|
|
|
|
Index++;
|
|
|
|
} else {
|
|
|
|
// Strip out any characters other than alphanumeric characters.
|
|
|
|
NewName.erase(Index, 1);
|
|
|
|
AtWordBoundary = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Generate a fixit hint if the new name is different.
|
|
|
|
if (NewName != Name)
|
|
|
|
return FixItHint::CreateReplacement(
|
|
|
|
CharSourceRange::getTokenRange(SourceRange(Decl->getLocation())),
|
|
|
|
llvm::StringRef(NewName));
|
|
|
|
|
|
|
|
return FixItHint();
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
void FunctionNamingCheck::registerMatchers(MatchFinder *Finder) {
|
2019-02-21 08:34:01 +08:00
|
|
|
// Enforce Objective-C function naming conventions on all functions except:
|
|
|
|
// • Functions defined in system headers.
|
|
|
|
// • C++ member functions.
|
|
|
|
// • Namespaced functions.
|
|
|
|
// • Implicitly defined functions.
|
|
|
|
// • The main function.
|
2018-11-17 10:37:21 +08:00
|
|
|
Finder->addMatcher(
|
|
|
|
functionDecl(
|
2018-12-05 07:40:42 +08:00
|
|
|
unless(anyOf(isExpansionInSystemHeader(), cxxMethodDecl(),
|
2019-02-21 08:34:01 +08:00
|
|
|
hasAncestor(namespaceDecl()), isMain(), isImplicit(),
|
2018-12-05 07:40:42 +08:00
|
|
|
matchesName(validFunctionNameRegex(true)),
|
2018-11-17 10:37:21 +08:00
|
|
|
allOf(isStaticStorageClass(),
|
|
|
|
matchesName(validFunctionNameRegex(false))))))
|
|
|
|
.bind("function"),
|
|
|
|
this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void FunctionNamingCheck::check(const MatchFinder::MatchResult &Result) {
|
|
|
|
const auto *MatchedDecl = Result.Nodes.getNodeAs<FunctionDecl>("function");
|
|
|
|
|
2021-01-05 06:17:45 +08:00
|
|
|
bool IsGlobal = MatchedDecl->getStorageClass() != SC_Static;
|
2018-11-17 10:37:21 +08:00
|
|
|
diag(MatchedDecl->getLocation(),
|
2018-12-14 11:13:31 +08:00
|
|
|
"%select{static function|function in global namespace}1 named %0 must "
|
|
|
|
"%select{be in|have an appropriate prefix followed by}1 Pascal case as "
|
|
|
|
"required by Google Objective-C style guide")
|
|
|
|
<< MatchedDecl << IsGlobal << generateFixItHint(MatchedDecl);
|
2018-11-17 10:37:21 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace objc
|
|
|
|
} // namespace google
|
|
|
|
} // namespace tidy
|
|
|
|
} // namespace clang
|