@devaloop/devalang 0.0.1-alpha.9 → 0.0.1-beta.2

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 (322) hide show
  1. package/.cargo/config.toml +2 -0
  2. package/.devalang +6 -1
  3. package/.github/workflows/ci.yml +103 -0
  4. package/Cargo.toml +81 -48
  5. package/README.md +137 -154
  6. package/docs/CHANGELOG.md +428 -1
  7. package/docs/CONTRIBUTING.md +101 -0
  8. package/docs/ROADMAP.md +14 -7
  9. package/docs/TODO.md +16 -15
  10. package/examples/automation.deva +42 -0
  11. package/examples/bank.deva +7 -0
  12. package/examples/bus.deva +10 -0
  13. package/examples/duration.deva +9 -0
  14. package/examples/effect.deva +2 -0
  15. package/examples/events.deva +12 -0
  16. package/examples/filter.deva +11 -0
  17. package/examples/function.deva +15 -0
  18. package/examples/index.deva +57 -12
  19. package/examples/lfo.deva +9 -0
  20. package/examples/loop.deva +5 -12
  21. package/examples/pattern.deva +8 -0
  22. package/examples/plugin.deva +16 -0
  23. package/examples/synth.deva +11 -1
  24. package/examples/synth_types.deva +17 -0
  25. package/examples/variables.deva +1 -1
  26. package/out-tsc/bin/index.d.ts +2 -0
  27. package/out-tsc/bin/index.js +51 -7
  28. package/out-tsc/core/functions/index.d.ts +42 -0
  29. package/out-tsc/core/functions/index.js +87 -0
  30. package/out-tsc/core/index.d.ts +6 -0
  31. package/out-tsc/core/index.js +22 -0
  32. package/out-tsc/core/types/index.d.ts +4 -0
  33. package/out-tsc/core/types/index.js +20 -0
  34. package/out-tsc/core/types/plugin.d.ts +18 -0
  35. package/out-tsc/core/types/plugin.js +2 -0
  36. package/out-tsc/core/types/result.d.ts +27 -0
  37. package/out-tsc/core/types/result.js +2 -0
  38. package/out-tsc/core/types/statement.d.ts +106 -0
  39. package/out-tsc/core/types/statement.js +2 -0
  40. package/out-tsc/core/types/value.d.ts +43 -0
  41. package/out-tsc/core/types/value.js +2 -0
  42. package/out-tsc/index.d.ts +7 -0
  43. package/out-tsc/index.js +42 -1
  44. package/out-tsc/pkg/devalang_core.d.ts +15 -0
  45. package/out-tsc/pkg/devalang_core.js +65 -0
  46. package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +34 -0
  47. package/out-tsc/scripts/copy-wasm-dts.d.ts +1 -0
  48. package/out-tsc/scripts/copy-wasm-dts.js +73 -0
  49. package/out-tsc/scripts/postinstall.d.ts +1 -0
  50. package/out-tsc/scripts/postinstall.js +83 -0
  51. package/out-tsc/scripts/version/bump.d.ts +1 -0
  52. package/out-tsc/scripts/version/fetch.d.ts +1 -0
  53. package/out-tsc/scripts/version/index.d.ts +1 -0
  54. package/out-tsc/scripts/version/sync.d.ts +1 -0
  55. package/package.json +28 -7
  56. package/project-version.json +4 -4
  57. package/rust/cli/bank/api.rs +122 -0
  58. package/rust/cli/bank/commands.rs +306 -0
  59. package/rust/cli/bank/mod.rs +29 -0
  60. package/rust/cli/build/commands.rs +153 -0
  61. package/rust/cli/build/mod.rs +2 -0
  62. package/rust/cli/build/process.rs +165 -0
  63. package/rust/cli/check/mod.rs +208 -0
  64. package/rust/cli/discover/commands.rs +253 -0
  65. package/rust/cli/discover/config.rs +111 -0
  66. package/rust/cli/discover/fs.rs +19 -0
  67. package/rust/cli/discover/install.rs +103 -0
  68. package/rust/cli/discover/metadata.rs +48 -0
  69. package/rust/cli/discover/mod.rs +5 -0
  70. package/rust/cli/{init.rs → init/commands.rs} +32 -23
  71. package/rust/cli/init/mod.rs +1 -0
  72. package/rust/cli/install/addon.rs +118 -0
  73. package/rust/cli/install/bank.rs +72 -0
  74. package/rust/cli/install/commands.rs +35 -0
  75. package/rust/cli/install/mod.rs +4 -0
  76. package/rust/cli/install/plugin.rs +80 -0
  77. package/rust/cli/login/commands.rs +124 -0
  78. package/rust/cli/login/mod.rs +1 -0
  79. package/rust/cli/mod.rs +9 -202
  80. package/rust/cli/parser.rs +359 -0
  81. package/rust/cli/play/commands.rs +375 -0
  82. package/rust/cli/play/io.rs +17 -0
  83. package/rust/cli/play/mod.rs +5 -0
  84. package/rust/cli/play/process.rs +159 -0
  85. package/rust/cli/play/realtime.rs +91 -0
  86. package/rust/cli/play/utils.rs +23 -0
  87. package/rust/cli/telemetry/commands.rs +22 -0
  88. package/rust/cli/telemetry/event_creator.rs +80 -0
  89. package/rust/cli/telemetry/mod.rs +3 -0
  90. package/rust/cli/telemetry/send.rs +51 -0
  91. package/rust/cli/{template.rs → template/commands.rs} +17 -5
  92. package/rust/cli/template/mod.rs +1 -0
  93. package/rust/cli/update/commands.rs +6 -0
  94. package/rust/cli/update/mod.rs +1 -0
  95. package/rust/config/driver.rs +112 -0
  96. package/rust/config/mod.rs +3 -16
  97. package/rust/config/ops.rs +26 -0
  98. package/rust/config/settings.rs +101 -0
  99. package/rust/core/audio/engine/driver.rs +220 -0
  100. package/rust/core/audio/engine/export.rs +169 -0
  101. package/rust/core/audio/engine/helpers.rs +178 -0
  102. package/rust/core/audio/engine/mod.rs +56 -0
  103. package/rust/core/audio/engine/notes/dsp.rs +85 -0
  104. package/rust/core/audio/engine/notes/mod.rs +44 -0
  105. package/rust/core/audio/engine/notes/params.rs +294 -0
  106. package/rust/core/audio/engine/sample/insert.rs +199 -0
  107. package/rust/core/audio/engine/sample/mod.rs +40 -0
  108. package/rust/core/audio/engine/sample/padding.rs +170 -0
  109. package/rust/core/audio/evaluator/condition.rs +61 -0
  110. package/rust/core/audio/evaluator/mod.rs +9 -0
  111. package/rust/core/audio/evaluator/numeric.rs +152 -0
  112. package/rust/core/audio/evaluator/rhs.rs +16 -0
  113. package/rust/core/audio/evaluator/string_expr.rs +94 -0
  114. package/rust/core/audio/interpreter/driver.rs +574 -216
  115. package/rust/core/audio/interpreter/mod.rs +2 -12
  116. package/rust/core/audio/interpreter/statements/arrow_call/interprete.rs +175 -0
  117. package/rust/core/audio/interpreter/statements/arrow_call/methods/chord.rs +384 -0
  118. package/rust/core/audio/interpreter/statements/arrow_call/methods/mod.rs +2 -0
  119. package/rust/core/audio/interpreter/statements/arrow_call/methods/note.rs +316 -0
  120. package/rust/core/audio/interpreter/statements/arrow_call/mod.rs +3 -0
  121. package/rust/core/audio/interpreter/statements/arrow_call/types/arp.rs +192 -0
  122. package/rust/core/audio/interpreter/statements/arrow_call/types/mod.rs +24 -0
  123. package/rust/core/audio/interpreter/statements/arrow_call/types/pad.rs +116 -0
  124. package/rust/core/audio/interpreter/statements/arrow_call/types/pluck.rs +97 -0
  125. package/rust/core/audio/interpreter/statements/arrow_call/types/sub.rs +100 -0
  126. package/rust/core/audio/interpreter/statements/automate.rs +16 -0
  127. package/rust/core/audio/interpreter/statements/call.rs +295 -0
  128. package/rust/core/audio/interpreter/{condition.rs → statements/condition.rs} +72 -69
  129. package/rust/core/audio/interpreter/statements/function.rs +24 -0
  130. package/rust/core/audio/interpreter/statements/let_.rs +36 -0
  131. package/rust/core/audio/interpreter/statements/load.rs +17 -0
  132. package/rust/core/audio/interpreter/statements/loop_.rs +115 -0
  133. package/rust/core/audio/interpreter/statements/mod.rs +12 -0
  134. package/rust/core/audio/interpreter/statements/sleep.rs +28 -0
  135. package/rust/core/audio/interpreter/statements/spawn.rs +253 -0
  136. package/rust/core/audio/interpreter/statements/tempo.rs +40 -0
  137. package/rust/core/audio/interpreter/statements/trigger.rs +239 -0
  138. package/rust/core/audio/loader/mod.rs +1 -1
  139. package/rust/core/audio/loader/trigger.rs +98 -52
  140. package/rust/core/audio/mod.rs +2 -2
  141. package/rust/core/audio/player.rs +28 -12
  142. package/rust/core/audio/special/easing.rs +189 -0
  143. package/rust/core/audio/special/env.rs +45 -0
  144. package/rust/core/audio/special/math.rs +134 -0
  145. package/rust/core/audio/special/mod.rs +9 -0
  146. package/rust/core/audio/special/modulator.rs +143 -0
  147. package/rust/core/builder/mod.rs +129 -80
  148. package/rust/core/debugger/lexer.rs +4 -4
  149. package/rust/core/debugger/logs.rs +52 -0
  150. package/rust/core/debugger/mod.rs +11 -2
  151. package/rust/core/debugger/preprocessor.rs +4 -4
  152. package/rust/core/debugger/store.rs +38 -25
  153. package/rust/core/error/mod.rs +221 -12
  154. package/rust/core/lexer/driver.rs +59 -0
  155. package/rust/core/lexer/handler/arrow.rs +62 -11
  156. package/rust/core/lexer/handler/at.rs +5 -5
  157. package/rust/core/lexer/handler/brace.rs +11 -11
  158. package/rust/core/lexer/handler/colon.rs +5 -5
  159. package/rust/core/lexer/handler/comment.rs +3 -3
  160. package/rust/core/lexer/handler/dot.rs +6 -6
  161. package/rust/core/lexer/handler/driver.rs +143 -32
  162. package/rust/core/lexer/handler/identifier.rs +11 -5
  163. package/rust/core/lexer/handler/indent.rs +18 -4
  164. package/rust/core/lexer/handler/mod.rs +6 -5
  165. package/rust/core/lexer/handler/newline.rs +3 -3
  166. package/rust/core/lexer/handler/number.rs +5 -5
  167. package/rust/core/lexer/handler/operator.rs +5 -3
  168. package/rust/core/lexer/handler/parenthesis.rs +41 -0
  169. package/rust/core/lexer/handler/slash.rs +21 -0
  170. package/rust/core/lexer/handler/string.rs +3 -3
  171. package/rust/core/lexer/mod.rs +1 -49
  172. package/rust/core/lexer/token.rs +17 -12
  173. package/rust/core/mod.rs +9 -10
  174. package/rust/core/parser/driver/block.rs +111 -0
  175. package/rust/core/parser/driver/cursor.rs +82 -0
  176. package/rust/core/parser/driver/driver_impl.rs +139 -0
  177. package/rust/core/parser/driver/mod.rs +6 -0
  178. package/rust/core/parser/driver/parse_array.rs +120 -0
  179. package/rust/core/parser/driver/parse_map.rs +223 -0
  180. package/rust/core/parser/driver/parser.rs +160 -0
  181. package/rust/core/parser/handler/arrow_call.rs +277 -126
  182. package/rust/core/parser/handler/at.rs +142 -25
  183. package/rust/core/parser/handler/bank.rs +83 -20
  184. package/rust/core/parser/handler/condition.rs +14 -5
  185. package/rust/core/parser/handler/dot.rs +111 -75
  186. package/rust/core/parser/handler/identifier/automate.rs +254 -0
  187. package/rust/core/parser/handler/identifier/call.rs +74 -24
  188. package/rust/core/parser/handler/identifier/emit.rs +70 -0
  189. package/rust/core/parser/handler/identifier/function.rs +113 -0
  190. package/rust/core/parser/handler/identifier/group.rs +28 -14
  191. package/rust/core/parser/handler/identifier/let_.rs +61 -21
  192. package/rust/core/parser/handler/identifier/mod.rs +24 -20
  193. package/rust/core/parser/handler/identifier/on.rs +107 -0
  194. package/rust/core/parser/handler/identifier/print.rs +49 -0
  195. package/rust/core/parser/handler/identifier/sleep.rs +77 -14
  196. package/rust/core/parser/handler/identifier/spawn.rs +81 -31
  197. package/rust/core/parser/handler/identifier/synth.rs +102 -32
  198. package/rust/core/parser/handler/loop_.rs +144 -22
  199. package/rust/core/parser/handler/mod.rs +6 -5
  200. package/rust/core/parser/handler/pattern.rs +74 -0
  201. package/rust/core/parser/handler/tempo.rs +67 -9
  202. package/rust/core/parser/mod.rs +3 -4
  203. package/rust/core/parser/statement.rs +6 -92
  204. package/rust/core/plugin/loader.rs +137 -0
  205. package/rust/core/plugin/mod.rs +2 -0
  206. package/rust/core/plugin/runner/mod.rs +11 -0
  207. package/rust/core/plugin/runner/non_wasm.rs +297 -0
  208. package/rust/core/plugin/runner/wasm32.rs +43 -0
  209. package/rust/core/preprocessor/loader/inject.rs +278 -0
  210. package/rust/core/preprocessor/loader/loader_helpers.rs +110 -0
  211. package/rust/core/preprocessor/loader/mod.rs +235 -0
  212. package/rust/core/preprocessor/mod.rs +4 -4
  213. package/rust/core/preprocessor/module.rs +55 -50
  214. package/rust/core/preprocessor/processor/handlers.rs +107 -0
  215. package/rust/core/preprocessor/processor/mod.rs +1 -0
  216. package/rust/core/preprocessor/resolver/bank.rs +14 -12
  217. package/rust/core/preprocessor/resolver/call.rs +106 -105
  218. package/rust/core/preprocessor/resolver/condition.rs +13 -10
  219. package/rust/core/preprocessor/resolver/driver.rs +145 -48
  220. package/rust/core/preprocessor/resolver/function.rs +69 -0
  221. package/rust/core/preprocessor/resolver/group.rs +122 -61
  222. package/rust/core/preprocessor/resolver/let_.rs +13 -12
  223. package/rust/core/preprocessor/resolver/loop_.rs +240 -13
  224. package/rust/core/preprocessor/resolver/mod.rs +8 -6
  225. package/rust/core/preprocessor/resolver/pattern.rs +83 -0
  226. package/rust/core/preprocessor/resolver/spawn.rs +83 -42
  227. package/rust/core/preprocessor/resolver/synth.rs +15 -11
  228. package/rust/core/preprocessor/resolver/tempo.rs +13 -14
  229. package/rust/core/preprocessor/resolver/trigger.rs +32 -28
  230. package/rust/core/preprocessor/resolver/value.rs +111 -13
  231. package/rust/core/store/global.rs +57 -39
  232. package/rust/core/store/mod.rs +0 -3
  233. package/rust/lib.rs +323 -117
  234. package/rust/main.rs +388 -65
  235. package/rust/types/Cargo.toml +11 -0
  236. package/rust/types/src/addons.rs +55 -0
  237. package/rust/types/src/ast.rs +202 -0
  238. package/rust/types/src/config.rs +84 -0
  239. package/rust/types/src/lib.rs +15 -0
  240. package/rust/types/src/plugin.rs +20 -0
  241. package/rust/types/src/store.rs +139 -0
  242. package/rust/types/src/telemetry.rs +85 -0
  243. package/rust/utils/Cargo.toml +26 -0
  244. package/rust/utils/src/error.rs +186 -0
  245. package/rust/utils/src/file.rs +94 -0
  246. package/rust/utils/src/first_usage.rs +97 -0
  247. package/rust/utils/{mod.rs → src/lib.rs} +6 -3
  248. package/rust/utils/{logger.rs → src/logger.rs} +94 -17
  249. package/rust/utils/src/path.rs +129 -0
  250. package/rust/utils/src/signature.rs +41 -0
  251. package/rust/utils/{spinner.rs → src/spinner.rs} +7 -8
  252. package/rust/utils/src/version.rs +27 -0
  253. package/rust/utils/{watcher.rs → src/watcher.rs} +17 -4
  254. package/rust/web/api.rs +5 -0
  255. package/rust/web/cdn.rs +34 -0
  256. package/rust/web/mod.rs +3 -0
  257. package/rust/web/sso.rs +5 -0
  258. package/templates/minimal/README.md +143 -127
  259. package/templates/welcome/README.md +143 -127
  260. package/templates/welcome/src/index.deva +56 -8
  261. package/templates/welcome/src/variables.deva +2 -4
  262. package/tests/integration.rs +21 -0
  263. package/tests/rust/cli_check_build.rs +21 -0
  264. package/tests/rust/cli_help.rs +12 -0
  265. package/tests/rust/cli_template_list.rs +10 -0
  266. package/tests/rust/cli_version.rs +11 -0
  267. package/tests/typescript/index.spec.ts +136 -0
  268. package/tests/typescript/playhead.spec.ts +36 -0
  269. package/tests/typescript/render_e2e.spec.ts +77 -0
  270. package/tsconfig.json +12 -10
  271. package/typescript/bin/index.ts +19 -5
  272. package/typescript/core/functions/index.ts +94 -0
  273. package/typescript/core/index.ts +6 -0
  274. package/typescript/core/types/index.ts +4 -0
  275. package/typescript/core/types/plugin.ts +19 -0
  276. package/typescript/core/types/result.ts +29 -0
  277. package/typescript/core/types/statement.ts +47 -0
  278. package/typescript/core/types/value.ts +29 -0
  279. package/typescript/index.ts +8 -1
  280. package/typescript/pkg/devalang_core.d.ts +4 -0
  281. package/typescript/pkg/devalang_core.ts +65 -0
  282. package/typescript/scripts/copy-wasm-dts.ts +41 -0
  283. package/typescript/scripts/postinstall.ts +85 -0
  284. package/typescript/scripts/version/bump.ts +0 -1
  285. package/typescript/scripts/version/index.ts +0 -1
  286. package/docs/COMMANDS.md +0 -85
  287. package/docs/CONFIG.md +0 -30
  288. package/docs/SYNTAX.md +0 -210
  289. package/out-tsc/bin/devalang.exe +0 -0
  290. package/out-tsc/scripts/postbuild.js +0 -11
  291. package/rust/cli/build.rs +0 -137
  292. package/rust/cli/check.rs +0 -117
  293. package/rust/cli/play.rs +0 -193
  294. package/rust/config/loader.rs +0 -13
  295. package/rust/core/audio/engine.rs +0 -203
  296. package/rust/core/audio/evaluator.rs +0 -31
  297. package/rust/core/audio/interpreter/arrow_call.rs +0 -129
  298. package/rust/core/audio/interpreter/call.rs +0 -64
  299. package/rust/core/audio/interpreter/let_.rs +0 -19
  300. package/rust/core/audio/interpreter/load.rs +0 -18
  301. package/rust/core/audio/interpreter/loop_.rs +0 -67
  302. package/rust/core/audio/interpreter/sleep.rs +0 -36
  303. package/rust/core/audio/interpreter/spawn.rs +0 -66
  304. package/rust/core/audio/interpreter/tempo.rs +0 -16
  305. package/rust/core/audio/interpreter/trigger.rs +0 -69
  306. package/rust/core/audio/renderer.rs +0 -54
  307. package/rust/core/parser/driver.rs +0 -331
  308. package/rust/core/preprocessor/loader.rs +0 -193
  309. package/rust/core/preprocessor/processor.rs +0 -76
  310. package/rust/core/shared/duration.rs +0 -8
  311. package/rust/core/shared/mod.rs +0 -2
  312. package/rust/core/shared/value.rs +0 -18
  313. package/rust/core/store/export.rs +0 -28
  314. package/rust/core/store/import.rs +0 -28
  315. package/rust/core/store/variable.rs +0 -28
  316. package/rust/core/utils/mod.rs +0 -2
  317. package/rust/core/utils/path.rs +0 -31
  318. package/rust/core/utils/validation.rs +0 -37
  319. package/rust/utils/file.rs +0 -35
  320. package/rust/utils/signature.rs +0 -17
  321. package/rust/utils/version.rs +0 -15
  322. package/typescript/scripts/postbuild.ts +0 -8
