omtcyf0 updated this revision to Diff 48613. omtcyf0 added a comment. Thanks for a review, Richard! Fixed all the issues you pointed to!
Thanks for the hint, Eugene! I'll try to add this functionality to this check later on. http://reviews.llvm.org/D17484 Files: clang-tidy/modernize/CMakeLists.txt clang-tidy/modernize/DeprecatedHeadersCheck.cpp clang-tidy/modernize/DeprecatedHeadersCheck.h clang-tidy/modernize/ModernizeTidyModule.cpp docs/clang-tidy/checks/list.rst docs/clang-tidy/checks/modernize-deprecated-headers.rst test/clang-tidy/modernize-deprecated-headers-cxx03.cpp test/clang-tidy/modernize-deprecated-headers-cxx11.cpp
Index: test/clang-tidy/modernize-deprecated-headers-cxx11.cpp =================================================================== --- /dev/null +++ test/clang-tidy/modernize-deprecated-headers-cxx11.cpp @@ -0,0 +1,109 @@ +// RUN: %check_clang_tidy %s modernize-deprecated-headers %t -- -- -std=c++11 -isystem %S/Inputs/Headers + +#include <assert.h> +#include <complex.h> +#include <ctype.h> +#include <errno.h> +#include <fenv.h> +#include <float.h> +#include <inttypes.h> +#include <iso646.h> +#include <limits.h> +#include <locale.h> +#include <math.h> +#include <setjmp.h> +#include <signal.h> +#include <stdalign.h> +#include <stdarg.h> +#include <stdbool.h> +#include <stddef.h> +#include <stdint.h> +#include <stdio.h> +#include <stdlib.h> +#include <string.h> +#include <tgmath.h> +#include <time.h> +#include <uchar.h> +#include <wchar.h> +#include <wctype.h> + +// CHECK-FIXES: #include <cassert> +// CHECK-FIXES-NEXT: #include <ccomplex> +// CHECK-FIXES-NEXT: #include <cctype> +// CHECK-FIXES-NEXT: #include <cerrno> +// CHECK-FIXES-NEXT: #include <cfenv> +// CHECK-FIXES-NEXT: #include <cfloat> +// CHECK-FIXES-NEXT: #include <cinttypes> +// CHECK-FIXES-NEXT: #include <ciso646> +// CHECK-FIXES-NEXT: #include <climits> +// CHECK-FIXES-NEXT: #include <clocale> +// CHECK-FIXES-NEXT: #include <cmath> +// CHECK-FIXES-NEXT: #include <csetjmp> +// CHECK-FIXES-NEXT: #include <csignal> +// CHECK-FIXES-NEXT: #include <cstdalign> +// CHECK-FIXES-NEXT: #include <cstdarg> +// CHECK-FIXES-NEXT: #include <cstdbool> +// CHECK-FIXES-NEXT: #include <cstddef> +// CHECK-FIXES-NEXT: #include <cstdint> +// CHECK-FIXES-NEXT: #include <cstdio> +// CHECK-FIXES-NEXT: #include <cstdlib> +// CHECK-FIXES-NEXT: #include <cstring> +// CHECK-FIXES-NEXT: #include <ctgmath> +// CHECK-FIXES-NEXT: #include <ctime> +// CHECK-FIXES-NEXT: #include <cuchar> +// CHECK-FIXES-NEXT: #include <cwchar> +// CHECK-FIXES-NEXT: #include <cwctype> + +#include "assert.h" +#include "complex.h" +#include "ctype.h" +#include "errno.h" +#include "fenv.h" +#include "float.h" +#include "inttypes.h" +#include "iso646.h" +#include "limits.h" +#include "locale.h" +#include "math.h" +#include "setjmp.h" +#include "signal.h" +#include "stdalign.h" +#include "stdarg.h" +#include "stdbool.h" +#include "stddef.h" +#include "stdint.h" +#include "stdio.h" +#include "stdlib.h" +#include "string.h" +#include "tgmath.h" +#include "time.h" +#include "uchar.h" +#include "wchar.h" +#include "wctype.h" + +// CHECK-FIXES: #include <cassert> +// CHECK-FIXES-NEXT: #include <ccomplex> +// CHECK-FIXES-NEXT: #include <cctype> +// CHECK-FIXES-NEXT: #include <cerrno> +// CHECK-FIXES-NEXT: #include <cfenv> +// CHECK-FIXES-NEXT: #include <cfloat> +// CHECK-FIXES-NEXT: #include <cinttypes> +// CHECK-FIXES-NEXT: #include <ciso646> +// CHECK-FIXES-NEXT: #include <climits> +// CHECK-FIXES-NEXT: #include <clocale> +// CHECK-FIXES-NEXT: #include <cmath> +// CHECK-FIXES-NEXT: #include <csetjmp> +// CHECK-FIXES-NEXT: #include <csignal> +// CHECK-FIXES-NEXT: #include <cstdalign> +// CHECK-FIXES-NEXT: #include <cstdarg> +// CHECK-FIXES-NEXT: #include <cstdbool> +// CHECK-FIXES-NEXT: #include <cstddef> +// CHECK-FIXES-NEXT: #include <cstdint> +// CHECK-FIXES-NEXT: #include <cstdio> +// CHECK-FIXES-NEXT: #include <cstdlib> +// CHECK-FIXES-NEXT: #include <cstring> +// CHECK-FIXES-NEXT: #include <ctgmath> +// CHECK-FIXES-NEXT: #include <ctime> +// CHECK-FIXES-NEXT: #include <cuchar> +// CHECK-FIXES-NEXT: #include <cwchar> +// CHECK-FIXES-NEXT: #include <cwctype> Index: test/clang-tidy/modernize-deprecated-headers-cxx03.cpp =================================================================== --- /dev/null +++ test/clang-tidy/modernize-deprecated-headers-cxx03.cpp @@ -0,0 +1,103 @@ +// RUN: %check_clang_tidy %s modernize-deprecated-headers %t -- -- -std=c++03 -isystem %S/Inputs/Headers + +#include <assert.h> +#include <complex.h> +#include <ctype.h> +#include <errno.h> +#include <float.h> +#include <inttypes.h> +#include <iso646.h> +#include <limits.h> +#include <locale.h> +#include <math.h> +#include <setjmp.h> +#include <signal.h> +#include <stdarg.h> +#include <stddef.h> +#include <stdint.h> +#include <stdio.h> +#include <stdlib.h> +#include <string.h> +#include <time.h> +#include <wchar.h> +#include <wctype.h> + +// Headers deprecated since C++11; expect no diagnostics +#include <fenv.h> +#include <stdalign.h> +#include <stdbool.h> +#include <tgmath.h> +#include <uchar.h> + +// CHECK-FIXES: #include <cassert> +// CHECK-FIXES-NEXT: #include <ccomplex> +// CHECK-FIXES-NEXT: #include <cctype> +// CHECK-FIXES-NEXT: #include <cerrno> +// CHECK-FIXES-NEXT: #include <cfloat> +// CHECK-FIXES-NEXT: #include <cinttypes> +// CHECK-FIXES-NEXT: #include <ciso646> +// CHECK-FIXES-NEXT: #include <climits> +// CHECK-FIXES-NEXT: #include <clocale> +// CHECK-FIXES-NEXT: #include <cmath> +// CHECK-FIXES-NEXT: #include <csetjmp> +// CHECK-FIXES-NEXT: #include <csignal> +// CHECK-FIXES-NEXT: #include <cstdarg> +// CHECK-FIXES-NEXT: #include <cstddef> +// CHECK-FIXES-NEXT: #include <cstdint> +// CHECK-FIXES-NEXT: #include <cstdio> +// CHECK-FIXES-NEXT: #include <cstdlib> +// CHECK-FIXES-NEXT: #include <cstring> +// CHECK-FIXES-NEXT: #include <ctime> +// CHECK-FIXES-NEXT: #include <cwchar> +// CHECK-FIXES-NEXT: #include <cwctype> + +#include "assert.h" +#include "complex.h" +#include "ctype.h" +#include "errno.h" +#include "float.h" +#include "inttypes.h" +#include "iso646.h" +#include "limits.h" +#include "locale.h" +#include "math.h" +#include "setjmp.h" +#include "signal.h" +#include "stdarg.h" +#include "stddef.h" +#include "stdint.h" +#include "stdio.h" +#include "stdlib.h" +#include "string.h" +#include "time.h" +#include "wchar.h" +#include "wctype.h" + +// Headers deprecated since C++11; expect no diagnostics +#include "fenv.h" +#include "stdalign.h" +#include "stdbool.h" +#include "tgmath.h" +#include "uchar.h" + +// CHECK-FIXES: #include <cassert> +// CHECK-FIXES-NEXT: #include <ccomplex> +// CHECK-FIXES-NEXT: #include <cctype> +// CHECK-FIXES-NEXT: #include <cerrno> +// CHECK-FIXES-NEXT: #include <cfloat> +// CHECK-FIXES-NEXT: #include <cinttypes> +// CHECK-FIXES-NEXT: #include <ciso646> +// CHECK-FIXES-NEXT: #include <climits> +// CHECK-FIXES-NEXT: #include <clocale> +// CHECK-FIXES-NEXT: #include <cmath> +// CHECK-FIXES-NEXT: #include <csetjmp> +// CHECK-FIXES-NEXT: #include <csignal> +// CHECK-FIXES-NEXT: #include <cstdarg> +// CHECK-FIXES-NEXT: #include <cstddef> +// CHECK-FIXES-NEXT: #include <cstdint> +// CHECK-FIXES-NEXT: #include <cstdio> +// CHECK-FIXES-NEXT: #include <cstdlib> +// CHECK-FIXES-NEXT: #include <cstring> +// CHECK-FIXES-NEXT: #include <ctime> +// CHECK-FIXES-NEXT: #include <cwchar> +// CHECK-FIXES-NEXT: #include <cwctype> Index: docs/clang-tidy/checks/modernize-deprecated-headers.rst =================================================================== --- /dev/null +++ docs/clang-tidy/checks/modernize-deprecated-headers.rst @@ -0,0 +1,62 @@ +.. title:: clang-tidy - modernize-deprecated-headers + +modernize-deprecated-headers +========================== + +This check replaces C standard library headers with their C++ alternatives. + + Annex D (normative) + Compatibility features [depr] + + D.5 C standard library headers [depr.c.headers] + 1 For compatibility with the C standard library and the C Unicode TR, the C++ + standard library provides the 26 C headers, as shown in Table 155 + + ... + + Every C header, each of which has a name of the form name.h, behaves as if + each name placed in the standard library namespace by the corresponding cname + header is placed within the global namespace scope. It is unspecified whether + these names are first declared or defined within namespace scope (3.3.6) of + the namespace std and are then injected into the global namespace scope by + explicit using-declarations (7.3.3). + + [ Example: The header <cstdlib> assuredly provides its declarations and + definitions within the namespace std. It may also provide these names within + the global namespace. The header <stdlib.h> assuredly provides the same + declarations and definitions within the global namespace, much as in the C + Standard. It may also provide these names within the namespace std. + —end example ] + + -- C++14 Standard + +* `<assert.h>` +* `<complex.h>` +* `<ctype.h>` +* `<errno.h>` +* `<fenv.h>` // deprecated since C++11 +* `<float.h>` +* `<inttypes.h>` +* `<iso646.h>` +* `<limits.h>` +* `<locale.h>` +* `<math.h>` +* `<setjmp.h>` +* `<signal.h>` +* `<stdalign.h>` // deprecated since C++11 +* `<stdarg.h>` +* `<stdbool.h>` // deprecated since C++11 +* `<stddef.h>` +* `<stdint.h>` +* `<stdio.h>` +* `<stdlib.h>` +* `<string.h>` +* `<tgmath.h>` // deprecated since C++11 +* `<time.h>` +* `<uchar.h>` // deprecated since C++11 +* `<wchar.h>` +* `<wctype.h>` + +If the specified standard is older than C++11 the check will only replace +headers deprecated before C++11, otherwise -- every header that appeared in +the list. Index: docs/clang-tidy/checks/list.rst =================================================================== --- docs/clang-tidy/checks/list.rst +++ docs/clang-tidy/checks/list.rst @@ -72,6 +72,7 @@ misc-unused-parameters misc-unused-raii misc-virtual-near-miss + modernize-deprecated-headers modernize-loop-convert modernize-make-unique modernize-pass-by-value Index: clang-tidy/modernize/ModernizeTidyModule.cpp =================================================================== --- clang-tidy/modernize/ModernizeTidyModule.cpp +++ clang-tidy/modernize/ModernizeTidyModule.cpp @@ -10,6 +10,7 @@ #include "../ClangTidy.h" #include "../ClangTidyModule.h" #include "../ClangTidyModuleRegistry.h" +#include "DeprecatedHeadersCheck.h" #include "LoopConvertCheck.h" #include "MakeUniqueCheck.h" #include "PassByValueCheck.h" @@ -30,6 +31,8 @@ class ModernizeModule : public ClangTidyModule { public: void addCheckFactories(ClangTidyCheckFactories &CheckFactories) override { + CheckFactories.registerCheck<DeprecatedHeadersCheck>( + "modernize-deprecated-headers"); CheckFactories.registerCheck<LoopConvertCheck>("modernize-loop-convert"); CheckFactories.registerCheck<MakeUniqueCheck>("modernize-make-unique"); CheckFactories.registerCheck<PassByValueCheck>("modernize-pass-by-value"); Index: clang-tidy/modernize/DeprecatedHeadersCheck.h =================================================================== --- /dev/null +++ clang-tidy/modernize/DeprecatedHeadersCheck.h @@ -0,0 +1,42 @@ +//===--- DeprecatedHeadersCheck.h - clang-tidy-------------------*- C++ -*-===// +// +// The LLVM Compiler Infrastructure +// +// This file is distributed under the University of Illinois Open Source +// License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// + +#ifndef LLVM_CLANG_TOOLS_EXTRA_CLANG_TIDY_MODERNIZE_C_HEADERS_TO_CXX_H +#define LLVM_CLANG_TOOLS_EXTRA_CLANG_TIDY_MODERNIZE_C_HEADERS_TO_CXX_H + +#include "../ClangTidy.h" + +namespace clang { +namespace tidy { +namespace modernize { + +/// This check replaces deprecated C library headers with their C++ STL +/// alternatives. +/// +/// Before: +/// #include <header.h> +/// After: +/// #include <cheader> +/// +/// Example: <stdio.h> => <cstdio> +/// +/// For the user-facing documentation see: +/// http://clang.llvm.org/extra/clang-tidy/checks/modernize-deprecated-headers.html +class DeprecatedHeadersCheck : public ClangTidyCheck { +public: + DeprecatedHeadersCheck(StringRef Name, ClangTidyContext *Context) + : ClangTidyCheck(Name, Context) {} + void registerPPCallbacks(CompilerInstance &Compiler) override; +}; + +} // namespace modernize +} // namespace tidy +} // namespace clang + +#endif // LLVM_CLANG_TOOLS_EXTRA_CLANG_TIDY_MODERNIZE_C_HEADERS_TO_CXX_H Index: clang-tidy/modernize/DeprecatedHeadersCheck.cpp =================================================================== --- /dev/null +++ clang-tidy/modernize/DeprecatedHeadersCheck.cpp @@ -0,0 +1,105 @@ +//===--- DeprecatedHeadersCheck.cpp - clang-tidy---------------------------===// +// +// The LLVM Compiler Infrastructure +// +// This file is distributed under the University of Illinois Open Source +// License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// + +#include "DeprecatedHeadersCheck.h" +#include "clang/Frontend/CompilerInstance.h" +#include "clang/Lex/PPCallbacks.h" +#include "clang/Lex/Preprocessor.h" + +#include <iostream> +#include <vector> +#include <map> + +namespace clang { +namespace tidy { +namespace modernize { + +namespace { +class IncludeModernizePPCallbacks : public PPCallbacks { +public: + explicit IncludeModernizePPCallbacks(ClangTidyCheck &Check, + LangOptions LangOpts); + + void InclusionDirective(SourceLocation HashLoc, const Token &IncludeTok, + StringRef FileName, bool IsAngled, + CharSourceRange FilenameRange, const FileEntry *File, + StringRef SearchPath, StringRef RelativePath, + const Module *Imported) override; + +private: + std::map<std::string, std::string> CStyledHeaderToCxx; + + ClangTidyCheck &Check; + LangOptions LangOpts; +}; +} // namespace + +void DeprecatedHeadersCheck::registerPPCallbacks(CompilerInstance &Compiler) { + if (this->getLangOpts().CPlusPlus) { + Compiler.getPreprocessor().addPPCallbacks( + ::llvm::make_unique<IncludeModernizePPCallbacks>(*this, + this->getLangOpts())); + } +} + +IncludeModernizePPCallbacks::IncludeModernizePPCallbacks(ClangTidyCheck &Check, + LangOptions LangOpts) + : Check(Check), LangOpts(LangOpts) { + CStyledHeaderToCxx = { + {"assert.h", "cassert"}, + {"complex.h", "ccomplex"}, + {"ctype.h", "cctype"}, + {"errno.h", "cerrno"}, + {"float.h", "cfloat"}, + {"inttypes.h", "cinttypes"}, + {"iso646.h", "ciso646"}, + {"limits.h", "climits"}, + {"locale.h", "clocale"}, + {"math.h", "cmath"}, + {"setjmp.h", "csetjmp"}, + {"signal.h", "csignal"}, + {"stdarg.h", "cstdarg"}, + {"stddef.h", "cstddef"}, + {"stdint.h", "cstdint"}, + {"stdio.h", "cstdio"}, + {"stdlib.h", "cstdlib"}, + {"string.h", "cstring"}, + {"time.h", "ctime"}, + {"wchar.h", "cwchar"}, + {"wctype.h", "cwctype"}}; + + // Add C++ 11 headers + if (LangOpts.CPlusPlus11) { + std::map<std::string, std::string> Cxx11DeprecatedHeaders = { + {"fenv.h", "cfenv"}, + {"stdalign.h", "cstdalign"}, + {"stdbool.h", "cstdbool"}, + {"tgmath.h", "ctgmath"}, + {"uchar.h", "cuchar"}}; + CStyledHeaderToCxx.insert(begin(Cxx11DeprecatedHeaders), + end(Cxx11DeprecatedHeaders)); + } +} + +void IncludeModernizePPCallbacks::InclusionDirective( + SourceLocation HashLoc, const Token &IncludeTok, StringRef FileName, + bool IsAngled, CharSourceRange FilenameRange, const FileEntry *File, + StringRef SearchPath, StringRef RelativePath, const Module *Imported) { + if (CStyledHeaderToCxx.count(std::string(FileName)) != 0) { + std::string Replacement = + "<" + CStyledHeaderToCxx[std::string(FileName)] + ">"; + Check.diag(FilenameRange.getBegin(), "including deprecated C++ header") + << FixItHint::CreateReplacement(FilenameRange.getAsRange(), + Replacement); + } +} + +} // namespace modernize +} // namespace tidy +} // namespace clang Index: clang-tidy/modernize/CMakeLists.txt =================================================================== --- clang-tidy/modernize/CMakeLists.txt +++ clang-tidy/modernize/CMakeLists.txt @@ -1,6 +1,7 @@ set(LLVM_LINK_COMPONENTS support) add_clang_library(clangTidyModernizeModule + DeprecatedHeadersCheck.cpp LoopConvertCheck.cpp LoopConvertUtils.cpp MakeUniqueCheck.cpp
_______________________________________________ cfe-commits mailing list cfe-commits@lists.llvm.org http://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits