Przeglądaj źródła

added ELSE condition

Simon Van Mierlo 7 lat temu
rodzic
commit
399bdd9d59

+ 171 - 0
examples/my-tests/python/else_transition.py

@@ -0,0 +1,171 @@
+"""
+Generated by Statechart compiler by Glenn De Jonghe, Joeri Exelmans, Simon Van Mierlo, and Yentl Van Tendeloo (for the inspiration)
+
+Date:   Wed May 30 12:03:02 2018
+
+Model name:   else_transition_test
+
+"""
+
+from sccd.runtime.statecharts_core import *
+
+# package "else_transition_test"
+
+class A(RuntimeClassBase):
+    def __init__(self, controller):
+        RuntimeClassBase.__init__(self, controller)
+        
+        self.semantics.big_step_maximality = StatechartSemantics.TakeMany
+        self.semantics.internal_event_lifeline = StatechartSemantics.Queue
+        self.semantics.input_event_lifeline = StatechartSemantics.FirstComboStep
+        self.semantics.priority = StatechartSemantics.SourceParent
+        self.semantics.concurrency = StatechartSemantics.Single
+        
+        # build Statechart structure
+        self.build_statechart_structure()
+        
+        # call user defined constructor
+        A.user_defined_constructor(self)
+    
+    def user_defined_constructor(self):
+        self.x = 3
+    
+    def user_defined_destructor(self):
+        pass
+    
+    
+    # builds Statechart structure
+    def build_statechart_structure(self):
+        
+        # state <root>
+        self.states[""] = State(0, "", self)
+        
+        # state /main
+        self.states["/main"] = ParallelState(1, "/main", self)
+        
+        # state /main/one
+        self.states["/main/one"] = State(2, "/main/one", self)
+        
+        # state /main/one/A
+        self.states["/main/one/A"] = State(3, "/main/one/A", self)
+        
+        # state /main/one/B
+        self.states["/main/one/B"] = State(4, "/main/one/B", self)
+        self.states["/main/one/B"].setEnter(self._main_one_B_enter)
+        self.states["/main/one/B"].setExit(self._main_one_B_exit)
+        
+        # state /main/one/C
+        self.states["/main/one/C"] = State(5, "/main/one/C", self)
+        self.states["/main/one/C"].setEnter(self._main_one_C_enter)
+        self.states["/main/one/C"].setExit(self._main_one_C_exit)
+        
+        # state /main/one/D
+        self.states["/main/one/D"] = State(6, "/main/one/D", self)
+        self.states["/main/one/D"].setEnter(self._main_one_D_enter)
+        
+        # state /main/two
+        self.states["/main/two"] = State(7, "/main/two", self)
+        
+        # state /main/two/A
+        self.states["/main/two/A"] = State(8, "/main/two/A", self)
+        
+        # state /main/two/B
+        self.states["/main/two/B"] = State(9, "/main/two/B", self)
+        
+        # add children
+        self.states[""].addChild(self.states["/main"])
+        self.states["/main"].addChild(self.states["/main/one"])
+        self.states["/main"].addChild(self.states["/main/two"])
+        self.states["/main/one"].addChild(self.states["/main/one/A"])
+        self.states["/main/one"].addChild(self.states["/main/one/B"])
+        self.states["/main/one"].addChild(self.states["/main/one/C"])
+        self.states["/main/one"].addChild(self.states["/main/one/D"])
+        self.states["/main/two"].addChild(self.states["/main/two/A"])
+        self.states["/main/two"].addChild(self.states["/main/two/B"])
+        self.states[""].fixTree()
+        self.states[""].default_state = self.states["/main"]
+        self.states["/main/one"].default_state = self.states["/main/one/A"]
+        self.states["/main/two"].default_state = self.states["/main/two/A"]
+        
+        # transition /main/one/A
+        _main_one_A_0 = Transition(self, self.states["/main/one/A"], [self.states["/main/one/B"]])
+        _main_one_A_0.setTrigger(Event("a", None))
+        _main_one_A_0.setGuard(self._main_one_A_0_guard)
+        self.states["/main/one/A"].addTransition(_main_one_A_0)
+        _main_one_A_1 = Transition(self, self.states["/main/one/A"], [self.states["/main/one/C"]])
+        _main_one_A_1.setTrigger(Event("a", None))
+        _main_one_A_1.setGuard(self._main_one_A_1_guard)
+        self.states["/main/one/A"].addTransition(_main_one_A_1)
+        _main_one_A_2 = Transition(self, self.states["/main/one/A"], [self.states["/main/one/D"]])
+        _main_one_A_2.setTrigger(None)
+        _main_one_A_2.setGuard(self._main_one_A_2_guard)
+        self.states["/main/one/A"].addTransition(_main_one_A_2)
+        
+        # transition /main/one/B
+        _main_one_B_0 = Transition(self, self.states["/main/one/B"], [self.states["/main/one/A"]])
+        _main_one_B_0.setTrigger(Event("_0after"))
+        self.states["/main/one/B"].addTransition(_main_one_B_0)
+        
+        # transition /main/one/C
+        _main_one_C_0 = Transition(self, self.states["/main/one/C"], [self.states["/main/one/A"]])
+        _main_one_C_0.setTrigger(Event("_1after"))
+        self.states["/main/one/C"].addTransition(_main_one_C_0)
+        
+        # transition /main/two/A
+        _main_two_A_0 = Transition(self, self.states["/main/two/A"], [self.states["/main/two/B"]])
+        _main_two_A_0.setAction(self._main_two_A_0_exec)
+        _main_two_A_0.setTrigger(None)
+        self.states["/main/two/A"].addTransition(_main_two_A_0)
+    
+    def _main_one_B_enter(self):
+        print "in B"
+        self.addTimer(0, 0.05)
+    
+    def _main_one_B_exit(self):
+        self.removeTimer(0)
+    
+    def _main_one_C_enter(self):
+        print "in C"
+        self.addTimer(1, 0.05)
+    
+    def _main_one_C_exit(self):
+        self.removeTimer(1)
+    
+    def _main_one_D_enter(self):
+        print "in D"
+    
+    def _main_one_A_0_guard(self, parameters):
+        return self.x == 5
+    
+    def _main_one_A_1_guard(self, parameters):
+        return "ELSE_GUARD"
+    
+    def _main_one_A_2_guard(self, parameters):
+        return "ELSE_GUARD"
+    
+    def _main_two_A_0_exec(self, parameters):
+        self.raiseInternalEvent(Event("a", None, []))
+    
+    def initializeStatechart(self):
+        # enter default state
+        self.default_targets = self.states["/main"].getEffectiveTargetStates()
+        RuntimeClassBase.initializeStatechart(self)
+
+class ObjectManager(ObjectManagerBase):
+    def __init__(self, controller):
+        ObjectManagerBase.__init__(self, controller)
+    
+    def instantiate(self, class_name, construct_params):
+        if class_name == "A":
+            instance = A(self.controller)
+            instance.associations = {}
+        else:
+            raise Exception("Cannot instantiate class " + class_name)
+        return instance
+
+class Controller(ThreadsControllerBase):
+    def __init__(self, keep_running = None, behind_schedule_callback = None):
+        if keep_running == None: keep_running = True
+        if behind_schedule_callback == None: behind_schedule_callback = None
+        ThreadsControllerBase.__init__(self, ObjectManager(self), keep_running, behind_schedule_callback)
+        self.object_manager.createInstance("A", [])

