Program Listing for File statemap.hpp

Return to documentation for file (include/smclib/statemap.hpp)

// Copyright (C) 2000 - 2007. Charles W. Rapp.
// All Rights Reserved.
//
// The contents of this file are subject to the Mozilla Public
// License Version 1.1 (the "License"); you may not use this file
// except in compliance with the License. You may obtain a copy
// of the License at http://www.mozilla.org/MPL/
//
// Software distributed under the License is distributed on an
// "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
// implied. See the License for the specific language governing
// rights and limitations under the License.
//
// The Original Code is State Machine Compiler (SMC).
//
// The Initial Developer of the Original Code is Charles W. Rapp.
// Portions created by Charles W. Rapp are
// Copyright (C) 2000 - 2007. Charles W. Rapp.
// All Rights Reserved.
//
// Contributor(s):
//
// Namespace
//  statemap
//
// Description
//  This namespace contains the finite state machine context
//  class. The user can derive FSM contexts from this class and
//  interface to them with the methods of this class.
//
// Notes
//  The finite state machine needs to be initialized to the
//  starting state of the FSM.  This must be done manually in
//  the constructor of the derived class.
//
// Author
//  C. W. Rapp
//
// RCS ID
// $Id: statemap.h,v 1.15 2009/11/24 20:42:39 cwrapp Exp $
//
// CHANGE LOG
// (See bottom of file)
//

#ifndef SMCLIB__STATEMAP_HPP_
#define SMCLIB__STATEMAP_HPP_

#if (__GNUC__ > 3 || (__GNUC__ == 3 && __GNUC_MINOR__ >= 1))
#if defined(SMC_NO_EXCEPTIONS)
#include <cassert>
#endif  // SMC_NO_EXCEPTIONS
#include <cstdio>
#elif defined(WIN32)
#include <windows.h>
#if defined(SMC_NO_EXCEPTIONS)
#include <cassert>
#endif  // SMC_NO_EXCEPTIONS
#else
#if defined(SMC_NO_EXCEPTIONS)
#include <assert.h>
#endif  // SMC_NO_EXCEPTIONS
#include <stdio.h>
#endif
#if !defined(SMC_NO_EXCEPTIONS)
#include <stdexcept>
#include <cstring>
#endif

#include <string>
#include <iostream>
// Limit names to 100 ASCII characters.
// Why 100? Because it is a round number.
#define MAX_NAME_LEN 100

