summaryrefslogtreecommitdiff
path: root/libs/type_index/test/type_index_runtime_cast_test.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'libs/type_index/test/type_index_runtime_cast_test.cpp')
-rw-r--r--libs/type_index/test/type_index_runtime_cast_test.cpp296
1 files changed, 296 insertions, 0 deletions
diff --git a/libs/type_index/test/type_index_runtime_cast_test.cpp b/libs/type_index/test/type_index_runtime_cast_test.cpp
new file mode 100644
index 0000000000..1f5344ef5e
--- /dev/null
+++ b/libs/type_index/test/type_index_runtime_cast_test.cpp
@@ -0,0 +1,296 @@
+//
+// Copyright Chris Glover, 2016.
+//
+// Distributed under 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)
+
+// #include <boost/type_index/runtime_cast.hpp>
+// #include <boost/type_index/runtime_reference_cast.hpp>
+
+#include <boost/type_index/runtime_cast.hpp>
+#include <boost/type_index/runtime_cast/boost_shared_ptr_cast.hpp>
+#include <boost/smart_ptr/make_shared.hpp>
+
+#include <boost/core/lightweight_test.hpp>
+
+#if !defined(BOOST_NO_CXX11_SMART_PTR)
+# include <boost/type_index/runtime_cast/std_shared_ptr_cast.hpp>
+#endif
+
+// Classes include a member variable "name" with the
+// name of the class hard coded so we can be sure that
+// the pointer offsets are all working, since we're doing
+// a cast from void* at some point.
+
+#define IMPLEMENT_CLASS(type_name) \
+ type_name() : name( #type_name ) {} \
+ std::string name;
+
+struct base {
+ BOOST_TYPE_INDEX_IMPLEMENT_RUNTIME_CAST(BOOST_TYPE_INDEX_NO_BASE_CLASS)
+ IMPLEMENT_CLASS(base)
+};
+
+struct single_derived : base {
+ BOOST_TYPE_INDEX_IMPLEMENT_RUNTIME_CAST((base))
+ IMPLEMENT_CLASS(single_derived)
+};
+
+struct base1 {
+ BOOST_TYPE_INDEX_IMPLEMENT_RUNTIME_CAST(BOOST_TYPE_INDEX_NO_BASE_CLASS)
+ IMPLEMENT_CLASS(base1)
+};
+
+struct base2 {
+ BOOST_TYPE_INDEX_IMPLEMENT_RUNTIME_CAST(BOOST_TYPE_INDEX_NO_BASE_CLASS)
+ IMPLEMENT_CLASS(base2)
+};
+
+struct multiple_derived : base1, base2 {
+ BOOST_TYPE_INDEX_IMPLEMENT_RUNTIME_CAST((base1)(base2))
+ IMPLEMENT_CLASS(multiple_derived)
+};
+
+struct baseV1 : virtual base {
+ BOOST_TYPE_INDEX_IMPLEMENT_RUNTIME_CAST((base))
+ IMPLEMENT_CLASS(baseV1)
+};
+
+struct baseV2 : virtual base {
+ BOOST_TYPE_INDEX_IMPLEMENT_RUNTIME_CAST((base))
+ IMPLEMENT_CLASS(baseV2)
+};
+
+struct multiple_virtual_derived : baseV1, baseV2 {
+ BOOST_TYPE_INDEX_IMPLEMENT_RUNTIME_CAST((baseV1)(baseV2))
+ IMPLEMENT_CLASS(multiple_virtual_derived)
+};
+
+struct unrelated {
+ BOOST_TYPE_INDEX_IMPLEMENT_RUNTIME_CAST(BOOST_TYPE_INDEX_NO_BASE_CLASS)
+ IMPLEMENT_CLASS(unrelated)
+};
+
+struct unrelated_with_base : base {
+ BOOST_TYPE_INDEX_IMPLEMENT_RUNTIME_CAST((base))
+ IMPLEMENT_CLASS(unrelated_with_base)
+};
+
+struct unrelatedV1 : virtual base {
+ BOOST_TYPE_INDEX_IMPLEMENT_RUNTIME_CAST((base))
+ IMPLEMENT_CLASS(unrelatedV1)
+};
+
+struct level1_a : base {
+ BOOST_TYPE_INDEX_IMPLEMENT_RUNTIME_CAST((base))
+ IMPLEMENT_CLASS(level1_a)
+};
+
+struct level1_b : base {
+ BOOST_TYPE_INDEX_IMPLEMENT_RUNTIME_CAST((base))
+ IMPLEMENT_CLASS(level1_b)
+};
+
+struct level2 : level1_a, level1_b {
+ BOOST_TYPE_INDEX_IMPLEMENT_RUNTIME_CAST((level1_a)(level1_b))
+ IMPLEMENT_CLASS(level2)
+};
+
+struct reg_base {
+ BOOST_TYPE_INDEX_REGISTER_RUNTIME_CLASS(BOOST_TYPE_INDEX_NO_BASE_CLASS)
+};
+
+struct reg_derived : reg_base {
+ BOOST_TYPE_INDEX_REGISTER_RUNTIME_CLASS((reg_base))
+};
+
+void no_base()
+{
+ using namespace boost::typeindex;
+ base b;
+ base* b2 = runtime_pointer_cast<base>(&b);
+ BOOST_TEST_NE(b2, (base*)NULL);
+ BOOST_TEST_EQ(b2->name, "base");
+
+ BOOST_TEST_EQ(runtime_pointer_cast<unrelated>(&b), (unrelated*)NULL);
+ BOOST_TEST_EQ(runtime_pointer_cast<single_derived>(&b), (single_derived*)NULL);
+ BOOST_TEST_EQ(runtime_pointer_cast<unrelatedV1>(&b), (unrelatedV1*)NULL);
+ BOOST_TEST_EQ(runtime_pointer_cast<unrelated_with_base>(&b), (unrelated_with_base*)NULL);
+}
+
+void single_base()
+{
+ using namespace boost::typeindex;
+ single_derived d;
+ base* b = &d;
+ single_derived* d2 = runtime_pointer_cast<single_derived>(b);
+ BOOST_TEST_NE(d2, (single_derived*)NULL);
+ BOOST_TEST_EQ(d2->name, "single_derived");
+
+ BOOST_TEST_EQ(runtime_pointer_cast<unrelated>(&d), (unrelated*)NULL);
+ BOOST_TEST_EQ(runtime_pointer_cast<unrelated>(b), (unrelated*)NULL);
+ BOOST_TEST_EQ(runtime_pointer_cast<unrelated_with_base>(b), (unrelated_with_base*)NULL);
+}
+
+void multiple_base()
+{
+ using namespace boost::typeindex;
+ multiple_derived d;
+ base1* b1 = &d;
+ multiple_derived* d2 = runtime_pointer_cast<multiple_derived>(b1);
+ BOOST_TEST_NE(d2, (multiple_derived*)NULL);
+ BOOST_TEST_EQ(d2->name, "multiple_derived");
+
+ base2* b2 = runtime_pointer_cast<base2>(b1);
+ BOOST_TEST_NE(b2, (base2*)NULL);
+ BOOST_TEST_EQ(b2->name, "base2");
+
+ BOOST_TEST_EQ(runtime_pointer_cast<unrelated>(&d), (unrelated*)NULL);
+ BOOST_TEST_EQ(runtime_pointer_cast<unrelated>(b1), (unrelated*)NULL);
+ BOOST_TEST_EQ(runtime_pointer_cast<unrelated_with_base>(b1), (unrelated_with_base*)NULL);
+}
+
+void virtual_base()
+{
+ using namespace boost::typeindex;
+ multiple_virtual_derived d;
+ base* b = &d;
+ multiple_virtual_derived* d2 = runtime_pointer_cast<multiple_virtual_derived>(b);
+ baseV1* bv1 = runtime_pointer_cast<baseV1>(b);
+ baseV2* bv2 = runtime_pointer_cast<baseV2>(b);
+
+ BOOST_TEST_NE(d2, (multiple_virtual_derived*)NULL);
+ BOOST_TEST_EQ(d2->name, "multiple_virtual_derived");
+
+ BOOST_TEST_NE(bv1, (baseV1*)NULL);
+ BOOST_TEST_EQ(bv1->name, "baseV1");
+
+ BOOST_TEST_NE(bv2, (baseV2*)NULL);
+ BOOST_TEST_EQ(bv2->name, "baseV2");
+
+ BOOST_TEST_EQ(runtime_pointer_cast<unrelated>(b), (unrelated*)NULL);
+ BOOST_TEST_EQ(runtime_pointer_cast<unrelated>(&d), (unrelated*)NULL);
+ BOOST_TEST_EQ(runtime_pointer_cast<unrelated_with_base>(b), (unrelated_with_base*)NULL);
+}
+
+void pointer_interface()
+{
+ using namespace boost::typeindex;
+ single_derived d;
+ base* b = &d;
+ single_derived* d2 = runtime_cast<single_derived*>(b);
+ BOOST_TEST_NE(d2, (single_derived*)NULL);
+ BOOST_TEST_EQ(d2->name, "single_derived");
+ BOOST_TEST_EQ(runtime_pointer_cast<unrelated>(b), (unrelated*)NULL);
+}
+
+void reference_interface()
+{
+ using namespace boost::typeindex;
+ single_derived d;
+ base& b = d;
+ single_derived& d2 = runtime_cast<single_derived&>(b);
+ BOOST_TEST_EQ(d2.name, "single_derived");
+
+ try {
+ unrelated& u = runtime_cast<unrelated&>(b);
+ (void)u;
+ BOOST_TEST(!"should throw bad_runtime_cast");
+ }
+ catch(boost::typeindex::bad_runtime_cast&) {
+ }
+ catch(...) {
+ BOOST_TEST(!"should throw bad_runtime_cast");
+ }
+}
+
+void const_pointer_interface()
+{
+ using namespace boost::typeindex;
+ const single_derived d;
+ base const* b = &d;
+ single_derived const* d2 = runtime_cast<single_derived const*>(b);
+ BOOST_TEST_NE(d2, (single_derived*)NULL);
+ BOOST_TEST_EQ(d2->name, "single_derived");
+ BOOST_TEST_EQ(runtime_pointer_cast<unrelated>(b), (unrelated*)NULL);
+}
+
+void const_reference_interface()
+{
+ using namespace boost::typeindex;
+ const single_derived d;
+ base const& b = d;
+ single_derived const& d2 = runtime_cast<single_derived const&>(b);
+ BOOST_TEST_EQ(d2.name, "single_derived");
+
+ try {
+ unrelated const& u = runtime_cast<unrelated const&>(b);
+ (void)u;
+ BOOST_TEST(!"should throw bad_runtime_cast");
+ }
+ catch(boost::typeindex::bad_runtime_cast&) {
+ }
+ catch(...) {
+ BOOST_TEST(!"should throw bad_runtime_cast");
+ }
+}
+
+void diamond_non_virtual()
+{
+ using namespace boost::typeindex;
+ level2 inst;
+ level1_a* l1a = &inst;
+ base* b1 = l1a;
+ level1_b* l1_b = runtime_cast<level1_b*>(b1);
+ BOOST_TEST_NE(l1_b, (level1_b*)NULL);
+ BOOST_TEST_EQ(l1_b->name, "level1_b");
+}
+
+void boost_shared_ptr()
+{
+ using namespace boost::typeindex;
+ boost::shared_ptr<single_derived> d = boost::make_shared<single_derived>();
+ boost::shared_ptr<base> b = d;
+ boost::shared_ptr<single_derived> d2 = runtime_pointer_cast<single_derived>(b);
+ BOOST_TEST_NE(d2, boost::shared_ptr<single_derived>());
+ BOOST_TEST_EQ(d2->name, "single_derived");
+}
+
+void std_shared_ptr()
+{
+#if !defined(BOOST_NO_CXX11_SMART_PTR)
+ using namespace boost::typeindex;
+ std::shared_ptr<single_derived> d = std::make_shared<single_derived>();
+ std::shared_ptr<base> b = d;
+ std::shared_ptr<single_derived> d2 = runtime_pointer_cast<single_derived>(b);
+ BOOST_TEST_NE(d2, std::shared_ptr<single_derived>());
+ BOOST_TEST_EQ(d2->name, "single_derived");
+#endif
+}
+
+void register_runtime_class()
+{
+ using namespace boost::typeindex;
+ reg_derived rd;
+ reg_base* rb = &rd;
+ reg_derived* prd = runtime_pointer_cast<reg_derived>(rb);
+ BOOST_TEST_NE(prd, (reg_derived*)NULL);
+ BOOST_TEST_EQ(type_id_runtime(*prd), type_id<reg_derived>());
+}
+
+int main() {
+ no_base();
+ single_derived();
+ multiple_base();
+ virtual_base();
+ pointer_interface();
+ reference_interface();
+ const_pointer_interface();
+ const_reference_interface();
+ diamond_non_virtual();
+ boost_shared_ptr();
+ std_shared_ptr();
+ register_runtime_class();
+ return boost::report_errors();
+}