started to implement bind for free and member functions, but without subcontracting yet

This commit is contained in:
Lorenzo Caminiti 2015-05-10 20:37:01 -07:00
parent 71ed05596c
commit effcbf157e
58 changed files with 1051 additions and 1895 deletions

View File

@ -8,8 +8,10 @@
/* PUBLIC */
#define BOOST_CONTRACT_ASSERT(condition) \
if(!(condition)) throw boost::contract::assertion_failure( \
__FILE__, __LINE__, BOOST_PP_STRINGIZE(condition));
if(!(condition)) { \
throw boost::contract::assertion_failure(__FILE__, __LINE__, \
BOOST_PP_STRINGIZE(cond)); \
}
#endif // #include guard

View File

@ -0,0 +1,30 @@
#ifndef BOOST_CONTRACT_AUX_CALL_HPP_
#define BOOST_CONTRACT_AUX_CALL_HPP_
#include <boost/shared_ptr.hpp>
#include <boost/noncopyable.hpp>
#include <queue>
namespace boost { namespace contract { namespace aux {
struct call : private boost::noncopyable { // Do not copy queue.
enum action_enum {
// Call always hold by ptr so null ptr (not enum) used for user-call.
copy_oldof,
check_entry_inv,
check_pre,
check_post,
check_exit_inv
};
explicit call() : action() {}
action_enum action;
std::queue<boost::shared_ptr<void> > old_values;
};
} } } // namespace
#endif // #include guard

View File

@ -1,55 +0,0 @@
#ifndef BOOST_CONTRACT_AUX_CHECK_PRE_POST_HPP_
#define BOOST_CONTRACT_AUX_CHECK_PRE_POST_HPP_
#include <boost/contract/aux_/exception.hpp>
#include <boost/function.hpp>
namespace boost { namespace contract { namespace aux { namespace check {
class pre_post {
public:
explicit pre_post(boost::contract::from const& from) : from_(from) {}
virtual ~pre_post() {}
template<typename Pre>
void set_pre(Pre const& pre) { pre_ = pre; pre_available(); }
template<typename Post>
void set_post(Post const& post) { post_ = post; post_available(); }
protected:
// Allow to throw on failure for relaxing subcontracted pre.
void check_pre(bool const throw_on_failure = false) {
if(pre_) {
try { pre_(); }
catch(...) {
if(throw_on_failure) throw;
boost::contract::aux::pre_failure_handler(from());
}
}
}
void check_post() {
if(post_) {
try { post_(); }
catch(...) { boost::contract::aux::post_failure_handler(from()); }
}
}
virtual void pre_available() {}
virtual void post_available() {}
boost::contract::from from() { return from_; }
private:
boost::contract::from const from_;
boost::function<void ()> pre_;
boost::function<void ()> post_;
};
} } } } // namespace
#endif // #include guard

View File

@ -1,83 +0,0 @@
#ifndef BOOST_CONTRACT_AUX_CHECK_PRE_POST_INV_HPP_
#define BOOST_CONTRACT_AUX_CHECK_PRE_POST_INV_HPP_
#include <boost/contract/core/config.hpp>
#include <boost/contract/aux_/check/pre_post.hpp>
#include <boost/contract/aux_/exception.hpp>
#include <boost/contract/aux_/type_traits/invariant.hpp>
#include <boost/contract/aux_/debug.hpp>
#include <boost/type_traits/add_const.hpp>
#include <boost/mpl/bool.hpp>
namespace boost { namespace contract { namespace aux { namespace check {
template<class Class>
class pre_post_inv : public boost::contract::aux::check::pre_post {
public:
// Allow object to be 0 (i.e., no object), for static members.
explicit pre_post_inv(boost::contract::from const& from,
Class* const obj = 0) :
boost::contract::aux::check::pre_post(from),
obj_(obj)
{}
virtual ~pre_post_inv() {}
protected:
void check_entry_inv(bool const static_inv_only = false) {
check_inv(/* on_entry = */ true, static_inv_only);
}
void check_exit_inv(bool const static_inv_only = false) {
check_inv(/* on_entry = */ false, static_inv_only);
}
// Could return null pointer (for static members).
Class* const object() const { return obj_; }
private:
void check_inv(bool const on_entry, bool const static_inv_only = false) {
check_static_inv(on_entry, boost::mpl::bool_<boost::contract::aux::
type_traits::has_static_invariant<Class>::value>());
if(!static_inv_only) {
check_cv_inv(on_entry, boost::mpl::bool_<boost::contract::aux::
type_traits::has_const_invariant<Class>::value>());
}
}
// TODO: Add const volatile invariant checking here.
void check_static_inv(bool const on_entry, boost::mpl::false_ const&) {}
void check_static_inv(bool const on_entry, boost::mpl::true_ const&) {
try { Class::BOOST_CONTRACT_CONFIG_STATIC_INVARIANT(); }
catch(...) {
if(on_entry) {
boost::contract::aux::static_entry_inv_failure_handler(from());
} else {
boost::contract::aux::static_entry_inv_failure_handler(from());
}
}
}
void check_cv_inv(bool const on_entry, boost::mpl::false_ const&) {}
void check_cv_inv(bool const on_entry, boost::mpl::true_ const&) {
typename boost::add_const<Class>::type* const const_obj = obj_;
BOOST_CONTRACT_AUX_DEBUG(const_obj);
try { const_obj->BOOST_CONTRACT_CONFIG_INVARIANT(); }
catch(...) {
if(on_entry) {
boost::contract::aux::const_entry_inv_failure_handler(from());
} else {
boost::contract::aux::const_exit_inv_failure_handler(from());
}
}
}
Class* const obj_;
};
} } } } // namespace
#endif // #include guard

View File

@ -1,216 +0,0 @@
#ifndef BOOST_CONTRACT_AUX_CHECK_SUBCONTRACTED_PRE_POST_INV_HPP_
#define BOOST_CONTRACT_AUX_CHECK_SUBCONTRACTED_PRE_POST_INV_HPP_
// TODO: Double check that this lib header should only #include from core/*
// and aux_/*, while headers directly from contract/* should NOT be #included
// by other headers of this lib (so they remain 100% for user's #inclusion).
// Consider generating a Doxygen #include diagram...
#include <boost/contract/core/var.hpp>
#include <boost/contract/core/virtual.hpp>
#include <boost/contract/aux_/check/pre_post_inv.hpp>
#include <boost/contract/aux_/type_traits/bases.hpp>
#include <boost/contract/aux_/exception.hpp>
#include <boost/contract/aux_/none.hpp>
#include <boost/contract/aux_/debug.hpp>
#include <boost/function_types/result_type.hpp>
#include <boost/function_types/parameter_types.hpp>
#include <boost/function_types/member_function_pointer.hpp>
#include <boost/function_types/property_tags.hpp>
#include <boost/type_traits/add_pointer.hpp>
#include <boost/type_traits/is_same.hpp>
#include <boost/mpl/for_each.hpp>
#include <boost/mpl/vector.hpp>
#include <boost/mpl/transform.hpp>
#include <boost/mpl/pop_front.hpp>
#include <boost/mpl/push_front.hpp>
#include <boost/mpl/push_back.hpp>
#include <boost/mpl/back.hpp>
#include <boost/mpl/bool.hpp>
#include <boost/mpl/identity.hpp>
#include <boost/mpl/placeholders.hpp>
#include <boost/mpl/eval_if.hpp>
#include <boost/noncopyable.hpp>
namespace boost { namespace contract { namespace aux { namespace check {
template<
class Class,
class Intro = boost::contract::aux::none,
typename Arg0 = boost::contract::aux::none
> class subcontracted_pre_post_inv :
public boost::contract::aux::check::pre_post_inv<Class>,
private boost::noncopyable // Avoid copying captured function arguments.
{
// Base types as pointers because mpl::for_each will construct them.
typedef typename boost::mpl::transform<
typename boost::mpl::eval_if<boost::contract::aux::type_traits::
has_bases<Class>,
boost::contract::aux::type_traits::bases_of<Class>
,
boost::mpl::identity<boost::mpl::vector<> >
>::type,
boost::add_pointer<boost::mpl::placeholders::_1>
>::type base_ptrs;
public:
explicit subcontracted_pre_post_inv(boost::contract::from const from,
boost::contract::virtual_* const virt, Class* const obj, Arg0& arg0
) :
boost::contract::aux::check::pre_post_inv<Class>(from, obj),
virt_(virt),
arg0_(arg0)
{ init(); }
virtual ~subcontracted_pre_post_inv() { if(own_virt_) delete virt_; }
protected:
void copy_subcontracted_oldof() {
boost::mpl::for_each<base_ptrs>(check_base_.action(
boost::contract::virtual_::copy_oldof));
// Old-of values always copied on stack for derived func being called.
}
void check_subcontracted_entry_inv() {
boost::mpl::for_each<base_ptrs>(check_base_.action(
boost::contract::virtual_::check_entry_inv));
this->check_entry_inv();
}
// Allow to throw on failure for relaxing subcontracted pre.
void check_subcontracted_pre(bool const throw_on_failure = false) {
try {
boost::mpl::for_each<base_ptrs>(check_base_.action(
boost::contract::virtual_::check_pre));
// Pre logic-or: Last check, error also throws.
this->check_pre(throw_on_failure);
} catch(boost::contract::aux::no_error const&) {
// Pre logic-or: Stop at 1st no_error (thrown by callee).
}
}
void check_subcontracted_exit_inv() {
boost::mpl::for_each<base_ptrs>(check_base_.action(
boost::contract::virtual_::check_exit_inv));
this->check_exit_inv();
}
void check_subcontracted_post() {
boost::mpl::for_each<base_ptrs>(check_base_.action(
boost::contract::virtual_::check_post));
std::clog << "post checking" << std::endl;
check_base_.action(boost::contract::virtual_::check_this_post)(this->object());
std::clog << "post checked" << std::endl;
//this->check_post();
}
// Invariant: virt_ never null after init (so can always deref it here).
boost::contract::virtual_& virtual_call() { return *virt_; }
private:
void init() {
if(virt_) own_virt_ = false;
else {
virt_ = new boost::contract::virtual_();
own_virt_ = true;
}
check_base_.nest(this);
}
class check_base {
public:
explicit check_base() : nest_(0),
action_(boost::contract::virtual_::user_call) {}
void nest(subcontracted_pre_post_inv* n) { nest_ = n; }
check_base& action(boost::contract::virtual_::action_enum const a) {
action_ = a;
return *this;
}
template<class Base>
void operator()(Base*) {
typedef boost::contract::var result_type;
typedef boost::mpl::vector<Arg0 const&, boost::contract::virtual_*>
virtual_arg_types;
//std::clog << "*** func: " <<
// typeid(result_type).name() << " " <<
// typeid(Base).name() << " " <<
// typeid(virtual_arg_types).name() <<
//std::endl;
call_base_function<Base, result_type, virtual_arg_types>(
boost::mpl::bool_<
Intro::template has_member_function<
Base,
result_type,
virtual_arg_types,
boost::function_types::const_qualified
>::value
>()
);
}
private:
template<class Base, typename ResultType, class VirtualArgTypes>
void call_base_function(boost::mpl::false_ const&) {
//std::clog << "*** base function not found" << std::endl;
}
template<class Base, typename ResultType, class VirtualArgTypes>
void call_base_function(boost::mpl::true_ const&) {
//std::clog << "*** base function found" << std::endl;
typedef typename boost::mpl::push_front<
typename boost::mpl::push_front<VirtualArgTypes, Base*>::type,
ResultType
>::type base_virtual_func_types;
typedef typename boost::function_types::member_function_pointer<
base_virtual_func_types,
boost::function_types::const_qualified
>::type base_virtual_func_ptr;
base_virtual_func_ptr base_virtual_func = Intro::template
member_function_address<Base, base_virtual_func_ptr>();
BOOST_CONTRACT_AUX_DEBUG(base_virtual_func);
BOOST_CONTRACT_AUX_DEBUG(nest_);
Base const* const base = nest_->object();
BOOST_CONTRACT_AUX_DEBUG(base);
boost::contract::virtual_::action_enum reset_action =
nest_->virtual_call().action_;
nest_->virtual_call().action_ = action_;
try {
(base->*base_virtual_func)(nest_->arg0_, nest_->virt_);
} catch(boost::contract::aux::no_error const&) {
if(nest_->virtual_call().action_ ==
boost::contract::virtual_::check_pre) {
nest_->virtual_call().action_ = reset_action;
throw; // Pre logic-or: 1st no_err stops (throw to caller).
}
} catch(...) {
if(nest_->virtual_call().action_ ==
boost::contract::virtual_::check_pre) {
// Pre logic-or: Ignore err, possibly checks up to caller.
}
}
nest_->virtual_call().action_ = reset_action;
}
subcontracted_pre_post_inv* nest_;
boost::contract::virtual_::action_enum action_;
};
bool own_virt_;
boost::contract::virtual_* virt_;
check_base check_base_;
// TODO: Support 0-to-n args.
Arg0& arg0_; // TODO: Should this be & or perfect fwd?
};
} } } } // namespace
#endif // #include guard

View File

@ -0,0 +1,15 @@
#ifndef BOOST_CONTRACT_AUX_CHECK_NOTHING_HPP_
#define BOOST_CONTRACT_AUX_CHECK_NOTHING_HPP_
namespace boost { namespace contract { namespace aux {
class check_nothing { // Used as base for all contracts (and holder for RAII).
public:
virtual ~check_nothing() {}
};
} } } // namespace
#endif // #include guard

View File

@ -0,0 +1,94 @@
#ifndef BOOST_CONTRACT_AUX_CHECK_PRE_POST_HPP_
#define BOOST_CONTRACT_AUX_CHECK_PRE_POST_HPP_
#include <boost/contract/core/exception.hpp>
#include <boost/contract/aux_/condition/check_nothing.hpp>
#include <boost/contract/aux_/call.hpp>
#include <boost/contract/aux_/exception.hpp>
#include <boost/shared_ptr.hpp>
#include <boost/function.hpp>
namespace boost { namespace contract { namespace aux {
// TODO: Does this need to be copied? Copying function<> could copy captures
// (if function is a lambda) and it could be expensive... check all classes
// that MUST be copiable, make sure their copies are effecient, make all other
// calsses noncopyable.
class check_pre_post : public boost::contract::aux::check_nothing {
public:
explicit check_pre_post(boost::contract::from from) : from_(from) {}
explicit check_pre_post(boost::contract::from from, boost::shared_ptr<
boost::contract::aux::call> call) : from_(from), call_(call) {}
virtual ~check_pre_post() {}
template<typename F>
void set_pre(F f) { pre_ = f; pre_available(); }
template<typename F>
void set_post(F f) { post_ = f; post_available(); }
protected:
void check_pre(bool throw_on_failure = false) {
if(!call_ || call_->action == boost::contract::aux::call::check_pre) {
if(pre_) {
try { pre_(); }
catch(...) {
// Subcontracted pre must throw on failure (instead of
// calling failure handler) so to be checked in logic-or.
if(throw_on_failure) throw;
boost::contract::precondition_failed(from_);
}
}
if(call_) throw boost::contract::aux::no_error();
}
}
// TODO: I should be able to get rid of throw no_error everywhere (because
// the checking call_ allows the rest of the code to just do nothing). That
// way post and exit_inv checking can go in dtor also when bind is used.
// That will require to use scoped contract = ... also in contract decl
// functions.
// Then I should be able to set a state in c to false when a
// contract decl func is first called and then to true after the
// scoped contract = ... assignment is done. OLDOF can check that state so
// if I add from bind a call with action coped_entry_oldof and
// copy_oldof_after_pre_and_inv, I can then support copying old values
// after pre/inv have been checked also in contract decl func when users
// simply assign old values old_x = OLDOF(c, ...) after the contract decl
// (old_x variable will always have to be decl before the contract decl, but
// assigned before the contract decl to be copied before pre/inv, or after
// the contract decl to be copied after pre/inv). The same syntax, assigned
// before/after contract decl, can be used when bind is not used (that
// should work without changing current impl).
// If call(), can't call from a dtor (as throw no_error on OK).
void check_post() {
if(!call_ || call_->action == boost::contract::aux::call::check_post) {
if(post_) {
try { post_(); }
catch(...) { boost::contract::postcondition_failed(from_); }
}
if(call_) throw boost::contract::aux::no_error();
}
}
virtual void pre_available() {}
virtual void post_available() {}
boost::contract::from from() const { return from_; }
boost::shared_ptr<boost::contract::aux::call> call() { return call_; }
private:
boost::function<void ()> pre_;
boost::function<void ()> post_;
boost::contract::from from_;
boost::shared_ptr<boost::contract::aux::call> call_;
};
} } }
#endif // #include guard

