#ifndef BOOST_CONTRACT_DETAIL_DESTRUCTOR_HPP_ #define BOOST_CONTRACT_DETAIL_DESTRUCTOR_HPP_ // Copyright (C) 2008-2018 Lorenzo Caminiti // Distributed under the Boost Software License, Version 1.0 (see accompanying // file LICENSE_1_0.txt or a copy at http://www.boost.org/LICENSE_1_0.txt). // See: http://www.boost.org/doc/libs/release/libs/contract/doc/html/index.html #include #include #include #include #include #if !defined(BOOST_CONTRACT_ALL_DISABLE_NO_ASSERTION) && ( \ !defined(BOOST_CONTRACT_NO_INVARIANTS) || \ !defined(BOOST_CONTRACT_NO_POSTCONDITIONS) || \ !defined(BOOST_CONTRACT_NO_EXCEPTS)) #include #endif #if !defined(BOOST_CONTRACT_NO_EXIT_INVARIANTS) || \ !defined(BOOST_CONTRACT_NO_POSTCONDITIONS) || \ !defined(BOOST_CONTRACT_NO_EXCEPTS) #include #include #endif namespace boost { namespace contract { namespace detail { // Dtor subcontracting impl via C++ obj destruction mechanism. template // Non-copyable base. class destructor : public cond_inv { public: explicit destructor(C* obj) : cond_inv( boost::contract::from_destructor, obj) {} private: #if !defined(BOOST_CONTRACT_NO_ENTRY_INVARIANTS) || \ !defined(BOOST_CONTRACT_NO_OLDS) void init() /* override */ { #ifndef BOOST_CONTRACT_ALL_DISABLE_NO_ASSERTION if(checking::already()) return; #endif #ifndef BOOST_CONTRACT_NO_ENTRY_INVARIANTS { #ifndef BOOST_CONTRACT_ALL_DISABLE_NO_ASSERTION checking k; #endif // Obj exists (before dtor body), check static and non- inv. this->check_entry_all_inv(); // Dtor cannot have pre because it has no parameters. } #endif #ifndef BOOST_CONTRACT_NO_OLDS this->copy_old(); #endif } #endif public: #if !defined(BOOST_CONTRACT_NO_EXIT_INVARIANTS) || \ !defined(BOOST_CONTRACT_NO_POSTCONDITIONS) || \ !defined(BOOST_CONTRACT_NO_EXCEPTS) ~destructor() BOOST_NOEXCEPT_IF(false) { this->assert_initialized(); #ifndef BOOST_CONTRACT_ALL_DISABLE_NO_ASSERTION if(checking::already()) return; checking k; #endif // If dtor body threw, obj still exists so check subcontracted // static and non- inv (but no post because of throw). Otherwise, // obj destructed so check static inv and post (even if there is no // obj after dtor body, this library allows dtor post, for example // to check static members for an instance counter class). // NOTE: In theory C++ destructors should not throw, but the // language allows for that (even if in C++11 dtors declarations are // implicitly noexcept(true) unless specified otherwise) so this // library must handle such a case. if(uncaught_exception()) { #ifndef BOOST_CONTRACT_NO_EXIT_INVARIANTS this->check_exit_all_inv(); #endif #ifndef BOOST_CONTRACT_NO_EXCEPTS this->check_except(); #endif } else { #ifndef BOOST_CONTRACT_NO_EXIT_INVARIANTS this->check_exit_static_inv(); #endif #ifndef BOOST_CONTRACT_NO_POSTCONDITIONS this->check_post(none()); #endif } } #endif }; } } } // namespace #endif // #include guard