/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #ifndef INCLUDED_COMPILERPLUGINS_CLANG_CHECK_HXX #define INCLUDED_COMPILERPLUGINS_CLANG_CHECK_HXX #include #include #include #include #include namespace loplugin { class ContextCheck; class TerminalCheck; namespace detail { inline ContextCheck checkRecordDecl( clang::Decl const * decl, clang::TagTypeKind tag, llvm::StringRef id); } class TypeCheck { public: explicit TypeCheck(clang::QualType type): type_(type) {} explicit TypeCheck(clang::Type const * type): type_(type, 0) {} explicit TypeCheck(clang::TypeDecl const * decl): type_(decl->getTypeForDecl(), 0) {} explicit operator bool() const { return !type_.isNull(); } TypeCheck NonConst() const; TypeCheck NonConstVolatile() const; TypeCheck Const() const; TypeCheck Volatile() const; TypeCheck ConstVolatile() const; TerminalCheck Void() const; TerminalCheck Char() const; TerminalCheck AnyBoolean() const; TypeCheck Pointer() const; TerminalCheck Enum() const; TypeCheck LvalueReference() const; inline ContextCheck Class(llvm::StringRef id) const; inline ContextCheck Struct(llvm::StringRef id) const; TypeCheck Typedef() const; inline ContextCheck Typedef(llvm::StringRef id) const; TypeCheck NotSubstTemplateTypeParmType() const; private: TypeCheck() = default; clang::QualType const type_{}; }; class DeclCheck { public: explicit DeclCheck(clang::Decl const * decl): decl_(decl) {} explicit operator bool() const { return decl_ != nullptr; } inline ContextCheck Class(llvm::StringRef id) const; inline ContextCheck Struct(llvm::StringRef id) const; inline ContextCheck Union(llvm::StringRef id) const; inline ContextCheck Function(llvm::StringRef id) const; ContextCheck Operator(clang::OverloadedOperatorKind op) const; inline ContextCheck Var(llvm::StringRef id) const; ContextCheck MemberFunction() const; private: clang::Decl const * const decl_; }; class ContextCheck { public: explicit ContextCheck(clang::DeclContext const * context = nullptr): context_(context) {} explicit operator bool() const { return context_ != nullptr; } TerminalCheck GlobalNamespace() const; inline ContextCheck Namespace(llvm::StringRef id) const; TerminalCheck StdNamespace() const; TerminalCheck StdOrNestedNamespace() const; ContextCheck AnonymousNamespace() const; inline ContextCheck Class(llvm::StringRef id) const; inline ContextCheck Struct(llvm::StringRef id) const; explicit ContextCheck(const clang::NamespaceDecl * decl ) : context_( decl ) {} private: clang::DeclContext const * const context_; }; class TerminalCheck { public: explicit operator bool() const { return satisfied_; } private: friend ContextCheck; friend TypeCheck; explicit TerminalCheck(bool satisfied): satisfied_(satisfied) {} bool const satisfied_; }; typedef std::function DeclChecker; // Returns true if the class has a base matching the checker, or if the class itself matches. bool isDerivedFrom(const clang::CXXRecordDecl *decl, DeclChecker base); namespace detail { ContextCheck checkRecordDecl( clang::Decl const * decl, clang::TagTypeKind tag, llvm::StringRef id) { auto r = llvm::dyn_cast_or_null(decl); if (r != nullptr && r->getTagKind() == tag) { auto const i = r->getIdentifier(); if (i != nullptr && i->getName() == id) { return ContextCheck(r->getDeclContext()); } } return ContextCheck(); } } ContextCheck TypeCheck::Class(llvm::StringRef id) const { if (!type_.isNull()) { auto const t = type_->getAs(); if (t != nullptr) { return detail::checkRecordDecl(t->getDecl(), clang::TTK_Class, id); } } return ContextCheck(); } ContextCheck TypeCheck::Struct(llvm::StringRef id) const { if (!type_.isNull()) { auto const t = type_->getAs(); if (t != nullptr) { return detail::checkRecordDecl(t->getDecl(), clang::TTK_Struct, id); } } return ContextCheck(); } ContextCheck TypeCheck::Typedef(llvm::StringRef id) const { if (!type_.isNull()) { if (auto const t = type_->getAs()) { auto const d = t->getDecl(); auto const i = d->getIdentifier(); assert(i != nullptr); if (i->getName() == id) { return ContextCheck(d->getDeclContext()); } } } return ContextCheck(); } ContextCheck DeclCheck::Class(llvm::StringRef id) const { return detail::checkRecordDecl(decl_, clang::TTK_Class, id); } ContextCheck DeclCheck::Struct(llvm::StringRef id) const { return detail::checkRecordDecl(decl_, clang::TTK_Struct, id); } ContextCheck DeclCheck::Union(llvm::StringRef id) const { return detail::checkRecordDecl(decl_, clang::TTK_Union, id); } ContextCheck DeclCheck::Function(llvm::StringRef id) const { auto f = llvm::dyn_cast_or_null(decl_); if (f != nullptr) { auto const i = f->getIdentifier(); if (i != nullptr && i->getName() == id) { return ContextCheck(f->getDeclContext()); } } return ContextCheck(); } ContextCheck DeclCheck::Var(llvm::StringRef id) const { auto f = llvm::dyn_cast_or_null(decl_); if (f != nullptr) { auto const i = f->getIdentifier(); if (i != nullptr && i->getName() == id) { return ContextCheck(f->getDeclContext()); } } return ContextCheck(); } ContextCheck ContextCheck::Namespace(llvm::StringRef id) const { if (context_) { auto n = llvm::dyn_cast(context_); if (n != nullptr) { auto const i = n->getIdentifier(); if (i != nullptr && i->getName() == id) { return ContextCheck(n->getParent()); } } } return ContextCheck(); } ContextCheck ContextCheck::Class(llvm::StringRef id) const { return detail::checkRecordDecl( llvm::dyn_cast_or_null(context_), clang::TTK_Class, id); } ContextCheck ContextCheck::Struct(llvm::StringRef id) const { return detail::checkRecordDecl( llvm::dyn_cast_or_null(context_), clang::TTK_Struct, id); } bool isExtraWarnUnusedType(clang::QualType type); bool isOkToRemoveArithmeticCast( clang::ASTContext & context, clang::QualType t1, clang::QualType t2, const clang::Expr* subExpr); } #endif /* vim:set shiftwidth=4 softtabstop=4 expandtab: */