@@ -0,0 +1,137 @@
1
+ use devalang_types::{plugin::PluginExport, plugin::PluginInfo as SharedPluginInfo};
2
+ use devalang_utils::path as path_utils;
3
+ use serde::Deserialize;
4
+ use toml::Value as TomlValue;
5
+
6
+ #[derive(Debug, Deserialize, Clone)]
7
+ struct LocalExportEntry {
8
+ pub name: String,
9
+ #[serde(rename = "type")]
10
+ pub kind: String,
11
+ #[serde(default)]
12
+ pub default: Option<TomlValue>,
13
+ }
14
+
15
+ #[derive(Debug, Deserialize, Clone)]
16
+ struct LocalPluginFile {
17
+ pub plugin: LocalPluginInfo,
18
+ #[serde(default)]
19
+ pub export: Vec<LocalExportEntry>,
20
+ }
21
+
22
+ #[derive(Debug, Deserialize, Clone)]
23
+ struct LocalPluginInfo {
24
+ pub name: String,
25
+ pub version: Option<String>,
26
+ pub description: Option<String>,
27
+ pub author: Option<String>,
28
+ }
29
+
30
+ pub fn load_plugin(author: &str, name: &str) -> Result<(SharedPluginInfo, Vec<u8>), String> {
31
+ let root = path_utils::get_deva_dir()?;
32
+ let plugin_dir_preferred = root.join("plugins").join(format!("{}.{}", author, name));
33
+ let toml_path_preferred = plugin_dir_preferred.join("plugin.toml");
34
+ let wasm_path_preferred_bg = plugin_dir_preferred.join(format!("{}_bg.wasm", name));
35
+ let wasm_path_preferred_plain = plugin_dir_preferred.join(format!("{}.wasm", name));
36
+
37
+ // Legacy layout (fallback): ./.deva/plugin/<author>/<name>/
38
+ let plugin_dir_fallback = root.join("plugins").join(author).join(name);
39
+ let toml_path_fallback = plugin_dir_fallback.join("plugin.toml");
40
+ let wasm_path_fallback_bg = plugin_dir_fallback.join(format!("{}_bg.wasm", name));
41
+ let wasm_path_fallback_plain = plugin_dir_fallback.join(format!("{}.wasm", name));
42
+
43
+ // Resolve actual paths to use
44
+ let (toml_path, wasm_path) = if toml_path_preferred.exists() && wasm_path_preferred_bg.exists()
45
+ {
46
+ (toml_path_preferred, wasm_path_preferred_bg)
47
+ } else if toml_path_preferred.exists() && wasm_path_preferred_plain.exists() {
48
+ (toml_path_preferred, wasm_path_preferred_plain)
49
+ } else if toml_path_fallback.exists() && wasm_path_fallback_bg.exists() {
50
+ (toml_path_fallback, wasm_path_fallback_bg)
51
+ } else if toml_path_fallback.exists() && wasm_path_fallback_plain.exists() {
52
+ (toml_path_fallback, wasm_path_fallback_plain)
53
+ } else {
54
+ // If either file is missing in both layouts, produce specific errors for missing files in preferred layout
55
+ if !toml_path_preferred.exists() {
56
+ return Err(format!(
57
+ "❌ Plugin file not found: {}",
58
+ toml_path_preferred.display()
59
+ ));
60
+ }
61
+ if !wasm_path_preferred_bg.exists() && !wasm_path_preferred_plain.exists() {
62
+ return Err(format!(
63
+ "❌ Plugin wasm not found: '{}' or '{}'",
64
+ wasm_path_preferred_bg.display(),
65
+ wasm_path_preferred_plain.display()
66
+ ));
67
+ }
68
+ unreachable!();
69
+ };
70
+
71
+ let toml_content = std::fs::read_to_string(&toml_path)
72
+ .map_err(|e| format!("Failed to read '{}': {}", toml_path.display(), e))?;
73
+ let plugin_file: LocalPluginFile = toml::from_str(&toml_content)
74
+ .map_err(|e| format!("Failed to parse '{}': {}", toml_path.display(), e))?;
75
+
76
+ let wasm_bytes = std::fs::read(&wasm_path)
77
+ .map_err(|e| format!("Failed to read '{}': {}", wasm_path.display(), e))?;
78
+
79
+ // Map local parsed plugin info to shared PluginInfo
80
+ let mut exports: Vec<PluginExport> = Vec::new();
81
+ for e in plugin_file.export.iter() {
82
+ exports.push(PluginExport {
83
+ name: e.name.clone(),
84
+ kind: e.kind.clone(),
85
+ default: e.default.clone(),
86
+ });
87
+ }
88
+
89
+ let info = SharedPluginInfo {
90
+ author: plugin_file
91
+ .plugin
92
+ .author
93
+ .unwrap_or_else(|| author.to_string()),
94
+ name: plugin_file.plugin.name.clone(),
95
+ version: plugin_file.plugin.version.clone(),
96
+ description: plugin_file.plugin.description.clone(),
97
+ exports,
98
+ };
99
+
100
+ Ok((info, wasm_bytes))
101
+ }
102
+
103
+ /// Load a plugin from dot notation: "author.name"
104
+ pub fn load_plugin_from_dot(dot: &str) -> Result<(SharedPluginInfo, Vec<u8>), String> {
105
+ let mut parts = dot.split('.');
106
+ let author = parts
107
+ .next()
108
+ .ok_or_else(|| "Invalid plugin name, missing author".to_string())?;
109
+ let name = parts
110
+ .next()
111
+ .ok_or_else(|| "Invalid plugin name, missing name".to_string())?;
112
+ if parts.next().is_some() {
113
+ return Err("Invalid plugin name format, expected <author>.<name>".into());
114
+ }
115
+ load_plugin(author, name)
116
+ }
117
+
118
+ pub fn load_plugin_from_uri(uri: &str) -> Result<(SharedPluginInfo, Vec<u8>), String> {
119
+ if !uri.starts_with("devalang://plugin/") {
120
+ return Err("Invalid plugin URI".into());
121
+ }
122
+
123
+ // Expect format: devalang://plugin/author.name
124
+ let payload = uri.trim_start_matches("devalang://plugin/");
125
+ let mut parts = payload.split('.');
126
+ let author = parts
127
+ .next()
128
+ .ok_or_else(|| "Invalid plugin URI, missing author".to_string())?;
129
+ let name = parts
130
+ .next()
131
+ .ok_or_else(|| "Invalid plugin URI, missing name".to_string())?;
132
+ if parts.next().is_some() {
133
+ return Err("Invalid plugin URI format, expected devalang://plugin/<author>.<name>".into());
134
+ }
135
+
136
+ load_plugin(author, name)
137
+ }
@@ -0,0 +1,2 @@
1
+ pub mod loader;
2
+ pub mod runner;
@@ -0,0 +1,11 @@
1
+ #[cfg(not(target_arch = "wasm32"))]
2
+ mod non_wasm;
3
+
4
+ #[cfg(target_arch = "wasm32")]
5
+ mod wasm32;
6
+
7
+ #[cfg(not(target_arch = "wasm32"))]
8
+ pub use non_wasm::WasmPluginRunner;
9
+
10
+ #[cfg(target_arch = "wasm32")]
11
+ pub use wasm32::WasmPluginRunner;
@@ -0,0 +1,297 @@
1
+ use std::collections::HashMap;
2
+
3
+ use wasmtime::{Engine, Instance, Linker, Module, Store, TypedFunc};
4
+
5
+ type RenderFunc = TypedFunc<(i32, i32, f32, f32, i32, i32, i32), ()>;
6
+
7
+ pub struct WasmPluginRunner {
8
+ engine: Engine,
9
+ }
10
+
11
+ impl Default for WasmPluginRunner {
12
+ fn default() -> Self {
13
+ Self::new()
14
+ }
15
+ }
16
+
17
+ impl WasmPluginRunner {
18
+ pub fn new() -> Self {
19
+ let engine = Engine::default();
20
+ Self { engine }
21
+ }
22
+
23
+ pub fn process_in_place(&self, wasm_bytes: &[u8], buffer: &mut [f32]) -> Result<(), String> {
24
+ let module = Module::new(&self.engine, wasm_bytes)
25
+ .map_err(|e| format!("Failed to compile wasm: {e}"))?;
26
+
27
+ let mut store = Store::new(&self.engine, ());
28
+ let linker = Linker::new(&self.engine);
29
+
30
+ let instance = linker
31
+ .instantiate(&mut store, &module)
32
+ .map_err(|e| format!("Failed to instantiate wasm: {e}"))?;
33
+
34
+ let memory = instance
35
+ .get_memory(&mut store, "memory")
36
+ .ok_or_else(|| "WASM memory export not found".to_string())?;
37
+
38
+ let func = instance
39
+ .get_typed_func::<(i32, i32), ()>(&mut store, "process")
40
+ .map_err(|_| "Exported function `process(i32,i32)` not found".to_string())?;
41
+
42
+ let byte_len = std::mem::size_of_val(buffer) as i32;
43
+ let ptr = Self::alloc_temp(&mut store, &instance, &memory, byte_len as usize)? as i32;
44
+ let mem_slice = memory
45
+ .data_mut(&mut store)
46
+ .get_mut(ptr as usize..(ptr as usize) + (byte_len as usize))
47
+ .ok_or_else(|| "Failed to get memory slice".to_string())?;
48
+
49
+ let src_bytes =
50
+ unsafe { std::slice::from_raw_parts(buffer.as_ptr() as *const u8, byte_len as usize) };
51
+ mem_slice.copy_from_slice(src_bytes);
52
+
53
+ func.call(&mut store, (ptr, buffer.len() as i32))
54
+ .map_err(|e| format!("Error calling `process`: {e}"))?;
55
+
56
+ let mem_slice_after = memory
57
+ .data(&store)
58
+ .get(ptr as usize..(ptr as usize) + (byte_len as usize))
59
+ .ok_or_else(|| "Failed to get memory slice after".to_string())?;
60
+ let dst_bytes = unsafe {
61
+ std::slice::from_raw_parts_mut(buffer.as_mut_ptr() as *mut u8, byte_len as usize)
62
+ };
63
+ dst_bytes.copy_from_slice(mem_slice_after);
64
+
65
+ Ok(())
66
+ }
67
+
68
+ pub fn render_note_in_place(
69
+ &self,
70
+ wasm_bytes: &[u8],
71
+ buffer: &mut [f32],
72
+ synth_name: Option<&str>,
73
+ freq: f32,
74
+ amp: f32,
75
+ duration_ms: i32,
76
+ sample_rate: i32,
77
+ channels: i32,
78
+ ) -> Result<(), String> {
79
+ let module = Module::new(&self.engine, wasm_bytes)
80
+ .map_err(|e| format!("Failed to compile wasm: {e}"))?;
81
+
82
+ let mut store = Store::new(&self.engine, ());
83
+ let linker = Linker::new(&self.engine);
84
+
85
+ let instance = linker
86
+ .instantiate(&mut store, &module)
87
+ .map_err(|e| format!("Failed to instantiate wasm: {e}"))?;
88
+
89
+ let memory = instance
90
+ .get_memory(&mut store, "memory")
91
+ .ok_or_else(|| "WASM memory export not found".to_string())?;
92
+
93
+ // Try specific function first
94
+ let mut func_opt: Option<RenderFunc> = None;
95
+ if let Some(name) = synth_name {
96
+ let specific = format!("render_note_{}", name);
97
+ if let Ok(f) = instance
98
+ .get_typed_func::<(i32, i32, f32, f32, i32, i32, i32), ()>(&mut store, &specific)
99
+ {
100
+ func_opt = Some(f);
101
+ }
102
+ }
103
+ if func_opt.is_none() {
104
+ // fallback to generic name
105
+ if let Ok(f) = instance.get_typed_func::<(i32, i32, f32, f32, i32, i32, i32), ()>(
106
+ &mut store,
107
+ "render_note",
108
+ ) {
109
+ func_opt = Some(f);
110
+ }
111
+ }
112
+
113
+ let func =
114
+ func_opt.ok_or_else(|| "Exported function `render_note` not found".to_string())?;
115
+
116
+ // Copy host buffer into wasm memory
117
+ let byte_len = std::mem::size_of_val(buffer) as i32;
118
+ let ptr = Self::alloc_temp(&mut store, &instance, &memory, byte_len as usize)? as i32;
119
+ let mem_slice = memory
120
+ .data_mut(&mut store)
121
+ .get_mut(ptr as usize..(ptr as usize) + (byte_len as usize))
122
+ .ok_or_else(|| "Failed to get memory slice".to_string())?;
123
+ let src_bytes =
124
+ unsafe { std::slice::from_raw_parts(buffer.as_ptr() as *const u8, byte_len as usize) };
125
+ mem_slice.copy_from_slice(src_bytes);
126
+
127
+ // Call render
128
+ func.call(
129
+ &mut store,
130
+ (
131
+ ptr,
132
+ buffer.len() as i32,
133
+ freq,
134
+ amp,
135
+ duration_ms,
136
+ sample_rate,
137
+ channels,
138
+ ),
139
+ )
140
+ .map_err(|e| format!("Error calling `render_note`: {e}"))?;
141
+
142
+ // Copy back
143
+ let mem_slice_after = memory
144
+ .data(&store)
145
+ .get(ptr as usize..(ptr as usize) + (byte_len as usize))
146
+ .ok_or_else(|| "Failed to get memory slice after".to_string())?;
147
+ let dst_bytes = unsafe {
148
+ std::slice::from_raw_parts_mut(buffer.as_mut_ptr() as *mut u8, byte_len as usize)
149
+ };
150
+ dst_bytes.copy_from_slice(mem_slice_after);
151
+
152
+ Ok(())
153
+ }
154
+
155
+ /// Same as render_note_in_place, but first tries to call exported setters `set_<param>(f32)`
156
+ /// for each provided param before rendering. Ignored if setter is missing.
157
+ pub fn render_note_with_params_in_place(
158
+ &self,
159
+ wasm_bytes: &[u8],
160
+ buffer: &mut [f32],
161
+ synth_name: Option<&str>,
162
+ freq: f32,
163
+ amp: f32,
164
+ duration_ms: i32,
165
+ sample_rate: i32,
166
+ channels: i32,
167
+ params_num: &HashMap<String, f32>,
168
+ params_str: Option<&HashMap<String, String>>,
169
+ ) -> Result<(), String> {
170
+ let module = Module::new(&self.engine, wasm_bytes)
171
+ .map_err(|e| format!("Failed to compile wasm: {e}"))?;
172
+
173
+ let mut store = Store::new(&self.engine, ());
174
+ let linker = Linker::new(&self.engine);
175
+
176
+ let instance = linker
177
+ .instantiate(&mut store, &module)
178
+ .map_err(|e| format!("Failed to instantiate wasm: {e}"))?;
179
+
180
+ let memory = instance
181
+ .get_memory(&mut store, "memory")
182
+ .ok_or_else(|| "WASM memory export not found".to_string())?;
183
+
184
+ // Call numeric setters if present: set_<param>(f32)
185
+ for (k, v) in params_num.iter() {
186
+ let fname = format!("set_{}", k);
187
+ if let Ok(setter) = instance.get_typed_func::<f32, ()>(&mut store, &fname) {
188
+ let _ = setter.call(&mut store, *v);
189
+ }
190
+ }
191
+
192
+ // Call string setters if present: set_<param>_str(ptr: i32, len: i32)
193
+ if let Some(smap) = params_str {
194
+ for (k, v) in smap.iter() {
195
+ let fname = format!("set_{}_str", k);
196
+ if let Ok(setter) = instance.get_typed_func::<(i32, i32), ()>(&mut store, &fname) {
197
+ // Allocate and copy UTF-8 bytes into wasm memory
198
+ let bytes = v.as_bytes();
199
+ let ptr = Self::alloc_temp(&mut store, &instance, &memory, bytes.len())? as i32;
200
+ let mem_slice = memory
201
+ .data_mut(&mut store)
202
+ .get_mut(ptr as usize..(ptr as usize) + bytes.len())
203
+ .ok_or_else(|| "Failed to get memory slice for string".to_string())?;
204
+ mem_slice.copy_from_slice(bytes);
205
+ let _ = setter.call(&mut store, (ptr, bytes.len() as i32));
206
+ }
207
+ }
208
+ }
209
+
210
+ // Try specific or generic render function
211
+ let mut func_opt: Option<RenderFunc> = None;
212
+ if let Some(name) = synth_name {
213
+ let specific = format!("render_note_{}", name);
214
+ if let Ok(f) = instance
215
+ .get_typed_func::<(i32, i32, f32, f32, i32, i32, i32), ()>(&mut store, &specific)
216
+ {
217
+ func_opt = Some(f);
218
+ }
219
+ }
220
+ if func_opt.is_none() {
221
+ if let Ok(f) = instance.get_typed_func::<(i32, i32, f32, f32, i32, i32, i32), ()>(
222
+ &mut store,
223
+ "render_note",
224
+ ) {
225
+ func_opt = Some(f);
226
+ }
227
+ }
228
+ let func =
229
+ func_opt.ok_or_else(|| "Exported function `render_note` not found".to_string())?;
230
+
231
+ // Copy host buffer into wasm memory
232
+ let byte_len = std::mem::size_of_val(buffer) as i32;
233
+ let ptr = Self::alloc_temp(&mut store, &instance, &memory, byte_len as usize)? as i32;
234
+ let mem_slice = memory
235
+ .data_mut(&mut store)
236
+ .get_mut(ptr as usize..(ptr as usize) + (byte_len as usize))
237
+ .ok_or_else(|| "Failed to get memory slice".to_string())?;
238
+ let src_bytes =
239
+ unsafe { std::slice::from_raw_parts(buffer.as_ptr() as *const u8, byte_len as usize) };
240
+ mem_slice.copy_from_slice(src_bytes);
241
+
242
+ // Call render
243
+ func.call(
244
+ &mut store,
245
+ (
246
+ ptr,
247
+ buffer.len() as i32,
248
+ freq,
249
+ amp,
250
+ duration_ms,
251
+ sample_rate,
252
+ channels,
253
+ ),
254
+ )
255
+ .map_err(|e| format!("Error calling `render_note`: {e}"))?;
256
+
257
+ // Copy back
258
+ let mem_slice_after = memory
259
+ .data(&store)
260
+ .get(ptr as usize..(ptr as usize) + (byte_len as usize))
261
+ .ok_or_else(|| "Failed to get memory slice after".to_string())?;
262
+ let dst_bytes = unsafe {
263
+ std::slice::from_raw_parts_mut(buffer.as_mut_ptr() as *mut u8, byte_len as usize)
264
+ };
265
+ dst_bytes.copy_from_slice(mem_slice_after);
266
+
267
+ Ok(())
268
+ }
269
+
270
+ fn alloc_temp(
271
+ store: &mut Store<()>,
272
+ instance: &Instance,
273
+ memory: &wasmtime::Memory,
274
+ size: usize,
275
+ ) -> Result<usize, String> {
276
+ // Try to use an exported `__wbindgen_malloc` if present; otherwise, grow memory manually.
277
+ if let Ok(malloc) = instance.get_typed_func::<i32, i32>(&mut *store, "__wbindgen_malloc") {
278
+ let ptr = malloc
279
+ .call(&mut *store, size as i32)
280
+ .map_err(|e| format!("malloc failed: {e}"))? as usize;
281
+ return Ok(ptr);
282
+ }
283
+
284
+ // Fallback: grow memory and use end of memory as scratch space
285
+ let current_len = memory.data_size(&mut *store);
286
+ let need = size;
287
+ let pages_needed = (current_len + need).div_ceil(0x10000) as u64; // 64KiB pages
288
+ let current_pages = memory.size(&mut *store);
289
+ if pages_needed > current_pages {
290
+ let to_grow = pages_needed - current_pages;
291
+ memory
292
+ .grow(&mut *store, to_grow)
293
+ .map_err(|e| format!("memory.grow failed: {e}"))?;
294
+ }
295
+ Ok(current_len)
296
+ }
297
+ }
@@ -0,0 +1,43 @@
1
+ use std::collections::HashMap;
2
+
3
+ pub struct WasmPluginRunner;
4
+
5
+ impl WasmPluginRunner {
6
+ pub fn new() -> Self {
7
+ WasmPluginRunner
8
+ }
9
+
10
+ pub fn process_in_place(&self, _wasm_bytes: &[u8], _buffer: &mut [f32]) -> Result<(), String> {
11
+ Err("Wasm plugin execution is not available in wasm builds".to_string())
12
+ }
13
+
14
+ pub fn render_note_in_place(
15
+ &self,
16
+ _wasm_bytes: &[u8],
17
+ _buffer: &mut [f32],
18
+ _synth_name: Option<&str>,
19
+ _freq: f32,
20
+ _amp: f32,
21
+ _duration_ms: i32,
22
+ _sample_rate: i32,
23
+ _channels: i32,
24
+ ) -> Result<(), String> {
25
+ Err("Wasm plugin rendering is not available in wasm builds".to_string())
26
+ }
27
+
28
+ pub fn render_note_with_params_in_place(
29
+ &self,
30
+ _wasm_bytes: &[u8],
31
+ _buffer: &mut [f32],
32
+ _synth_name: Option<&str>,
33
+ _freq: f32,
34
+ _amp: f32,
35
+ _duration_ms: i32,
36
+ _sample_rate: i32,
37
+ _channels: i32,
38
+ _params_num: &HashMap<String, f32>,
39
+ _params_str: Option<&HashMap<String, String>>,
40
+ ) -> Result<(), String> {
41
+ Err("Wasm plugin rendering is not available in wasm builds".to_string())
42
+ }
43
+ }