| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201 | // Copyright 2008 Christophe Henry// henry UNDERSCORE christophe AT hotmail DOT com// This is an extended version of the state machine available in the boost::mpl library// Distributed under the same license as the original.// Copyright for the original version:// Copyright 2005 David Abrahams and Aleksey Gurtovoy. Distributed// under the Boost Software License, Version 1.0. (See accompanying// file LICENSE_1_0.txt or copy at// http://www.boost.org/LICENSE_1_0.txt)#ifndef BOOST_MSM_BACK_HISTORY_POLICIES_H#define BOOST_MSM_BACK_HISTORY_POLICIES_H#include <boost/mpl/contains.hpp>namespace boost { namespace msm { namespace back{// policy classes// Default: no history usedtemplate <int NumberOfRegions>class NoHistoryImpl{public:    NoHistoryImpl(){}    ~NoHistoryImpl(){}    void set_initial_states(int* const initial_states)    {        for (int i=0;i<NumberOfRegions;++i)            m_initialStates[i] = initial_states[i];    }    void history_exit(int* const )    {        // ignore    }    // returns the state where the state machine should be at start    template <class Event>    const int* history_entry(Event const& )    {        // always come back to the original state        return m_initialStates;    }    NoHistoryImpl<NumberOfRegions>& operator=(NoHistoryImpl<NumberOfRegions> const& rhs)    {         for (int i=0; i<NumberOfRegions;++i)         {             m_initialStates[i] = rhs.m_initialStates[i];         }         return *this;    }    // this policy deletes all waiting deferred events    template <class Event>    bool process_deferred_events(Event const&)const    {        return false;    }    template<class Archive>    void serialize(Archive & ar, const unsigned int)    {        ar & m_initialStates;    }private:    int m_initialStates[NumberOfRegions];};// not UML standard. Always activates history, no matter which event generated the transitiontemplate <int NumberOfRegions>class AlwaysHistoryImpl{public:    AlwaysHistoryImpl(){}    ~AlwaysHistoryImpl(){}    void set_initial_states(int* const initial_states)    {        for (int i=0;i<NumberOfRegions;++i)            m_initialStates[i] = initial_states[i];    }    void history_exit(int* const current_states)    {        for (int i=0;i<NumberOfRegions;++i)            m_initialStates[i] = current_states[i];    }    // returns the state where the state machine should be at start    template <class Event>    const int* history_entry(Event const& )    {        // always load back the last active state        return m_initialStates;    }    AlwaysHistoryImpl<NumberOfRegions>& operator=(AlwaysHistoryImpl<NumberOfRegions> const& rhs)    {         for (int i=0; i<NumberOfRegions;++i)         {             m_initialStates[i] = rhs.m_initialStates[i];         }         return *this;    }    // the history policy keeps all deferred events until next reentry    template <class Event>    bool process_deferred_events(Event const&)const    {        return true;    }    template<class Archive>    void serialize(Archive & ar, const unsigned int)    {        ar & m_initialStates;    }private:    int m_initialStates[NumberOfRegions];};// UML Shallow history. For deep history, just use this policy for all the contained state machinestemplate <class Events,int NumberOfRegions>class ShallowHistoryImpl{public:    ShallowHistoryImpl(){}    ~ShallowHistoryImpl(){}    void set_initial_states(int* const initial_states)    {        for (int i=0;i<NumberOfRegions;++i)        {            m_currentStates[i] = initial_states[i];            m_initialStates[i] = initial_states[i];        }    }    void history_exit(int* const current_states)    {        for (int i=0;i<NumberOfRegions;++i)            m_currentStates[i] = current_states[i];    }    // returns the state where the state machine should be at start    template <class Event>    const int* history_entry(Event const&)    {        if ( ::boost::mpl::contains<Events,Event>::value)        {            return m_currentStates;        }        // not one of our events, no history        return m_initialStates;    }    ShallowHistoryImpl<Events,NumberOfRegions>& operator=(ShallowHistoryImpl<Events,NumberOfRegions> const& rhs)    {         for (int i=0; i<NumberOfRegions;++i)         {             m_initialStates[i] = rhs.m_initialStates[i];             m_currentStates[i] = rhs.m_currentStates[i];         }         return *this;    }    // the history policy keeps deferred events until next reentry if coming from our history event    template <class Event>    bool process_deferred_events(Event const&)const    {        return ::boost::mpl::contains<Events,Event>::value;    }    template<class Archive>    void serialize(Archive & ar, const unsigned int)    {        ar & m_initialStates;        ar & m_currentStates;    }private:    int m_initialStates[NumberOfRegions];    int m_currentStates[NumberOfRegions];};struct NoHistory{    typedef int history_policy;    template <int NumberOfRegions>    struct apply    {        typedef NoHistoryImpl<NumberOfRegions> type;    };};struct AlwaysHistory{    typedef int history_policy;    template <int NumberOfRegions>    struct apply    {        typedef AlwaysHistoryImpl<NumberOfRegions> type;    };};template <class Events>struct ShallowHistory{    typedef int history_policy;    template <int NumberOfRegions>    struct apply    {        typedef ShallowHistoryImpl<Events,NumberOfRegions> type;    };};} } }//boost::msm::back#endif //BOOST_MSM_BACK_HISTORY_POLICIES_H
 |