View File

@ -0,0 +1,92 @@
#ifndef BOOST_CONTRACT_AUX_CHECK_PRE_POST_INV_HPP_
#define BOOST_CONTRACT_AUX_CHECK_PRE_POST_INV_HPP_
#include <boost/contract/core/config.hpp>
#include <boost/contract/core/exception.hpp>
#include <boost/contract/aux_/condition/check_pre_post.hpp>
#include <boost/contract/aux_/call.hpp>
#include <boost/contract/aux_/type_traits/invariant.hpp>
#include <boost/contract/aux_/exception.hpp>
#include <boost/mpl/bool.hpp>
#include <boost/shared_ptr.hpp>
namespace boost { namespace contract { namespace aux {
template<class C>
class check_pre_post_inv : // Copyable (as shallow ptr).
public boost::contract::aux::check_pre_post {
public:
explicit check_pre_post_inv(boost::contract::from from, C const* obj) :
boost::contract::aux::check_pre_post(from), obj_(obj) {}
explicit check_pre_post_inv(boost::contract::from from, boost::shared_ptr<
boost::contract::aux::call> call, C const* obj) :
boost::contract::aux::check_pre_post(from, call), obj_(obj)
{}
virtual ~check_pre_post_inv() {}
protected:
void check_entry_inv(bool static_inv_only = false) {
if(!this->call() || this->call()->action ==
boost::contract::aux::call::check_entry_inv) {
check_inv(/* on_entry = */ true, static_inv_only);
if(this->call()) throw no_error();
}
}
// If call(), can't call from dtor (as throw no_error on OK).
void check_exit_inv(bool static_inv_only = false) {
if(!this->call() || this->call()->action ==
boost::contract::aux::call::check_exit_inv) {
check_inv(/* on_entry = */ false, static_inv_only);
if(this->call()) throw no_error();
}
}
C const* object() const { return obj_; }
private:
void check_inv(bool on_entry, bool static_inv_only) {
check_static_inv(on_entry, boost::mpl::bool_<boost::contract::aux::
has_static_invariant<C>::value>());
if(!static_inv_only) {
check_const_inv(on_entry, boost::mpl::bool_<boost::contract::aux::
has_const_invariant<C>::value>());
}
}
void check_static_inv(bool, boost::mpl::false_ const&) {}
void check_static_inv(bool on_entry, boost::mpl::true_ const&) {
try { C::BOOST_CONTRACT_CONFIG_STATIC_INVARIANT(); }
catch(...) {
if(on_entry) {
boost::contract::static_entry_invariant_failed(from());
} else {
boost::contract::static_exit_invariant_failed(from());
}
}
}
void check_const_inv(bool, boost::mpl::false_ const&) {}
void check_const_inv(bool on_entry, boost::mpl::true_ const&) {
try { obj_->BOOST_CONTRACT_CONFIG_INVARIANT(); }
catch(...) {
if(on_entry) {
boost::contract::const_entry_invariant_failed(from());
} else {
boost::contract::const_exit_invariant_failed(from());
}
}
}
// TODO: Add volatile inv here...
C const* obj_;
};
} } }
#endif // #include guard

View File

@ -2,8 +2,13 @@
#ifndef BOOST_CONTRACT_AUX_CONFIG_HPP_
#define BOOST_CONTRACT_AUX_CONFIG_HPP_
#ifndef BOOST_CONTRACT_AUX_CONFIG_DEBUG
# define BOOST_CONTRACT_AUX_CONFIG_DEBUG 1
// Disable lib's implementation assertions to check against internal errors.
// #define BOOST_CONTRACT_AUX_CONFIG_NDEBUG
// IMPORTANT: Cannot use '_' infix to prevent forming reserved C++ names
// (containing "__" if user `name` starts/ends with '_' already) so using 'X'.
#ifndef BOOST_CONTRACT_AUX_CONFIG_NAME_PREFIX
# define BOOST_CONTRACT_AUX_CONFIG_NAME_PREFIX boost_contract_auxX
#endif
#endif // #include guard

View File

@ -3,11 +3,12 @@
#define BOOST_CONTRACT_AUX_DEBUG_HPP_
#include <boost/contract/aux_/config.hpp>
#if BOOST_CONTRACT_AUX_CONFIG_DEBUG
# include <cassert>
# define BOOST_CONTRACT_AUX_DEBUG(cond) assert(cond)
#ifdef BOOST_CONTRACT_AUX_CONFIG_NDEBUG
# define BOOST_CONTRACT_AUX_DEBUG(cond)
#else
# define BOOST_COTNRACT_AUX_DEBUG(cond)
# include <cassert>
# define BOOST_CONTRACT_AUX_DEBUG(cond) (assert(cond))
#endif
#endif // #include guard

View File

@ -3,85 +3,11 @@
#define BOOST_CONTRACT_AUX_EXCEPTION_HPP_
#include <boost/contract/core/exception.hpp>
#include <exception>
#include <string>
#include <iostream>
namespace boost { namespace contract { namespace aux {
struct no_error {}; // Do not inherit from anything (this is internal only).
namespace exception_ {
enum failure_handler_key {
pre_key,
post_key,
const_entry_inv_key,
const_volatile_entry_inv_key,
static_entry_inv_key,
const_exit_inv_key,
const_volatile_exit_inv_key,
static_exit_inv_key,
};
template<failure_handler_key Key, typename From>
void default_failure_handler(From const) {
std::string s = "";
switch(Key) {
case pre_key:
s = "precondition "; break;
case post_key:
s = "postcondition "; break;
case const_entry_inv_key:
s = "const entry invariant "; break;
case const_volatile_entry_inv_key:
s = "const volatile entry invariant "; break;
case static_entry_inv_key:
s = "static entry invariant "; break;
case const_exit_inv_key:
s= "const exit invariant "; break;
case const_volatile_exit_inv_key:
s= "const volatile exit invariant "; break;
case static_exit_inv_key:
s = "static exit invariant "; break;
// No default (so compiler warning/error on missing enum case).
}
try {
throw;
} catch(boost::contract::assertion_failure const& error) {
// what = 'assertion "..." failed: ...'.
std::cerr << s << error.what() << std::endl;
} catch(std::exception const& error) {
std::cerr << s << "checking threw standard exception with " <<
"what(): " << error.what() << std::endl;
} catch(...) {
std::cerr << s << "checking threw unknown exception" <<
std::endl;
}
std::terminate(); // Default handlers log and call terminate.
}
}
// TODO: These are part of the lib state. They should prob be put in a .cpp and
// exported (so they are the same across DLL, etc.), plus they should be
// protected by mutexes.
failure_handler pre_failure_handler = &exception_::
default_failure_handler<exception_::pre_key>;
failure_handler post_failure_handler = &exception_::
default_failure_handler<exception_::post_key>;
failure_handler const_entry_inv_failure_handler = &exception_::
default_failure_handler<exception_::const_entry_inv_key>;
failure_handler const_volatile_entry_inv_failure_handler = &exception_::
default_failure_handler<exception_::const_volatile_entry_inv_key>;
failure_handler static_entry_inv_failure_handler = &exception_::
default_failure_handler<exception_::static_entry_inv_key>;
failure_handler const_exit_inv_failure_handler = &exception_::
default_failure_handler<exception_::const_exit_inv_key>;
failure_handler const_volatile_exit_inv_failure_handler = &exception_::
default_failure_handler<exception_::const_volatile_exit_inv_key>;
failure_handler static_exit_inv_failure_handler = &exception_::
default_failure_handler<exception_::static_exit_inv_key>;
class no_error {}; // Must have no bases (just a tag type to signal no error).
} } } // namespace

View File

@ -0,0 +1,62 @@
#ifndef BOOST_CONTRACT_AUX_BIND_HPP_
#define BOOST_CONTRACT_AUX_BIND_HPP_
#include <boost/contract/core/call.hpp>
#include <boost/contract/aux_/condition/check_nothing.hpp>
#include <boost/contract/aux_/call.hpp>
#include <boost/contract/aux_/exception.hpp>
#include <boost/make_shared.hpp>
#include <boost/function_types/is_member_pointer.hpp>
namespace boost { namespace contract { namespace aux {
template<typename F, typename A0, typename A1>
class bind : // Shallow ptr copies or &.
public boost::contract::aux::check_nothing {
public:
explicit bind(F f, A0 const& a0, A1 const& a1) :
c_(boost::make_shared<boost::contract::aux::call>()),
f_(f), a0_(a0), a1_(a1)
{
execute(boost::contract::aux::call::copy_oldof);
execute(boost::contract::aux::call::check_entry_inv);
execute(boost::contract::aux::call::check_pre);
}
~bind() {
if(!std::uncaught_exception()) { // Body did not throw.
execute(boost::contract::aux::call::check_exit_inv);
execute(boost::contract::aux::call::check_post);
}
}
private:
void execute(boost::contract::aux::call::action_enum a) {
c_.call_->action = a;
try { // Call contract decl func for given action.
call(boost::mpl::bool_<boost::function_types::
is_member_pointer<F>::value>());
} catch(boost::contract::aux::no_error&) {
// no_error exception used to signal OK (so just continue).
} // Else: If other exceptions, throw them to caller.
}
void call(boost::mpl::false_ const&) {
(*f_)(a0_, a1_, c_);
}
void call(boost::mpl::true_ const&) {
(a0_->*f_)(a1_, c_);
}
boost::contract::call c_; // Copy as ptr.
F f_; // Copy as func. ptr.
A0 const& a0_; // Object for member func ptrs, 1st func arg otherwise.
A1 const& a1_;
};
} } } // namespace
#endif // #include guard

View File

@ -1,46 +0,0 @@
#ifndef BOOST_CONTRACT_AUX_FUNCTION_CONSTRUCTOR_HPP_
#define BOOST_CONTRACT_AUX_FUNCTION_CONSTRUCTOR_HPP_
#include <boost/contract/core/exception.hpp>
#include <boost/contract/aux_/check/pre_post_inv.hpp>
#include <exception>
namespace boost { namespace contract { namespace aux { namespace function {
template<class Class>
class constructor : public boost::contract::aux::check::pre_post_inv<Class> {
public:
explicit constructor(Class* const obj) : boost::contract::aux::check::
pre_post_inv<Class>(boost::contract::from_constructor, obj) {
entry();
}
~constructor() { exit(); }
private:
// No object before ctor body so only static inv at entry.
void entry() { this->check_entry_inv(/* static_inv_only = */ true); }
// Ctor pre checked by constructor_precondition at start of init list.
void pre_available() /* override */ { BOOST_CONTRACT_AUX_DEBUG(false); }
// Ctor post always checked later, at exit (see below).
void post_available() /* override */ {}
// If ctor body threw, only check static inv, otherwise obj constructed so
// check also non-static inv and post (subcontracting implemented
// automatically via C++ object constructor mechanism, so no calls to
// check_subcontracted_... in this case).
void exit() {
bool const body_threw = std::uncaught_exception();
this->check_exit_inv(/* static_inv_only = */ body_threw);
if(!body_threw) this->check_post();
}
};
} } } } // namespace
#endif // #include guard

View File

@ -1,53 +0,0 @@
#ifndef BOOST_CONTRACT_AUX_FUNCTION_DESTRUCTOR_HPP_
#define BOOST_CONTRACT_AUX_FUNCTION_DESTRUCTOR_HPP_
#include <boost/contract/core/exception.hpp>
#include <boost/contract/aux_/check/pre_post_inv.hpp>
#include <boost/contract/aux_/debug.hpp>
#include <exception>
namespace boost { namespace contract { namespace aux { namespace function {
template<class Class>
class destructor : public boost::contract::aux::check::pre_post_inv<Class> {
public:
explicit destructor(Class* const obj) : boost::contract::aux::check::
pre_post_inv<Class>(boost::contract::from_destructor, obj) {
entry();
}
~destructor() { exit(); }
private:
// Obj still exists (before dtor body) so check static and non-static inv
// (subcontracting implemented automatically via C++ object destruction
// mechanism, so no calls to check_subcontracted_... in this case).
void entry() { this->check_entry_inv(); }
// Dtor cannot have pre because it has no parameters.
void pre_available() /* override */ { BOOST_CONTRACT_AUX_DEBUG(false); }
// Ctor post always checked after body, at exit (see below).
// NOTE: Even if there is no obj after dtor body, this library allows for
// dtor post (e.g., to check static members for an instance counter class).
void post_available() /* override */ {}
// If dtor body threw, obj still exists so check subcontracted static and
// non-static inv (but no post because of throw), otherwise obj destructed
// so check static static inv and post (subcontracting implemented
// automatically via C++ object destruction mechanism, so no calls to
// check_subcontracted_... in this case).
// NOTE: In theory C++ destructors should not throw, but the language allows
// for that so this library must handle such a case.
void exit() {
bool const body_threw = std::uncaught_exception();
this->check_exit_inv(/* static_inv_only = */ !body_threw);
if(!body_threw) this->check_post();
}
};
} } } } // namespace
#endif // #include guard

View File

