/*
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 .
*/
#include "config.h"
#include "config-api.h"
#include "dspmath.h"
#include "module.h"
#include "rt.h"
#include
#include
#include
#include
#include
#include
#include
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)