// using namespace std;
namespace statemap
{
// ---------------------------------------------------------------
// Routines.
//

inline char * copyString(const char * s)
{
  char * retval = NULL;
  if (s != NULL) {
    size_t copy_len = strlen(s);
    if (copy_len > MAX_NAME_LEN) {
      copy_len = MAX_NAME_LEN;
    }
    retval = new char[copy_len + 1];
    memcpy(retval, s, copy_len);
    retval[copy_len] = '\0';
  }

  return retval;
}

// ---------------------------------------------------------------
// Exception Classes.
//

#ifndef SMC_NO_EXCEPTIONS
// Base class for all SMC exceptions.
class SmcException
  : public std::runtime_error
{
// -----------------------------------------------------------
// Member methods
//

protected:
  // Constructor.
  explicit SmcException(const std::string & reason)
  : std::runtime_error(reason)
  {}

private:
  // Default construction not allowed.
  SmcException();
};

// This class is thrown when a pop is issued on an empty
// state stack.
class PopOnEmptyStateStackException
  : public SmcException
{
  // -----------------------------------------------------------
  // Member methods.
  //

public:
  // Default constructor.
  PopOnEmptyStateStackException()
  : SmcException("no state to pop from state stack")
  {}
};

// This class is thrown when a transition is issued
// but there is no current state. This happens when
// a transition is issued from within a transition
// action.
class StateUndefinedException
  : public SmcException
{
  // -----------------------------------------------------------
  // Member methods.
  //

public:
  // Default constructor.
  StateUndefinedException()
  : SmcException("transition invoked while in transition")
  {}
};

// This class is thrown when a transition is issued
// but there is no code to handle it.
class TransitionUndefinedException
  : public SmcException
{
  // -----------------------------------------------------------
  // Member methods.
  //

public:
  // Default constructor.
  TransitionUndefinedException()
  : SmcException("no such transition in current state"),
    _state(NULL),
    _transition(NULL)
  {}

  // Construct an exception using the specified state
  // and transition.
  TransitionUndefinedException(
    const char * state,
    const char * transition)
  : SmcException("no such transition in current state"),
    _state(copyString(state)),
    _transition(copyString(transition))
  {}

  // Copy constructor.
  TransitionUndefinedException(
    const TransitionUndefinedException & ex)
  : SmcException("no such transition in current state"),
    _state(copyString(ex._state)),
    _transition(copyString(ex._transition))
  {}

  // Destructor.
  virtual ~TransitionUndefinedException() noexcept
  {
    if (_state != NULL) {
      delete[] _state;
      _state = NULL;
    }

    if (_transition != NULL) {
      delete[] _transition;
      _transition = NULL;
    }
  }

  // Assignment operator.
  const TransitionUndefinedException &
  operator=(const TransitionUndefinedException & ex)
  {
    // Don't do self assignment.
    if (this != &ex) {
      if (_state != NULL) {
        delete[] _state;
        _state = NULL;
      }

      if (_transition != NULL) {
        delete[] _transition;
        _transition = NULL;
      }

      _state = copyString(ex._state);
      _transition = copyString(ex._transition);
    }

    return *this;
  }

  // Returns the state. May be NULL.
  const char * getState() const
  {
    return _state;
  }

  // Returns the transition. May be NULL.
  const char * getTransition() const
  {
    return _transition;
  }

private:
  char * _state;
  char * _transition;
};

// This class is thrown when a state ID is either less than
// the minimal value or greater than the maximal value.
class IndexOutOfBoundsException
  : public SmcException
{
  // -----------------------------------------------------------
  // Member methods.
  //

public:
  // Default constructor.
  IndexOutOfBoundsException()
  : SmcException("index out of bounds"),
    _index(0),
    _minIndex(0),
    _maxIndex(0)
  {}

  // Constructs an exception using the specified index,
  // minimum index and maximum index.
  IndexOutOfBoundsException(
    const int index,
    const int minIndex,
    const int maxIndex)
  : SmcException("index out of bounds"),
    _index(index),
    _minIndex(minIndex),
    _maxIndex(maxIndex)
  {}

  // Copy constructor.
  IndexOutOfBoundsException(
    const IndexOutOfBoundsException & ex)
  : SmcException("index out of bounds"),
    _index(ex._index),
    _minIndex(ex._minIndex),
    _maxIndex(ex._maxIndex)
  {}

  // Destructor.
  virtual ~IndexOutOfBoundsException() noexcept
  {}

  // Assignment operator.
  const IndexOutOfBoundsException &
  operator=(const IndexOutOfBoundsException & ex)
  {
    // Don't do self assignment.
    if (this != &ex) {
      _index = ex._index;
      _minIndex = ex._minIndex;
      _maxIndex = ex._maxIndex;
    }

    return *this;
  }

  // Returns the out-of-bounds index.
  int getIndex() const
  {
    return _index;
  }

  // Returns the minimum allowed index value.
  int getMinIndex() const
  {
    return _minIndex;
  }

  // Returns the maximum allowed index value.
  int getMaxIndex() const
  {
    return _maxIndex;
  }

private:
  int _index;
  int _minIndex;
  int _maxIndex;
};
#endif  // !SMC_NO_EXCEPTIONS

//
// end of Exception Classes.
//---------------------------------------------------------------

class State
{
//-----------------------------------------------------------
// Member functions.
//

public:
  const char * getName() const
  {
    return _name;
  }

  int getId() const
  {
    return _stateId;
  }

protected:
  State(const char * name, int stateId)
  : _name(NULL),
    _stateId(stateId)
  {
    if (name != NULL) {
      _name = copyString(name);
    } else {
      _name = copyString("NAME NOT SET");
    }
  }

  virtual ~State()
  {
    if (_name != NULL) {
      delete[] _name;
      _name = NULL;
    }
  }

private:
  // Make the default and copy constructors private to
  // prevent their use.
  State() {}
  State(const State &) {}

//  -----------------------------------------------------------
// Member data.
//

protected:
  // This state's printable name.
  char * _name;

  // This state's unique identifier.
  int _stateId;
};

class FSMContext
{
//-----------------------------------------------------------
// Nested classes.
//

private:
  // Implements the state stack.
  class StateEntry
  {
    //-------------------------------------------------------
    // Member functions.
    //

public:
    StateEntry(State * state, StateEntry * next)
    : _state(state),
      _next(next)
    {}

    ~StateEntry()
    {
      _state = NULL;
      _next = NULL;
    }

    State * getState()
    {
      return _state;
    }

    StateEntry * getNext()
    {
      return _next;
    }

private:
    State * _state;
    StateEntry * _next;

    //-------------------------------------------------------
    // Friends
    //
    friend class FSMContext;
  };  // end of class StateEntry

// -----------------------------------------------------------
// Member functions
//

public:
  // Destructor.
  virtual ~FSMContext()
  {
    StateEntry * state;

    if (_transition != NULL) {
      delete[] _transition;
      _transition = NULL;
    }

    // But we did allocate the state stack.
    while (_state_stack != NULL) {
      state = _state_stack;
      _state_stack = _state_stack->_next;
      delete state;
    }
  }

