Compare commits
4 commits
8b189914f7
...
6bc54f8e46
Author | SHA1 | Date | |
---|---|---|---|
6bc54f8e46 | |||
20aeedeef2 | |||
9a5d4f14e5 | |||
c534721e88 |
6 changed files with 138 additions and 23 deletions
|
@ -23,6 +23,8 @@ where in the file you want the plugin to be ran.
|
|||
so, if you did `plugin 3 1...`, it would split the file in 3, then get the
|
||||
part that is of index 1 (starting at 0).
|
||||
|
||||
**Keep in mind parts start from the bottom of the file.**
|
||||
|
||||
## `load path_or_arg`
|
||||
|
||||
Load a file into memory. The file MUST end with the bmp extension (and so MUST
|
||||
|
@ -34,7 +36,29 @@ Run the eg-amp plugin over the given slice of the file.
|
|||
|
||||
## `rflanger split index delay_depth_avg law_freq`
|
||||
|
||||
Runs the Retro Flanger script from the SWH plugins.
|
||||
Run the Retro Flanger script from the SWH plugins.
|
||||
|
||||
- `delay_depth_avg` is for the `Average stall (ms)` parameter of the plugin.
|
||||
- `law_freq` is for the `Flange frequency` parameter of the plugin.
|
||||
|
||||
## `eq split index lo mid hi`
|
||||
|
||||
Run the DJ EQ plugin from the SWH plugins.
|
||||
|
||||
`lo`, `mid`, and `hi` are the respective dB gains for each frequency range.
|
||||
|
||||
All three ranges accept gains from -70dB to +6dB.
|
||||
Default is 0 for all (no action taken).
|
||||
|
||||
## `phaser split index lfo_rate lfo_depth fb spread`
|
||||
|
||||
Run the LFO Phaser plugin from the SWH plugins.
|
||||
|
||||
Parameters:
|
||||
- `lfo_rate`: LFO Rate (Hz), 0..100, default 25
|
||||
- `lfo_depth`: LFO depth, 0..1, default 0.25
|
||||
- `fb`: Feedback, -1..1, default 0
|
||||
- `spread`: Spread (octaves), 0..2, default 1
|
||||
|
||||
## TODO `echo split index delay`
|
||||
|
||||
|
|
4
examples/eq.scri
Normal file
4
examples/eq.scri
Normal file
|
@ -0,0 +1,4 @@
|
|||
load :0;
|
||||
eq 3 1 2 -0.3 0.7;
|
||||
eq 10 3 3 -0.9 2;
|
||||
quicksave;
|
3
examples/phaser.scri
Normal file
3
examples/phaser.scri
Normal file
|
@ -0,0 +1,3 @@
|
|||
load :0;
|
||||
phaser 3 1 25 0.25 0 1;
|
||||
quicksave;
|
|
@ -187,7 +187,7 @@ pub const Image = struct {
|
|||
|
||||
var sym = c.lilv_new_string(ctx.world, sym_cstr.ptr);
|
||||
const port = c.lilv_plugin_get_port_by_symbol(ctx.plugin, sym) orelse blk: {
|
||||
std.debug.warn("assert fail: symbol not found on port");
|
||||
std.debug.warn("assert fail: symbol {} not found on port\n", param.sym);
|
||||
return ImageError.InvalidSymbol;
|
||||
};
|
||||
|
||||
|
|
63
src/lang.zig
63
src/lang.zig
|
@ -1,5 +1,7 @@
|
|||
const std = @import("std");
|
||||
|
||||
const plugin = @import("plugin.zig");
|
||||
|
||||
pub const ParseError = error{
|
||||
NoCommandGiven,
|
||||
UnknownCommand,
|
||||
|
@ -9,9 +11,12 @@ pub const ParseError = error{
|
|||
pub const CommandType = enum {
|
||||
Noop,
|
||||
Load,
|
||||
Quicksave,
|
||||
|
||||
Amp,
|
||||
RFlanger,
|
||||
Quicksave,
|
||||
Eq,
|
||||
Phaser,
|
||||
};
|
||||
|
||||
pub const Command = struct {
|
||||
|
@ -38,6 +43,13 @@ pub const Command = struct {
|
|||
return try std.fmt.parseInt(usize, arg, 10);
|
||||
}
|
||||
|
||||
pub fn consumePosition(self: *const Command) !plugin.Position {
|
||||
return plugin.Position{
|
||||
.split = try self.usizeArgAt(0),
|
||||
.index = try self.usizeArgAt(1),
|
||||
};
|
||||
}
|
||||
|
||||
pub fn intArgAt(self: *const Command, idx: usize) !i32 {
|
||||
var arg = try self.argAt(idx);
|
||||
return try std.fmt.parseInt(i32, arg, 10);
|
||||
|
@ -52,33 +64,34 @@ pub const Command = struct {
|
|||
pub const CommandList = std.ArrayList(*Command);
|
||||
pub const ArgList = std.ArrayList([]const u8);
|
||||
|
||||
pub fn commandToCmdType(command: []const u8) ParseError!CommandType {
|
||||
// TODO make a hashmap for those
|
||||
if (std.mem.eql(u8, command, "noop")) {
|
||||
return CommandType.Noop;
|
||||
} else if (std.mem.eql(u8, command, "load")) {
|
||||
return CommandType.Load;
|
||||
} else if (std.mem.eql(u8, command, "amp")) {
|
||||
return CommandType.Amp;
|
||||
} else if (std.mem.eql(u8, command, "rflanger")) {
|
||||
return CommandType.RFlanger;
|
||||
} else if (std.mem.eql(u8, command, "quicksave")) {
|
||||
return CommandType.Quicksave;
|
||||
} else {
|
||||
std.debug.warn("Unknown command: '{}'\n", command);
|
||||
return ParseError.UnknownCommand;
|
||||
}
|
||||
}
|
||||
pub const PluginKeywords = [_]CommandType{ .Amp, .RFlanger };
|
||||
pub const KeywordMap = std.AutoHashMap([]const u8, CommandType);
|
||||
|
||||
pub const Lang = struct {
|
||||
allocator: *std.mem.Allocator,
|
||||
keywords: KeywordMap,
|
||||
|
||||
pub fn init(allocator: *std.mem.Allocator) Lang {
|
||||
return Lang{ .allocator = allocator };
|
||||
return Lang{
|
||||
.allocator = allocator,
|
||||
.keywords = KeywordMap.init(allocator),
|
||||
};
|
||||
}
|
||||
|
||||
fn fillKeywords(self: *Lang) !void {
|
||||
_ = try self.keywords.put("noop", .Noop);
|
||||
_ = try self.keywords.put("load", .Load);
|
||||
_ = try self.keywords.put("quicksave", .Quicksave);
|
||||
|
||||
_ = try self.keywords.put("amp", .Amp);
|
||||
_ = try self.keywords.put("rflanger", .RFlanger);
|
||||
_ = try self.keywords.put("eq", .Eq);
|
||||
_ = try self.keywords.put("phaser", .Phaser);
|
||||
}
|
||||
|
||||
pub fn parse(self: *Lang, data: []const u8) !CommandList {
|
||||
var splitted_it = std.mem.separate(data, ";");
|
||||
try self.fillKeywords();
|
||||
var cmds = CommandList.init(self.allocator);
|
||||
|
||||
while (splitted_it.next()) |stmt_orig| {
|
||||
|
@ -93,9 +106,17 @@ pub const Lang = struct {
|
|||
|
||||
var cmd_opt = tok_it.next();
|
||||
if (cmd_opt == null) return ParseError.NoCommandGiven;
|
||||
|
||||
var command = cmd_opt.?;
|
||||
var ctype = try commandToCmdType(command);
|
||||
|
||||
var kv_opt = self.keywords.get(command);
|
||||
var ctype: CommandType = undefined;
|
||||
if (kv_opt) |kv| {
|
||||
ctype = kv.value;
|
||||
} else {
|
||||
std.debug.warn("Unknown command: '{}'\n", command);
|
||||
return ParseError.UnknownCommand;
|
||||
}
|
||||
|
||||
var args = ArgList.init(self.allocator);
|
||||
|
||||
while (tok_it.next()) |arg| {
|
||||
|
|
|
@ -11,6 +11,10 @@ pub const RunError = error{
|
|||
ImageRequired,
|
||||
};
|
||||
|
||||
fn appendParam(params: *plugin.ParamList, sym: []const u8, value: f32) !void {
|
||||
try params.append(plugin.Param{ .sym = sym, .value = value });
|
||||
}
|
||||
|
||||
pub const Runner = struct {
|
||||
allocator: *std.mem.Allocator,
|
||||
image: ?*Image = null,
|
||||
|
@ -189,6 +193,44 @@ pub const Runner = struct {
|
|||
);
|
||||
}
|
||||
|
||||
fn eqCmd(
|
||||
self: *Runner,
|
||||
position: plugin.Position,
|
||||
lo: f32,
|
||||
mid: f32,
|
||||
high: f32,
|
||||
) !void {
|
||||
var image = try self.getImage();
|
||||
var params = plugin.ParamList.init(self.allocator);
|
||||
defer params.deinit();
|
||||
|
||||
try appendParam(¶ms, "lo", lo);
|
||||
try appendParam(¶ms, "mid", mid);
|
||||
try appendParam(¶ms, "hi", high);
|
||||
|
||||
try image.runPlugin("http://plugin.org.uk/swh-plugins/dj_eq_mono", position, params);
|
||||
}
|
||||
|
||||
fn phaserCmd(
|
||||
self: *Runner,
|
||||
position: plugin.Position,
|
||||
lfo_rate: f32,
|
||||
lfo_depth: f32,
|
||||
fb: f32,
|
||||
spread: f32,
|
||||
) !void {
|
||||
var image = try self.getImage();
|
||||
var params = plugin.ParamList.init(self.allocator);
|
||||
defer params.deinit();
|
||||
|
||||
try appendParam(¶ms, "lfo_rate", lfo_rate);
|
||||
try appendParam(¶ms, "lfo_depth", lfo_depth);
|
||||
try appendParam(¶ms, "fb", fb);
|
||||
try appendParam(¶ms, "spread", spread);
|
||||
|
||||
try image.runPlugin("http://plugin.org.uk/swh-plugins/lfoPhaser", position, params);
|
||||
}
|
||||
|
||||
fn runCommand(self: *Runner, cmd: *lang.Command) !void {
|
||||
return switch (cmd.command) {
|
||||
.Noop => {},
|
||||
|
@ -215,6 +257,27 @@ pub const Runner = struct {
|
|||
try self.rFlangerCmd(split, index, delay_depth_avg, law_freq);
|
||||
},
|
||||
|
||||
.Eq => blk: {
|
||||
const pos = try cmd.consumePosition();
|
||||
|
||||
const lo = try cmd.floatArgAt(2);
|
||||
const mid = try cmd.floatArgAt(3);
|
||||
const high = try cmd.floatArgAt(4);
|
||||
|
||||
try self.eqCmd(pos, lo, mid, high);
|
||||
},
|
||||
|
||||
.Phaser => blk: {
|
||||
const pos = try cmd.consumePosition();
|
||||
|
||||
const lfo_rate = try cmd.floatArgAt(2);
|
||||
const lfo_depth = try cmd.floatArgAt(3);
|
||||
const fb = try cmd.floatArgAt(4);
|
||||
const spread = try cmd.floatArgAt(5);
|
||||
|
||||
try self.phaserCmd(pos, lfo_rate, lfo_depth, fb, spread);
|
||||
},
|
||||
|
||||
else => blk: {
|
||||
std.debug.warn("Unknown command: {}\n", cmd.command);
|
||||
break :blk RunError.UnknownCommand;
|
||||
|
|
Loading…
Reference in a new issue