+ 51 - 0
examples/my-tests/python/else_transition.xml

@@ -0,0 +1,51 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<diagram name="else_transition_test">
+    <class name="A" default="true">
+        <constructor>
+            <body>
+                self.x = 3
+            </body>
+        </constructor>
+        <scxml initial="main">
+            <parallel id="main">
+                <state id="one" initial="A">
+                    <state id="A">
+                        <transition target="../B" event="a" cond="self.x == 5" />
+                        <transition target="../C" event="a" cond="ELSE" />
+                    </state>
+                    <state id="B">
+                        <onentry>
+                            <script>
+                                print "in B"
+                            </script>
+                        </onentry>
+                        <transition target="../A" after="0.05" />
+                    </state>
+                    <state id="C">
+                        <onentry>
+                            <script>
+                                print "in C"
+                            </script>
+                        </onentry>
+                        <transition target="../A" after="0.05" />
+                    </state>
+                    <state id="D">
+                        <onentry>
+                            <script>
+                                print "in D"
+                            </script>
+                        </onentry>
+                    </state>
+                </state>
+                <state id="two" initial="A">
+                    <state id="A">
+                        <transition target="../B">
+                            <raise event="a" />
+                        </transition>
+                    </state>
+                    <state id="B" />
+                </state>
+            </parallel>
+        </scxml>
+    </class>
+</diagram>

+ 3 - 0
examples/my-tests/python/runner_else_transition.py

@@ -0,0 +1,3 @@
+import else_transition
+controller = else_transition.Controller(False)
+controller.start()