  // Comparison and assignment operators
  // Assignment operator
  FSMContext & operator=(const FSMContext & fsm)
  {
    // Don't do the assignment if the left and right
    // hand sides are the same object.
    if (this != &fsm) {
      _state = fsm._state;
    }

    return *this;
  }

  // Starts the finite state machine running by executing
  // the initial state's entry actions.
  virtual void enterStartState() = 0;

  // Exact same object (is it me?)
  int same(const FSMContext & fsm) const
  {
    return this == &fsm;
  }

  // Returns the debug flag's current setting.
  bool getDebugFlag()
  {
    return _debug_flag;
  }

  // Sets the debug flag. A true value means debugging
  // is on and false means off.
  void setDebugFlag(bool flag)
  {
    _debug_flag = flag;
  }

#ifdef SMC_USES_IOSTREAMS
  // Returns the stream to which debug output is written.
  std::ostream & getDebugStream()
  {
    return *_debug_stream;
  }

  // Sets the debug output stream.
  void setDebugStream(std::ostream & debug_stream)
  {
    _debug_stream = &debug_stream;
  }
#endif  // SMC_USES_IOSTREAMS

  // Is this state machine already inside a transition?
  // Yes if state is null.
  bool isInTransition() const
  {
    return _state == NULL ? true : false;
  }

  // Returns the current transition's name.
  // Used only for debugging purposes.
  char * getTransition()
  {
    return _transition;
  }

  // Saves away the transition name only if debugging
  // is turned on.
  void setTransition(const char * transition)
  {
    if (_transition != NULL) {
      delete[] _transition;
      _transition = NULL;
    }

    _transition = copyString(transition);
  }

  // Clears the current state.
  void clearState()
  {
    _previous_state = _state;
    _state = NULL;
  }

  // Returns the state which a transition left.
  // May be NULL.
  State * getPreviousState()
  {
    return _previous_state;
  }

  // Sets the current state to the specified state.
  void setState(const State & state)
  {
    _state = const_cast<State *>(&state);

    if (_debug_flag == true) {
#ifdef SMC_USES_IOSTREAMS
      *_debug_stream << "ENTER STATE     : " <<
        _state->getName() <<
        std::endl;
#else
      TRACE(
        "ENTER STATE     : %s\n\r",
        _state->getName());
#endif  // SMC_USES_IOSTREAMS
    }
  }

  // Returns true if the state stack is empty and false
  // otherwise.
  bool isStateStackEmpty() const
  {
    return _state_stack == NULL;
  }

  // Returns the state stack's depth.
  int getStateStackDepth() const
  {
    StateEntry * state_ptr;
    int retval;

    for (state_ptr = _state_stack, retval = 0;
      state_ptr != NULL;
      state_ptr = state_ptr->getNext(), ++retval) {}

    return retval;
  }

  // Push the current state on top of the state stack
  // and make the specified state the current state.
  void pushState(const State & state)
  {
    StateEntry * new_entry;

    // Do the push only if there is a state to be pushed
    // on the stack.
    if (_state != NULL) {
      new_entry = new StateEntry(_state, _state_stack);
      _state_stack = new_entry;
    }

    _state = const_cast<State *>(&state);

    if (_debug_flag == true) {
#ifdef SMC_USES_IOSTREAMS
      *_debug_stream << "PUSH TO STATE   : " <<
        _state->getName() <<
        std::endl;
#else
      TRACE(
        "PUSH TO STATE   : %s\n\r",
        _state->getName());
#endif  // SMC_USES_IOSTREAMS
    }
  }

  // Make the state on top of the state stack the
  // current state.
  void popState()
  {
    StateEntry * entry;

    // Popping when there was no previous push is an error.
#ifdef SMC_NO_EXCEPTIONS
    assert(_state_stack != NULL);
#else
    if (_state_stack == NULL) {
      throw PopOnEmptyStateStackException();
    }
#endif  // SMC_NO_EXCEPTIONS

    _state = _state_stack->getState();
    entry = _state_stack;
    _state_stack = _state_stack->getNext();
    delete entry;

    if (_debug_flag == true) {
#ifdef SMC_USES_IOSTREAMS
      *_debug_stream << "POP TO STATE    : " <<
        _state->getName() <<
        std::endl;
#else
      TRACE(
        "POP TO STATE    : %s\n\r",
        _state->getName());
#endif  // SMC_USES_IOSTREAMS
    }
  }

