redscript-mc 1.2.30 → 2.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (593) hide show
  1. package/.claudeignore +21 -0
  2. package/.github/workflows/ci.yml +1 -0
  3. package/README.md +12 -16
  4. package/README.zh.md +2 -2
  5. package/demo.gif +0 -0
  6. package/dist/cli.js +2 -554
  7. package/dist/compile.js +2 -266
  8. package/dist/index.js +2 -159
  9. package/dist/src/__tests__/budget.test.js +261 -0
  10. package/dist/src/__tests__/cli.test.js +104 -0
  11. package/dist/{__tests__ → src/__tests__}/dce.test.js +11 -47
  12. package/dist/{__tests__ → src/__tests__}/diagnostics.test.js +67 -40
  13. package/dist/src/__tests__/e2e/basic.test.d.ts +8 -0
  14. package/dist/src/__tests__/e2e/basic.test.js +140 -0
  15. package/dist/src/__tests__/e2e/coroutine.test.d.ts +7 -0
  16. package/dist/src/__tests__/e2e/coroutine.test.js +132 -0
  17. package/dist/src/__tests__/e2e/macros.test.d.ts +9 -0
  18. package/dist/src/__tests__/e2e/macros.test.js +182 -0
  19. package/dist/src/__tests__/e2e/migrate.test.d.ts +13 -0
  20. package/dist/src/__tests__/e2e/migrate.test.js +2739 -0
  21. package/dist/src/__tests__/e2e/stdlib-e2e.test.d.ts +10 -0
  22. package/dist/src/__tests__/e2e/stdlib-e2e.test.js +324 -0
  23. package/dist/src/__tests__/enum.test.d.ts +10 -0
  24. package/dist/src/__tests__/enum.test.js +389 -0
  25. package/dist/src/__tests__/generics.test.d.ts +14 -0
  26. package/dist/src/__tests__/generics.test.js +367 -0
  27. package/dist/src/__tests__/hir/desugar.test.js +234 -0
  28. package/dist/src/__tests__/incremental.test.d.ts +5 -0
  29. package/dist/src/__tests__/incremental.test.js +308 -0
  30. package/dist/src/__tests__/lir/lower.test.js +559 -0
  31. package/dist/src/__tests__/lir/types.test.js +185 -0
  32. package/dist/src/__tests__/lir/verify.test.js +221 -0
  33. package/dist/src/__tests__/lsp.test.d.ts +7 -0
  34. package/dist/src/__tests__/lsp.test.js +245 -0
  35. package/dist/{__tests__ → src/__tests__}/mc-integration.test.js +1 -3
  36. package/dist/src/__tests__/mc-version.test.d.ts +10 -0
  37. package/dist/src/__tests__/mc-version.test.js +154 -0
  38. package/dist/src/__tests__/mir/arithmetic.test.js +130 -0
  39. package/dist/src/__tests__/mir/control-flow.test.js +205 -0
  40. package/dist/src/__tests__/mir/verify.test.js +223 -0
  41. package/dist/src/__tests__/modules.test.d.ts +7 -0
  42. package/dist/src/__tests__/modules.test.js +333 -0
  43. package/dist/src/__tests__/optimizer/block_merge.test.js +78 -0
  44. package/dist/src/__tests__/optimizer/branch_simplify.test.js +58 -0
  45. package/dist/src/__tests__/optimizer/constant_fold.test.js +131 -0
  46. package/dist/src/__tests__/optimizer/copy_prop.test.js +91 -0
  47. package/dist/src/__tests__/optimizer/coroutine.test.d.ts +12 -0
  48. package/dist/src/__tests__/optimizer/coroutine.test.js +251 -0
  49. package/dist/src/__tests__/optimizer/dce.test.d.ts +1 -0
  50. package/dist/src/__tests__/optimizer/dce.test.js +76 -0
  51. package/dist/src/__tests__/optimizer/interprocedural.test.d.ts +1 -0
  52. package/dist/src/__tests__/optimizer/interprocedural.test.js +145 -0
  53. package/dist/src/__tests__/optimizer/lir/const_imm.test.d.ts +1 -0
  54. package/dist/src/__tests__/optimizer/lir/const_imm.test.js +138 -0
  55. package/dist/src/__tests__/optimizer/lir/dead_slot.test.d.ts +1 -0
  56. package/dist/src/__tests__/optimizer/lir/dead_slot.test.js +141 -0
  57. package/dist/src/__tests__/optimizer/lir/peephole.test.d.ts +1 -0
  58. package/dist/src/__tests__/optimizer/lir/peephole.test.js +126 -0
  59. package/dist/src/__tests__/optimizer/lir/pipeline.test.d.ts +1 -0
  60. package/dist/src/__tests__/optimizer/lir/pipeline.test.js +84 -0
  61. package/dist/src/__tests__/optimizer/nbt-batch.test.d.ts +1 -0
  62. package/dist/src/__tests__/optimizer/nbt-batch.test.js +110 -0
  63. package/dist/src/__tests__/optimizer/pipeline.test.d.ts +1 -0
  64. package/dist/src/__tests__/optimizer/pipeline.test.js +102 -0
  65. package/dist/src/__tests__/optimizer/selector-cache.test.d.ts +1 -0
  66. package/dist/src/__tests__/optimizer/selector-cache.test.js +103 -0
  67. package/dist/src/__tests__/optimizer/unroll.test.d.ts +1 -0
  68. package/dist/src/__tests__/optimizer/unroll.test.js +206 -0
  69. package/dist/src/__tests__/option.test.d.ts +14 -0
  70. package/dist/src/__tests__/option.test.js +275 -0
  71. package/dist/src/__tests__/parser.test.d.ts +1 -0
  72. package/dist/src/__tests__/repl.test.d.ts +1 -0
  73. package/dist/src/__tests__/schedule.test.d.ts +7 -0
  74. package/dist/src/__tests__/schedule.test.js +98 -0
  75. package/dist/src/__tests__/sourcemap.test.d.ts +7 -0
  76. package/dist/src/__tests__/sourcemap.test.js +227 -0
  77. package/dist/src/__tests__/tuple.test.d.ts +11 -0
  78. package/dist/src/__tests__/tuple.test.js +202 -0
  79. package/dist/src/__tests__/typechecker-strict.test.d.ts +10 -0
  80. package/dist/src/__tests__/typechecker-strict.test.js +197 -0
  81. package/dist/src/__tests__/typechecker.test.d.ts +1 -0
  82. package/dist/{ast → src/ast}/types.d.ts +58 -3
  83. package/dist/src/cache/deps.d.ts +41 -0
  84. package/dist/src/cache/deps.js +158 -0
  85. package/dist/src/cache/incremental.d.ts +35 -0
  86. package/dist/src/cache/incremental.js +165 -0
  87. package/dist/src/cache/index.d.ts +37 -0
  88. package/dist/src/cache/index.js +152 -0
  89. package/dist/{cli.d.ts → src/cli.d.ts} +1 -1
  90. package/dist/src/cli.js +474 -0
  91. package/dist/src/compile.d.ts +37 -0
  92. package/dist/src/compile.js +165 -0
  93. package/dist/{diagnostics → src/diagnostics}/index.d.ts +1 -1
  94. package/dist/{diagnostics → src/diagnostics}/index.js +8 -11
  95. package/dist/src/emit/compile.d.ts +29 -0
  96. package/dist/src/emit/compile.js +143 -0
  97. package/dist/src/emit/index.d.ts +26 -0
  98. package/dist/src/emit/index.js +223 -0
  99. package/dist/src/emit/modules.d.ts +29 -0
  100. package/dist/src/emit/modules.js +492 -0
  101. package/dist/src/emit/sourcemap.d.ts +53 -0
  102. package/dist/src/emit/sourcemap.js +73 -0
  103. package/dist/src/hir/lower.d.ts +15 -0
  104. package/dist/src/hir/lower.js +399 -0
  105. package/dist/src/hir/monomorphize.d.ts +22 -0
  106. package/dist/src/hir/monomorphize.js +379 -0
  107. package/dist/src/hir/types.d.ts +406 -0
  108. package/dist/src/hir/types.js +16 -0
  109. package/dist/src/index.d.ts +39 -0
  110. package/dist/src/index.js +67 -0
  111. package/dist/{lexer → src/lexer}/index.d.ts +1 -1
  112. package/dist/{lexer → src/lexer}/index.js +1 -0
  113. package/dist/src/lir/budget.d.ts +37 -0
  114. package/dist/src/lir/budget.js +280 -0
  115. package/dist/src/lir/lower.d.ts +15 -0
  116. package/dist/src/lir/lower.js +472 -0
  117. package/dist/src/lir/types.d.ts +139 -0
  118. package/dist/src/lir/types.js +11 -0
  119. package/dist/src/lir/verify.d.ts +14 -0
  120. package/dist/src/lir/verify.js +113 -0
  121. package/dist/src/lsp/main.d.ts +8 -0
  122. package/dist/src/lsp/main.js +11 -0
  123. package/dist/src/lsp/server.d.ts +11 -0
  124. package/dist/src/lsp/server.js +352 -0
  125. package/dist/{mc-test → src/mc-test}/runner.js +4 -3
  126. package/dist/src/mir/lower.d.ts +9 -0
  127. package/dist/src/mir/lower.js +1264 -0
  128. package/dist/src/mir/macro.d.ts +22 -0
  129. package/dist/src/mir/macro.js +168 -0
  130. package/dist/src/mir/types.d.ts +191 -0
  131. package/dist/src/mir/types.js +11 -0
  132. package/dist/src/mir/verify.d.ts +16 -0
  133. package/dist/src/mir/verify.js +216 -0
  134. package/dist/src/optimizer/block_merge.d.ts +12 -0
  135. package/dist/src/optimizer/block_merge.js +84 -0
  136. package/dist/src/optimizer/branch_simplify.d.ts +9 -0
  137. package/dist/src/optimizer/branch_simplify.js +28 -0
  138. package/dist/src/optimizer/constant_fold.d.ts +10 -0
  139. package/dist/src/optimizer/constant_fold.js +85 -0
  140. package/dist/src/optimizer/copy_prop.d.ts +9 -0
  141. package/dist/src/optimizer/copy_prop.js +113 -0
  142. package/dist/src/optimizer/coroutine.d.ts +34 -0
  143. package/dist/src/optimizer/coroutine.js +789 -0
  144. package/dist/src/optimizer/dce.d.ts +8 -0
  145. package/dist/src/optimizer/dce.js +156 -0
  146. package/dist/src/optimizer/interprocedural.d.ts +14 -0
  147. package/dist/src/optimizer/interprocedural.js +186 -0
  148. package/dist/src/optimizer/lir/const_imm.d.ts +12 -0
  149. package/dist/src/optimizer/lir/const_imm.js +139 -0
  150. package/dist/src/optimizer/lir/dead_slot.d.ts +14 -0
  151. package/dist/src/optimizer/lir/dead_slot.js +130 -0
  152. package/dist/src/optimizer/lir/peephole.d.ts +21 -0
  153. package/dist/src/optimizer/lir/peephole.js +52 -0
  154. package/dist/src/optimizer/lir/pipeline.d.ts +10 -0
  155. package/dist/src/optimizer/lir/pipeline.js +34 -0
  156. package/dist/src/optimizer/nbt-batch.d.ts +11 -0
  157. package/dist/src/optimizer/nbt-batch.js +51 -0
  158. package/dist/src/optimizer/pipeline.d.ts +14 -0
  159. package/dist/src/optimizer/pipeline.js +58 -0
  160. package/dist/src/optimizer/selector-cache.d.ts +22 -0
  161. package/dist/src/optimizer/selector-cache.js +100 -0
  162. package/dist/src/optimizer/unroll.d.ts +32 -0
  163. package/dist/src/optimizer/unroll.js +348 -0
  164. package/dist/{parser → src/parser}/index.d.ts +8 -0
  165. package/dist/{parser → src/parser}/index.js +204 -14
  166. package/dist/{repl.d.ts → src/repl.d.ts} +1 -1
  167. package/dist/{runtime → src/runtime}/index.js +1 -1
  168. package/dist/{typechecker → src/typechecker}/index.d.ts +4 -0
  169. package/dist/{typechecker → src/typechecker}/index.js +198 -13
  170. package/dist/src/types/mc-version.d.ts +24 -0
  171. package/dist/src/types/mc-version.js +49 -0
  172. package/docs/ROADMAP.md +395 -0
  173. package/docs/compiler-pipeline-redesign.md +2260 -0
  174. package/docs/optimization-ideas.md +1076 -0
  175. package/editors/vscode/out/extension.js +25176 -8000
  176. package/editors/vscode/package-lock.json +90 -6
  177. package/editors/vscode/package.json +3 -2
  178. package/editors/vscode/src/extension.ts +97 -67
  179. package/examples/showcase.mcrs +3 -3
  180. package/package.json +13 -6
  181. package/scripts/postbuild.js +15 -0
  182. package/src/__tests__/budget.test.ts +297 -0
  183. package/src/__tests__/cli.test.ts +8 -220
  184. package/src/__tests__/dce.test.ts +11 -56
  185. package/src/__tests__/diagnostics.test.ts +61 -41
  186. package/src/__tests__/e2e/basic.test.ts +154 -0
  187. package/src/__tests__/e2e/coroutine.test.ts +142 -0
  188. package/src/__tests__/e2e/macros.test.ts +199 -0
  189. package/src/__tests__/e2e/migrate.test.ts +3008 -0
  190. package/src/__tests__/e2e/stdlib-e2e.test.ts +348 -0
  191. package/src/__tests__/enum.test.ts +425 -0
  192. package/src/__tests__/generics.test.ts +390 -0
  193. package/src/__tests__/hir/desugar.test.ts +263 -0
  194. package/src/__tests__/incremental.test.ts +337 -0
  195. package/src/__tests__/lir/lower.test.ts +619 -0
  196. package/src/__tests__/lir/types.test.ts +207 -0
  197. package/src/__tests__/lir/verify.test.ts +249 -0
  198. package/src/__tests__/lsp.test.ts +270 -0
  199. package/src/__tests__/mc-integration.test.ts +1 -2
  200. package/src/__tests__/mc-version.test.ts +178 -0
  201. package/src/__tests__/mir/arithmetic.test.ts +156 -0
  202. package/src/__tests__/mir/control-flow.test.ts +242 -0
  203. package/src/__tests__/mir/verify.test.ts +254 -0
  204. package/src/__tests__/modules.test.ts +365 -0
  205. package/src/__tests__/optimizer/block_merge.test.ts +84 -0
  206. package/src/__tests__/optimizer/branch_simplify.test.ts +64 -0
  207. package/src/__tests__/optimizer/constant_fold.test.ts +145 -0
  208. package/src/__tests__/optimizer/copy_prop.test.ts +99 -0
  209. package/src/__tests__/optimizer/coroutine.test.ts +312 -0
  210. package/src/__tests__/optimizer/dce.test.ts +83 -0
  211. package/src/__tests__/optimizer/interprocedural.test.ts +174 -0
  212. package/src/__tests__/optimizer/lir/const_imm.test.ts +151 -0
  213. package/src/__tests__/optimizer/lir/dead_slot.test.ts +156 -0
  214. package/src/__tests__/optimizer/lir/peephole.test.ts +136 -0
  215. package/src/__tests__/optimizer/lir/pipeline.test.ts +113 -0
  216. package/src/__tests__/optimizer/nbt-batch.test.ts +119 -0
  217. package/src/__tests__/optimizer/pipeline.test.ts +116 -0
  218. package/src/__tests__/optimizer/selector-cache.test.ts +112 -0
  219. package/src/__tests__/optimizer/unroll.test.ts +231 -0
  220. package/src/__tests__/option.test.ts +299 -0
  221. package/src/__tests__/schedule.test.ts +105 -0
  222. package/src/__tests__/sourcemap.test.ts +254 -0
  223. package/src/__tests__/tuple.test.ts +220 -0
  224. package/src/__tests__/typechecker-strict.test.ts +216 -0
  225. package/src/ast/types.ts +39 -3
  226. package/src/cache/deps.ts +132 -0
  227. package/src/cache/incremental.ts +173 -0
  228. package/src/cache/index.ts +135 -0
  229. package/src/cli.ts +111 -195
  230. package/src/compile.ts +6 -162
  231. package/src/diagnostics/index.ts +8 -11
  232. package/src/emit/compile.ts +177 -0
  233. package/src/emit/index.ts +286 -0
  234. package/src/emit/modules.ts +581 -0
  235. package/src/emit/sourcemap.ts +101 -0
  236. package/src/hir/lower.ts +455 -0
  237. package/src/hir/monomorphize.ts +416 -0
  238. package/src/hir/types.ts +228 -0
  239. package/src/index.ts +37 -182
  240. package/src/lexer/index.ts +2 -1
  241. package/src/lir/budget.ts +321 -0
  242. package/src/lir/lower.ts +587 -0
  243. package/src/lir/types.ts +113 -0
  244. package/src/lir/verify.ts +129 -0
  245. package/src/lsp/main.ts +9 -0
  246. package/src/lsp/server.ts +414 -0
  247. package/src/mc-test/runner.ts +4 -3
  248. package/src/mir/lower.ts +1403 -0
  249. package/src/mir/macro.ts +167 -0
  250. package/src/mir/types.ts +117 -0
  251. package/src/mir/verify.ts +218 -0
  252. package/src/optimizer/block_merge.ts +93 -0
  253. package/src/optimizer/branch_simplify.ts +27 -0
  254. package/src/optimizer/constant_fold.ts +88 -0
  255. package/src/optimizer/copy_prop.ts +106 -0
  256. package/src/optimizer/coroutine.ts +996 -0
  257. package/src/optimizer/dce.ts +108 -653
  258. package/src/optimizer/interprocedural.ts +177 -0
  259. package/src/optimizer/lir/const_imm.ts +143 -0
  260. package/src/optimizer/lir/dead_slot.ts +123 -0
  261. package/src/optimizer/lir/peephole.ts +57 -0
  262. package/src/optimizer/lir/pipeline.ts +37 -0
  263. package/src/optimizer/nbt-batch.ts +50 -0
  264. package/src/optimizer/pipeline.ts +59 -0
  265. package/src/optimizer/selector-cache.ts +103 -0
  266. package/src/optimizer/unroll.ts +386 -0
  267. package/src/parser/index.ts +213 -16
  268. package/src/repl.ts +1 -1
  269. package/src/runtime/index.ts +1 -1
  270. package/src/stdlib/math.mcrs +4 -4
  271. package/src/templates/quest.mcrs +4 -4
  272. package/src/typechecker/index.ts +215 -15
  273. package/src/types/mc-version.ts +46 -0
  274. package/tsconfig.json +1 -1
  275. package/dist/__tests__/cli.test.js +0 -278
  276. package/dist/__tests__/codegen.test.js +0 -152
  277. package/dist/__tests__/e2e.test.d.ts +0 -6
  278. package/dist/__tests__/e2e.test.js +0 -1847
  279. package/dist/__tests__/entity-types.test.js +0 -203
  280. package/dist/__tests__/lowering.test.js +0 -1015
  281. package/dist/__tests__/macro.test.d.ts +0 -8
  282. package/dist/__tests__/macro.test.js +0 -305
  283. package/dist/__tests__/nbt.test.js +0 -82
  284. package/dist/__tests__/optimizer-advanced.test.js +0 -124
  285. package/dist/__tests__/optimizer.test.js +0 -149
  286. package/dist/__tests__/runtime.test.js +0 -289
  287. package/dist/__tests__/stdlib-advanced.test.d.ts +0 -4
  288. package/dist/__tests__/stdlib-advanced.test.js +0 -378
  289. package/dist/__tests__/stdlib-bigint.test.d.ts +0 -7
  290. package/dist/__tests__/stdlib-bigint.test.js +0 -428
  291. package/dist/__tests__/stdlib-math.test.d.ts +0 -7
  292. package/dist/__tests__/stdlib-math.test.js +0 -352
  293. package/dist/__tests__/stdlib-vec.test.d.ts +0 -4
  294. package/dist/__tests__/stdlib-vec.test.js +0 -264
  295. package/dist/__tests__/structure-optimizer.test.js +0 -33
  296. package/dist/__tests__/var-allocator.test.js +0 -69
  297. package/dist/codegen/cmdblock/index.d.ts +0 -26
  298. package/dist/codegen/cmdblock/index.js +0 -45
  299. package/dist/codegen/mcfunction/index.d.ts +0 -40
  300. package/dist/codegen/mcfunction/index.js +0 -606
  301. package/dist/codegen/structure/index.d.ts +0 -24
  302. package/dist/codegen/structure/index.js +0 -279
  303. package/dist/codegen/var-allocator.d.ts +0 -45
  304. package/dist/codegen/var-allocator.js +0 -104
  305. package/dist/compile.d.ts +0 -68
  306. package/dist/data/arena/function/__load.mcfunction +0 -6
  307. package/dist/data/arena/function/__tick.mcfunction +0 -2
  308. package/dist/data/arena/function/announce_leaders/else_1.mcfunction +0 -3
  309. package/dist/data/arena/function/announce_leaders/foreach_0/merge_2.mcfunction +0 -1
  310. package/dist/data/arena/function/announce_leaders/foreach_0/then_0.mcfunction +0 -3
  311. package/dist/data/arena/function/announce_leaders/foreach_0.mcfunction +0 -7
  312. package/dist/data/arena/function/announce_leaders/foreach_1/merge_2.mcfunction +0 -1
  313. package/dist/data/arena/function/announce_leaders/foreach_1/then_0.mcfunction +0 -4
  314. package/dist/data/arena/function/announce_leaders/foreach_1.mcfunction +0 -6
  315. package/dist/data/arena/function/announce_leaders/merge_2.mcfunction +0 -1
  316. package/dist/data/arena/function/announce_leaders/then_0.mcfunction +0 -4
  317. package/dist/data/arena/function/announce_leaders.mcfunction +0 -6
  318. package/dist/data/arena/function/arena_tick/merge_2.mcfunction +0 -1
  319. package/dist/data/arena/function/arena_tick/then_0.mcfunction +0 -4
  320. package/dist/data/arena/function/arena_tick.mcfunction +0 -11
  321. package/dist/data/counter/function/__load.mcfunction +0 -5
  322. package/dist/data/counter/function/__tick.mcfunction +0 -2
  323. package/dist/data/counter/function/counter_tick/merge_2.mcfunction +0 -1
  324. package/dist/data/counter/function/counter_tick/then_0.mcfunction +0 -3
  325. package/dist/data/counter/function/counter_tick.mcfunction +0 -11
  326. package/dist/data/gcd2/function/__load.mcfunction +0 -3
  327. package/dist/data/gcd2/function/abs/merge_2.mcfunction +0 -3
  328. package/dist/data/gcd2/function/abs/then_0.mcfunction +0 -5
  329. package/dist/data/gcd2/function/abs.mcfunction +0 -7
  330. package/dist/data/gcd2/function/gcd/loop_body_1.mcfunction +0 -7
  331. package/dist/data/gcd2/function/gcd/loop_check_0.mcfunction +0 -5
  332. package/dist/data/gcd2/function/gcd/loop_exit_2.mcfunction +0 -3
  333. package/dist/data/gcd2/function/gcd.mcfunction +0 -14
  334. package/dist/data/gcd3/function/__load.mcfunction +0 -3
  335. package/dist/data/gcd3/function/abs/merge_2.mcfunction +0 -3
  336. package/dist/data/gcd3/function/abs/then_0.mcfunction +0 -5
  337. package/dist/data/gcd3/function/abs.mcfunction +0 -7
  338. package/dist/data/gcd3/function/gcd/loop_body_1.mcfunction +0 -7
  339. package/dist/data/gcd3/function/gcd/loop_check_0.mcfunction +0 -5
  340. package/dist/data/gcd3/function/gcd/loop_exit_2.mcfunction +0 -3
  341. package/dist/data/gcd3/function/gcd.mcfunction +0 -14
  342. package/dist/data/gcd3/function/test.mcfunction +0 -7
  343. package/dist/data/gcd3nm/function/__load.mcfunction +0 -3
  344. package/dist/data/gcd3nm/function/abs/merge_2.mcfunction +0 -3
  345. package/dist/data/gcd3nm/function/abs/then_0.mcfunction +0 -5
  346. package/dist/data/gcd3nm/function/abs.mcfunction +0 -7
  347. package/dist/data/gcd3nm/function/gcd/loop_body_1.mcfunction +0 -7
  348. package/dist/data/gcd3nm/function/gcd/loop_check_0.mcfunction +0 -5
  349. package/dist/data/gcd3nm/function/gcd/loop_exit_2.mcfunction +0 -3
  350. package/dist/data/gcd3nm/function/gcd.mcfunction +0 -14
  351. package/dist/data/gcd3nm/function/test.mcfunction +0 -7
  352. package/dist/data/gcd_test/function/__load.mcfunction +0 -3
  353. package/dist/data/gcd_test/function/abs/merge_2.mcfunction +0 -3
  354. package/dist/data/gcd_test/function/abs/then_0.mcfunction +0 -5
  355. package/dist/data/gcd_test/function/abs.mcfunction +0 -7
  356. package/dist/data/gcd_test/function/gcd/loop_body_1.mcfunction +0 -7
  357. package/dist/data/gcd_test/function/gcd/loop_check_0.mcfunction +0 -5
  358. package/dist/data/gcd_test/function/gcd/loop_exit_2.mcfunction +0 -3
  359. package/dist/data/gcd_test/function/gcd.mcfunction +0 -14
  360. package/dist/data/isqrttest/function/__load.mcfunction +0 -6
  361. package/dist/data/isqrttest/function/isqrt/loop_body_4.mcfunction +0 -12
  362. package/dist/data/isqrttest/function/isqrt/loop_check_3.mcfunction +0 -5
  363. package/dist/data/isqrttest/function/isqrt/loop_exit_5.mcfunction +0 -3
  364. package/dist/data/isqrttest/function/isqrt/merge_2.mcfunction +0 -4
  365. package/dist/data/isqrttest/function/isqrt/merge_8.mcfunction +0 -6
  366. package/dist/data/isqrttest/function/isqrt/then_0.mcfunction +0 -3
  367. package/dist/data/isqrttest/function/isqrt/then_6.mcfunction +0 -3
  368. package/dist/data/isqrttest/function/isqrt.mcfunction +0 -7
  369. package/dist/data/isqrttest/function/test.mcfunction +0 -6
  370. package/dist/data/mathtest/function/__load.mcfunction +0 -3
  371. package/dist/data/mathtest/function/abs/merge_2.mcfunction +0 -3
  372. package/dist/data/mathtest/function/abs/then_0.mcfunction +0 -5
  373. package/dist/data/mathtest/function/abs.mcfunction +0 -6
  374. package/dist/data/mathtest/function/test.mcfunction +0 -5
  375. package/dist/data/minecraft/tags/function/load.json +0 -5
  376. package/dist/data/minecraft/tags/function/tick.json +0 -5
  377. package/dist/data/mypack/function/__load.mcfunction +0 -13
  378. package/dist/data/mypack/function/_atan_init.mcfunction +0 -2
  379. package/dist/data/mypack/function/abs/merge_2.mcfunction +0 -3
  380. package/dist/data/mypack/function/abs/then_0.mcfunction +0 -5
  381. package/dist/data/mypack/function/abs.mcfunction +0 -6
  382. package/dist/data/mypack/function/atan2_fixed/__sgi_1.mcfunction +0 -2
  383. package/dist/data/mypack/function/atan2_fixed/else_34.mcfunction +0 -3
  384. package/dist/data/mypack/function/atan2_fixed/loop_body_31.mcfunction +0 -19
  385. package/dist/data/mypack/function/atan2_fixed/loop_check_30.mcfunction +0 -5
  386. package/dist/data/mypack/function/atan2_fixed/loop_exit_32.mcfunction +0 -6
  387. package/dist/data/mypack/function/atan2_fixed/merge_11.mcfunction +0 -6
  388. package/dist/data/mypack/function/atan2_fixed/merge_14.mcfunction +0 -3
  389. package/dist/data/mypack/function/atan2_fixed/merge_17.mcfunction +0 -6
  390. package/dist/data/mypack/function/atan2_fixed/merge_2.mcfunction +0 -5
  391. package/dist/data/mypack/function/atan2_fixed/merge_20.mcfunction +0 -5
  392. package/dist/data/mypack/function/atan2_fixed/merge_23.mcfunction +0 -5
  393. package/dist/data/mypack/function/atan2_fixed/merge_26.mcfunction +0 -6
  394. package/dist/data/mypack/function/atan2_fixed/merge_29.mcfunction +0 -4
  395. package/dist/data/mypack/function/atan2_fixed/merge_38.mcfunction +0 -5
  396. package/dist/data/mypack/function/atan2_fixed/merge_41.mcfunction +0 -5
  397. package/dist/data/mypack/function/atan2_fixed/merge_44.mcfunction +0 -5
  398. package/dist/data/mypack/function/atan2_fixed/merge_47.mcfunction +0 -5
  399. package/dist/data/mypack/function/atan2_fixed/merge_5.mcfunction +0 -5
  400. package/dist/data/mypack/function/atan2_fixed/merge_8.mcfunction +0 -3
  401. package/dist/data/mypack/function/atan2_fixed/then_0.mcfunction +0 -5
  402. package/dist/data/mypack/function/atan2_fixed/then_12.mcfunction +0 -3
  403. package/dist/data/mypack/function/atan2_fixed/then_15.mcfunction +0 -5
  404. package/dist/data/mypack/function/atan2_fixed/then_18.mcfunction +0 -5
  405. package/dist/data/mypack/function/atan2_fixed/then_21.mcfunction +0 -3
  406. package/dist/data/mypack/function/atan2_fixed/then_24.mcfunction +0 -3
  407. package/dist/data/mypack/function/atan2_fixed/then_27.mcfunction +0 -6
  408. package/dist/data/mypack/function/atan2_fixed/then_3.mcfunction +0 -3
  409. package/dist/data/mypack/function/atan2_fixed/then_33.mcfunction +0 -5
  410. package/dist/data/mypack/function/atan2_fixed/then_36.mcfunction +0 -5
  411. package/dist/data/mypack/function/atan2_fixed/then_39.mcfunction +0 -5
  412. package/dist/data/mypack/function/atan2_fixed/then_42.mcfunction +0 -3
  413. package/dist/data/mypack/function/atan2_fixed/then_45.mcfunction +0 -5
  414. package/dist/data/mypack/function/atan2_fixed/then_6.mcfunction +0 -3
  415. package/dist/data/mypack/function/atan2_fixed/then_9.mcfunction +0 -5
  416. package/dist/data/mypack/function/atan2_fixed.mcfunction +0 -7
  417. package/dist/data/mypack/function/my_game.mcfunction +0 -10
  418. package/dist/data/quiz/function/__load.mcfunction +0 -16
  419. package/dist/data/quiz/function/__tick.mcfunction +0 -6
  420. package/dist/data/quiz/function/__trigger_quiz_a_dispatch.mcfunction +0 -4
  421. package/dist/data/quiz/function/__trigger_quiz_b_dispatch.mcfunction +0 -4
  422. package/dist/data/quiz/function/__trigger_quiz_c_dispatch.mcfunction +0 -4
  423. package/dist/data/quiz/function/__trigger_quiz_start_dispatch.mcfunction +0 -4
  424. package/dist/data/quiz/function/answer_a.mcfunction +0 -4
  425. package/dist/data/quiz/function/answer_b.mcfunction +0 -4
  426. package/dist/data/quiz/function/answer_c.mcfunction +0 -4
  427. package/dist/data/quiz/function/ask_question/else_1.mcfunction +0 -5
  428. package/dist/data/quiz/function/ask_question/else_4.mcfunction +0 -5
  429. package/dist/data/quiz/function/ask_question/else_7.mcfunction +0 -4
  430. package/dist/data/quiz/function/ask_question/merge_2.mcfunction +0 -1
  431. package/dist/data/quiz/function/ask_question/merge_5.mcfunction +0 -2
  432. package/dist/data/quiz/function/ask_question/merge_8.mcfunction +0 -2
  433. package/dist/data/quiz/function/ask_question/then_0.mcfunction +0 -4
  434. package/dist/data/quiz/function/ask_question/then_3.mcfunction +0 -4
  435. package/dist/data/quiz/function/ask_question/then_6.mcfunction +0 -4
  436. package/dist/data/quiz/function/ask_question.mcfunction +0 -7
  437. package/dist/data/quiz/function/finish_quiz.mcfunction +0 -6
  438. package/dist/data/quiz/function/handle_answer/else_1.mcfunction +0 -5
  439. package/dist/data/quiz/function/handle_answer/else_10.mcfunction +0 -3
  440. package/dist/data/quiz/function/handle_answer/else_16.mcfunction +0 -3
  441. package/dist/data/quiz/function/handle_answer/else_4.mcfunction +0 -3
  442. package/dist/data/quiz/function/handle_answer/else_7.mcfunction +0 -5
  443. package/dist/data/quiz/function/handle_answer/merge_11.mcfunction +0 -2
  444. package/dist/data/quiz/function/handle_answer/merge_14.mcfunction +0 -2
  445. package/dist/data/quiz/function/handle_answer/merge_17.mcfunction +0 -2
  446. package/dist/data/quiz/function/handle_answer/merge_2.mcfunction +0 -8
  447. package/dist/data/quiz/function/handle_answer/merge_5.mcfunction +0 -2
  448. package/dist/data/quiz/function/handle_answer/merge_8.mcfunction +0 -2
  449. package/dist/data/quiz/function/handle_answer/then_0.mcfunction +0 -5
  450. package/dist/data/quiz/function/handle_answer/then_12.mcfunction +0 -5
  451. package/dist/data/quiz/function/handle_answer/then_15.mcfunction +0 -6
  452. package/dist/data/quiz/function/handle_answer/then_3.mcfunction +0 -6
  453. package/dist/data/quiz/function/handle_answer/then_6.mcfunction +0 -5
  454. package/dist/data/quiz/function/handle_answer/then_9.mcfunction +0 -6
  455. package/dist/data/quiz/function/handle_answer.mcfunction +0 -11
  456. package/dist/data/quiz/function/start_quiz.mcfunction +0 -5
  457. package/dist/data/reqtest/function/__load.mcfunction +0 -4
  458. package/dist/data/reqtest/function/_table_init.mcfunction +0 -2
  459. package/dist/data/reqtest/function/no_trig.mcfunction +0 -3
  460. package/dist/data/reqtest/function/use_table.mcfunction +0 -4
  461. package/dist/data/reqtest2/function/__load.mcfunction +0 -3
  462. package/dist/data/reqtest2/function/no_trig.mcfunction +0 -3
  463. package/dist/data/runtime/function/__load.mcfunction +0 -5
  464. package/dist/data/runtime/function/__tick.mcfunction +0 -2
  465. package/dist/data/runtime/function/counter_tick/then_0.mcfunction +0 -3
  466. package/dist/data/runtime/function/counter_tick.mcfunction +0 -13
  467. package/dist/data/shop/function/__load.mcfunction +0 -7
  468. package/dist/data/shop/function/__tick.mcfunction +0 -3
  469. package/dist/data/shop/function/__trigger_shop_buy_dispatch.mcfunction +0 -4
  470. package/dist/data/shop/function/complete_purchase/else_1.mcfunction +0 -5
  471. package/dist/data/shop/function/complete_purchase/else_4.mcfunction +0 -5
  472. package/dist/data/shop/function/complete_purchase/else_7.mcfunction +0 -3
  473. package/dist/data/shop/function/complete_purchase/merge_2.mcfunction +0 -2
  474. package/dist/data/shop/function/complete_purchase/merge_5.mcfunction +0 -2
  475. package/dist/data/shop/function/complete_purchase/merge_8.mcfunction +0 -2
  476. package/dist/data/shop/function/complete_purchase/then_0.mcfunction +0 -4
  477. package/dist/data/shop/function/complete_purchase/then_3.mcfunction +0 -4
  478. package/dist/data/shop/function/complete_purchase/then_6.mcfunction +0 -4
  479. package/dist/data/shop/function/complete_purchase.mcfunction +0 -7
  480. package/dist/data/shop/function/handle_shop_trigger.mcfunction +0 -3
  481. package/dist/data/swap_test/function/__load.mcfunction +0 -3
  482. package/dist/data/swap_test/function/gcd_old/loop_body_1.mcfunction +0 -7
  483. package/dist/data/swap_test/function/gcd_old/loop_check_0.mcfunction +0 -5
  484. package/dist/data/swap_test/function/gcd_old/loop_exit_2.mcfunction +0 -3
  485. package/dist/data/swap_test/function/gcd_old.mcfunction +0 -8
  486. package/dist/data/turret/function/__load.mcfunction +0 -5
  487. package/dist/data/turret/function/__tick.mcfunction +0 -4
  488. package/dist/data/turret/function/__trigger_deploy_turret_dispatch.mcfunction +0 -4
  489. package/dist/data/turret/function/deploy_turret.mcfunction +0 -8
  490. package/dist/data/turret/function/turret_tick/at_1.mcfunction +0 -2
  491. package/dist/data/turret/function/turret_tick/foreach_0.mcfunction +0 -2
  492. package/dist/data/turret/function/turret_tick/foreach_2.mcfunction +0 -2
  493. package/dist/data/turret/function/turret_tick/tick_body.mcfunction +0 -3
  494. package/dist/data/turret/function/turret_tick/tick_skip.mcfunction +0 -1
  495. package/dist/data/turret/function/turret_tick.mcfunction +0 -5
  496. package/dist/gcd2.map.json +0 -15
  497. package/dist/gcd3.map.json +0 -17
  498. package/dist/gcd_test.map.json +0 -15
  499. package/dist/index.d.ts +0 -62
  500. package/dist/ir/builder.d.ts +0 -33
  501. package/dist/ir/builder.js +0 -99
  502. package/dist/ir/types.d.ts +0 -132
  503. package/dist/ir/types.js +0 -15
  504. package/dist/isqrttest.map.json +0 -15
  505. package/dist/lowering/index.d.ts +0 -188
  506. package/dist/lowering/index.js +0 -3403
  507. package/dist/mathtest.map.json +0 -6
  508. package/dist/mypack.map.json +0 -27
  509. package/dist/optimizer/commands.d.ts +0 -38
  510. package/dist/optimizer/commands.js +0 -451
  511. package/dist/optimizer/dce.d.ts +0 -34
  512. package/dist/optimizer/dce.js +0 -639
  513. package/dist/optimizer/passes.d.ts +0 -34
  514. package/dist/optimizer/passes.js +0 -243
  515. package/dist/optimizer/structure.d.ts +0 -9
  516. package/dist/optimizer/structure.js +0 -356
  517. package/dist/pack.mcmeta +0 -6
  518. package/dist/reqtest.map.json +0 -4
  519. package/dist/reqtest2.map.json +0 -4
  520. package/dist/runtime.map.json +0 -7
  521. package/dist/swap_test.map.json +0 -14
  522. package/src/__tests__/codegen.test.ts +0 -161
  523. package/src/__tests__/e2e.test.ts +0 -2039
  524. package/src/__tests__/entity-types.test.ts +0 -236
  525. package/src/__tests__/lowering.test.ts +0 -1185
  526. package/src/__tests__/macro.test.ts +0 -343
  527. package/src/__tests__/nbt.test.ts +0 -58
  528. package/src/__tests__/optimizer-advanced.test.ts +0 -144
  529. package/src/__tests__/optimizer.test.ts +0 -162
  530. package/src/__tests__/runtime.test.ts +0 -305
  531. package/src/__tests__/stdlib-advanced.test.ts +0 -379
  532. package/src/__tests__/stdlib-bigint.test.ts +0 -427
  533. package/src/__tests__/stdlib-math.test.ts +0 -374
  534. package/src/__tests__/stdlib-vec.test.ts +0 -259
  535. package/src/__tests__/structure-optimizer.test.ts +0 -38
  536. package/src/__tests__/var-allocator.test.ts +0 -75
  537. package/src/codegen/cmdblock/index.ts +0 -63
  538. package/src/codegen/mcfunction/index.ts +0 -662
  539. package/src/codegen/structure/index.ts +0 -346
  540. package/src/codegen/var-allocator.ts +0 -104
  541. package/src/ir/builder.ts +0 -116
  542. package/src/ir/types.ts +0 -134
  543. package/src/lowering/index.ts +0 -3876
  544. package/src/optimizer/commands.ts +0 -534
  545. package/src/optimizer/passes.ts +0 -250
  546. package/src/optimizer/structure.ts +0 -450
  547. /package/dist/{__tests__/cli.test.d.ts → src/__tests__/budget.test.d.ts} +0 -0
  548. /package/dist/{__tests__/codegen.test.d.ts → src/__tests__/cli.test.d.ts} +0 -0
  549. /package/dist/{__tests__ → src/__tests__}/compile-all.test.d.ts +0 -0
  550. /package/dist/{__tests__ → src/__tests__}/compile-all.test.js +0 -0
  551. /package/dist/{__tests__ → src/__tests__}/dce.test.d.ts +0 -0
  552. /package/dist/{__tests__ → src/__tests__}/diagnostics.test.d.ts +0 -0
  553. /package/dist/{__tests__ → src/__tests__}/formatter.test.d.ts +0 -0
  554. /package/dist/{__tests__ → src/__tests__}/formatter.test.js +0 -0
  555. /package/dist/{__tests__/entity-types.test.d.ts → src/__tests__/hir/desugar.test.d.ts} +0 -0
  556. /package/dist/{__tests__ → src/__tests__}/lexer.test.d.ts +0 -0
  557. /package/dist/{__tests__ → src/__tests__}/lexer.test.js +0 -0
  558. /package/dist/{__tests__/lowering.test.d.ts → src/__tests__/lir/lower.test.d.ts} +0 -0
  559. /package/dist/{__tests__/mc-syntax.test.d.ts → src/__tests__/lir/types.test.d.ts} +0 -0
  560. /package/dist/{__tests__/nbt.test.d.ts → src/__tests__/lir/verify.test.d.ts} +0 -0
  561. /package/dist/{__tests__ → src/__tests__}/mc-integration.test.d.ts +0 -0
  562. /package/dist/{__tests__/optimizer-advanced.test.d.ts → src/__tests__/mc-syntax.test.d.ts} +0 -0
  563. /package/dist/{__tests__ → src/__tests__}/mc-syntax.test.js +0 -0
  564. /package/dist/{__tests__/optimizer.test.d.ts → src/__tests__/mir/arithmetic.test.d.ts} +0 -0
  565. /package/dist/{__tests__/parser.test.d.ts → src/__tests__/mir/control-flow.test.d.ts} +0 -0
  566. /package/dist/{__tests__/repl.test.d.ts → src/__tests__/mir/verify.test.d.ts} +0 -0
  567. /package/dist/{__tests__/runtime.test.d.ts → src/__tests__/optimizer/block_merge.test.d.ts} +0 -0
  568. /package/dist/{__tests__/structure-optimizer.test.d.ts → src/__tests__/optimizer/branch_simplify.test.d.ts} +0 -0
  569. /package/dist/{__tests__/typechecker.test.d.ts → src/__tests__/optimizer/constant_fold.test.d.ts} +0 -0
  570. /package/dist/{__tests__/var-allocator.test.d.ts → src/__tests__/optimizer/copy_prop.test.d.ts} +0 -0
  571. /package/dist/{__tests__ → src/__tests__}/parser.test.js +0 -0
  572. /package/dist/{__tests__ → src/__tests__}/repl.test.js +0 -0
  573. /package/dist/{__tests__ → src/__tests__}/typechecker.test.js +0 -0
  574. /package/dist/{ast → src/ast}/types.js +0 -0
  575. /package/dist/{builtins → src/builtins}/metadata.d.ts +0 -0
  576. /package/dist/{builtins → src/builtins}/metadata.js +0 -0
  577. /package/dist/{events → src/events}/types.d.ts +0 -0
  578. /package/dist/{events → src/events}/types.js +0 -0
  579. /package/dist/{formatter → src/formatter}/index.d.ts +0 -0
  580. /package/dist/{formatter → src/formatter}/index.js +0 -0
  581. /package/dist/{mc-test → src/mc-test}/client.d.ts +0 -0
  582. /package/dist/{mc-test → src/mc-test}/client.js +0 -0
  583. /package/dist/{mc-test → src/mc-test}/runner.d.ts +0 -0
  584. /package/dist/{mc-test → src/mc-test}/setup.d.ts +0 -0
  585. /package/dist/{mc-test → src/mc-test}/setup.js +0 -0
  586. /package/dist/{mc-validator → src/mc-validator}/index.d.ts +0 -0
  587. /package/dist/{mc-validator → src/mc-validator}/index.js +0 -0
  588. /package/dist/{nbt → src/nbt}/index.d.ts +0 -0
  589. /package/dist/{nbt → src/nbt}/index.js +0 -0
  590. /package/dist/{repl.js → src/repl.js} +0 -0
  591. /package/dist/{runtime → src/runtime}/index.d.ts +0 -0
  592. /package/dist/{types → src/types}/entity-hierarchy.d.ts +0 -0
  593. /package/dist/{types → src/types}/entity-hierarchy.js +0 -0
