1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/ossrv/ossrv_pub/boost_apis/boost/statechart/state_machine.hpp Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,1054 @@
1.4 +#ifndef BOOST_STATECHART_STATE_MACHINE_HPP_INCLUDED
1.5 +#define BOOST_STATECHART_STATE_MACHINE_HPP_INCLUDED
1.6 +//////////////////////////////////////////////////////////////////////////////
1.7 +// Copyright 2002-2006 Andreas Huber Doenni
1.8 +// Distributed under the Boost Software License, Version 1.0. (See accompany-
1.9 +// ing file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
1.10 +//////////////////////////////////////////////////////////////////////////////
1.11 +
1.12 +
1.13 +
1.14 +#include <boost/statechart/event.hpp>
1.15 +#include <boost/statechart/null_exception_translator.hpp>
1.16 +#include <boost/statechart/result.hpp>
1.17 +
1.18 +#include <boost/statechart/detail/rtti_policy.hpp>
1.19 +#include <boost/statechart/detail/state_base.hpp>
1.20 +#include <boost/statechart/detail/leaf_state.hpp>
1.21 +#include <boost/statechart/detail/node_state.hpp>
1.22 +#include <boost/statechart/detail/constructor.hpp>
1.23 +#include <boost/statechart/detail/avoid_unused_warning.hpp>
1.24 +
1.25 +#include <boost/mpl/list.hpp>
1.26 +#include <boost/mpl/clear.hpp>
1.27 +#include <boost/mpl/if.hpp>
1.28 +#include <boost/mpl/at.hpp>
1.29 +#include <boost/mpl/integral_c.hpp>
1.30 +#include <boost/mpl/minus.hpp>
1.31 +#include <boost/mpl/equal_to.hpp>
1.32 +
1.33 +#include <boost/intrusive_ptr.hpp>
1.34 +#include <boost/type_traits/is_pointer.hpp>
1.35 +#include <boost/type_traits/remove_reference.hpp>
1.36 +#include <boost/noncopyable.hpp>
1.37 +#include <boost/assert.hpp>
1.38 +#include <boost/static_assert.hpp>
1.39 +#include <boost/cast.hpp> // boost::polymorphic_downcast
1.40 +// BOOST_NO_EXCEPTIONS, BOOST_MSVC, BOOST_MSVC_STD_ITERATOR
1.41 +#include <boost/config.hpp>
1.42 +
1.43 +#include <boost/detail/allocator_utilities.hpp>
1.44 +
1.45 +#ifdef BOOST_MSVC
1.46 +# pragma warning( push )
1.47 +# pragma warning( disable: 4702 ) // unreachable code (in release mode only)
1.48 +#endif
1.49 +
1.50 +#include <map>
1.51 +
1.52 +#ifdef BOOST_MSVC
1.53 +# pragma warning( pop )
1.54 +#endif
1.55 +
1.56 +#include <memory> // std::allocator
1.57 +#include <typeinfo> // std::bad_cast
1.58 +#include <functional> // std::less
1.59 +#include <iterator>
1.60 +
1.61 +
1.62 +
1.63 +#ifdef BOOST_MSVC
1.64 +// We permanently turn off the following level 4 warnings because users will
1.65 +// have to do so themselves anyway if we turn them back on
1.66 +# pragma warning( disable: 4511 ) // copy constructor could not be generated
1.67 +# pragma warning( disable: 4512 ) // assignment op could not be generated
1.68 +#endif
1.69 +
1.70 +
1.71 +
1.72 +namespace boost
1.73 +{
1.74 +namespace statechart
1.75 +{
1.76 +namespace detail
1.77 +{
1.78 +
1.79 +
1.80 +
1.81 +//////////////////////////////////////////////////////////////////////////////
1.82 +template< class StateBaseType, class EventBaseType, class IdType >
1.83 +class send_function
1.84 +{
1.85 + public:
1.86 + //////////////////////////////////////////////////////////////////////////
1.87 + send_function(
1.88 + StateBaseType & toState,
1.89 + const EventBaseType & evt,
1.90 + IdType eventType
1.91 + ) :
1.92 + toState_( toState ), evt_( evt ), eventType_( eventType )
1.93 + {
1.94 + }
1.95 +
1.96 + result operator()()
1.97 + {
1.98 + return detail::result_utility::make_result(
1.99 + toState_.react_impl( evt_, eventType_ ) );
1.100 + }
1.101 +
1.102 + private:
1.103 + //////////////////////////////////////////////////////////////////////////
1.104 + StateBaseType & toState_;
1.105 + const EventBaseType & evt_;
1.106 + IdType eventType_;
1.107 +};
1.108 +
1.109 +
1.110 +//////////////////////////////////////////////////////////////////////////////
1.111 +struct state_cast_impl_pointer_target
1.112 +{
1.113 + public:
1.114 + //////////////////////////////////////////////////////////////////////////
1.115 + template< class StateBaseType >
1.116 + static const StateBaseType * deref_if_necessary(
1.117 + const StateBaseType * pState )
1.118 + {
1.119 + return pState;
1.120 + }
1.121 +
1.122 + template< class Target, class IdType >
1.123 + static IdType type_id()
1.124 + {
1.125 + Target p = 0;
1.126 + return type_id_impl< IdType >( p );
1.127 + }
1.128 +
1.129 + static bool found( const void * pFound )
1.130 + {
1.131 + return pFound != 0;
1.132 + }
1.133 +
1.134 + template< class Target >
1.135 + static Target not_found()
1.136 + {
1.137 + return 0;
1.138 + }
1.139 +
1.140 + private:
1.141 + //////////////////////////////////////////////////////////////////////////
1.142 + template< class IdType, class Type >
1.143 + static IdType type_id_impl( const Type * )
1.144 + {
1.145 + return Type::static_type();
1.146 + }
1.147 +};
1.148 +
1.149 +struct state_cast_impl_reference_target
1.150 +{
1.151 + template< class StateBaseType >
1.152 + static const StateBaseType & deref_if_necessary(
1.153 + const StateBaseType * pState )
1.154 + {
1.155 + return *pState;
1.156 + }
1.157 +
1.158 + template< class Target, class IdType >
1.159 + static IdType type_id()
1.160 + {
1.161 + return remove_reference< Target >::type::static_type();
1.162 + }
1.163 +
1.164 + template< class Dummy >
1.165 + static bool found( const Dummy & )
1.166 + {
1.167 + return true;
1.168 + }
1.169 +
1.170 + template< class Target >
1.171 + static Target not_found()
1.172 + {
1.173 + throw std::bad_cast();
1.174 + }
1.175 +};
1.176 +
1.177 +template< class Target >
1.178 +struct state_cast_impl : public mpl::if_<
1.179 + is_pointer< Target >,
1.180 + state_cast_impl_pointer_target,
1.181 + state_cast_impl_reference_target
1.182 +>::type {};
1.183 +
1.184 +
1.185 +//////////////////////////////////////////////////////////////////////////////
1.186 +template< class RttiPolicy >
1.187 +class history_key
1.188 +{
1.189 + public:
1.190 + //////////////////////////////////////////////////////////////////////////
1.191 + template< class HistorizedState >
1.192 + static history_key make_history_key()
1.193 + {
1.194 + return history_key(
1.195 + HistorizedState::context_type::static_type(),
1.196 + HistorizedState::orthogonal_position::value );
1.197 + }
1.198 +
1.199 + typename RttiPolicy::id_type history_context_type() const
1.200 + {
1.201 + return historyContextType_;
1.202 + }
1.203 +
1.204 + friend bool operator<(
1.205 + const history_key & left, const history_key & right )
1.206 + {
1.207 + return
1.208 + std::less< typename RttiPolicy::id_type >()(
1.209 + left.historyContextType_, right.historyContextType_ ) ||
1.210 + ( ( left.historyContextType_ == right.historyContextType_ ) &&
1.211 + ( left.historizedOrthogonalRegion_ <
1.212 + right.historizedOrthogonalRegion_ ) );
1.213 + }
1.214 +
1.215 + private:
1.216 + //////////////////////////////////////////////////////////////////////////
1.217 + history_key(
1.218 + typename RttiPolicy::id_type historyContextType,
1.219 + orthogonal_position_type historizedOrthogonalRegion
1.220 + ) :
1.221 + historyContextType_( historyContextType ),
1.222 + historizedOrthogonalRegion_( historizedOrthogonalRegion )
1.223 + {
1.224 + }
1.225 +
1.226 + const typename RttiPolicy::id_type historyContextType_;
1.227 + const orthogonal_position_type historizedOrthogonalRegion_;
1.228 +};
1.229 +
1.230 +
1.231 +
1.232 +} // namespace detail
1.233 +
1.234 +
1.235 +
1.236 +//////////////////////////////////////////////////////////////////////////////
1.237 +template< class MostDerived,
1.238 + class InitialState,
1.239 + class Allocator = std::allocator< void >,
1.240 + class ExceptionTranslator = null_exception_translator >
1.241 +class state_machine : noncopyable
1.242 +{
1.243 + public:
1.244 + //////////////////////////////////////////////////////////////////////////
1.245 + typedef Allocator allocator_type;
1.246 + typedef detail::rtti_policy rtti_policy_type;
1.247 + typedef event_base event_base_type;
1.248 + typedef intrusive_ptr< const event_base_type > event_base_ptr_type;
1.249 +
1.250 + void initiate()
1.251 + {
1.252 + terminate();
1.253 +
1.254 + {
1.255 + terminator guard( *this );
1.256 + detail::result_utility::get_result( translator_(
1.257 + initial_construct_function( *this ),
1.258 + exception_event_handler( *this ) ) );
1.259 + guard.dismiss();
1.260 + }
1.261 +
1.262 + process_queued_events();
1.263 + }
1.264 +
1.265 + void terminate()
1.266 + {
1.267 + terminator guard( *this );
1.268 + detail::result_utility::get_result( translator_(
1.269 + terminate_function( *this ),
1.270 + exception_event_handler( *this ) ) );
1.271 + guard.dismiss();
1.272 + }
1.273 +
1.274 + bool terminated() const
1.275 + {
1.276 + return pOutermostState_ == 0;
1.277 + }
1.278 +
1.279 + void process_event( const event_base_type & evt )
1.280 + {
1.281 + send_event( evt );
1.282 + process_queued_events();
1.283 + }
1.284 +
1.285 + template< class Target >
1.286 + Target state_cast() const
1.287 + {
1.288 + typedef detail::state_cast_impl< Target > impl;
1.289 +
1.290 + for ( typename state_list_type::const_iterator pCurrentLeafState =
1.291 + currentStates_.begin();
1.292 + pCurrentLeafState != currentStatesEnd_;
1.293 + ++pCurrentLeafState )
1.294 + {
1.295 + const state_base_type * pCurrentState(
1.296 + get_pointer( *pCurrentLeafState ) );
1.297 +
1.298 + while ( pCurrentState != 0 )
1.299 + {
1.300 + // The unnecessary try/catch overhead for pointer targets is
1.301 + // typically small compared to the cycles dynamic_cast needs
1.302 + #ifndef BOOST_NO_EXCEPTIONS
1.303 + try
1.304 + #endif
1.305 + {
1.306 + Target result = dynamic_cast< Target >(
1.307 + impl::deref_if_necessary( pCurrentState ) );
1.308 +
1.309 + if ( impl::found( result ) )
1.310 + {
1.311 + return result;
1.312 + }
1.313 + }
1.314 + #ifndef BOOST_NO_EXCEPTIONS
1.315 + // Intentionally swallow std::bad_cast exceptions. We'll throw one
1.316 + // ourselves when we fail to find a state that can be cast to Target
1.317 + catch ( const std::bad_cast & ) {}
1.318 + #endif
1.319 +
1.320 + pCurrentState = pCurrentState->outer_state_ptr();
1.321 + }
1.322 + }
1.323 +
1.324 + return impl::template not_found< Target >();
1.325 + }
1.326 +
1.327 + template< class Target >
1.328 + Target state_downcast() const
1.329 + {
1.330 + typedef detail::state_cast_impl< Target > impl;
1.331 +
1.332 + typename rtti_policy_type::id_type targetType =
1.333 + impl::template type_id< Target, rtti_policy_type::id_type >();
1.334 +
1.335 + for ( typename state_list_type::const_iterator pCurrentLeafState =
1.336 + currentStates_.begin();
1.337 + pCurrentLeafState != currentStatesEnd_;
1.338 + ++pCurrentLeafState )
1.339 + {
1.340 + const state_base_type * pCurrentState(
1.341 + get_pointer( *pCurrentLeafState ) );
1.342 +
1.343 + while ( pCurrentState != 0 )
1.344 + {
1.345 + if ( pCurrentState->dynamic_type() == targetType )
1.346 + {
1.347 + return static_cast< Target >(
1.348 + impl::deref_if_necessary( pCurrentState ) );
1.349 + }
1.350 +
1.351 + pCurrentState = pCurrentState->outer_state_ptr();
1.352 + }
1.353 + }
1.354 +
1.355 + return impl::template not_found< Target >();
1.356 + }
1.357 +
1.358 + typedef detail::state_base< allocator_type, rtti_policy_type >
1.359 + state_base_type;
1.360 +
1.361 + class state_iterator : public std::iterator<
1.362 + std::forward_iterator_tag,
1.363 + state_base_type, std::ptrdiff_t
1.364 + #ifndef BOOST_MSVC_STD_ITERATOR
1.365 + , const state_base_type *, const state_base_type &
1.366 + #endif
1.367 + >
1.368 + {
1.369 + public:
1.370 + //////////////////////////////////////////////////////////////////////
1.371 + explicit state_iterator(
1.372 + typename state_base_type::state_list_type::const_iterator
1.373 + baseIterator
1.374 + ) : baseIterator_( baseIterator ) {}
1.375 +
1.376 + const state_base_type & operator*() const { return **baseIterator_; }
1.377 + const state_base_type * operator->() const
1.378 + {
1.379 + return &**baseIterator_;
1.380 + }
1.381 +
1.382 + state_iterator & operator++() { ++baseIterator_; return *this; }
1.383 + state_iterator operator++( int )
1.384 + {
1.385 + return state_iterator( baseIterator_++ );
1.386 + }
1.387 +
1.388 + bool operator==( const state_iterator & right ) const
1.389 + {
1.390 + return baseIterator_ == right.baseIterator_;
1.391 + }
1.392 + bool operator!=( const state_iterator & right ) const
1.393 + {
1.394 + return !( *this == right );
1.395 + }
1.396 +
1.397 + private:
1.398 + typename state_base_type::state_list_type::const_iterator
1.399 + baseIterator_;
1.400 + };
1.401 +
1.402 + state_iterator state_begin() const
1.403 + {
1.404 + return state_iterator( currentStates_.begin() );
1.405 + }
1.406 +
1.407 + state_iterator state_end() const
1.408 + {
1.409 + return state_iterator( currentStatesEnd_ );
1.410 + }
1.411 +
1.412 + void unconsumed_event( const event_base & ) {}
1.413 +
1.414 + protected:
1.415 + //////////////////////////////////////////////////////////////////////////
1.416 + state_machine() :
1.417 + currentStatesEnd_( currentStates_.end() ),
1.418 + pOutermostState_( 0 ),
1.419 + isInnermostCommonOuter_( false ),
1.420 + performFullExit_( true )
1.421 + {
1.422 + }
1.423 +
1.424 + // This destructor was only made virtual so that that
1.425 + // polymorphic_downcast can be used to cast to MostDerived.
1.426 + virtual ~state_machine()
1.427 + {
1.428 + terminate_impl( false );
1.429 + }
1.430 +
1.431 + public:
1.432 + //////////////////////////////////////////////////////////////////////////
1.433 + // The following declarations should be protected.
1.434 + // They are only public because many compilers lack template friends.
1.435 + //////////////////////////////////////////////////////////////////////////
1.436 + void post_event( const event_base_ptr_type & pEvent )
1.437 + {
1.438 + BOOST_ASSERT( get_pointer( pEvent ) != 0 );
1.439 + eventQueue_.push_back( pEvent );
1.440 + }
1.441 +
1.442 + void post_event( const event_base & evt )
1.443 + {
1.444 + post_event( evt.intrusive_from_this() );
1.445 + }
1.446 +
1.447 + public:
1.448 + //////////////////////////////////////////////////////////////////////////
1.449 + // The following declarations should be private.
1.450 + // They are only public because many compilers lack template friends.
1.451 + //////////////////////////////////////////////////////////////////////////
1.452 + typedef MostDerived inner_context_type;
1.453 + typedef mpl::integral_c< detail::orthogonal_position_type, 0 >
1.454 + inner_orthogonal_position;
1.455 + typedef mpl::integral_c< detail::orthogonal_position_type, 1 >
1.456 + no_of_orthogonal_regions;
1.457 +
1.458 + typedef MostDerived outermost_context_type;
1.459 + typedef state_machine outermost_context_base_type;
1.460 + typedef state_machine * inner_context_ptr_type;
1.461 + typedef typename state_base_type::node_state_base_ptr_type
1.462 + node_state_base_ptr_type;
1.463 + typedef typename state_base_type::leaf_state_ptr_type leaf_state_ptr_type;
1.464 + typedef typename state_base_type::state_list_type state_list_type;
1.465 +
1.466 + typedef mpl::clear< mpl::list<> >::type context_type_list;
1.467 +
1.468 + typedef mpl::bool_< false > shallow_history;
1.469 + typedef mpl::bool_< false > deep_history;
1.470 + typedef mpl::bool_< false > inherited_deep_history;
1.471 +
1.472 + detail::reaction_result react_impl(
1.473 + const event_base_type &,
1.474 + typename rtti_policy_type::id_type )
1.475 + {
1.476 + return detail::do_forward_event;
1.477 + }
1.478 +
1.479 + void exit_impl(
1.480 + inner_context_ptr_type &,
1.481 + typename state_base_type::node_state_base_ptr_type &,
1.482 + bool ) {}
1.483 +
1.484 + void set_outermost_unstable_state(
1.485 + typename state_base_type::node_state_base_ptr_type &
1.486 + pOutermostUnstableState )
1.487 + {
1.488 + pOutermostUnstableState = 0;
1.489 + }
1.490 +
1.491 + // Returns a reference to the context identified by the template
1.492 + // parameter. This can either be _this_ object or one of its direct or
1.493 + // indirect contexts.
1.494 + template< class Context >
1.495 + Context & context()
1.496 + {
1.497 + // As we are in the outermost context here, only this object can be
1.498 + // returned.
1.499 + return *polymorphic_downcast< MostDerived * >( this );
1.500 + }
1.501 +
1.502 + template< class Context >
1.503 + const Context & context() const
1.504 + {
1.505 + // As we are in the outermost context here, only this object can be
1.506 + // returned.
1.507 + return *polymorphic_downcast< const MostDerived * >( this );
1.508 + }
1.509 +
1.510 + outermost_context_type & outermost_context()
1.511 + {
1.512 + return *polymorphic_downcast< MostDerived * >( this );
1.513 + }
1.514 +
1.515 + const outermost_context_type & outermost_context() const
1.516 + {
1.517 + return *polymorphic_downcast< const MostDerived * >( this );
1.518 + }
1.519 +
1.520 + outermost_context_base_type & outermost_context_base()
1.521 + {
1.522 + return *this;
1.523 + }
1.524 +
1.525 + const outermost_context_base_type & outermost_context_base() const
1.526 + {
1.527 + return *this;
1.528 + }
1.529 +
1.530 + void terminate_as_reaction( state_base_type & theState )
1.531 + {
1.532 + terminate_impl( theState, performFullExit_ );
1.533 + pOutermostUnstableState_ = 0;
1.534 + }
1.535 +
1.536 + void terminate_as_part_of_transit( state_base_type & theState )
1.537 + {
1.538 + terminate_impl( theState, performFullExit_ );
1.539 + isInnermostCommonOuter_ = true;
1.540 + }
1.541 +
1.542 + void terminate_as_part_of_transit( state_machine & )
1.543 + {
1.544 + terminate_impl( *pOutermostState_, performFullExit_ );
1.545 + isInnermostCommonOuter_ = true;
1.546 + }
1.547 +
1.548 +
1.549 + template< class State >
1.550 + void add( const intrusive_ptr< State > & pState )
1.551 + {
1.552 + // The second dummy argument is necessary because the call to the
1.553 + // overloaded function add_impl would otherwise be ambiguous.
1.554 + node_state_base_ptr_type pNewOutermostUnstableStateCandidate =
1.555 + add_impl( pState, *pState );
1.556 +
1.557 + if ( isInnermostCommonOuter_ ||
1.558 + is_in_highest_orthogonal_region< State >() &&
1.559 + ( get_pointer( pOutermostUnstableState_ ) ==
1.560 + pState->State::outer_state_ptr() ) )
1.561 + {
1.562 + isInnermostCommonOuter_ = false;
1.563 + pOutermostUnstableState_ = pNewOutermostUnstableStateCandidate;
1.564 + }
1.565 + }
1.566 +
1.567 +
1.568 + void add_inner_state(
1.569 + detail::orthogonal_position_type position,
1.570 + state_base_type * pOutermostState )
1.571 + {
1.572 + BOOST_ASSERT( position == 0 );
1.573 + detail::avoid_unused_warning( position );
1.574 + pOutermostState_ = pOutermostState;
1.575 + }
1.576 +
1.577 + void remove_inner_state( detail::orthogonal_position_type position )
1.578 + {
1.579 + BOOST_ASSERT( position == 0 );
1.580 + detail::avoid_unused_warning( position );
1.581 + pOutermostState_ = 0;
1.582 + }
1.583 +
1.584 +
1.585 + void defer_event(
1.586 + const event_base_type & evt,
1.587 + const state_base_type * pForState )
1.588 + {
1.589 + deferredMap_[ pForState ].push_back( evt.intrusive_from_this() );
1.590 + }
1.591 +
1.592 + void release_events( const state_base_type * pForState )
1.593 + {
1.594 + const typename deferred_map_type::iterator pFound =
1.595 + deferredMap_.find( pForState );
1.596 +
1.597 + // We are not guaranteed to find an entry because a state is marked for
1.598 + // having deferred events _before_ the event is actually deferred. An
1.599 + // exception might be thrown during deferral.
1.600 + if ( pFound != deferredMap_.end() )
1.601 + {
1.602 + eventQueue_.splice( eventQueue_.end(), pFound->second );
1.603 + deferredMap_.erase( pFound );
1.604 + }
1.605 + }
1.606 +
1.607 +
1.608 + template< class HistorizedState >
1.609 + void store_shallow_history()
1.610 + {
1.611 + // 5.2.10.6 declares that reinterpret_casting a function pointer to a
1.612 + // different function pointer and back must yield the same value. The
1.613 + // following reinterpret_cast is the first half of such a sequence.
1.614 + store_history_impl(
1.615 + shallowHistoryMap_,
1.616 + history_key_type::make_history_key< HistorizedState >(),
1.617 + reinterpret_cast< void (*)() >( &HistorizedState::deep_construct ) );
1.618 + }
1.619 +
1.620 + template<
1.621 + class HistoryContext,
1.622 + detail::orthogonal_position_type orthogonalPosition >
1.623 + void clear_shallow_history()
1.624 + {
1.625 + // If you receive a
1.626 + // "use of undefined type 'boost::STATIC_ASSERTION_FAILURE<x>'" or
1.627 + // similar compiler error here then you tried to clear shallow history
1.628 + // for a state that does not have shallow history. That is, the state
1.629 + // does not pass either statechart::has_shallow_history or
1.630 + // statechart::has_full_history to its base class template.
1.631 + BOOST_STATIC_ASSERT( HistoryContext::shallow_history::value );
1.632 +
1.633 + typedef typename mpl::at_c<
1.634 + typename HistoryContext::inner_initial_list,
1.635 + orthogonalPosition >::type historized_state;
1.636 +
1.637 + store_history_impl(
1.638 + shallowHistoryMap_,
1.639 + history_key_type::make_history_key< historized_state >(),
1.640 + 0 );
1.641 + }
1.642 +
1.643 + template< class DefaultState >
1.644 + void construct_with_shallow_history(
1.645 + const typename DefaultState::context_ptr_type & pContext )
1.646 + {
1.647 + construct_with_history_impl< DefaultState >(
1.648 + shallowHistoryMap_, pContext );
1.649 + }
1.650 +
1.651 +
1.652 + template< class HistorizedState, class LeafState >
1.653 + void store_deep_history()
1.654 + {
1.655 + typedef typename detail::make_context_list<
1.656 + typename HistorizedState::context_type,
1.657 + LeafState >::type history_context_list;
1.658 + typedef detail::constructor<
1.659 + history_context_list, outermost_context_base_type > constructor_type;
1.660 + // 5.2.10.6 declares that reinterpret_casting a function pointer to a
1.661 + // different function pointer and back must yield the same value. The
1.662 + // following reinterpret_cast is the first half of such a sequence.
1.663 + store_history_impl(
1.664 + deepHistoryMap_,
1.665 + history_key_type::make_history_key< HistorizedState >(),
1.666 + reinterpret_cast< void (*)() >( &constructor_type::construct ) );
1.667 + }
1.668 +
1.669 + template<
1.670 + class HistoryContext,
1.671 + detail::orthogonal_position_type orthogonalPosition >
1.672 + void clear_deep_history()
1.673 + {
1.674 + // If you receive a
1.675 + // "use of undefined type 'boost::STATIC_ASSERTION_FAILURE<x>'" or
1.676 + // similar compiler error here then you tried to clear deep history for
1.677 + // a state that does not have deep history. That is, the state does not
1.678 + // pass either statechart::has_deep_history or
1.679 + // statechart::has_full_history to its base class template
1.680 + BOOST_STATIC_ASSERT( HistoryContext::deep_history::value );
1.681 +
1.682 + typedef typename mpl::at_c<
1.683 + typename HistoryContext::inner_initial_list,
1.684 + orthogonalPosition >::type historized_state;
1.685 +
1.686 + store_history_impl(
1.687 + deepHistoryMap_,
1.688 + history_key_type::make_history_key< historized_state >(),
1.689 + 0 );
1.690 + }
1.691 +
1.692 + template< class DefaultState >
1.693 + void construct_with_deep_history(
1.694 + const typename DefaultState::context_ptr_type & pContext )
1.695 + {
1.696 + construct_with_history_impl< DefaultState >(
1.697 + deepHistoryMap_, pContext );
1.698 + }
1.699 +
1.700 + private: // implementation
1.701 + //////////////////////////////////////////////////////////////////////////
1.702 + void initial_construct()
1.703 + {
1.704 + InitialState::initial_deep_construct(
1.705 + *polymorphic_downcast< MostDerived * >( this ) );
1.706 + }
1.707 +
1.708 + class initial_construct_function
1.709 + {
1.710 + public:
1.711 + //////////////////////////////////////////////////////////////////////
1.712 + initial_construct_function( state_machine & machine ) :
1.713 + machine_( machine )
1.714 + {
1.715 + }
1.716 +
1.717 + result operator()()
1.718 + {
1.719 + machine_.initial_construct();
1.720 + return detail::result_utility::make_result(
1.721 + detail::do_discard_event ); // there is nothing to be consumed
1.722 + }
1.723 +
1.724 + private:
1.725 + //////////////////////////////////////////////////////////////////////
1.726 + state_machine & machine_;
1.727 + };
1.728 + friend class initial_construct_function;
1.729 +
1.730 + class terminate_function
1.731 + {
1.732 + public:
1.733 + //////////////////////////////////////////////////////////////////////
1.734 + terminate_function( state_machine & machine ) : machine_( machine ) {}
1.735 +
1.736 + result operator()()
1.737 + {
1.738 + machine_.terminate_impl( true );
1.739 + return detail::result_utility::make_result(
1.740 + detail::do_discard_event ); // there is nothing to be consumed
1.741 + }
1.742 +
1.743 + private:
1.744 + //////////////////////////////////////////////////////////////////////
1.745 + state_machine & machine_;
1.746 + };
1.747 + friend class terminate_function;
1.748 +
1.749 + template< class ExceptionEvent >
1.750 + detail::reaction_result handle_exception_event(
1.751 + const ExceptionEvent & exceptionEvent,
1.752 + state_base_type * pCurrentState )
1.753 + {
1.754 + if ( terminated() )
1.755 + {
1.756 + // there is no state that could handle the exception -> bail out
1.757 + throw;
1.758 + }
1.759 +
1.760 + // If we are stable, an event handler has thrown.
1.761 + // Otherwise, either a state constructor, a transition action or an exit
1.762 + // function has thrown and the state machine is now in an invalid state.
1.763 + // This situation can be resolved by the exception event handler
1.764 + // function by orderly transiting to another state or terminating.
1.765 + // As a result of this, the machine must not be unstable when this
1.766 + // function is left.
1.767 + state_base_type * const pOutermostUnstableState =
1.768 + get_pointer( pOutermostUnstableState_ );
1.769 + state_base_type * const pHandlingState = pOutermostUnstableState == 0 ?
1.770 + pCurrentState : pOutermostUnstableState;
1.771 +
1.772 + BOOST_ASSERT( pHandlingState != 0 );
1.773 +
1.774 + // Setting a member variable to a special value for the duration of a
1.775 + // call surely looks like a kludge (normally it should be a parameter of
1.776 + // the call). However, in this case it is unavoidable because the call
1.777 + // below could result in a call to user code where passing through an
1.778 + // additional bool parameter is not acceptable.
1.779 + performFullExit_ = false;
1.780 + const detail::reaction_result reactionResult = pHandlingState->react_impl(
1.781 + exceptionEvent, exceptionEvent.dynamic_type() );
1.782 + // If the above call throws then performFullExit_ will obviously not be
1.783 + // set back to true. In this case the termination triggered by the
1.784 + // scope guard further up in the call stack will take care of this.
1.785 + performFullExit_ = true;
1.786 +
1.787 + if ( ( reactionResult != detail::do_discard_event ) ||
1.788 + ( get_pointer( pOutermostUnstableState_ ) != 0 ) )
1.789 + {
1.790 + throw;
1.791 + }
1.792 +
1.793 + return detail::do_discard_event;
1.794 + }
1.795 +
1.796 + class exception_event_handler
1.797 + {
1.798 + public:
1.799 + //////////////////////////////////////////////////////////////////////
1.800 + exception_event_handler(
1.801 + state_machine & machine,
1.802 + state_base_type * pCurrentState = 0
1.803 + ) :
1.804 + machine_( machine ),
1.805 + pCurrentState_( pCurrentState )
1.806 + {
1.807 + }
1.808 +
1.809 + template< class ExceptionEvent >
1.810 + result operator()(
1.811 + const ExceptionEvent & exceptionEvent )
1.812 + {
1.813 + return detail::result_utility::make_result(
1.814 + machine_.handle_exception_event(
1.815 + exceptionEvent, pCurrentState_ ) );
1.816 + }
1.817 +
1.818 + private:
1.819 + //////////////////////////////////////////////////////////////////////
1.820 + state_machine & machine_;
1.821 + state_base_type * pCurrentState_;
1.822 + };
1.823 + friend class exception_event_handler;
1.824 +
1.825 + class terminator
1.826 + {
1.827 + public:
1.828 + terminator( state_machine & machine ) :
1.829 + machine_( machine ), dismissed_( false ) {}
1.830 + ~terminator()
1.831 + {
1.832 + if ( !dismissed_ ) { machine_.terminate_impl( false ); }
1.833 + }
1.834 + void dismiss() { dismissed_ = true; }
1.835 +
1.836 + private:
1.837 + state_machine & machine_;
1.838 + bool dismissed_;
1.839 + };
1.840 + friend class terminator;
1.841 +
1.842 +
1.843 + void send_event( const event_base_type & evt )
1.844 + {
1.845 + terminator guard( *this );
1.846 + BOOST_ASSERT( get_pointer( pOutermostUnstableState_ ) == 0 );
1.847 + const typename rtti_policy_type::id_type eventType = evt.dynamic_type();
1.848 + detail::reaction_result reactionResult = detail::do_forward_event;
1.849 +
1.850 + for (
1.851 + typename state_list_type::iterator pState = currentStates_.begin();
1.852 + ( reactionResult == detail::do_forward_event ) &&
1.853 + ( pState != currentStatesEnd_ );
1.854 + ++pState )
1.855 + {
1.856 + // CAUTION: The following statement could modify our state list!
1.857 + // We must not continue iterating if the event was consumed
1.858 + reactionResult = detail::result_utility::get_result( translator_(
1.859 + detail::send_function<
1.860 + state_base_type, event_base_type, rtti_policy_type::id_type >(
1.861 + **pState, evt, eventType ),
1.862 + exception_event_handler( *this, get_pointer( *pState ) ) ) );
1.863 + }
1.864 +
1.865 + guard.dismiss();
1.866 +
1.867 + if ( reactionResult == detail::do_forward_event )
1.868 + {
1.869 + polymorphic_downcast< MostDerived * >( this )->unconsumed_event( evt );
1.870 + }
1.871 + }
1.872 +
1.873 +
1.874 + void process_queued_events()
1.875 + {
1.876 + while ( !eventQueue_.empty() )
1.877 + {
1.878 + const event_base_ptr_type pCurrentEvent( eventQueue_.front() );
1.879 + eventQueue_.pop_front();
1.880 + send_event( *pCurrentEvent );
1.881 + }
1.882 + }
1.883 +
1.884 +
1.885 + void terminate_impl( bool performFullExit )
1.886 + {
1.887 + performFullExit_ = true;
1.888 +
1.889 + if ( !terminated() )
1.890 + {
1.891 + // this also empties deferredMap_
1.892 + terminate_impl( *pOutermostState_, performFullExit );
1.893 + }
1.894 +
1.895 + eventQueue_.clear();
1.896 + shallowHistoryMap_.clear();
1.897 + deepHistoryMap_.clear();
1.898 + }
1.899 +
1.900 + void terminate_impl( state_base_type & theState, bool performFullExit )
1.901 + {
1.902 + isInnermostCommonOuter_ = false;
1.903 +
1.904 + // If pOutermostUnstableState_ == 0, we know for sure that
1.905 + // currentStates_.size() > 0, otherwise theState couldn't be alive any
1.906 + // more
1.907 + if ( get_pointer( pOutermostUnstableState_ ) != 0 )
1.908 + {
1.909 + theState.remove_from_state_list(
1.910 + currentStatesEnd_, pOutermostUnstableState_, performFullExit );
1.911 + }
1.912 + // Optimization: We want to find out whether currentStates_ has size 1
1.913 + // and if yes use the optimized implementation below. Since
1.914 + // list<>::size() is implemented quite inefficiently in some std libs
1.915 + // it is best to just decrement the currentStatesEnd_ here and
1.916 + // increment it again, if the test failed.
1.917 + else if ( currentStates_.begin() == --currentStatesEnd_ )
1.918 + {
1.919 + // The machine is stable and there is exactly one innermost state.
1.920 + // The following optimization is only correct for a stable machine
1.921 + // without orthogonal regions.
1.922 + leaf_state_ptr_type & pState = *currentStatesEnd_;
1.923 + pState->exit_impl(
1.924 + pState, pOutermostUnstableState_, performFullExit );
1.925 + }
1.926 + else
1.927 + {
1.928 + BOOST_ASSERT( currentStates_.size() > 1 );
1.929 + // The machine is stable and there are multiple innermost states
1.930 + theState.remove_from_state_list(
1.931 + ++currentStatesEnd_, pOutermostUnstableState_, performFullExit );
1.932 + }
1.933 + }
1.934 +
1.935 +
1.936 + node_state_base_ptr_type add_impl(
1.937 + const leaf_state_ptr_type & pState,
1.938 + detail::leaf_state< allocator_type, rtti_policy_type > & )
1.939 + {
1.940 + if ( currentStatesEnd_ == currentStates_.end() )
1.941 + {
1.942 + pState->set_list_position(
1.943 + currentStates_.insert( currentStatesEnd_, pState ) );
1.944 + }
1.945 + else
1.946 + {
1.947 + *currentStatesEnd_ = pState;
1.948 + pState->set_list_position( currentStatesEnd_ );
1.949 + ++currentStatesEnd_;
1.950 + }
1.951 +
1.952 + return 0;
1.953 + }
1.954 +
1.955 + node_state_base_ptr_type add_impl(
1.956 + const node_state_base_ptr_type & pState,
1.957 + state_base_type & )
1.958 + {
1.959 + return pState;
1.960 + }
1.961 +
1.962 + template< class State >
1.963 + static bool is_in_highest_orthogonal_region()
1.964 + {
1.965 + return mpl::equal_to<
1.966 + typename State::orthogonal_position,
1.967 + mpl::minus<
1.968 + typename State::context_type::no_of_orthogonal_regions,
1.969 + mpl::integral_c< detail::orthogonal_position_type, 1 > >
1.970 + >::value;
1.971 + }
1.972 +
1.973 +
1.974 + typedef detail::history_key< rtti_policy_type > history_key_type;
1.975 +
1.976 + typedef std::map<
1.977 + history_key_type, void (*)(),
1.978 + std::less< history_key_type >,
1.979 + typename boost::detail::allocator::rebind_to<
1.980 + allocator_type, std::pair< const history_key_type, void (*)() >
1.981 + >::type
1.982 + > history_map_type;
1.983 +
1.984 + void store_history_impl(
1.985 + history_map_type & historyMap,
1.986 + const history_key_type & historyId,
1.987 + void (*pConstructFunction)() )
1.988 + {
1.989 + historyMap[ historyId ] = pConstructFunction;
1.990 + }
1.991 +
1.992 + template< class DefaultState >
1.993 + void construct_with_history_impl(
1.994 + history_map_type & historyMap,
1.995 + const typename DefaultState::context_ptr_type & pContext )
1.996 + {
1.997 + typename history_map_type::iterator pFoundSlot = historyMap.find(
1.998 + history_key_type::make_history_key< DefaultState >() );
1.999 +
1.1000 + if ( ( pFoundSlot == historyMap.end() ) || ( pFoundSlot->second == 0 ) )
1.1001 + {
1.1002 + // We have never entered this state before or history was cleared
1.1003 + DefaultState::deep_construct(
1.1004 + pContext, *polymorphic_downcast< MostDerived * >( this ) );
1.1005 + }
1.1006 + else
1.1007 + {
1.1008 + typedef void construct_function(
1.1009 + const typename DefaultState::context_ptr_type &,
1.1010 + typename DefaultState::outermost_context_base_type & );
1.1011 + // 5.2.10.6 declares that reinterpret_casting a function pointer to a
1.1012 + // different function pointer and back must yield the same value. The
1.1013 + // following reinterpret_cast is the second half of such a sequence.
1.1014 + construct_function * const pConstructFunction =
1.1015 + reinterpret_cast< construct_function * >( pFoundSlot->second );
1.1016 + (*pConstructFunction)(
1.1017 + pContext, *polymorphic_downcast< MostDerived * >( this ) );
1.1018 + }
1.1019 + }
1.1020 +
1.1021 + typedef std::list<
1.1022 + event_base_ptr_type,
1.1023 + typename boost::detail::allocator::rebind_to<
1.1024 + allocator_type, event_base_ptr_type >::type
1.1025 + > event_queue_type;
1.1026 +
1.1027 + typedef std::map<
1.1028 + const state_base_type *, event_queue_type,
1.1029 + std::less< const state_base_type * >,
1.1030 + typename boost::detail::allocator::rebind_to<
1.1031 + allocator_type,
1.1032 + std::pair< const state_base_type * const, event_queue_type >
1.1033 + >::type
1.1034 + > deferred_map_type;
1.1035 +
1.1036 +
1.1037 + event_queue_type eventQueue_;
1.1038 + deferred_map_type deferredMap_;
1.1039 + state_list_type currentStates_;
1.1040 + typename state_list_type::iterator currentStatesEnd_;
1.1041 + state_base_type * pOutermostState_;
1.1042 + bool isInnermostCommonOuter_;
1.1043 + node_state_base_ptr_type pOutermostUnstableState_;
1.1044 + ExceptionTranslator translator_;
1.1045 + bool performFullExit_;
1.1046 + history_map_type shallowHistoryMap_;
1.1047 + history_map_type deepHistoryMap_;
1.1048 +};
1.1049 +
1.1050 +
1.1051 +
1.1052 +} // namespace statechart
1.1053 +} // namespace boost
1.1054 +
1.1055 +
1.1056 +
1.1057 +#endif