@devaloop/devalang 0.0.1-beta.2 → 0.0.1-beta.3

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 (159) hide show
  1. package/Cargo.toml +84 -81
  2. package/README.md +3 -2
  3. package/docs/CHANGELOG.md +41 -0
  4. package/docs/ROADMAP.md +3 -3
  5. package/examples/chain.deva +19 -0
  6. package/examples/plugin.deva +10 -10
  7. package/examples/routing.deva +23 -0
  8. package/out-tsc/bin/project-version.json +6 -0
  9. package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +8 -8
  10. package/out-tsc/scripts/version/copy-to-binary.d.ts +1 -0
  11. package/out-tsc/scripts/version/copy-to-binary.js +79 -0
  12. package/package.json +23 -10
  13. package/project-version.json +3 -3
  14. package/rust/bindings/Cargo.toml +9 -0
  15. package/rust/bindings/src/lib.rs +86 -0
  16. package/rust/cli/addon/commands.rs +35 -0
  17. package/rust/cli/addon/download.rs +234 -0
  18. package/rust/cli/addon/install.rs +33 -0
  19. package/rust/cli/addon/list.rs +224 -0
  20. package/rust/cli/addon/metadata.rs +124 -0
  21. package/rust/cli/addon/mod.rs +8 -0
  22. package/rust/cli/addon/remove.rs +271 -0
  23. package/rust/cli/addon/update.rs +305 -0
  24. package/rust/cli/{install/addon.rs → addon/utils.rs} +109 -118
  25. package/rust/cli/build/commands.rs +153 -153
  26. package/rust/cli/build/process.rs +165 -165
  27. package/rust/cli/check/mod.rs +208 -208
  28. package/rust/cli/discover/commands.rs +275 -253
  29. package/rust/cli/discover/config.rs +109 -111
  30. package/rust/cli/discover/fs.rs +19 -19
  31. package/rust/cli/discover/install.rs +214 -103
  32. package/rust/cli/discover/metadata.rs +48 -48
  33. package/rust/cli/discover/mod.rs +5 -5
  34. package/rust/cli/me/commands.rs +52 -0
  35. package/rust/cli/me/mod.rs +1 -0
  36. package/rust/cli/mod.rs +12 -12
  37. package/rust/cli/parser.rs +30 -69
  38. package/rust/cli/play/commands.rs +375 -375
  39. package/rust/cli/play/process.rs +159 -159
  40. package/rust/core/audio/engine/driver.rs +19 -2
  41. package/rust/core/audio/engine/export.rs +169 -169
  42. package/rust/core/audio/engine/mod.rs +56 -56
  43. package/rust/core/audio/engine/notes/dsp.rs +88 -85
  44. package/rust/core/audio/engine/notes/mod.rs +53 -44
  45. package/rust/core/audio/engine/notes/params.rs +294 -294
  46. package/rust/core/audio/engine/sample/insert.rs +148 -47
  47. package/rust/core/audio/engine/sample/mod.rs +40 -40
  48. package/rust/core/audio/engine/sample/padding.rs +170 -170
  49. package/rust/core/audio/evaluator/condition.rs +61 -61
  50. package/rust/core/audio/evaluator/numeric.rs +152 -152
  51. package/rust/core/audio/evaluator/rhs.rs +16 -16
  52. package/rust/core/audio/evaluator/string_expr.rs +94 -94
  53. package/rust/core/audio/interpreter/driver.rs +574 -574
  54. package/rust/core/audio/interpreter/mod.rs +2 -2
  55. package/rust/core/audio/interpreter/statements/arrow_call/interprete.rs +9 -5
  56. package/rust/core/audio/interpreter/statements/arrow_call/methods/chord.rs +398 -384
  57. package/rust/core/audio/interpreter/statements/arrow_call/methods/effects.rs +323 -0
  58. package/rust/core/audio/interpreter/statements/arrow_call/methods/mod.rs +1 -0
  59. package/rust/core/audio/interpreter/statements/arrow_call/methods/note.rs +66 -11
  60. package/rust/core/audio/interpreter/statements/arrow_call/mod.rs +3 -3
  61. package/rust/core/audio/interpreter/statements/arrow_call/types/arp.rs +192 -192
  62. package/rust/core/audio/interpreter/statements/arrow_call/types/mod.rs +24 -24
  63. package/rust/core/audio/interpreter/statements/arrow_call/types/pad.rs +116 -116
  64. package/rust/core/audio/interpreter/statements/arrow_call/types/pluck.rs +97 -97
  65. package/rust/core/audio/interpreter/statements/arrow_call/types/sub.rs +100 -100
  66. package/rust/core/audio/interpreter/statements/automate.rs +16 -16
  67. package/rust/core/audio/interpreter/statements/call.rs +31 -1
  68. package/rust/core/audio/interpreter/statements/condition.rs +72 -72
  69. package/rust/core/audio/interpreter/statements/function.rs +24 -24
  70. package/rust/core/audio/interpreter/statements/let_.rs +36 -36
  71. package/rust/core/audio/interpreter/statements/load.rs +17 -17
  72. package/rust/core/audio/interpreter/statements/loop_.rs +115 -115
  73. package/rust/core/audio/interpreter/statements/spawn.rs +51 -2
  74. package/rust/core/audio/interpreter/statements/trigger.rs +242 -239
  75. package/rust/core/audio/loader/trigger.rs +98 -98
  76. package/rust/core/audio/player.rs +70 -70
  77. package/rust/core/audio/special/mod.rs +9 -9
  78. package/rust/core/builder/mod.rs +129 -129
  79. package/rust/core/debugger/lexer.rs +27 -27
  80. package/rust/core/debugger/logs.rs +52 -52
  81. package/rust/core/debugger/preprocessor.rs +27 -27
  82. package/rust/core/debugger/store.rs +38 -38
  83. package/rust/core/lexer/driver.rs +59 -59
  84. package/rust/core/lexer/handler/arrow.rs +82 -82
  85. package/rust/core/lexer/handler/at.rs +21 -21
  86. package/rust/core/lexer/handler/brace.rs +41 -41
  87. package/rust/core/lexer/handler/colon.rs +21 -21
  88. package/rust/core/lexer/handler/comment.rs +30 -30
  89. package/rust/core/lexer/handler/dot.rs +21 -21
  90. package/rust/core/lexer/handler/driver.rs +337 -337
  91. package/rust/core/lexer/handler/identifier.rs +47 -47
  92. package/rust/core/lexer/handler/indent.rs +66 -66
  93. package/rust/core/lexer/handler/mod.rs +15 -15
  94. package/rust/core/lexer/handler/newline.rs +23 -23
  95. package/rust/core/lexer/handler/number.rs +31 -31
  96. package/rust/core/lexer/handler/operator.rs +46 -46
  97. package/rust/core/lexer/handler/parenthesis.rs +41 -41
  98. package/rust/core/lexer/handler/slash.rs +21 -21
  99. package/rust/core/lexer/handler/string.rs +63 -63
  100. package/rust/core/lexer/mod.rs +3 -3
  101. package/rust/core/mod.rs +9 -9
  102. package/rust/core/parser/driver/block.rs +111 -111
  103. package/rust/core/parser/driver/cursor.rs +82 -82
  104. package/rust/core/parser/driver/driver_impl.rs +21 -1
  105. package/rust/core/parser/driver/mod.rs +6 -6
  106. package/rust/core/parser/driver/parse_array.rs +120 -120
  107. package/rust/core/parser/driver/parse_map.rs +247 -223
  108. package/rust/core/parser/driver/parser.rs +160 -160
  109. package/rust/core/parser/handler/arrow_call.rs +65 -14
  110. package/rust/core/parser/handler/identifier/synth.rs +171 -135
  111. package/rust/core/parser/handler/mod.rs +9 -9
  112. package/rust/core/parser/handler/pattern.rs +24 -1
  113. package/rust/core/plugin/loader.rs +137 -137
  114. package/rust/core/plugin/mod.rs +2 -2
  115. package/rust/core/plugin/runner/non_wasm.rs +481 -297
  116. package/rust/core/plugin/runner/wasm32.rs +1 -0
  117. package/rust/core/preprocessor/loader/inject.rs +313 -278
  118. package/rust/core/preprocessor/loader/loader_helpers.rs +110 -110
  119. package/rust/core/preprocessor/loader/mod.rs +235 -235
  120. package/rust/core/preprocessor/module.rs +55 -55
  121. package/rust/core/preprocessor/processor/handlers.rs +107 -107
  122. package/rust/core/preprocessor/resolver/bank.rs +49 -49
  123. package/rust/core/preprocessor/resolver/call.rs +124 -124
  124. package/rust/core/preprocessor/resolver/condition.rs +95 -95
  125. package/rust/core/preprocessor/resolver/driver.rs +324 -324
  126. package/rust/core/preprocessor/resolver/function.rs +69 -69
  127. package/rust/core/preprocessor/resolver/group.rs +122 -122
  128. package/rust/core/preprocessor/resolver/let_.rs +32 -32
  129. package/rust/core/preprocessor/resolver/loop_.rs +318 -318
  130. package/rust/core/preprocessor/resolver/mod.rs +16 -16
  131. package/rust/core/preprocessor/resolver/pattern.rs +95 -83
  132. package/rust/core/preprocessor/resolver/spawn.rs +99 -99
  133. package/rust/core/preprocessor/resolver/synth.rs +54 -54
  134. package/rust/core/preprocessor/resolver/tempo.rs +48 -48
  135. package/rust/core/preprocessor/resolver/trigger.rs +116 -116
  136. package/rust/core/preprocessor/resolver/value.rs +176 -176
  137. package/rust/core/store/global.rs +57 -57
  138. package/rust/lib.rs +323 -323
  139. package/rust/macros/Cargo.toml +14 -0
  140. package/rust/macros/src/lib.rs +52 -0
  141. package/rust/main.rs +311 -142
  142. package/rust/types/Cargo.toml +1 -1
  143. package/rust/types/src/addons.rs +3 -1
  144. package/rust/types/src/config.rs +1 -3
  145. package/rust/utils/Cargo.toml +5 -2
  146. package/rust/utils/src/file.rs +397 -14
  147. package/rust/utils/src/path.rs +31 -2
  148. package/rust/utils/src/version.rs +38 -7
  149. package/rust/web/auth.rs +5 -0
  150. package/rust/web/forge.rs +5 -0
  151. package/rust/web/mod.rs +5 -3
  152. package/typescript/scripts/version/copy-to-binary.ts +82 -0
  153. package/rust/cli/bank/api.rs +0 -122
  154. package/rust/cli/bank/commands.rs +0 -306
  155. package/rust/cli/bank/mod.rs +0 -29
  156. package/rust/cli/install/bank.rs +0 -72
  157. package/rust/cli/install/commands.rs +0 -35
  158. package/rust/cli/install/mod.rs +0 -4
  159. package/rust/cli/install/plugin.rs +0 -80
