sccd.xml 58 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086
  1. <?xml version="1.0" ?>
  2. <diagram author="Simon Van Mierlo/Yentl Van Tendeloo" name="Dynamic Structure DEVS simulator">
  3. <description>
  4. A restricted PythonPDEVS simulator modelled in SCCD for classic, dynamic structure DEVS models.
  5. </description>
  6. <top>
  7. import cPickle as pickle
  8. import time
  9. import copy
  10. # We basically just interface with the basesimulator
  11. from scheduler import Scheduler
  12. from DEVS import directConnect, CoupledDEVS, AtomicDEVS, RootDEVS
  13. from classicDEVSWrapper import ClassicDEVSWrapper
  14. from tracer import trace
  15. class Breakpoint(object):
  16. def __init__(self, breakpoint_id, function, enabled, disable_on_trigger):
  17. self.id = breakpoint_id
  18. self.function = function
  19. self.enabled = enabled
  20. self.disable_on_trigger = disable_on_trigger
  21. </top>
  22. <inport name="request"/>
  23. <outport name="reply"/>
  24. <class name="SCCDSimulator" default="true">
  25. <!-- Define the constructor, taking the model to simulate as a parameter -->
  26. <constructor>
  27. <parameter name="model"/>
  28. <body>
  29. <![CDATA[
  30. self.model = model
  31. self.save_model = pickle.dumps(self.model, pickle.HIGHEST_PROTOCOL)
  32. self.listeners_by_string = {}
  33. self.initialize_simulation()
  34. ]]>
  35. </body>
  36. </constructor>
  37. <method name="initialize_simulation">
  38. <body>
  39. <![CDATA[
  40. # Simulation variables
  41. self.termination_time = None
  42. self.termination_condition = None
  43. self.simulation_time = (0.0, 0)
  44. self.listeners = {}
  45. self.root_model = self.model
  46. self.realtime_scale = 1.0
  47. # Values to be set during simulation
  48. self.realtime_starttime = None
  49. self.inject_queue = []
  50. # Model initialization
  51. self.model_ids = []
  52. self.model.finalize(name="", model_counter=0, model_ids=self.model_ids, locations={None: []}, select_hierarchy=[])
  53. ]]>
  54. </body>
  55. </method>
  56. <!-- Serialize the output values in something human-readable instead of internal objects.
  57. Note that this doesn't alter the internal structure, as that is still used for simulation. -->
  58. <method name="serialize">
  59. <parameter name="type"/>
  60. <parameter name="object"/>
  61. <body>
  62. <![CDATA[
  63. if type == "imminents":
  64. return [m.getModelFullName() for m in object]
  65. elif type == "imminent":
  66. return object.getModelFullName()
  67. elif type == "outbag":
  68. return {m.getPortFullName(): [object[m]] for m in object}
  69. elif type == "inbags":
  70. return {m.getPortFullName(): ([object[m][0].getPortFullName(), object[m][1]] if (object[m][0] is not None) else [object[m][1]]) for m in object}
  71. elif type == "new_tn" or type == "new_states":
  72. return {m.getModelFullName(): object[m] for m in object if m.model_id is not None}
  73. elif type == "transitioning":
  74. return {m.getModelFullName(): {1: "INTERNAL", 2: "EXTERNAL"}[object[m]] for m in object if m.model_id is not None}
  75. elif type == "transitioned":
  76. return {m: {1: "INTERNAL", 2: "EXTERNAL"}[object[m]] for m in object if m.model_id is not None}
  77. ]]>
  78. </body>
  79. </method>
  80. <!-- Find a port based on a fully qualified name. -->
  81. <method name="find_port_with_name">
  82. <parameter name="name"/>
  83. <body>
  84. <![CDATA[
  85. for model in self.model.component_set:
  86. if name.startswith(model.getModelFullName()):
  87. # Found a potential model
  88. # We can't simply check for equality, as portnames might contain dots too
  89. for port in model.IPorts:
  90. if port.getPortFullName() == name:
  91. # Now everything matches
  92. return port
  93. # Nothing found
  94. return None
  95. ]]>
  96. </body>
  97. </method>
  98. <!-- Find a model based on a fully qualified name. -->
  99. <method name="find_model_with_name">
  100. <parameter name="name"/>
  101. <body>
  102. <![CDATA[
  103. for model in self.model.component_set:
  104. if name == model.getModelFullName():
  105. # Found exact match
  106. return model
  107. return None
  108. ]]>
  109. </body>
  110. </method>
  111. <!-- Calculate the time to wait before triggering the next transition.
  112. This method is also called in non-realtime simulation, so make sure that it returns infinity in that case. -->
  113. <method name="calculate_after">
  114. <body>
  115. <![CDATA[
  116. try:
  117. # Process in parts of 100 milliseconds to repeatedly check the termination condition
  118. if self.interrupt_string:
  119. nexttime = 0.0
  120. else:
  121. nexttime = (self.time_next[0] - (time.time() - self.realtime_starttime) / self.realtime_scale) * self.realtime_scale
  122. x = min(0.1, nexttime)
  123. return x
  124. except TypeError, AttributeError:
  125. # We are probably not simulating in realtime...
  126. return float('inf')
  127. ]]>
  128. </body>
  129. </method>
  130. <!-- Parse a list of options that can be passed together with the request -->
  131. <method name="parse_options">
  132. <parameter name="configuration"/>
  133. <body>
  134. <![CDATA[
  135. self.termination_condition = None if "termination_condition" not in configuration else configuration["termination_condition"]
  136. self.termination_time = None if "termination_time" not in configuration else configuration["termination_time"]
  137. self.realtime_scale = 1.0 if "realtime_scale" not in configuration else 1.0/configuration["realtime_scale"]
  138. # Subtract the current simulation time to allow for pausing
  139. self.realtime_starttime = (time.time() - self.simulation_time[0]*self.realtime_scale)
  140. # Reset the time used in the waiting, as it might not get recomputed
  141. self.the_time = 0.00001
  142. ]]>
  143. </body>
  144. </method>
  145. <!-- Utility function to determine whether or not simulation is finished. -->
  146. <method name="should_terminate">
  147. <parameter name="realtime"/>
  148. <body>
  149. <![CDATA[
  150. # Now that it includes breakpoints, results are to be interpretted as follows:
  151. # -2 --> continue simulation
  152. # -1 --> termination condition
  153. # else --> breakpoint
  154. if realtime:
  155. check_time = self.simulation_time
  156. else:
  157. self.compute_timeNext()
  158. check_time = self.time_next
  159. # Just access the 'transitioned' dictionary
  160. # Kind of dirty though...
  161. if self.transitioning is None:
  162. transitioned = set()
  163. else:
  164. transitioned = self.transitioning
  165. outbag = self.outbag if self.outbag is not None else {}
  166. inbags = self.inbags if self.inbags is not None else {}
  167. if check_time[0] == float('inf'):
  168. # Always terminate if we have reached infinity
  169. terminate = True
  170. elif self.termination_condition is not None:
  171. terminate = self.termination_condition(check_time, self.root_model, transitioned)
  172. else:
  173. terminate = self.termination_time < check_time[0]
  174. if terminate:
  175. # Always terminate, so don't check breakpoints
  176. return -1
  177. else:
  178. # Have to check breakpoints for termination
  179. for bp in self.breakpoints:
  180. if not bp.enabled:
  181. continue
  182. # Include the function in the scope
  183. exec(bp.function)
  184. # And execute it, note that the breakpoint thus has to start with "def breakpoint("
  185. # TODO: report to the user if a breakpoint is invalid (catching exceptions).
  186. if breakpoint(check_time, self.root_model, self.serialize('transitioned', transitioned), outbag, inbags):
  187. # Triggered!
  188. return bp.id
  189. else:
  190. # Not triggered, so continue
  191. continue
  192. return -2
  193. ]]>
  194. </body>
  195. </method>
  196. <!-- Compute all models of which the internal transition function has to be triggered. -->
  197. <method name="find_internal_imminents">
  198. <body>
  199. <![CDATA[
  200. self.imminents = self.model.scheduler.getImminent(self.simulation_time)
  201. self.reschedule = set(self.imminents)
  202. self.transition_times.append(self.simulation_time)
  203. for model in self.imminents:
  204. model.time_next = (model.time_next[0], model.time_next[1] + 1)
  205. ]]>
  206. </body>
  207. </method>
  208. <!-- Select one of the imminent components -->
  209. <method name="select_imminent">
  210. <body>
  211. <![CDATA[
  212. self.imminent = None
  213. if len(self.imminents) > 1:
  214. # Perform all selects
  215. self.imminents.sort()
  216. pending = self.imminents
  217. level = 1
  218. while len(pending) > 1:
  219. # Take the model each time, as we need to make sure that the selectHierarchy is valid everywhere
  220. model = pending[0]
  221. # Make a set first to remove duplicates
  222. colliding = list(set([m.select_hierarchy[level] for m in pending]))
  223. chosen = model.select_hierarchy[level-1].select(
  224. sorted(colliding, key=lambda i:i.getModelFullName()))
  225. pending = [m for m in pending
  226. if m.select_hierarchy[level] == chosen]
  227. level += 1
  228. self.imminent = pending[0]
  229. else:
  230. self.imminent = self.imminents[0]
  231. self.imminent.time_next = (self.imminent.time_next[0], self.imminent.time_next[1] - 1)
  232. ]]>
  233. </body>
  234. </method>
  235. <!-- Trigger all output functions of imminent models. -->
  236. <method name="compute_outputfunction">
  237. <body>
  238. <![CDATA[
  239. self.outbag = ClassicDEVSWrapper(self.imminent).outputFnc()
  240. ]]>
  241. </body>
  242. </method>
  243. <!-- Route all messages and apply the required transfer functions. -->
  244. <method name="route_messages">
  245. <body>
  246. <![CDATA[
  247. self.inbags = {}
  248. for outport in self.outbag:
  249. payload = self.outbag[outport]
  250. for inport, z in outport.routing_outline:
  251. if z is not None:
  252. payload = [z(pickle.loads(pickle.dumps(m))) for m in payload]
  253. self.inbags[inport] = [outport, payload]
  254. ]]>
  255. </body>
  256. </method>
  257. <!-- Process the injects on the inject_queue.
  258. Also process realtime interrupts.
  259. TODO: What about events injected on a port which already has an event, or on a time at which a transition was scheduled? Should this be seen as a separate component that needs priority?
  260. TODO: Should rejected injects be communicated to the user? -->
  261. <method name="process_injects">
  262. <body>
  263. <![CDATA[
  264. while self.inject_queue:
  265. if self.inject_queue[0]["time"] > self.simulation_time:
  266. break
  267. config = self.inject_queue.pop(0)
  268. portname = config["port"]
  269. event = config["event"]
  270. port = self.find_port_with_name(portname)
  271. if port == None:
  272. break
  273. self.inbags[port] = [None, [event]]
  274. if self.interrupt_string:
  275. portname, event_value = self.interrupt_string.split(" ")
  276. for inport, z in self.root_model.ports[portname].routing_outline:
  277. ev = event_value if z is None else z(event_value)
  278. self.inbags[inport] = [self.root_model.ports[portname], [ev]]
  279. self.interrupt_string = None
  280. ]]>
  281. </body>
  282. </method>
  283. <!-- Combine the information from the routed messages to determine the external and internal transition functions to trigger. -->
  284. <method name="find_all_imminents">
  285. <body>
  286. <![CDATA[
  287. # Internal codes:
  288. # 1 --> internal transition
  289. # 2 --> external transition
  290. # 3 --> confluent transition
  291. # These codes are a legacy of efficient PyPDEVS, but is kept here for consistency
  292. self.transitioning = {self.imminent: 1}
  293. for inport in self.inbags:
  294. aDEVS = inport.host_DEVS
  295. aDEVS.my_input[inport] = self.inbags[inport][1]
  296. self.transitioning[aDEVS] = 2
  297. self.reschedule.add(aDEVS)
  298. self.transitioning = {ClassicDEVSWrapper(m): self.transitioning[m]
  299. for m in self.transitioning}
  300. for m in self.transitioning:
  301. m.server = self
  302. self.dc_altered = set()
  303. ]]>
  304. </body>
  305. </method>
  306. <!-- Trigger all transition functions. -->
  307. <method name="compute_transitions">
  308. <body>
  309. <![CDATA[
  310. self.new_states = {}
  311. for aDEVS in self.transitioning:
  312. aDEVS.my_input = {key: pickle.loads(pickle.dumps(aDEVS.my_input[key], pickle.HIGHEST_PROTOCOL)) for key in aDEVS.my_input}
  313. if self.transitioning[aDEVS] == 1:
  314. aDEVS.state = aDEVS.intTransition()
  315. elif self.transitioning[aDEVS] == 2:
  316. aDEVS.elapsed = self.simulation_time[0] - aDEVS.time_last[0]
  317. aDEVS.state = aDEVS.extTransition(aDEVS.my_input)
  318. aDEVS.old_states.append((self.simulation_time, pickle.dumps(aDEVS.state)))
  319. aDEVS.my_input = {}
  320. self.new_states[aDEVS] = aDEVS.state
  321. ]]>
  322. </body>
  323. </method>
  324. <!-- Trigger all timeAdvance functions. -->
  325. <method name="compute_ta">
  326. <body>
  327. <![CDATA[
  328. self.new_tn = {}
  329. t, age = self.simulation_time
  330. for aDEVS in self.transitioning:
  331. ta = aDEVS.timeAdvance()
  332. aDEVS.time_last = self.simulation_time
  333. aDEVS.time_next = (t + ta, 1 if ta else (age + 1))
  334. self.new_tn[aDEVS] = aDEVS.time_next
  335. trace(self.trace_file, self.transitioning[aDEVS], aDEVS)
  336. self.model.scheduler.massReschedule(self.reschedule)
  337. self.time_next = self.model.scheduler.readFirst()
  338. ]]>
  339. </body>
  340. </method>
  341. <method name="dsAddPort">
  342. <parameter name="port" />
  343. <body>
  344. <![CDATA[
  345. self.dc_altered.add(port)
  346. self.structural_changes.setdefault('CREATED_PORTS', []).append((port.getPortFullName(), port.is_input))
  347. ]]>
  348. </body>
  349. </method>
  350. <method name="dsRemovePort">
  351. <parameter name="port" />
  352. <body>
  353. <![CDATA[
  354. for iport in port.inline:
  355. iport.outline = [p for p in iport.outline if p != port]
  356. self.dsDisconnectPorts(iport, port);
  357. for oport in port.outline:
  358. oport.inline = [p for p in oport.inline if p != port]
  359. self.dsDisconnectPorts(port, oport);
  360. self.dc_altered.add(port)
  361. self.structural_changes.setdefault('DELETED_PORTS', []).append(port.getPortFullName())
  362. ]]>
  363. </body>
  364. </method>
  365. <method name="dsDisconnectPorts">
  366. <parameter name="p1" />
  367. <parameter name="p2" />
  368. <body>
  369. <![CDATA[
  370. self.dc_altered.add(p1)
  371. self.structural_changes.setdefault('DISCONNECTED_PORTS', []).append((p1.getPortFullName(), p2.getPortFullName()))
  372. ]]>
  373. </body>
  374. </method>
  375. <method name="dsConnectPorts">
  376. <parameter name="p1" />
  377. <parameter name="p2" />
  378. <body>
  379. <![CDATA[
  380. self.dc_altered.add(p1)
  381. self.structural_changes.setdefault('CONNECTED_PORTS', []).append((p1.getPortFullName(), p2.getPortFullName()))
  382. ]]>
  383. </body>
  384. </method>
  385. <method name="dsUnscheduleModel">
  386. <parameter name="model" />
  387. <body>
  388. <![CDATA[
  389. if isinstance(model, CoupledDEVS):
  390. for m in model.component_set:
  391. self.dsUnscheduleModel(m, False)
  392. elif isinstance(model, AtomicDEVS):
  393. self.model.component_set.remove(model)
  394. self.model.models.remove(model)
  395. # The model is removed, so remove it from the scheduler
  396. self.model.scheduler.unschedule(model)
  397. self.model_ids[model.model_id] = None
  398. self.model.local_model_ids.remove(model.model_id)
  399. else:
  400. raise DEVSException("Unknown model to schedule: %s" % model)
  401. for port in model.IPorts:
  402. self.dsRemovePort(port)
  403. for port in model.OPorts:
  404. self.dsRemovePort(port)
  405. self.structural_changes.setdefault('DELETED_MODELS', []).append(model.getModelFullName())
  406. ]]>
  407. </body>
  408. </method>
  409. <method name="dsScheduleModel">
  410. <parameter name="model" />
  411. <body>
  412. <![CDATA[
  413. if isinstance(model, CoupledDEVS):
  414. model.full_name = model.parent.full_name + "." + model.getModelName()
  415. for m in model.component_set:
  416. self.dsScheduleModel(m)
  417. elif isinstance(model, AtomicDEVS):
  418. model.model_id = len(self.model_ids)
  419. model.full_name = model.parent.full_name + "." + model.getModelName()
  420. self.model_ids.append(model)
  421. self.model.component_set.append(model)
  422. self.model.models.append(model)
  423. self.model.local_model_ids.add(model.model_id)
  424. model.time_last = (self.simulation_time[0] - model.elapsed, 1)
  425. ta = model.timeAdvance()
  426. if ta < 0:
  427. raise DEVSException("Negative time advance in atomic model '" + \
  428. model.getModelFullName() + "' with value " + \
  429. str(ta) + " at initialisation")
  430. model.time_next = (model.time_last[0] + ta, 1)
  431. model.old_states.append((self.simulation_time, pickle.dumps(model.state)))
  432. p = model.parent
  433. model.select_hierarchy = [model]
  434. while p != None:
  435. model.select_hierarchy = [p] + model.select_hierarchy
  436. p = p.parent
  437. if model.time_next[0] == self.simulation_time[0]:
  438. model.time_next = (model.time_next[0], self.simulation_time[1])
  439. # If scheduled for 'now', update the age manually
  440. # It is a new model, so add it to the scheduler too
  441. self.model.scheduler.schedule(model)
  442. else:
  443. raise DEVSException("Unknown model to schedule: %s" % model)
  444. self.structural_changes.setdefault('CREATED_MODELS', []).append(model.getModelFullName())
  445. self.structural_changes.setdefault('CREATED_PORTS', []).extend([(port.getPortFullName(), port.is_input) for port in model.ports.itervalues()])
  446. ]]>
  447. </body>
  448. </method>
  449. <method name="getSelfProxy">
  450. <body>
  451. <![CDATA[
  452. return self
  453. ]]>
  454. </body>
  455. </method>
  456. <!-- Process dynamic structure changes. -->
  457. <method name="process_structure_changes">
  458. <body>
  459. <![CDATA[
  460. dsdevs_dict = {}
  461. iterlist = [aDEVS.parent for aDEVS in self.transitioning
  462. if aDEVS.modelTransition(dsdevs_dict)]
  463. # Contains all models that are already checked, to prevent duplicate checking.
  464. # This was not necessary for atomic models, as they are guaranteed to only be called
  465. # once, as they have no children to induce a structural change on them
  466. checked = set()
  467. while iterlist:
  468. new_iterlist = []
  469. for cDEVS in iterlist:
  470. cDEVS.server = self
  471. if cDEVS is None:
  472. # Problematic
  473. #assert warning("Root DEVS returned True in the modelTransition method; ignoring")
  474. continue
  475. if cDEVS in checked:
  476. continue
  477. checked.add(cDEVS)
  478. if cDEVS.modelTransition(dsdevs_dict):
  479. new_iterlist.append(cDEVS.parent)
  480. cDEVS.server = None
  481. # Don't update the iterlist while we are iterating over it
  482. iterlist = new_iterlist
  483. if self.dc_altered:
  484. self.model.redoDirectConnection(self.dc_altered)
  485. for m in self.transitioning:
  486. m.server = None
  487. ]]>
  488. </body>
  489. </method>
  490. <method name="flush_file">
  491. <body>
  492. <![CDATA[
  493. if self.trace_file is not None:
  494. self.trace_file.flush()
  495. ]]>
  496. </body>
  497. </method>
  498. <method name="process_breakpoints">
  499. <parameter name="realtime"/>
  500. <body>
  501. <![CDATA[
  502. breakpoint_id = self.should_terminate(realtime)
  503. for breakpoint in self.breakpoints:
  504. if breakpoint.id == breakpoint_id:
  505. if breakpoint.disable_on_trigger:
  506. breakpoint.enabled = False
  507. return breakpoint_id
  508. ]]>
  509. </body>
  510. </method>
  511. <method name="compute_timeNext">
  512. <body>
  513. <![CDATA[
  514. model_timeNext = self.model.scheduler.readFirst()
  515. if len(self.inject_queue) > 0:
  516. self.time_next = min(model_timeNext, self.inject_queue[0]["time"])
  517. else:
  518. self.time_next = model_timeNext
  519. ]]>
  520. </body>
  521. </method>
  522. <method name="rollback_step">
  523. <body>
  524. <![CDATA[
  525. if len(self.transition_times) == 0:
  526. return
  527. new_time = self.transition_times.pop()
  528. for model in self.model.component_set:
  529. if model.old_states[-1][0] == new_time:
  530. # Remove the current state
  531. del model.old_states[-1]
  532. # Set the new (old...) state
  533. new_state = model.old_states[-1]
  534. model.state = pickle.loads(new_state[1])
  535. model.time_last = new_state[0]
  536. ta = model.timeAdvance()
  537. model.time_next = (model.time_last[0] + ta, model.time_last[1] + 1 if ta == 0 else 0)
  538. self.model.scheduler.massReschedule([model])
  539. self.simulation_time = self.transition_times[-1] if len(self.transition_times) > 0 else (0.0, 0)
  540. ]]>
  541. </body>
  542. </method>
  543. <scxml initial="initializing">
  544. <state id="initializing">
  545. <transition target="../main">
  546. <script>
  547. <![CDATA[
  548. # Direct connection
  549. if isinstance(self.model, CoupledDEVS):
  550. self.model.component_set = directConnect(self.model.component_set, self.listeners)
  551. self.model = RootDEVS(self.model.component_set, self.model.component_set, None)
  552. self.model.listeners = self.listeners
  553. for m in self.model.component_set:
  554. m.time_last = (-m.elapsed, 1)
  555. ta = m.timeAdvance()
  556. m.time_next = (m.time_last[0] + ta, 1)
  557. m.old_states = [(m.time_last, pickle.dumps(m.state))]
  558. elif isinstance(self.model, AtomicDEVS):
  559. for p in self.model.IPorts:
  560. p.routing_inline = []
  561. p.routing_outline = []
  562. for p in self.model.OPorts:
  563. p.routing_inline = []
  564. p.routing_outline = []
  565. self.model = RootDEVS([self.model], [self.model], None)
  566. self.model.time_last = (-self.model.elapsed, 1)
  567. ta = self.model.timeAdvance()
  568. self.model.time_next = (self.model.time_last[0] + ta, 1)
  569. # Fixed configuration options
  570. self.model.scheduler = Scheduler(self.model.component_set, 1e-6, len(self.model.component_set))
  571. self.time_next = self.model.scheduler.readFirst()
  572. # Cached values
  573. self.imminents = None
  574. self.outbag = None
  575. self.inbags = None
  576. self.transitioning = None
  577. self.new_states = None
  578. self.new_tn = None
  579. # Verbose trace file
  580. self.trace_file = None
  581. # Breakpoint management
  582. self.breakpoints = []
  583. # For a reset
  584. # self.save_model = {model: (model.elapsed, pickle.dumps(model.state, pickle.HIGHEST_PROTOCOL)) for model in self.model.component_set}
  585. # self.save_model = pickle.dumps(self.model, pickle.HIGHEST_PROTOCOL)
  586. self.transition_times = []
  587. self.interrupt_string = None
  588. ]]>
  589. </script>
  590. </transition>
  591. </state>
  592. <!-- Parallel states: one of them controls the simulation flow, the other the type of simulation being performed. -->
  593. <parallel id="main">
  594. <!-- When an injection is received, just append it to the list of pending injections.
  595. These will be processed as soon as the current simulation step is finished.
  596. Afterwards, return to the previous state, as there was no change of state. -->
  597. <state id="injection_monitor">
  598. <state id="inject">
  599. <transition port="request" event="inject" target=".">
  600. <parameter name="configuration"/>
  601. <script>
  602. configuration["time"] = (configuration["time"], 1)
  603. self.inject_queue.append(configuration)
  604. self.inject_queue.sort(key=lambda i: i["time"])
  605. </script>
  606. <raise scope="output" port="reply" event="inject_ok"/>
  607. </transition>
  608. </state>
  609. </state>
  610. <state id="tracer_monitor">
  611. <state id="trace">
  612. <transition port="request" event="trace" target=".">
  613. <parameter name="filename"/>
  614. <script>
  615. if filename is not None:
  616. self.trace_file = open(filename, 'w')
  617. else:
  618. self.trace_file = None
  619. </script>
  620. <raise scope="output" port="reply" event="trace_config_ok"/>
  621. </transition>
  622. </state>
  623. </state>
  624. <state id="realtime_interrupt_monitor">
  625. <state id="realtime_interrupt">
  626. <transition target="." port="request" event="realtime_interrupt">
  627. <parameter name="interrupt_string" />
  628. <script>
  629. self.interrupt_string = interrupt_string
  630. </script>
  631. </transition>
  632. </state>
  633. </state>
  634. <!-- The main parallel component: the simulation flow. -->
  635. <state id="simulation_flow" initial="initialize">
  636. <state id="initialize">
  637. <onentry>
  638. <script>
  639. self.structural_changes = {'CREATED_MODELS': [], 'CREATED_PORTS': [], 'CONNECTED_PORTS': []}
  640. queue = [self.root_model]
  641. while queue:
  642. item = queue.pop()
  643. self.structural_changes['CREATED_MODELS'].append(item.getModelFullName())
  644. self.structural_changes['CREATED_PORTS'].extend([(port.getPortFullName(), port.is_input) for port in item.ports.itervalues()])
  645. self.structural_changes['CONNECTED_PORTS'].extend([(outport.getPortFullName(), inport.getPortFullName()) for outport in item.OPorts for inport in outport.outline])
  646. if isinstance(item, CoupledDEVS):
  647. self.structural_changes['CONNECTED_PORTS'].extend([(inport.getPortFullName(), inport2.getPortFullName()) for inport in item.IPorts for inport2 in inport.outline])
  648. queue.extend(item.component_set)
  649. </script>
  650. </onentry>
  651. <transition target="../check_termination">
  652. <raise scope="output" port="reply" event="all_states_reset">
  653. <parameter expr="self.simulation_time"/>
  654. <parameter expr="{m.getModelFullName(): (m.time_next, m.state) for m in self.model.component_set}"/>
  655. <parameter expr="self.structural_changes"/>
  656. <parameter expr="self.breakpoints" />
  657. </raise>
  658. <script>
  659. self.structural_changes = {}
  660. </script>
  661. </transition>
  662. </state>
  663. <state id="check_termination" initial="workaround">
  664. <onexit>
  665. <script>
  666. self.simulation_time = self.time_next
  667. </script>
  668. </onexit>
  669. <state id="workaround">
  670. <transition after="0" target="../check_termination"/>
  671. </state>
  672. <state id="wait">
  673. <onexit>
  674. <script>
  675. diff = time.time() - self.realtime_starttime
  676. self.simulation_time = (diff / self.realtime_scale, 1)
  677. </script>
  678. </onexit>
  679. <transition after="self.calculate_after()" target="../check_termination"/>
  680. <transition cond="INSTATE('../../../simulation_state/paused')" target="../check_termination"/>
  681. </state>
  682. <state id="small_step_check">
  683. <transition cond="self.should_terminate(False) == -2" target="../../do_simulation"/>
  684. <transition cond="self.should_terminate(False) == -1" target="../check_termination">
  685. <raise event="termination_condition"/>
  686. </transition>
  687. <transition cond="self.should_terminate(False) &gt; -1" target="../check_termination">
  688. <script>
  689. breakpoint_id = self.process_breakpoints(False)
  690. </script>
  691. <raise scope="output" port="reply" event="breakpoint_triggered">
  692. <parameter expr="breakpoint_id"/>
  693. <parameter expr="self.serialize('transitioning', self.transitioning)"/>
  694. <parameter expr="self.serialize('outbag', self.outbag)"/>
  695. <parameter expr="self.serialize('inbags', self.inbags)"/>
  696. </raise>
  697. <raise event="termination_condition"/>
  698. </transition>
  699. </state>
  700. <state id="check_termination">
  701. <onentry>
  702. <script>
  703. self.compute_timeNext()
  704. self.the_time = self.calculate_after()
  705. </script>
  706. </onentry>
  707. <!-- Continue simulation -->
  708. <transition cond="INSTATE('../../../simulation_state/continuous') and (self.should_terminate(False) == -2)" target="../../do_simulation"/>
  709. <transition cond="INSTATE('../../../simulation_state/big_step') and (self.should_terminate(False) == -2)" target="../../do_simulation"/>
  710. <!-- Realtime and didn't reach the required time_next yet -->
  711. <transition cond="INSTATE('../../../simulation_state/realtime') and (self.should_terminate(True) == -2) and (self.the_time &gt; 0.0)" target="../wait"/>
  712. <transition cond="INSTATE('../../../simulation_state/realtime') and (self.should_terminate(True) == -2) and (self.the_time &lt;= 0.0)" target="../../do_simulation"/>
  713. <transition port="request" event="small_step" target="../small_step_check">
  714. <parameter name="configuration" type="dict" default="{}"/>
  715. <script>
  716. self.parse_options(configuration)
  717. self.listeners = {getattr(self.root_model, portname): listener for (portname, listener) in self.listeners_by_string.iteritems()}
  718. if not (len(self.listeners) == len(self.model.listeners)):
  719. self.model.listeners = self.listeners
  720. self.model.component_set = directConnect(self.model.models, self.listeners)
  721. </script>
  722. </transition>
  723. <!-- Pause simulation -->
  724. <transition cond="(not INSTATE('../../../simulation_state/paused') and INSTATE('../../../simulation_state/realtime') and (self.should_terminate(True) == -1))" target="../workaround">
  725. <raise event="termination_condition"/>
  726. </transition>
  727. <transition cond="(not INSTATE('../../../simulation_state/paused') and not INSTATE('../../../simulation_state/realtime') and (self.should_terminate(False) == -1))" target="../workaround">
  728. <raise event="termination_condition"/>
  729. </transition>
  730. <transition cond="(not INSTATE('../../../simulation_state/paused')) and INSTATE('../../../simulation_state/realtime') and (self.should_terminate(True) &gt; -1)" target="../workaround">
  731. <script>
  732. breakpoint_id = self.process_breakpoints(True)
  733. </script>
  734. <raise scope="output" port="reply" event="breakpoint_triggered">
  735. <parameter expr="breakpoint_id"/>
  736. <parameter expr="self.serialize('transitioning', self.transitioning)"/>
  737. <parameter expr="self.serialize('outbag', self.outbag)"/>
  738. <parameter expr="self.serialize('inbags', self.inbags)"/>
  739. </raise>
  740. <raise event="termination_condition"/>
  741. </transition>
  742. <transition cond="(not INSTATE('../../../simulation_state/paused')) and not INSTATE('../../../simulation_state/realtime') and (self.should_terminate(False) &gt; -1)" target="../workaround">
  743. <script>
  744. breakpoint_id = self.process_breakpoints(False)
  745. </script>
  746. <raise scope="output" port="reply" event="breakpoint_triggered">
  747. <parameter expr="breakpoint_id"/>
  748. <parameter expr="self.serialize('transitioning', self.transitioning)"/>
  749. <parameter expr="self.serialize('outbag', self.outbag)"/>
  750. <parameter expr="self.serialize('inbags', self.inbags)"/>
  751. </raise>
  752. <raise event="termination_condition"/>
  753. </transition>
  754. <!-- Process god event -->
  755. <transition cond="INSTATE('../../../simulation_state/paused')" port="request" event="god_event" target="../workaround">
  756. <parameter name="configuration"/>
  757. <script>
  758. modelname = configuration["model"]
  759. state_attribute = configuration["attribute"]
  760. new_value = configuration["value"]
  761. model = self.find_model_with_name(modelname)
  762. setattr(model.state, state_attribute, new_value)
  763. # Call the timeadvance method again and compute new ta
  764. ta = model.timeAdvance()
  765. model.time_next = (model.time_last[0] + ta, 1 if ta else (model.time_last[1] + 1))
  766. self.model.scheduler.massReschedule([model])
  767. # self.simulation_time = self.model.scheduler.readFirst()
  768. </script>
  769. <raise scope="output" port="reply" event="god_event_ok">
  770. <parameter expr="{model.getModelFullName(): model.state}"/>
  771. </raise>
  772. <raise scope="output" port="reply" event="new_tn">
  773. <parameter expr="self.simulation_time"/>
  774. <parameter expr="{model.getModelFullName(): model.time_next}"/>
  775. </raise>
  776. </transition>
  777. </state>
  778. </state>
  779. <state id="do_simulation" initial="init">
  780. <state id="init">
  781. <onexit>
  782. <script>
  783. self.find_internal_imminents()
  784. </script>
  785. </onexit>
  786. <transition cond="not INSTATE('../../../simulation_state/paused')" target="../found_internal_imminents"/>
  787. <!-- Always output this if paused, as we only got here because a small step was fired previously -->
  788. <transition cond="INSTATE('../../../simulation_state/paused')" target="../found_internal_imminents">
  789. <raise scope="output" port="reply" event="imminents">
  790. <parameter expr="self.simulation_time"/>
  791. <parameter expr="self.serialize('imminents', self.imminents)"/>
  792. </raise>
  793. </transition>
  794. </state>
  795. <state id="found_internal_imminents">
  796. <onexit>
  797. <script>
  798. self.select_imminent()
  799. </script>
  800. </onexit>
  801. <transition cond="not INSTATE('../../../simulation_state/paused')" target="../selected_imminent"/>
  802. <transition port="request" event="small_step" target="../selected_imminent">
  803. <raise scope="output" port="reply" event="selected_imminent">
  804. <parameter expr="self.simulation_time"/>
  805. <parameter expr="self.serialize('imminent', self.imminent)"/>
  806. </raise>
  807. </transition>
  808. </state>
  809. <state id="selected_imminent">
  810. <onexit>
  811. <script>
  812. self.compute_outputfunction()
  813. </script>
  814. </onexit>
  815. <transition cond="not INSTATE('../../../simulation_state/paused')" target="../computed_outputfunction"/>
  816. <transition port="request" event="small_step" target="../computed_outputfunction">
  817. <raise scope="output" port="reply" event="outbag">
  818. <parameter expr="self.simulation_time"/>
  819. <parameter expr="self.serialize('outbag', self.outbag)"/>
  820. </raise>
  821. </transition>
  822. </state>
  823. <state id="computed_outputfunction">
  824. <onexit>
  825. <script>
  826. self.route_messages()
  827. self.process_injects()
  828. </script>
  829. </onexit>
  830. <transition cond="not INSTATE('../../../simulation_state/paused')" target="../routed_messages">
  831. </transition>
  832. <transition port="request" event="small_step" target="../routed_messages">
  833. <raise scope="output" port="reply" event="inbags">
  834. <parameter expr="self.simulation_time"/>
  835. <parameter expr="self.serialize('inbags', self.inbags)"/>
  836. </raise>
  837. </transition>
  838. </state>
  839. <state id="routed_messages">
  840. <onexit>
  841. <script>
  842. self.find_all_imminents()
  843. </script>
  844. </onexit>
  845. <transition cond="not INSTATE('../../../simulation_state/paused')" target="../found_all_imminents">
  846. </transition>
  847. <transition port="request" event="small_step" target="../found_all_imminents">
  848. <raise scope="output" port="reply" event="transitioning">
  849. <parameter expr="self.simulation_time"/>
  850. <parameter expr="self.serialize('transitioning', self.transitioning)"/>
  851. </raise>
  852. </transition>
  853. </state>
  854. <state id="found_all_imminents">
  855. <onexit>
  856. <script>
  857. self.compute_transitions()
  858. </script>
  859. </onexit>
  860. <transition cond="not INSTATE('../../../simulation_state/paused')" target="../computed_transitions"/>
  861. <transition port="request" event="small_step" target="../computed_transitions">
  862. <raise scope="output" port="reply" event="new_internal_states">
  863. <parameter expr="self.simulation_time"/>
  864. <parameter expr="self.serialize('new_states', self.new_states)"/>
  865. </raise>
  866. </transition>
  867. </state>
  868. <state id="computed_transitions">
  869. <onexit>
  870. <script>
  871. self.compute_ta()
  872. </script>
  873. </onexit>
  874. <transition cond="not INSTATE('../../../simulation_state/paused')" target="../computed_ta"/>
  875. <transition port="request" event="small_step" target="../computed_ta">
  876. <raise scope="output" port="reply" event="new_tn">
  877. <parameter expr="self.simulation_time"/>
  878. <parameter expr="self.serialize('new_tn', self.new_tn)"/>
  879. </raise>
  880. </transition>
  881. </state>
  882. <state id="computed_ta">
  883. <onexit>
  884. <script>
  885. self.process_structure_changes()
  886. </script>
  887. </onexit>
  888. <transition cond="INSTATE('../../../simulation_state/continuous')" target="../../check_termination"/>
  889. <transition port="request" event="small_step" target="../../check_termination">
  890. <raise scope="output" port="reply" event="structural_changes">
  891. <parameter expr="self.simulation_time"/>
  892. <parameter expr="self.structural_changes"/>
  893. </raise>
  894. <script>
  895. self.structural_changes = {}
  896. </script>
  897. </transition>
  898. <transition cond="INSTATE('../../../simulation_state/realtime') or INSTATE('../../../simulation_state/big_step')" target="../../check_termination">
  899. <raise event="big_step_done"/>
  900. <raise scope="output" port="reply" event="new_states">
  901. <parameter expr="self.simulation_time"/>
  902. <parameter expr="self.serialize('new_states', self.new_states)"/>
  903. <parameter expr="self.serialize('new_tn', self.new_tn)"/>
  904. <parameter expr="self.structural_changes"/>
  905. </raise>
  906. <script>
  907. self.structural_changes = {}
  908. </script>
  909. </transition>
  910. </state>
  911. </state>
  912. </state>
  913. <state id="simulation_state" initial="paused">
  914. <state id="paused">
  915. <transition port="request" event="simulate" target="../continuous">
  916. <parameter name="configuration" type="dict" default="{}"/>
  917. <script>
  918. self.parse_options(configuration)
  919. </script>
  920. </transition>
  921. <transition port="request" event="realtime" target="../realtime">
  922. <parameter name="configuration" type="dict" default="{}"/>
  923. <script>
  924. self.parse_options(configuration)
  925. </script>
  926. </transition>
  927. <transition port="request" event="big_step" target="../big_step">
  928. <parameter name="configuration" type="dict" default="{}"/>
  929. <script>
  930. self.parse_options(configuration)
  931. </script>
  932. </transition>
  933. </state>
  934. <state id="continuous">
  935. <onentry>
  936. <script>
  937. self.listeners = {getattr(self.root_model, portname): listener for (portname, listener) in self.listeners_by_string.iteritems()}
  938. self.model.listeners = self.listeners
  939. self.model.component_set = directConnect(self.model.models, self.listeners)
  940. </script>
  941. </onentry>
  942. <transition port="request" event="pause" target=".">
  943. <script>
  944. # Just override termination condition
  945. self.termination_condition = lambda i, j, k : True
  946. self.termination_time = None
  947. </script>
  948. </transition>
  949. <transition event="termination_condition" target="../paused">
  950. <raise port="reply" event="terminate">
  951. <parameter expr="self.simulation_time"/>
  952. </raise>
  953. <script>
  954. self.flush_file()
  955. </script>
  956. <raise scope="output" port="reply" event="all_states">
  957. <parameter expr="self.simulation_time"/>
  958. <parameter expr="{m.getModelFullName(): (m.time_next, m.state) for m in self.model.component_set}"/>
  959. <parameter expr="self.structural_changes"/>
  960. </raise>
  961. <script>
  962. self.structural_changes = {}
  963. </script>
  964. </transition>
  965. </state>
  966. <state id="realtime">
  967. <onentry>
  968. <script>
  969. self.listeners = {getattr(self.root_model, portname): listener for (portname, listener) in self.listeners_by_string.iteritems()}
  970. self.model.listeners = self.listeners
  971. self.model.component_set = directConnect(self.model.models, self.listeners)
  972. </script>
  973. </onentry>
  974. <transition port="request" event="pause" target=".">
  975. <script>
  976. # Just override termination condition
  977. self.termination_condition = lambda i, j, k : True
  978. self.termination_time = None
  979. </script>
  980. </transition>
  981. <transition event="termination_condition" target="../paused">
  982. <raise port="reply" event="terminate">
  983. <parameter expr="self.simulation_time"/>
  984. </raise>
  985. <script>
  986. self.flush_file()
  987. </script>
  988. </transition>
  989. </state>
  990. <state id="big_step">
  991. <onentry>
  992. <script>
  993. self.listeners = {getattr(self.root_model, portname): listener for (portname, listener) in self.listeners_by_string.iteritems()}
  994. self.model.listeners = self.listeners
  995. self.model.component_set = directConnect(self.model.models, self.listeners)
  996. </script>
  997. </onentry>
  998. <transition event="big_step_done" target="../paused"/>
  999. <transition event="termination_condition" target="../paused">
  1000. <raise port="reply" event="terminate">
  1001. <parameter expr="self.simulation_time"/>
  1002. </raise>
  1003. <script>
  1004. self.flush_file()
  1005. </script>
  1006. </transition>
  1007. </state>
  1008. </state>
  1009. <state id="breakpoint_manager">
  1010. <state id="breakpoint_manage">
  1011. <transition port="request" event="add_breakpoint" target=".">
  1012. <parameter name="breakpoint_id"/>
  1013. <parameter name="function"/>
  1014. <parameter name="enabled"/>
  1015. <parameter name="disable_on_trigger"/>
  1016. <script>
  1017. # TODO: report to the user if a breakpoint is invalid (trying to include in scope and reporting any exceptions).
  1018. self.breakpoints.append(Breakpoint(breakpoint_id, function, enabled, disable_on_trigger))
  1019. </script>
  1020. </transition>
  1021. <transition port="request" event="del_breakpoint" target=".">
  1022. <parameter name="del_breakpoint_id"/>
  1023. <script>
  1024. self.breakpoints = [breakpoint for breakpoint in self.breakpoints if breakpoint.id != del_breakpoint_id]
  1025. </script>
  1026. </transition>
  1027. <transition port="request" event="toggle_breakpoint" target=".">
  1028. <parameter name="breakpoint_id"/>
  1029. <parameter name="enabled"/>
  1030. <script>
  1031. for breakpoint in self.breakpoints:
  1032. if breakpoint.id == breakpoint_id:
  1033. breakpoint.enabled = enabled
  1034. break
  1035. </script>
  1036. </transition>
  1037. </state>
  1038. </state>
  1039. <state id="listeners">
  1040. <state id="listening">
  1041. <transition target="." event="set_listen_ports">
  1042. <parameter name="port" />
  1043. <parameter name="function" />
  1044. <script>
  1045. self.listeners_by_string[port] = function
  1046. </script>
  1047. </transition>
  1048. </state>
  1049. </state>
  1050. <state id="reset">
  1051. <state id="reset">
  1052. <transition port="request" event="reset" target="../../../initializing">
  1053. <script>
  1054. self.model = pickle.loads(self.save_model)
  1055. self.initialize_simulation()
  1056. </script>
  1057. </transition>
  1058. </state>
  1059. </state>
  1060. </parallel>
  1061. </scxml>
  1062. </class>
  1063. </diagram>