mirror of
				https://github.com/boostorg/smart_ptr.git
				synced 2025-10-31 15:51:38 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			210 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			210 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| //
 | |
| //  cpp11_pointer_cast_test.cpp - a test for boost/pointer_cast.hpp with std::shared_ptr and std::unique_ptr
 | |
| //
 | |
| //  Copyright (c) 2016 Karolin Varner
 | |
| //
 | |
| // 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/pointer_cast.hpp>
 | |
| #include <boost/shared_ptr.hpp>
 | |
| #include <boost/get_pointer.hpp>
 | |
| #include <boost/core/lightweight_test.hpp>
 | |
| #include <boost/config.hpp>
 | |
| #include <memory>
 | |
| #include <utility>
 | |
| #include <functional>
 | |
| 
 | |
| namespace
 | |
| {
 | |
| 
 | |
| // Let's create these inheritance relationship:
 | |
| //
 | |
| //    base   base2
 | |
| //      |     |
 | |
| //      derived
 | |
| //         |
 | |
| //  derived_derived
 | |
| //
 | |
| 
 | |
| class base
 | |
| {
 | |
|    public:
 | |
|    virtual ~base(){}
 | |
|    int filler [5];
 | |
| };
 | |
| 
 | |
| class base2
 | |
| {
 | |
| public:
 | |
| 
 | |
|     virtual ~base2(){}
 | |
|     int filler [5];
 | |
| };
 | |
| 
 | |
| class derived
 | |
|    : public base, public base2
 | |
| {
 | |
|     int filler [5];
 | |
| };
 | |
| 
 | |
| class derived_derived
 | |
|    : public derived
 | |
| {
 | |
|     int filler [5];
 | |
| };
 | |
| 
 | |
| // And now some simple check functions
 | |
| 
 | |
| #if !defined( BOOST_NO_RTTI )
 | |
| 
 | |
| template <class BasePtr>
 | |
| bool check_dynamic_pointer_cast(const BasePtr &ptr)
 | |
| {
 | |
|    //Check that dynamic_pointer_cast versus dynamic_cast
 | |
|    return
 | |
|    //Correct cast with dynamic_pointer_cast
 | |
|    boost::get_pointer(boost::dynamic_pointer_cast<derived>(ptr)) ==
 | |
|       //Correct cast with dynamic_cast
 | |
|       dynamic_cast<derived*>(boost::get_pointer(ptr))
 | |
|    &&
 | |
|    //Incorrect cast with dynamic_pointer_cast
 | |
|    boost::get_pointer(boost::dynamic_pointer_cast<derived_derived>(ptr)) ==
 | |
|       //Incorrect cast with dynamic_cast
 | |
|       dynamic_cast<derived_derived*>(boost::get_pointer(ptr));
 | |
| }
 | |
| 
 | |
| #endif
 | |
| 
 | |
| template <class BasePtr>
 | |
| bool check_static_pointer_cast(const BasePtr &ptr)
 | |
| {
 | |
|    return
 | |
|    //Cast base -> derived -> base2 using static_pointer_cast
 | |
|    boost::get_pointer(
 | |
|             boost::static_pointer_cast<base2>(
 | |
|                boost::static_pointer_cast<derived>(ptr))) ==
 | |
|    //Now the same with static_cast
 | |
|    static_cast<base2*>(static_cast<derived*>(boost::get_pointer(ptr)));
 | |
| }
 | |
| 
 | |
| template <class BasePtr>
 | |
| bool check_const_pointer_cast(const BasePtr &ptr)
 | |
| {
 | |
|    return
 | |
|    //Unconst and const again using const_pointer_cast
 | |
|    boost::get_pointer(
 | |
|       boost::const_pointer_cast<const base>
 | |
|          (boost::const_pointer_cast<base>(ptr))) ==
 | |
|    //Now the same with const_cast
 | |
|    const_cast<const base*>(const_cast<base*>(boost::get_pointer(ptr)));
 | |
| }
 | |
| 
 | |
| template <class BasePtr>
 | |
| void check_all_copy_casts(const BasePtr &ptr)
 | |
| {
 | |
| #if !defined( BOOST_NO_RTTI )
 | |
|    BOOST_TEST( check_dynamic_pointer_cast( ptr ) );
 | |
| #endif
 | |
|    BOOST_TEST( check_static_pointer_cast( ptr ) );
 | |
|    BOOST_TEST( check_const_pointer_cast( ptr ) );
 | |
| }
 | |
| 
 | |
| 
 | |
| #if !defined( BOOST_NO_RTTI )
 | |
| 
 | |
| template <class BasePtr>
 | |
| bool check_dynamic_moving_pointer_cast(std::function<BasePtr()> f)
 | |
| {
 | |
|    BasePtr smart1 = f(), smart2 = f();
 | |
|    derived* expect1 = dynamic_cast<derived*>(boost::get_pointer(smart1));
 | |
|    derived_derived* expect2 = dynamic_cast<derived_derived*>(boost::get_pointer(smart2));
 | |
|    //Check that dynamic_pointer_cast versus dynamic_cast
 | |
|    return
 | |
|    //Correct cast with dynamic_pointer_cast
 | |
|    boost::get_pointer(boost::dynamic_pointer_cast<derived>( std::move(smart1) )) == expect1
 | |
|    &&
 | |
|    //Incorrect cast with dynamic_pointer_cast
 | |
|    boost::get_pointer(boost::dynamic_pointer_cast<derived_derived>( std::move(smart2) )) == expect2;
 | |
| }
 | |
| 
 | |
| #endif
 | |
| 
 | |
| template <class BasePtr>
 | |
| bool check_static_moving_pointer_cast(std::function<BasePtr()> f)
 | |
| {
 | |
|    BasePtr smart = f();
 | |
|    base2 *expect = static_cast<base2*>(static_cast<derived*>(boost::get_pointer(smart)));
 | |
| 
 | |
|    return
 | |
|    //Cast base -> derived -> base2 using static_pointer_cast
 | |
|    boost::get_pointer(
 | |
|             boost::static_pointer_cast<base2>(
 | |
|                boost::static_pointer_cast<derived>( std::move(smart) ))) ==
 | |
|    //Now the same with static_cast
 | |
|    expect;
 | |
| }
 | |
| 
 | |
| template <class BasePtr>
 | |
| bool check_const_moving_pointer_cast(std::function<BasePtr()> f)
 | |
| {
 | |
|    BasePtr smart = f();
 | |
|    const base *expect = const_cast<const base*>(const_cast<base*>(boost::get_pointer(smart)));
 | |
|    return
 | |
|    //Unconst and const again using const_pointer_cast
 | |
|    boost::get_pointer(
 | |
|       boost::const_pointer_cast<const base>
 | |
|          (boost::const_pointer_cast<base>( std::move(smart) ))) ==
 | |
|    //Now the same with const_cast
 | |
|    expect;
 | |
| }
 | |
| 
 | |
| template <class BasePtr>
 | |
| void check_all_moving_casts(std::function<BasePtr()> f) {
 | |
| #if !defined( BOOST_NO_RTTI )
 | |
|    BOOST_TEST( check_dynamic_moving_pointer_cast( f ) );
 | |
| #endif
 | |
|    BOOST_TEST( check_static_moving_pointer_cast( f ) );
 | |
|    BOOST_TEST( check_const_moving_pointer_cast( f ) );
 | |
| }
 | |
| 
 | |
| }
 | |
