Mercurial > audlegacy-plugins
view src/echo_plugin/echo.c @ 3191:a65f440cbed3
alsa-ng: Fix possible race conditions, sluggish pause and seek.
author | John Lindgren <john.lindgren@tds.net> |
---|---|
date | Mon, 22 Jun 2009 16:05:57 -0400 |
parents | 3134a0987162 |
children |
line wrap: on
line source
#include "config.h" #include <audlegacy/plugin.h> #include <stdlib.h> #include <stdio.h> #include <string.h> #include <gtk/gtk.h> #include <audlegacy/i18n.h> #include "echo.h" static void init(void); static void cleanup(void); static int mod_samples(gpointer * d, gint length, AFormat afmt, gint srate, gint nch); static void query_format(AFormat * fmt, gint * rate, gint * nch); #define MAX_SRATE 50000 #define MAX_CHANNELS 2 #define BYTES_PS 2 #define BUFFER_SAMPLES (MAX_SRATE * MAX_DELAY / 1000) #define BUFFER_SHORTS (BUFFER_SAMPLES * MAX_CHANNELS) #define BUFFER_BYTES (BUFFER_SHORTS * BYTES_PS) EffectPlugin echo_ep = { .description = "Echo Plugin", /* Description */ .init = init, .cleanup = cleanup, .about = echo_about, .configure = echo_configure, .mod_samples = mod_samples, .query_format = query_format, }; static gint16 *buffer = NULL; gint echo_delay = 500, echo_feedback = 50, echo_volume = 50; gboolean echo_surround_enable = FALSE; static int w_ofs; EffectPlugin *echo_eplist[] = { &echo_ep, NULL }; DECLARE_PLUGIN(echo, NULL, NULL, NULL, NULL, echo_eplist, NULL, NULL, NULL); static void init(void) { mcs_handle_t *cfg; if (sizeof(short) != sizeof(gint16)) abort(); cfg = aud_cfg_db_open(); aud_cfg_db_get_int(cfg, "echo_plugin", "delay", &echo_delay); aud_cfg_db_get_int(cfg, "echo_plugin", "feedback", &echo_feedback); aud_cfg_db_get_int(cfg, "echo_plugin", "volume", &echo_volume); aud_cfg_db_get_bool(cfg, "echo_plugin", "enable_surround", &echo_surround_enable); aud_cfg_db_close(cfg); } static void cleanup(void) { g_free(buffer); buffer = NULL; } static void query_format(AFormat * fmt, gint * rate, gint * nch) { if (!(*fmt == FMT_S16_NE || (*fmt == FMT_S16_LE && G_BYTE_ORDER == G_LITTLE_ENDIAN) || (*fmt == FMT_S16_BE && G_BYTE_ORDER == G_BIG_ENDIAN))) *fmt = FMT_S16_NE; } static int mod_samples(gpointer * d, gint length, AFormat afmt, gint srate, gint nch) { gint i, in, out, buf, r_ofs, fb_div; gint16 *data = (gint16 *) * d; static gint old_srate, old_nch; if (!(afmt == FMT_S16_NE || (afmt == FMT_S16_LE && G_BYTE_ORDER == G_LITTLE_ENDIAN) || (afmt == FMT_S16_BE && G_BYTE_ORDER == G_BIG_ENDIAN))) return length; if (!buffer) buffer = g_malloc0(BUFFER_BYTES + 2); if (nch != old_nch || srate != old_srate) { memset(buffer, 0, BUFFER_BYTES); w_ofs = 0; old_nch = nch; old_srate = srate; } if (echo_surround_enable && nch == 2) fb_div = 200; else fb_div = 100; r_ofs = w_ofs - (srate * echo_delay / 1000) * nch; if (r_ofs < 0) r_ofs += BUFFER_SHORTS; for (i = 0; i < length / BYTES_PS; i++) { in = data[i]; buf = buffer[r_ofs]; if (echo_surround_enable && nch == 2) { if (i & 1) buf -= buffer[r_ofs - 1]; else buf -= buffer[r_ofs + 1]; } out = in + buf * echo_volume / 100; buf = in + buf * echo_feedback / fb_div; out = CLAMP(out, -32768, 32767); buf = CLAMP(buf, -32768, 32767); buffer[w_ofs] = buf; data[i] = out; if (++r_ofs >= BUFFER_SHORTS) r_ofs -= BUFFER_SHORTS; if (++w_ofs >= BUFFER_SHORTS) w_ofs -= BUFFER_SHORTS; } return length; }