serialization/test/test_shared_ptr.cpp
Robert Ramey a838b0821f tweaked tests to suppress some warning noise
fixed sequential lists to handle types without default constructors
adjusted CMakeLists.txt to correctly generate c++ selection for Xcode
2014-12-16 15:40:24 -08:00

323 lines
7.8 KiB
C++

/////////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 <cstddef> // NULL
#include <cstdio> // remove
#include <fstream>
#include <boost/config.hpp>
#if defined(BOOST_NO_STDC_NAMESPACE)
namespace std{
using ::remove;
}
#endif
#include <boost/serialization/nvp.hpp>
#include <boost/serialization/export.hpp>
#include <boost/serialization/shared_ptr.hpp>
#include <boost/serialization/weak_ptr.hpp>
#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<class Archive>
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<class Archive>
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<class Archive>
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<class SP>
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<class SP>
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<class SP>
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<class SP>
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<class SP>
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<class SP>
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<class SP, class WP>
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<class SP, class WP>
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<class SP, class WP>
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<class SP>
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<class SP>
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<class SP>
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<template<class T> class SPT , template<class T> class WPT >
bool test(){
{
SPT<A> spa;
// These are our shared_ptrs
spa = SPT<A>(new A);
SPT<A> spa1 = spa;
spa1 = spa;
}
{
// These are our shared_ptrs
SPT<A> spa;
// trivial test 1
save_and_load(spa);
//trivival test 2
spa = SPT<A>(new A);
save_and_load(spa);
// Try to save and load pointers to As
spa = SPT<A>(new A);
SPT<A> spa1 = spa;
save_and_load2(spa, spa1);
// Try to save and load pointers to Bs
spa = SPT<A>(new B);
spa1 = spa;
save_and_load2(spa, spa1);
// test a weak pointer
spa = SPT<A>(new A);
spa1 = spa;
WPT<A> 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<C> spc;
spc = SPT<C>(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<boost::shared_ptr, boost::weak_ptr>();
#ifndef BOOST_NO_CXX11_SMART_PTR
result &= test<std::shared_ptr, std::weak_ptr>();
#endif
return result ? EXIT_SUCCESS : EXIT_FAILURE;
}