  // Remove all states from the state stack.
  void emptyStateStack()
  {
    StateEntry * state_ptr,
      * next_ptr;

    for (state_ptr = _state_stack;
      state_ptr != NULL;
      state_ptr = next_ptr)
    {
      next_ptr = state_ptr->getNext();
      delete state_ptr;
    }

    _state_stack = NULL;
  }

protected:
  // Default constructor.
  explicit FSMContext(const State & state)
  : _state(const_cast<State *>(&state)),
    _previous_state(NULL),
    _state_stack(NULL),
    _transition(NULL),
#ifdef SMC_USES_IOSTREAMS
    _debug_flag(false),
    _debug_stream(&std::cerr)
#else
    _debug_flag(false)
#endif  // SMC_USES_IOSTREAMS
  {}

private:
  // I don't believe that it makes sense to copy a
  // context. It may make sense to copy the application
  // class but the new object is *not* in the same
  // state as the old - the new object must start in
  // the FSM's initial state. Therefore, the copy
  // constructor is private in order to prevent it
  // being used.
  FSMContext(const FSMContext &)
  {}

// -----------------------------------------------------------
// Member data
//

protected:
  // The current state of the finite state machine.
  State * _state;

  // Remember which state a transition left.
  State * _previous_state;

  // The stack of pushed states.
  StateEntry * _state_stack;

  // The current transition *name*. Use for debugging
  // purposes.
  char * _transition;

private:
  // When this flag is set to true, this class will print
  // out debug messages.
  bool _debug_flag;

// Include the following only if C++ iostreams are being used.
#ifdef SMC_USES_IOSTREAMS
  // When FSM debugging is on, debug messages will be
  // written to this output stream. This stream is set to
  // standard error by default.
  std::ostream * _debug_stream;
#endif  // SMC_USES_IOSTREAMS
};  // end of class FSMContext
}  // namespace statemap

//
// CHANGE LOG
// $Log: statemap.h,v $
// Revision 1.15  2009/11/24 20:42:39  cwrapp
// v. 6.0.1 update
//
// Revision 1.14  2009/03/01 18:20:40  cwrapp
// Preliminary v. 6.0.0 commit.
//
// Revision 1.13  2008/05/20 18:31:12  cwrapp
// ----------------------------------------------------------------------
//
// Committing release 5.1.0.
//
// Modified Files:
//  Makefile README.txt smc.mk tar_list.txt bin/Smc.jar
//  examples/Ant/EX1/build.xml examples/Ant/EX2/build.xml
//  examples/Ant/EX3/build.xml examples/Ant/EX4/build.xml
//  examples/Ant/EX5/build.xml examples/Ant/EX6/build.xml
//  examples/Ant/EX7/build.xml examples/Ant/EX7/src/Telephone.java
//  examples/Java/EX1/Makefile examples/Java/EX4/Makefile
//  examples/Java/EX5/Makefile examples/Java/EX6/Makefile
//  examples/Java/EX7/Makefile examples/Ruby/EX1/Makefile
//  lib/statemap.jar lib/C++/statemap.h lib/Java/Makefile
//  lib/Php/statemap.php lib/Scala/Makefile
//  lib/Scala/statemap.scala net/sf/smc/CODE_README.txt
//  net/sf/smc/README.txt net/sf/smc/Smc.java
// ----------------------------------------------------------------------
//
// Revision 1.12  2007/12/28 12:34:40  cwrapp
// Version 5.0.1 check-in.
//
// Revision 1.11  2007/08/05 12:58:54  cwrapp
// Version 5.0.1 check-in. See net/sf/smc/CODE_README.txt for more information.
//
// Revision 1.10  2007/01/15 00:23:50  cwrapp
// Release 4.4.0 initial commit.
//
// Revision 1.9  2006/07/11 18:28:22  cwrapp
// Move SmcException::copyString() to a package-wide routine.
//
// Revision 1.8  2006/04/22 12:45:24  cwrapp
// Version 4.3.1
//
// Revision 1.7  2005/06/08 11:09:14  cwrapp
// + Updated Python code generator to place "pass" in methods with empty
//   bodies.
// + Corrected FSM errors in Python example 7.
// + Removed unnecessary includes from C++ examples.
// + Corrected errors in top-level makefile's distribution build.
//
// Revision 1.6  2005/05/28 18:44:13  cwrapp
// Updated C++, Java and Tcl libraries, added CSharp, Python and VB.
//
// Revision 1.2  2005/02/21 19:01:42  charlesr
// Changed State::_id to State::_stateId because of Object-C++
// reserved word conflict.
//
// Revision 1.1  2004/05/31 13:44:41  charlesr
// Added support for non-iostreams output.
//
// Revision 1.0  2003/12/14 20:37:49  charlesr
// Initial revision

#endif  // SMCLIB__STATEMAP_HPP_