@@ -0,0 +1,619 @@
1
+ import { lowerToLIR } from '../../lir/lower'
2
+ import type { MIRModule, MIRFunction, MIRBlock, MIRInstr, Operand } from '../../mir/types'
3
+ import type { LIRInstr, Slot } from '../../lir/types'
4
+
5
+ // ---------------------------------------------------------------------------
6
+ // Helpers
7
+ // ---------------------------------------------------------------------------
8
+
9
+ const OBJ = '__test'
10
+ const NS = 'test'
11
+
12
+ function mkModule(functions: MIRFunction[]): MIRModule {
13
+ return { functions, namespace: NS, objective: OBJ }
14
+ }
15
+
16
+ function mkFn(
17
+ name: string,
18
+ blocks: MIRBlock[],
19
+ params: MIRFunction['params'] = [],
20
+ isMacro = false,
21
+ ): MIRFunction {
22
+ return { name, params, blocks, entry: 'entry', isMacro }
23
+ }
24
+
25
+ function mkBlock(id: string, instrs: MIRInstr[], term: MIRInstr, preds: string[] = []): MIRBlock {
26
+ return { id, instrs, term, preds }
27
+ }
28
+
29
+ const c = (v: number): Operand => ({ kind: 'const', value: v })
30
+ const t = (n: string): Operand => ({ kind: 'temp', name: n })
31
+
32
+ /** Build a slot with the function-name prefix that LIR lowering adds */
33
+ function slot(name: string, fn = 'main'): Slot {
34
+ return { player: `$${fn}_${name}`, obj: OBJ }
35
+ }
36
+
37
+ function findInstr(instrs: LIRInstr[], kind: string): LIRInstr | undefined {
38
+ return instrs.find(i => i.kind === kind)
39
+ }
40
+
41
+ function findAllInstrs(instrs: LIRInstr[], kind: string): LIRInstr[] {
42
+ return instrs.filter(i => i.kind === kind)
43
+ }
44
+
45
+ // ---------------------------------------------------------------------------
46
+ // Constant lowering
47
+ // ---------------------------------------------------------------------------
48
+
49
+ describe('MIR→LIR lowering — constants', () => {
50
+ test('const → score_set', () => {
51
+ const mod = mkModule([
52
+ mkFn('main', [
53
+ mkBlock('entry', [
54
+ { kind: 'const', dst: 't0', value: 42 },
55
+ ], { kind: 'return', value: null }),
56
+ ]),
57
+ ])
58
+ const lir = lowerToLIR(mod)
59
+ const main = lir.functions.find(f => f.name === 'main')!
60
+ expect(main.instructions[0]).toEqual({
61
+ kind: 'score_set',
62
+ dst: slot('t0'),
63
+ value: 42,
64
+ })
65
+ })
66
+ })
67
+
68
+ // ---------------------------------------------------------------------------
69
+ // Copy lowering
70
+ // ---------------------------------------------------------------------------
71
+
72
+ describe('MIR→LIR lowering — copy', () => {
73
+ test('copy temp → score_copy', () => {
74
+ const mod = mkModule([
75
+ mkFn('main', [
76
+ mkBlock('entry', [
77
+ { kind: 'const', dst: 't0', value: 5 },
78
+ { kind: 'copy', dst: 't1', src: t('t0') },
79
+ ], { kind: 'return', value: null }),
80
+ ]),
81
+ ])
82
+ const lir = lowerToLIR(mod)
83
+ const main = lir.functions.find(f => f.name === 'main')!
84
+ expect(main.instructions[1]).toEqual({
85
+ kind: 'score_copy',
86
+ dst: slot('t1'),
87
+ src: slot('t0'),
88
+ })
89
+ })
90
+
91
+ test('copy const → score_set', () => {
92
+ const mod = mkModule([
93
+ mkFn('main', [
94
+ mkBlock('entry', [
95
+ { kind: 'copy', dst: 't0', src: c(10) },
96
+ ], { kind: 'return', value: null }),
97
+ ]),
98
+ ])
99
+ const lir = lowerToLIR(mod)
100
+ const main = lir.functions.find(f => f.name === 'main')!
101
+ expect(main.instructions[0]).toEqual({
102
+ kind: 'score_set',
103
+ dst: slot('t0'),
104
+ value: 10,
105
+ })
106
+ })
107
+ })
108
+
109
+ // ---------------------------------------------------------------------------
110
+ // Arithmetic lowering
111
+ // ---------------------------------------------------------------------------
112
+
113
+ describe('MIR→LIR lowering — arithmetic', () => {
114
+ test('add temps → score_copy + score_add', () => {
115
+ const mod = mkModule([
116
+ mkFn('main', [
117
+ mkBlock('entry', [
118
+ { kind: 'const', dst: 'a', value: 3 },
119
+ { kind: 'const', dst: 'b', value: 4 },
120
+ { kind: 'add', dst: 'r', a: t('a'), b: t('b') },
121
+ ], { kind: 'return', value: null }),
122
+ ]),
123
+ ])
124
+ const lir = lowerToLIR(mod)
125
+ const main = lir.functions.find(f => f.name === 'main')!
126
+ // After two score_sets for a and b, we should see:
127
+ // score_copy $r = $a
128
+ // score_add $r += $b
129
+ const instrs = main.instructions
130
+ expect(instrs[2]).toEqual({ kind: 'score_copy', dst: slot('r'), src: slot('a') })
131
+ expect(instrs[3]).toEqual({ kind: 'score_add', dst: slot('r'), src: slot('b') })
132
+ })
133
+
134
+ test('add with const operand → score_set + score_copy + score_set const + score_add', () => {
135
+ const mod = mkModule([
136
+ mkFn('main', [
137
+ mkBlock('entry', [
138
+ { kind: 'const', dst: 'a', value: 3 },
139
+ { kind: 'add', dst: 'r', a: t('a'), b: c(7) },
140
+ ], { kind: 'return', value: null }),
141
+ ]),
142
+ ])
143
+ const lir = lowerToLIR(mod)
144
+ const main = lir.functions.find(f => f.name === 'main')!
145
+ const instrs = main.instructions
146
+ // score_set $a 3
147
+ // score_copy $r = $a
148
+ // score_set $__const_7 7
149
+ // score_add $r += $__const_7
150
+ expect(instrs[1]).toEqual({ kind: 'score_copy', dst: slot('r'), src: slot('a') })
151
+ expect(instrs[2]).toEqual({ kind: 'score_set', dst: { player: '$__const_7', obj: OBJ }, value: 7 })
152
+ expect(instrs[3]).toEqual({ kind: 'score_add', dst: slot('r'), src: { player: '$__const_7', obj: OBJ } })
153
+ })
154
+
155
+ test('sub, mul, div, mod all produce correct score ops', () => {
156
+ const ops = ['sub', 'mul', 'div', 'mod'] as const
157
+ const scoreOps = ['score_sub', 'score_mul', 'score_div', 'score_mod'] as const
158
+
159
+ for (let i = 0; i < ops.length; i++) {
160
+ const mod = mkModule([
161
+ mkFn('main', [
162
+ mkBlock('entry', [
163
+ { kind: 'const', dst: 'a', value: 10 },
164
+ { kind: 'const', dst: 'b', value: 3 },
165
+ { kind: ops[i], dst: 'r', a: t('a'), b: t('b') } as MIRInstr,
166
+ ], { kind: 'return', value: null }),
167
+ ]),
168
+ ])
169
+ const lir = lowerToLIR(mod)
170
+ const main = lir.functions.find(f => f.name === 'main')!
171
+ const instrs = main.instructions
172
+ expect(instrs[2]).toEqual({ kind: 'score_copy', dst: slot('r'), src: slot('a') })
173
+ expect(instrs[3]).toEqual({ kind: scoreOps[i], dst: slot('r'), src: slot('b') })
174
+ }
175
+ })
176
+ })
177
+
178
+ // ---------------------------------------------------------------------------
179
+ // Negation
180
+ // ---------------------------------------------------------------------------
181
+
182
+ describe('MIR→LIR lowering — negation', () => {
183
+ test('neg → score_set 0, score_sub', () => {
184
+ const mod = mkModule([
185
+ mkFn('main', [
186
+ mkBlock('entry', [
187
+ { kind: 'const', dst: 'x', value: 5 },
188
+ { kind: 'neg', dst: 'r', src: t('x') },
189
+ ], { kind: 'return', value: null }),
190
+ ]),
191
+ ])
192
+ const lir = lowerToLIR(mod)
193
+ const main = lir.functions.find(f => f.name === 'main')!
194
+ const instrs = main.instructions
195
+ // score_set $x 5
196
+ // score_set $r 0
197
+ // score_sub $r -= $x
198
+ expect(instrs[1]).toEqual({ kind: 'score_set', dst: slot('r'), value: 0 })
199
+ expect(instrs[2]).toEqual({ kind: 'score_sub', dst: slot('r'), src: slot('x') })
200
+ })
201
+ })
202
+
203
+ // ---------------------------------------------------------------------------
204
+ // Comparison
205
+ // ---------------------------------------------------------------------------
206
+
207
+ describe('MIR→LIR lowering — comparison', () => {
208
+ test('cmp eq → score_set 0, raw execute if score', () => {
209
+ const mod = mkModule([
210
+ mkFn('main', [
211
+ mkBlock('entry', [
212
+ { kind: 'const', dst: 'a', value: 1 },
213
+ { kind: 'const', dst: 'b', value: 2 },
214
+ { kind: 'cmp', dst: 'r', op: 'eq', a: t('a'), b: t('b') },
215
+ ], { kind: 'return', value: null }),
216
+ ]),
217
+ ])
218
+ const lir = lowerToLIR(mod)
219
+ const main = lir.functions.find(f => f.name === 'main')!
220
+ const instrs = main.instructions
221
+ // score_set $r 0, then raw execute if score
222
+ const setCmp = instrs.find(i => i.kind === 'score_set' && (i as any).dst.player === '$main_r')
223
+ expect(setCmp).toEqual({ kind: 'score_set', dst: slot('r'), value: 0 })
224
+ const raw = instrs.find(i => i.kind === 'raw') as any
225
+ expect(raw).toBeDefined()
226
+ expect(raw.cmd).toContain('execute if score')
227
+ expect(raw.cmd).toContain('$main_a')
228
+ expect(raw.cmd).toContain('$main_b')
229
+ expect(raw.cmd).toContain('scoreboard players set $main_r')
230
+ })
231
+ })
232
+
233
+ // ---------------------------------------------------------------------------
234
+ // Boolean logic
235
+ // ---------------------------------------------------------------------------
236
+
237
+ describe('MIR→LIR lowering — boolean logic', () => {
238
+ test('and → score_copy + score_mul', () => {
239
+ const mod = mkModule([
240
+ mkFn('main', [
241
+ mkBlock('entry', [
242
+ { kind: 'const', dst: 'a', value: 1 },
243
+ { kind: 'const', dst: 'b', value: 1 },
244
+ { kind: 'and', dst: 'r', a: t('a'), b: t('b') },
245
+ ], { kind: 'return', value: null }),
246
+ ]),
247
+ ])
248
+ const lir = lowerToLIR(mod)
249
+ const main = lir.functions.find(f => f.name === 'main')!
250
+ const instrs = main.instructions
251
+ expect(instrs[2]).toEqual({ kind: 'score_copy', dst: slot('r'), src: slot('a') })
252
+ expect(instrs[3]).toEqual({ kind: 'score_mul', dst: slot('r'), src: slot('b') })
253
+ })
254
+
255
+ test('not → score_set 1, score_sub', () => {
256
+ const mod = mkModule([
257
+ mkFn('main', [
258
+ mkBlock('entry', [
259
+ { kind: 'const', dst: 'x', value: 1 },
260
+ { kind: 'not', dst: 'r', src: t('x') },
261
+ ], { kind: 'return', value: null }),
262
+ ]),
263
+ ])
264
+ const lir = lowerToLIR(mod)
265
+ const main = lir.functions.find(f => f.name === 'main')!
266
+ const instrs = main.instructions
267
+ // score_set $x 1
268
+ // score_set $r 1
269
+ // score_sub $r -= $x
270
+ expect(instrs[1]).toEqual({ kind: 'score_set', dst: slot('r'), value: 1 })
271
+ expect(instrs[2]).toEqual({ kind: 'score_sub', dst: slot('r'), src: slot('x') })
272
+ })
273
+
274
+ test('or → score_copy + score_add + score_min(1)', () => {
275
+ const mod = mkModule([
276
+ mkFn('main', [
277
+ mkBlock('entry', [
278
+ { kind: 'const', dst: 'a', value: 1 },
279
+ { kind: 'const', dst: 'b', value: 0 },
280
+ { kind: 'or', dst: 'r', a: t('a'), b: t('b') },
281
+ ], { kind: 'return', value: null }),
282
+ ]),
283
+ ])
284
+ const lir = lowerToLIR(mod)
285
+ const main = lir.functions.find(f => f.name === 'main')!
286
+ const instrs = main.instructions
287
+ // score_set $a 1, score_set $b 0
288
+ // score_copy $r = $a
289
+ // score_add $r += $b
290
+ // score_set $__const_1 1
291
+ // score_min $r, $__const_1
292
+ expect(instrs[2]).toEqual({ kind: 'score_copy', dst: slot('r'), src: slot('a') })
293
+ expect(instrs[3]).toEqual({ kind: 'score_add', dst: slot('r'), src: slot('b') })
294
+ const minInstr = instrs.find(i => i.kind === 'score_min') as any
295
+ expect(minInstr).toBeDefined()
296
+ expect(minInstr.dst).toEqual(slot('r'))
297
+ })
298
+ })
299
+
300
+ // ---------------------------------------------------------------------------
301
+ // NBT operations
302
+ // ---------------------------------------------------------------------------
303
+
304
+ describe('MIR→LIR lowering — NBT', () => {
305
+ test('nbt_read → store_nbt_to_score', () => {
306
+ const mod = mkModule([
307
+ mkFn('main', [
308
+ mkBlock('entry', [
309
+ { kind: 'nbt_read', dst: 't0', ns: 'rs:data', path: 'val', scale: 1 },
310
+ ], { kind: 'return', value: null }),
311
+ ]),
312
+ ])
313
+ const lir = lowerToLIR(mod)
314
+ const main = lir.functions.find(f => f.name === 'main')!
315
+ expect(main.instructions[0]).toEqual({
316
+ kind: 'store_nbt_to_score',
317
+ dst: slot('t0'),
318
+ ns: 'rs:data',
319
+ path: 'val',
320
+ scale: 1,
321
+ })
322
+ })
323
+
324
+ test('nbt_write → store_score_to_nbt', () => {
325
+ const mod = mkModule([
326
+ mkFn('main', [
327
+ mkBlock('entry', [
328
+ { kind: 'const', dst: 't0', value: 10 },
329
+ { kind: 'nbt_write', ns: 'rs:data', path: 'val', type: 'int', scale: 1, src: t('t0') },
330
+ ], { kind: 'return', value: null }),
331
+ ]),
332
+ ])
333
+ const lir = lowerToLIR(mod)
334
+ const main = lir.functions.find(f => f.name === 'main')!
335
+ expect(main.instructions[1]).toEqual({
336
+ kind: 'store_score_to_nbt',
337
+ ns: 'rs:data',
338
+ path: 'val',
339
+ type: 'int',
340
+ scale: 1,
341
+ src: slot('t0'),
342
+ })
343
+ })
344
+ })
345
+
346
+ // ---------------------------------------------------------------------------
347
+ // Function calls
348
+ // ---------------------------------------------------------------------------
349
+
350
+ describe('MIR→LIR lowering — calls', () => {
351
+ test('call with args → set $p0, $p1, call, copy $ret', () => {
352
+ const mod = mkModule([
353
+ mkFn('main', [
354
+ mkBlock('entry', [
355
+ { kind: 'const', dst: 'a', value: 1 },
356
+ { kind: 'const', dst: 'b', value: 2 },
357
+ { kind: 'call', dst: 'r', fn: 'add', args: [t('a'), t('b')] },
358
+ ], { kind: 'return', value: null }),
359
+ ]),
360
+ mkFn('add', [
361
+ mkBlock('entry', [], { kind: 'return', value: null }),
362
+ ]),
363
+ ])
364
+ const lir = lowerToLIR(mod)
365
+ const main = lir.functions.find(f => f.name === 'main')!
366
+ const instrs = main.instructions
367
+
368
+ // $p0 = $a
369
+ expect(instrs[2]).toEqual({
370
+ kind: 'score_copy',
371
+ dst: { player: '$p0', obj: OBJ },
372
+ src: slot('a'),
373
+ })
374
+ // $p1 = $b
375
+ expect(instrs[3]).toEqual({
376
+ kind: 'score_copy',
377
+ dst: { player: '$p1', obj: OBJ },
378
+ src: slot('b'),
379
+ })
380
+ // call test:add
381
+ expect(instrs[4]).toEqual({ kind: 'call', fn: 'test:add' })
382
+ // $r = $ret
383
+ expect(instrs[5]).toEqual({
384
+ kind: 'score_copy',
385
+ dst: slot('r'),
386
+ src: { player: '$ret', obj: OBJ },
387
+ })
388
+ })
389
+
390
+ test('call with no dst does not copy $ret', () => {
391
+ const mod = mkModule([
392
+ mkFn('main', [
393
+ mkBlock('entry', [
394
+ { kind: 'call', dst: null, fn: 'side_effect', args: [] },
395
+ ], { kind: 'return', value: null }),
396
+ ]),
397
+ mkFn('side_effect', [
398
+ mkBlock('entry', [], { kind: 'return', value: null }),
399
+ ]),
400
+ ])
401
+ const lir = lowerToLIR(mod)
402
+ const main = lir.functions.find(f => f.name === 'main')!
403
+ expect(main.instructions).toEqual([
404
+ { kind: 'call', fn: 'test:side_effect' },
405
+ ])
406
+ })
407
+
408
+ test('raw command via __raw: prefix', () => {
409
+ const mod = mkModule([
410
+ mkFn('main', [
411
+ mkBlock('entry', [
412
+ { kind: 'call', dst: null, fn: '__raw:say hello', args: [] },
413
+ ], { kind: 'return', value: null }),
414
+ ]),
415
+ ])
416
+ const lir = lowerToLIR(mod)
417
+ const main = lir.functions.find(f => f.name === 'main')!
418
+ expect(main.instructions[0]).toEqual({ kind: 'raw', cmd: 'say hello' })
419
+ })
420
+ })
421
+
422
+ // ---------------------------------------------------------------------------
423
+ // Macro calls
424
+ // ---------------------------------------------------------------------------
425
+
426
+ describe('MIR→LIR lowering — macro calls', () => {
427
+ test('call_macro → store args to NBT + call_macro', () => {
428
+ const mod = mkModule([
429
+ mkFn('main', [
430
+ mkBlock('entry', [
431
+ { kind: 'const', dst: 'px', value: 100 },
432
+ {
433
+ kind: 'call_macro',
434
+ dst: null,
435
+ fn: 'draw_pt',
436
+ args: [{ name: 'px', value: t('px'), type: 'int' as const, scale: 1 }],
437
+ },
438
+ ], { kind: 'return', value: null }),
439
+ ]),
440
+ mkFn('draw_pt', [
441
+ mkBlock('entry', [], { kind: 'return', value: null }),
442
+ ], [{ name: 'px', isMacroParam: true }], true),
443
+ ])
444
+ const lir = lowerToLIR(mod)
445
+ const main = lir.functions.find(f => f.name === 'main')!
446
+ const instrs = main.instructions
447
+
448
+ // store_score_to_nbt for px
449
+ const storeNbt = instrs.find(i => i.kind === 'store_score_to_nbt') as any
450
+ expect(storeNbt).toBeDefined()
451
+ expect(storeNbt.ns).toBe('rs:macro_args')
452
+ expect(storeNbt.path).toBe('px')
453
+
454
+ // call_macro
455
+ const callMacro = instrs.find(i => i.kind === 'call_macro') as any
456
+ expect(callMacro).toBeDefined()
457
+ expect(callMacro.fn).toBe('test:draw_pt')
458
+ expect(callMacro.storage).toBe('rs:macro_args')
459
+ })
460
+ })
461
+
462
+ // ---------------------------------------------------------------------------
463
+ // Context calls
464
+ // ---------------------------------------------------------------------------
465
+
466
+ describe('MIR→LIR lowering — context calls', () => {
467
+ test('call_context → call_context with qualified name', () => {
468
+ const mod = mkModule([
469
+ mkFn('main', [
470
+ mkBlock('entry', [
471
+ {
472
+ kind: 'call_context',
473
+ fn: 'helper',
474
+ subcommands: [{ kind: 'as', selector: '@e[tag=foo]' }, { kind: 'at_self' }],
475
+ },
476
+ ], { kind: 'return', value: null }),
477
+ ]),
478
+ mkFn('helper', [
479
+ mkBlock('entry', [], { kind: 'return', value: null }),
480
+ ]),
481
+ ])
482
+ const lir = lowerToLIR(mod)
483
+ const main = lir.functions.find(f => f.name === 'main')!
484
+ expect(main.instructions[0]).toEqual({
485
+ kind: 'call_context',
486
+ fn: 'test:helper',
487
+ subcommands: [{ kind: 'as', selector: '@e[tag=foo]' }, { kind: 'at_self' }],
488
+ })
489
+ })
490
+ })
491
+
492
+ // ---------------------------------------------------------------------------
493
+ // Return
494
+ // ---------------------------------------------------------------------------
495
+
496
+ describe('MIR→LIR lowering — return', () => {
497
+ test('return with value → return_value', () => {
498
+ const mod = mkModule([
499
+ mkFn('main', [
500
+ mkBlock('entry', [
501
+ { kind: 'const', dst: 't0', value: 42 },
502
+ ], { kind: 'return', value: t('t0') }),
503
+ ]),
504
+ ])
505
+ const lir = lowerToLIR(mod)
506
+ const main = lir.functions.find(f => f.name === 'main')!
507
+ const ret = main.instructions.find(i => i.kind === 'return_value') as any
508
+ expect(ret).toBeDefined()
509
+ expect(ret.slot).toEqual(slot('t0'))
510
+ })
511
+
512
+ test('return void → no return_value instruction', () => {
513
+ const mod = mkModule([
514
+ mkFn('main', [
515
+ mkBlock('entry', [], { kind: 'return', value: null }),
516
+ ]),
517
+ ])
518
+ const lir = lowerToLIR(mod)
519
+ const main = lir.functions.find(f => f.name === 'main')!
520
+ expect(main.instructions.find(i => i.kind === 'return_value')).toBeUndefined()
521
+ })
522
+ })
523
+
524
+ // ---------------------------------------------------------------------------
525
+ // Control flow — jump
526
+ // ---------------------------------------------------------------------------
527
+
528
+ describe('MIR→LIR lowering — jump (inlining)', () => {
529
+ test('jump to single-pred block inlines instructions', () => {
530
+ const mod = mkModule([
531
+ mkFn('main', [
532
+ mkBlock('entry', [
533
+ { kind: 'const', dst: 't0', value: 1 },
534
+ ], { kind: 'jump', target: 'next' }),
535
+ mkBlock('next', [
536
+ { kind: 'const', dst: 't1', value: 2 },
537
+ ], { kind: 'return', value: null }),
538
+ ]),
539
+ ])
540
+ const lir = lowerToLIR(mod)
541
+ const main = lir.functions.find(f => f.name === 'main')!
542
+ // Both blocks inlined: score_set t0, score_set t1
543
+ expect(main.instructions).toEqual([
544
+ { kind: 'score_set', dst: slot('t0'), value: 1 },
545
+ { kind: 'score_set', dst: slot('t1'), value: 2 },
546
+ ])
547
+ })
548
+ })
549
+
550
+ // ---------------------------------------------------------------------------
551
+ // Control flow — branch
552
+ // ---------------------------------------------------------------------------
553
+
554
+ describe('MIR→LIR lowering — branch', () => {
555
+ test('branch emits call_if_matches and call_unless_matches', () => {
556
+ const mod = mkModule([
557
+ mkFn('main', [
558
+ mkBlock('entry', [
559
+ { kind: 'const', dst: 'cond', value: 1 },
560
+ ], { kind: 'branch', cond: t('cond'), then: 'yes', else: 'no' }),
561
+ mkBlock('yes', [
562
+ { kind: 'const', dst: 'a', value: 10 },
563
+ ], { kind: 'return', value: null }),
564
+ mkBlock('no', [
565
+ { kind: 'const', dst: 'b', value: 20 },
566
+ ], { kind: 'return', value: null }),
567
+ ]),
568
+ ])
569
+ const lir = lowerToLIR(mod)
570
+ const main = lir.functions.find(f => f.name === 'main')!
571
+
572
+ // Then-branch uses `return run function` (raw instruction) to prevent
573
+ // fallthrough when recursive calls clobber the condition slot.
574
+ const rawReturn = main.instructions.find(
575
+ i => i.kind === 'raw' && (i as any).cmd.includes('return run function'),
576
+ ) as any
577
+ expect(rawReturn).toBeDefined()
578
+ expect(rawReturn.cmd).toContain('matches 1')
579
+
580
+ // Else-branch is an unconditional call (fallthrough only reached when cond≠1)
581
+ const elseCall = main.instructions.find(i => i.kind === 'call') as any
582
+ expect(elseCall).toBeDefined()
583
+
584
+ // The then/else blocks should be separate functions
585
+ const yesFn = lir.functions.find(f => f.name.includes('yes'))
586
+ const noFn = lir.functions.find(f => f.name.includes('no'))
587
+ expect(yesFn).toBeDefined()
588
+ expect(noFn).toBeDefined()
589
+ })
590
+ })
591
+
592
+ // ---------------------------------------------------------------------------
593
+ // Module structure
594
+ // ---------------------------------------------------------------------------
595
+
596
+ describe('MIR→LIR lowering — module', () => {
597
+ test('preserves namespace and objective', () => {
598
+ const mod = mkModule([
599
+ mkFn('main', [
600
+ mkBlock('entry', [], { kind: 'return', value: null }),
601
+ ]),
602
+ ])
603
+ const lir = lowerToLIR(mod)
604
+ expect(lir.namespace).toBe(NS)
605
+ expect(lir.objective).toBe(OBJ)
606
+ })
607
+
608
+ test('macro function carries isMacro and macroParams', () => {
609
+ const mod = mkModule([
610
+ mkFn('draw', [
611
+ mkBlock('entry', [], { kind: 'return', value: null }),
612
+ ], [{ name: 'px', isMacroParam: true }, { name: 'py', isMacroParam: true }], true),
613
+ ])
614
+ const lir = lowerToLIR(mod)
615
+ const drawFn = lir.functions.find(f => f.name === 'draw')!
616
+ expect(drawFn.isMacro).toBe(true)
617
+ expect(drawFn.macroParams).toEqual(['px', 'py'])
618
+ })
619
+ })