You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
231 lines
7.7 KiB
231 lines
7.7 KiB
/*
|
|
Calf Box, an open source musical instrument.
|
|
Copyright (C) 2010-2011 Krzysztof Foltman
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
#include "config-api.h"
|
|
#include "dspmath.h"
|
|
#include "module.h"
|
|
#include "rt.h"
|
|
#include <glib.h>
|
|
#include <malloc.h>
|
|
#include <math.h>
|
|
#include <memory.h>
|
|
#include <sndfile.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
|
|
struct fxchain_module
|
|
{
|
|
struct cbox_module module;
|
|
|
|
struct cbox_module **modules;
|
|
uint32_t module_count;
|
|
};
|
|
|
|
void fxchain_move(struct fxchain_module *m, unsigned int oldpos, unsigned int newpos)
|
|
{
|
|
if (oldpos == newpos)
|
|
return;
|
|
struct cbox_module **modules = malloc(sizeof(struct cbox_module *) * m->module_count);
|
|
for (uint32_t i = 0; i < m->module_count; i++)
|
|
{
|
|
int s;
|
|
if (i == newpos)
|
|
s = oldpos;
|
|
else
|
|
{
|
|
if (oldpos < newpos)
|
|
s = (i < oldpos || i > newpos) ? i : i + 1;
|
|
else
|
|
s = (i < newpos || i > oldpos) ? i : i - 1;
|
|
}
|
|
modules[i] = m->modules[s];
|
|
}
|
|
free(cbox_rt_swap_pointers(m->module.rt, (void **)&m->modules, modules));
|
|
}
|
|
|
|
gboolean fxchain_process_cmd(struct cbox_command_target *ct, struct cbox_command_target *fb, struct cbox_osc_command *cmd, GError **error)
|
|
{
|
|
struct fxchain_module *m = (struct fxchain_module *)ct->user_data;
|
|
const char *subcommand = NULL;
|
|
int index = 0;
|
|
|
|
//EFFECT_PARAM("/module_count", "i", stages, int, , 1, 12) else
|
|
if (!strcmp(cmd->command, "/status") && !strcmp(cmd->arg_types, ""))
|
|
{
|
|
if (!cbox_check_fb_channel(fb, cmd->command, error))
|
|
return FALSE;
|
|
for (uint32_t i = 0; i < m->module_count; i++)
|
|
{
|
|
gboolean res = FALSE;
|
|
if (m->modules[i])
|
|
res = cbox_execute_on(fb, NULL, "/module", "ss", error, m->modules[i]->engine_name, m->modules[i]->instance_name);
|
|
else
|
|
res = cbox_execute_on(fb, NULL, "/module", "ss", error, "", "");
|
|
if (!res)
|
|
return FALSE;
|
|
res = cbox_execute_on(fb, NULL, "/bypass", "ii", error, i + 1, m->modules[i] ? m->modules[i]->bypass : 0);
|
|
}
|
|
return CBOX_OBJECT_DEFAULT_STATUS(&m->module, fb, error);
|
|
}
|
|
else if (cbox_parse_path_part_int(cmd, "/module/", &subcommand, &index, 1, m->module_count, error))
|
|
{
|
|
if (!subcommand)
|
|
return FALSE;
|
|
return cbox_module_slot_process_cmd(&m->modules[index - 1], fb, cmd, subcommand, CBOX_GET_DOCUMENT(&m->module), m->module.rt, m->module.engine, error);
|
|
}
|
|
else if (!strcmp(cmd->command, "/insert") && !strcmp(cmd->arg_types, "i"))
|
|
{
|
|
int pos = CBOX_ARG_I(cmd, 0) - 1;
|
|
struct cbox_module **new_modules = malloc((m->module_count + 1) * sizeof(struct cbox_module *));
|
|
memcpy(new_modules, m->modules, pos * sizeof(struct cbox_module *));
|
|
new_modules[pos] = NULL;
|
|
memcpy(new_modules + pos + 1, m->modules + pos, (m->module_count - pos) * sizeof(struct cbox_module *));
|
|
void *old_modules = cbox_rt_swap_pointers_and_update_count(m->module.rt, (void **)&m->modules, new_modules, &m->module_count, m->module_count + 1);
|
|
free(old_modules);
|
|
return TRUE;
|
|
}
|
|
else if (!strcmp(cmd->command, "/delete") && !strcmp(cmd->arg_types, "i"))
|
|
{
|
|
int pos = CBOX_ARG_I(cmd, 0) - 1;
|
|
struct cbox_module **new_modules = malloc((m->module_count + 1) * sizeof(struct cbox_module *));
|
|
memcpy(new_modules, m->modules, pos * sizeof(struct cbox_module *));
|
|
memcpy(new_modules + pos, m->modules + pos + 1, (m->module_count - pos - 1) * sizeof(struct cbox_module *));
|
|
struct cbox_module *deleted_module = m->modules[pos];
|
|
void *old_modules = cbox_rt_swap_pointers_and_update_count(m->module.rt, (void **)&m->modules, new_modules, &m->module_count, m->module_count - 1);
|
|
free(old_modules);
|
|
if (deleted_module)
|
|
CBOX_DELETE(deleted_module);
|
|
return TRUE;
|
|
}
|
|
else if (!strcmp(cmd->command, "/move") && !strcmp(cmd->arg_types, "ii"))
|
|
{
|
|
int oldpos = CBOX_ARG_I(cmd, 0) - 1;
|
|
int newpos = CBOX_ARG_I(cmd, 1) - 1;
|
|
fxchain_move(m, oldpos, newpos);
|
|
}
|
|
else
|
|
return cbox_object_default_process_cmd(ct, fb, cmd, error);
|
|
return TRUE;
|
|
}
|
|
|
|
void fxchain_process_event(struct cbox_module *module, const uint8_t *data, uint32_t len)
|
|
{
|
|
// struct fxchain_module *m = module->user_data;
|
|
}
|
|
|
|
void fxchain_process_block(struct cbox_module *module, cbox_sample_t **inputs, cbox_sample_t **outputs)
|
|
{
|
|
struct fxchain_module *m = module->user_data;
|
|
|
|
float bufs[2][2][CBOX_BLOCK_SIZE];
|
|
|
|
for (uint32_t i = 0; i < m->module_count; i++)
|
|
{
|
|
float *input_bufs[2], *output_bufs[2];
|
|
for (int c = 0; c < 2; c++)
|
|
{
|
|
input_bufs[c] = i == 0 ? inputs[c] : bufs[i & 1][c];
|
|
output_bufs[c] = i == m->module_count - 1 ? outputs[c] : bufs[(i + 1) & 1][c];
|
|
}
|
|
if (m->modules[i] && !m->modules[i]->bypass)
|
|
m->modules[i]->process_block(m->modules[i]->user_data, input_bufs, output_bufs);
|
|
else
|
|
{
|
|
// this is not eficient at all, but empty modules aren't likely to be used except
|
|
// when setting up a chain.
|
|
for (int c = 0; c < 2; c++)
|
|
memcpy(output_bufs[c], input_bufs[c], CBOX_BLOCK_SIZE * sizeof(float));
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
static void fxchain_destroyfunc(struct cbox_module *module)
|
|
{
|
|
struct fxchain_module *m = module->user_data;
|
|
for (uint32_t i = 0; i < m->module_count; i++)
|
|
{
|
|
CBOX_DELETE(m->modules[i]);
|
|
m->modules[i] = NULL;
|
|
}
|
|
free(m->modules);
|
|
}
|
|
|
|
MODULE_CREATE_FUNCTION(fxchain)
|
|
{
|
|
static int inited = 0;
|
|
if (!inited)
|
|
{
|
|
inited = 1;
|
|
}
|
|
|
|
int i, fx_count = 0;
|
|
for (i = 0; ; i++)
|
|
{
|
|
gchar *name = g_strdup_printf("effect%d", i + 1);
|
|
const char *fx_name = cbox_config_get_string(cfg_section, name);
|
|
g_free(name);
|
|
if (!fx_name)
|
|
break;
|
|
}
|
|
fx_count = i;
|
|
if (cfg_section && !fx_count)
|
|
{
|
|
g_set_error(error, CBOX_MODULE_ERROR, CBOX_MODULE_ERROR_FAILED, "No effects defined");
|
|
return NULL;
|
|
}
|
|
|
|
struct fxchain_module *m = malloc(sizeof(struct fxchain_module));
|
|
CALL_MODULE_INIT(m, 2, 2, fxchain);
|
|
m->module.process_event = fxchain_process_event;
|
|
m->module.process_block = fxchain_process_block;
|
|
m->modules = malloc(sizeof(struct cbox_module *) * fx_count);
|
|
m->module_count = fx_count;
|
|
|
|
for (i = 0; i < fx_count; i++)
|
|
m->modules[i] = NULL;
|
|
|
|
for (i = 0; i < fx_count; i++)
|
|
{
|
|
gchar *name = g_strdup_printf("effect%d", i + 1);
|
|
const char *fx_preset_name = cbox_config_get_string(cfg_section, name);
|
|
g_free(name);
|
|
m->modules[i] = cbox_module_new_from_fx_preset(fx_preset_name, doc, rt, engine, error);
|
|
if (!m->modules[i])
|
|
goto failed;
|
|
}
|
|
fx_count = i;
|
|
|
|
return &m->module;
|
|
|
|
failed:
|
|
m->module_count = i;
|
|
CBOX_DELETE(&m->module);
|
|
return NULL;
|
|
}
|
|
|
|
|
|
struct cbox_module_keyrange_metadata fxchain_keyranges[] = {
|
|
};
|
|
|
|
struct cbox_module_livecontroller_metadata fxchain_controllers[] = {
|
|
};
|
|
|
|
DEFINE_MODULE(fxchain, 0, 2)
|
|
|
|
|