Files
optional/test/optional_test_ref_portable_minimum.cpp

245 lines
5.0 KiB
C++
Raw Normal View History

// Copyright (C) 2014 Andrzej Krzemienski.
//
// 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/lib/optional for documentation.
//
// You are welcome to contact the author at: akrzemi1@gmail.com
#include "boost/optional/optional.hpp"
#ifdef __BORLANDC__
#pragma hdrstop
#endif
#include "boost/core/addressof.hpp"
2014-12-01 18:03:46 +01:00
#include "boost/core/enable_if.hpp"
#include "boost/core/lightweight_test.hpp"
using boost::optional;
using boost::none;
// testable classes
struct ScopeGuard // no copy/move ctor/assign
{
2014-12-01 18:03:46 +01:00
int val_;
explicit ScopeGuard(int v) : val_(v) {}
int& val() { return val_; }
private:
ScopeGuard(ScopeGuard const&);
void operator=(ScopeGuard const&);
};
struct Abstract
{
virtual int& val() = 0;
virtual ~Abstract() {}
Abstract(){}
2014-12-01 18:03:46 +01:00
private:
Abstract(Abstract const&);
void operator=(Abstract const&);
};
struct Impl : Abstract
{
int val_;
Impl(int v) : val_(v) {}
int& val() { return val_; }
};
template <typename T>
struct concrete_type_of
{
typedef T type;
};
template <>
struct concrete_type_of<Abstract>
{
typedef Impl type;
};
2014-12-01 18:03:46 +01:00
template <typename T>
struct has_arrow
{
static const bool value = true;
};
template <>
struct has_arrow<int>
{
static const bool value = false;
};
int& val(int& i) { return i; }
int& val(Abstract& a) { return a.val(); }
2014-12-01 18:03:46 +01:00
int& val(ScopeGuard& g) { return g.val(); }
// end testable classes
template <typename T>
typename boost::enable_if< has_arrow<T> >::type
test_arrow()
{
typename concrete_type_of<T>::type v(2);
optional<T&> o(v);
BOOST_TEST(o);
BOOST_TEST_EQ(o->val(), 2);
o->val() = 3;
BOOST_TEST(o);
BOOST_TEST_EQ(o->val(), 3);
BOOST_TEST_EQ(v.val(), 3);
}
template <typename T>
typename boost::disable_if< has_arrow<T> >::type
test_arrow()
{
}
template <typename T>
void test_not_containing_value_for()
{
optional<T&> o1;
optional<T&> o2 = none;
optional<T&> o3 = o1;
BOOST_TEST(!o1);
BOOST_TEST(!o2);
BOOST_TEST(!o3);
BOOST_TEST(o1 == none);
BOOST_TEST(o2 == none);
BOOST_TEST(o3 == none);
}
template <typename T>
void test_direct_init_for()
{
typename concrete_type_of<T>::type v(2);
optional<T&> o(v);
BOOST_TEST(o);
BOOST_TEST(o != none);
BOOST_TEST(boost::addressof(*o) == boost::addressof(v));
2014-12-01 18:03:46 +01:00
BOOST_TEST_EQ(val(*o), val(v));
BOOST_TEST_EQ(val(*o), 2);
val(v) = 9;
BOOST_TEST(boost::addressof(*o) == boost::addressof(v));
2014-12-01 18:03:46 +01:00
BOOST_TEST_EQ(val(*o), val(v));
BOOST_TEST_EQ(val(*o), 9);
BOOST_TEST_EQ(val(v), 9);
val(*o) = 7;
BOOST_TEST(boost::addressof(*o) == boost::addressof(v));
2014-12-01 18:03:46 +01:00
BOOST_TEST_EQ(val(*o), val(v));
BOOST_TEST_EQ(val(*o), 7);
BOOST_TEST_EQ(val(v), 7);
}
template <typename T>
void test_clearing_the_value()
{
typename concrete_type_of<T>::type v(2);
optional<T&> o1(v), o2(v);
BOOST_TEST(o1);
BOOST_TEST(o1 != none);
BOOST_TEST(o2);
BOOST_TEST(o2 != none);
o1 = none;
BOOST_TEST(!o1);
BOOST_TEST(o1 == none);
BOOST_TEST(o2);
BOOST_TEST(o2 != none);
BOOST_TEST_EQ(val(*o2), 2);
BOOST_TEST(boost::addressof(*o2) == boost::addressof(v));
BOOST_TEST_EQ(val(v), 2);
}
template <typename T>
void test_copy_assignment_for()
{
typename concrete_type_of<T>::type v(2);
optional<T&> o;
o = optional<T&>(v);
BOOST_TEST(o);
BOOST_TEST(o != none);
BOOST_TEST(boost::addressof(*o) == boost::addressof(v));
BOOST_TEST(val(*o) == val(v));
BOOST_TEST(val(*o) == 2);
val(v) = 9;
BOOST_TEST(boost::addressof(*o) == boost::addressof(v));
2014-12-01 18:03:46 +01:00
BOOST_TEST_EQ(val(*o), val(v));
BOOST_TEST_EQ(val(*o), 9);
BOOST_TEST_EQ(val(v), 9);
val(*o) = 7;
BOOST_TEST(boost::addressof(*o) == boost::addressof(v));
2014-12-01 18:03:46 +01:00
BOOST_TEST_EQ(val(*o), val(v));
BOOST_TEST_EQ(val(*o), 7);
BOOST_TEST_EQ(val(v), 7);
}
template <typename T>
void test_rebinding_assignment_semantics()
{
typename concrete_type_of<T>::type v(2), w(7);
optional<T&> o(v);
BOOST_TEST(o);
BOOST_TEST(boost::addressof(*o) == boost::addressof(v));
BOOST_TEST_EQ(val(*o), val(v));
BOOST_TEST_EQ(val(*o), 2);
o = optional<T&>(w);
BOOST_TEST_EQ(val(v), 2);
BOOST_TEST(o);
BOOST_TEST(boost::addressof(*o) != boost::addressof(v));
BOOST_TEST_NE(val(*o), val(v));
BOOST_TEST_NE(val(*o), 2);
BOOST_TEST(boost::addressof(*o) == boost::addressof(w));
BOOST_TEST_EQ(val(*o), val(w));
BOOST_TEST_EQ(val(*o), 7);
val(*o) = 8;
BOOST_TEST(boost::addressof(*o) == boost::addressof(w));
BOOST_TEST_EQ(val(*o), val(w));
BOOST_TEST_EQ(val(*o), 8);
BOOST_TEST_EQ(val(w), 8);
BOOST_TEST_EQ(val(v), 2);
}
template <typename T>
void test_optional_ref()
{
test_not_containing_value_for<T>();
test_direct_init_for<T>();
test_copy_assignment_for<T>();
2014-12-01 18:03:46 +01:00
test_rebinding_assignment_semantics<T>();
test_clearing_the_value<T>();
test_arrow<T>();
}
int main()
{
test_optional_ref<int>();
test_optional_ref<ScopeGuard>();
2014-12-01 18:03:46 +01:00
//test_optional_ref<Abstract>();
return boost::report_errors();
}