mvkcontroller.xml 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321
  1. <class name="MvKController">
  2. <relationships>
  3. <association name="to_mvi" class="Server" min="1" max="1"/>
  4. </relationships>
  5. <constructor>
  6. <parameter name="params"/>
  7. <body>
  8. <![CDATA[
  9. self.mvs = ModelverseState("../bootstrap/bootstrap.m.gz")
  10. # Enable Garbage Collection
  11. self.mvs.GC = True
  12. self.root = self.mvs.read_root()
  13. # Instantiate the kernel.
  14. self.mvk = ModelverseKernel(self.root)
  15. # Parse kernel-related options
  16. default_kernel_type = 'adaptive-jit'
  17. kernel_opts = [default_kernel_type]
  18. for parameter in params:
  19. if parameter.startswith('--kernel='):
  20. kernel_opts = parameter[len('--kernel='):].split(',')
  21. for opt in kernel_opts:
  22. if opt == 'legacy-interpreter':
  23. self.mvk = LegacyModelverseKernel(self.root)
  24. elif opt == 'interpreter':
  25. self.mvk.jit.set_jit_enabled(False)
  26. elif opt == 'no-thunks':
  27. self.mvk.jit.enable_thunks(False)
  28. elif opt == 'thunks':
  29. self.mvk.jit.enable_thunks()
  30. elif opt == 'no-source-maps':
  31. self.mvk.jit.enable_source_maps(False)
  32. elif opt == 'source-maps':
  33. self.mvk.jit.enable_source_maps()
  34. elif opt == 'no-insert-nops':
  35. self.mvk.jit.enable_nop_insertion(False)
  36. elif opt == 'insert-nops':
  37. self.mvk.jit.enable_nop_insertion()
  38. elif opt == 'trace':
  39. self.mvk.jit.enable_tracing()
  40. elif opt == 'bytecode-interpreter':
  41. self.mvk.jit.set_function_body_compiler(jit.compile_function_body_interpret)
  42. elif opt == 'baseline-jit':
  43. self.mvk.jit.set_function_body_compiler(jit.compile_function_body_baseline)
  44. elif opt == 'fast-jit':
  45. self.mvk.jit.set_function_body_compiler(jit.compile_function_body_fast)
  46. elif opt == 'adaptive-jit-favor-large-functions':
  47. self.mvk.jit.set_function_body_compiler(
  48. lambda *args: jit.compile_function_body_adaptive(
  49. *args, temperature_heuristic=jit.favor_large_functions))
  50. elif opt == 'adaptive-jit-favor-small-functions':
  51. self.mvk.jit.set_function_body_compiler(
  52. lambda *args: jit.compile_function_body_adaptive(
  53. *args, temperature_heuristic=jit.favor_small_functions))
  54. elif opt == 'adaptive-jit-favor-loops':
  55. self.mvk.jit.set_function_body_compiler(
  56. lambda *args: jit.compile_function_body_adaptive(
  57. *args, temperature_heuristic=jit.favor_loops))
  58. elif opt == 'adaptive-jit' or opt == 'adaptive-jit-favor-small-loops':
  59. self.mvk.jit.set_function_body_compiler(
  60. lambda *args: jit.compile_function_body_adaptive(
  61. *args, temperature_heuristic=jit.favor_small_loops))
  62. else:
  63. print("warning: unknown kernel option '%s'." % opt)
  64. self.all_failed = False
  65. self.timeout = False
  66. self.tasks = set()
  67. self.input_queue = defaultdict(list)
  68. self.output_queue = defaultdict(list)
  69. self.source = None
  70. self.port = int(sys.argv[1])
  71. self.count = 0
  72. self.debugged_tasks = set()
  73. self.debug_info = {}
  74. self.done_something = False
  75. self.mvs_operations = {
  76. "CN": self.mvs.create_node,
  77. "CE": self.mvs.create_edge,
  78. "CNV": self.mvs.create_nodevalue,
  79. "CD": self.mvs.create_dict,
  80. "RV": self.mvs.read_value,
  81. "RO": self.mvs.read_outgoing,
  82. "RI": self.mvs.read_incoming,
  83. "RE": self.mvs.read_edge,
  84. "RD": self.mvs.read_dict,
  85. "RDN": self.mvs.read_dict_node,
  86. "RDNE": self.mvs.read_dict_node_edge,
  87. "RDE": self.mvs.read_dict_edge,
  88. "RRD": self.mvs.read_reverse_dict,
  89. "RR": self.mvs.read_root,
  90. "RDK": self.mvs.read_dict_keys,
  91. "DE": self.mvs.delete_edge,
  92. "DN": self.mvs.delete_node,
  93. }
  94. self.execute_modelverse("", "load_primitives", [])
  95. ]]>
  96. </body>
  97. </constructor>
  98. <method name="execute_modelverse">
  99. <parameter name="taskname"/>
  100. <parameter name="operation"/>
  101. <parameter name="params"/>
  102. <body>
  103. <![CDATA[
  104. reply = None
  105. commands = []
  106. mvk = self.mvk
  107. mvs_operations = self.mvs_operations
  108. try:
  109. while 1:
  110. commands = mvk.execute_yields(taskname, operation, params, reply)
  111. if commands is None:
  112. break
  113. reply = [mvs_operations[command[0]](*(command[1])) for command in commands]
  114. except:
  115. print("ERROR: " + str(self.mvk.debug_info.get(taskname, "Unknown taskname")))
  116. raise
  117. ]]>
  118. </body>
  119. </method>
  120. <scxml initial="init_server">
  121. <state id="init_server">
  122. <onentry>
  123. <raise scope="cd" event="create_instance">
  124. <parameter expr="'to_mvi'"/>
  125. <parameter expr="'Server'"/>
  126. <parameter expr="''"/>
  127. <parameter expr="self.port"/>
  128. </raise>
  129. </onentry>
  130. <transition event="instance_created" target="../running">
  131. <parameter name="instancename"/>
  132. <raise scope="cd" event="start_instance">
  133. <parameter expr="instancename"/>
  134. </raise>
  135. </transition>
  136. </state>
  137. <parallel id="running">
  138. <state id="wait_for_requests">
  139. <state id="wait">
  140. <transition event="from_mvi" target=".">
  141. <parameter name="source"/>
  142. <parameter name="data"/>
  143. <script>
  144. # No JSON encoding necessary, as it is not complex
  145. try:
  146. self.done_something = False
  147. if data["op"] == "set_input":
  148. if "value" in data:
  149. value = [json.loads(data["value"])]
  150. else:
  151. value = json.loads(data["data"])
  152. self.input_queue[data["taskname"]].append((source, value))
  153. elif data["op"] == "get_output":
  154. self.output_queue[data["taskname"]].append(source)
  155. elif data["op"] == "attach_debugger":
  156. self.debugged_tasks.add(data["taskname"])
  157. self.done_something = True
  158. self.source = source
  159. self.debug_info[data["taskname"]] = {'state': 'running', 'breakpoints': []}
  160. elif data["op"] == "detach_debugger":
  161. self.debugged_tasks.discard(data["taskname"])
  162. self.done_something = True
  163. self.source = source
  164. del self.debug_info[data["taskname"]]
  165. elif data["op"] == "pause":
  166. if data["taskname"] in self.debugged_tasks:
  167. self.debug_info[data["taskname"]]['state'] = 'paused'
  168. self.done_something = True
  169. self.source = source
  170. elif data["op"] == "resume":
  171. if data["taskname"] in self.debugged_tasks:
  172. self.debug_info[data["taskname"]]['state'] = 'running'
  173. self.done_something = True
  174. self.source = source
  175. elif data["op"] == "step_over":
  176. pass
  177. self.done_something = True
  178. self.source = source
  179. elif data["op"] == "step_into":
  180. pass
  181. self.done_something = True
  182. self.source = source
  183. else:
  184. raise Exception("DROPPING unknown operation: " + str(data["op"]))
  185. except ValueError:
  186. print("Error when deserializing request: " + str(data))
  187. raise
  188. </script>
  189. </transition>
  190. <transition cond="self.done_something" target=".">
  191. <raise event="HTTP_input" scope="narrow" target="'to_mvi/%s' % self.source">
  192. <parameter expr="json.dumps(True)"/>
  193. </raise>
  194. <script>
  195. self.done_something = False
  196. </script>
  197. </transition>
  198. </state>
  199. </state>
  200. <state id="execution" initial="execution">
  201. <state id="execution">
  202. <onentry>
  203. <script>
  204. #start = time.time()
  205. self.outputs = []
  206. self.timeout = False
  207. if self.tasks:
  208. task = self.tasks.pop()
  209. if not task in self.debugged_tasks or self.debug_info[task]['state'] == 'running':
  210. # Check if there are values to input
  211. while self.input_queue[task]:
  212. source, args = self.input_queue[task].pop(0)
  213. for args_entry in args:
  214. self.execute_modelverse(task, "set_input", [args_entry])
  215. self.outputs.append((source, "OK"))
  216. self.all_failed = False
  217. if task in self.debugged_tasks:
  218. start_time = -float('inf')
  219. else:
  220. start_time = time.time()
  221. # Grant each task some milliseconds of execution
  222. while (time.time() - start_time &lt; 0.05):
  223. self.execute_modelverse(task, "execute_rule", [])
  224. if not self.mvk.success:
  225. # Blocking or broken, so quit already to stop wasting CPU
  226. break
  227. # Could at least execute one instruction, so mark it as "not failed"
  228. self.all_failed = False
  229. # Perform output if there is anything
  230. while self.output_queue[task]:
  231. self.execute_modelverse(task, "get_output", [])
  232. if self.mvk.success:
  233. self.outputs.append((self.output_queue[task].pop(0), self.mvk.returnvalue))
  234. self.all_failed = False
  235. else:
  236. break
  237. else:
  238. if self.count >= 2000:
  239. self.count = 0
  240. self.mvs.purge()
  241. else:
  242. self.count += 1
  243. self.mvs.garbage_collect()
  244. out = self.mvs.read_outgoing(self.root)
  245. for m in out:
  246. src, task = self.mvs.read_edge(m)
  247. outgoing = self.mvs.read_outgoing(m)
  248. first = self.mvs.read_edge(outgoing[0])
  249. dest = first[1]
  250. name = self.mvs.read_value(dest)
  251. if name.startswith("__"):
  252. continue
  253. self.tasks.add(name)
  254. self.timeout = self.all_failed
  255. self.all_failed = True
  256. </script>
  257. </onentry>
  258. <transition cond="self.outputs" target="../process_data"/>
  259. <transition cond="not self.timeout and not self.outputs" target="../timeout_none"/>
  260. <transition cond="self.timeout and not self.outputs" target="../timeout_long"/>
  261. </state>
  262. <state id="timeout_long">
  263. <transition after="self.sccd_yield() + 0.05" target="../execution"/>
  264. </state>
  265. <state id="timeout_none">
  266. <transition after="self.sccd_yield()" target="../execution"/>
  267. </state>
  268. <state id="process_data">
  269. <transition cond="self.outputs" target=".">
  270. <script>
  271. destination, value = self.outputs.pop()
  272. </script>
  273. <raise event="HTTP_input" scope="narrow" target="'to_mvi/%s' % destination">
  274. <parameter expr="json.dumps(value)"/>
  275. </raise>
  276. </transition>
  277. <transition cond="not self.outputs" target="../execution"/>
  278. </state>
  279. </state>
  280. <state id="remove_sockets">
  281. <state id="remove_sockets">
  282. <transition event="delete_socket" target=".">
  283. <parameter name="socket"/>
  284. <script>
  285. for task in self.output_queue.keys():
  286. self.output_queue[task] = [s for s in self.output_queue[task] if s != socket]
  287. </script>
  288. </transition>
  289. </state>
  290. </state>
  291. </parallel>
  292. </scxml>
  293. </class>