os/ossrv/ossrv_pub/boost_apis/boost/statechart/fifo_scheduler.hpp
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
sl@0
     1
#ifndef BOOST_STATECHART_FIFO_SCHEDULER_HPP_INCLUDED
sl@0
     2
#define BOOST_STATECHART_FIFO_SCHEDULER_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_base.hpp>
sl@0
    12
#include <boost/statechart/fifo_worker.hpp>
sl@0
    13
#include <boost/statechart/processor_container.hpp>
sl@0
    14
sl@0
    15
#include <boost/intrusive_ptr.hpp>
sl@0
    16
#include <boost/noncopyable.hpp>
sl@0
    17
#include <boost/config.hpp> // BOOST_HAS_THREADS
sl@0
    18
sl@0
    19
sl@0
    20
sl@0
    21
namespace boost
sl@0
    22
{
sl@0
    23
namespace statechart
sl@0
    24
{
sl@0
    25
sl@0
    26
sl@0
    27
sl@0
    28
//////////////////////////////////////////////////////////////////////////////
sl@0
    29
template<
sl@0
    30
  class FifoWorker = fifo_worker<>,
sl@0
    31
  class Allocator = std::allocator< void > >
sl@0
    32
class fifo_scheduler : noncopyable
sl@0
    33
{
sl@0
    34
  typedef processor_container<
sl@0
    35
    fifo_scheduler, typename FifoWorker::work_item, Allocator > container;
sl@0
    36
  public:
sl@0
    37
    //////////////////////////////////////////////////////////////////////////
sl@0
    38
    #ifdef BOOST_HAS_THREADS
sl@0
    39
    fifo_scheduler( bool waitOnEmptyQueue = false ) :
sl@0
    40
      worker_( waitOnEmptyQueue )
sl@0
    41
    {
sl@0
    42
    }
sl@0
    43
    #endif
sl@0
    44
sl@0
    45
    typedef typename container::processor_handle processor_handle;
sl@0
    46
    typedef typename container::processor_context processor_context;
sl@0
    47
sl@0
    48
    template< class Processor >
sl@0
    49
    processor_handle create_processor()
sl@0
    50
    {
sl@0
    51
      processor_handle result;
sl@0
    52
      work_item item =
sl@0
    53
        container_.template create_processor< Processor >( result, *this );
sl@0
    54
      worker_.queue_work_item( item );
sl@0
    55
      return result;
sl@0
    56
    }
sl@0
    57
sl@0
    58
    template< class Processor, typename Arg1 >
sl@0
    59
    processor_handle create_processor( Arg1 arg1 )
sl@0
    60
    {
sl@0
    61
      processor_handle result;
sl@0
    62
      work_item item = container_.template create_processor< Processor >(
sl@0
    63
        result, *this, arg1 );
sl@0
    64
      worker_.queue_work_item( item );
sl@0
    65
      return result;
sl@0
    66
    }
sl@0
    67
sl@0
    68
    template< class Processor, typename Arg1, typename Arg2 >
sl@0
    69
    processor_handle create_processor( Arg1 arg1, Arg2 arg2 )
sl@0
    70
    {
sl@0
    71
      processor_handle result;
sl@0
    72
      work_item item = container_.template create_processor< Processor >(
sl@0
    73
        result, *this, arg1, arg2 );
sl@0
    74
      worker_.queue_work_item( item );
sl@0
    75
      return result;
sl@0
    76
    }
sl@0
    77
sl@0
    78
    template< class Processor, typename Arg1, typename Arg2, typename Arg3 >
sl@0
    79
    processor_handle create_processor( Arg1 arg1, Arg2 arg2, Arg3 arg3 )
sl@0
    80
    {
sl@0
    81
      processor_handle result;
sl@0
    82
      work_item item = container_.template create_processor< Processor >(
sl@0
    83
        result, *this, arg1, arg2, arg3 );
sl@0
    84
      worker_.queue_work_item( item );
sl@0
    85
      return result;
sl@0
    86
    }
sl@0
    87
sl@0
    88
    template<
sl@0
    89
      class Processor, typename Arg1, typename Arg2,
sl@0
    90
      typename Arg3, typename Arg4 >
sl@0
    91
    processor_handle create_processor(
sl@0
    92
      Arg1 arg1, Arg2 arg2, Arg3 arg3, Arg4 arg4 )
sl@0
    93
    {
sl@0
    94
      processor_handle result;
sl@0
    95
      work_item item = container_.template create_processor< Processor >(
sl@0
    96
        result, *this, arg1, arg2, arg3, arg4 );
sl@0
    97
      worker_.queue_work_item( item );
sl@0
    98
      return result;
sl@0
    99
    }
sl@0
   100
sl@0
   101
    template<
sl@0
   102
      class Processor, typename Arg1, typename Arg2,
sl@0
   103
      typename Arg3, typename Arg4, typename Arg5 >
sl@0
   104
    processor_handle create_processor(
sl@0
   105
      Arg1 arg1, Arg2 arg2, Arg3 arg3, Arg4 arg4, Arg5 arg5 )
sl@0
   106
    {
sl@0
   107
      processor_handle result;
sl@0
   108
      work_item item = container_.template create_processor< Processor >(
sl@0
   109
        result, *this, arg1, arg2, arg3, arg4, arg5 );
sl@0
   110
      worker_.queue_work_item( item );
sl@0
   111
      return result;
sl@0
   112
    }
sl@0
   113
sl@0
   114
    template<
sl@0
   115
      class Processor, typename Arg1, typename Arg2,
sl@0
   116
      typename Arg3, typename Arg4, typename Arg5, typename Arg6 >
sl@0
   117
    processor_handle create_processor(
sl@0
   118
      Arg1 arg1, Arg2 arg2, Arg3 arg3, Arg4 arg4, Arg5 arg5, Arg6 arg6 )
sl@0
   119
    {
sl@0
   120
      processor_handle result;
sl@0
   121
      work_item item = container_.template create_processor< Processor >(
sl@0
   122
        result, *this, arg1, arg2, arg3, arg4, arg5, arg6 );
sl@0
   123
      worker_.queue_work_item( item );
sl@0
   124
      return result;
sl@0
   125
    }
sl@0
   126
sl@0
   127
    void destroy_processor( const processor_handle & processor )
sl@0
   128
    {
sl@0
   129
      work_item item = container_.destroy_processor( processor );
sl@0
   130
      worker_.queue_work_item( item );
sl@0
   131
    }
sl@0
   132
sl@0
   133
    void initiate_processor( const processor_handle & processor )
sl@0
   134
    {
sl@0
   135
      work_item item = container_.initiate_processor( processor );
sl@0
   136
      worker_.queue_work_item( item );
sl@0
   137
    }
sl@0
   138
sl@0
   139
    void terminate_processor( const processor_handle & processor )
sl@0
   140
    {
sl@0
   141
      work_item item = container_.terminate_processor( processor );
sl@0
   142
      worker_.queue_work_item( item );
sl@0
   143
    }
sl@0
   144
sl@0
   145
    typedef intrusive_ptr< const event_base > event_ptr_type;
sl@0
   146
sl@0
   147
    void queue_event(
sl@0
   148
      const processor_handle & processor, const event_ptr_type & pEvent )
sl@0
   149
    {
sl@0
   150
      work_item item = container_.queue_event( processor, pEvent );
sl@0
   151
      worker_.queue_work_item( item );
sl@0
   152
    }
sl@0
   153
sl@0
   154
    typedef typename FifoWorker::work_item work_item;
sl@0
   155
sl@0
   156
    // We take a non-const reference so that we can move (i.e. swap) the item
sl@0
   157
    // into the queue, what avoids copying the (possibly heap-allocated)
sl@0
   158
    // implementation object inside work_item.
sl@0
   159
    void queue_work_item( work_item & item )
sl@0
   160
    {
sl@0
   161
      worker_.queue_work_item( item );
sl@0
   162
    }
sl@0
   163
sl@0
   164
    // Convenience overload so that temporary objects can be passed directly
sl@0
   165
    // instead of having to create a work_item object first. Under most
sl@0
   166
    // circumstances, this will lead to one unnecessary copy of the
sl@0
   167
    // function implementation object.
sl@0
   168
    void queue_work_item( const work_item & item )
sl@0
   169
    {
sl@0
   170
      worker_.queue_work_item( item );
sl@0
   171
    }
sl@0
   172
sl@0
   173
    void terminate()
sl@0
   174
    {
sl@0
   175
      worker_.terminate();
sl@0
   176
    }
sl@0
   177
sl@0
   178
    // Is not mutex-protected! Must only be called from the thread that also
sl@0
   179
    // calls operator().
sl@0
   180
    bool terminated() const
sl@0
   181
    {
sl@0
   182
      return worker_.terminated();
sl@0
   183
    }
sl@0
   184
sl@0
   185
    unsigned long operator()( unsigned long maxEventCount = 0 )
sl@0
   186
    {
sl@0
   187
      return worker_( maxEventCount );
sl@0
   188
    }
sl@0
   189
sl@0
   190
  private:
sl@0
   191
    //////////////////////////////////////////////////////////////////////////
sl@0
   192
    container container_;
sl@0
   193
    FifoWorker worker_;
sl@0
   194
};
sl@0
   195
sl@0
   196
sl@0
   197
sl@0
   198
} // namespace statechart
sl@0
   199
} // namespace boost
sl@0
   200
sl@0
   201
sl@0
   202
sl@0
   203
#endif