@devaloop/devalang 0.0.1-beta.1 → 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 (220) hide show
  1. package/.devalang +9 -10
  2. package/Cargo.toml +5 -4
  3. package/README.md +7 -5
  4. package/docs/CHANGELOG.md +42 -0
  5. package/docs/ROADMAP.md +5 -1
  6. package/docs/TODO.md +3 -14
  7. package/examples/bus.deva +10 -0
  8. package/examples/effect.deva +2 -0
  9. package/examples/filter.deva +11 -0
  10. package/examples/lfo.deva +9 -0
  11. package/examples/synth.deva +11 -1
  12. package/examples/synth_types.deva +17 -0
  13. package/out-tsc/core/functions/index.d.ts +5 -0
  14. package/out-tsc/core/functions/index.js +11 -0
  15. package/out-tsc/pkg/devalang_core.d.ts +2 -0
  16. package/out-tsc/pkg/devalang_core.js +17 -2
  17. package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +8 -7
  18. package/package.json +1 -1
  19. package/project-version.json +3 -3
  20. package/rust/cli/bank/api.rs +122 -122
  21. package/rust/cli/bank/commands.rs +33 -2
  22. package/rust/cli/bank/mod.rs +29 -29
  23. package/rust/cli/build/commands.rs +53 -3
  24. package/rust/cli/build/mod.rs +2 -2
  25. package/rust/cli/build/process.rs +26 -7
  26. package/rust/cli/check/mod.rs +2 -2
  27. package/rust/cli/discover/commands.rs +253 -253
  28. package/rust/cli/discover/config.rs +111 -111
  29. package/rust/cli/discover/fs.rs +19 -19
  30. package/rust/cli/discover/install.rs +103 -103
  31. package/rust/cli/discover/metadata.rs +48 -48
  32. package/rust/cli/discover/mod.rs +5 -5
  33. package/rust/cli/install/addon.rs +118 -118
  34. package/rust/cli/install/bank.rs +22 -3
  35. package/rust/cli/install/commands.rs +35 -35
  36. package/rust/cli/install/mod.rs +4 -4
  37. package/rust/cli/install/plugin.rs +80 -61
  38. package/rust/cli/login/commands.rs +124 -124
  39. package/rust/cli/mod.rs +12 -12
  40. package/rust/cli/parser.rs +46 -1
  41. package/rust/cli/play/commands.rs +71 -20
  42. package/rust/cli/play/mod.rs +5 -5
  43. package/rust/cli/play/process.rs +14 -5
  44. package/rust/cli/play/realtime.rs +91 -91
  45. package/rust/cli/telemetry/commands.rs +22 -22
  46. package/rust/cli/telemetry/event_creator.rs +80 -80
  47. package/rust/cli/telemetry/mod.rs +3 -3
  48. package/rust/cli/telemetry/send.rs +51 -51
  49. package/rust/cli/template/commands.rs +69 -69
  50. package/rust/config/driver.rs +112 -103
  51. package/rust/config/mod.rs +3 -3
  52. package/rust/config/ops.rs +26 -26
  53. package/rust/config/settings.rs +101 -101
  54. package/rust/core/audio/engine/driver.rs +220 -0
  55. package/rust/core/audio/engine/export.rs +169 -0
  56. package/rust/core/audio/engine/helpers.rs +178 -170
  57. package/rust/core/audio/engine/mod.rs +51 -2
  58. package/rust/core/audio/engine/notes/dsp.rs +85 -0
  59. package/rust/core/audio/engine/notes/mod.rs +44 -0
  60. package/rust/core/audio/engine/notes/params.rs +294 -0
  61. package/rust/core/audio/engine/sample/insert.rs +199 -0
  62. package/rust/core/audio/engine/sample/mod.rs +40 -0
  63. package/rust/core/audio/engine/sample/padding.rs +170 -0
  64. package/rust/core/audio/evaluator/condition.rs +61 -0
  65. package/rust/core/audio/evaluator/mod.rs +9 -0
  66. package/rust/core/audio/{evaluator.rs → evaluator/numeric.rs} +1 -159
  67. package/rust/core/audio/evaluator/rhs.rs +16 -0
  68. package/rust/core/audio/evaluator/string_expr.rs +94 -0
  69. package/rust/core/audio/interpreter/driver.rs +55 -23
  70. package/rust/core/audio/interpreter/mod.rs +1 -13
  71. package/rust/core/audio/interpreter/statements/arrow_call/interprete.rs +175 -0
  72. package/rust/core/audio/interpreter/statements/arrow_call/methods/chord.rs +384 -0
  73. package/rust/core/audio/interpreter/statements/arrow_call/methods/mod.rs +2 -0
  74. package/rust/core/audio/interpreter/statements/arrow_call/methods/note.rs +316 -0
  75. package/rust/core/audio/interpreter/statements/arrow_call/mod.rs +3 -0
  76. package/rust/core/audio/interpreter/statements/arrow_call/types/arp.rs +192 -0
  77. package/rust/core/audio/interpreter/statements/arrow_call/types/mod.rs +24 -0
  78. package/rust/core/audio/interpreter/statements/arrow_call/types/pad.rs +116 -0
  79. package/rust/core/audio/interpreter/statements/arrow_call/types/pluck.rs +97 -0
  80. package/rust/core/audio/interpreter/statements/arrow_call/types/sub.rs +100 -0
  81. package/rust/core/audio/interpreter/{automate.rs → statements/automate.rs} +16 -18
  82. package/rust/core/audio/interpreter/{call.rs → statements/call.rs} +5 -4
  83. package/rust/core/audio/interpreter/{condition.rs → statements/condition.rs} +2 -1
  84. package/rust/core/audio/interpreter/{function.rs → statements/function.rs} +2 -4
  85. package/rust/core/audio/interpreter/{let_.rs → statements/let_.rs} +2 -4
  86. package/rust/core/audio/interpreter/{load.rs → statements/load.rs} +2 -4
  87. package/rust/core/audio/interpreter/{loop_.rs → statements/loop_.rs} +2 -1
  88. package/rust/core/audio/interpreter/statements/mod.rs +12 -0
  89. package/rust/core/audio/interpreter/{sleep.rs → statements/sleep.rs} +28 -28
  90. package/rust/core/audio/interpreter/{spawn.rs → statements/spawn.rs} +3 -2
  91. package/rust/core/audio/interpreter/{tempo.rs → statements/tempo.rs} +40 -40
  92. package/rust/core/audio/interpreter/{trigger.rs → statements/trigger.rs} +1 -1
  93. package/rust/core/audio/loader/trigger.rs +2 -1
  94. package/rust/core/audio/mod.rs +6 -7
  95. package/rust/core/audio/player.rs +70 -70
  96. package/rust/core/audio/special/easing.rs +189 -189
  97. package/rust/core/audio/special/env.rs +45 -45
  98. package/rust/core/audio/special/math.rs +134 -134
  99. package/rust/core/audio/special/mod.rs +9 -9
  100. package/rust/core/audio/special/modulator.rs +143 -143
  101. package/rust/core/builder/mod.rs +45 -2
  102. package/rust/core/debugger/lexer.rs +27 -27
  103. package/rust/core/debugger/{module.rs → logs.rs} +3 -6
  104. package/rust/core/debugger/mod.rs +30 -30
  105. package/rust/core/debugger/preprocessor.rs +27 -27
  106. package/rust/core/debugger/store.rs +2 -4
  107. package/rust/core/error/mod.rs +269 -269
  108. package/rust/core/lexer/driver.rs +59 -61
  109. package/rust/core/lexer/handler/arrow.rs +82 -82
  110. package/rust/core/lexer/handler/at.rs +21 -21
  111. package/rust/core/lexer/handler/brace.rs +41 -41
  112. package/rust/core/lexer/handler/colon.rs +21 -21
  113. package/rust/core/lexer/handler/comment.rs +30 -30
  114. package/rust/core/lexer/handler/dot.rs +21 -21
  115. package/rust/core/lexer/handler/driver.rs +337 -337
  116. package/rust/core/lexer/handler/identifier.rs +47 -47
  117. package/rust/core/lexer/handler/indent.rs +66 -66
  118. package/rust/core/lexer/handler/mod.rs +15 -15
  119. package/rust/core/lexer/handler/newline.rs +23 -23
  120. package/rust/core/lexer/handler/number.rs +31 -31
  121. package/rust/core/lexer/handler/operator.rs +46 -46
  122. package/rust/core/lexer/handler/parenthesis.rs +41 -41
  123. package/rust/core/lexer/handler/slash.rs +21 -21
  124. package/rust/core/lexer/handler/string.rs +63 -63
  125. package/rust/core/lexer/mod.rs +3 -3
  126. package/rust/core/mod.rs +0 -1
  127. package/rust/core/parser/driver/block.rs +111 -0
  128. package/rust/core/parser/driver/cursor.rs +82 -0
  129. package/rust/core/parser/driver/driver_impl.rs +139 -0
  130. package/rust/core/parser/driver/mod.rs +6 -0
  131. package/rust/core/parser/driver/parse_array.rs +120 -0
  132. package/rust/core/parser/driver/parse_map.rs +223 -0
  133. package/rust/core/parser/driver/parser.rs +160 -0
  134. package/rust/core/parser/handler/arrow_call.rs +28 -4
  135. package/rust/core/parser/handler/at.rs +279 -279
  136. package/rust/core/parser/handler/bank.rs +104 -104
  137. package/rust/core/parser/handler/condition.rs +83 -83
  138. package/rust/core/parser/handler/dot.rs +148 -148
  139. package/rust/core/parser/handler/identifier/automate.rs +254 -254
  140. package/rust/core/parser/handler/identifier/call.rs +91 -91
  141. package/rust/core/parser/handler/identifier/emit.rs +70 -70
  142. package/rust/core/parser/handler/identifier/function.rs +113 -113
  143. package/rust/core/parser/handler/identifier/group.rs +89 -89
  144. package/rust/core/parser/handler/identifier/let_.rs +173 -173
  145. package/rust/core/parser/handler/identifier/mod.rs +55 -55
  146. package/rust/core/parser/handler/identifier/on.rs +107 -107
  147. package/rust/core/parser/handler/identifier/print.rs +49 -49
  148. package/rust/core/parser/handler/identifier/sleep.rs +96 -43
  149. package/rust/core/parser/handler/identifier/spawn.rs +91 -91
  150. package/rust/core/parser/handler/identifier/synth.rs +135 -135
  151. package/rust/core/parser/handler/loop_.rs +194 -194
  152. package/rust/core/parser/handler/mod.rs +9 -9
  153. package/rust/core/parser/handler/pattern.rs +1 -1
  154. package/rust/core/parser/handler/tempo.rs +105 -57
  155. package/rust/core/parser/statement.rs +10 -11
  156. package/rust/core/plugin/loader.rs +1 -1
  157. package/rust/core/plugin/mod.rs +2 -2
  158. package/rust/core/plugin/runner/mod.rs +11 -0
  159. package/rust/core/plugin/{runner.rs → runner/non_wasm.rs} +297 -347
  160. package/rust/core/plugin/runner/wasm32.rs +43 -0
  161. package/rust/core/preprocessor/loader/inject.rs +278 -0
  162. package/rust/core/preprocessor/loader/loader_helpers.rs +110 -0
  163. package/rust/core/preprocessor/loader/mod.rs +235 -0
  164. package/rust/core/preprocessor/module.rs +2 -7
  165. package/rust/core/preprocessor/{processor.rs → processor/handlers.rs} +6 -13
  166. package/rust/core/preprocessor/processor/mod.rs +1 -0
  167. package/rust/core/preprocessor/resolver/bank.rs +49 -49
  168. package/rust/core/preprocessor/resolver/call.rs +124 -124
  169. package/rust/core/preprocessor/resolver/condition.rs +95 -95
  170. package/rust/core/preprocessor/resolver/driver.rs +324 -324
  171. package/rust/core/preprocessor/resolver/function.rs +2 -2
  172. package/rust/core/preprocessor/resolver/group.rs +46 -18
  173. package/rust/core/preprocessor/resolver/let_.rs +32 -32
  174. package/rust/core/preprocessor/resolver/loop_.rs +318 -318
  175. package/rust/core/preprocessor/resolver/mod.rs +16 -16
  176. package/rust/core/preprocessor/resolver/pattern.rs +83 -83
  177. package/rust/core/preprocessor/resolver/spawn.rs +99 -99
  178. package/rust/core/preprocessor/resolver/synth.rs +54 -54
  179. package/rust/core/preprocessor/resolver/tempo.rs +48 -48
  180. package/rust/core/preprocessor/resolver/trigger.rs +116 -116
  181. package/rust/core/preprocessor/resolver/value.rs +176 -176
  182. package/rust/core/store/global.rs +2 -6
  183. package/rust/core/store/mod.rs +1 -5
  184. package/rust/lib.rs +18 -3
  185. package/rust/main.rs +27 -3
  186. package/rust/types/Cargo.toml +1 -1
  187. package/rust/types/src/addons.rs +55 -55
  188. package/rust/types/src/config.rs +84 -74
  189. package/rust/types/src/lib.rs +15 -12
  190. package/rust/types/src/plugin.rs +20 -0
  191. package/rust/types/src/store.rs +139 -0
  192. package/rust/types/src/telemetry.rs +85 -85
  193. package/rust/utils/Cargo.toml +2 -2
  194. package/rust/utils/src/file.rs +94 -94
  195. package/rust/utils/src/first_usage.rs +97 -97
  196. package/rust/utils/src/lib.rs +9 -9
  197. package/rust/utils/src/logger.rs +200 -200
  198. package/rust/utils/src/path.rs +129 -88
  199. package/rust/utils/src/signature.rs +41 -41
  200. package/rust/utils/src/spinner.rs +20 -20
  201. package/rust/utils/src/version.rs +27 -27
  202. package/rust/utils/src/watcher.rs +46 -46
  203. package/rust/web/api.rs +5 -5
  204. package/rust/web/cdn.rs +34 -34
  205. package/rust/web/mod.rs +3 -3
  206. package/tests/integration.rs +21 -21
  207. package/typescript/core/functions/index.ts +11 -0
  208. package/typescript/pkg/devalang_core.ts +20 -4
  209. package/rust/core/audio/engine/sample.rs +0 -366
  210. package/rust/core/audio/engine/synth.rs +0 -325
  211. package/rust/core/audio/interpreter/arrow_call.rs +0 -311
  212. package/rust/core/audio/renderer.rs +0 -54
  213. package/rust/core/parser/driver.rs +0 -584
  214. package/rust/core/preprocessor/loader.rs +0 -637
  215. package/rust/core/store/export.rs +0 -28
  216. package/rust/core/store/function.rs +0 -40
  217. package/rust/core/store/import.rs +0 -28
  218. package/rust/core/store/variable.rs +0 -51
  219. package/rust/core/utils/mod.rs +0 -1
  220. package/rust/core/utils/path.rs +0 -37
