os/ossrv/ossrv_pub/boost_apis/boost/statechart/state_machine.hpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
#ifndef BOOST_STATECHART_STATE_MACHINE_HPP_INCLUDED
sl@0
     2
#define BOOST_STATECHART_STATE_MACHINE_HPP_INCLUDED
sl@0
     3
//////////////////////////////////////////////////////////////////////////////
sl@0
     4
// Copyright 2002-2006 Andreas Huber Doenni
sl@0
     5
// Distributed under the Boost Software License, Version 1.0. (See accompany-
sl@0
     6
// ing file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
sl@0
     7
//////////////////////////////////////////////////////////////////////////////
sl@0
     8
sl@0
     9
sl@0
    10
sl@0
    11
#include <boost/statechart/event.hpp>
sl@0
    12
#include <boost/statechart/null_exception_translator.hpp>
sl@0
    13
#include <boost/statechart/result.hpp>
sl@0
    14
sl@0
    15
#include <boost/statechart/detail/rtti_policy.hpp>
sl@0
    16
#include <boost/statechart/detail/state_base.hpp>
sl@0
    17
#include <boost/statechart/detail/leaf_state.hpp>
sl@0
    18
#include <boost/statechart/detail/node_state.hpp>
sl@0
    19
#include <boost/statechart/detail/constructor.hpp>
sl@0
    20
#include <boost/statechart/detail/avoid_unused_warning.hpp>
sl@0
    21
sl@0
    22
#include <boost/mpl/list.hpp>
sl@0
    23
#include <boost/mpl/clear.hpp>
sl@0
    24
#include <boost/mpl/if.hpp>
sl@0
    25
#include <boost/mpl/at.hpp>
sl@0
    26
#include <boost/mpl/integral_c.hpp>
sl@0
    27
#include <boost/mpl/minus.hpp>
sl@0
    28
#include <boost/mpl/equal_to.hpp>
sl@0
    29
sl@0
    30
#include <boost/intrusive_ptr.hpp>
sl@0
    31
#include <boost/type_traits/is_pointer.hpp>
sl@0
    32
#include <boost/type_traits/remove_reference.hpp>
sl@0
    33
#include <boost/noncopyable.hpp>
sl@0
    34
#include <boost/assert.hpp>
sl@0
    35
#include <boost/static_assert.hpp>
sl@0
    36
#include <boost/cast.hpp> // boost::polymorphic_downcast
sl@0
    37
// BOOST_NO_EXCEPTIONS, BOOST_MSVC, BOOST_MSVC_STD_ITERATOR
sl@0
    38
#include <boost/config.hpp>
sl@0
    39
sl@0
    40
#include <boost/detail/allocator_utilities.hpp>
sl@0
    41
sl@0
    42
#ifdef BOOST_MSVC
sl@0
    43
#  pragma warning( push )
sl@0
    44
#  pragma warning( disable: 4702 ) // unreachable code (in release mode only)
sl@0
    45
#endif
sl@0
    46
sl@0
    47
#include <map>
sl@0
    48
sl@0
    49
#ifdef BOOST_MSVC
sl@0
    50
#  pragma warning( pop )
sl@0
    51
#endif
sl@0
    52
sl@0
    53
#include <memory>   // std::allocator
sl@0
    54
#include <typeinfo> // std::bad_cast
sl@0
    55
#include <functional> // std::less
sl@0
    56
#include <iterator>
sl@0
    57
sl@0
    58
sl@0
    59
sl@0
    60
#ifdef BOOST_MSVC
sl@0
    61
// We permanently turn off the following level 4 warnings because users will
sl@0
    62
// have to do so themselves anyway if we turn them back on
sl@0
    63
#  pragma warning( disable: 4511 ) // copy constructor could not be generated
sl@0
    64
#  pragma warning( disable: 4512 ) // assignment op could not be generated
sl@0
    65
#endif
sl@0
    66
sl@0
    67
sl@0
    68
sl@0
    69
