/////////1/////////2/////////3/////////4/////////5/////////6/////////7/////////8 // test_shared_ptr_multi_base.cpp // (C) Copyright 2002 Robert Ramey- http://www.rrsd.com - David Tonge . // (C) Copyright 2009 Takatoshi Kondo. // Use, modification and distribution is subject to the Boost Software // License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at // http://www.boost.org/LICENSE_1_0.txt) // // See http://www.boost.org for updates, documentation, and revision history. #include // NULL #include // remove #include #include #if defined(BOOST_NO_STDC_NAMESPACE) namespace std{ using ::remove; } #endif #include #include #include #include #include #include #include #include #include #include #include #include #include #include "test_tools.hpp" struct Base1 { Base1() {} Base1(int x) : m_x(1 + x) {} virtual ~Base1() {} int m_x; // serialize friend class boost::serialization::access; template void serialize(Archive &ar, const unsigned int /* file_version */) { ar & BOOST_SERIALIZATION_NVP(m_x); } }; //BOOST_BROKEN_COMPILER_TYPE_TRAITS_SPECIALIZATION(Base1) BOOST_SERIALIZATION_SHARED_PTR(Base1) struct Base2 { Base2() {} Base2(int x) : m_x(2 + x) {} int m_x; virtual ~Base2() {} // serialize friend class boost::serialization::access; template void serialize(Archive &ar, const unsigned int /* file_version */) { ar & BOOST_SERIALIZATION_NVP(m_x); } }; //BOOST_BROKEN_COMPILER_TYPE_TRAITS_SPECIALIZATION(Base2) BOOST_SERIALIZATION_SHARED_PTR(Base2) struct Base3 { Base3() {} Base3(int x) : m_x(3 + x) {} virtual ~Base3() {} int m_x; // serialize friend class boost::serialization::access; template void serialize(Archive &ar, const unsigned int /* file_version */) { ar & BOOST_SERIALIZATION_NVP(m_x); } }; //BOOST_BROKEN_COMPILER_TYPE_TRAITS_SPECIALIZATION(Base3) BOOST_SERIALIZATION_SHARED_PTR(Base3) // Sub is a subclass of Base1, Base1 and Base3. struct Sub:public Base1, public Base2, public Base3 { Sub() {} Sub(int x) : Base1(x), Base2(x), m_x(x) {} virtual ~Sub() {} int m_x; // serialize friend class boost::serialization::access; template void serialize(Archive &ar, const unsigned int /* file_version */) { ar & BOOST_SERIALIZATION_BASE_OBJECT_NVP(Base1); ar & BOOST_SERIALIZATION_BASE_OBJECT_NVP(Base2); ar & BOOST_SERIALIZATION_BASE_OBJECT_NVP(Base3); ar & BOOST_SERIALIZATION_NVP(m_x); } }; // Sub needs to be exported because its serialized via a base class pointer BOOST_CLASS_EXPORT(Sub) BOOST_SERIALIZATION_SHARED_PTR(Sub) template void save2( const char * testfile, const FIRST& first, const SECOND& second ){ test_ostream os(testfile, TEST_STREAM_FLAGS); test_oarchive oa(os, TEST_ARCHIVE_FLAGS); oa << BOOST_SERIALIZATION_NVP(first); oa << BOOST_SERIALIZATION_NVP(second); } template void load2( const char * testfile, FIRST& first, SECOND& second) { test_istream is(testfile, TEST_STREAM_FLAGS); test_iarchive ia(is, TEST_ARCHIVE_FLAGS); ia >> BOOST_SERIALIZATION_NVP(first); ia >> BOOST_SERIALIZATION_NVP(second); } // Run tests by serializing two shared_ptrs into an archive, // clearing them (deleting the objects) and then reloading the // objects back from an archive. // Serialization sequence // First, shared_ptr // Second, weak_ptr template void shared_weak( boost::shared_ptr& first, boost::weak_ptr& second ){ const char * testfile = boost::archive::tmpnam(NULL); BOOST_REQUIRE(NULL != testfile); int firstm = first->m_x; int secondm = second.lock()->m_x; save2(testfile, first, second); // Clear the pointers, thereby destroying the objects they contain second.reset(); first.reset(); load2(testfile, first, second); // Check data member BOOST_CHECK(firstm == first->m_x); BOOST_CHECK(secondm == second.lock()->m_x); // Check pointer to vtable BOOST_CHECK(boost::dynamic_pointer_cast(first)); BOOST_CHECK(boost::dynamic_pointer_cast(second.lock())); std::remove(testfile); } // Serialization sequence // First, weak_ptr // Second, shared_ptr template void weak_shared( boost::weak_ptr& first, boost::shared_ptr& second ){ const char * testfile = boost::archive::tmpnam(NULL); BOOST_REQUIRE(NULL != testfile); int firstm = first.lock()->m_x; int secondm = second->m_x; save2(testfile, first, second); // Clear the pointers, thereby destroying the objects they contain first.reset(); second.reset(); load2(testfile, first, second); // Check data member BOOST_CHECK(firstm == first.lock()->m_x); BOOST_CHECK(secondm == second->m_x); // Check pointer to vtable BOOST_CHECK(boost::dynamic_pointer_cast(first.lock())); BOOST_CHECK(boost::dynamic_pointer_cast(second)); std::remove(testfile); } // This does the tests int test_main(int /* argc */, char * /* argv */[]) { // Both Sub boost::shared_ptr tc1_sp(new Sub(10)); boost::weak_ptr tc1_wp(tc1_sp); shared_weak(tc1_sp, tc1_wp); weak_shared(tc1_wp, tc1_sp); // Sub and Base1 boost::shared_ptr tc2_sp(new Sub(10)); boost::weak_ptr tc2_wp(tc2_sp); shared_weak(tc2_sp, tc2_wp); weak_shared(tc2_wp, tc2_sp); // Sub and Base2 boost::shared_ptr tc3_sp(new Sub(10)); boost::weak_ptr tc3_wp(tc3_sp); shared_weak(tc3_sp, tc3_wp); weak_shared(tc3_wp, tc3_sp); // Sub and Base3 boost::shared_ptr tc4_sp(new Sub(10)); boost::weak_ptr tc4_wp(tc4_sp); shared_weak(tc4_sp, tc4_wp); weak_shared(tc4_wp, tc4_sp); // Base1 and Base2 boost::shared_ptr tc5_sp_tmp(new Sub(10)); boost::shared_ptr tc5_sp(tc5_sp_tmp); boost::weak_ptr tc5_wp(tc5_sp_tmp); tc5_sp_tmp.reset(); shared_weak(tc5_sp, tc5_wp); weak_shared(tc5_wp, tc5_sp); // Base2 and Base3 boost::shared_ptr tc6_sp_tmp(new Sub(10)); boost::shared_ptr tc6_sp(tc6_sp_tmp); boost::weak_ptr tc6_wp(tc6_sp_tmp); tc6_sp_tmp.reset(); shared_weak(tc6_sp, tc6_wp); weak_shared(tc6_wp, tc6_sp); // Base3 and Base1 boost::shared_ptr tc7_sp_tmp(new Sub(10)); boost::shared_ptr tc7_sp(tc7_sp_tmp); boost::weak_ptr tc7_wp(tc7_sp_tmp); tc7_sp_tmp.reset(); shared_weak(tc7_sp, tc7_wp); weak_shared(tc7_wp, tc7_sp); return EXIT_SUCCESS; }