/////////1/////////2/////////3/////////4/////////5/////////6/////////7/////////8 // test_shared_ptr.cpp // (C) Copyright 2002 Robert Ramey- http://www.rrsd.com - David Tonge . // 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 "test_tools.hpp" // This is a simple class. It contains a counter of the number // of objects of this class which have been instantiated. class A { private: friend class boost::serialization::access; int x; template void serialize(Archive & ar, const unsigned int /* file_version */){ ar & BOOST_SERIALIZATION_NVP(x); } A(A const & rhs); A& operator=(A const & rhs); public: static int count; bool operator==(const A & rhs) const { return x == rhs.x; } A(){++count;} // default constructor virtual ~A(){--count;} // default destructor }; BOOST_SERIALIZATION_SHARED_PTR(A) int A::count = 0; // B is a subclass of A class B : public A { private: friend class boost::serialization::access; template void serialize(Archive & ar, const unsigned int /* file_version */){ ar & BOOST_SERIALIZATION_BASE_OBJECT_NVP(A); } public: static int count; B() : A() {}; virtual ~B() {}; }; // B needs to be exported because its serialized via a base class pointer BOOST_CLASS_EXPORT(B) BOOST_SERIALIZATION_SHARED_PTR(B) // test a non-polymorphic class class C { private: friend class boost::serialization::access; int z; template void serialize(Archive & ar, const unsigned int /* file_version */){ ar & BOOST_SERIALIZATION_NVP(z); } public: static int count; bool operator==(const C & rhs) const { return z == rhs.z; } C() : z(++count) // default constructor {} virtual ~C(){--count;} // default destructor }; int C::count = 0; template void save(const char * testfile, const SP & spa) { test_ostream os(testfile, TEST_STREAM_FLAGS); test_oarchive oa(os, TEST_ARCHIVE_FLAGS); oa << BOOST_SERIALIZATION_NVP(spa); } template void load(const char * testfile, SP & spa) { test_istream is(testfile, TEST_STREAM_FLAGS); test_iarchive ia(is, TEST_ARCHIVE_FLAGS); ia >> BOOST_SERIALIZATION_NVP(spa); } // trivial test template void save_and_load(SP & spa) { const char * testfile = boost::archive::tmpnam(NULL); BOOST_REQUIRE(NULL != testfile); save(testfile, spa); SP spa1; load(testfile, spa1); BOOST_CHECK( (spa.get() == NULL && spa1.get() == NULL) || * spa == * spa1 ); std::remove(testfile); } template void save2( const char * testfile, const SP & first, const SP & 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, SP & first, SP & 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. template void save_and_load2(SP & first, SP & second) { const char * testfile = boost::archive::tmpnam(NULL); BOOST_REQUIRE(NULL != testfile); save2(testfile, first, second); // Clear the pointers, thereby destroying the objects they contain first.reset(); second.reset(); load2(testfile, first, second); BOOST_CHECK(first == second); std::remove(testfile); } template void save3( const char * testfile, SP & first, SP & second, WP & third ){ test_ostream os(testfile, TEST_STREAM_FLAGS); test_oarchive oa(os, TEST_ARCHIVE_FLAGS); oa << BOOST_SERIALIZATION_NVP(third); oa << BOOST_SERIALIZATION_NVP(first); oa << BOOST_SERIALIZATION_NVP(second); } template void load3( const char * testfile, SP & first, SP & second, WP & third ){ test_istream is(testfile, TEST_STREAM_FLAGS); test_iarchive ia(is, TEST_ARCHIVE_FLAGS); // note that we serialize the weak pointer first ia >> BOOST_SERIALIZATION_NVP(third); // inorder to test that a temporarily solitary weak pointer // correctly restored. ia >> BOOST_SERIALIZATION_NVP(first); ia >> BOOST_SERIALIZATION_NVP(second); } template void save_and_load3( SP & first, SP & second, WP & third ){ const char * testfile = boost::archive::tmpnam(NULL); BOOST_REQUIRE(NULL != testfile); save3(testfile, first, second, third); // Clear the pointers, thereby destroying the objects they contain first.reset(); second.reset(); third.reset(); load3(testfile, first, second, third); BOOST_CHECK(first == second); BOOST_CHECK(first == third.lock()); std::remove(testfile); } template void save4(const char * testfile, const SP & spc) { test_ostream os(testfile, TEST_STREAM_FLAGS); test_oarchive oa(os, TEST_ARCHIVE_FLAGS); oa << BOOST_SERIALIZATION_NVP(spc); } template void load4(const char * testfile, SP & spc) { test_istream is(testfile, TEST_STREAM_FLAGS); test_iarchive ia(is, TEST_ARCHIVE_FLAGS); ia >> BOOST_SERIALIZATION_NVP(spc); } // trivial test template void save_and_load4(SP & spc) { const char * testfile = boost::archive::tmpnam(NULL); BOOST_REQUIRE(NULL != testfile); save4(testfile, spc); SP spc1; load4(testfile, spc1); BOOST_CHECK( (spc.get() == NULL && spc1.get() == NULL) || * spc == * spc1 ); std::remove(testfile); } // This does the tests template class SPT , template class WPT > bool test(){ { SPT spa; // These are our shared_ptrs spa = SPT(new A); SPT spa1 = spa; spa1 = spa; } { // These are our shared_ptrs SPT spa; // trivial test 1 save_and_load(spa); //trivival test 2 spa = SPT(new A); save_and_load(spa); // Try to save and load pointers to As spa = SPT(new A); SPT spa1 = spa; save_and_load2(spa, spa1); // Try to save and load pointers to Bs spa = SPT(new B); spa1 = spa; save_and_load2(spa, spa1); // test a weak pointer spa = SPT(new A); spa1 = spa; WPT wp = spa; save_and_load3(spa, spa1, wp); // obj of type B gets destroyed // as smart_ptr goes out of scope } BOOST_CHECK(A::count == 0); { // Try to save and load pointers to Cs SPT spc; spc = SPT(new C); save_and_load4(spc); } BOOST_CHECK(C::count == 0); return true; } // This does the tests int test_main(int /* argc */, char * /* argv */[]) { bool result = true; result &= test(); #ifndef BOOST_NO_CXX11_SMART_PTR result &= test(); #endif return result ? EXIT_SUCCESS : EXIT_FAILURE; }