contract/example/features/private_protected_virtual_multi.cpp
2017-12-10 16:31:15 -08:00

210 lines
6.0 KiB
C++

// Copyright (C) 2008-2018 Lorenzo Caminiti
// Distributed under the Boost Software License, Version 1.0 (see accompanying
// file LICENSE_1_0.txt or a copy at http://www.boost.org/LICENSE_1_0.txt).
// See: http://www.boost.org/doc/libs/release/libs/contract/doc/html/index.html
#include <boost/config.hpp>
#ifdef BOOST_MSVC
// WARNING: MSVC (at least up to VS 2015) gives a compile-time error if SFINAE
// cannot introspect a member because of its private or protected access level.
// That is incorrect, SFINAE should fail in these cases without generating
// compile-time errors like GCC and CLang do. Therefore, currently it is not
// possible to override a member that is public in one base but private or
// protected in other base using this library on MSVC (that can be done instead
// using this library on GCC or CLang).
int main() { return 0; } // Trivial program for MSVC.
#else
#include <boost/contract.hpp>
#include <limits>
#include <cassert>
class counter {
// Virtual private and protected functions still declare extra
// `virtual_* = 0` parameter (otherwise they cannot be overridden).
protected:
virtual void set(int n, boost::contract::virtual_* = 0) {
boost::contract::check c = boost::contract::function()
.precondition([&] {
BOOST_CONTRACT_ASSERT(n <= 0);
})
.postcondition([&] {
BOOST_CONTRACT_ASSERT(get() == n);
})
;
n_ = n;
}
private:
virtual void dec(boost::contract::virtual_* = 0) {
boost::contract::old_ptr<int> old_get = BOOST_CONTRACT_OLDOF(get());
boost::contract::check c = boost::contract::function()
.precondition([&] {
BOOST_CONTRACT_ASSERT(
get() + 1 >= std::numeric_limits<int>::min());
})
.postcondition([&] {
BOOST_CONTRACT_ASSERT(get() == *old_get - 1);
})
;
set(get() - 1);
}
int n_;
public:
virtual int get(boost::contract::virtual_* v = 0) const {
int result;
boost::contract::check c = boost::contract::public_function(
v, result, this)
.postcondition([&] (int const result) {
BOOST_CONTRACT_ASSERT(result <= 0);
BOOST_CONTRACT_ASSERT(result == n_);
})
;
return result = n_;
}
counter() : n_(0) {} // Should contract constructor and destructor too.
void invariant() const {
BOOST_CONTRACT_ASSERT(get() <= 0);
}
friend int main();
};
//[private_protected_virtual_multi_countable
class countable {
public:
void invariant() const {
BOOST_CONTRACT_ASSERT(get() <= 0);
}
virtual void dec(boost::contract::virtual_* v = 0) = 0;
virtual void set(int n, boost::contract::virtual_* v = 0) = 0;
virtual int get(boost::contract::virtual_* v = 0) const = 0;
};
/* ... */
//]
void countable::dec(boost::contract::virtual_* v) {
boost::contract::old_ptr<int> old_get = BOOST_CONTRACT_OLDOF(v, get());
boost::contract::check c = boost::contract::public_function(v, this)
.precondition([&] {
BOOST_CONTRACT_ASSERT(get() > std::numeric_limits<int>::min());
})
.postcondition([&] {
BOOST_CONTRACT_ASSERT(get() < *old_get);
})
;
assert(false); // Never executed by this library.
}
void countable::set(int n, boost::contract::virtual_* v) {
boost::contract::check c = boost::contract::public_function(
v, this)
.precondition([&] {
BOOST_CONTRACT_ASSERT(n <= 0);
})
.postcondition([&] {
BOOST_CONTRACT_ASSERT(get() == n);
})
;
assert(false); // Never executed by this library.
}
int countable::get(boost::contract::virtual_* v) const {
int result;
boost::contract::check c = boost::contract::public_function(
v, result, this);
assert(false); // Never executed by this library.
}
//[private_protected_virtual_multi_counter10
class counter10
#define BASES public countable, public counter // Multiple inheritance.
: BASES
{
public:
typedef BOOST_CONTRACT_BASE_TYPES(BASES) base_types;
#undef BASES
// Overriding from public members from `countable` so use `override_...`.
virtual void set(int n, boost::contract::virtual_* v = 0) /* override */ {
boost::contract::check c = boost::contract::public_function<
override_set>(v, &counter10::set, this, n)
.precondition([&] {
BOOST_CONTRACT_ASSERT(n % 10 == 0);
})
.postcondition([&] {
BOOST_CONTRACT_ASSERT(get() == n);
})
;
counter::set(n);
}
virtual void dec(boost::contract::virtual_* v = 0) /* override */ {
boost::contract::old_ptr<int> old_get = BOOST_CONTRACT_OLDOF(v, get());
boost::contract::check c = boost::contract::public_function<
override_dec>(v, &counter10::dec, this)
.precondition([&] {
BOOST_CONTRACT_ASSERT(
get() + 10 >= std::numeric_limits<int>::min());
})
.postcondition([&] {
BOOST_CONTRACT_ASSERT(get() == *old_get - 10);
})
;
set(get() - 10);
}
BOOST_CONTRACT_OVERRIDES(set, dec)
/* ... */
//]
virtual int get(boost::contract::virtual_* v = 0) const {
int result;
boost::contract::check c = boost::contract::public_function<
override_get>(v, result, &counter10::get, this);
return result = counter::get();
}
BOOST_CONTRACT_OVERRIDE(get)
// Should contract default constructor and destructor too.
void invariant() const {
BOOST_CONTRACT_ASSERT(get() % 10 == 0);
}
};
int main() {
counter cnt;
assert(cnt.get() == 0);
cnt.dec();
assert(cnt.get() == -1);
counter10 cnt10;
countable& b = cnt10; // Polymorphic calls.
assert(b.get() == 0);
b.dec();
assert(b.get() == -10);
return 0;
}
#endif // MSVC