@@ -1,170 +1,178 @@
1
- use devalang_types::Value;
2
- use std::collections::HashMap;
3
-
4
- type OptMap = Option<HashMap<String, Value>>;
5
-
6
- pub fn env_maps_from_automation(
7
- automation: &Option<HashMap<String, Value>>,
8
- ) -> (OptMap, OptMap, OptMap) {
9
- if let Some(auto) = automation {
10
- let vol = match auto.get("volume") {
11
- Some(Value::Map(m)) => Some(m.clone()),
12
- _ => None,
13
- };
14
- let pan = match auto.get("pan") {
15
- Some(Value::Map(m)) => Some(m.clone()),
16
- _ => None,
17
- };
18
- let pit = match auto.get("pitch") {
19
- Some(Value::Map(m)) => Some(m.clone()),
20
- _ => None,
21
- };
22
- (vol, pan, pit)
23
- } else {
24
- (None, None, None)
25
- }
26
- }
27
-
28
- pub fn eval_env_map(
29
- env_opt: &Option<HashMap<String, Value>>,
30
- progress: f32,
31
- default_val: f32,
32
- ) -> f32 {
33
- let env = match env_opt {
34
- Some(m) => m,
35
- None => {
36
- return default_val;
37
- }
38
- };
39
- let mut points: Vec<(f32, f32)> = Vec::with_capacity(env.len());
40
- for (k, v) in env.iter() {
41
- let key = if k.ends_with('%') {
42
- &k[..k.len() - 1]
43
- } else {
44
- &k[..]
45
- };
46
- if let Ok(mut p) = key.parse::<f32>() {
47
- p = (p / 100.0).clamp(0.0, 1.0);
48
- let val = match v {
49
- Value::Number(n) => *n,
50
- Value::String(s) => s.parse::<f32>().unwrap_or(default_val),
51
- Value::Identifier(s) => s.parse::<f32>().unwrap_or(default_val),
52
- _ => default_val,
53
- };
54
- points.push((p, val));
55
- }
56
- }
57
- if points.is_empty() {
58
- return default_val;
59
- }
60
- points.sort_by(|a, b| a.0.partial_cmp(&b.0).unwrap());
61
- let t = progress.clamp(0.0, 1.0);
62
- if t <= points[0].0 {
63
- return points[0].1;
64
- }
65
- if t >= points[points.len() - 1].0 {
66
- return points[points.len() - 1].1;
67
- }
68
- for w in points.windows(2) {
69
- let (p0, v0) = w[0];
70
- let (p1, v1) = w[1];
71
- if t >= p0 && t <= p1 {
72
- let ratio = if (p1 - p0).abs() < f32::EPSILON {
73
- 0.0
74
- } else {
75
- (t - p0) / (p1 - p0)
76
- };
77
- return v0 + (v1 - v0) * ratio;
78
- }
79
- }
80
- default_val
81
- }
82
-
83
- pub fn oscillator_sample(waveform: &str, current_freq: f32, t: f32) -> f32 {
84
- let phase = 2.0 * std::f32::consts::PI * current_freq * t;
85
- match waveform {
86
- "sine" => phase.sin(),
87
- "square" => {
88
- if phase.sin() >= 0.0 {
89
- 1.0
90
- } else {
91
- -1.0
92
- }
93
- }
94
- "saw" => 2.0 * (current_freq * t - (current_freq * t + 0.5).floor()),
95
- "triangle" => (2.0 * (2.0 * (current_freq * t).fract() - 1.0)).abs() * 2.0 - 1.0,
96
- _ => 0.0,
97
- }
98
- }
99
-
100
- pub fn adsr_envelope_value(
101
- i: usize,
102
- attack_samples: usize,
103
- decay_samples: usize,
104
- sustain_samples: usize,
105
- release_samples: usize,
106
- sustain_level: f32,
107
- ) -> f32 {
108
- let attack_start = 0usize;
109
- let decay_start = attack_samples;
110
- let sustain_start = attack_samples + decay_samples;
111
- let release_start = attack_samples + decay_samples + sustain_samples;
112
-
113
- if i < attack_start + attack_samples && attack_samples > 0 {
114
- let k = i - attack_start;
115
- let denom = if attack_samples > 1 {
116
- (attack_samples - 1) as f32
117
- } else {
118
- 1.0
119
- };
120
- (k as f32) / denom
121
- } else if i < decay_start + decay_samples && decay_samples > 0 {
122
- let k = i - decay_start;
123
- let denom = if decay_samples > 1 {
124
- (decay_samples - 1) as f32
125
- } else {
126
- 1.0
127
- };
128
- let ratio = (k as f32) / denom;
129
- 1.0 - (1.0 - sustain_level) * ratio
130
- } else if i < sustain_start + sustain_samples {
131
- sustain_level
132
- } else if release_samples > 0 {
133
- // release: interpolate from sustain_level down to 0 inclusive
134
- let k = i.saturating_sub(release_start);
135
- let denom = if release_samples > 1 {
136
- (release_samples - 1) as f32
137
- } else {
138
- 1.0
139
- };
140
- let ratio = (k as f32) / denom;
141
- let val = sustain_level * (1.0 - ratio);
142
- if val < 0.0 { 0.0 } else { val }
143
- } else {
144
- 0.0
145
- }
146
- }
147
-
148
- pub fn pan_gains(pan_val: f32) -> (f32, f32) {
149
- let left_gain = 1.0 - pan_val.max(0.0);
150
- let right_gain = 1.0 + pan_val.min(0.0);
151
- (left_gain, right_gain)
152
- }
153
-
154
- pub fn mix_stereo_samples_into_buffer(
155
- engine: &mut super::synth::AudioEngine,
156
- start_sample: usize,
157
- channels: usize,
158
- stereo_samples: &[i16],
159
- ) {
160
- let offset = start_sample * channels;
161
- let required_len = offset + stereo_samples.len();
162
-
163
- if engine.buffer.len() < required_len {
164
- engine.buffer.resize(required_len, 0);
165
- }
166
-
167
- for (i, sample) in stereo_samples.iter().enumerate() {
168
- engine.buffer[offset + i] = engine.buffer[offset + i].saturating_add(*sample);
169
- }
170
- }
1
+ use devalang_types::Value;
2
+ use std::collections::HashMap;
3
+
4
+ type OptMap = Option<HashMap<String, Value>>;
5
+
6
+ pub fn env_maps_from_automation(
7
+ automation: &Option<HashMap<String, Value>>,
8
+ ) -> (OptMap, OptMap, OptMap) {
9
+ if let Some(auto) = automation {
10
+ let vol = match auto.get("volume") {
11
+ Some(Value::Map(m)) => Some(m.clone()),
12
+ _ => None,
13
+ };
14
+ let pan = match auto.get("pan") {
15
+ Some(Value::Map(m)) => Some(m.clone()),
16
+ _ => None,
17
+ };
18
+ let pit = match auto.get("pitch") {
19
+ Some(Value::Map(m)) => Some(m.clone()),
20
+ _ => None,
21
+ };
22
+ (vol, pan, pit)
23
+ } else {
24
+ (None, None, None)
25
+ }
26
+ }
27
+
28
+ /// Convert an Option<HashMap<..>> into a HashMap (empty if None).
29
+ pub fn env_map_to_hash(opt: &Option<HashMap<String, Value>>) -> HashMap<String, Value> {
30
+ match opt {
31
+ Some(m) => m.clone(),
32
+ None => HashMap::new(),
33
+ }
34
+ }
35
+
36
+ pub fn eval_env_map(
37
+ env_opt: &Option<HashMap<String, Value>>,
38
+ progress: f32,
39
+ default_val: f32,
40
+ ) -> f32 {
41
+ let env = match env_opt {
42
+ Some(m) => m,
43
+ None => {
44
+ return default_val;
45
+ }
46
+ };
47
+ let mut points: Vec<(f32, f32)> = Vec::with_capacity(env.len());
48
+ for (k, v) in env.iter() {
49
+ let key = if k.ends_with('%') {
50
+ &k[..k.len() - 1]
51
+ } else {
52
+ &k[..]
53
+ };
54
+ if let Ok(mut p) = key.parse::<f32>() {
55
+ p = (p / 100.0).clamp(0.0, 1.0);
56
+ let val = match v {
57
+ Value::Number(n) => *n,
58
+ Value::String(s) => s.parse::<f32>().unwrap_or(default_val),
59
+ Value::Identifier(s) => s.parse::<f32>().unwrap_or(default_val),
60
+ _ => default_val,
61
+ };
62
+ points.push((p, val));
63
+ }
64
+ }
65
+ if points.is_empty() {
66
+ return default_val;
67
+ }
68
+ points.sort_by(|a, b| a.0.partial_cmp(&b.0).unwrap());
69
+ let t = progress.clamp(0.0, 1.0);
70
+ if t <= points[0].0 {
71
+ return points[0].1;
72
+ }
73
+ if t >= points[points.len() - 1].0 {
74
+ return points[points.len() - 1].1;
75
+ }
76
+ for w in points.windows(2) {
77
+ let (p0, v0) = w[0];
78
+ let (p1, v1) = w[1];
79
+ if t >= p0 && t <= p1 {
80
+ let ratio = if (p1 - p0).abs() < f32::EPSILON {
81
+ 0.0
82
+ } else {
83
+ (t - p0) / (p1 - p0)
84
+ };
85
+ return v0 + (v1 - v0) * ratio;
86
+ }
87
+ }
88
+ default_val
89
+ }
90
+
91
+ pub fn oscillator_sample(waveform: &str, current_freq: f32, t: f32) -> f32 {
92
+ let phase = 2.0 * std::f32::consts::PI * current_freq * t;
93
+ match waveform {
94
+ "sine" => phase.sin(),
95
+ "square" => {
96
+ if phase.sin() >= 0.0 {
97
+ 1.0
98
+ } else {
99
+ -1.0
100
+ }
101
+ }
102
+ "saw" => 2.0 * (current_freq * t - (current_freq * t + 0.5).floor()),
103
+ "triangle" => (2.0 * (2.0 * (current_freq * t).fract() - 1.0)).abs() * 2.0 - 1.0,
104
+ _ => 0.0,
105
+ }
106
+ }
107
+
108
+ pub fn adsr_envelope_value(
109
+ i: usize,
110
+ attack_samples: usize,
111
+ decay_samples: usize,
112
+ sustain_samples: usize,
113
+ release_samples: usize,
114
+ sustain_level: f32,
115
+ ) -> f32 {
116
+ let attack_start = 0usize;
117
+ let decay_start = attack_samples;
118
+ let sustain_start = attack_samples + decay_samples;
119
+ let release_start = attack_samples + decay_samples + sustain_samples;
120
+
121
+ if i < attack_start + attack_samples && attack_samples > 0 {
122
+ let k = i - attack_start;
123
+ let denom = if attack_samples > 1 {
124
+ (attack_samples - 1) as f32
125
+ } else {
126
+ 1.0
127
+ };
128
+ (k as f32) / denom
129
+ } else if i < decay_start + decay_samples && decay_samples > 0 {
130
+ let k = i - decay_start;
131
+ let denom = if decay_samples > 1 {
132
+ (decay_samples - 1) as f32
133
+ } else {
134
+ 1.0
135
+ };
136
+ let ratio = (k as f32) / denom;
137
+ 1.0 - (1.0 - sustain_level) * ratio
138
+ } else if i < sustain_start + sustain_samples {
139
+ sustain_level
140
+ } else if release_samples > 0 {
141
+ // release: interpolate from sustain_level down to 0 inclusive
142
+ let k = i.saturating_sub(release_start);
143
+ let denom = if release_samples > 1 {
144
+ (release_samples - 1) as f32
145
+ } else {
146
+ 1.0
147
+ };
148
+ let ratio = (k as f32) / denom;
149
+ let val = sustain_level * (1.0 - ratio);
150
+ if val < 0.0 { 0.0 } else { val }
151
+ } else {
152
+ 0.0
153
+ }
154
+ }
155
+
156
+ pub fn pan_gains(pan_val: f32) -> (f32, f32) {
157
+ let left_gain = 1.0 - pan_val.max(0.0);
158
+ let right_gain = 1.0 + pan_val.min(0.0);
159
+ (left_gain, right_gain)
160
+ }
161
+
162
+ pub fn mix_stereo_samples_into_buffer(
163
+ engine: &mut super::driver::AudioEngine,
164
+ start_sample: usize,
165
+ channels: usize,
166
+ stereo_samples: &[i16],
167
+ ) {
168
+ let offset = start_sample * channels;
169
+ let required_len = offset + stereo_samples.len();
170
+
171
+ if engine.buffer.len() < required_len {
172
+ engine.buffer.resize(required_len, 0);
173
+ }
174
+
175
+ for (i, sample) in stereo_samples.iter().enumerate() {
176
+ engine.buffer[offset + i] = engine.buffer[offset + i].saturating_add(*sample);
177
+ }
178
+ }
@@ -1,7 +1,56 @@
1
+ pub mod driver;
2
+ pub mod export;
1
3
  pub mod helpers;