@ -1,41 +1,43 @@
#ifndef BOOST_CONTRACT_AUX_FUNCTION_FREE_FUNCTION_HPP_
#define BOOST_CONTRACT_AUX_FUNCTION_FREE_FUNCTION_HPP_
#ifndef BOOST_CONTRACT_AUX_FREE_FUNCTION_HPP_
#define BOOST_CONTRACT_AUX_FREE_FUNCTION_HPP_
#include <boost/contract/core/exception.hpp>
#include <boost/contract/aux_/check/pre_post.hpp>
#include <boost/contract/core/call.hpp>
#include <boost/contract/aux_/condition/check_pre_post.hpp>
#include <boost/shared_ptr.hpp>
#include <exception>
namespace boost { namespace contract { namespace aux { namespace function {
namespace boost { namespace contract { namespace aux {
class free_function : public boost::contract::aux::check::pre_post {
class free_function : public boost::contract::aux::check_pre_post {
public:
explicit free_function(boost::contract::from const from =
boost::contract::from_free_function) :
boost::contract::aux::check::pre_post(from)
{ entry(); }
explicit free_function() :
boost::contract::aux::check_pre_post(
boost::contract::from_free_function)
{}
explicit free_function(boost::shared_ptr<boost::contract::aux::call> call) :
boost::contract::aux::check_pre_post(
boost::contract::from_free_function, call)
{}
virtual ~free_function() { exit(); }
private:
// Do nothing (not a public member so no inv to check, nor subcontracting).
void entry() {}
// Check pre (as soon as related functor set).
void pre_available() /* override */ { check_pre(); }
void post_available() /* override */ {
if(call()) check_post(); // Throw (so not in dtor).
}
// Post always checked after body, at exit (see below).
void post_available() /* override */ {}
// If body did not throw, check post (not a public member so no inv to
// check, nor subcontracting).
void exit() {
bool const body_threw = std::uncaught_exception();
if(!body_threw) check_post();
public:
~free_function() {
if(!call() && !std::uncaught_exception()) { // Body didn't throw.
check_post();
}
}
};
} } } } // namespace
} } } // namespace
#endif // #include guard

View File

@ -1,23 +0,0 @@
#ifndef BOOST_CONTRACT_AUX_FUNCTION_PRIVATE_MEMBER_HPP_
#define BOOST_CONTRACT_AUX_FUNCTION_PRIVATE_MEMBER_HPP_
#include <boost/contract/core/exception.hpp>
#include <boost/contract/aux_/function/free_function.hpp>
namespace boost { namespace contract { namespace aux { namespace function {
// Not public members (i.e., private members) are allowed to fail class
// invariants (so no inv) and they do not participate in virtual function
// polymorphism according to substitution principle (so no subcontracting).
// Therefore, their contracts behave like contracts of free functions.
class private_member : public free_function {
public:
explicit private_member() :
free_function(boost::contract::from_private_member) {}
};
} } } } // namespace
#endif // #include guard

View File

@ -1,23 +0,0 @@
#ifndef BOOST_CONTRACT_AUX_FUNCTION_PROTECTED_MEMBER_HPP_
#define BOOST_CONTRACT_AUX_FUNCTION_PROTECTED_MEMBER_HPP_
#include <boost/contract/core/exception.hpp>
#include <boost/contract/aux_/function/free_function.hpp>
namespace boost { namespace contract { namespace aux { namespace function {
// Not public members (i.e., protected members) are allowed to fail class
// invariants (so no inv) and they do not participate in virtual function
// polymorphism according to substitution principle (so no subcontracting).
// Therefore, their contracts behave like contracts of free functions.
class protected_member : public free_function {
public:
explicit protected_member() :
free_function(boost::contract::from_protected_member) {}
};
} } } } // namespace
#endif // #include guard

View File

@ -1,147 +1,50 @@
#ifndef BOOST_CONTRACT_AUX_FUNCTION_PUBLIC_MEMBER_HPP_
#define BOOST_CONTRACT_AUX_FUNCTION_PUBLIC_MEMBER_HPP_
#ifndef BOOST_CONTRACT_AUX_PUBLIC_MEMBER_HPP_
#define BOOST_CONTRACT_AUX_PUBLIC_MEMBER_HPP_
#include <boost/contract/core/virtual.hpp>
#include <boost/contract/core/exception.hpp>
#include <boost/contract/aux_/check/subcontracted_pre_post_inv.hpp>
#include <boost/contract/aux_/exception.hpp>
#include <boost/contract/aux_/none.hpp>
#include <boost/contract/aux_/debug.hpp>
#include <boost/contract/aux_/call.hpp>
#include <boost/contract/aux_/condition/check_pre_post_inv.hpp>
#include <boost/shared_ptr.hpp>
#include <exception>
// TODO: Consider passing params as in (by mutable value or const&), inout
// (by & or *).
namespace boost { namespace contract { namespace aux {
namespace boost { namespace contract { namespace aux { namespace function {
template<
class Class,
class Intro = boost::contract::aux::none,
typename Arg0 = boost::contract::aux::none
> class public_member : public boost::contract::aux::check::
subcontracted_pre_post_inv<Class, Intro, Arg0> {
template<class C>
class public_member : public boost::contract::aux::check_pre_post_inv<C> {
public:
// Must be used when bases and virtual body (but can also always be used).
explicit public_member(boost::contract::virtual_* const virt,
Class* const obj, Arg0& arg0) :
boost::contract::aux::check::subcontracted_pre_post_inv<Class, Intro,
Arg0>(boost::contract::from_public_member, virt, obj,
arg0
)
{ entry(); }
// Can be used when bases and no virtual body.
explicit public_member(Class* const obj, Arg0& arg0) :
boost::contract::aux::check::subcontracted_pre_post_inv<Class, Intro,
Arg0>(boost::contract::from_public_member, /* virt = */ 0, obj,
arg0
)
{ entry(); }
// Can be used when no bases and virtual body.
explicit public_member(boost::contract::virtual_* const virt,
Class* const obj) :
boost::contract::aux::check::subcontracted_pre_post_inv<Class, Intro,
Arg0>(boost::contract::from_public_member, virt, obj,
boost::contract::aux::none::value
)
{ entry(); }
// Can be used when no bases and no virtual body.
explicit public_member(Class* const obj) :
boost::contract::aux::check::subcontracted_pre_post_inv<Class, Intro,
Arg0>(boost::contract::from_public_member, /* virt = */ 0, obj,
boost::contract::aux::none()
)
{ entry(); }
~public_member() { exit(); }
explicit public_member(C const* obj) :
boost::contract::aux::check_pre_post_inv<C>(
boost::contract::from_public_member, obj)
{ init(); }
explicit public_member(boost::shared_ptr<boost::contract::aux::call> call,
C const* obj) :
boost::contract::aux::check_pre_post_inv<C>(
boost::contract::from_public_member, call, obj)
{ init(); }
private:
// Check static and non-static subcontracted inv.
void entry() {
if(
this->virtual_call().action_ ==
boost::contract::virtual_::user_call ||
this->virtual_call().action_ ==
boost::contract::virtual_::copy_oldof
) {
this->copy_subcontracted_oldof();
if(this->virtual_call().action_ !=
boost::contract::virtual_::user_call) {
throw boost::contract::aux::no_error();
}
}
if( // When inv only, also must check exit inv here to skip body.
this->virtual_call().action_ ==
boost::contract::virtual_::user_call ||
this->virtual_call().action_ ==
boost::contract::virtual_::check_entry_inv ||
this->virtual_call().action_ ==
boost::contract::virtual_::check_exit_inv
) {
if(this->virtual_call().action_ ==
boost::contract::virtual_::check_exit_inv) {
this->check_subcontracted_exit_inv();
} else { // For both user call and entry inv only.
this->check_subcontracted_entry_inv();
}
if(this->virtual_call().action_ !=
boost::contract::virtual_::user_call) {
throw boost::contract::aux::no_error();
}
} // Else (check only pre, post, etc.) do nothing.
void init() {
this->check_entry_inv();
if(this->call()) this->check_exit_inv(); // Throw (so not in dtor).
}
// Check subcontracted pre (as soon as related functor set).
void pre_available() /* override */ {
if(
this->virtual_call().action_ ==
boost::contract::virtual_::user_call ||
this->virtual_call().action_ ==
boost::contract::virtual_::check_pre
) {
this->check_subcontracted_pre(
/* throw_on_failure = */ this->virtual_call().action_ !=
boost::contract::virtual_::user_call
);
if(this->virtual_call().action_ !=
boost::contract::virtual_::user_call) {
throw boost::contract::aux::no_error();
}
} // Else (check only inv, post, etc.) do nothing.
}
// Check post here only if check-post-only mode (otherwise check at exit).
void pre_available() /* override */ { this->check_pre(); }
void post_available() /* override */ {
if(this->virtual_call().action_ ==
boost::contract::virtual_::check_post) {
this->check_subcontracted_post();
throw boost::contract::aux::no_error();
} // Else (check only inv, pre, etc.) do nothing.
if(this->virtual_call().action_ ==
boost::contract::virtual_::check_this_post) {
std::clog << "****************" << std::endl;
this->check_post();
throw boost::contract::aux::no_error();
} // Else (check only inv, pre, etc.) do nothing.
if(this->call()) this->check_post(); // Throw (so not in dtor).
}
// Check static and non-static subcontracted invariant and, if body did not
// throw, also check subcontracted post.
void exit() {
bool const body_threw = std::uncaught_exception();
if(this->virtual_call().action_ ==
boost::contract::virtual_::user_call) {
this->check_subcontracted_exit_inv();
if(!body_threw) this->check_subcontracted_post();
public:
~public_member() {
if(!this->call() && !std::uncaught_exception()) { // Body didn't throw.
this->check_exit_inv();
this->check_post();
}
}
};
} } } } // namespace
} } } // namespace
#endif // #include guard

View File

@ -1,47 +0,0 @@
#ifndef BOOST_CONTRACT_AUX_FUNCTION_PUBLIC_STATIC_MEMBER_HPP_
#define BOOST_CONTRACT_AUX_FUNCTION_PUBLIC_STATIC_MEMBER_HPP_
#include <boost/contract/core/exception.hpp>
#include <boost/contract/aux_/check/pre_post_inv.hpp>
#include <exception>
namespace boost { namespace contract { namespace aux { namespace function {
template<class Class> class public_static_member :
public boost::contract::aux::check::pre_post_inv<Class> {
public:
explicit public_static_member() : boost::contract::aux::check::
pre_post_inv<Class>(boost::contract::from_public_member) {
entry();
}
~public_static_member() { exit(); }
private:
// Static members have no object so they do not participate in virtual
// function polymorphism according to substitution principle (so no
// subcontracting for any of the checks below).
// Static so no object so check static inv only.
void entry() { this->check_entry_inv(/* static_inv_only = */ true); }
// Check pre (as soon as related functor set).
void pre_available() /* override */ { this->check_pre(); }
// Post always checked after body, at exit (see below).
void post_available() /* override */ {}
// Static so no object and only static inv checked, plus check post but
// only if body did not throw.
void exit() {
bool const body_threw = std::uncaught_exception();
this->check_exit_inv(/* static_inv_only = */ true);
if(!body_threw) this->check_post();
}
};
} } } } // namespace
#endif // #include guard

View File

@ -2,18 +2,16 @@
#ifndef BOOST_CONTRACT_AUX_NAME_HPP_
#define BOOST_CONTRACT_AUX_NAME_HPP_
#include <boost/contract/aux_/config.hpp>
#include <boost/preprocessor/cat.hpp>
/* PUBLIC */
// IMPORTANT: Cannot use '_' infix to prevent forming reserved C++ names
// (containing "__" if user `name` starts/ends with '_' already) so using 'X'.
// NOTE: Explicitly list number of names to concatenate using ..._NAME-n
// (instead of using ..._SEQ_CAT or similar) for optimal speed and reentrancy.
#define BOOST_CONTRACT_AUX_NAME1(name1) \
BOOST_PP_CAT(boost_contract_auxX, name1)
BOOST_PP_CAT(BOOST_CONTRACT_AUX_CONFIG_NAME_PREFIX, name1)
#endif // #include guard

View File

@ -4,7 +4,12 @@
namespace boost { namespace contract { namespace aux {
struct none { static none value; };
struct none { // Tag for "no type".
static none value; // Tag for "no value".
// Must contain no functionality (so it cannot actually be used by mistake).
};
none none::value;
} } } // namespace

View File

@ -1,68 +0,0 @@
#ifndef BOOST_CONTRACT_AUX_OLDOF_HPP_
#define BOOST_CONTRACT_AUX_OLDOF_HPP_
#include <boost/contract/core/virtual.hpp>
#include <boost/contract/aux_/debug.hpp>
#include <boost/make_shared.hpp>
#include <boost/shared_ptr.hpp>
namespace boost { namespace contract { namespace aux {
class oldof { // Must allow copies (shallow pointer copies).
public:
oldof() : virt_(), value_() {} // Null value ptr when no oldof.
explicit oldof(boost::contract::virtual_* const virt, oldof const& old) :
virt_(virt), value_(old.value_) {}
template<typename T>
/* implicit */ oldof(T const& old_value) :
value_(boost::make_shared<T>(old_value)) {} // One single copy of T.
// Pointer so it can be null (for when no post) and shared because owned by
// this lib for virtual calls (stored in v) but owned by user's stack value
// for non-virtual calls (so shared ownership to handle both cases).
template<typename T>
operator boost::shared_ptr<T const>() {
if(!virt_) {
BOOST_CONTRACT_AUX_DEBUG(value_);
boost::shared_ptr<T const> result =
boost::static_pointer_cast<T const>(value_);
BOOST_CONTRACT_AUX_DEBUG(result);
return result;
} else if(
virt_->action_ == boost::contract::virtual_::user_call ||
virt_->action_ == boost::contract::virtual_::copy_oldof
) {
BOOST_CONTRACT_AUX_DEBUG(value_);
virt_->old_values_.push(value_);
std::clog << "pushed" << std::endl;
return boost::shared_ptr<T const>();
} else if(
virt_->action_ == boost::contract::virtual_::check_post ||
virt_->action_ == boost::contract::virtual_::check_this_post
) {
BOOST_CONTRACT_AUX_DEBUG(!value_);
boost::shared_ptr<void> value = virt_->old_values_.front();
virt_->old_values_.pop();
std::clog << "popped" << std::endl;
BOOST_CONTRACT_AUX_DEBUG(value);
boost::shared_ptr<T const> result =
boost::static_pointer_cast<T const>(value);
BOOST_CONTRACT_AUX_DEBUG(result);
return result;
}
BOOST_CONTRACT_AUX_DEBUG(!value_);
return boost::shared_ptr<T>();
}
private:
boost::contract::virtual_* virt_;
boost::shared_ptr<void> value_; // Type erased to `void*`.
};
} } } // namespace
#endif // #include guard

View File

@ -1,29 +0,0 @@
#ifndef BOOST_CONTRACT_AUX_PP_KEYWORD_PRIVATE_HPP_
#define BOOST_CONTRACT_AUX_PP_KEYWORD_PRIVATE_HPP_
#include <boost/contract/aux_/preprocessor/keyword/utility/is.hpp>
#include <boost/preprocessor/cat.hpp>
/* PUBLIC */
// Precondition: tokens must start with a token concatenable to a macro name
// (e.g., a literal or integral token).
#define BOOST_CONTRACT_AUX_PP_KEYWORD_IS_PRIVATE(tokens) \
BOOST_CONTRACT_AUX_PP_KEYWORD_UTILITY_IS( \
BOOST_CONTRACT_AUX_PP_KEYWORD_PRIVATE_CAT_TO_COMMA, tokens)
// Precondition: tokens must start with `private` (this can be
// checked with `..._IS_PRIVATE` macro above).
#define BOOST_CONTRACT_AUX_PP_KEYWORD_REMOVE_PRIVATE(tokens) \
BOOST_PP_CAT(BOOST_CONTRACT_AUX_PP_KEYWORD_PRIVATE_CAT_TO_EMPTY, tokens)
/* PRIVATE */
// Must expand to a single comma `,` (not local macros, do not #undefine).
#define BOOST_CONTRACT_AUX_PP_KEYWORD_PRIVATE_CAT_TO_COMMAprivate ,
// Must expand to empty `` (not local macros, do not #undefine).
#define BOOST_CONTRACT_AUX_PP_KEYWORD_PRIVATE_CAT_TO_EMPTYprivate
#endif // #include guard

View File

@ -1,29 +0,0 @@
#ifndef BOOST_CONTRACT_AUX_PP_KEYWORD_PROTECTED_HPP_
#define BOOST_CONTRACT_AUX_PP_KEYWORD_PROTECTED_HPP_
#include <boost/contract/aux_/preprocessor/keyword/utility/is.hpp>
#include <boost/preprocessor/cat.hpp>
/* PUBLIC */
// Precondition: tokens must start with a token concatenable to a macro name
// (e.g., a literal or integral token).
#define BOOST_CONTRACT_AUX_PP_KEYWORD_IS_PROTECTED(tokens) \
BOOST_CONTRACT_AUX_PP_KEYWORD_UTILITY_IS( \
BOOST_CONTRACT_AUX_PP_KEYWORD_PROTECTED_CAT_TO_COMMA, tokens)
// Precondition: tokens must start with `protected` (this can be
// checked with `..._IS_PROTECTED` macro above).
#define BOOST_CONTRACT_AUX_PP_KEYWORD_REMOVE_PROTECTED(tokens) \
BOOST_PP_CAT(BOOST_CONTRACT_AUX_PP_KEYWORD_PROTECTED_CAT_TO_EMPTY, tokens)
/* PRIVATE */
// Must expand to a single comma `,` (not local macros, do not #undefine).
#define BOOST_CONTRACT_AUX_PP_KEYWORD_PROTECTED_CAT_TO_COMMAprotected ,
// Must expand to empty `` (not local macros, do not #undefine).
#define BOOST_CONTRACT_AUX_PP_KEYWORD_PROTECTED_CAT_TO_EMPTYprotected
#endif // #include guard

View File

@ -1,29 +0,0 @@
#ifndef BOOST_CONTRACT_AUX_PP_KEYWORD_PUBLIC_HPP_
#define BOOST_CONTRACT_AUX_PP_KEYWORD_PUBLIC_HPP_
#include <boost/contract/aux_/preprocessor/keyword/utility/is.hpp>
#include <boost/preprocessor/cat.hpp>
/* PUBLIC */
// Precondition: tokens must start with a token concatenable to a macro name
// (e.g., a literal or integral token).
#define BOOST_CONTRACT_AUX_PP_KEYWORD_IS_PUBLIC(tokens) \
BOOST_CONTRACT_AUX_PP_KEYWORD_UTILITY_IS( \
BOOST_CONTRACT_AUX_PP_KEYWORD_PUBLIC_CAT_TO_COMMA, tokens)
// Precondition: tokens must start with `public` (this can be
// checked with `..._IS_PUBLIC` macro above).
#define BOOST_CONTRACT_AUX_PP_KEYWORD_REMOVE_PUBLIC(tokens) \
BOOST_PP_CAT(BOOST_CONTRACT_AUX_PP_KEYWORD_PUBLIC_CAT_TO_EMPTY, tokens)
/* PRIVATE */
// Must expand to a single comma `,` (not local macros, do not #undefine).
#define BOOST_CONTRACT_AUX_PP_KEYWORD_PUBLIC_CAT_TO_COMMApublic ,
// Must expand to empty `` (not local macros, do not #undefine).
#define BOOST_CONTRACT_AUX_PP_KEYWORD_PUBLIC_CAT_TO_EMPTYpublic
#endif // #include guard

View File

@ -1,24 +0,0 @@
#ifndef BOOST_CONTRACT_EXT_PP_KEYWORD_UTILITY_IS_HPP_
#define BOOST_CONTRACT_EXT_PP_KEYWORD_UTILITY_IS_HPP_
#include <boost/preprocessor/cat.hpp>
#include <boost/preprocessor/variadic/size.hpp>
/* PUBLIC */
// Precondition: A macro named `cat_to_comma_prefix ## token-to-check` must be
// #defined to expand to `,`.
// Precondition: tokens must start with a token concatenable to a macro name
// (e.g., a literal or integral token).
#define BOOST_CONTRACT_AUX_PP_KEYWORD_UTILITY_IS(cat_to_comma_prefix, tokens) \
BOOST_PP_CAT(BOOST_CONTRACT_AUX_PP_KEYWORD_UTILITY_IS_, \
BOOST_PP_VARIADIC_SIZE(BOOST_PP_CAT(cat_to_comma_prefix, tokens)))
/* PRIVATE */
#define BOOST_CONTRACT_AUX_PP_KEYWORD_UTILITY_IS_1 0
#define BOOST_CONTRACT_AUX_PP_KEYWORD_UTILITY_IS_2 1
#endif // #include guard

View File

@ -1,28 +0,0 @@
#ifndef BOOST_CONTRACT_AUX_PP_KEYWORD_VIRTUAL_HPP_
#define BOOST_CONTRACT_AUX_PP_KEYWORD_VIRTUAL_HPP_
#include <boost/contract/aux_/preprocessor/keyword/utility/is.hpp>
/* PUBLIC */
// Precondition: tokens must start with a token concatenable to a macro name
// (e.g., a literal or integral token).
#define BOOST_CONTRACT_AUX_PP_KEYWORD_IS_VIRTUAL(tokens) \
BOOST_CONTRACT_AUX_PP_KEYWORD_UTILITY_IS( \
BOOST_CONTRACT_AUX_PP_KEYWORD_VIRTUAL_CAT_TO_COMMA, tokens)
// Precondition: tokens must start with `virtual` (this can be
// checked with `..._IS_VIRTUAL` macro above).
#define BOOST_CONTRACT_AUX_PP_KEYWORD_REMOVE_VIRTUAL(tokens) \
BOOST_PP_CAT(BOOST_CONTRACT_AUX_PP_KEYWORD_VIRTUAL_CAT_TO_EMPTY, tokens)
/* PRIVATE */
// Must expand to a single comma `,` (not local macros, do not #undefine).
#define BOOST_CONTRACT_AUX_PP_KEYWORD_VIRTUAL_CAT_TO_COMMAvirtual ,
// Must expand to empty `` (not local macros, do not #undefine).
#define BOOST_CONTRACT_AUX_PP_KEYWORD_VIRTUAL_CAT_TO_EMPTYvirtual
#endif // #include guard

View File

@ -1,19 +0,0 @@
#ifndef BOOST_CONTRACT_AUX_TYPE_TRAITS_BASES_HPP_
#define BOOST_CONTRACT_AUX_TYPE_TRAITS_BASES_HPP_
#include <boost/contract/core/config.hpp>
#include <boost/contract/aux_/type_traits/introspection.hpp>
namespace boost { namespace contract { namespace aux { namespace type_traits {
BOOST_CONTRACT_AUX_TYPE_TRAITS_INTROSPECTION_HAS_TYPE(has_bases,
BOOST_CONTRACT_CONFIG_BASE_TYPES)
template< class C >
struct bases_of { typedef typename C::BOOST_CONTRACT_CONFIG_BASE_TYPES type; };
} } } } // namespace
#endif // #include guard

View File

@ -1,6 +1,6 @@
#ifndef BOOST_CONTRACT_AUX_TYPE_TRAITS_INTROSPECTION_HPP_
#define BOOST_CONTRACT_AUX_TYPE_TRAITS_INTROSPECTION_HPP_
#ifndef BOOST_CONTRACT_AUX_INTROSPECTION_HPP_
#define BOOST_CONTRACT_AUX_INTROSPECTION_HPP_
#include <boost/contract/aux_/name.hpp>
#include <boost/function_types/member_function_pointer.hpp>
@ -19,32 +19,29 @@
/* PUBLIC */
#define BOOST_CONTRACT_AUX_TYPE_TRAITS_INTROSPECTION_HAS_TYPE(trait, type_name)\
#define BOOST_CONTRACT_AUX_INTROSPECTION_HAS_TYPE(trait, type_name)\
template<typename BOOST_CONTRACT_AUX_NAME1(T)> \
class trait { \
template<class BOOST_CONTRACT_AUX_NAME1(C)> \
static boost::contract::aux::type_traits::introspection::yes& check( \
static boost::contract::aux::introspection::yes& check( \
typename BOOST_CONTRACT_AUX_NAME1(C)::type_name*); \
\
BOOST_CONTRACT_AUX_TYPE_TRAITS_INTROSPECTION_END_( \
BOOST_CONTRACT_AUX_NAME1(T)) \
BOOST_CONTRACT_AUX_INTROSPECTION_END_(BOOST_CONTRACT_AUX_NAME1(T)) \
};
#define BOOST_CONTRACT_AUX_TYPE_TRAITS_INTROSPECTION_HAS_MEMBER_FUNCTION( \
#define BOOST_CONTRACT_AUX_INTROSPECTION_HAS_MEMBER_FUNCTION( \
trait, func_name) \
BOOST_CONTRACT_AUX_TYPE_TRAITS_INTROSPECTION_HAS_MEMBER_FUNCTION_( \
/* is_static = */ 0, trait, func_name)
BOOST_CONTRACT_AUX_INTROSPECTION_HAS_MEMBER_FUNCTION_(/* is_static = */ 0, \
trait, func_name)
#define \
BOOST_CONTRACT_AUX_TYPE_TRAITS_INTROSPECTION_HAS_STATIC_MEMBER_FUNCTION(trait, \
#define BOOST_CONTRACT_AUX_INTROSPECTION_HAS_STATIC_MEMBER_FUNCTION(trait, \
func_name) \
BOOST_CONTRACT_AUX_TYPE_TRAITS_INTROSPECTION_HAS_MEMBER_FUNCTION_( \
/* is_static = */ 1, trait, func_name)
BOOST_CONTRACT_AUX_INTROSPECTION_HAS_MEMBER_FUNCTION_(/* is_static = */ 1, \
trait, func_name)
/* PRIVATE */
#define BOOST_CONTRACT_AUX_TYPE_TRAITS_INTROSPECTION_HAS_MEMBER_FUNCTION_( \
is_static, trait, func_name) \
#define BOOST_CONTRACT_AUX_INTROSPECTION_HAS_MEMBER_FUNCTION_(is_static, \
trait, func_name) \
template< \
typename BOOST_CONTRACT_AUX_NAME1(T), \
typename BOOST_CONTRACT_AUX_NAME1(R), \
@ -53,8 +50,8 @@ BOOST_CONTRACT_AUX_TYPE_TRAITS_INTROSPECTION_HAS_STATIC_MEMBER_FUNCTION(trait, \
> \
class trait { \
template<class BOOST_CONTRACT_AUX_NAME1(C)> \
static boost::contract::aux::type_traits::introspection::yes& check( \
boost::contract::aux::type_traits::introspection::check_function< \
static boost::contract::aux::introspection::yes& check( \
boost::contract::aux::introspection::check_function< \
typename \
BOOST_PP_IIF(is_static, \
boost::function_types::function_pointer \
@ -81,33 +78,27 @@ BOOST_CONTRACT_AUX_TYPE_TRAITS_INTROSPECTION_HAS_STATIC_MEMBER_FUNCTION(trait, \
&BOOST_CONTRACT_AUX_NAME1(C)::func_name \
>* \
); \
\
BOOST_CONTRACT_AUX_TYPE_TRAITS_INTROSPECTION_END_( \
BOOST_CONTRACT_AUX_NAME1(T)) \
BOOST_CONTRACT_AUX_INTROSPECTION_END_(BOOST_CONTRACT_AUX_NAME1(T)) \
};
#define BOOST_CONTRACT_AUX_TYPE_TRAITS_INTROSPECTION_END_(tparam) \
#define BOOST_CONTRACT_AUX_INTROSPECTION_END_(tparam) \
template<typename> \
static boost::contract::aux::type_traits::introspection::no& check( \
...); \
\
static boost::contract::aux::introspection::no& check(...); \
public: \
static bool const value = sizeof(check<tparam>(0)) == \
sizeof(boost::contract::aux::type_traits::introspection::yes); \
sizeof(boost::contract::aux::introspection::yes); \
typedef boost::mpl::bool_<value> type;
/* CODE */
namespace boost { namespace contract { namespace aux { namespace type_traits {
namespace introspection {
namespace boost { namespace contract { namespace aux { namespace introspection {
typedef struct {} yes;
typedef yes no[2];
template<typename F, F>
struct check_function;
template<typename F, F> struct check_function;
} } } } } // namespace
} } } } // namespace
#endif // #include guard

View File

@ -1,22 +1,22 @@
#ifndef BOOST_CONTRACT_AUX_TYPE_TRAITS_INVARIANT_HPP_
#define BOOST_CONTRACT_AUX_TYPE_TRAITS_INVARIANT_HPP_
#ifndef BOOST_CONTRACT_AUX_INVARIANT_HPP_
#define BOOST_CONTRACT_AUX_INVARIANT_HPP_
#include <boost/contract/core/config.hpp>
#include <boost/contract/aux_/type_traits/introspection.hpp>
#include <boost/function_types/property_tags.hpp>
#include <boost/mpl/vector.hpp>
namespace boost { namespace contract { namespace aux { namespace type_traits {
namespace boost { namespace contract { namespace aux {
namespace invariant_ {
BOOST_CONTRACT_AUX_TYPE_TRAITS_INTROSPECTION_HAS_MEMBER_FUNCTION(
BOOST_CONTRACT_AUX_INTROSPECTION_HAS_MEMBER_FUNCTION(
has_invariant, BOOST_CONTRACT_CONFIG_INVARIANT)
BOOST_CONTRACT_AUX_TYPE_TRAITS_INTROSPECTION_HAS_MEMBER_FUNCTION(
BOOST_CONTRACT_AUX_INTROSPECTION_HAS_MEMBER_FUNCTION(
has_non_static_invariant, BOOST_CONTRACT_CONFIG_STATIC_INVARIANT)
BOOST_CONTRACT_AUX_TYPE_TRAITS_INTROSPECTION_HAS_STATIC_MEMBER_FUNCTION(
BOOST_CONTRACT_AUX_INTROSPECTION_HAS_STATIC_MEMBER_FUNCTION(
has_static_invariant, BOOST_CONTRACT_CONFIG_STATIC_INVARIANT)
}
@ -45,7 +45,7 @@ template<typename T>
struct has_non_static_invariant : invariant_::has_non_static_invariant<T, void,
boost::mpl::vector<> > {};
} } } } // namespace
} } } // namespace
#endif // #include guard

View File

@ -1,142 +0,0 @@
#ifndef BOOST_CONTRACT_BASE_TYPES_HPP_
#define BOOST_CONTRACT_BASE_TYPES_HPP_
#include <boost/preprocessor/config/config.hpp>
#if !BOOST_PP_VARIADICS
#define BOOST_CONTRACT_BASE_TYPES BOOST_CONTRACT_ERROR_variadic_macros_required_by_BOOST_CONTRACT_BASE_TYPES_otherwise_manually_program_base_types
#else
#include <boost/contract/aux_/preprocessor/keyword/public.hpp>
#include <boost/contract/aux_/preprocessor/keyword/protected.hpp>
#include <boost/contract/aux_/preprocessor/keyword/private.hpp>
#include <boost/contract/aux_/preprocessor/keyword/virtual.hpp>
#include <boost/mpl/vector.hpp>
#include <boost/preprocessor/seq/enum.hpp>
#include <boost/preprocessor/seq/fold_left.hpp>
#include <boost/preprocessor/seq/push_back.hpp>
#include <boost/preprocessor/seq/size.hpp>
#include <boost/preprocessor/seq/seq.hpp>
#include <boost/preprocessor/variadic/to_seq.hpp>
#include <boost/preprocessor/tuple/eat.hpp>
#include <boost/preprocessor/tuple/rem.hpp>
#include <boost/preprocessor/tuple/elem.hpp>
#include <boost/preprocessor/comparison/equal.hpp>
#include <boost/preprocessor/control/iif.hpp>
#include <boost/preprocessor/facilities/expand.hpp>
/* PUBLIC */
#define BOOST_CONTRACT_BASE_TYPES(...) \
BOOST_CONTRACT_BASE_TYPES_CHECK_((__VA_ARGS__), \
BOOST_PP_VARIADIC_TO_SEQ(__VA_ARGS__))
/* PRIVATE */
// Cannot check that all base types have access specifiers (unless users have to
// specify bases using pp-seq, because user specified base list can have
// unwrapped commas between bases but also within a given base type, when base
// types are templates), but at least check the very first base type explicitly
// specifies access `[virtual] public | protected | private [virtual] ...`.
#define BOOST_CONTRACT_BASE_TYPES_CHECK_(bases_tuple, bases_seq) \
BOOST_PP_IIF(BOOST_CONTRACT_BASE_TYPES_IS_ACCESS_( \
BOOST_CONTRACT_BASE_TYPES_REMOVE_VIRTUAL_(BOOST_PP_SEQ_HEAD( \
bases_seq))), \
BOOST_CONTRACT_BASE_TYPES_OK_ \
, \
BOOST_CONTRACT_BASE_TYPES_ERR_ \
)(bases_tuple, bases_seq)
#define BOOST_CONTRACT_BASE_TYPES_ERR_(bases_tuple, bases_seq) \
BOOST_CONTRACT_ERROR_all_bases_must_explicitly_specify_public_protected_or_private base_tuple
#define BOOST_CONTRACT_BASE_TYPES_OK_(base_tuple, bases_seq) \
boost::mpl::vector< \
BOOST_CONTRACT_BASE_TYPES_RETURN_(BOOST_PP_TUPLE_ELEM(2, 1, \
BOOST_PP_SEQ_FOLD_LEFT( \
BOOST_CONTRACT_BASE_TYPES_, \
(0, (BOOST_PP_NIL)), \
bases_seq \
) \
)) \
>
#define BOOST_CONTRACT_BASE_TYPES_RETURN_(types_nilseq) \
BOOST_PP_IIF(BOOST_PP_EQUAL(BOOST_PP_SEQ_SIZE(types_nilseq), 1), \
BOOST_PP_TUPLE_EAT(1) \
, \
BOOST_CONTRACT_BASE_TYPES_RETURN_YES_ \
)(types_nilseq)
#define BOOST_CONTRACT_BASE_TYPES_RETURN_YES_(types_nilseq) \
BOOST_PP_SEQ_ENUM(BOOST_PP_SEQ_TAIL(types_nilseq))
#define BOOST_CONTRACT_BASE_TYPES_(s, public_types, base) \
BOOST_CONTRACT_BASE_TYPES_ACCESS_( \
BOOST_PP_TUPLE_ELEM(2, 0, public_types), \
BOOST_PP_TUPLE_ELEM(2, 1, public_types), \
BOOST_CONTRACT_BASE_TYPES_REMOVE_VIRTUAL_(base) \
)
#define BOOST_CONTRACT_BASE_TYPES_ACCESS_(is_public, types_nilseq, base) \
BOOST_PP_IIF(BOOST_CONTRACT_AUX_PP_KEYWORD_IS_PUBLIC(base), \
BOOST_CONTRACT_BASE_TYPES_PUSH_BACK_PUBLIC_ \
, BOOST_PP_IIF(BOOST_CONTRACT_AUX_PP_KEYWORD_IS_PROTECTED(base), \
BOOST_CONTRACT_BASE_TYPES_SKIP_NOT_PUBLIC_ \
, BOOST_PP_IIF(BOOST_CONTRACT_AUX_PP_KEYWORD_IS_PRIVATE(base), \
BOOST_CONTRACT_BASE_TYPES_SKIP_NOT_PUBLIC_ \
, \
BOOST_CONTRACT_BASE_TYPES_PUSH_BACK_IF_ \
)))(is_public, types_nilseq, base)
// Precondition: base = `public [virtual] ...`.
#define BOOST_CONTRACT_BASE_TYPES_PUSH_BACK_PUBLIC_(is_public, types_nilseq, \
base) \
( \
1, \
BOOST_PP_SEQ_PUSH_BACK(types_nilseq, \
BOOST_CONTRACT_BASE_TYPES_REMOVE_VIRTUAL_( \
BOOST_CONTRACT_AUX_PP_KEYWORD_REMOVE_PUBLIC(base)) \
) \
)
#define BOOST_CONTRACT_BASE_TYPES_SKIP_NOT_PUBLIC_(is_public, types_nilseq, \
base) \
(0, types_nilseq)
#define BOOST_CONTRACT_BASE_TYPES_PUSH_BACK_IF_(is_public, types_nilseq, base) \
( \
is_public, \
BOOST_PP_IIF(is_public, \
BOOST_PP_SEQ_PUSH_BACK \
, \
types_nilseq BOOST_PP_TUPLE_EAT(2) \
)(types_nilseq, base) \
)
#define BOOST_CONTRACT_BASE_TYPES_REMOVE_VIRTUAL_(base) \
BOOST_PP_EXPAND( \
BOOST_PP_IIF(BOOST_CONTRACT_AUX_PP_KEYWORD_IS_VIRTUAL(base), \
BOOST_CONTRACT_AUX_PP_KEYWORD_REMOVE_VIRTUAL \
, \
BOOST_PP_TUPLE_REM(1) \
)(base) \
)
#define BOOST_CONTRACT_BASE_TYPES_IS_ACCESS_(base) \
BOOST_PP_IIF(BOOST_CONTRACT_AUX_PP_KEYWORD_IS_PUBLIC(base), \
1 \
, BOOST_PP_IIF(BOOST_CONTRACT_AUX_PP_KEYWORD_IS_PROTECTED(base), \
1 \
, BOOST_PP_IIF(BOOST_CONTRACT_AUX_PP_KEYWORD_IS_PRIVATE(base), \
1 \
, \
0 \
)))
#endif // BOOST_PP_VARIADICS
#endif // #include guard

View File

@ -0,0 +1,22 @@
#ifndef BOOST_CONTRACT_BIND_HPP_
#define BOOST_CONTRACT_BIND_HPP_
#include <boost/contract/core/set_nothing.hpp>
#include <boost/contract/aux_/function/bind.hpp>
#include <boost/make_shared.hpp>
namespace boost { namespace contract {
// A0 is object ptr if F is member func ptr, 1st func arg otherwise.
template<typename F, typename A0, typename A1>
boost::contract::set_nothing bind(F f, A0 const& a0, A1 const& a1) {
return boost::contract::set_nothing(boost::make_shared<boost::contract::
aux::bind<F, A0, A1> >(f, a0, a1));
}
} } // namespace
#endif // #include guard

View File

@ -1,37 +0,0 @@
#ifndef BOOST_CONTRACT_CONSTRUCTOR_HPP_
#define BOOST_CONTRACT_CONSTRUCTOR_HPP_
#include <boost/contract/core/var.hpp> // Here so user can store contract.
#include <boost/contract/core/set_postcondition_only.hpp>
#include <boost/contract/aux_/exception.hpp>
#include <boost/contract/aux_/function/constructor.hpp>
#include <boost/make_shared.hpp>
namespace boost { namespace contract {
template<class Class>
boost::contract::set_postcondition_only constructor(Class* const object) {
return boost::contract::set_postcondition_only(boost::make_shared<
boost::contract::aux::function::constructor<Class> >(object));
}
// Uses Class tparam to avoid multiple inheritance from same type.
template<class Class>
struct constructor_precondition {
constructor_precondition() {} // For overloaded constructors with no pre.
template<typename Precondition>
explicit constructor_precondition(Precondition const& f) {
try { f(); }
catch(...) {
boost::contract::aux::pre_failure_handler(
boost::contract::from_constructor);
}
}
};
} } // namespace
#endif // #include guard

View File

@ -0,0 +1,51 @@
#ifndef BOOST_CONTRACT_CALL_HPP_
#define BOOST_CONTRACT_CALL_HPP_
#include <boost/contract/aux_/call.hpp>
#include <boost/shared_ptr.hpp>
namespace boost {
namespace contract {
class set_precondition_postcondition;
namespace aux {
template<typename, typename, typename>
class bind;
}
}
}
namespace boost { namespace contract {
class call {
public:
// Nothing here (opaque object that users cannot directly use or create).
private: // Friendship to minimize lib's public API.
friend bool copy_oldof(call const&);
friend class oldof;
template<typename, typename, typename>
friend class boost::contract::aux::bind;
friend boost::contract::set_precondition_postcondition free_function(
call const&);
template<class C>
friend boost::contract::set_precondition_postcondition public_member(
boost::contract::call const&, C const*);
explicit call(boost::shared_ptr<boost::contract::aux::call> _call) :
call_(_call) {}
call(call const& other) : call_(other.call_) {} // Shallow ptr copy.
call& operator=(call const&); // No copy operator.
boost::shared_ptr<boost::contract::aux::call> call_;
};
} } // namespace
#endif // #include guard

View File

@ -2,30 +2,15 @@
#ifndef BOOST_CONTRACT_CONFIG_HPP_
#define BOOST_CONTRACT_CONFIG_HPP_
// TODO: Implement NO_... macros to disable pre/post/inv-entry/inv-exit/etc.
// The name of the invariant member function ("invariant" by default).
#ifndef BOOST_CONTRACT_CONFIG_INVARIANT
# define BOOST_CONTRACT_CONFIG_INVARIANT invariant
#endif
// The name of the static invariant member function ("static invariant" by
// default).
// NOTE: C++ forbids function overload based on static classifier so a new
// function name, different from invariant's function name, had to be used.
// C++ does not allow to overload member functions based on static classifier,
// so a name different from the non-static class invariant member must be used.
#ifndef BOOST_CONTRACT_CONFIG_STATIC_INVARIANT
# define BOOST_CONTRACT_CONFIG_STATIC_INVARIANT static_invariant
#endif
// The name of the base types type-def ("base_types" by default).
#ifndef BOOST_CONTRACT_CONFIG_BASE_TYPES
# define BOOST_CONTRACT_CONFIG_BASE_TYPES base_types
#endif
// Disable some sanity checks on usage of APIs (0 by default).
#ifndef BOOST_CONTRACT_CONFIG_PERMISSIVE
# define BOOST_CONTRACT_CONFIG_PERMISSIVE 0
#endif
#endif // #include guard

View File

@ -2,14 +2,15 @@
#ifndef BOOST_CONTRACT_EXCEPTION_HPP_
#define BOOST_CONTRACT_EXCEPTION_HPP_
//#include <boost/thread/mutex.hpp>
#include <boost/config.hpp>
#include <exception>
#include <sstream>
#include <string>
#include <iostream>
namespace boost { namespace contract {
// Exceptions.
struct assertion_failure : public std::exception {
explicit assertion_failure(char const* const file = "",
unsigned long const line = 0, char const* const code = "") :
@ -36,7 +37,7 @@ private:
if(std::string(code_) != "") text << " \"" << code_ << "\"";
text << " failed";
if(std::string(file_) != "") {
text << ": \"" << file_ << "\"";
text << ": file \"" << file_ << "\"";
if(line_ != 0) text << ", line " << line_;
}
what_ = text.str();
@ -48,8 +49,6 @@ private:
std::string what_;
};
// Handlers.
enum from {
from_constructor,
from_destructor,
@ -59,104 +58,241 @@ enum from {
from_free_function
};
typedef void (*failure_handler)(from);
typedef void (*assertion_failed_handler)(from);
namespace aux {
extern failure_handler pre_failure_handler;
extern failure_handler post_failure_handler;
namespace exception_ {
enum failed_key {
pre_key,
post_key,
const_entry_inv_key,
const_volatile_entry_inv_key,
static_entry_inv_key,
const_exit_inv_key,
const_volatile_exit_inv_key,
static_exit_inv_key,
};
extern failure_handler const_entry_inv_failure_handler;
extern failure_handler const_volatile_entry_inv_failure_handler;
extern failure_handler static_entry_inv_failure_handler;
template<failed_key Key>
void default_handler(from) {
std::string s = "";
switch(Key) {
case pre_key:
s = "precondition "; break;
case post_key:
s = "postcondition "; break;
case const_entry_inv_key:
s = "const entry invariant "; break;
case const_volatile_entry_inv_key:
s = "const volatile entry invariant "; break;
case static_entry_inv_key:
s = "static entry invariant "; break;
case const_exit_inv_key:
s= "const exit invariant "; break;
case const_volatile_exit_inv_key:
s= "const volatile exit invariant "; break;
case static_exit_inv_key:
s = "static exit invariant "; break;
// No default (so compiler warning/error on missing enum case).
}
try {
throw;
} catch(boost::contract::assertion_failure const& error) {
// what = 'assertion "..." failed: ...'.
std::cerr << s << error.what() << std::endl;
} catch(std::exception const& error) {
std::cerr << s << "checking threw standard exception with " <<
"what(): " << error.what() << std::endl;
} catch(...) {
std::cerr << s << "checking threw unknown exception" <<
std::endl;
}
std::terminate(); // Default handlers log and call terminate.
}
extern failure_handler const_exit_inv_failure_handler;
extern failure_handler const_volatile_exit_inv_failure_handler;
extern failure_handler static_exit_inv_failure_handler;
// TODO: These (and some of the related def code) should be moved in a .cpp.
//boost::mutex pre_failed_mutex;
assertion_failed_handler pre_failed_handler =
&default_handler<pre_key>;
//boost::mutex post_failed_mutex;
assertion_failed_handler post_failed_handler =
&default_handler<post_key>;
//boost::mutex const_entry_inv_failed_mutex;
assertion_failed_handler const_entry_inv_failed_handler =
&default_handler<const_entry_inv_key>;
//boost::mutex const_volatile_entry_inv_failed_mutex;
assertion_failed_handler const_volatile_entry_inv_failed_handler =
&default_handler<const_volatile_entry_inv_key>;
//boost::mutex static_entry_inv_failed_mutex;
assertion_failed_handler static_entry_inv_failed_handler =
&default_handler<static_entry_inv_key>;
//boost::mutex const_exit_inv_failed_mutex;
assertion_failed_handler const_exit_inv_failed_handler =
&default_handler<const_exit_inv_key>;
//boost::mutex const_volatile_exit_inv_failed_mutex;
assertion_failed_handler const_volatile_exit_inv_failed_handler =
&default_handler<const_volatile_exit_inv_key>;
//boost::mutex static_exit_inv_failed_mutex;
assertion_failed_handler static_exit_inv_failed_handler =
&default_handler<static_exit_inv_key>;
}
failure_handler set_precondition_failure(failure_handler f)
BOOST_NOEXCEPT_OR_NOTHROW {
failure_handler result = boost::contract::aux::pre_failure_handler;
boost::contract::aux::pre_failure_handler = f;
#define BOOST_CONTRACT_EXCEPTION_SET_HANDLER_(_mutex, handler, f) \
/*boost::mutex::scoped_lock lock(exception_::_mutex);*/ \
assertion_failed_handler result = exception_::handler; \
exception_::handler = f; \
return result;
}
#define BOOST_CONTRACT_EXCEPTION_GET_HANDLER_(_mutex, handler) \
/*boost::mutex::scoped_lock lock(exception_::_mutex)*/; \
return exception_::handler;
#define BOOST_CONTRACT_EXCEPTION_CALL_HANDLER_(_mutex, handler, where) \
/*boost::mutex::scoped_lock lock(exception_::_mutex);*/ \
exception_::handler(where);
failure_handler set_postcondition_failure(failure_handler f)
assertion_failed_handler set_precondition_failed(assertion_failed_handler f)
BOOST_NOEXCEPT_OR_NOTHROW {
failure_handler result = boost::contract::aux::post_failure_handler;
boost::contract::aux::post_failure_handler = f;
return result;
BOOST_CONTRACT_EXCEPTION_SET_HANDLER_(pre_failed_mutex, pre_failed_handler,
f)
}
assertion_failed_handler get_precondition_failed() BOOST_NOEXCEPT_OR_NOTHROW {
BOOST_CONTRACT_EXCEPTION_GET_HANDLER_(pre_failed_mutex, pre_failed_handler)
}
void precondition_failed(from where) /* can throw */ {
BOOST_CONTRACT_EXCEPTION_CALL_HANDLER_(pre_failed_mutex, pre_failed_handler,
where)
}
// Entry invariants.
failure_handler set_const_entry_invariant_failure(failure_handler f)
assertion_failed_handler set_postcondition_failed(assertion_failed_handler f)
BOOST_NOEXCEPT_OR_NOTHROW {
failure_handler result = boost::contract::aux::
const_entry_inv_failure_handler;
boost::contract::aux::const_entry_inv_failure_handler = f;
return result;
BOOST_CONTRACT_EXCEPTION_SET_HANDLER_(post_failed_mutex,
post_failed_handler, f)
}
assertion_failed_handler get_postcondition_failed() BOOST_NOEXCEPT_OR_NOTHROW {
BOOST_CONTRACT_EXCEPTION_GET_HANDLER_(post_failed_mutex,
post_failed_handler)
}
void postcondition_failed(from where) /* can throw */ {
BOOST_CONTRACT_EXCEPTION_CALL_HANDLER_(post_failed_mutex,
post_failed_handler, where)
}
failure_handler set_const_volatile_entry_invariant_failure(failure_handler f)
assertion_failed_handler set_const_entry_invariant_failed(
assertion_failed_handler f) BOOST_NOEXCEPT_OR_NOTHROW {
BOOST_CONTRACT_EXCEPTION_SET_HANDLER_(const_entry_inv_failed_mutex,
const_entry_inv_failed_handler, f)
}
assertion_failed_handler get_const_entry_invariant_failed()
BOOST_NOEXCEPT_OR_NOTHROW {
failure_handler result = boost::contract::aux::
const_volatile_entry_inv_failure_handler;
boost::contract::aux::const_volatile_entry_inv_failure_handler = f;
return result;
BOOST_CONTRACT_EXCEPTION_GET_HANDLER_(const_entry_inv_failed_mutex,
const_entry_inv_failed_handler)
}
void const_entry_invariant_failed(from where) /* can throw */ {
BOOST_CONTRACT_EXCEPTION_CALL_HANDLER_(const_entry_inv_failed_mutex,
const_entry_inv_failed_handler, where)
}
failure_handler set_static_entry_invariant_failure(failure_handler f)
assertion_failed_handler set_const_volatile_entry_invariant_failed(
assertion_failed_handler f) BOOST_NOEXCEPT_OR_NOTHROW {
BOOST_CONTRACT_EXCEPTION_SET_HANDLER_(const_volatile_entry_inv_failed_mutex,
const_volatile_entry_inv_failed_handler, f)
}
assertion_failed_handler get_const_volatile_entry_invariant_failed()
BOOST_NOEXCEPT_OR_NOTHROW {
failure_handler result = boost::contract::aux::
static_entry_inv_failure_handler;
boost::contract::aux::static_entry_inv_failure_handler = f;
return result;
BOOST_CONTRACT_EXCEPTION_GET_HANDLER_(const_volatile_entry_inv_failed_mutex,
const_volatile_entry_inv_failed_handler)
}
void const_volatile_entry_invariant_failed(from where) /* can throw */ {
BOOST_CONTRACT_EXCEPTION_CALL_HANDLER_(
const_volatile_entry_inv_failed_mutex,
const_volatile_entry_inv_failed_handler,
where
)
}
void set_entry_invariant_failure(failure_handler f) BOOST_NOEXCEPT_OR_NOTHROW {
set_const_entry_invariant_failure(f);
set_const_volatile_entry_invariant_failure(f);
set_static_entry_invariant_failure(f);
assertion_failed_handler set_static_entry_invariant_failed(
assertion_failed_handler f) BOOST_NOEXCEPT_OR_NOTHROW {
BOOST_CONTRACT_EXCEPTION_SET_HANDLER_(static_entry_inv_failed_mutex,
static_entry_inv_failed_handler, f)
}
// Exit invariants.
failure_handler set_const_exit_invariant_failure(failure_handler f)
assertion_failed_handler get_static_entry_invariant_failed()
BOOST_NOEXCEPT_OR_NOTHROW {
failure_handler result = boost::contract::aux::
const_exit_inv_failure_handler;
boost::contract::aux::const_exit_inv_failure_handler = f;
return result;
BOOST_CONTRACT_EXCEPTION_GET_HANDLER_(static_entry_inv_failed_mutex,
static_entry_inv_failed_handler)
}
void static_entry_invariant_failed(from where) /* can throw */ {
BOOST_CONTRACT_EXCEPTION_CALL_HANDLER_(static_entry_inv_failed_mutex,
static_entry_inv_failed_handler, where)
}
failure_handler set_const_volatile_exit_invariant_failure(failure_handler f)
assertion_failed_handler set_const_exit_invariant_failed(
assertion_failed_handler f) BOOST_NOEXCEPT_OR_NOTHROW {
BOOST_CONTRACT_EXCEPTION_SET_HANDLER_(const_exit_inv_failed_mutex,
const_exit_inv_failed_handler, f)
}
assertion_failed_handler get_const_exit_invariant_failed()
BOOST_NOEXCEPT_OR_NOTHROW {
failure_handler result = boost::contract::aux::
const_volatile_exit_inv_failure_handler;
boost::contract::aux::const_volatile_exit_inv_failure_handler = f;
return result;
BOOST_CONTRACT_EXCEPTION_GET_HANDLER_(const_exit_inv_failed_mutex,
const_exit_inv_failed_handler)
}
void const_exit_invariant_failed(from where) /* can throw */ {
BOOST_CONTRACT_EXCEPTION_CALL_HANDLER_(const_exit_inv_failed_mutex,
const_exit_inv_failed_handler, where)
}
failure_handler set_static_exit_invariant_failure(failure_handler f)
assertion_failed_handler set_const_volatile_exit_invariant_failed(
assertion_failed_handler f) BOOST_NOEXCEPT_OR_NOTHROW {
BOOST_CONTRACT_EXCEPTION_SET_HANDLER_(const_volatile_exit_inv_failed_mutex,
const_volatile_exit_inv_failed_handler, f)
}
assertion_failed_handler get_const_volatile_exit_invariant_failed()
BOOST_NOEXCEPT_OR_NOTHROW {
failure_handler result = boost::contract::aux::
static_exit_inv_failure_handler;
boost::contract::aux::static_exit_inv_failure_handler = f;
return result;
BOOST_CONTRACT_EXCEPTION_GET_HANDLER_(const_volatile_exit_inv_failed_mutex,
const_volatile_exit_inv_failed_handler)
}
void const_volatile_exit_invariant_failed(from where) /* can throw */ {
BOOST_CONTRACT_EXCEPTION_CALL_HANDLER_(const_volatile_exit_inv_failed_mutex,
const_volatile_exit_inv_failed_handler, where)
}
void set_exit_invariant_failure(failure_handler f) BOOST_NOEXCEPT_OR_NOTHROW {
set_const_exit_invariant_failure(f);
set_const_volatile_exit_invariant_failure(f);
set_static_exit_invariant_failure(f);
assertion_failed_handler set_static_exit_invariant_failed(
assertion_failed_handler f) BOOST_NOEXCEPT_OR_NOTHROW {
BOOST_CONTRACT_EXCEPTION_SET_HANDLER_(static_exit_inv_failed_mutex,
static_exit_inv_failed_handler, f)
}
assertion_failed_handler get_static_exit_invariant_failed()
BOOST_NOEXCEPT_OR_NOTHROW {
BOOST_CONTRACT_EXCEPTION_GET_HANDLER_(static_exit_inv_failed_mutex,
static_exit_inv_failed_handler)
}
void static_exit_invariant_failed(from where) /* can throw */ {
BOOST_CONTRACT_EXCEPTION_CALL_HANDLER_(static_exit_inv_failed_mutex,
static_exit_inv_failed_handler, where)
}
// All invariants.
void set_invariant_failure(failure_handler f) BOOST_NOEXCEPT_OR_NOTHROW {
set_entry_invariant_failure(f);
set_exit_invariant_failure(f);
void set_entry_invariant_failed(assertion_failed_handler f)
BOOST_NOEXCEPT_OR_NOTHROW {
set_const_entry_invariant_failed(f);
set_const_volatile_entry_invariant_failed(f);
set_static_entry_invariant_failed(f);
}
void set_exit_invariant_failed(assertion_failed_handler f)
BOOST_NOEXCEPT_OR_NOTHROW {
set_const_exit_invariant_failed(f);
set_const_volatile_exit_invariant_failed(f);
set_static_exit_invariant_failed(f);
}
void set_invariant_failed(assertion_failed_handler f)
BOOST_NOEXCEPT_OR_NOTHROW {
set_entry_invariant_failed(f);
set_exit_invariant_failed(f);
}
} } // namespace

View File

@ -2,25 +2,27 @@
#ifndef BOOST_CONTRACT_SET_NOTHING_HPP_
#define BOOST_CONTRACT_SET_NOTHING_HPP_
#include <boost/contract/aux_/check/pre_post.hpp>
#include <boost/contract/aux_/condition/check_nothing.hpp>
#include <boost/shared_ptr.hpp>
namespace boost { namespace contract {
class set_nothing { // Allow (shallow ptr) copy for `auto c = ...`.
class set_nothing {
public:
// Allow to set nothing (neither precondition, nor postcondition).
// No set function member here.
private:
// Use friendship and deleted constructors to limit public API.
friend class var;
friend class set_precondition_only;
friend class set_postcondition_only;
explicit set_nothing(boost::shared_ptr<boost::contract::aux::check::
pre_post> const contract) : contract_(contract) {}
private: // Friendship to minimize lib's public API.
friend class scoped;
friend class set_precondition;
friend class set_postcondition;
boost::shared_ptr<boost::contract::aux::check::pre_post> contract_;
template<typename F, typename A0, typename A1>
friend set_nothing bind(F, A0 const&, A1 const&);
explicit set_nothing(boost::shared_ptr<boost::contract::
aux::check_nothing> check) : check_(check) {}
boost::shared_ptr<boost::contract::aux::check_nothing> check_;
};
} } // namespace

View File

@ -0,0 +1,32 @@
#ifndef BOOST_CONTRACT_SET_POSTCONDITION_HPP_
#define BOOST_CONTRACT_SET_POSTCONDITION_HPP_
#include <boost/contract/core/set_nothing.hpp>
#include <boost/contract/aux_/condition/check_pre_post.hpp>
#include <boost/shared_ptr.hpp>
namespace boost { namespace contract {
class set_postcondition {
public:
template<typename F>
set_nothing postcondition(F f) {
check_->set_post(f);
return set_nothing(check_);
}
private: // Friendship to minimize lib's public API.
friend class scoped;
friend class set_precondition_postcondition;
explicit set_postcondition(boost::shared_ptr<boost::contract::aux::
check_pre_post> check) : check_(check) {}
boost::shared_ptr<boost::contract::aux::check_pre_post> check_;
};
} } // namespace
#endif // #include guard

View File

@ -1,39 +0,0 @@
#ifndef BOOST_CONTRACT_SET_POSTCONDITION_ONLY_HPP_
#define BOOST_CONTRACT_SET_POSTCONDITION_ONLY_HPP_
#include <boost/contract/core/set_nothing.hpp>
#include <boost/contract/aux_/check/pre_post.hpp>
#include <boost/shared_ptr.hpp>
namespace boost { namespace contract {
class set_postcondition_only { // Allow (shallow ptr) copy for `auto c = ...`.
public:
template<typename Postcondition>
boost::contract::set_nothing postcondition(Postcondition const& f) {
contract_->set_post(f);
return boost::contract::set_nothing(contract_);
}
private:
// Use friendship and deleted constructors to limit public API.
friend class var;
friend class set_precondition_postcondition;
template<class Class>
friend set_postcondition_only constructor(Class* const);
template<class Class>
friend set_postcondition_only destructor(Class* const);
explicit set_postcondition_only(boost::shared_ptr<boost::contract::aux::
check::pre_post> const contract) : contract_(contract) {}
boost::shared_ptr<boost::contract::aux::check::pre_post> contract_;
};
} } // namespace
#endif // #include guard

View File

@ -0,0 +1,32 @@
#ifndef BOOST_CONTRACT_SET_PRECONDITION_HPP_
#define BOOST_CONTRACT_SET_PRECONDITION_HPP_
#include <boost/contract/core/set_nothing.hpp>
#include <boost/contract/aux_/condition/check_pre_post.hpp>
#include <boost/shared_ptr.hpp>
namespace boost { namespace contract {
class set_precondition {
public:
template<typename F>
set_nothing precondition(F f) {
check_->set_pre(f);
return set_nothing(check_);
}
private: // Friendship to minimize lib's public API.
friend class scoped;
friend class set_precondition_postcondition;
explicit set_precondition(boost::shared_ptr<boost::contract::
aux::check_pre_post> check) : check_(check) {}
boost::shared_ptr<boost::contract::aux::check_pre_post> check_;
};
} } // namespace
#endif // #include guard

View File

@ -1,33 +0,0 @@
#ifndef BOOST_CONTRACT_SET_PRECONDITION_ONLY_HPP_
#define BOOST_CONTRACT_SET_PRECONDITION_ONLY_HPP_
#include <boost/contract/core/set_nothing.hpp>
#include <boost/contract/aux_/check/pre_post.hpp>
#include <boost/shared_ptr.hpp>
namespace boost { namespace contract {
class set_precondition_only { // Allow (shallow ptr) copy for `auto c = ...`.
public:
template<typename Precondition>
boost::contract::set_nothing precondition(Precondition const& f) {
contract_->set_pre(f);
return boost::contract::set_nothing(contract_);
}
private:
// Use friendship and deleted constructors to limit public API.
friend class var;
friend class set_precondition_postcondition;
explicit set_precondition_only(boost::shared_ptr<boost::contract::aux::
check::pre_post> const contract) : contract_(contract) {}
boost::shared_ptr<boost::contract::aux::check::pre_post> contract_;
};
} } // namespace
#endif // #include guard

View File

@ -2,61 +2,43 @@
#ifndef BOOST_CONTRACT_SET_PRECONDITION_POSTCONDITION_HPP_
#define BOOST_CONTRACT_SET_PRECONDITION_POSTCONDITION_HPP_
#include <boost/contract/core/set_precondition_only.hpp>
#include <boost/contract/core/set_postcondition_only.hpp>
#include <boost/contract/core/virtual.hpp>
#include <boost/contract/aux_/check/pre_post.hpp>
#include <boost/contract/core/set_precondition.hpp>
#include <boost/contract/core/set_postcondition.hpp>
#include <boost/contract/core/call.hpp>
#include <boost/contract/aux_/condition/check_pre_post.hpp>
#include <boost/shared_ptr.hpp>
namespace boost { namespace contract {
// This type allows (shallow ptr) copies for `auto c = ...`.
class set_precondition_postcondition {
public:
template<typename Precondition>
boost::contract::set_postcondition_only precondition(
Precondition const& f) {
contract_->set_pre(f);
return boost::contract::set_postcondition_only(contract_);
}
template<typename Postcondition>
boost::contract::set_precondition_only postcondition(
Postcondition const& f) {
contract_->set_post(f);
return boost::contract::set_precondition_only(contract_);
template<typename F>
boost::contract::set_postcondition precondition(F f) {
check_->set_pre(f);
return boost::contract::set_postcondition(check_);
}
private:
// Use friendship and deleted constructors to limit public API.
friend class var;
template<typename F>
boost::contract::set_precondition postcondition(F f) {
check_->set_post(f);
return boost::contract::set_precondition(check_);
}
template<class Itrospection, class Class, typename Argument0>
friend set_precondition_postcondition public_member(boost::contract::
virtual_* const, Class* const, Argument0&);
template<class Introspection, class Class, typename Argument0>
friend set_precondition_postcondition public_member(Class *const,
Argument0&);
template<class Class>
friend set_precondition_postcondition public_member(boost::contract::
virtual_* const, Class* const);
template<class Class>
friend set_precondition_postcondition public_member(Class* const);
template<class Class>
friend set_precondition_postcondition public_member();
friend set_precondition_postcondition protected_member();
friend set_precondition_postcondition private_member();
private: // Friendship to minimize lib's public API.
friend class scoped;
friend set_precondition_postcondition free_function(
boost::contract::call const&);
friend set_precondition_postcondition free_function();
template<class C>
friend set_precondition_postcondition public_member(
boost::contract::call const&, C const*);
explicit set_precondition_postcondition(boost::shared_ptr<boost::contract::
aux::check::pre_post> const contract) : contract_(contract) {}
aux::check_pre_post> check) : check_(check) {}
boost::shared_ptr<boost::contract::aux::check::pre_post> contract_;
boost::shared_ptr<boost::contract::aux::check_pre_post> check_;
};
} } // namespace

View File

@ -1,35 +0,0 @@
#ifndef BOOST_CONTRACT_VAR_HPP_
#define BOOST_CONTRACT_VAR_HPP_
#include <boost/contract/core/set_precondition_postcondition.hpp>
#include <boost/contract/core/set_precondition_only.hpp>
#include <boost/contract/core/set_precondition_only.hpp>
#include <boost/contract/core/set_nothing.hpp>
#include <boost/contract/aux_/check/pre_post.hpp>
#include <boost/shared_ptr.hpp>
namespace boost { namespace contract {
class var { // Allow (shallow ptr) copy for `var c = ...`.
public:
/* implicit */ var(boost::contract::set_precondition_postcondition
const& c) : contract_(c.contract_) {}
/* implicit */ var(boost::contract::set_precondition_only
const& c) : contract_(c.contract_) {}
/* implicit */ var(boost::contract::set_postcondition_only
const& c) : contract_(c.contract_) {}
/* implicit */ var(boost::contract::set_nothing
const& c) : contract_(c.contract_) {}
private:
boost::shared_ptr<boost::contract::aux::check::pre_post> contract_;
};
} } // namespace
#endif // #include guard

View File

@ -1,68 +0,0 @@
#ifndef BOOST_CONTRACT_VIRTUAL_HPP_
#define BOOST_CONTRACT_VIRTUAL_HPP_
#include <boost/shared_ptr.hpp>
#include <boost/noncopyable.hpp>
#include <queue>
namespace boost {
namespace contract {
namespace aux {
class oldof;
namespace function {
template<class, class, typename>
class public_member;
}
namespace check {
template<class, class, typename>
class subcontracted_pre_post_inv;
}
namespace test {
class friend_;
}
}
}
}
namespace boost { namespace contract {
class virtual_ : private boost::noncopyable { // To avoid copying queue.
// No public operation (users can only assign pointer to this type to null.
//private:
public:
// Friendship used to limit public library API.
template<class, class, typename>
friend class boost::contract::aux::function::public_member;
template<class, class, typename>
friend class boost::contract::aux::check::subcontracted_pre_post_inv;
friend class boost::contract::aux::oldof;
friend bool copy_oldof(virtual_* const);
friend class boost::contract::aux::test::friend_; // For regression tests.
enum action_enum {
user_call, // User call if either v == null || v->action == user_call.
copy_oldof,
check_entry_inv,
check_pre,
check_post,
check_this_post,
check_exit_inv
};
explicit virtual_() : action_(user_call), old_values_() {}
// State passed across virtual calls (friends access data members directly).
action_enum action_;
std::queue<boost::shared_ptr<void> > old_values_;
};
} } // namespace
#endif // #include guard

View File

@ -1,21 +0,0 @@
#ifndef BOOST_CONTRACT_DESTRUCTOR_HPP_
#define BOOST_CONTRACT_DESTRUCTOR_HPP_
#include <boost/contract/core/var.hpp> // Here so user can store contract.
#include <boost/contract/core/set_postcondition_only.hpp>
#include <boost/contract/aux_/function/destructor.hpp>
#include <boost/make_shared.hpp>
namespace boost { namespace contract {
template<class Class>
boost::contract::set_postcondition_only destructor(Class* const object) {
return boost::contract::set_postcondition_only(boost::make_shared<
boost::contract::aux::function::destructor<Class> >(object));
}
} } // namespace
#endif // #include guard

View File

@ -2,16 +2,22 @@
#ifndef BOOST_CONTRACT_FREE_FUNCTION_HPP_
#define BOOST_CONTRACT_FREE_FUNCTION_HPP_
#include <boost/contract/core/var.hpp> // Here so user can store contract.
#include <boost/contract/core/call.hpp>
#include <boost/contract/core/set_precondition_postcondition.hpp>
#include <boost/contract/aux_/function/free_function.hpp>
#include <boost/make_shared.hpp>
namespace boost { namespace contract {
boost::contract::set_precondition_postcondition free_function(
boost::contract::call const& c) {
return boost::contract::set_precondition_postcondition(boost::make_shared<
boost::contract::aux::free_function>(c.call_));
}
boost::contract::set_precondition_postcondition free_function() {
return boost::contract::set_precondition_postcondition(boost::make_shared<
boost::contract::aux::function::free_function>());
boost::contract::aux::free_function>());
}
} } // namespace

View File

@ -1,28 +0,0 @@
#ifndef BOOST_CONTRACT_INTROSPECT_HPP_
#define BOOST_CONTRACT_INTROSPECT_HPP_
#include <boost/contract/aux_/type_traits/introspection.hpp>
#include <boost/contract/aux_/name.hpp>
#include <boost/preprocessor/cat.hpp>
/* PUBLIC */
#define BOOST_CONTRACT_INTROSPECT(function_name) \
BOOST_CONTRACT_INTROSPECT_TRAIT(BOOST_PP_CAT(introspect_, function_name), \
function_name)
#define BOOST_CONTRACT_INTROSPECT_TRAIT(trait_name, function_name) \
struct trait_name { \
BOOST_CONTRACT_AUX_TYPE_TRAITS_INTROSPECTION_HAS_MEMBER_FUNCTION( \
has_member_function, function_name) \
\
template<class BOOST_CONTRACT_AUX_NAME1(C), \
typename BOOST_CONTRACT_AUX_NAME1(F)> \
static BOOST_CONTRACT_AUX_NAME1(F) member_function_address() { \
return &BOOST_CONTRACT_AUX_NAME1(C)::function_name; \
} \
};
#endif // #include guard

View File

@ -2,59 +2,55 @@
#ifndef BOOST_CONTRACT_OLDOF_HPP_
#define BOOST_CONTRACT_OLDOF_HPP_
#include <boost/contract/core/virtual.hpp>
#include <boost/contract/core/config.hpp>
#include <boost/contract/aux_/oldof.hpp>
#include <boost/type_traits/remove_reference.hpp>
#include <boost/optional.hpp>
#include <boost/contract/core/call.hpp>
#include <boost/contract/aux_/call.hpp>
#include <boost/contract/aux_/debug.hpp>
#include <boost/make_shared.hpp>
#include <boost/shared_ptr.hpp>
#include <boost/preprocessor/config/config.hpp>
#include <boost/config.hpp>
// IMPORTANT: The following makes sure old values expressions are evaluated
// only once, they are copied only once, and only when preconditions are
// enabled.
#if !BOOST_PP_VARIADICS
#define BOOST_CONTRACT_OLDOF BOOST_CONTRACT_ERROR_variadic_macros_required_by_BOOST_CONTRACT_OLDOF_otherwise_manually_program_old_values
# define BOOST_CONTRACT_OLDOF \
BOOST_CONTRACT_ERROR_variadic_macros_required_by_BOOST_CONTRACT_OLDOF_otherwise_manually_program_old_values
#else
#include <boost/config.hpp>
#include <boost/preprocessor/facilities/overload.hpp>
#include <boost/preprocessor/facilities/empty.hpp>
#include <boost/preprocessor/cat.hpp>
/* PUBLIC */
#define BOOST_CONTRACT_OLDOF(...) \
BOOST_PP_CAT(BOOST_PP_OVERLOAD( /* CAT(... EMPTY()) workaround for MSVC */ \
BOOST_CONTRACT_ERROR_macro_BOOST_CONTRACT_OLDOF_invalid_number_of_arguments_, \
__VA_ARGS__)(__VA_ARGS__), BOOST_PP_EMPTY())
BOOST_PP_CAT( /* CAT(..., EMTPY()) required on MSVC */ \
BOOST_PP_OVERLOAD( \
BOOST_CONTRACT_ERROR_invoked_macro_BOOST_CONTRACT_OLDOF_with_invalid_number_of_arguments_, \
__VA_ARGS__ \
)(__VA_ARGS__), \
BOOST_PP_EMPTY() \
)
/* PRIVATE */
// NOTE: Following are not local macros, do NOT #undef them. These macro names
// are use so all variadic macro argument numbers > 2 will generate appropriate
// ERROR symbols.
#define \
BOOST_CONTRACT_ERROR_macro_BOOST_CONTRACT_OLDOF_invalid_number_of_arguments_1( \
value) \
(boost::contract::oldof BOOST_CONTRACT_OLDOF_TYPEOF_(value) ( \
boost::contract::copy_oldof() ? value : boost::contract::oldof() \
BOOST_CONTRACT_ERROR_invoked_macro_BOOST_CONTRACT_OLDOF_with_invalid_number_of_arguments_1(value) \
BOOST_CONTRACT_OLDOF_AUTO_(value)(boost::contract::oldof( \
boost::contract::copy_oldof() ? (value) : boost::contract::oldof() \
))
#define \
BOOST_CONTRACT_ERROR_macro_BOOST_CONTRACT_OLDOF_invalid_number_of_arguments_2( \
v, value) \
(boost::contract::oldof BOOST_CONTRACT_OLDOF_TYPEOF_(value) ( \
v, boost::contract::copy_oldof(v) ? value : boost::contract::oldof() \
BOOST_CONTRACT_ERROR_invoked_macro_BOOST_CONTRACT_OLDOF_with_invalid_number_of_arguments_2(c, value) \
BOOST_CONTRACT_OLDOF_AUTO_(value)(boost::contract::oldof(c, \
boost::contract::copy_oldof(c) ? (value) : boost::contract::oldof() \
))
#ifdef BOOST_NO_CXX11_AUTO_DECLARATIONS
# define BOOST_CONTRACT_OLDOF_TYPEOF_(value) /* nothing */
# define BOOST_CONTRACT_OLDOF_AUTO_(value) /* nothing */
#else
# include <boost/typeof/typeof.hpp>
# define BOOST_CONTRACT_OLDOF_TYPEOF_(value) <BOOST_TYPEOF(value)>
// Explicitly force shared_ptr<T const> conversion to allow for C++11 auto decl.
# define BOOST_CONTRACT_OLDOF_AUTO_(value) \
boost::shared_ptr<BOOST_TYPEOF(value) const>
#endif
#endif // VARIADICS
@ -63,43 +59,72 @@ BOOST_CONTRACT_ERROR_macro_BOOST_CONTRACT_OLDOF_invalid_number_of_arguments_2( \
namespace boost { namespace contract {
bool copy_oldof(boost::contract::virtual_* const v = 0) {
bool copy_oldof() {
#ifdef BOOST_CONTRACT_CONFIG_NO_POSTCONDITIONS
return false; // Never check post, so old-of never copied.
return false; // Post checking disabled, so never copy old values.
#else
// Copy if user call (so also if !v) or virtual contract call for copy.
return !v || v->action_ == boost::contract::virtual_::user_call ||
v->action_ == boost::contract::virtual_::copy_oldof;
return true;
#endif
}
boost::contract::aux::oldof oldof() { return boost::contract::aux::oldof(); }
// Un-erasure via explicit type T so to allow to use C++11 auto decl.
template<typename T>
boost::shared_ptr<typename boost::remove_reference<T>::type const>
oldof(boost::contract::virtual_* const v,
boost::contract::aux::oldof const& old) {
return boost::contract::aux::oldof(v, old);
bool copy_oldof(boost::contract::call const& c) {
#ifdef BOOST_CONTRACT_CONFIG_NO_POSTCONDITIONS
return false; // Post checking disabled, so never copy old values.
#else
return c.call_ && c.call_->action == boost::contract::aux::call::copy_oldof;
#endif
}
// Un-erasure will be done based on explicit decl type (no auto allowed).
boost::contract::aux::oldof oldof(boost::contract::virtual_* const v,
boost::contract::aux::oldof const& old) {
return boost::contract::aux::oldof(v, old);
}
class oldof {
public:
explicit oldof() : call_(), value_() {}
// Un-erasure via explicit type T so to allow to use C++11 auto decl.
template<typename T>
boost::shared_ptr<typename boost::remove_reference<T>::type const> oldof(
boost::contract::aux::oldof const& old) {
return boost::contract::aux::oldof(0, old);
}
explicit oldof(boost::contract::call const& c, oldof const& old) :
call_(c.call_), value_(old.value_) {}
// Un-erasure will be done based on explicit decl type (no auto allowed).
boost::contract::aux::oldof oldof(boost::contract::aux::oldof const& old) {
return boost::contract::aux::oldof(0, old);
}
template<typename T>
/* implicit */ oldof(T const& old_value) :
value_(boost::make_shared<T>(old_value)) { // T's one single copy.
std::clog << "oldof copy" << std::endl;
}
// TODO: I might be able to use unique_ptr here instead of shared_ptr. That
// might be the true for the pointer that holds contract and call as well...
// do some testing to figure that out (unique_ptr adds less overhead).
template<typename T>
operator boost::shared_ptr<T const>() {
if(!call_) {
BOOST_CONTRACT_AUX_DEBUG(value_);
boost::shared_ptr<T const> old_value =
boost::static_pointer_cast<T const>(value_);
BOOST_CONTRACT_AUX_DEBUG(old_value);
std::clog << "oldof un-erased: " << *old_value << std::endl;
return old_value;
} else if(call_->action == boost::contract::aux::call::copy_oldof) {
BOOST_CONTRACT_AUX_DEBUG(value_);
call_->old_values.push(value_);
std::clog << "oldof push" << std::endl;
return boost::shared_ptr<T const>();
} else if(call_->action == boost::contract::aux::call::check_post) {
BOOST_CONTRACT_AUX_DEBUG(!value_);
boost::shared_ptr<void> value = call_->old_values.front();
BOOST_CONTRACT_AUX_DEBUG(value);
call_->old_values.pop();
boost::shared_ptr<T const> old_value =
boost::static_pointer_cast<T const>(value);
BOOST_CONTRACT_AUX_DEBUG(old_value);
std::clog << "oldof pop: " << *old_value << std::endl;
return old_value;
}
BOOST_CONTRACT_AUX_DEBUG(!value_);
return boost::shared_ptr<T const>();
}
private:
boost::shared_ptr<boost::contract::aux::call> call_;
boost::shared_ptr<void> value_;
};
} } // namespace

View File

@ -1,22 +0,0 @@
#ifndef BOOST_CONTRACT_PRIVATE_MEMBER_HPP_
#define BOOST_CONTRACT_PRIVATE_MEMBER_HPP_
#include <boost/contract/core/var.hpp> // Here so user can store contract.
#include <boost/contract/core/set_precondition_postcondition.hpp>
#include <boost/contract/aux_/function/private_member.hpp>
#include <boost/make_shared.hpp>
// TOOD: On C++11 Clang... these could static_assert enclosing func is not pub?
namespace boost { namespace contract {
boost::contract::set_precondition_postcondition private_member() {
return boost::contract::set_precondition_postcondition(boost::make_shared<
boost::contract::aux::function::private_member>());
}
} } // namespace
#endif //

View File

@ -1,22 +0,0 @@
#ifndef BOOST_CONTRACT_PROTECTED_MEMBER_HPP_
#define BOOST_CONTRACT_PROTECTED_MEMBER_HPP_
#include <boost/contract/core/var.hpp> // Here so user can store contract.
#include <boost/contract/core/set_precondition_postcondition.hpp>
#include <boost/contract/aux_/function/protected_member.hpp>
#include <boost/make_shared.hpp>
// TOOD: On C++11 Clang... these could static_assert enclosing func is not pub?
namespace boost { namespace contract {
boost::contract::set_precondition_postcondition protected_member() {
return boost::contract::set_precondition_postcondition(boost::make_shared<
boost::contract::aux::function::protected_member>());
}
} } // namespace
#endif //

View File

@ -2,62 +2,24 @@
#ifndef BOOST_CONTRACT_PUBLIC_MEMBER_HPP_
#define BOOST_CONTRACT_PUBLIC_MEMBER_HPP_
#include <boost/contract/core/var.hpp> // Here so user can store contract.
#include <boost/contract/core/call.hpp>
#include <boost/contract/core/set_precondition_postcondition.hpp>
#include <boost/contract/core/virtual.hpp>
#include <boost/contract/aux_/function/public_static_member.hpp>
#include <boost/contract/aux_/function/public_member.hpp>
#include <boost/make_shared.hpp>
namespace boost { namespace contract {
// Contract for public member functions with a virtual body and members of a
// class that inherits from one or more base classes (optionally, it can be used
// also if there are no base classes).
template<class Introspection, class Class, typename Argument0>
boost::contract::set_precondition_postcondition public_member(boost::contract::
virtual_* const v, Class* const object, Argument0& argument0) {
return boost::contract::set_precondition_postcondition(boost::make_shared<
boost::contract::aux::function::public_member<Class, Introspection,
Argument0> >(v, object, argument0)
);
}
// Contract for public member functions with a non-virtual body and members of a
// class that inherits from one or more base classes (optionally, it can be used
// also if there are no base classes).
template<class Introspection, class Class, typename Argument0>
template<class C>
boost::contract::set_precondition_postcondition public_member(
Class* const object, Argument0& argument0) {
boost::contract::call const& c, C const* obj) {
return boost::contract::set_precondition_postcondition(boost::make_shared<
boost::contract::aux::function::public_member<Class, Introspection,
Argument0> >(object, argument0)
);
boost::contract::aux::public_member<C> >(c.call_, obj));
}
// Contract for public member functions with a virtual body and members of a
// class that does not inherits from any base class.
template<class Class>
boost::contract::set_precondition_postcondition public_member(
boost::contract::virtual_* const v, Class* const object) {
template<class C>
boost::contract::set_precondition_postcondition public_member(C const* obj) {
return boost::contract::set_precondition_postcondition(boost::make_shared<
boost::contract::aux::function::public_member<Class> >(v, object));
}
// Contract for public member functions with a non-virtual body and members of a
// class that does not inherits from any base class.
template<class Class>
boost::contract::set_precondition_postcondition public_member(
Class* const object) {
return boost::contract::set_precondition_postcondition(boost::make_shared<
boost::contract::aux::function::public_member<Class> >(object));
}
// Contract for public static member functions.
template<class Class>
boost::contract::set_precondition_postcondition public_member() {
return boost::contract::set_precondition_postcondition(boost::make_shared<
boost::contract::aux::function::public_static_member<Class> >());
boost::contract::aux::public_member<C> >(obj));
}
} } // namespace

View File

@ -0,0 +1,37 @@
#ifndef BOOST_CONTRACT_SCOPED_HPP_
#define BOOST_CONTRACT_SCOPED_HPP_
#include <boost/contract/core/set_precondition_postcondition.hpp>
#include <boost/contract/core/set_precondition.hpp>
#include <boost/contract/core/set_postcondition.hpp>
#include <boost/contract/core/set_nothing.hpp>
#include <boost/contract/aux_/condition/check_nothing.hpp>
#include <boost/shared_ptr.hpp>
namespace boost { namespace contract {
class scoped { // Copyable (shallow shared ptr copies).
public:
// All implicit to allow `scoped c = ...`.
/* implicit */ scoped(boost::contract::set_precondition_postcondition
const& contract) : check_(contract.check_) {}
/* implicit */ scoped(boost::contract::set_precondition
const& contract) : check_(contract.check_) {}
/* implicit */ scoped(boost::contract::set_postcondition
const& contract) : check_(contract.check_) {}
/* implicit */ scoped(boost::contract::set_nothing
const& contract) : check_(contract.check_) {}
private:
boost::shared_ptr<boost::contract::aux::check_nothing> check_;
};
} } // namespace
#endif // #include guard

View File

@ -1,30 +0,0 @@
#ifndef BOOST_CONTRACT_ALWAYS_HPP_
#define BOOST_CONTRACT_ALWAYS_HPP_
#include <boost/shared_ptr.hpp>
#include <boost/make_shared.hpp>
namespace boost { namespace contract {
// Result must be copy constructible (always true for func. result types).
template< typename Result >
struct always {
typedef Result result_type;
explicit always ( Result const& result ) :
result_(boost::make_shared<Result>(result)) {}
// No matter if functor called with 0 to n args, it always returns result.
Result operator() ( ... ) const { return *result_; }
private:
// Pointer so copies of this object do not keep copying result (above
// operator() already returns new copies for result to users).
boost::shared_ptr<Result> const result_;
};
} } // namespace
#endif // #include guard

View File

@ -1,94 +0,0 @@
#ifndef BOOST_CONTRACT_CALL_IF_HPP_
#define BOOST_CONTRACT_CALL_IF_HPP_
#include <boost/utility/result_of.hpp>
#include <boost/shared_ptr.hpp>
#include <boost/make_shared.hpp>
// TODO: Once I get this to work 100% right... then consider using it instead
// of the extra mpl::bool_ param inside the lib impl too.
namespace boost { namespace contract {
namespace call_if_ {
// Result must be copy constructible (always true for func. result types).
template< bool Cond, typename Result >
struct call_if_c {
template< typename Then >
explicit call_if_c ( Then const& t ) :
result_(boost::make_shared<Result>(t())) {}
operator Result ( ) const { return *result_; }
template< typename Else >
Result else_ ( Else const& ) const { return *result_; }
private:
// Pointer so copies of this object do not keep copying result (above
// operator() already returns new copies of result to users).
boost::shared_ptr<Result> const result_;
};
template< typename Result >
struct call_if_c<false, Result> {
template< typename Then >
explicit call_if_c ( Then const& ) {}
// Result must be default constructible when else_ is not used.
operator Result ( ) { return Result(); }
template< typename Else >
Result else_ ( Else const& e ) const { return e(); }
};
// Specializations (optimized) for void result type.
template< >
struct call_if_c<true, void> {
template< typename Then >
explicit call_if_c ( Then const& t ) { t(); }
template< typename Else >
void else_ ( Else const& ) const {}
};
template< >
struct call_if_c<false, void> {
template< typename Then >
explicit call_if_c ( Then const& ) {}
template< typename Else >
void else_ ( Else const& e ) const { e(); }
};
}
template< bool Cond, typename Result, typename Then >
call_if_::call_if_c<Cond, Result> call_if_c ( Then const& t ) {
return call_if_::call_if_c<Cond, Result>(t);
}
template< bool Cond, typename Then >
call_if_::call_if_c<Cond, typename boost::result_of<Then>::type> call_if_c (
Then const& t ) {
return call_if_::call_if_c<Cond, typename
boost::result_of<Then>::type>(t);
}
template< class Cond, typename Result, typename Then >
call_if_::call_if_c<Cond::value, Result> call_if ( Then const& t ) {
return call_if_::call_if_c<Cond::value, Result>(t);
}
template< class Cond, typename Then >
call_if_::call_if_c<Cond::value, typename boost::result_of<Then>::type> call_if(
Then const& t ) {
return call_if_::call_if_c<Cond::value,
typename boost::result_of<Then>::type>(t);
}
} } // namespace
#endif // #include guard

View File

@ -1,4 +1,8 @@
subdir-run const : free_function ;
subdir-run const : public_member ;
subdir-run const : public_virtual_member ;
subdir-run function : constructor ;
subdir-run function : destructor ;
subdir-run function : public_member ;

View File

@ -1,27 +1,33 @@
#include "../aux_/oteststream.hpp"
#include <iostream>
#include <boost/contract/oldof.hpp>
#include <boost/contract/free_function.hpp>
#include <boost/detail/lightweight_test.hpp>
#include <sstream>
#include <boost/contract/assert.hpp>
#include <boost/contract/scoped.hpp>
#include <boost/contract/bind.hpp>
#include <boost/shared_ptr.hpp>
// Test free-function.
boost::contract::aux::test::oteststream out;
void inv1_contract(int const& x, int const& result,
boost::contract::decl c = 0) {
void inv1_contract(int const& x, int const& result, boost::contract::call c) {
boost::shared_ptr<int const> old_x = BOOST_CONTRACT_OLDOF(c, x);
boost::contract::free_function(c)
.precondition([&] { BOOST_CONTRACT_ASSERT(x >= 0); })
.precondition([&] {
std::clog << "inv1::pre" << std::endl;
BOOST_CONTRACT_ASSERT(x >= 0);
})
.postcondition([&] {
std::clog << "inv1::post" << std::endl;
BOOST_CONTRACT_ASSERT(-x == *old_x);
BOOST_CONTRACT_ASSERT(result == *old_x);
})
;
}
int inv1(int& x) {
std::clog << "inv1" << std::endl;
int result;
boost::contract::var contract(inv1_contract, x, result);
boost::contract::scoped contract = boost::contract::bind(
&inv1_contract, x, result);
std::clog << "inv1::body" << std::endl;
result = x;
x = -x;
return result;
@ -30,32 +36,31 @@ int inv1(int& x) {
int inv2(int& x) {
int result;
boost::shared_ptr<int const> old_x = BOOST_CONTRACT_OLDOF(x);
boost::contract::var contract = boost::contract::free_function()
.precondition([&] { BOOST_CONTRACT_ASSERT(x >= 0); })
boost::contract::scoped contract = boost::contract::free_function()
.precondition([&] {
std::clog << "inv2::pre" << std::endl;
BOOST_CONTRACT_ASSERT(x >= 0);
})
.postcondition([&] {
std::clog << "inv2::post" << std::endl;
BOOST_CONTRACT_ASSERT(-x == *old_x);
BOOST_CONTRACT_ASSERT(result == *old_x);
})
;
std::clog << "inv2::body" << std::endl;
result = x;
x = -x;
return result;
}
int main() {
std::ostringstream ok;
int x = 123;
inv1(x);
std::clog << std::endl;
out.str("");
f();
ok.str("");
ok
<< "f::pre" << std::endl
<< "f::body" << std::endl
<< "f::post" << std::endl
;
BOOST_TEST(out.check(ok.str()));
return boost::report_errors();
x = 456;
inv2(x);
return 0;
}

View File

@ -0,0 +1,53 @@
#include <iostream>
#include <boost/contract/public_member.hpp>
#include <boost/contract/oldof.hpp>
#include <boost/contract/assert.hpp>
#include <boost/contract/scoped.hpp>
#include <boost/contract/bind.hpp>
struct a {
int x;
void invariant() const {
std::clog << "a::inv" << std::endl;
BOOST_CONTRACT_ASSERT(x > 0);
}
static void static_invariant() {
std::clog << "a::static_inv" << std::endl;
}
void swap_contract(int const& y, boost::contract::call c) const {
boost::shared_ptr<int const> old_x = BOOST_CONTRACT_OLDOF(c, x);
boost::shared_ptr<int const> old_y = BOOST_CONTRACT_OLDOF(c, y);
boost::contract::public_member(c, this)
.precondition([&] {
std::clog << "a::swap::pre" << std::endl;
BOOST_CONTRACT_ASSERT(x != y);
})
.postcondition([&] {
std::clog << "a::swap::post" << std::endl;
BOOST_CONTRACT_ASSERT(x == *old_y);
BOOST_CONTRACT_ASSERT(y == *old_x);
})
;
}
void swap(int& y) {
boost::contract::scoped contract = boost::contract::bind(
&a::swap_contract, this, y);
std::clog << "a::swap::body" << std::endl;
int save_x = x;
x = y;
y = save_x;
}
};
int main() {
a aa;
aa.x = 10;
int y = 20;
aa.swap(y);
return 0;
}

View File

@ -0,0 +1,39 @@
#include <iostream>
#include <boost/contract/public_member.hpp>
#include <boost/contract/oldof.hpp>
#include <boost/contract/assert.hpp>
#include <boost/contract/scoped.hpp>
#include <boost/contract/bind.hpp>
struct a {
std::string x;
a() : x("a") {}
void invariant() const { std::clog << "a::inv" << std::endl; }
static void static_invariant() { std::clog << "a::static_inv" << std::endl;}
void f_contract(int const& s, boost::contract::call c) const {
boost::shared_ptr<int const> old_x = BOOST_CONTRACT_OLDOF(c, x);
boost::contract::public_member(c, this)
.precondition([&] { std::clog << "a::f::pre" << std::endd; })
.postcondition([&] {
std::clog << "a::f::post" << std::endl;
BOOST_CONTRACT_ASSERT(x == old_x + s);
})
;
}
virtual void f(int s) {
boost::contract::scoped contract = boost::contract::bind(
&a::f, this, s);
std::clog << "a::f::body" << std::endl;
x = x + s;
}
};
int main() {
a aa;
aa.f("s");
return 0;
}