| 
 | |
| int main()
 | |
| {
 | |
| 
 | |
|    std::shared_ptr<base> std_shared(new derived);
 | |
|    boost::shared_ptr<base> boost_shared(new derived);
 | |
|    base *plain = boost_shared.get();
 | |
| 
 | |
|    // plain & boost::shared_ptr moving pointer_cast checks; there
 | |
|    // is no specific handleing for those types at the moment; this
 | |
|    // test just makes sure they won't break when std::move() is used
 | |
|    // in generic code
 | |
| 
 | |
|    check_all_moving_casts<boost::shared_ptr<base>>([&boost_shared]() {
 | |
|        return boost_shared;
 | |
|    });
 | |
| 
 | |
|    check_all_moving_casts<base*>([plain]() {
 | |
|        return plain;
 | |
|    });
 | |
| 
 | |
|    // std::shared_ptr casts
 | |
| 
 | |
|    check_all_copy_casts(std_shared);
 | |
|    check_all_moving_casts<std::shared_ptr<base>>([&std_shared]() {
 | |
|        return std_shared;
 | |
|    });
 | |
| 
 | |
|    // std::unique_ptr casts
 | |
| 
 | |
|    check_all_moving_casts<std::unique_ptr<base>>([]() {
 | |
|        return std::unique_ptr<base>(new derived);
 | |
|    });
 | |
| 
 | |
|     return boost::report_errors();
 | |
| }
 |