4
+ pub mod notes;
2
5
  pub mod sample;
3
- pub mod synth;
4
6
 
5
- pub use synth::AudioEngine;
7
+ pub use driver::AudioEngine;
6
8
 
9
+ pub use driver::CHANNELS;
10
+ pub use driver::MidiNoteEvent;
11
+ pub use driver::SAMPLE_RATE;
7
12
  pub use helpers::*;
13
+
14
+ use crate::core::audio::interpreter::driver::run_audio_program;
15
+ use crate::core::parser::statement::Statement;
16
+ use crate::core::store::global::GlobalStore;
17
+ use devalang_types::{FunctionTable, VariableTable};
18
+ use std::collections::HashMap;
19
+
20
+ /// Render audio for a set of parsed modules.
21
+ ///
22
+ /// For each entry in `modules` (module path -> statements), create an
23
+ /// AudioEngine, setup empty module-local VariableTable and FunctionTable,
24
+ /// invoke the interpreter driver to schedule audio into the engine, and
25
+ /// return a map of module name -> AudioEngine.
26
+ pub fn render_audio_with_modules(
27
+ modules: HashMap<String, Vec<Statement>>,
28
+ _output_dir: &str,
29
+ global_store: &mut GlobalStore,
30
+ ) -> HashMap<String, AudioEngine> {
31
+ let mut result: HashMap<String, AudioEngine> = HashMap::new();
32
+
33
+ for (module_name, statements) in modules.into_iter() {
34
+ // Create engine named after module (used for diagnostics)
35
+ let mut engine = AudioEngine::new(module_name.clone());
36
+
37
+ // Create empty module-local tables; interpreter expects these as starting context
38
+ let module_vars = VariableTable::new();
39
+ let module_funcs = FunctionTable::new();
40
+
41
+ // Run interpreter which will populate the engine.buffer and midi events
42
+ let (_max_end_time, _cursor_time) = run_audio_program(
43
+ &statements,
44
+ &mut engine,
45
+ module_name.clone(),
46
+ module_name.clone(),
47
+ module_vars,
48
+ module_funcs,
49
+ global_store,
50
+ );
51
+
52
+ result.insert(module_name, engine);
53
+ }
54
+
55
+ result
56
+ }
@@ -0,0 +1,85 @@
1
+ use crate::core::audio::engine::notes::params::NoteSetup;
2
+ use devalang_types::Value;
3
+ use std::collections::HashMap;
4
+
5
+ pub fn render_notes_into_buffer(
6
+ engine: &mut crate::core::audio::engine::AudioEngine,
7
+ _waveform: &str,
8
+ _freq: f32,
9
+ _amp: f32,
10
+ _start_time_ms: f32,
11
+ _duration_ms: f32,
12
+ _synth_params: HashMap<String, Value>,
13
+ _note_params: HashMap<String, Value>,
14
+ _automation: Option<HashMap<String, Value>>,
15
+ setup: NoteSetup,
16
+ ) {
17
+ use crate::core::audio::engine::helpers;
18
+
19
+ let sample_rate = setup.sample_rate;
20
+ let channels = setup.channels;
21
+ let total_samples = setup.total_samples;
22
+ let start_sample = setup.start_sample;
23
+
24
+ let mut stereo_samples: Vec<i16> = Vec::with_capacity(total_samples * channels);
25
+ let fade_len = (sample_rate * 0.01) as usize; // 10ms fade
26
+
27
+ for i in 0..total_samples {
28
+ let t = ((start_sample + i) as f32) / sample_rate;
29
+
30
+ // simplified voice/unison and oscillator sampling
31
+ let mut value = helpers::oscillator_sample(_waveform, _freq, t);
32
+
33
+ // apply ADSR envelope
34
+ let envelope = helpers::adsr_envelope_value(
35
+ i,
36
+ setup.attack_samples,
37
+ setup.decay_samples,
38
+ if total_samples > setup.attack_samples + setup.decay_samples + setup.release_samples {
39
+ total_samples - setup.attack_samples - setup.decay_samples - setup.release_samples
40
+ } else {
41
+ 0
42
+ },
43
+ setup.release_samples,
44
+ setup.sustain_level,
45
+ );
46
+
47
+ value *= envelope * (i16::MAX as f32) * _amp;
48
+
49
+ if fade_len > 0 && i < fade_len {
50
+ if fade_len == 1 {
51
+ value *= 0.0;
52
+ } else {
53
+ value *= (i as f32) / (fade_len as f32);
54
+ }
55
+ } else if fade_len > 0 && i >= total_samples.saturating_sub(fade_len) {
56
+ if fade_len == 1 {
57
+ value *= 0.0;
58
+ } else {
59
+ value *= ((total_samples - 1 - i) as f32) / ((fade_len - 1) as f32);
60
+ }
61
+ }
62
+
63
+ let (left_gain, right_gain) = helpers::pan_gains(0.0);
64
+ let left = (value * left_gain)
65
+ .round()
66
+ .clamp(i16::MIN as f32, i16::MAX as f32) as i16;
67
+ let right = (value * right_gain)
68
+ .round()
69
+ .clamp(i16::MIN as f32, i16::MAX as f32) as i16;
70
+
71
+ // push samples interleaved for channels=2, fallback zeros for other channel counts
72
+ if channels >= 2 {
73
+ stereo_samples.push(left);
74
+ stereo_samples.push(right);
75
+ } else if channels == 1 {
76
+ stereo_samples.push(((left as i32 + right as i32) / 2) as i16);
77
+ } else {
78
+ stereo_samples.push(left);
79
+ stereo_samples.push(right);
80
+ }
81
+ }
82
+
83
+ engine.note_count = engine.note_count.saturating_add(1);
84
+ helpers::mix_stereo_samples_into_buffer(engine, start_sample, channels, &stereo_samples);
85
+ }
@@ -0,0 +1,44 @@
1
+ use devalang_types::Value;
2
+ use std::collections::HashMap;
3
+
4
+ pub mod dsp;
5
+ pub mod params;
6
+
7
+ // Minimal safe facade: parse the setup then call DSP renderer.
8
+
9
+ pub fn insert_note_impl(
10
+ engine: &mut crate::core::audio::engine::AudioEngine,
11
+ waveform: String,
12
+ freq: f32,
13
+ amp: f32,
14
+ start_time_ms: f32,
15
+ duration_ms: f32,
16
+ synth_params: HashMap<String, Value>,
17
+ note_params: HashMap<String, Value>,
18
+ automation: Option<HashMap<String, Value>>,
19
+ ) {
20
+ let setup = params::build_note_setup(
21
+ engine,
22
+ &waveform,
23
+ freq,
24
+ amp,
25
+ start_time_ms,
26
+ duration_ms,
27
+ &synth_params,
28
+ &note_params,
29
+ &automation,
30
+ );
31
+
32
+ dsp::render_notes_into_buffer(
33
+ engine,
34
+ &waveform,
35
+ freq,
36
+ amp,
37
+ start_time_ms,
38
+ duration_ms,
39
+ synth_params,
40
+ note_params,
41
+ automation,
42
+ setup,
43
+ );
44
+ }