/* Copyright (c) 2012 The Chromium OS Authors. All rights reserved.
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include <stdlib.h>
#include "cras_dsp_module.h"
#include "drc.h"
#include "dsp_util.h"
#include "dcblock.h"
#include "eq.h"
#include "eq2.h"
/*
* empty module functions (for source and sink)
*/
static int empty_instantiate(struct dsp_module *module,
unsigned long sample_rate)
{
return 0;
}
static void empty_connect_port(struct dsp_module *module, unsigned long port,
float *data_location) {}
static int empty_get_delay(struct dsp_module *module)
{
return 0;
}
static void empty_run(struct dsp_module *module, unsigned long sample_count) {}
static void empty_deinstantiate(struct dsp_module *module) {}
static void empty_free_module(struct dsp_module *module)
{
free(module);
}
static int empty_get_properties(struct dsp_module *module) { return 0; }
static void empty_dump(struct dsp_module *module, struct dumper *d)
{
dumpf(d, "built-in module\n");
}
static void empty_init_module(struct dsp_module *module)
{
module->instantiate = &empty_instantiate;
module->connect_port = &empty_connect_port;
module->get_delay = &empty_get_delay;
module->run = &empty_run;
module->deinstantiate = &empty_deinstantiate;
module->free_module = &empty_free_module;
module->get_properties = &empty_get_properties;
module->dump = &empty_dump;
}
/*
* swap_lr module functions
*/
static int swap_lr_instantiate(struct dsp_module *module,
unsigned long sample_rate)
{
module->data = calloc(4, sizeof(float*));
return 0;
}
static void swap_lr_connect_port(struct dsp_module *module,
unsigned long port, float *data_location)
{
float **ports;
ports = (float **)module->data;
ports[port] = data_location;
}
static void swap_lr_run(struct dsp_module *module,
unsigned long sample_count)
{
size_t i;
float **ports = (float **)module->data;
/* This module runs dsp in-place, so ports[0] == ports[2],
* ports[1] == ports[3]. Here we swap data on two channels.
*/
for (i = 0; i < sample_count; i++) {
float temp = ports[0][i];
ports[2][i] = ports[1][i];
ports[3][i] = temp;
}
}
static void swap_lr_deinstantiate(struct dsp_module *module)
{
free(module->data);
}
static void swap_lr_init_module(struct dsp_module *module)
{
module->instantiate = &swap_lr_instantiate;
module->connect_port = &swap_lr_connect_port;
module->get_delay = &empty_get_delay;
module->run = &swap_lr_run;
module->deinstantiate = &swap_lr_deinstantiate;
module->free_module = &empty_free_module;
module->get_properties = &empty_get_properties;
}
/*
* invert_lr module functions
*/
static int invert_lr_instantiate(struct dsp_module *module,
unsigned long sample_rate)
{
module->data = calloc(4, sizeof(float*));
return 0;
}
static void invert_lr_connect_port(struct dsp_module *module,
unsigned long port, float *data_location)
{
float **ports;
ports = (float **)module->data;
ports[port] = data_location;
}
static void invert_lr_run(struct dsp_module *module,
unsigned long sample_count)
{
size_t i;
float **ports = (float **)module->data;
for (i = 0; i < sample_count; i++) {
ports[2][i] = -ports[0][i];
ports[3][i] = ports[1][i];
}
}
static void invert_lr_deinstantiate(struct dsp_module *module)
{
free(module->data);
}
static void invert_lr_init_module(struct dsp_module *module)
{
module->instantiate = &invert_lr_instantiate;
module->connect_port = &invert_lr_connect_port;
module->get_delay = &empty_get_delay;
module->run = &invert_lr_run;
module->deinstantiate = &invert_lr_deinstantiate;
module->free_module = &empty_free_module;
module->get_properties = &empty_get_properties;
}
/*
* mix_stereo module functions
*/
static int mix_stereo_instantiate(struct dsp_module *module,
unsigned long sample_rate)
{
module->data = calloc(4, sizeof(float*));
return 0;
}
static void mix_stereo_connect_port(struct dsp_module *module,
unsigned long port, float *data_location)
{
float **ports;
ports = (float **)module->data;
ports[port] = data_location;
}
static void mix_stereo_run(struct dsp_module *module,
unsigned long sample_count)
{
size_t i;
float tmp;
float **ports = (float **)module->data;
for (i = 0; i < sample_count; i++) {
tmp = ports[0][i] + ports[1][i];
ports[2][i] = tmp;
ports[3][i] = tmp;
}
}
static void mix_stereo_deinstantiate(struct dsp_module *module)
{
free(module->data);
}
static void mix_stereo_init_module(struct dsp_module *module)
{
module->instantiate = &mix_stereo_instantiate;
module->connect_port = &mix_stereo_connect_port;
module->get_delay = &empty_get_delay;
module->run = &mix_stereo_run;
module->deinstantiate = &mix_stereo_deinstantiate;
module->free_module = &empty_free_module;
module->get_properties = &empty_get_properties;
module->dump = &empty_dump;
}
/*
* dcblock module functions
*/
struct dcblock_data {
struct dcblock *dcblockl;
struct dcblock *dcblockr;
/* One port for input, one for output, and 1 parameter */
float *ports[5];
};
static int dcblock_instantiate(struct dsp_module *module,
unsigned long sample_rate)
{
module->data = calloc(1, sizeof(struct dcblock_data));
return 0;
}
static void dcblock_connect_port(struct dsp_module *module,
unsigned long port, float *data_location)
{
struct dcblock_data *data = (struct dcblock_data *) module->data;
data->ports[port] = data_location;
}
static void dcblock_run(struct dsp_module *module, unsigned long sample_count)
{
struct dcblock_data *data = (struct dcblock_data *) module->data;
if (!data->dcblockl)
data->dcblockl = dcblock_new(*data->ports[4]);
if (!data->dcblockr)
data->dcblockr = dcblock_new(*data->ports[4]);
if (data->ports[0] != data->ports[2])
memcpy(data->ports[2], data->ports[0],
sizeof(float) * sample_count);
if (data->ports[1] != data->ports[3])
memcpy(data->ports[3], data->ports[1],
sizeof(float) * sample_count);
dcblock_process(data->dcblockl, data->ports[2], (int) sample_count);
dcblock_process(data->dcblockr, data->ports[3], (int) sample_count);
}
static void dcblock_deinstantiate(struct dsp_module *module)
{
struct dcblock_data *data = (struct dcblock_data *) module->data;
if (data->dcblockl)
dcblock_free(data->dcblockl);
if (data->dcblockr)
dcblock_free(data->dcblockr);
free(data);
}
static void dcblock_init_module(struct dsp_module *module)
{
module->instantiate = &dcblock_instantiate;
module->connect_port = &dcblock_connect_port;
module->get_delay = &empty_get_delay;
module->run = &dcblock_run;
module->deinstantiate = &dcblock_deinstantiate;
module->free_module = &empty_free_module;
module->get_properties = &empty_get_properties;
module->dump = &empty_dump;
}
/*
* eq module functions
*/
struct eq_data {
int sample_rate;
struct eq *eq; /* Initialized in the first call of eq_run() */
/* One port for input, one for output, and 4 parameters per eq */
float *ports[2 + MAX_BIQUADS_PER_EQ * 4];
};
static int eq_instantiate(struct dsp_module *module, unsigned long sample_rate)
{
struct eq_data *data;
module->data = calloc(1, sizeof(struct eq_data));
data = (struct eq_data *) module->data;
data->sample_rate = (int) sample_rate;
return 0;
}
static void eq_connect_port(struct dsp_module *module,
unsigned long port, float *data_location)
{
struct eq_data *data = (struct eq_data *) module->data;
data->ports[port] = data_location;
}
static void eq_run(struct dsp_module *module, unsigned long sample_count)
{
struct eq_data *data = (struct eq_data *) module->data;
if (!data->eq) {
float nyquist = data->sample_rate / 2;
int i;
data->eq = eq_new();
for (i = 2; i < 2 + MAX_BIQUADS_PER_EQ * 4; i += 4) {
if (!data->ports[i])
break;
int type = (int) *data->ports[i];
float freq = *data->ports[i+1];
float Q = *data->ports[i+2];
float gain = *data->ports[i+3];
eq_append_biquad(data->eq, type, freq / nyquist, Q,
gain);
}
}
if (data->ports[0] != data->ports[1])
memcpy(data->ports[1], data->ports[0],
sizeof(float) * sample_count);
eq_process(data->eq, data->ports[1], (int) sample_count);
}
static void eq_deinstantiate(struct dsp_module *module)
{
struct eq_data *data = (struct eq_data *) module->data;
if (data->eq)
eq_free(data->eq);
free(data);
}
static void eq_init_module(struct dsp_module *module)
{
module->instantiate = &eq_instantiate;
module->connect_port = &eq_connect_port;
module->get_delay = &empty_get_delay;
module->run = &eq_run;
module->deinstantiate = &eq_deinstantiate;
module->free_module = &empty_free_module;
module->get_properties = &empty_get_properties;
module->dump = &empty_dump;
}
/*
* eq2 module functions
*/
struct eq2_data {
int sample_rate;
struct eq2 *eq2; /* Initialized in the first call of eq2_run() */
/* Two ports for input, two for output, and 8 parameters per eq pair */
float *ports[4 + MAX_BIQUADS_PER_EQ2 * 8];
};
static int eq2_instantiate(struct dsp_module *module, unsigned long sample_rate)
{
struct eq2_data *data;
module->data = calloc(1, sizeof(struct eq2_data));
data = (struct eq2_data *) module->data;
data->sample_rate = (int) sample_rate;
return 0;
}
static void eq2_connect_port(struct dsp_module *module,
unsigned long port, float *data_location)
{
struct eq2_data *data = (struct eq2_data *) module->data;
data->ports[port] = data_location;
}
static void eq2_run(struct dsp_module *module, unsigned long sample_count)
{
struct eq2_data *data = (struct eq2_data *) module->data;
if (!data->eq2) {
float nyquist = data->sample_rate / 2;
int i, channel;
data->eq2 = eq2_new();
for (i = 4; i < 4 + MAX_BIQUADS_PER_EQ2 * 8; i += 8) {
if (!data->ports[i])
break;
for (channel = 0; channel < 2; channel++) {
int k = i + channel * 4;
int type = (int) *data->ports[k];
float freq = *data->ports[k+1];
float Q = *data->ports[k+2];
float gain = *data->ports[k+3];
eq2_append_biquad(data->eq2, channel, type,
freq / nyquist, Q, gain);
}
}
}
if (data->ports[0] != data->ports[2])
memcpy(data->ports[2], data->ports[0],
sizeof(float) * sample_count);
if (data->ports[3] != data->ports[1])
memcpy(data->ports[3], data->ports[1],
sizeof(float) * sample_count);
eq2_process(data->eq2, data->ports[2], data->ports[3],
(int) sample_count);
}
static void eq2_deinstantiate(struct dsp_module *module)
{
struct eq2_data *data = (struct eq2_data *) module->data;
if (data->eq2)
eq2_free(data->eq2);
free(data);
}
static void eq2_init_module(struct dsp_module *module)
{
module->instantiate = &eq2_instantiate;
module->connect_port = &eq2_connect_port;
module->get_delay = &empty_get_delay;
module->run = &eq2_run;
module->deinstantiate = &eq2_deinstantiate;
module->free_module = &empty_free_module;
module->get_properties = &empty_get_properties;
module->dump = &empty_dump;
}
/*
* drc module functions
*/
struct drc_data {
int sample_rate;
struct drc *drc; /* Initialized in the first call of drc_run() */
/* Two ports for input, two for output, one for disable_emphasis,
* and 8 parameters each band */
float *ports[4 + 1 + 8 * 3];
};
static int drc_instantiate(struct dsp_module *module, unsigned long sample_rate)
{
struct drc_data *data;
module->data = calloc(1, sizeof(struct drc_data));
data = (struct drc_data *) module->data;
data->sample_rate = (int) sample_rate;
return 0;
}
static void drc_connect_port(struct dsp_module *module,
unsigned long port, float *data_location)
{
struct drc_data *data = (struct drc_data *) module->data;
data->ports[port] = data_location;
}
static int drc_get_delay(struct dsp_module *module)
{
struct drc_data *data = (struct drc_data *) module->data;
return DRC_DEFAULT_PRE_DELAY * data->sample_rate;
}
static void drc_run(struct dsp_module *module, unsigned long sample_count)
{
struct drc_data *data = (struct drc_data *) module->data;
if (!data->drc) {
int i;
float nyquist = data->sample_rate / 2;
struct drc *drc = drc_new(data->sample_rate);
data->drc = drc;
drc->emphasis_disabled = (int) *data->ports[4];
for (i = 0; i < 3; i++) {
int k = 5 + i * 8;
float f = *data->ports[k];
float enable = *data->ports[k+1];
float threshold = *data->ports[k+2];
float knee = *data->ports[k+3];
float ratio = *data->ports[k+4];
float attack = *data->ports[k+5];
float release = *data->ports[k+6];
float boost = *data->ports[k+7];
drc_set_param(drc, i, PARAM_CROSSOVER_LOWER_FREQ,
f / nyquist);
drc_set_param(drc, i, PARAM_ENABLED, enable);
drc_set_param(drc, i, PARAM_THRESHOLD, threshold);
drc_set_param(drc, i, PARAM_KNEE, knee);
drc_set_param(drc, i, PARAM_RATIO, ratio);
drc_set_param(drc, i, PARAM_ATTACK, attack);
drc_set_param(drc, i, PARAM_RELEASE, release);
drc_set_param(drc, i, PARAM_POST_GAIN, boost);
}
drc_init(drc);
}
if (data->ports[0] != data->ports[2])
memcpy(data->ports[2], data->ports[0],
sizeof(float) * sample_count);
if (data->ports[1] != data->ports[3])
memcpy(data->ports[3], data->ports[1],
sizeof(float) * sample_count);
drc_process(data->drc, &data->ports[2], (int) sample_count);
}
static void drc_deinstantiate(struct dsp_module *module)
{
struct drc_data *data = (struct drc_data *) module->data;
if (data->drc)
drc_free(data->drc);
free(data);
}
static void drc_init_module(struct dsp_module *module)
{
module->instantiate = &drc_instantiate;
module->connect_port = &drc_connect_port;
module->get_delay = &drc_get_delay;
module->run = &drc_run;
module->deinstantiate = &drc_deinstantiate;
module->free_module = &empty_free_module;
module->get_properties = &empty_get_properties;
module->dump = &empty_dump;
}
/*
* sink module functions
*/
struct sink_data {
struct ext_dsp_module *ext_module;
float *ports[MAX_EXT_DSP_PORTS];
};
static int sink_instantiate(struct dsp_module *module,
unsigned long sample_rate)
{
module->data = (struct sink_data *)calloc(1, sizeof(struct sink_data));
return 0;
}
static void sink_deinstantiate(struct dsp_module *module)
{
struct sink_data *data = (struct sink_data *)module->data;
free(data);
}
static void sink_connect_port(struct dsp_module *module, unsigned long port,
float *data_location)
{
struct sink_data *data = (struct sink_data *)module->data;
data->ports[port] = data_location;
}
static void sink_run(struct dsp_module *module, unsigned long sample_count)
{
struct sink_data *data = (struct sink_data *)module->data;
if (!data->ext_module)
return;
data->ext_module->run(data->ext_module, sample_count);
}
static void sink_init_module(struct dsp_module *module)
{
module->instantiate = &sink_instantiate;
module->connect_port = &sink_connect_port;
module->get_delay = &empty_get_delay;
module->run = &sink_run;
module->deinstantiate = &sink_deinstantiate;
module->free_module = &empty_free_module;
module->get_properties = &empty_get_properties;
module->dump = &empty_dump;
}
void cras_dsp_module_set_sink_ext_module(struct dsp_module *module,
struct ext_dsp_module *ext_module)
{
struct sink_data *data = (struct sink_data *)module->data;
int i;
data->ext_module = ext_module;
for (i = 0; i < MAX_EXT_DSP_PORTS; i++)
ext_module->ports[i] = data->ports[i];
}
/*
* builtin module dispatcher
*/
struct dsp_module *cras_dsp_module_load_builtin(struct plugin *plugin)
{
struct dsp_module *module;
if (strcmp(plugin->library, "builtin") != 0)
return NULL;
module = calloc(1, sizeof(struct dsp_module));
if (strcmp(plugin->label, "mix_stereo") == 0) {
mix_stereo_init_module(module);
} else if (strcmp(plugin->label, "invert_lr") == 0) {
invert_lr_init_module(module);
} else if (strcmp(plugin->label, "dcblock") == 0) {
dcblock_init_module(module);
} else if (strcmp(plugin->label, "eq") == 0) {
eq_init_module(module);
} else if (strcmp(plugin->label, "eq2") == 0) {
eq2_init_module(module);
} else if (strcmp(plugin->label, "drc") == 0) {
drc_init_module(module);
} else if (strcmp(plugin->label, "swap_lr") == 0) {
swap_lr_init_module(module);
} else if (strcmp(plugin->label, "sink") == 0) {
sink_init_module(module);
} else {
empty_init_module(module);
}
return module;
}