2018-01-12 21:47:58 +01:00

206 lines
4.7 KiB
C++

//////////////////////////////////////////////////////////////////////////////
// Copyright 2002-2006 Andreas Huber Doenni
// Distributed under the Boost Software License, Version 1.0. (See accompany-
// ing file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
//////////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////////
// This is a quick-and-dirty handcrafted state machine with two states and two
// transitions employing GOF-visitation (two virtual calls per event).
// It is used to make speed comparisons with Boost.Statechart machines.
//////////////////////////////////////////////////////////////////////////////
#include <boost/config.hpp>
#include <iostream>
#include <iomanip>
#include <ctime>
#ifdef BOOST_NO_STDC_NAMESPACE
namespace std
{
using ::clock_t;
using ::clock;
}
#endif
#ifdef BOOST_INTEL
# pragma warning( disable: 304 ) // access control not specified
#endif
//////////////////////////////////////////////////////////////////////////////
class EvFlipBit;
class state_base
{
public:
virtual ~state_base() {};
virtual const state_base & react( const EvFlipBit & toEvent ) const = 0;
protected:
state_base() {}
};
template< class Derived >
class state : public state_base
{
public:
state() : state_base() { }
static const Derived & instance()
{
return instance_;
}
private:
static const Derived instance_;
};
template< class Derived >
const Derived state< Derived >::instance_;
//////////////////////////////////////////////////////////////////////////////
class event_base
{
public:
virtual ~event_base() {}
protected:
event_base() {}
public:
virtual const state_base & send( const state_base & toState ) const = 0;
};
template< class Derived >
class event : public event_base
{
protected:
event() {}
private:
virtual const state_base & send( const state_base & toState ) const
{
return toState.react( *static_cast< const Derived * >( this ) );
}
};
//////////////////////////////////////////////////////////////////////////////
class EvFlipBit : public event< EvFlipBit > {
public:
EvFlipBit() : event < EvFlipBit >() { }
};
const EvFlipBit flip;
class BitMachine
{
public:
//////////////////////////////////////////////////////////////////////////
BitMachine() : pCurrentState_( &Off::instance() ) {}
void process_event( const event_base & evt )
{
pCurrentState_ = &evt.send( *pCurrentState_ );
}
private:
//////////////////////////////////////////////////////////////////////////
struct On : state< On >
{
On() : state<On>() { }
virtual const state_base & react( const EvFlipBit & ) const
{
return Off::instance();
}
};
struct Off : state< Off >
{
Off() : state<Off>() { }
virtual const state_base & react( const EvFlipBit & ) const
{
return On::instance();
}
};
const state_base * pCurrentState_;
};
//////////////////////////////////////////////////////////////////////////////
char GetKey()
{
char key;
std::cin >> key;
return key;
}
//////////////////////////////////////////////////////////////////////////////
int main()
{
// common prime factors of 2^n-1 for n in [1,8]
const unsigned int noOfEvents = 3 * 3 * 5 * 7 * 17 * 31 * 127;
unsigned long eventsSentTotal = 0;
std::cout << "Boost.Statechart Handcrafted example\n";
std::cout << "Machine configuration: " << 2 <<
" states interconnected with " << 2 << " transitions.\n\n";
std::cout << "p<CR>: Performance test\n";
std::cout << "e<CR>: Exits the program\n\n";
std::cout <<
"You may chain commands, e.g. pe<CR> performs a test and then exits the program\n\n";
BitMachine bitMachine;
char key = GetKey();
while ( key != 'e' )
{
switch ( key )
{
case 'p':
{
std::cout << "\nSending " << noOfEvents <<
" events. Please wait...\n";
const unsigned long startEvents2 = eventsSentTotal;
const std::clock_t startTime2 = std::clock();
for ( unsigned int eventNo = 0; eventNo < noOfEvents; ++eventNo )
{
bitMachine.process_event( flip );
++eventsSentTotal;
}
const std::clock_t elapsedTime2 = std::clock() - startTime2;
const unsigned int eventsSent2 = eventsSentTotal - startEvents2;
std::cout << "Time to dispatch one event and\n" <<
"perform the resulting transition: ";
std::cout << elapsedTime2 * 1000.0 / eventsSent2 << " microseconds\n\n";
}
break;
default:
{
std::cout << "Invalid key!\n";
}
}
key = GetKey();
}
return 0;
}