serialization/test/test_dll_plugin.cpp

212 lines
6.3 KiB
C++

/////////1/////////2/////////3/////////4/////////5/////////6/////////7/////////8
// test_dll_plugin.cpp
// (C) Copyright 2002 Robert Ramey - http://www.rrsd.com .
// 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)
// should pass compilation and execution
// Note this test creates, serializes, and destroys
// a class instance while knowing nothing more than its
// exported class ID (GUID) and a base class from which
// it is derived. This is referred to as a "plugin"
// since the same program could, without recompilation,
// manipulate any number of derived types - even those
// which have not been yet been created.
#include <fstream>
#include <cstdio> // remove
#include <boost/config.hpp>
#if defined(BOOST_NO_STDC_NAMESPACE)
namespace std{
using ::remove;
}
#endif
#include <boost/archive/archive_exception.hpp>
// for now, only test with simple text and polymorphic archive
#include "test_tools.hpp"
#include <boost/serialization/base_object.hpp>
#include <boost/serialization/export.hpp>
#include <boost/serialization/type_info_implementation.hpp>
#include <boost/serialization/access.hpp>
#include <boost/serialization/void_cast.hpp>
#include <boost/serialization/extended_type_info.hpp>
#include "polymorphic_base.hpp"
// declare and implement a derived class in our own executable
class polymorphic_derived1 : public polymorphic_base
{
friend class boost::serialization::access;
template<class Archive>
void serialize(Archive &ar, const unsigned int /* file_version */){
ar & BOOST_SERIALIZATION_BASE_OBJECT_NVP(polymorphic_base);
}
const char * get_key() const{
return
boost::serialization::type_info_implementation<
polymorphic_derived1
>::type::get_const_instance().get_key();
}
public:
virtual ~polymorphic_derived1(){}
};
// This class is derived from polymorphic_base which uses the no_rtti system
// rather than the typeid system. This system uses the exported name as the
// type identifier key. This MUST be exported!!!
BOOST_CLASS_EXPORT(polymorphic_derived1)
// MWerks users can do this to make their code work
BOOST_SERIALIZATION_MWERKS_BASE_AND_DERIVED(polymorphic_base, polymorphic_derived1)
// save exported polymorphic class
void save_exported(const char *testfile)
{
test_ostream os(testfile, TEST_STREAM_FLAGS);
test_oarchive oa(os, TEST_ARCHIVE_FLAGS);
polymorphic_base *rb1 = new polymorphic_derived1;
// get the eti record for the exported type "polymorphic_derived2"
boost::serialization::extended_type_info const * const d2_eti =
boost::serialization::extended_type_info::find(
"polymorphic_derived2"
);
assert(NULL != d2_eti);
// create a new instance of the type referred to by this record.
// in this example, we happen to know that the class constructor
// takes no arguments.
void const * const rd2 = d2_eti->construct();
assert(NULL != rd2);
// transform the pointer to a pointer to the base class
polymorphic_base const * const rb2
= static_cast<polymorphic_base const *>(
boost::serialization::void_upcast(
* d2_eti,
boost::serialization::type_info_implementation<polymorphic_base>
::type::get_const_instance(),
rd2
)
);
// export will permit correct serialization
// through a pointer to a base class
oa << BOOST_SERIALIZATION_NVP(rb1);
oa << BOOST_SERIALIZATION_NVP(rb2);
// don't need these any more - don't leak memory
delete rb1;
// note delete original handle - not runtime cast one !!!
//delete rb2;
d2_eti->destroy(rd2);
}
// save exported polymorphic class
void load_exported(const char *testfile)
{
test_istream is(testfile, TEST_STREAM_FLAGS);
test_iarchive ia(is, TEST_ARCHIVE_FLAGS);
polymorphic_base *rb1 = NULL;
polymorphic_base *rb2 = NULL;
// export will permit correct serialization
// through a pointer to a base class
ia >> BOOST_SERIALIZATION_NVP(rb1);
BOOST_CHECK_MESSAGE(
boost::serialization::type_info_implementation<polymorphic_derived1>
::type::get_const_instance()
==
* boost::serialization::type_info_implementation<polymorphic_base>
::type::get_const_instance().get_derived_extended_type_info(*rb1),
"restored pointer b1 not of correct type"
);
ia >> BOOST_SERIALIZATION_NVP(rb2);
// get the eti record for the exported type "polymorphic_derived2"
boost::serialization::extended_type_info const * const d2_eti =
boost::serialization::extended_type_info::find(
"polymorphic_derived2"
);
assert(NULL != d2_eti);
BOOST_CHECK_MESSAGE(
* d2_eti
==
* boost::serialization::type_info_implementation<polymorphic_base>
::type::get_const_instance().get_derived_extended_type_info(*rb2),
"restored pointer b2 not of correct type"
);
delete rb1;
delete rb2;
}
#ifdef BOOST_WINDOWS
#define WIN32_LEAN_AND_MEAN
#include <TCHAR.H>
#include <windows.h>
int
test_main( int /* argc */, char* /* argv */[] )
{
const char * testfile = boost::archive::tmpnam(NULL);
BOOST_REQUIRE(NULL != testfile);
HINSTANCE hDLL; // Handle to DLL
hDLL = LoadLibrary(_T("polymorphic_derived2.dll"));
BOOST_CHECK_MESSAGE(
(0 != hDLL),
"Failed to find/load polymorphic_derived2"
);
if(0 == hDLL)
return EXIT_FAILURE;
save_exported(testfile);
load_exported(testfile);
FreeLibrary(hDLL);
std::remove(testfile);
return EXIT_SUCCESS;
}
#else // presume *nix
#include <dlfcn.h>
int
test_main( int /* argc */, char* /* argv */[] )
{
const char * testfile = boost::archive::tmpnam(NULL);
BOOST_REQUIRE(NULL != testfile);
void * hDLL; // Handle to DLL
hDLL = dlopen("polymorphic_derived2.so", RTLD_NOW | RTLD_GLOBAL);
BOOST_CHECK_MESSAGE((0 != hDLL), "Failed to find/load plugin_polymorphic_derived2" );
BOOST_CHECK_MESSAGE((0 != hDLL), dlerror() );
if(0 == hDLL)
return EXIT_FAILURE;
save_exported(testfile);
load_exported(testfile);
dlclose(hDLL);
std::remove(testfile);
return EXIT_SUCCESS;
}
#endif
// EOF