@@ -1,297 +1,481 @@
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
- }
1
+ use devalang_utils::logger::{LogLevel, Logger};
2
+ use std::collections::HashMap;
3
+
4
+ use wasmtime::{Engine, Instance, Linker, Module, Store, TypedFunc};
5
+
6
+ type RenderFunc = TypedFunc<(i32, i32, f32, f32, i32, i32, i32), ()>;
7
+
8
+ pub struct WasmPluginRunner {
9
+ engine: Engine,
10
+ }
11
+
12
+ impl Default for WasmPluginRunner {
13
+ fn default() -> Self {
14
+ Self::new()
15
+ }
16
+ }
17
+
18
+ impl WasmPluginRunner {
19
+ pub fn new() -> Self {
20
+ let engine = Engine::default();
21
+ Self { engine }
22
+ }
23
+
24
+ pub fn process_in_place(&self, wasm_bytes: &[u8], buffer: &mut [f32]) -> Result<(), String> {
25
+ let module = Module::new(&self.engine, wasm_bytes)
26
+ .map_err(|e| format!("Failed to compile wasm: {e}"))?;
27
+
28
+ let mut store = Store::new(&self.engine, ());
29
+ let linker = Linker::new(&self.engine);
30
+
31
+ let instance = linker
32
+ .instantiate(&mut store, &module)
33
+ .map_err(|e| format!("Failed to instantiate wasm: {e}"))?;
34
+
35
+ let memory = instance
36
+ .get_memory(&mut store, "memory")
37
+ .ok_or_else(|| "WASM memory export not found".to_string())?;
38
+
39
+ let func = instance
40
+ .get_typed_func::<(i32, i32), ()>(&mut store, "process")
41
+ .map_err(|_| "Exported function `process(i32,i32)` not found".to_string())?;
42
+
43
+ let byte_len = std::mem::size_of_val(buffer) as i32;
44
+ let ptr = Self::alloc_temp(&mut store, &instance, &memory, byte_len as usize)? as i32;
45
+ let mem_slice = memory
46
+ .data_mut(&mut store)
47
+ .get_mut(ptr as usize..(ptr as usize) + (byte_len as usize))
48
+ .ok_or_else(|| "Failed to get memory slice".to_string())?;
49
+
50
+ let src_bytes =
51
+ unsafe { std::slice::from_raw_parts(buffer.as_ptr() as *const u8, byte_len as usize) };
52
+ mem_slice.copy_from_slice(src_bytes);
53
+
54
+ func.call(&mut store, (ptr, buffer.len() as i32))
55
+ .map_err(|e| format!("Error calling `process`: {e}"))?;
56
+
57
+ let mem_slice_after = memory
58
+ .data(&store)
59
+ .get(ptr as usize..(ptr as usize) + (byte_len as usize))
60
+ .ok_or_else(|| "Failed to get memory slice after".to_string())?;
61
+ let dst_bytes = unsafe {
62
+ std::slice::from_raw_parts_mut(buffer.as_mut_ptr() as *mut u8, byte_len as usize)
63
+ };
64
+ dst_bytes.copy_from_slice(mem_slice_after);
65
+
66
+ Ok(())
67
+ }
68
+
69
+ pub fn render_note_in_place(
70
+ &self,
71
+ wasm_bytes: &[u8],
72
+ buffer: &mut [f32],
73
+ synth_name: Option<&str>,
74
+ freq: f32,
75
+ amp: f32,
76
+ duration_ms: i32,
77
+ sample_rate: i32,
78
+ channels: i32,
79
+ ) -> Result<(), String> {
80
+ let module = Module::new(&self.engine, wasm_bytes)
81
+ .map_err(|e| format!("Failed to compile wasm: {e}"))?;
82
+
83
+ let mut store = Store::new(&self.engine, ());
84
+ let linker = Linker::new(&self.engine);
85
+
86
+ let instance = linker
87
+ .instantiate(&mut store, &module)
88
+ .map_err(|e| format!("Failed to instantiate wasm: {e}"))?;
89
+
90
+ let memory = instance
91
+ .get_memory(&mut store, "memory")
92
+ .ok_or_else(|| "WASM memory export not found".to_string())?;
93
+
94
+ // Try specific + generic entry points in order of preference.
95
+ // 1) render_note_<name>
96
+ // 2) render_note
97
+ // 3) synth_<name>
98
+ // 4) synth
99
+ let mut func_opt: Option<RenderFunc> = None;
100
+ if let Some(name) = synth_name {
101
+ let specific = format!("render_note_{}", name);
102
+ if let Ok(f) = instance
103
+ .get_typed_func::<(i32, i32, f32, f32, i32, i32, i32), ()>(&mut store, &specific)
104
+ {
105
+ func_opt = Some(f);
106
+ }
107
+ }
108
+ if func_opt.is_none() {
109
+ if let Ok(f) = instance.get_typed_func::<(i32, i32, f32, f32, i32, i32, i32), ()>(
110
+ &mut store,
111
+ "render_note",
112
+ ) {
113
+ func_opt = Some(f);
114
+ }
115
+ }
116
+ if func_opt.is_none() {
117
+ if let Some(name) = synth_name {
118
+ let specific = format!("synth_{}", name);
119
+ if let Ok(f) = instance.get_typed_func::<(i32, i32, f32, f32, i32, i32, i32), ()>(
120
+ &mut store, &specific,
121
+ ) {
122
+ func_opt = Some(f);
123
+ }
124
+ }
125
+ }
126
+ if func_opt.is_none() {
127
+ if let Ok(f) = instance
128
+ .get_typed_func::<(i32, i32, f32, f32, i32, i32, i32), ()>(&mut store, "synth")
129
+ {
130
+ func_opt = Some(f);
131
+ }
132
+ }
133
+
134
+ let func = func_opt.ok_or_else(|| {
135
+ "Exported function not found: tried render_note[_<name>] and synth[_<name>]".to_string()
136
+ })?;
137
+
138
+ // Copy host buffer into wasm memory
139
+ let byte_len = std::mem::size_of_val(buffer) as i32;
140
+ let ptr = Self::alloc_temp(&mut store, &instance, &memory, byte_len as usize)? as i32;
141
+ let mem_slice = memory
142
+ .data_mut(&mut store)
143
+ .get_mut(ptr as usize..(ptr as usize) + (byte_len as usize))
144
+ .ok_or_else(|| "Failed to get memory slice".to_string())?;
145
+ let src_bytes =
146
+ unsafe { std::slice::from_raw_parts(buffer.as_ptr() as *const u8, byte_len as usize) };
147
+ mem_slice.copy_from_slice(src_bytes);
148
+
149
+ // Call render
150
+ func.call(
151
+ &mut store,
152
+ (
153
+ ptr,
154
+ buffer.len() as i32,
155
+ freq,
156
+ amp,
157
+ duration_ms,
158
+ sample_rate,
159
+ channels,
160
+ ),
161
+ )
162
+ .map_err(|e| format!("Error calling `render_note`: {e}"))?;
163
+
164
+ // Copy back
165
+ let mem_slice_after = memory
166
+ .data(&store)
167
+ .get(ptr as usize..(ptr as usize) + (byte_len as usize))
168
+ .ok_or_else(|| "Failed to get memory slice after".to_string())?;
169
+ let dst_bytes = unsafe {
170
+ std::slice::from_raw_parts_mut(buffer.as_mut_ptr() as *mut u8, byte_len as usize)
171
+ };
172
+ dst_bytes.copy_from_slice(mem_slice_after);
173
+
174
+ Ok(())
175
+ }
176
+
177
+ /// Same as render_note_in_place, but first tries to call exported setters `set_<param>(f32)`
178
+ /// for each provided param before rendering. Ignored if setter is missing.
179
+ pub fn render_note_with_params_in_place(
180
+ &self,
181
+ wasm_bytes: &[u8],
182
+ buffer: &mut [f32],
183
+ synth_name: Option<&str>,
184
+ freq: f32,
185
+ amp: f32,
186
+ duration_ms: i32,
187
+ sample_rate: i32,
188
+ channels: i32,
189
+ params_num: &HashMap<String, f32>,
190
+ params_str: Option<&HashMap<String, String>>,
191
+ exported_names: Option<&[String]>,
192
+ ) -> Result<(), String> {
193
+ let module = Module::new(&self.engine, wasm_bytes)
194
+ .map_err(|e| format!("Failed to compile wasm: {e}"))?;
195
+
196
+ let mut store = Store::new(&self.engine, ());
197
+ let linker = Linker::new(&self.engine);
198
+
199
+ let instance = linker
200
+ .instantiate(&mut store, &module)
201
+ .map_err(|e| format!("Failed to instantiate wasm: {e}"))?;
202
+
203
+ let memory = instance
204
+ .get_memory(&mut store, "memory")
205
+ .ok_or_else(|| "WASM memory export not found".to_string())?;
206
+
207
+ // Call numeric setters if present: set_<param>(f32)
208
+ let logger = Logger::new();
209
+ for (k, v) in params_num.iter() {
210
+ // Candidate patterns in order of preference
211
+ let candidates = [
212
+ format!("set_synth_{}", k),
213
+ format!("set_{}", k),
214
+ format!("set_note_{}", k),
215
+ ];
216
+
217
+ let mut any_called = false;
218
+
219
+ // If exported_names provided, try only declared exports first
220
+ if let Some(exports) = exported_names {
221
+ logger.log_message(
222
+ LogLevel::Debug,
223
+ &format!("Plugin exports provided ({} names)", exports.len()),
224
+ );
225
+ for c in &candidates {
226
+ if exports.iter().any(|e| e == c) {
227
+ match instance.get_typed_func::<f32, ()>(&mut store, c) {
228
+ Ok(setter) => {
229
+ let _ = setter.call(&mut store, *v);
230
+ any_called = true;
231
+ logger.log_message(
232
+ LogLevel::Debug,
233
+ &format!("Called setter '{}' with {}", c, v),
234
+ );
235
+ }
236
+ Err(_) => {
237
+ logger.log_message(
238
+ LogLevel::Debug,
239
+ &format!("Export '{}' declared but signature lookup failed", c),
240
+ );
241
+ }
242
+ }
243
+ }
244
+ }
245
+ }
246
+
247
+ // If nothing was called using the metadata, fall back to dynamic lookup
248
+ if !any_called {
249
+ for fname in &candidates {
250
+ match instance.get_typed_func::<f32, ()>(&mut store, fname) {
251
+ Ok(setter) => {
252
+ let _ = setter.call(&mut store, *v);
253
+ any_called = true;
254
+ logger.log_message(
255
+ LogLevel::Debug,
256
+ &format!("Dynamically called setter '{}' with {}", fname, v),
257
+ );
258
+ }
259
+ Err(_) => {
260
+ // no-op, continue
261
+ }
262
+ }
263
+ }
264
+ }
265
+
266
+ if !any_called {
267
+ logger.log_message(
268
+ LogLevel::Warning,
269
+ &format!("No setter found/called for numeric param '{}'", k),
270
+ );
271
+ }
272
+ }
273
+
274
+ // Call string setters if present: support set_synth_<param>_str, set_<param>_str, set_note_<param>_str
275
+ if let Some(smap) = params_str {
276
+ for (k, v) in smap.iter() {
277
+ let candidates = [
278
+ format!("set_synth_{}_str", k),
279
+ format!("set_{}_str", k),
280
+ format!("set_note_{}_str", k),
281
+ ];
282
+
283
+ let logger = Logger::new();
284
+ for (k, v) in smap.iter() {
285
+ let candidates = [
286
+ format!("set_synth_{}_str", k),
287
+ format!("set_{}_str", k),
288
+ format!("set_note_{}_str", k),
289
+ ];
290
+
291
+ let mut any_called = false;
292
+
293
+ if let Some(exports) = exported_names {
294
+ for c in &candidates {
295
+ if exports.iter().any(|e| e == c) {
296
+ match instance.get_typed_func::<(i32, i32), ()>(&mut store, c) {
297
+ Ok(setter) => {
298
+ let bytes = v.as_bytes();
299
+ let ptr = Self::alloc_temp(
300
+ &mut store,
301
+ &instance,
302
+ &memory,
303
+ bytes.len(),
304
+ )? as i32;
305
+ let mem_slice = memory
306
+ .data_mut(&mut store)
307
+ .get_mut(ptr as usize..(ptr as usize) + bytes.len())
308
+ .ok_or_else(|| {
309
+ "Failed to get memory slice for string".to_string()
310
+ })?;
311
+ mem_slice.copy_from_slice(bytes);
312
+ let _ = setter.call(&mut store, (ptr, bytes.len() as i32));
313
+ any_called = true;
314
+ logger.log_message(
315
+ LogLevel::Debug,
316
+ &format!("Called string setter '{}' with '{}'", c, v),
317
+ );
318
+ }
319
+ Err(_) => {
320
+ logger.log_message(
321
+ LogLevel::Debug,
322
+ &format!(
323
+ "Export '{}' declared but signature lookup failed",
324
+ c
325
+ ),
326
+ );
327
+ }
328
+ }
329
+ }
330
+ }
331
+ }
332
+
333
+ if !any_called {
334
+ for fname in &candidates {
335
+ if let Ok(setter) =
336
+ instance.get_typed_func::<(i32, i32), ()>(&mut store, fname)
337
+ {
338
+ let bytes = v.as_bytes();
339
+ let ptr =
340
+ Self::alloc_temp(&mut store, &instance, &memory, bytes.len())?
341
+ as i32;
342
+ let mem_slice = memory
343
+ .data_mut(&mut store)
344
+ .get_mut(ptr as usize..(ptr as usize) + bytes.len())
345
+ .ok_or_else(|| {
346
+ "Failed to get memory slice for string".to_string()
347
+ })?;
348
+ mem_slice.copy_from_slice(bytes);
349
+ let _ = setter.call(&mut store, (ptr, bytes.len() as i32));
350
+ any_called = true;
351
+ logger.log_message(
352
+ LogLevel::Debug,
353
+ &format!(
354
+ "Dynamically called string setter '{}' with '{}'",
355
+ fname, v
356
+ ),
357
+ );
358
+ }
359
+ }
360
+ }
361
+
362
+ if !any_called {
363
+ logger.log_message(
364
+ LogLevel::Warning,
365
+ &format!("No string setter found/called for param '{}'", k),
366
+ );
367
+ }
368
+ }
369
+ }
370
+ }
371
+
372
+ // Try specific + generic entry points in order of preference.
373
+ // 1) render_note_<name>
374
+ // 2) render_note
375
+ // 3) synth_<name>
376
+ // 4) synth
377
+ let mut func_opt: Option<RenderFunc> = None;
378
+ if let Some(name) = synth_name {
379
+ let specific = format!("render_note_{}", name);
380
+ if let Ok(f) = instance
381
+ .get_typed_func::<(i32, i32, f32, f32, i32, i32, i32), ()>(&mut store, &specific)
382
+ {
383
+ func_opt = Some(f);
384
+ }
385
+ }
386
+ if func_opt.is_none() {
387
+ if let Ok(f) = instance.get_typed_func::<(i32, i32, f32, f32, i32, i32, i32), ()>(
388
+ &mut store,
389
+ "render_note",
390
+ ) {
391
+ func_opt = Some(f);
392
+ }
393
+ }
394
+ if func_opt.is_none() {
395
+ if let Some(name) = synth_name {
396
+ let specific = format!("synth_{}", name);
397
+ if let Ok(f) = instance.get_typed_func::<(i32, i32, f32, f32, i32, i32, i32), ()>(
398
+ &mut store, &specific,
399
+ ) {
400
+ func_opt = Some(f);
401
+ }
402
+ }
403
+ }
404
+ if func_opt.is_none() {
405
+ if let Ok(f) = instance
406
+ .get_typed_func::<(i32, i32, f32, f32, i32, i32, i32), ()>(&mut store, "synth")
407
+ {
408
+ func_opt = Some(f);
409
+ }
410
+ }
411
+ let func = func_opt.ok_or_else(|| {
412
+ "Exported function not found: tried render_note[_<name>] and synth[_<name>]".to_string()
413
+ })?;
414
+
415
+ // Copy host buffer into wasm memory
416
+ let byte_len = std::mem::size_of_val(buffer) as i32;
417
+ let ptr = Self::alloc_temp(&mut store, &instance, &memory, byte_len as usize)? as i32;
418
+ let mem_slice = memory
419
+ .data_mut(&mut store)
420
+ .get_mut(ptr as usize..(ptr as usize) + (byte_len as usize))
421
+ .ok_or_else(|| "Failed to get memory slice".to_string())?;
422
+ let src_bytes =
423
+ unsafe { std::slice::from_raw_parts(buffer.as_ptr() as *const u8, byte_len as usize) };
424
+ mem_slice.copy_from_slice(src_bytes);
425
+
426
+ // Call render
427
+ func.call(
428
+ &mut store,
429
+ (
430
+ ptr,
431
+ buffer.len() as i32,
432
+ freq,
433
+ amp,
434
+ duration_ms,
435
+ sample_rate,
436
+ channels,
437
+ ),
438
+ )
439
+ .map_err(|e| format!("Error calling `render_note`: {e}"))?;
440
+
441
+ // Copy back
442
+ let mem_slice_after = memory
443
+ .data(&store)
444
+ .get(ptr as usize..(ptr as usize) + (byte_len as usize))
445
+ .ok_or_else(|| "Failed to get memory slice after".to_string())?;
446
+ let dst_bytes = unsafe {
447
+ std::slice::from_raw_parts_mut(buffer.as_mut_ptr() as *mut u8, byte_len as usize)
448
+ };
449
+ dst_bytes.copy_from_slice(mem_slice_after);
450
+
451
+ Ok(())
452
+ }
453
+
454
+ fn alloc_temp(
455
+ store: &mut Store<()>,
456
+ instance: &Instance,
457
+ memory: &wasmtime::Memory,
458
+ size: usize,
459
+ ) -> Result<usize, String> {
460
+ // Try to use an exported `__wbindgen_malloc` if present; otherwise, grow memory manually.
461
+ if let Ok(malloc) = instance.get_typed_func::<i32, i32>(&mut *store, "__wbindgen_malloc") {
462
+ let ptr = malloc
463
+ .call(&mut *store, size as i32)
464
+ .map_err(|e| format!("malloc failed: {e}"))? as usize;
465
+ return Ok(ptr);
466
+ }
467
+
468
+ // Fallback: grow memory and use end of memory as scratch space
469
+ let current_len = memory.data_size(&mut *store);
470
+ let need = size;
471
+ let pages_needed = (current_len + need).div_ceil(0x10000) as u64; // 64KiB pages
472
+ let current_pages = memory.size(&mut *store);
473
+ if pages_needed > current_pages {
474
+ let to_grow = pages_needed - current_pages;
475
+ memory
476
+ .grow(&mut *store, to_grow)
477
+ .map_err(|e| format!("memory.grow failed: {e}"))?;
478
+ }
479
+ Ok(current_len)
480
+ }
481
+ }