namespace boost
sl@0
    70
{
sl@0
    71
namespace statechart
sl@0
    72
{
sl@0
    73
namespace detail
sl@0
    74
{
sl@0
    75
sl@0
    76
sl@0
    77
sl@0
    78
//////////////////////////////////////////////////////////////////////////////
sl@0
    79
template< class StateBaseType, class EventBaseType, class IdType >
sl@0
    80
class send_function
sl@0
    81
{
sl@0
    82
  public:
sl@0
    83
    //////////////////////////////////////////////////////////////////////////
sl@0
    84
    send_function(
sl@0
    85
      StateBaseType & toState,
sl@0
    86
      const EventBaseType & evt,
sl@0
    87
      IdType eventType
sl@0
    88
    ) :
sl@0
    89
      toState_( toState ), evt_( evt ), eventType_( eventType )
sl@0
    90
    {
sl@0
    91
    }
sl@0
    92
sl@0
    93
    result operator()()
sl@0
    94
    {
sl@0
    95
      return detail::result_utility::make_result(
sl@0
    96
        toState_.react_impl( evt_, eventType_ ) );
sl@0
    97
    }
sl@0
    98
sl@0
    99
  private:
sl@0
   100
    //////////////////////////////////////////////////////////////////////////
sl@0
   101
    StateBaseType & toState_;
sl@0
   102
    const EventBaseType & evt_;
sl@0
   103
    IdType eventType_;
sl@0
   104
};
sl@0
   105
sl@0
   106
sl@0
   107
//////////////////////////////////////////////////////////////////////////////
sl@0
   108
struct state_cast_impl_pointer_target
sl@0
   109
{
sl@0
   110
  public:
sl@0
   111
    //////////////////////////////////////////////////////////////////////////
sl@0
   112
    template< class StateBaseType >
sl@0
   113
    static const StateBaseType * deref_if_necessary(
sl@0
   114
      const StateBaseType * pState )
sl@0
   115
    {
sl@0
   116
      return pState;
sl@0
   117
    }
sl@0
   118
sl@0
   119
    template< class Target, class IdType >
sl@0
   120
    static IdType type_id()
sl@0
   121
    {
sl@0
   122
      Target p = 0;
sl@0
   123
      return type_id_impl< IdType >( p );
sl@0
   124
    }
sl@0
   125
sl@0
   126
    static bool found( const void * pFound )
sl@0
   127
    {
sl@0
   128
      return pFound != 0;
sl@0
   129
    }
sl@0
   130
sl@0
   131
    template< class Target >
sl@0
   132
    static Target not_found()
sl@0
   133
    {
sl@0
   134
      return 0;
sl@0
   135
    }
sl@0
   136
sl@0
   137
  private:
sl@0
   138
    //////////////////////////////////////////////////////////////////////////
sl@0
   139
    template< class IdType, class Type >
sl@0
   140
    static IdType type_id_impl( const Type * )
sl@0
   141
    {
sl@0
   142
      return Type::static_type();
sl@0
   143
    }
sl@0
   144
};
sl@0
   145
sl@0
   146
struct state_cast_impl_reference_target
sl@0
   147
{
sl@0
   148
  template< class StateBaseType >
sl@0
   149
  static const StateBaseType & deref_if_necessary(
sl@0
   150
    const StateBaseType * pState )
sl@0
   151
  {
sl@0
   152
    return *pState;
sl@0
   153
  }
sl@0
   154
sl@0
   155
  template< class Target, class IdType >
sl@0
   156
  static IdType type_id()
sl@0
   157
  {
sl@0
   158
    return remove_reference< Target >::type::static_type();
sl@0
   159
  }
sl@0
   160
sl@0
   161
  template< class Dummy >
sl@0
   162
  static bool found( const Dummy & )
sl@0
   163
  {
sl@0
   164
    return true;
sl@0
   165
  }
sl@0
   166
sl@0
   167
  template< class Target >
sl@0
   168
  static Target not_found()
sl@0
   169
  {
sl@0
   170
    throw std::bad_cast();
sl@0
   171
  }
sl@0
   172
};
sl@0
   173
sl@0
   174
template< class Target >
sl@0
   175
struct state_cast_impl : public mpl::if_<
sl@0
   176
  is_pointer< Target >,
sl@0
   177
  state_cast_impl_pointer_target,
sl@0
   178
  state_cast_impl_reference_target
sl@0
   179
>::type {};
sl@0
   180
sl@0
   181
sl@0
   182
//////////////////////////////////////////////////////////////////////////////
sl@0
   183
template< class RttiPolicy >
sl@0
   184
class history_key
sl@0
   185
{
sl@0
   186
  public:
sl@0
   187
    //////////////////////////////////////////////////////////////////////////
sl@0
   188
    template< class HistorizedState >
sl@0
   189
    static history_key make_history_key()
sl@0
   190
    {
sl@0
   191
      return history_key(
sl@0
   192
        HistorizedState::context_type::static_type(),
sl@0
   193
        HistorizedState::orthogonal_position::value );
sl@0
   194
    }
sl@0
   195
sl@0
   196
    typename RttiPolicy::id_type history_context_type() const
sl@0
   197
    {
sl@0
   198
      return historyContextType_;
sl@0
   199
    }
sl@0
   200
sl@0
   201
    friend bool operator<(
sl@0
   202
      const history_key & left, const history_key & right )
sl@0
   203
    {
sl@0
   204
      return
sl@0
   205
        std::less< typename RttiPolicy::id_type >()( 
sl@0
   206
          left.historyContextType_, right.historyContextType_ ) ||
sl@0
   207
        ( ( left.historyContextType_ == right.historyContextType_ ) &&
sl@0
   208
          ( left.historizedOrthogonalRegion_ <
sl@0
   209
            right.historizedOrthogonalRegion_ ) );
sl@0
   210
    }
sl@0
   211
sl@0
   212
  private:
sl@0
   213
    //////////////////////////////////////////////////////////////////////////
sl@0
   214
    history_key(
sl@0
   215
      typename RttiPolicy::id_type historyContextType, 
sl@0
   216
      orthogonal_position_type historizedOrthogonalRegion
sl@0
   217
    ) :
sl@0
   218
      historyContextType_( historyContextType ),
sl@0
   219
      historizedOrthogonalRegion_( historizedOrthogonalRegion )
sl@0
   220
    {
sl@0
   221
    }
sl@0
   222
sl@0
   223
    const typename RttiPolicy::id_type historyContextType_;
sl@0
   224
    const orthogonal_position_type historizedOrthogonalRegion_;
sl@0
   225
};
sl@0
   226
sl@0
   227
sl@0
   228
sl@0
   229
} // namespace detail
sl@0
   230
sl@0
   231
sl@0
   232
sl@0
   233
//////////////////////////////////////////////////////////////////////////////
sl@0
   234
template< class MostDerived,
sl@0
   235
          class InitialState, 
sl@0
   236
          class Allocator = std::allocator< void >,
sl@0
   237
          class ExceptionTranslator = null_exception_translator >
sl@0
   238
class state_machine : noncopyable
sl@0
   239
{
sl@0
   240
  public:
sl@0
   241
    //////////////////////////////////////////////////////////////////////////
sl@0
   242
    typedef Allocator allocator_type;
sl@0
   243
    typedef detail::rtti_policy rtti_policy_type;
sl@0
   244
    typedef event_base event_base_type;
sl@0
   245
    typedef intrusive_ptr< const event_base_type > event_base_ptr_type;
sl@0
   246
sl@0
   247
    void initiate()
sl@0
   248
    {
sl@0
   249
      terminate();
sl@0
   250
sl@0
   251
      {
sl@0
   252
        terminator guard( *this );
sl@0
   253
        detail::result_utility::get_result( translator_(
sl@0
   254
          initial_construct_function( *this ),
sl@0
   255
          exception_event_handler( *this ) ) );
sl@0
   256
        guard.dismiss();
sl@0
   257
      }
sl@0
   258
sl@0
   259
      process_queued_events();
sl@0
   260
    }
sl@0
   261
sl@0
   262
    void terminate()
sl@0
   263
    {
sl@0
   264
      terminator guard( *this );
sl@0
   265
      detail::result_utility::get_result( translator_(
sl@0
   266
        terminate_function( *this ),
sl@0
   267
        exception_event_handler( *this ) ) );
sl@0
   268
      guard.dismiss();
sl@0
   269
    }
sl@0
   270
sl@0
   271
    bool terminated() const
sl@0
   272
    {
sl@0
   273
      return pOutermostState_ == 0;
sl@0
   274
    }
sl@0
   275
sl@0
   276
    void process_event( const event_base_type & evt )
sl@0
   277
    {
sl@0
   278
      send_event( evt );
sl@0
   279
      process_queued_events();
sl@0
   280
    }
sl@0
   281
sl@0
   282
    template< class Target >
sl@0
   283
    Target state_cast() const
sl@0
   284
    {
sl@0
   285
      typedef detail::state_cast_impl< Target > impl;
sl@0
   286
sl@0
   287
      for ( typename state_list_type::const_iterator pCurrentLeafState =
sl@0
   288
              currentStates_.begin();
sl@0
   289
            pCurrentLeafState != currentStatesEnd_;
sl@0
   290
            ++pCurrentLeafState )
sl@0
   291
      {
sl@0
   292
        const state_base_type * pCurrentState(
sl@0
   293
          get_pointer( *pCurrentLeafState ) );
sl@0
   294
sl@0
   295
        while ( pCurrentState != 0 )
sl@0
   296
        {
sl@0
   297
          // The unnecessary try/catch overhead for pointer targets is
sl@0
   298
          // typically small compared to the cycles dynamic_cast needs
sl@0
   299
          #ifndef BOOST_NO_EXCEPTIONS
sl@0
   300
          try
sl@0
   301
          #endif
sl@0
   302
          {
sl@0
   303
            Target result = dynamic_cast< Target >(
sl@0
   304
              impl::deref_if_necessary( pCurrentState ) );
sl@0
   305
sl@0
   306
            if ( impl::found( result ) )
sl@0
   307
            {
sl@0
   308
              return result;
sl@0
   309
            }
sl@0
   310
          }
sl@0
   311
          #ifndef BOOST_NO_EXCEPTIONS
sl@0
   312
          // Intentionally swallow std::bad_cast exceptions. We'll throw one
sl@0
   313
          // ourselves when we fail to find a state that can be cast to Target
sl@0
   314
          catch ( const std::bad_cast & ) {}
sl@0
   315
          #endif
sl@0
   316
sl@0
   317
          pCurrentState = pCurrentState->outer_state_ptr();
sl@0
   318
        }
sl@0
   319
      }
sl@0
   320
sl@0
   321
      return impl::template not_found< Target >();
sl@0
   322
    }
sl@0
   323
sl@0
   324
    template< class Target >
sl@0
   325
    Target state_downcast() const
sl@0
   326
    {
sl@0
   327
      typedef detail::state_cast_impl< Target > impl;
sl@0
   328
sl@0
   329
      typename rtti_policy_type::id_type targetType =
sl@0
   330
        impl::template type_id< Target, rtti_policy_type::id_type >();
sl@0
   331
sl@0
   332
      for ( typename state_list_type::const_iterator pCurrentLeafState =
sl@0
   333
              currentStates_.begin();
sl@0
   334
            pCurrentLeafState != currentStatesEnd_;
sl@0
   335
            ++pCurrentLeafState )
sl@0
   336
      {
sl@0
   337
        const state_base_type * pCurrentState(
sl@0
   338
          get_pointer( *pCurrentLeafState ) );
sl@0
   339
sl@0
   340
        while ( pCurrentState != 0 )
sl@0
   341
        {
sl@0
   342
          if ( pCurrentState->dynamic_type() == targetType )
sl@0
   343
          {
sl@0
   344
            return static_cast< Target >(
sl@0
   345
              impl::deref_if_necessary( pCurrentState ) );
sl@0
   346
          }
sl@0
   347
sl@0
   348
          pCurrentState = pCurrentState->outer_state_ptr();
sl@0
   349
        }
sl@0
   350
      }
sl@0
   351
sl@0
   352
      return impl::template not_found< Target >();
sl@0
   353
    }
sl@0
   354
sl@0
   355
    typedef detail::state_base< allocator_type, rtti_policy_type >
sl@0
   356
      state_base_type;
sl@0
   357
sl@0
   358
    class state_iterator : public std::iterator<
sl@0
   359
      std::forward_iterator_tag,
sl@0
   360
      state_base_type, std::ptrdiff_t
sl@0
   361
      #ifndef BOOST_MSVC_STD_ITERATOR
sl@0
   362
      , const state_base_type *, const state_base_type &
sl@0
   363
      #endif
sl@0
   364
    >
sl@0
   365
    {
sl@0
   366
      public:
sl@0
   367
        //////////////////////////////////////////////////////////////////////
sl@0
   368
        explicit state_iterator(
sl@0
   369
          typename state_base_type::state_list_type::const_iterator 
sl@0
   370
            baseIterator
sl@0
   371
        ) : baseIterator_( baseIterator ) {}
sl@0
   372
sl@0
   373
        const state_base_type & operator*() const { return **baseIterator_; }
sl@0
   374
        const state_base_type * operator->() const
sl@0
   375
        {
sl@0
   376
          return &**baseIterator_;
sl@0
   377
        }
sl@0
   378
sl@0
   379
        state_iterator & operator++() { ++baseIterator_; return *this; }
sl@0
   380
        state_iterator operator++( int )
sl@0
   381
        {
sl@0
   382
          return state_iterator( baseIterator_++ );
sl@0
   383
        }
sl@0
   384
sl@0
   385
        bool operator==( const state_iterator & right ) const
sl@0
   386
        {
sl@0
   387
          return baseIterator_ == right.baseIterator_;
sl@0
   388
        }
sl@0
   389
        bool operator!=( const state_iterator & right ) const
sl@0
   390
        {
sl@0
   391
          return !( *this == right );
sl@0
   392
        }
sl@0
   393
sl@0
   394
      private:
sl@0
   395
        typename state_base_type::state_list_type::const_iterator
sl@0
   396
          baseIterator_;
sl@0
   397
    };
sl@0
   398
sl@0
   399
    state_iterator state_begin() const
sl@0
   400
    {
sl@0
   401
      return state_iterator( currentStates_.begin() );
sl@0
   402
    }
sl@0
   403
sl@0
   404
    state_iterator state_end() const
sl@0
   405
    {
sl@0
   406
      return state_iterator( currentStatesEnd_ );
sl@0
   407
    }
sl@0
   408
sl@0
   409
    void unconsumed_event( const event_base & ) {}
sl@0
   410
sl@0
   411
  protected:
sl@0
   412
    //////////////////////////////////////////////////////////////////////////
sl@0
   413
    state_machine() :
sl@0
   414
      currentStatesEnd_( currentStates_.end() ),
sl@0
   415
      pOutermostState_( 0 ),
sl@0
   416
      isInnermostCommonOuter_( false ),
sl@0
   417
      performFullExit_( true )
sl@0
   418
    {
sl@0
   419
    }
sl@0
   420
sl@0
   421
    // This destructor was only made virtual so that that
sl@0
   422
    // polymorphic_downcast can be used to cast to MostDerived.
sl@0
   423
    virtual ~state_machine()
sl@0
   424
    {
sl@0
   425
      terminate_impl( false );
sl@0
   426
    }
sl@0
   427
sl@0
   428
  public:
sl@0
   429
    //////////////////////////////////////////////////////////////////////////
sl@0
   430
    // The following declarations should be protected.
sl@0
   431
    // They are only public because many compilers lack template friends.
sl@0
   432
    //////////////////////////////////////////////////////////////////////////
sl@0
   433
    void post_event( const event_base_ptr_type & pEvent )
sl@0
   434
    {
sl@0
   435
      BOOST_ASSERT( get_pointer( pEvent ) != 0 );
sl@0
   436
      eventQueue_.push_back( pEvent );
sl@0
   437
    }
sl@0
   438
sl@0
   439
    void post_event( const event_base & evt )
sl@0
   440
    {
sl@0
   441
      post_event( evt.intrusive_from_this() );
sl@0
   442
    }
sl@0
   443
sl@0
   444
  public:
sl@0
   445
    //////////////////////////////////////////////////////////////////////////
sl@0
   446
    // The following declarations should be private.
sl@0
   447
    // They are only public because many compilers lack template friends.
sl@0
   448
    //////////////////////////////////////////////////////////////////////////
sl@0
   449
    typedef MostDerived inner_context_type;
sl@0
   450
    typedef mpl::integral_c< detail::orthogonal_position_type, 0 >
sl@0
   451
      inner_orthogonal_position;
sl@0
   452
    typedef mpl::integral_c< detail::orthogonal_position_type, 1 >
sl@0
   453
      no_of_orthogonal_regions;
sl@0
   454
sl@0
   455
    typedef MostDerived outermost_context_type;
sl@0
   456
    typedef state_machine outermost_context_base_type;
sl@0
   457
    typedef state_machine * inner_context_ptr_type;
sl@0
   458
    typedef typename state_base_type::node_state_base_ptr_type
sl@0
   459
      node_state_base_ptr_type;
sl@0
   460
    typedef typename state_base_type::leaf_state_ptr_type leaf_state_ptr_type;
sl@0
   461
    typedef typename state_base_type::state_list_type state_list_type;
sl@0
   462
sl@0
   463
    typedef mpl::clear< mpl::list<> >::type context_type_list;
sl@0
   464
sl@0
   465
    typedef mpl::bool_< false > shallow_history;
sl@0
   466
    typedef mpl::bool_< false > deep_history;
sl@0
   467
    typedef mpl::bool_< false > inherited_deep_history;
sl@0
   468
sl@0
   469
    detail::reaction_result react_impl(
sl@0
   470
      const event_base_type &,
sl@0
   471
      typename rtti_policy_type::id_type )
sl@0
   472
    {
sl@0
   473
      return detail::do_forward_event;
sl@0
   474
    }
sl@0
   475
sl@0
   476
    void exit_impl(
sl@0
   477
      inner_context_ptr_type &,
sl@0
   478
      typename state_base_type::node_state_base_ptr_type &,
sl@0
   479
      bool ) {}
sl@0
   480
sl@0
   481
    void set_outermost_unstable_state(
sl@0
   482
      typename state_base_type::node_state_base_ptr_type &
sl@0
   483
        pOutermostUnstableState )
sl@0
   484
    {
sl@0
   485
      pOutermostUnstableState = 0;
sl@0
   486
    }
sl@0
   487
sl@0
   488
    // Returns a reference to the context identified by the template
sl@0
   489
    // parameter. This can either be _this_ object or one of its direct or
sl@0
   490
    // indirect contexts.
sl@0
   491
    template< class Context >
sl@0
   492
    Context & context()
sl@0
   493
    {
sl@0
   494
      // As we are in the outermost context here, only this object can be
sl@0
   495
      // returned.
sl@0
   496
      return *polymorphic_downcast< MostDerived * >( this );
sl@0
   497
    }
sl@0
   498
sl@0
   499
    template< class Context >
sl@0
   500
    const Context & context() const
sl@0
   501
    {
sl@0
   502
      // As we are in the outermost context here, only this object can be
sl@0
   503
      // returned.
sl@0
   504
      return *polymorphic_downcast< const MostDerived * >( this );
sl@0
   505
    }
sl@0
   506
sl@0
   507
    outermost_context_type & outermost_context()
sl@0
   508
    {
sl@0
   509
      return *polymorphic_downcast< MostDerived * >( this );
sl@0
   510
    }
sl@0
   511
sl@0
   512
    const outermost_context_type & outermost_context() const
sl@0
   513
    {
sl@0
   514
      return *polymorphic_downcast< const MostDerived * >( this );
sl@0
   515
    }
sl@0
   516
sl@0
   517
    outermost_context_base_type & outermost_context_base()
sl@0
   518
    {
sl@0
   519
      return *this;
sl@0
   520
    }
sl@0
   521
sl@0
   522
    const outermost_context_base_type & outermost_context_base() const
sl@0
   523
    {
sl@0
   524
      return *this;
sl@0
   525
    }
sl@0
   526
sl@0
   527
    void terminate_as_reaction( state_base_type & theState )
sl@0
   528
    {
sl@0
   529
      terminate_impl( theState, performFullExit_ );
sl@0
   530
      pOutermostUnstableState_ = 0;
sl@0
   531
    }
sl@0
   532
sl@0
   533
    void terminate_as_part_of_transit( state_base_type & theState )
sl@0
   534
    {
sl@0
   535
      terminate_impl( theState, performFullExit_ );
sl@0
   536
      isInnermostCommonOuter_ = true;
sl@0
   537
    }
sl@0
   538
sl@0
   539
    void terminate_as_part_of_transit( state_machine & )
sl@0
   540
    {
sl@0
   541
      terminate_impl( *pOutermostState_, performFullExit_ );
sl@0
   542
      isInnermostCommonOuter_ = true;
sl@0
   543
    }
sl@0
   544
sl@0
   545
sl@0
   546
    template< class State >
sl@0
   547
    void add( const intrusive_ptr< State > & pState )
sl@0
   548
    {
sl@0
   549
      // The second dummy argument is necessary because the call to the
sl@0
   550
      // overloaded function add_impl would otherwise be ambiguous.
sl@0
   551
      node_state_base_ptr_type pNewOutermostUnstableStateCandidate =
sl@0
   552
        add_impl( pState, *pState );
sl@0
   553
sl@0
   554
      if ( isInnermostCommonOuter_ ||
sl@0
   555
        is_in_highest_orthogonal_region< State >() &&
sl@0
   556
        ( get_pointer( pOutermostUnstableState_ ) ==
sl@0
   557
          pState->State::outer_state_ptr() ) )
sl@0
   558
      {
sl@0
   559
        isInnermostCommonOuter_ = false;
sl@0
   560
        pOutermostUnstableState_ = pNewOutermostUnstableStateCandidate;
sl@0
   561
      }
sl@0
   562
    }
sl@0
   563
sl@0
   564
sl@0
   565
    void add_inner_state(
sl@0
   566
      detail::orthogonal_position_type position,
sl@0
   567
      state_base_type * pOutermostState )
sl@0
   568
    {
sl@0
   569
      BOOST_ASSERT( position == 0 );
sl@0
   570
      detail::avoid_unused_warning( position );
sl@0
   571
      pOutermostState_ = pOutermostState;
sl@0
   572
    }
sl@0
   573
sl@0
   574
    void remove_inner_state( detail::orthogonal_position_type position )
sl@0
   575
    {
sl@0
   576
      BOOST_ASSERT( position == 0 );
sl@0
   577
      detail::avoid_unused_warning( position );
sl@0
   578
      pOutermostState_ = 0;
sl@0
   579
    }
sl@0
   580
sl@0
   581
sl@0
   582
    void defer_event(
sl@0
   583
      const event_base_type & evt,
sl@0
   584
      const state_base_type * pForState )
sl@0
   585
    {
sl@0
   586
      deferredMap_[ pForState ].push_back( evt.intrusive_from_this() );
sl@0
   587
    }
sl@0
   588
sl@0
   589
    void release_events( const state_base_type * pForState )
sl@0
   590
    {
sl@0
   591
      const typename deferred_map_type::iterator pFound =
sl@0
   592
        deferredMap_.find( pForState );
sl@0
   593
sl@0
   594
      // We are not guaranteed to find an entry because a state is marked for
sl@0
   595
      // having deferred events _before_ the event is actually deferred. An
sl@0
   596
      // exception might be thrown during deferral.
sl@0
   597
      if ( pFound != deferredMap_.end() )
sl@0
   598
      {
sl@0
   599
        eventQueue_.splice( eventQueue_.end(), pFound->second );
sl@0
   600
        deferredMap_.erase( pFound );
sl@0
   601
      }
sl@0
   602
    }
sl@0
   603
sl@0
   604
sl@0
   605
    template< class HistorizedState >
sl@0
   606
    void store_shallow_history()
sl@0
   607
    {
sl@0
   608
      // 5.2.10.6 declares that reinterpret_casting a function pointer to a
sl@0
   609
      // different function pointer and back must yield the same value. The
sl@0
   610
      // following reinterpret_cast is the first half of such a sequence.
sl@0
   611
      store_history_impl(
sl@0
   612
        shallowHistoryMap_,
sl@0
   613
        history_key_type::make_history_key< HistorizedState >(),
sl@0
   614
        reinterpret_cast< void (*)() >( &HistorizedState::deep_construct ) );
sl@0
   615
    }
sl@0
   616
sl@0
   617
    template<
sl@0
   618
      class HistoryContext,
sl@0
   619
      detail::orthogonal_position_type orthogonalPosition >
sl@0
   620
    void clear_shallow_history()
sl@0
   621
    {
sl@0
   622
      // If you receive a
sl@0
   623
      // "use of undefined type 'boost::STATIC_ASSERTION_FAILURE<x>'" or
sl@0
   624
      // similar compiler error here then you tried to clear shallow history
sl@0
   625
      // for a state that does not have shallow history. That is, the state
sl@0
   626
      // does not pass either statechart::has_shallow_history or
sl@0
   627
      // statechart::has_full_history to its base class template.
sl@0
   628
      BOOST_STATIC_ASSERT( HistoryContext::shallow_history::value );
sl@0
   629
sl@0
   630
      typedef typename mpl::at_c<
sl@0
   631
        typename HistoryContext::inner_initial_list,
sl@0
   632
        orthogonalPosition >::type historized_state;
sl@0
   633
sl@0
   634
      store_history_impl(
sl@0
   635
        shallowHistoryMap_,
sl@0
   636
        history_key_type::make_history_key< historized_state >(),
sl@0
   637
        0 );
sl@0
   638
    }
sl@0
   639
sl@0
   640
    template< class DefaultState >
sl@0
   641
    void construct_with_shallow_history(
sl@0
   642
      const typename DefaultState::context_ptr_type & pContext )
sl@0
   643
    {
sl@0
   644
      construct_with_history_impl< DefaultState >(
sl@0
   645
        shallowHistoryMap_, pContext );
sl@0
   646
    }
sl@0
   647
sl@0
   648
sl@0
   649
    template< class HistorizedState, class LeafState >
sl@0
   650
    void store_deep_history()
sl@0
   651
    {
sl@0
   652
      typedef typename detail::make_context_list<
sl@0
   653
        typename HistorizedState::context_type,
sl@0
   654
        LeafState >::type history_context_list;
sl@0
   655
      typedef detail::constructor< 
sl@0
   656
        history_context_list, outermost_context_base_type > constructor_type;
sl@0
   657
      // 5.2.10.6 declares that reinterpret_casting a function pointer to a
sl@0
   658
      // different function pointer and back must yield the same value. The
sl@0
   659
      // following reinterpret_cast is the first half of such a sequence.
sl@0
   660
      store_history_impl(
sl@0
   661
        deepHistoryMap_, 
sl@0
   662
        history_key_type::make_history_key< HistorizedState >(),
sl@0
   663
        reinterpret_cast< void (*)() >( &constructor_type::construct ) );
sl@0
   664
    }
sl@0
   665
sl@0
   666
    template<
sl@0
   667
      class HistoryContext,
sl@0
   668
      detail::orthogonal_position_type orthogonalPosition >
sl@0
   669
    void clear_deep_history()
sl@0
   670
    {
sl@0
   671
      // If you receive a
sl@0
   672
      // "use of undefined type 'boost::STATIC_ASSERTION_FAILURE<x>'" or
sl@0
   673
      // similar compiler error here then you tried to clear deep history for
sl@0
   674
      // a state that does not have deep history. That is, the state does not
sl@0
   675
      // pass either statechart::has_deep_history or
sl@0
   676
      // statechart::has_full_history to its base class template
sl@0
   677
      BOOST_STATIC_ASSERT( HistoryContext::deep_history::value );
sl@0
   678
sl@0
   679
      typedef typename mpl::at_c<
sl@0
   680
        typename HistoryContext::inner_initial_list,
sl@0
   681
        orthogonalPosition >::type historized_state;
sl@0
   682
sl@0
   683
      store_history_impl(
sl@0
   684
        deepHistoryMap_,
sl@0
   685
        history_key_type::make_history_key< historized_state >(),
sl@0
   686
        0 );
sl@0
   687
    }
sl@0
   688
sl@0
   689
    template< class DefaultState >
sl@0
   690
    void construct_with_deep_history(
sl@0
   691
      const typename DefaultState::context_ptr_type & pContext )
sl@0
   692
    {
sl@0
   693
      construct_with_history_impl< DefaultState >(
sl@0
   694
        deepHistoryMap_, pContext );
sl@0
   695
    }
sl@0
   696
sl@0
   697
  private: // implementation
sl@0
   698
    //////////////////////////////////////////////////////////////////////////
sl@0
   699
    void initial_construct()
sl@0
   700
    {
sl@0
   701
      InitialState::initial_deep_construct(
sl@0
   702
        *polymorphic_downcast< MostDerived * >( this ) );
sl@0
   703
    }
sl@0
   704
sl@0
   705
    class initial_construct_function
sl@0
   706
    {
sl@0
   707
      public:
sl@0
   708
        //////////////////////////////////////////////////////////////////////
sl@0
   709
        initial_construct_function( state_machine & machine ) :
sl@0
   710
          machine_( machine )
sl@0
   711
        {
sl@0
   712
        }
sl@0
   713
sl@0
   714
        result operator()()
sl@0
   715
        {
sl@0
   716
          machine_.initial_construct();
sl@0
   717
          return detail::result_utility::make_result(
sl@0
   718
            detail::do_discard_event ); // there is nothing to be consumed
sl@0
   719
        }
sl@0
   720
sl@0
   721
      private:
sl@0
   722
        //////////////////////////////////////////////////////////////////////
sl@0
   723
        state_machine & machine_;
sl@0
   724
    };
sl@0
   725
    friend class initial_construct_function;
sl@0
   726
sl@0
   727
    class terminate_function
sl@0
   728
    {
sl@0
   729
      public:
sl@0
   730
        //////////////////////////////////////////////////////////////////////
sl@0
   731
        terminate_function( state_machine & machine ) : machine_( machine ) {}
sl@0
   732
sl@0
   733
        result operator()()
sl@0
   734
        {
sl@0
   735
          machine_.terminate_impl( true );
sl@0
   736
          return detail::result_utility::make_result(
sl@0
   737
            detail::do_discard_event ); // there is nothing to be consumed
sl@0
   738
        }
sl@0
   739
sl@0
   740
      private:
sl@0
   741
        //////////////////////////////////////////////////////////////////////
sl@0
   742
        state_machine & machine_;
sl@0
   743
    };
sl@0
   744
    friend class terminate_function;
sl@0
   745
sl@0
   746
    template< class ExceptionEvent >
sl@0
   747
    detail::reaction_result handle_exception_event(
sl@0
   748
      const ExceptionEvent & exceptionEvent,
sl@0
   749
      state_base_type * pCurrentState )
sl@0
   750
    {
sl@0
   751
      if ( terminated() )
sl@0
   752
      {
sl@0
   753
        // there is no state that could handle the exception -> bail out
sl@0
   754
        throw;
sl@0
   755
      }
sl@0
   756
sl@0
   757
      // If we are stable, an event handler has thrown.
sl@0
   758
      // Otherwise, either a state constructor, a transition action or an exit
sl@0
   759
      // function has thrown and the state machine is now in an invalid state.
sl@0
   760
      // This situation can be resolved by the exception event handler
sl@0
   761
      // function by orderly transiting to another state or terminating.
sl@0
   762
      // As a result of this, the machine must not be unstable when this
sl@0
   763
      // function is left.
sl@0
   764
      state_base_type * const pOutermostUnstableState =
sl@0
   765
        get_pointer( pOutermostUnstableState_ );
sl@0
   766
      state_base_type * const pHandlingState = pOutermostUnstableState == 0 ?
sl@0
   767
        pCurrentState : pOutermostUnstableState;
sl@0
   768
sl@0
   769
      BOOST_ASSERT( pHandlingState != 0 );
sl@0
   770
sl@0
   771
      // Setting a member variable to a special value for the duration of a
sl@0
   772
      // call surely looks like a kludge (normally it should be a parameter of
sl@0
   773
      // the call). However, in this case it is unavoidable because the call
sl@0
   774
      // below could result in a call to user code where passing through an
sl@0
   775
      // additional bool parameter is not acceptable.
sl@0
   776
      performFullExit_ = false;
sl@0
   777
      const detail::reaction_result reactionResult = pHandlingState->react_impl(
sl@0
   778
        exceptionEvent, exceptionEvent.dynamic_type() );
sl@0
   779
      // If the above call throws then performFullExit_ will obviously not be
sl@0
   780
      // set back to true. In this case the termination triggered by the
sl@0
   781
      // scope guard further up in the call stack will take care of this.
sl@0
   782
      performFullExit_ = true;
sl@0
   783
sl@0
   784
      if ( ( reactionResult != detail::do_discard_event ) ||
sl@0
   785
        ( get_pointer( pOutermostUnstableState_ ) != 0 ) )
sl@0
   786
      {
sl@0
   787
        throw;
sl@0
   788
      }
sl@0
   789
sl@0
   790
      return detail::do_discard_event;
sl@0
   791
    }
sl@0
   792
sl@0
   793
    class exception_event_handler
sl@0
   794
    {
sl@0
   795
      public:
sl@0
   796
        //////////////////////////////////////////////////////////////////////
sl@0
   797
        exception_event_handler(
sl@0
   798
          state_machine & machine,
sl@0
   799
          state_base_type * pCurrentState = 0
sl@0
   800
        ) :
sl@0
   801
          machine_( machine ),
sl@0
   802
          pCurrentState_( pCurrentState )
sl@0
   803
        {
sl@0
   804
        }
sl@0
   805
sl@0
   806
        template< class ExceptionEvent >
sl@0
   807
        result operator()(
sl@0
   808
          const ExceptionEvent & exceptionEvent )
sl@0
   809
        {
sl@0
   810
          return detail::result_utility::make_result(
sl@0
   811
            machine_.handle_exception_event(
sl@0
   812
              exceptionEvent, pCurrentState_ ) );
sl@0
   813
        }
sl@0
   814
sl@0
   815
      private:
sl@0
   816
        //////////////////////////////////////////////////////////////////////
sl@0
   817
        state_machine & machine_;
sl@0
   818
        state_base_type * pCurrentState_;
sl@0
   819
    };
sl@0
   820
    friend class exception_event_handler;
sl@0
   821
sl@0
   822
    class terminator
sl@0
   823
    {
sl@0
   824
      public:
sl@0
   825
        terminator( state_machine & machine ) :
sl@0
   826
          machine_( machine ), dismissed_( false ) {}
sl@0
   827
        ~terminator()
sl@0
   828
        {
sl@0
   829
          if ( !dismissed_ ) { machine_.terminate_impl( false ); }
sl@0
   830
        }
sl@0
   831
        void dismiss() { dismissed_ = true; }
sl@0
   832
sl@0
   833
      private:
sl@0
   834
        state_machine & machine_;
sl@0
   835
        bool dismissed_;
sl@0
   836
    };
sl@0
   837
    friend class terminator;
sl@0
   838
sl@0
   839
sl@0
   840
    void send_event( const event_base_type & evt )
sl@0
   841
    {
sl@0
   842
      terminator guard( *this );
sl@0
   843
      BOOST_ASSERT( get_pointer( pOutermostUnstableState_ ) == 0 );
sl@0
   844
      const typename rtti_policy_type::id_type eventType = evt.dynamic_type();
sl@0
   845
      detail::reaction_result reactionResult = detail::do_forward_event;
sl@0
   846
      
sl@0
   847
      for (
sl@0
   848
        typename state_list_type::iterator pState = currentStates_.begin();
sl@0
   849
        ( reactionResult == detail::do_forward_event ) &&
sl@0
   850
          ( pState != currentStatesEnd_ );
sl@0
   851
        ++pState )
sl@0
   852
      {
sl@0
   853
        // CAUTION: The following statement could modify our state list!
sl@0
   854
        // We must not continue iterating if the event was consumed
sl@0
   855
        reactionResult = detail::result_utility::get_result( translator_(
sl@0
   856
          detail::send_function<
sl@0
   857
            state_base_type, event_base_type, rtti_policy_type::id_type >(
sl@0
   858
              **pState, evt, eventType ),
sl@0
   859
          exception_event_handler( *this, get_pointer( *pState ) ) ) );
sl@0
   860
      }
sl@0
   861
sl@0
   862
      guard.dismiss();
sl@0
   863
sl@0
   864
      if ( reactionResult == detail::do_forward_event )
sl@0
   865
      {
sl@0
   866
        polymorphic_downcast< MostDerived * >( this )->unconsumed_event( evt );
sl@0
   867
      }
sl@0
   868
    }
sl@0
   869
sl@0
   870
sl@0
   871
    void process_queued_events()
sl@0
   872
    {
sl@0
   873
      while ( !eventQueue_.empty() )
sl@0
   874
      {
sl@0
   875
        const event_base_ptr_type pCurrentEvent( eventQueue_.front() );
sl@0
   876
        eventQueue_.pop_front();
sl@0
   877
        send_event( *pCurrentEvent );
sl@0
   878
      }
sl@0
   879
    }
sl@0
   880
sl@0
   881
sl@0
   882
    void terminate_impl( bool performFullExit )
sl@0
   883
    {
sl@0
   884
      performFullExit_ = true;
sl@0
   885
sl@0
   886
      if ( !terminated() )
sl@0
   887
      {
sl@0
   888
        // this also empties deferredMap_
sl@0
   889
        terminate_impl( *pOutermostState_, performFullExit );
sl@0
   890
      }
sl@0
   891
sl@0
   892
      eventQueue_.clear();
sl@0
   893
      shallowHistoryMap_.clear();
sl@0
   894
      deepHistoryMap_.clear();
sl@0
   895
    }
sl@0
   896
sl@0
   897
    void terminate_impl( state_base_type & theState, bool performFullExit )
sl@0
   898
    {
sl@0
   899
      isInnermostCommonOuter_ = false;
sl@0
   900
sl@0
   901
      // If pOutermostUnstableState_ == 0, we know for sure that
sl@0
   902
      // currentStates_.size() > 0, otherwise theState couldn't be alive any
sl@0
   903
      // more
sl@0
   904
      if ( get_pointer( pOutermostUnstableState_ ) != 0 )
sl@0
   905
      {
sl@0
   906
        theState.remove_from_state_list(
sl@0
   907
          currentStatesEnd_, pOutermostUnstableState_, performFullExit );
sl@0
   908
      }
sl@0
   909
      // Optimization: We want to find out whether currentStates_ has size 1
sl@0
   910
      // and if yes use the optimized implementation below. Since
sl@0
   911
      // list<>::size() is implemented quite inefficiently in some std libs
sl@0
   912
      // it is best to just decrement the currentStatesEnd_ here and
sl@0
   913
      // increment it again, if the test failed.
sl@0
   914
      else if ( currentStates_.begin() == --currentStatesEnd_ )
sl@0
   915
      {
sl@0
   916
        // The machine is stable and there is exactly one innermost state.
sl@0
   917
        // The following optimization is only correct for a stable machine
sl@0
   918
        // without orthogonal regions.
sl@0
   919
        leaf_state_ptr_type & pState = *currentStatesEnd_;
sl@0
   920
        pState->exit_impl(
sl@0
   921
          pState, pOutermostUnstableState_, performFullExit );
sl@0
   922
      }
sl@0
   923
      else
sl@0
   924
      {
sl@0
   925
        BOOST_ASSERT( currentStates_.size() > 1 );
sl@0
   926
        // The machine is stable and there are multiple innermost states
sl@0
   927
        theState.remove_from_state_list(
sl@0
   928
          ++currentStatesEnd_, pOutermostUnstableState_, performFullExit );
sl@0
   929
      }
sl@0
   930
    }
sl@0
   931
sl@0
   932
sl@0
   933
    node_state_base_ptr_type add_impl(
sl@0
   934
      const leaf_state_ptr_type & pState,
sl@0
   935
      detail::leaf_state< allocator_type, rtti_policy_type > & )
sl@0
   936
    {
sl@0
   937
      if ( currentStatesEnd_ == currentStates_.end() )
sl@0
   938
      {
sl@0
   939
        pState->set_list_position( 
sl@0
   940
          currentStates_.insert( currentStatesEnd_, pState ) );
sl@0
   941
      }
sl@0
   942
      else
sl@0
   943
      {
sl@0
   944
        *currentStatesEnd_ = pState;
sl@0
   945
        pState->set_list_position( currentStatesEnd_ );
sl@0
   946
        ++currentStatesEnd_;
sl@0
   947
      }
sl@0
   948
sl@0
   949
      return 0;
sl@0
   950
    }
sl@0
   951
sl@0
   952
    node_state_base_ptr_type add_impl(
sl@0
   953
      const node_state_base_ptr_type & pState,
sl@0
   954
      state_base_type & )
sl@0
   955
    {
sl@0
   956
      return pState;
sl@0
   957
    }
sl@0
   958
sl@0
   959
    template< class State >
sl@0
   960
    static bool is_in_highest_orthogonal_region()
sl@0
   961
    {
sl@0
   962
      return mpl::equal_to<
sl@0
   963
        typename State::orthogonal_position,
sl@0
   964
        mpl::minus< 
sl@0
   965
          typename State::context_type::no_of_orthogonal_regions,
sl@0
   966
          mpl::integral_c< detail::orthogonal_position_type, 1 > >
sl@0
   967
      >::value;
sl@0
   968
    }
sl@0
   969
sl@0
   970
sl@0
   971
    typedef detail::history_key< rtti_policy_type > history_key_type;
sl@0
   972
sl@0
   973
    typedef std::map<
sl@0
   974
      history_key_type, void (*)(),
sl@0
   975
      std::less< history_key_type >,
sl@0
   976
      typename boost::detail::allocator::rebind_to<
sl@0
   977
        allocator_type, std::pair< const history_key_type, void (*)() >
sl@0
   978
      >::type
sl@0
   979
    > history_map_type;
sl@0
   980
sl@0
   981
    void store_history_impl(
sl@0
   982
      history_map_type & historyMap,
sl@0
   983
      const history_key_type & historyId,
sl@0
   984
      void (*pConstructFunction)() )
sl@0
   985
    {
sl@0
   986
      historyMap[ historyId ] = pConstructFunction;
sl@0
   987
    }
sl@0
   988
sl@0
   989
    template< class DefaultState >
sl@0
   990
    void construct_with_history_impl(
sl@0
   991
      history_map_type & historyMap,
sl@0
   992
      const typename DefaultState::context_ptr_type & pContext )
sl@0
   993
    {
sl@0
   994
      typename history_map_type::iterator pFoundSlot = historyMap.find(
sl@0
   995
        history_key_type::make_history_key< DefaultState >() );
sl@0
   996
      
sl@0
   997
      if ( ( pFoundSlot == historyMap.end() ) || ( pFoundSlot->second == 0 ) )
sl@0
   998
      {
sl@0
   999
        // We have never entered this state before or history was cleared
sl@0
  1000
        DefaultState::deep_construct(
sl@0
  1001
          pContext, *polymorphic_downcast< MostDerived * >( this ) );
sl@0
  1002
      }
sl@0
  1003
      else
sl@0
  1004
      {
sl@0
  1005
        typedef void construct_function(
sl@0
  1006
          const typename DefaultState::context_ptr_type &,
sl@0
  1007
          typename DefaultState::outermost_context_base_type & );
sl@0
  1008
        // 5.2.10.6 declares that reinterpret_casting a function pointer to a
sl@0
  1009
        // different function pointer and back must yield the same value. The
sl@0
  1010
        // following reinterpret_cast is the second half of such a sequence.
sl@0
  1011
        construct_function * const pConstructFunction =
sl@0
  1012
          reinterpret_cast< construct_function * >( pFoundSlot->second );
sl@0
  1013
        (*pConstructFunction)(
sl@0
  1014
          pContext, *polymorphic_downcast< MostDerived * >( this ) );
sl@0
  1015
      }
sl@0
  1016
    }
sl@0
  1017
sl@0
  1018
    typedef std::list<
sl@0
  1019
      event_base_ptr_type,
sl@0
  1020
      typename boost::detail::allocator::rebind_to<
sl@0
  1021
        allocator_type, event_base_ptr_type >::type
sl@0
  1022
    > event_queue_type;
sl@0
  1023
sl@0
  1024
    typedef std::map<
sl@0
  1025
      const state_base_type *, event_queue_type,
sl@0
  1026
      std::less< const state_base_type * >,
sl@0
  1027
      typename boost::detail::allocator::rebind_to<
sl@0
  1028
        allocator_type,
sl@0
  1029
        std::pair< const state_base_type * const, event_queue_type >
sl@0
  1030
      >::type
sl@0
  1031
    > deferred_map_type;
sl@0
  1032
sl@0
  1033
sl@0
  1034
    event_queue_type eventQueue_;
sl@0
  1035
    deferred_map_type deferredMap_;
sl@0
  1036
    state_list_type currentStates_;
sl@0
  1037
    typename state_list_type::iterator currentStatesEnd_;
sl@0
  1038
    state_base_type * pOutermostState_;
sl@0
  1039
    bool isInnermostCommonOuter_;
sl@0
  1040
    node_state_base_ptr_type pOutermostUnstableState_;
sl@0
  1041
    ExceptionTranslator translator_;
sl@0
  1042
    bool performFullExit_;
sl@0
  1043
    history_map_type shallowHistoryMap_;
sl@0
  1044
    history_map_type deepHistoryMap_;
sl@0
  1045
};
sl@0
  1046
sl@0
  1047
sl@0
  1048
sl@0
  1049
} // namespace statechart
sl@0
  1050
} // namespace boost
sl@0
  1051
sl@0
  1052
sl@0
  1053
sl@0
  1054
#endif