+ 6 - 1
src/python_sccd/python_sccd_compiler/generic_generator.py

@@ -515,7 +515,7 @@ class GenericGenerator(Visitor):
             if s.transitions:
                 self.writer.addVSpace()
                 self.writer.addComment("transition %s" % s.new_full_name)
-            for (i, t) in enumerate(s.transitions):
+            for (i, t) in enumerate(s.transitions + s.else_transitions):
                 # instantiate new Transition instance
                 self.writer.addAssignment(
                     GLC.LocalVariableDeclaration(
@@ -773,6 +773,11 @@ class GenericGenerator(Visitor):
                 ]
             )
         )
+        
+    def visit_ElseGuard(self, else_guard):
+        self.writer.add(
+            GLC.String("ELSE_GUARD")
+        )
 
     def visit_Expression(self, expression):
         self.writer.startRecordingExpression()

+ 14 - 3
src/python_sccd/python_sccd_compiler/sccd_constructs.py

@@ -18,6 +18,7 @@ reserved = ["__init__", "__del__", "init", "transition", "microstep", "step", "i
 
 SELF_REFERENCE_SEQ = 'SELF'
 INSTATE_SEQ = 'INSTATE'
+ELSE_SEQ = 'ELSE'
 
 ##################################
 
@@ -42,6 +43,9 @@ class ExpressionPartString(ExpressionPart):
     
 class SelfReference(ExpressionPart):        
     pass
+
+class ElseGuard(ExpressionPart):
+    pass
     
 class InStateCall(ExpressionPart):
     def __init__(self, state_strings):
@@ -72,6 +76,8 @@ class Expression(Visitable):
                     raise CompilerException("Macro \"" + token.val + "\" not allowed here.")
                 elif token.val == SELF_REFERENCE_SEQ :
                     created_object = SelfReference()
+                elif token.val == ELSE_SEQ :
+                    created_object = ElseGuard()
                 elif token.val == INSTATE_SEQ :
                     created_object = self.parseInStateCall()
                     if created_object is None :
@@ -412,7 +418,7 @@ class StateChartTransition(Visitable):
         return self.guard is not None
     
     def getAction(self):
-        return self.action        
+        return self.action
 
 ##################################  
 
@@ -481,12 +487,17 @@ class StateChartNode(Visitable):
         
         #transitions
         self.transitions = []
+        self.else_transitions = []
         for transition_xml in xml_element.findall("transition"):
             transition = StateChartTransition(transition_xml, self)
-            self.transitions.append(transition)
+            if isinstance(transition.guard, ElseGuard):
+                self.else_transitions.append(transition)
+            else:
+                self.transitions.append(transition)
             if transition.trigger.isAfter():
                 self.has_timers = True
-            
+        
+        #TODO: Remove this...
         self.optimizeTransitions()
         self.generateChildren(xml_element)    
         self.calculateDefaults(xml_element)

+ 6 - 5
src/python_sccd/python_sccd_runtime/statecharts_core.py

@@ -16,6 +16,7 @@ from sccd.runtime.event_queue import EventQueue
 from sccd.runtime.accurate_time import AccurateTime
 
 DEBUG = False
+ELSE_GUARD = "ELSE_GUARD"
 
 def print_debug(msg):
     if DEBUG:
@@ -857,13 +858,13 @@ class Transition:
         self.enabled_event = None # the event that enabled this transition
         self.optimize()
     
-    def isEnabled(self, events):
+    def isEnabled(self, events, enabled_transitions):
         if self.trigger is None:
             self.enabled_event = None
-            return (self.guard is None) or self.guard([])
+            return (self.guard is None) or (self.guard == ELSE_GUARD and not enabled_transitions) or self.guard([])
         else:
             for event in events:
-                if (self.trigger.name == event.name and (not self.trigger.port or self.trigger.port == event.port)) and ((self.guard is None) or self.guard(event.parameters)):
+                if (self.trigger.name == event.name and (not self.trigger.port or self.trigger.port == event.port)) and ((self.guard is None) or (self.guard == ELSE_GUARD and not enabled_transitions) or self.guard(event.parameters)):
                     self.enabled_event = event
                     return True
     
@@ -1115,8 +1116,8 @@ class RuntimeClassBase(object):
         enabledEvents = self.getEnabledEvents()
         enabledTransitions = []
         for t in transitions:
-            if t.isEnabled(enabledEvents):
-                enabledTransitions.append(t)
+            if t.isEnabled(enabledEvents, enabledTransitions):
+				enabledTransitions.append(t)
         return enabledTransitions
 
     # @profile