Changeset 13834
- Timestamp:
- 05/23/12 17:02:12 (13 years ago)
- Location:
- trunk/FACT++/src
- Files:
-
- 2 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/FACT++/src/StateMachineImp.cc
r13217 r13834 515 515 // -------------------------------------------------------------------------- 516 516 // 517 //! Returns a pointer to a newly allocated object of base EventImp. 518 //! It is meant to be overloaded by derived classes to create their 519 //! own kind of events. 520 //! 521 //! @param targetstate 522 //! Defines the target state of the new transition. If \b must be 523 //! greater or equal zero. A negative target state is used to flag 524 //! commands which do not initiate a state transition. If this is 525 //! desired use AddEvent instead. 517 //! Calling this function, a new (named) event is added to the state 518 //! machine. Via a call to CreateEvent a new event is created with the 519 //! given targetstate, name and format. 520 //! 521 //! The allowed states are passed to the new event and a message 522 //! is written to the output-stream. 526 523 //! 527 524 //! @param name … … 529 526 //! will be constructed with the name given to the constructor and this 530 527 //! name, e.g. "DRIVE/CHANGE_STATE_TO_NEW_STATE" 528 //! 529 //! @param states 530 //! A comma sepeareted list of ints, e.g. "1, 4, 5, 9" with states 531 //! in which this new state transition is allowed and will be accepted. 531 532 //! 532 533 //! @param fmt … … 536 537 //! received commands is properly extracted. No check is done. 537 538 // 538 EventImp *StateMachineImp::CreateEvent(int targetstate, const char *, const char *) 539 { 540 return new EventImp(targetstate); 539 EventImp &StateMachineImp::AddEvent(const char *name, const char *states, const char *fmt) 540 { 541 EventImp *evt = CreateEvent(name, fmt); 542 543 evt->AddAllowedStates(states); 544 545 #ifdef DEBUG 546 Out() << ": " << Time().GetAsStr("%H:%M:%S.%f"); 547 Out() << " - Adding command " << evt->GetName(); 548 Out() << endl; 549 #endif 550 551 fListOfEvents.push_back(evt); 552 553 return *evt; 541 554 } 542 555 … … 544 557 // 545 558 //! Calling this function, a new (named) event is added to the state 546 //! machine. Via a call to CreateEvent a new event is created with the 547 //! given targetstate, name and format. 548 //! 549 //! The allowed states are passed to the new event and a message 550 //! is written to the output-stream. 551 //! 552 //! @param targetstate 553 //! Defines the target state (or name) of the new event. If \b must be 554 //! greater or equal zero. A negative target state is used to flag 555 //! commands which do not initiate a state transition. If this is 556 //! desired use the unnamed version of AddEvent instead. 559 //! machine. Therefore an instance of type DimEvent is created and added 560 //! to the list of available commands fListOfEvents. 557 561 //! 558 562 //! @param name … … 561 565 //! name, e.g. "DRIVE/CHANGE_STATE_TO_NEW_STATE" 562 566 //! 563 //! @param states 564 //! A comma sepeareted list of ints, e.g. "1, 4, 5, 9" with states 565 //! in which this new state transition is allowed and will be accepted. 567 //! @param s1, s2, s3, s4, s5 568 //! A list of states from which a transition to targetstate is allowed 569 //! by this command. 570 // 571 EventImp &StateMachineImp::AddEvent(const char *name, int s1, int s2, int s3, int s4, int s5) 572 { 573 ostringstream str; 574 str << s1 << ' ' << s2 << ' ' << s3 << ' ' << s4 << ' ' << s5; 575 return AddEvent(name, str.str().c_str(), ""); 576 } 577 578 // -------------------------------------------------------------------------- 579 // 580 //! Calling this function, a new (named) event is added to the state 581 //! machine. Therefore an instance of type DimEvent is created and added 582 //! to the list of available commands fListOfEvents. 583 //! 584 //! @param name 585 //! The command name which should initiate the transition. The DimCommand 586 //! will be constructed with the name given to the constructor and this 587 //! name, e.g. "DRIVE/CHANGE_STATE_TO_NEW_STATE" 566 588 //! 567 589 //! @param fmt … … 570 592 //! DimCommand object. However, the user must make sure that the data of 571 593 //! received commands is properly extracted. No check is done. 572 //573 EventImp &StateMachineImp::AddEvent(int targetstate, const char *name, const char *states, const char *fmt)574 {575 EventImp *evt = CreateEvent(targetstate, name, fmt);576 577 evt->AddAllowedStates(states);578 579 #ifdef DEBUG580 Out() << ": " << Time().GetAsStr("%H:%M:%S.%f");581 Out() << " - Adding command " << evt->GetName();582 if (evt->GetTargetState()>=0)583 Out() << " (transition to " << GetStateDescription(evt->GetTargetState()) << ")";584 Out() << endl;585 #endif586 587 fListOfEvents.push_back(evt);588 589 return *evt;590 }591 592 // --------------------------------------------------------------------------593 //594 //! Calling this function, a new (named) event is added to the state595 //! machine. Therefore an instance of type DimEvent is created and added596 //! to the list of available commands fListOfEvents.597 //!598 //! @param targetstate599 //! Defines the target state (or name) of the new event. If \b must be600 //! greater or equal zero. A negative target state is used to flag601 //! commands which do not initiate a state transition. If this is602 //! desired use the unnamed version of AddEvent instead.603 //!604 //! @param name605 //! The command name which should initiate the transition. The DimCommand606 //! will be constructed with the name given to the constructor and this607 //! name, e.g. "DRIVE/CHANGE_STATE_TO_NEW_STATE"608 594 //! 609 595 //! @param s1, s2, s3, s4, s5 … … 611 597 //! by this command. 612 598 // 613 EventImp &StateMachineImp::AddEvent( int targetstate, const char *name, int s1, int s2, int s3, int s4, int s5)599 EventImp &StateMachineImp::AddEvent(const char *name, const char *fmt, int s1, int s2, int s3, int s4, int s5) 614 600 { 615 601 ostringstream str; 616 602 str << s1 << ' ' << s2 << ' ' << s3 << ' ' << s4 << ' ' << s5; 617 return AddEvent(targetstate, name, str.str().c_str(), ""); 618 } 619 620 // -------------------------------------------------------------------------- 621 // 622 //! Calling this function, a new (named) event is added to the state 623 //! machine. Therefore an instance of type DimEvent is created and added 624 //! to the list of available commands fListOfEvents. 625 //! 626 //! @param targetstate 627 //! Defines the target state (or name) of the new event. If \b must be 628 //! greater or equal zero. A negative target state is used to flag 629 //! commands which do not initiate a state transition. If this is 630 //! desired use the unnamed version of AddEvent instead. 631 //! 632 //! @param name 633 //! The command name which should initiate the transition. The DimCommand 634 //! will be constructed with the name given to the constructor and this 635 //! name, e.g. "DRIVE/CHANGE_STATE_TO_NEW_STATE" 636 //! 637 //! @param fmt 638 //! A format as defined by the dim system can be given for the command. 639 //! However, it has no real meaning except that it is stored within the 640 //! DimCommand object. However, the user must make sure that the data of 641 //! received commands is properly extracted. No check is done. 642 //! 643 //! @param s1, s2, s3, s4, s5 644 //! A list of states from which a transition to targetstate is allowed 645 //! by this command. 646 // 647 EventImp &StateMachineImp::AddEvent(int targetstate, const char *name, const char *fmt, int s1, int s2, int s3, int s4, int s5) 648 { 649 ostringstream str; 650 str << s1 << ' ' << s2 << ' ' << s3 << ' ' << s4 << ' ' << s5; 651 return AddEvent(targetstate, name, str.str().c_str(), fmt); 652 } 653 654 // -------------------------------------------------------------------------- 655 // 656 //! This function calls AddEvent with a target-state of -1 (unnamed 657 //! event). This shell be used for configuration commands. As well as 658 //! in AddEvent the states in which such a configuration command is 659 //! accepted can be given. 660 //! 661 //! @param name 662 //! The command name which should initiate the transition. The DimCommand 663 //! will be constructed with the name given to the constructor and this 664 //! name, e.g. "DRIVE/CHANGE_STATE_TO_NEW_STATE" 665 //! 666 //! @param states 667 //! A comma sepeareted list of ints, e.g. "1, 4, 5, 9" with states 668 //! in which this new state transition is allowed and will be accepted. 669 //! 670 //! @param fmt 671 //! A format as defined by the dim system can be given for the command. 672 //! However, it has no real meaning except that it is stored within the 673 //! DimCommand object. However, the user must make sure that the data of 674 //! received commands is properly extracted. No check is done. 675 //! 676 EventImp &StateMachineImp::AddEvent(const char *name, const char *states, const char *fmt) 677 { 678 return AddEvent(-1, name, states, fmt); 679 } 680 681 // -------------------------------------------------------------------------- 682 // 683 //! This function calls AddEvent with a target-state of -1 (unnamed 684 //! event). This shell be used for configuration commands. As well as 685 //! in AddEvent the states in which such a configuration command is 686 //! accepted can be given. 687 //! 688 //! @param name 689 //! The command name which should initiate the transition. The DimCommand 690 //! will be constructed with the name given to the constructor and this 691 //! name, e.g. "DRIVE/CHANGE_STATE_TO_NEW_STATE" 692 //! 693 //! @param s1, s2, s3, s4, s5 694 //! A list of states from which a transition to targetstate is allowed 695 //! by this command. 696 // 697 EventImp &StateMachineImp::AddEvent(const char *name, int s1, int s2, int s3, int s4, int s5) 698 { 699 return AddEvent(-1, name, s1, s2, s3, s4, s5); 700 } 701 702 // -------------------------------------------------------------------------- 703 // 704 //! This function calls AddEvent with a target-state of -1 (unnamed 705 //! event). This shell be used for configuration commands. As well as 706 //! in AddEvent the states in which such a configuration command is 707 //! accepted can be given. 708 //! 709 //! @param name 710 //! The command name which should initiate the transition. The DimCommand 711 //! will be constructed with the name given to the constructor and this 712 //! name, e.g. "DRIVE/CHANGE_STATE_TO_NEW_STATE" 713 //! 714 //! @param fmt 715 //! A format as defined by the dim system can be given for the command. 716 //! However, it has no real meaning except that it is stored within the 717 //! DimCommand object. However, the user must make sure that the data of 718 //! received commands is properly extracted. No check is done. 719 //! 720 //! @param s1, s2, s3, s4, s5 721 //! A list of states from which a transition to targetstate is allowed 722 //! by this command. 723 // 724 EventImp &StateMachineImp::AddEvent(const char *name, const char *fmt, int s1, int s2, int s3, int s4, int s5) 725 { 726 return AddEvent(-1, name, fmt, s1, s2, s3, s4, s5); 603 return AddEvent(name, str.str().c_str(), fmt); 604 } 605 606 EventImp *StateMachineImp::CreateService(const char *) 607 { 608 return new EventImp(); 609 } 610 611 // -------------------------------------------------------------------------- 612 // 613 EventImp &StateMachineImp::Subscribe(const char *name) 614 { 615 EventImp *evt = CreateService(name); 616 fListOfEvents.push_back(evt); 617 return *evt; 727 618 } 728 619 … … 900 791 // 901 792 bool StateMachineImp::HandleNewState(int newstate, const EventImp *evt, 902 const char *txt , const char *alt)793 const char *txt) 903 794 { 904 795 if (newstate==kSM_FatalError) … … 908 799 return true; 909 800 910 if (!evt || !alt || newstate==evt->GetTargetState()) 911 SetCurrentState(newstate, txt, evt ? evt->GetName() : ""); 912 else 913 SetCurrentState(newstate, alt, evt->GetName()); 801 SetCurrentState(newstate, txt, evt ? evt->GetName() : ""); 914 802 915 803 return true; … … 954 842 bool StateMachineImp::HandleEvent(const EventImp &evt) 955 843 { 844 if (!evt.HasFunc()) 845 { 846 Warn(evt.GetName()+": No function assigned... ignored."); 847 return true; 848 849 } 850 851 #ifdef DEBUG 956 852 ostringstream out; 957 853 out << "Handle: " << evt.GetName() << "[" << evt.GetSize() << "]"; 958 854 Debug(out); 959 960 // Get the new state from the command 961 const int commandstate = evt.GetTargetState(); 855 #endif 962 856 963 857 // Check if the received command is allow in the current state 964 858 if (!evt.IsStateAllowed(fCurrentState)) 965 859 { 966 ostringstream msg; 967 msg << evt.GetName() << ": Not allowed in state "; 968 msg << GetStateDescription() << "... ignored."; 969 Warn(msg); 860 Warn(evt.GetName()+": Not allowed in state "+GetStateDescription()+"... ignored."); 970 861 return true; 971 862 } 972 863 973 if (evt.HasFunc()) 974 return HandleNewState(evt.ExecFunc(), &evt, 975 "by ExecFunc function-call"); 976 977 // Check if this is a configuration command (a command which 978 // intention is not to change the state of our state-machine 979 if (commandstate<0) 980 return HandleNewState(Configure(evt), &evt, "by Configure-command"); 981 else 982 return HandleNewState(Transition(evt), &evt, 983 "by Transition-command (expected)", 984 "by Transition-command (unexpected)"); 985 986 // This is a fatal error, because it can never happen 987 return false; 864 return HandleNewState(evt.ExecFunc(), &evt, 865 "by ExecFunc function-call"); 988 866 } 989 867 … … 1095 973 1096 974 // Execute a step in the current state of the state machine 1097 if (!HandleNewState(Execute(), "by Execute-command"))975 if (!HandleNewState(Execute(), 0, "by Execute-command")) 1098 976 break; 1099 977 -
trunk/FACT++/src/StateMachineImp.h
r12956 r13834 51 51 Event *PopEvent(); 52 52 53 bool HandleNewState(int newstate, const EventImp *evt, const char *txt, const char *alt=0); 54 bool HandleNewState(int newstate, const char *txt) 55 { 56 return HandleNewState(newstate, 0, txt); 57 } 58 59 virtual EventImp *CreateEvent(int targetstate, const char *name, const char *fmt); 53 bool HandleNewState(int newstate, const EventImp *evt, const char *txt); 54 55 virtual EventImp *CreateEvent(const char *name, const char *fmt) = 0; 56 virtual EventImp *CreateService(const char *); 60 57 61 58 /// Is called continously to execute actions in the current state 62 59 virtual int Execute() { return fCurrentState; } 63 60 /// Is called when a configuration event is to be processed (no transition of state) 64 virtual int Configure(const Event &) { return kSM_FatalError; }61 //virtual int Configure(const Event &) { return kSM_FatalError; } 65 62 /// Is called when a transition change event is to be processed (from one state to another) is received 66 virtual int Transition(const Event &) { return kSM_FatalError; }63 //virtual int Transition(const Event &) { return kSM_FatalError; } 67 64 68 65 virtual void Lock() { } 69 66 virtual void UnLock() { } 67 68 int Wrapper(const std::function<void(const EventImp &)> &f, const EventImp &imp) 69 { 70 f(imp); 71 return GetCurrentState(); 72 } 70 73 71 74 protected: … … 77 80 virtual std::string SetCurrentState(int state, const char *txt="", const std::string &cmd=""); 78 81 79 EventImp &AddEvent(int targetstate, const char *name, const char *states, const char *fmt);80 EventImp &AddEvent(int targetstate, const char *name, int s1=-1, int s2=-1, int s3=-1, int s4=-1, int s5=-1);81 EventImp &AddEvent(int targetstate, const char *name, const char *fmt, int s1=-1, int s2=-1, int s3=-1, int s4=-1, int s5=-1);82 83 82 EventImp &AddEvent(const char *name, const char *states, const char *fmt); 84 83 EventImp &AddEvent(const char *name, int s1=-1, int s2=-1, int s3=-1, int s4=-1, int s5=-1); … … 93 92 ~StateMachineImp(); 94 93 94 std::function<int(const EventImp &)> Wrap(const std::function<void(const EventImp &)> &func) 95 { 96 return bind(&StateMachineImp::Wrapper, this, func, std::placeholders::_1); 97 } 98 95 99 const std::string &GetName() const { return fName; } 100 101 EventImp &Subscribe(const char *name); 96 102 97 103 /// return the current state of the machine
Note:
See TracChangeset
for help on using the changeset viewer.