mvkcontroller.xml 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378
  1. <class name="MvKController">
  2. <relationships>
  3. <association name="to_mvi" class="Server" min="1" max="1"/>
  4. <association name="tasks" class="Task"/>
  5. </relationships>
  6. <constructor>
  7. <parameter name="params"/>
  8. <body>
  9. <![CDATA[
  10. self.mvs = ModelverseState("../bootstrap/bootstrap.m.gz")
  11. # Enable Garbage Collection
  12. self.mvs.GC = True
  13. self.root = self.mvs.read_root()
  14. # Instantiate the kernel.
  15. self.mvk = ModelverseKernel(self.root)
  16. # Parse kernel-related options
  17. default_kernel_type = 'adaptive-jit'
  18. kernel_opts = [default_kernel_type]
  19. for parameter in params:
  20. if parameter.startswith('--kernel='):
  21. kernel_opts = parameter[len('--kernel='):].split(',')
  22. for opt in kernel_opts:
  23. if opt == 'legacy-interpreter':
  24. self.mvk = LegacyModelverseKernel(self.root)
  25. elif opt == 'interpreter':
  26. self.mvk.jit.set_jit_enabled(False)
  27. elif opt == 'no-thunks':
  28. self.mvk.jit.enable_thunks(False)
  29. elif opt == 'thunks':
  30. self.mvk.jit.enable_thunks()
  31. elif opt == 'no-source-maps':
  32. self.mvk.jit.enable_source_maps(False)
  33. elif opt == 'source-maps':
  34. self.mvk.jit.enable_source_maps()
  35. elif opt == 'no-insert-nops':
  36. self.mvk.jit.enable_nop_insertion(False)
  37. elif opt == 'insert-nops':
  38. self.mvk.jit.enable_nop_insertion()
  39. elif opt == 'trace':
  40. self.mvk.jit.enable_tracing()
  41. elif opt == 'bytecode-interpreter':
  42. self.mvk.jit.set_function_body_compiler(jit.compile_function_body_interpret)
  43. elif opt == 'baseline-jit':
  44. self.mvk.jit.set_function_body_compiler(jit.compile_function_body_baseline)
  45. elif opt == 'fast-jit':
  46. self.mvk.jit.set_function_body_compiler(jit.compile_function_body_fast)
  47. elif opt == 'adaptive-jit-favor-large-functions':
  48. self.mvk.jit.set_function_body_compiler(
  49. lambda *args: jit.compile_function_body_adaptive(
  50. *args, temperature_heuristic=jit.favor_large_functions))
  51. elif opt == 'adaptive-jit-favor-small-functions':
  52. self.mvk.jit.set_function_body_compiler(
  53. lambda *args: jit.compile_function_body_adaptive(
  54. *args, temperature_heuristic=jit.favor_small_functions))
  55. elif opt == 'adaptive-jit-favor-loops':
  56. self.mvk.jit.set_function_body_compiler(
  57. lambda *args: jit.compile_function_body_adaptive(
  58. *args, temperature_heuristic=jit.favor_loops))
  59. elif opt == 'adaptive-jit' or opt == 'adaptive-jit-favor-small-loops':
  60. self.mvk.jit.set_function_body_compiler(
  61. lambda *args: jit.compile_function_body_adaptive(
  62. *args, temperature_heuristic=jit.favor_small_loops))
  63. else:
  64. print("warning: unknown kernel option '%s'." % opt)
  65. self.port = int(sys.argv[1])
  66. self.sc_map = {}
  67. self.forward = None
  68. self.HTTP_reply = None
  69. self.input_queue = defaultdict(list)
  70. self.output_queue = defaultdict(list)
  71. self.mvs_operations = {
  72. "CN": self.mvs.create_node,
  73. "CE": self.mvs.create_edge,
  74. "CNV": self.mvs.create_nodevalue,
  75. "CD": self.mvs.create_dict,
  76. "RV": self.mvs.read_value,
  77. "RO": self.mvs.read_outgoing,
  78. "RI": self.mvs.read_incoming,
  79. "RE": self.mvs.read_edge,
  80. "RD": self.mvs.read_dict,
  81. "RDN": self.mvs.read_dict_node,
  82. "RDNE": self.mvs.read_dict_node_edge,
  83. "RDE": self.mvs.read_dict_edge,
  84. "RRD": self.mvs.read_reverse_dict,
  85. "RR": self.mvs.read_root,
  86. "RDK": self.mvs.read_dict_keys,
  87. "DE": self.mvs.delete_edge,
  88. "DN": self.mvs.delete_node,
  89. }
  90. self.execute_modelverse("", "load_primitives", [])
  91. ]]>
  92. </body>
  93. </constructor>
  94. <method name="execute_modelverse">
  95. <parameter name="taskname"/>
  96. <parameter name="operation"/>
  97. <parameter name="params"/>
  98. <body>
  99. <![CDATA[
  100. reply = None
  101. commands = []
  102. mvk = self.mvk
  103. mvs_operations = self.mvs_operations
  104. try:
  105. while 1:
  106. commands = mvk.execute_yields(taskname, operation, params, reply)
  107. if commands is None:
  108. break
  109. reply = [mvs_operations[command[0]](*(command[1])) for command in commands]
  110. except:
  111. print("ERROR: " + str(self.mvk.debug_info.get(taskname, "Unknown taskname")))
  112. raise
  113. ]]>
  114. </body>
  115. </method>
  116. <scxml initial="init_server">
  117. <state id="init_server">
  118. <onentry>
  119. <raise scope="cd" event="create_instance">
  120. <parameter expr="'to_mvi'"/>
  121. <parameter expr="'Server'"/>
  122. <parameter expr="''"/>
  123. <parameter expr="self.port"/>
  124. </raise>
  125. </onentry>
  126. <transition event="instance_created" target="../running">
  127. <parameter name="instancename"/>
  128. <raise scope="cd" event="start_instance">
  129. <parameter expr="instancename"/>
  130. </raise>
  131. </transition>
  132. </state>
  133. <parallel id="running">
  134. <state id="forward_inputs">
  135. <state id="forward">
  136. <transition cond="set(self.input_queue) &amp; set(self.sc_map)" target=".">
  137. <script>
  138. self.input_task = (set(self.input_queue) &amp; set(self.sc_map)).pop()
  139. value = self.input_queue[self.input_task].pop(0)
  140. if not self.input_queue[self.input_task]:
  141. del self.input_queue[self.input_task]
  142. </script>
  143. <raise event="input" scope="narrow" target="self.sc_map[self.input_task]">
  144. <parameter expr="value"/>
  145. </raise>
  146. </transition>
  147. </state>
  148. </state>
  149. <state id="forward_outputs">
  150. <state id="forward">
  151. <transition cond="set(self.output_queue) &amp; set(self.sc_map)" target=".">
  152. <script>
  153. self.output_task = (set(self.output_queue) &amp; set(self.sc_map)).pop()
  154. value = self.output_queue[self.output_task].pop(0)
  155. if not self.output_queue[self.output_task]:
  156. del self.output_queue[self.output_task]
  157. </script>
  158. <raise event="output" scope="narrow" target="self.sc_map[self.output_task]">
  159. <parameter expr="value"/>
  160. </raise>
  161. </transition>
  162. </state>
  163. </state>
  164. <state id="wait_for_requests">
  165. <state id="wait">
  166. <transition event="from_mvi" target=".">
  167. <parameter name="source"/>
  168. <parameter name="data"/>
  169. <script>
  170. self.source = source
  171. # No JSON encoding necessary, as it is not complex
  172. try:
  173. if data["op"] == "set_input":
  174. if "value" in data:
  175. value = [json.loads(data["value"])]
  176. else:
  177. value = json.loads(data["data"])
  178. self.forward = (data["taskname"], "input", value)
  179. self.HTTP_reply = "OK"
  180. elif data["op"] == "get_output":
  181. self.forward = (data["taskname"], "output", self.source)
  182. self.HTTP_reply = None
  183. elif data["op"] == "pause":
  184. #TODO
  185. self.HTTP_reply = "NotImplemented"
  186. elif data["op"] == "resume":
  187. #TODO
  188. self.HTTP_reply = "NotImplemented"
  189. else:
  190. self.HTTP_reply = "Unknown command: %s" % data["op"]
  191. except ValueError as e:
  192. self.HTTP_reply = "Error when deserializing request: %s" % traceback.format_exc()
  193. except Exception as e:
  194. self.HTTP_reply = "Unknown Modelverse Error: %s" % traceback.format_exc()
  195. </script>
  196. </transition>
  197. <transition cond="not self.forward and self.HTTP_reply is not None" target=".">
  198. <raise event="HTTP_input" scope="narrow" target="'to_mvi/%s' % self.source">
  199. <parameter expr="json.dumps(self.HTTP_reply)"/>
  200. </raise>
  201. <script>
  202. self.HTTP_reply = None
  203. self.forward = None
  204. </script>
  205. </transition>
  206. <transition cond="self.forward and self.forward[1] == 'input' and self.forward[0] in self.sc_map" target=".">
  207. <raise event="input" scope="narrow" target="self.sc_map[self.forward[0]]">
  208. <parameter expr="self.forward[2]"/>
  209. </raise>
  210. <raise event="HTTP_input" scope="narrow" target="'to_mvi/%s' % self.source">
  211. <parameter expr="json.dumps(self.HTTP_reply)"/>
  212. </raise>
  213. <script>
  214. self.HTTP_reply = None
  215. self.forward = None
  216. </script>
  217. </transition>
  218. <transition cond="self.forward and self.forward[1] == 'input' and self.forward[0] not in self.sc_map" target=".">
  219. <raise event="HTTP_input" scope="narrow" target="'to_mvi/%s' % self.source">
  220. <parameter expr="json.dumps(self.HTTP_reply)"/>
  221. </raise>
  222. <script>
  223. self.input_queue[self.forward[0]].append(self.forward[2])
  224. self.HTTP_reply = None
  225. self.forward = None
  226. </script>
  227. </transition>
  228. <transition cond="self.forward and self.forward[1] == 'output' and self.forward[0] in self.sc_map" target=".">
  229. <raise event="output" scope="narrow" target="self.sc_map[self.forward[0]]">
  230. <parameter expr="self.forward[2]"/>
  231. </raise>
  232. <script>
  233. self.HTTP_reply = None
  234. self.forward = None
  235. </script>
  236. </transition>
  237. <transition cond="self.forward and self.forward[1] == 'output' and self.forward[0] not in self.sc_map" target=".">
  238. <script>
  239. self.output_queue[self.forward[0]].append(self.forward[2])
  240. self.HTTP_reply = None
  241. self.forward = None
  242. </script>
  243. </transition>
  244. </state>
  245. </state>
  246. <parallel id="execution">
  247. <state id="refresh_tasks" initial="refresh_tasks">
  248. <state id="refresh_tasks">
  249. <onentry>
  250. <script>
  251. out = self.mvs.read_outgoing(self.root)
  252. tasks = set()
  253. for m in out:
  254. src, task = self.mvs.read_edge(m)
  255. outgoing = self.mvs.read_outgoing(m)
  256. first = self.mvs.read_edge(outgoing[0])
  257. dest = first[1]
  258. name = self.mvs.read_value(dest)
  259. if name.startswith("__"):
  260. continue
  261. tasks.add(name)
  262. self.new_tasks = set([i for i in tasks if i not in self.sc_map])
  263. self.old_tasks = set([i for i in self.sc_map if i not in tasks])
  264. </script>
  265. </onentry>
  266. <transition cond="self.new_tasks" target="../create_SC"/>
  267. <transition cond="not self.new_tasks and self.old_tasks" target="../remove_SC"/>
  268. <transition after="1" target="."/>
  269. </state>
  270. <state id="create_SC" initial="creating">
  271. <state id="creating">
  272. <onentry>
  273. <script>
  274. self.task = self.new_tasks.pop()
  275. </script>
  276. <raise event="create_instance" scope="cd">
  277. <parameter expr="'tasks'"/>
  278. <parameter expr="'Task'"/>
  279. <parameter expr="self.task"/>
  280. <parameter expr="self.mvs_operations"/>
  281. <parameter expr="self.mvk"/>
  282. </raise>
  283. </onentry>
  284. <transition event="instance_created" target="../created">
  285. <parameter name="instancename"/>
  286. <raise scope="cd" event="start_instance">
  287. <parameter expr="instancename" />
  288. </raise>
  289. <script>
  290. self.sc_map[self.task] = instancename
  291. </script>
  292. </transition>
  293. </state>
  294. <state id="created">
  295. <transition cond="self.new_tasks" target="../creating"/>
  296. <transition cond="not self.new_tasks and self.old_tasks" target="../../remove_SC"/>
  297. <transition cond="not self.new_tasks and not self.old_tasks" target="../../refresh_tasks"/>
  298. </state>
  299. </state>
  300. <state id="remove_SC">
  301. <transition cond="self.old_tasks" target=".">
  302. <script>
  303. task = self.old_tasks.pop()
  304. </script>
  305. <raise scope="cd" event="delete_instance">
  306. <parameter expr="self.sc_map[task]"/>
  307. </raise>
  308. <script>
  309. del self.sc_map[task]
  310. </script>
  311. </transition>
  312. <transition cond="not self.old_tasks" target="../refresh_tasks"/>
  313. </state>
  314. </state>
  315. <state id="mvs_GC" initial="suspend_tasks">
  316. <state id="suspend_tasks">
  317. <onentry>
  318. <raise scope="broad" event="pause_task"/>
  319. </onentry>
  320. <transition target="../mvs_GC"/>
  321. </state>
  322. <state id="mvs_GC">
  323. <onentry>
  324. <script>
  325. self.mvs.garbage_collect()
  326. </script>
  327. <raise scope="broad" event="resume_task"/>
  328. </onentry>
  329. <transition after="self.sccd_yield() + 10" target="."/>
  330. </state>
  331. </state>
  332. </parallel>
  333. <state id="remove_sockets">
  334. <state id="remove_sockets">
  335. <transition event="delete_socket" target=".">
  336. <parameter name="socket"/>
  337. <script>
  338. for task in self.output_queue.keys():
  339. self.output_queue[task] = [s for s in self.output_queue[task] if s != socket]
  340. </script>
  341. </transition>
  342. </state>
  343. </state>
  344. </parallel>
  345. </scxml>
  346. </class>