Update contrib.
1 #ifndef BOOST_STATECHART_SIMPLE_STATE_HPP_INCLUDED
2 #define BOOST_STATECHART_SIMPLE_STATE_HPP_INCLUDED
3 //////////////////////////////////////////////////////////////////////////////
4 // Copyright 2002-2006 Andreas Huber Doenni
5 // Distributed under the Boost Software License, Version 1.0. (See accompany-
6 // ing file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
7 //////////////////////////////////////////////////////////////////////////////
11 #include <boost/statechart/event.hpp>
13 #include <boost/statechart/detail/leaf_state.hpp>
14 #include <boost/statechart/detail/node_state.hpp>
15 #include <boost/statechart/detail/constructor.hpp>
16 #include <boost/statechart/detail/memory.hpp>
18 #include <boost/mpl/eval_if.hpp>
19 #include <boost/mpl/if.hpp>
20 #include <boost/mpl/identity.hpp>
21 #include <boost/mpl/is_sequence.hpp>
22 #include <boost/mpl/list.hpp>
23 #include <boost/mpl/empty.hpp>
24 #include <boost/mpl/size.hpp>
25 #include <boost/mpl/front.hpp>
26 #include <boost/mpl/at.hpp>
27 #include <boost/mpl/find.hpp>
28 #include <boost/mpl/find_if.hpp>
29 #include <boost/mpl/contains.hpp>
30 #include <boost/mpl/distance.hpp>
31 #include <boost/mpl/deref.hpp>
32 #include <boost/mpl/pop_front.hpp>
33 #include <boost/mpl/push_front.hpp>
34 #include <boost/mpl/clear.hpp>
35 #include <boost/mpl/placeholders.hpp>
36 #include <boost/mpl/bool.hpp>
37 #include <boost/mpl/integral_c.hpp>
38 #include <boost/mpl/less.hpp>
39 #include <boost/mpl/equal_to.hpp>
40 #include <boost/mpl/not.hpp>
41 #include <boost/mpl/or.hpp>
43 #include <boost/mpl/plus.hpp>
44 #include <boost/mpl/max_element.hpp>
45 #include <boost/mpl/greater.hpp>
47 #include <boost/get_pointer.hpp>
48 #include <boost/intrusive_ptr.hpp>
49 #include <boost/assert.hpp>
50 #include <boost/type_traits/is_same.hpp>
51 #include <boost/static_assert.hpp>
52 #include <boost/cast.hpp> // boost::polymorphic_downcast
54 #include <cstddef> // std::size_t
67 //////////////////////////////////////////////////////////////////////////////
69 struct make_list : public mpl::eval_if<
70 mpl::is_sequence< T >,
72 mpl::identity< mpl::list< T > > > {};
74 //////////////////////////////////////////////////////////////////////////////
75 template< class MostDerived, class Context, class InnerInitial >
76 struct simple_state_base_type
79 typedef typename Context::outermost_context_base_type::allocator_type
81 typedef typename Context::outermost_context_base_type::rtti_policy_type
83 typedef typename detail::make_list< InnerInitial >::type
85 typedef typename mpl::size< inner_initial_list >::type
86 inner_initial_list_size;
89 typedef typename mpl::eval_if<
90 mpl::empty< inner_initial_list >,
91 mpl::identity< typename rtti_policy_type::
92 template rtti_derived_type< MostDerived, leaf_state<
94 rtti_policy_type > > >,
95 mpl::identity< typename rtti_policy_type::
96 template rtti_derived_type< MostDerived, node_state<
97 inner_initial_list_size,
99 rtti_policy_type > > > >::type type;
103 //////////////////////////////////////////////////////////////////////////////
104 struct no_transition_function
106 template< class CommonContext >
107 void operator()( CommonContext & ) const {}
110 template< class TransitionContext, class Event >
111 class transition_function
115 void ( TransitionContext::*pTransitionAction )( const Event & ),
118 pTransitionAction_( pTransitionAction ),
123 template< class CommonContext >
124 void operator()( CommonContext & commonContext ) const
126 ( commonContext.template context< TransitionContext >()
127 .*pTransitionAction_ )( evt_ );
131 void ( TransitionContext::*pTransitionAction_ )( const Event & );
136 template< bool contextHasInheritedDeepHistory, bool contextHasDeepHistory >
137 struct deep_history_storer
139 template< class HistorizedState, class LeafState, class Context >
140 static void store_deep_history( Context & ) {}
144 struct deep_history_storer< true, false >
146 template< class HistorizedState, class LeafState, class Context >
147 static void store_deep_history( Context & ctx )
149 ctx.template store_deep_history_impl< LeafState >();
154 struct deep_history_storer< true, true >
156 template< class HistorizedState, class LeafState, class Context >
157 static void store_deep_history( Context & ctx )
159 ctx.outermost_context_base().template store_deep_history<
160 HistorizedState, LeafState >();
161 ctx.template store_deep_history_impl< LeafState >();
167 } // namespace detail
171 //////////////////////////////////////////////////////////////////////////////
177 has_full_history // shallow & deep
182 //////////////////////////////////////////////////////////////////////////////
183 template< class MostDerived,
185 class InnerInitial = mpl::list<>,
186 history_mode historyMode = has_no_history >
187 class simple_state : public detail::simple_state_base_type< MostDerived,
188 typename Context::inner_context_type, InnerInitial >::type
190 typedef typename detail::simple_state_base_type<
191 MostDerived, typename Context::inner_context_type,
192 InnerInitial >::type base_type;
195 //////////////////////////////////////////////////////////////////////////
196 typedef mpl::list<> reactions;
198 typedef typename Context::inner_context_type context_type;
200 template< detail::orthogonal_position_type innerOrthogonalPosition >
203 typedef mpl::integral_c<
204 detail::orthogonal_position_type,
205 innerOrthogonalPosition > inner_orthogonal_position;
206 typedef MostDerived inner_context_type;
209 typedef typename context_type::outermost_context_type
210 outermost_context_type;
212 outermost_context_type & outermost_context()
214 // This assert fails when an attempt is made to access the state machine
215 // from a constructor of a state that is *not* a subtype of state<>.
216 // To correct this, derive from state<> instead of simple_state<>.
217 BOOST_ASSERT( get_pointer( pContext_ ) != 0 );
218 return pContext_->outermost_context();
221 const outermost_context_type & outermost_context() const
223 // This assert fails when an attempt is made to access the state machine
224 // from a constructor of a state that is *not* a subtype of state<>.
225 // To correct this, derive from state<> instead of simple_state<>.
226 BOOST_ASSERT( get_pointer( pContext_ ) != 0 );
227 return pContext_->outermost_context();
230 template< class OtherContext >
231 OtherContext & context()
233 typedef typename mpl::if_<
234 is_same< OtherContext, MostDerived >,
235 context_impl_this_context,
236 context_impl_other_context
238 return impl::template context_impl< OtherContext >( *this );
241 template< class OtherContext >
242 const OtherContext & context() const
244 typedef typename mpl::if_<
245 is_same< OtherContext, MostDerived >,
246 context_impl_this_context,
247 context_impl_other_context
249 return impl::template context_impl< OtherContext >( *this );
252 template< class Target >
253 Target state_cast() const
255 return outermost_context_base().template state_cast< Target >();
258 template< class Target >
259 Target state_downcast() const
261 return outermost_context_base().template state_downcast< Target >();
264 typedef typename context_type::state_base_type state_base_type;
265 typedef typename context_type::state_iterator state_iterator;
267 state_iterator state_begin() const
269 return outermost_context_base().state_begin();
272 state_iterator state_end() const
274 return outermost_context_base().state_end();
278 typedef typename context_type::event_base_ptr_type event_base_ptr_type;
280 void post_event( const event_base_ptr_type & pEvent )
282 outermost_context_base().post_event( pEvent );
285 void post_event( const event_base & evt )
287 outermost_context_base().post_event( evt );
290 result discard_event()
292 return detail::result_utility::make_result( detail::do_discard_event );
295 result forward_event()
297 return detail::result_utility::make_result( detail::do_forward_event );
302 this->state_base_type::defer_event();
303 return detail::result_utility::make_result( detail::do_defer_event );
306 template< class DestinationState >
309 return transit_impl< DestinationState, outermost_context_type >(
310 detail::no_transition_function() );
313 template< class DestinationState, class TransitionContext, class Event >
315 void ( TransitionContext::*pTransitionAction )( const Event & ),
318 return transit_impl< DestinationState, TransitionContext >(
319 detail::transition_function< TransitionContext, Event >(
320 pTransitionAction, evt ) );
325 outermost_context_base().terminate_as_reaction( *this );
326 return detail::result_utility::make_result( detail::do_discard_event );
330 class HistoryContext,
331 detail::orthogonal_position_type orthogonalPosition >
332 void clear_shallow_history()
334 outermost_context_base().template clear_shallow_history<
335 HistoryContext, orthogonalPosition >();
339 class HistoryContext,
340 detail::orthogonal_position_type orthogonalPosition >
341 void clear_deep_history()
343 outermost_context_base().template clear_deep_history<
344 HistoryContext, orthogonalPosition >();
348 //////////////////////////////////////////////////////////////////////////
349 simple_state() : pContext_( 0 ) {}
353 // As a result of a throwing derived class constructor, this destructor
354 // can be called before the context is set.
355 if ( get_pointer( pContext_ ) != 0 )
357 if ( this->deferred_events() )
359 outermost_context_base().release_events( this );
362 pContext_->remove_inner_state( orthogonal_position::value );
367 //////////////////////////////////////////////////////////////////////////
368 // The following declarations should be private.
369 // They are only public because many compilers lack template friends.
370 //////////////////////////////////////////////////////////////////////////
371 typedef typename Context::inner_orthogonal_position orthogonal_position;
374 // "use of undefined type 'boost::STATIC_ASSERTION_FAILURE<x>'" or similar
375 // compiler error here then either this state resides in a non-existent
376 // orthogonal region of the outer state or the outer state does not have
378 BOOST_STATIC_ASSERT( ( mpl::less<
380 typename context_type::no_of_orthogonal_regions >::value ) );
382 typedef MostDerived inner_context_type;
383 typedef mpl::integral_c< detail::orthogonal_position_type, 0 >
384 inner_orthogonal_position;
386 typedef typename context_type::event_base_type event_base_type;
387 typedef typename context_type::rtti_policy_type rtti_policy_type;
389 typedef typename context_type::outermost_context_base_type
390 outermost_context_base_type;
391 typedef typename context_type::inner_context_ptr_type context_ptr_type;
392 typedef typename context_type::state_list_type state_list_type;
393 typedef intrusive_ptr< inner_context_type > inner_context_ptr_type;
394 typedef typename detail::make_list< InnerInitial >::type
396 typedef typename mpl::size< inner_initial_list >::type
397 inner_initial_list_size;
398 typedef mpl::integral_c<
399 detail::orthogonal_position_type,
400 inner_initial_list_size::value > no_of_orthogonal_regions;
401 typedef typename mpl::push_front<
402 typename context_type::context_type_list,
403 context_type >::type context_type_list;
406 // "use of undefined type 'boost::STATIC_ASSERTION_FAILURE<x>'" or similar
407 // compiler error here then the direct or indirect context of this state
408 // has deep history _and_ this state has two or more orthogonal regions.
409 // Boost.Statechart does not currently support deep history in a state whose
410 // direct or indirect inner states have two or more orthogonal regions.
411 // Please consult the documentation on how to work around this limitation.
412 BOOST_STATIC_ASSERT( ( mpl::or_<
414 no_of_orthogonal_regions,
415 mpl::integral_c< detail::orthogonal_position_type, 2 > >,
417 typename context_type::inherited_deep_history > >::value ) );
419 typedef mpl::bool_< ( historyMode & has_shallow_history ) != 0 >
421 typedef typename context_type::shallow_history stores_shallow_history;
423 typedef mpl::bool_< ( historyMode & has_deep_history ) != 0 >
425 typedef typename mpl::or_<
427 typename context_type::inherited_deep_history
428 >::type inherited_deep_history;
429 typedef typename mpl::and_<
430 inherited_deep_history,
431 mpl::empty< inner_initial_list > >::type stores_deep_history;
433 void * operator new( std::size_t size )
435 return detail::allocate< MostDerived,
436 typename outermost_context_type::allocator_type >( size );
439 void operator delete( void * pState )
441 detail::deallocate< MostDerived,
442 typename outermost_context_type::allocator_type >( pState );
445 outermost_context_base_type & outermost_context_base()
447 // This assert fails when an attempt is made to access the state machine
448 // from a constructor of a state that is *not* a subtype of state<>.
449 // To correct this, derive from state<> instead of simple_state<>.
450 BOOST_ASSERT( get_pointer( pContext_ ) != 0 );
451 return pContext_->outermost_context_base();
454 const outermost_context_base_type & outermost_context_base() const
456 // This assert fails when an attempt is made to access the state machine
457 // from a constructor of a state that is *not* a subtype of state<>.
458 // To correct this, derive from state<> instead of simple_state<>.
459 BOOST_ASSERT( get_pointer( pContext_ ) != 0 );
460 return pContext_->outermost_context_base();
463 virtual const state_base_type * outer_state_ptr() const
465 typedef typename mpl::if_<
466 is_same< outermost_context_type, context_type >,
467 outer_state_ptr_impl_outermost,
468 outer_state_ptr_impl_non_outermost
470 return impl::outer_state_ptr_impl( *this );
473 virtual detail::reaction_result react_impl(
474 const event_base_type & evt,
475 typename rtti_policy_type::id_type eventType )
477 typedef typename detail::make_list<
478 typename MostDerived::reactions >::type reaction_list;
479 detail::reaction_result reactionResult =
480 local_react< reaction_list >( evt, eventType );
482 // At this point we can only safely access pContext_ if the handler did
483 // not return do_discard_event!
484 switch ( reactionResult )
486 case detail::do_forward_event:
487 // TODO: The following call to react_impl of our outer state should
488 // be made with a context_type:: prefix to call directly instead of
489 // virtually. For some reason the compiler complains...
490 reactionResult = pContext_->react_impl( evt, eventType );
492 case detail::do_defer_event:
493 outermost_context_base().defer_event( evt, this );
499 return reactionResult;
502 virtual void exit_impl(
503 typename base_type::direct_state_base_ptr_type & pSelf,
504 typename state_base_type::node_state_base_ptr_type &
505 pOutermostUnstableState,
506 bool performFullExit )
508 inner_context_ptr_type pMostDerivedSelf =
509 polymorphic_downcast< MostDerived * >( this );
511 exit_impl( pMostDerivedSelf, pOutermostUnstableState, performFullExit );
515 inner_context_ptr_type & pSelf,
516 typename state_base_type::node_state_base_ptr_type &
517 pOutermostUnstableState,
518 bool performFullExit )
520 switch ( this->ref_count() )
523 if ( get_pointer( pOutermostUnstableState ) ==
524 static_cast< state_base_type * >( this ) )
526 pContext_->set_outermost_unstable_state(
527 pOutermostUnstableState );
528 // fall through to next case intended
536 if ( get_pointer( pOutermostUnstableState ) == 0 )
538 pContext_->set_outermost_unstable_state(
539 pOutermostUnstableState );
542 if ( performFullExit )
545 check_store_shallow_history< stores_shallow_history >();
546 check_store_deep_history< stores_deep_history >();
549 context_ptr_type pContext = pContext_;
552 pContext, pOutermostUnstableState, performFullExit );
560 void set_outermost_unstable_state(
561 typename state_base_type::node_state_base_ptr_type &
562 pOutermostUnstableState )
564 pOutermostUnstableState = this;
567 template< class OtherContext >
568 const typename OtherContext::inner_context_ptr_type & context_ptr() const
570 typedef typename mpl::if_<
571 is_same< OtherContext, context_type >,
572 context_ptr_impl_my_context,
573 context_ptr_impl_other_context
576 return impl::template context_ptr_impl< OtherContext >( *this );
579 static void initial_deep_construct(
580 outermost_context_base_type & outermostContextBase )
582 deep_construct( &outermostContextBase, outermostContextBase );
585 static void deep_construct(
586 const context_ptr_type & pContext,
587 outermost_context_base_type & outermostContextBase )
589 const inner_context_ptr_type pInnerContext(
590 shallow_construct( pContext, outermostContextBase ) );
591 deep_construct_inner< inner_initial_list >(
592 pInnerContext, outermostContextBase );
595 static inner_context_ptr_type shallow_construct(
596 const context_ptr_type & pContext,
597 outermost_context_base_type & outermostContextBase )
599 const inner_context_ptr_type pInnerContext( new MostDerived );
600 pInnerContext->set_context( pContext );
601 outermostContextBase.add( pInnerContext );
602 return pInnerContext;
605 void set_context( const context_ptr_type & pContext )
607 BOOST_ASSERT( get_pointer( pContext ) != 0 );
608 pContext_ = pContext;
609 base_type::set_context(
610 orthogonal_position::value, get_pointer( pContext ) );
613 template< class InnerList >
614 static void deep_construct_inner(
615 const inner_context_ptr_type & pInnerContext,
616 outermost_context_base_type & outermostContextBase )
618 typedef typename mpl::if_<
619 mpl::empty< InnerList >,
620 deep_construct_inner_impl_empty,
621 deep_construct_inner_impl_non_empty
623 impl::template deep_construct_inner_impl< InnerList >(
624 pInnerContext, outermostContextBase );
627 template< class LeafState >
628 void store_deep_history_impl()
630 detail::deep_history_storer<
631 context_type::inherited_deep_history::value,
632 context_type::deep_history::value
633 >::template store_deep_history< MostDerived, LeafState >(
638 //////////////////////////////////////////////////////////////////////////
639 struct context_ptr_impl_other_context
641 template< class OtherContext, class State >
642 static const typename OtherContext::inner_context_ptr_type &
643 context_ptr_impl( const State & stt )
645 // This assert fails when an attempt is made to access an outer
646 // context from a constructor of a state that is *not* a subtype of
647 // state<>. To correct this, derive from state<> instead of
649 BOOST_ASSERT( get_pointer( stt.pContext_ ) != 0 );
650 return stt.pContext_->template context_ptr< OtherContext >();
653 friend struct context_ptr_impl_other_context;
655 struct context_ptr_impl_my_context
657 template< class OtherContext, class State >
658 static const typename OtherContext::inner_context_ptr_type &
659 context_ptr_impl( const State & stt )
661 // This assert fails when an attempt is made to access an outer
662 // context from a constructor of a state that is *not* a subtype of
663 // state<>. To correct this, derive from state<> instead of
665 BOOST_ASSERT( get_pointer( stt.pContext_ ) != 0 );
666 return stt.pContext_;
669 friend struct context_ptr_impl_my_context;
671 struct context_impl_other_context
673 template< class OtherContext, class State >
674 static OtherContext & context_impl( State & stt )
676 // This assert fails when an attempt is made to access an outer
677 // context from a constructor of a state that is *not* a subtype of
678 // state<>. To correct this, derive from state<> instead of
680 BOOST_ASSERT( get_pointer( stt.pContext_ ) != 0 );
681 return stt.pContext_->template context< OtherContext >();
684 friend struct context_impl_other_context;
686 struct context_impl_this_context
688 template< class OtherContext, class State >
689 static OtherContext & context_impl( State & stt )
691 return *polymorphic_downcast< MostDerived * >( &stt );
694 friend struct context_impl_this_context;
696 template< class DestinationState,
697 class TransitionContext,
698 class TransitionAction >
699 result transit_impl( const TransitionAction & transitionAction )
701 typedef typename mpl::find_if<
704 typename DestinationState::context_type_list,
705 mpl::placeholders::_ > >::type common_context_iter;
706 typedef typename mpl::deref< common_context_iter >::type
708 typedef typename mpl::distance<
709 typename mpl::begin< context_type_list >::type,
710 common_context_iter >::type termination_state_position;
711 typedef typename mpl::push_front< context_type_list, MostDerived >::type
712 possible_transition_contexts;
713 typedef typename mpl::at<
714 possible_transition_contexts,
715 termination_state_position >::type termination_state_type;
717 termination_state_type & terminationState(
718 context< termination_state_type >() );
720 common_context_type::inner_context_ptr_type pCommonContext(
721 terminationState.context_ptr< common_context_type >() );
722 outermost_context_base_type & outermostContextBase(
723 pCommonContext->outermost_context_base() );
725 #ifdef BOOST_STATECHART_RELAX_TRANSITION_CONTEXT
726 typedef typename mpl::distance<
727 typename mpl::begin< possible_transition_contexts >::type,
729 possible_transition_contexts, TransitionContext >::type
730 >::type proposed_transition_context_position;
732 typedef typename mpl::plus<
733 termination_state_position,
735 >::type uml_transition_context_position;
737 typedef typename mpl::deref< typename mpl::max_element<
739 proposed_transition_context_position,
740 uml_transition_context_position >,
741 mpl::greater< mpl::placeholders::_, mpl::placeholders::_ >
742 >::type >::type real_transition_context_position;
744 typedef typename mpl::at<
745 possible_transition_contexts,
746 real_transition_context_position >::type real_transition_context_type;
749 # pragma warning( push )
750 # pragma warning( disable: 4127 ) // conditional expression is constant
752 if ( ( proposed_transition_context_position::value == 0 ) &&
753 ( inner_initial_list_size::value == 0 ) )
755 transitionAction( *polymorphic_downcast< MostDerived * >( this ) );
756 outermostContextBase.terminate_as_part_of_transit( terminationState );
758 else if ( proposed_transition_context_position::value >=
759 uml_transition_context_position::value )
761 real_transition_context_type & transitionContext =
762 context< real_transition_context_type >();
763 outermostContextBase.terminate_as_part_of_transit( terminationState );
764 transitionAction( transitionContext );
768 typename real_transition_context_type::inner_context_ptr_type
769 pTransitionContext = context_ptr< real_transition_context_type >();
770 outermostContextBase.terminate_as_part_of_transit(
771 *pTransitionContext );
772 transitionAction( *pTransitionContext );
773 pTransitionContext = 0;
774 outermostContextBase.terminate_as_part_of_transit( terminationState );
777 # pragma warning( pop )
780 outermostContextBase.terminate_as_part_of_transit( terminationState );
781 transitionAction( *pCommonContext );
784 typedef typename detail::make_context_list<
785 common_context_type, DestinationState >::type context_list_type;
788 // "use of undefined type 'boost::STATIC_ASSERTION_FAILURE<x>'" or
789 // similar compiler error here then you tried to make an invalid
790 // transition between different orthogonal regions.
791 BOOST_STATIC_ASSERT( ( mpl::equal_to<
792 typename termination_state_type::orthogonal_position,
793 typename mpl::front< context_list_type >::type::orthogonal_position
797 context_list_type, outermost_context_base_type >::construct(
798 pCommonContext, outermostContextBase );
800 return detail::result_utility::make_result( detail::do_discard_event );
803 struct local_react_impl_non_empty
805 template< class ReactionList, class State >
806 static detail::reaction_result local_react_impl(
808 const event_base_type & evt,
809 typename rtti_policy_type::id_type eventType )
811 detail::reaction_result reactionResult =
812 mpl::front< ReactionList >::type::react(
813 *polymorphic_downcast< MostDerived * >( &stt ),
816 if ( reactionResult == detail::no_reaction )
818 reactionResult = stt.template local_react<
819 typename mpl::pop_front< ReactionList >::type >(
823 return reactionResult;
826 friend struct local_react_impl_non_empty;
828 struct local_react_impl_empty
830 template< class ReactionList, class State >
831 static detail::reaction_result local_react_impl(
832 State &, const event_base_type &, typename rtti_policy_type::id_type )
834 return detail::do_forward_event;
838 template< class ReactionList >
839 detail::reaction_result local_react(
840 const event_base_type & evt,
841 typename rtti_policy_type::id_type eventType )
843 typedef typename mpl::if_<
844 mpl::empty< ReactionList >,
845 local_react_impl_empty,
846 local_react_impl_non_empty
848 return impl::template local_react_impl< ReactionList >(
849 *this, evt, eventType );
852 struct outer_state_ptr_impl_non_outermost
854 template< class State >
855 static const state_base_type * outer_state_ptr_impl( const State & stt )
857 return get_pointer( stt.pContext_ );
860 friend struct outer_state_ptr_impl_non_outermost;
862 struct outer_state_ptr_impl_outermost
864 template< class State >
865 static const state_base_type * outer_state_ptr_impl( const State & )
871 struct deep_construct_inner_impl_non_empty
873 template< class InnerList >
874 static void deep_construct_inner_impl(
875 const inner_context_ptr_type & pInnerContext,
876 outermost_context_base_type & outermostContextBase )
878 typedef typename mpl::front< InnerList >::type current_inner;
881 // "use of undefined type 'boost::STATIC_ASSERTION_FAILURE<x>'" or
882 // similar compiler error here then there is a mismatch between the
883 // orthogonal position of a state and its position in the inner
884 // initial list of its outer state.
885 BOOST_STATIC_ASSERT( ( is_same<
888 typename current_inner::context_type::inner_initial_list,
889 typename current_inner::orthogonal_position >::type >::value ) );
891 current_inner::deep_construct( pInnerContext, outermostContextBase );
892 deep_construct_inner< typename mpl::pop_front< InnerList >::type >(
893 pInnerContext, outermostContextBase );
897 struct deep_construct_inner_impl_empty
899 template< class InnerList >
900 static void deep_construct_inner_impl(
901 const inner_context_ptr_type &, outermost_context_base_type & ) {}
904 struct check_store_shallow_history_impl_no
906 template< class State >
907 static void check_store_shallow_history_impl( State & ) {}
910 struct check_store_shallow_history_impl_yes
912 template< class State >
913 static void check_store_shallow_history_impl( State & stt )
915 stt.outermost_context_base().template store_shallow_history<
919 friend struct check_store_shallow_history_impl_yes;
921 template< class StoreShallowHistory >
922 void check_store_shallow_history()
924 typedef typename mpl::if_<
926 check_store_shallow_history_impl_yes,
927 check_store_shallow_history_impl_no
929 impl::check_store_shallow_history_impl( *this );
932 struct check_store_deep_history_impl_no
934 template< class State >
935 static void check_store_deep_history_impl( State & ) {}
938 struct check_store_deep_history_impl_yes
940 template< class State >
941 static void check_store_deep_history_impl( State & stt )
943 stt.store_deep_history_impl< MostDerived >();
946 friend struct check_store_deep_history_impl_yes;
948 template< class StoreDeepHistory >
949 void check_store_deep_history()
951 typedef typename mpl::if_<
953 check_store_deep_history_impl_yes,
954 check_store_deep_history_impl_no
956 impl::check_store_deep_history_impl( *this );
960 context_ptr_type pContext_;
965 #ifdef BOOST_NO_ARGUMENT_DEPENDENT_LOOKUP
966 } // namespace statechart
971 template< class MostDerived, class Context,
972 class InnerInitial, history_mode historyMode >
973 inline void intrusive_ptr_release( const ::boost::statechart::simple_state<
974 MostDerived, Context, InnerInitial, historyMode > * pBase )
976 if ( pBase->release() )
978 // The cast is necessary because the simple_state destructor is non-
979 // virtual (and inaccessible from this context)
980 delete polymorphic_downcast< const MostDerived * >( pBase );
986 #ifndef BOOST_NO_ARGUMENT_DEPENDENT_LOOKUP
987 } // namespace statechart