aaronpuchert created this revision. aaronpuchert added reviewers: aaron.ballman, efriedma, rsmith. aaronpuchert requested review of this revision. Herald added a project: clang. Herald added a subscriber: cfe-commits.
Being inline doesn't change anything about a function having internal linkage, see [basic.link]p3 for 'static' and p4 for unnamed namespaces. An internal linkage function can only be used in the same translation unit, so if it's not being used or needed, that's suspicious. Also don't recommend "static inline" since for all intents and purposes it has the same effect as just using "static", which should thus be preferred. Repository: rG LLVM Github Monorepo https://reviews.llvm.org/D92886 Files: clang/include/clang/Basic/DiagnosticSemaKinds.td clang/lib/Sema/Sema.cpp clang/lib/Sema/SemaDecl.cpp clang/test/SemaCXX/warn-static-function-inheader.cpp clang/test/SemaCXX/warn-static-function-inheader.h clang/test/SemaCXX/warn-unused-filescoped.cpp
Index: clang/test/SemaCXX/warn-unused-filescoped.cpp =================================================================== --- clang/test/SemaCXX/warn-unused-filescoped.cpp +++ clang/test/SemaCXX/warn-unused-filescoped.cpp @@ -5,26 +5,26 @@ #ifdef HEADER static void headerstatic() {} // expected-warning{{unused function 'headerstatic'}} -static inline void headerstaticinline() {} +static inline void headerstaticinline() {} // expected-warning{{unused function 'headerstaticinline'}} namespace { void headeranon() {} // expected-warning{{unused function 'headeranon'}} -inline void headerinlineanon() {} +inline void headerinlineanon() {} // expected-warning{{unused function 'headerinlineanon'}} } namespace test7 { template<typename T> - static inline void foo(T) { } + static inline void foo(T) { } // expected-warning{{unused function template 'foo'}} - // This should not emit an unused-function warning since it inherits + // This should emit an unused-function warning since it inherits // the static storage type from the base template. template<> - inline void foo(int) { } + inline void foo(int) { } // expected-warning{{unused function 'foo<int>'}} // Partial specialization template<typename T, typename U> - static inline void bar(T, U) { } + static inline void bar(T, U) { } // expected-warning{{unused function template 'bar'}} template<typename U> inline void bar(int, U) { } @@ -35,7 +35,7 @@ namespace pr19713 { #if __cplusplus >= 201103L - static constexpr int constexpr1() { return 1; } + static constexpr int constexpr1() { return 1; } // expected-warning{{unused function 'constexpr1'}} constexpr int constexpr2() { return 2; } #endif } Index: clang/test/SemaCXX/warn-static-function-inheader.h =================================================================== --- clang/test/SemaCXX/warn-static-function-inheader.h +++ clang/test/SemaCXX/warn-static-function-inheader.h @@ -1,3 +1,13 @@ -static void thing(void) { // expected-warning {{'static' function 'thing' declared in header file should be declared 'static inline'}} +static void thing(void) { // expected-warning {{function 'thing' declared in header file should be declared 'inline' instead of 'static'}} } +static inline void inline_thing(void) { // expected-warning {{function 'inline_thing' declared in header file should not be declared 'static'}} +} + +namespace { +void anon_thing(void) { // expected-warning {{function 'anon_thing' declared in header file should be declared 'inline' instead of in an anonymous namespace}} +} + +inline void anon_inline_thing(void) { // expected-warning {{function 'anon_inline_thing' declared in header file should not be declared in an anonymous namespace}} +} +} // namespace Index: clang/test/SemaCXX/warn-static-function-inheader.cpp =================================================================== --- clang/test/SemaCXX/warn-static-function-inheader.cpp +++ clang/test/SemaCXX/warn-static-function-inheader.cpp @@ -7,6 +7,9 @@ template <typename T> void foo(void) { - thing(); - another(); + thing(); + inline_thing(); + anon_thing(); + anon_inline_thing(); + another(); } Index: clang/lib/Sema/SemaDecl.cpp =================================================================== --- clang/lib/Sema/SemaDecl.cpp +++ clang/lib/Sema/SemaDecl.cpp @@ -1697,10 +1697,6 @@ if (const CXXMethodDecl *MD = dyn_cast<CXXMethodDecl>(FD)) { if (MD->isVirtual() || IsDisallowedCopyOrAssign(MD)) return false; - } else { - // 'static inline' functions are defined in headers; don't warn. - if (FD->isInlined() && !isMainFileLoc(*this, FD->getLocation())) - return false; } if (FD->doesThisDeclarationHaveABody() && Index: clang/lib/Sema/Sema.cpp =================================================================== --- clang/lib/Sema/Sema.cpp +++ clang/lib/Sema/Sema.cpp @@ -1211,13 +1211,12 @@ Diag(DiagD->getLocation(), diag::warn_unneeded_member_function) << DiagD; else { - if (FD->getStorageClass() == SC_Static && - !FD->isInlineSpecified() && - !SourceMgr.isInMainFile( - SourceMgr.getExpansionLoc(FD->getLocation()))) + if (!SourceMgr.isInMainFile( + SourceMgr.getExpansionLoc(FD->getLocation()))) Diag(DiagD->getLocation(), diag::warn_unneeded_static_internal_decl) - << DiagD; + << DiagD << FD->isInlineSpecified() + << (FD->getStorageClass() == SC_Static); else Diag(DiagD->getLocation(), diag::warn_unneeded_internal_decl) << /*function*/ 0 << DiagD; Index: clang/include/clang/Basic/DiagnosticSemaKinds.td =================================================================== --- clang/include/clang/Basic/DiagnosticSemaKinds.td +++ clang/include/clang/Basic/DiagnosticSemaKinds.td @@ -360,8 +360,9 @@ "%select{function|variable}0 %1 is not needed and will not be emitted">, InGroup<UnneededInternalDecl>, DefaultIgnore; def warn_unneeded_static_internal_decl : Warning< - "'static' function %0 declared in header file " - "should be declared 'static inline'">, + "function %0 declared in header file should " + "%select{be declared 'inline' instead of|not be declared}1 " + "%select{in an anonymous namespace|'static'}2">, InGroup<UnneededInternalDecl>, DefaultIgnore; def warn_unneeded_member_function : Warning< "member function %0 is not needed and will not be emitted">,
_______________________________________________ cfe-commits mailing list cfe-commits@lists.llvm.org https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits