Mercurial > audlegacy
view Plugins/Output/crossfade/volume.c @ 540:326de307e2eb trunk
[svn] add playback.c to build
author | nenolod |
---|---|
date | Sat, 28 Jan 2006 09:27:32 -0800 |
parents | 88b38e2414a1 |
children | 5b81b0f310e5 |
line wrap: on
line source
/* * XMMS Crossfade Plugin * Copyright (C) 2000-2004 Peter Eisenlohr <peter@eisenlohr.org> * * based on the original OSS Output Plugin * Copyright (C) 1998-2000 Peter Alm, Mikael Alm, Olle Hallnas, Thomas Nilsson and 4Front Technologies * * 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 2 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, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, * USA. */ /* * Volume to standard (16bit-le stereo) */ #ifdef HAVE_CONFIG_H # include "config.h" #endif #include "volume.h" #include <string.h> #include <math.h> /* for rintf */ #undef VERBOSE static void set_factor(volume_context_t *vc, gfloat factor) { vc->factor = factor; #if 0 if(vc->factor < 0.01f) vc->factor = 0.01f; if(vc->factor > 4.0f) vc->factor = 4.0f; #endif #ifdef VERBOSE DEBUG(("[crossfade] volume_set_factor: new factor=%.3f\n", vc->factor)); #endif } void volume_init(volume_context_t *vc) { memset(vc, 0, sizeof(*vc)); vc->active = FALSE; } void volume_set_active(volume_context_t *vc, gboolean active) { vc->active = active; #ifdef VERBOSE DEBUG(("[crossfade] volume_set_active: active=%d\n", vc->active)); #endif } void volume_set_target_rms(volume_context_t *vc, gint target_rms) { vc->target_rms = target_rms; if(vc->active && (vc->song_rms == 0)) { DEBUG(("[crossfade] volume_set_target_rms: WARNING: song_rms=0!\n")); vc->factor = 1; return; } set_factor(vc, (gfloat)vc->target_rms / (gfloat)vc->song_rms); } void volume_set_song_rms(volume_context_t *vc, gint song_rms) { vc->song_rms = song_rms; set_factor(vc, (gfloat)vc->target_rms / (gfloat)vc->song_rms); } gfloat volume_compute_factor(gint percent, gint dB_range) { if(percent>=100) return 1; if(percent<=0) return 0; gfloat dB = (percent-100)/100.0*dB_range; return pow(10, dB/20); } void volume_flow(volume_context_t *vc, gpointer *buffer, gint length) { gint16 *in = *buffer; struct timeval tv; glong dt; if(!vc->active) return; length /= 2; while(length--) { gint out = (gint) rintf((gfloat)*in * vc->factor); if(out > 32767) { *in++ = 32767; vc->clips++; } else if(out < -32768) { *in++ = -32768; vc->clips++; } else *in++ = out; } gettimeofday(&tv, NULL); dt = (tv.tv_sec - vc->tv_last.tv_sec) * 1000 + (tv.tv_usec - vc->tv_last.tv_usec) / 1000; if(((dt < 0) || (dt > 1000)) && (vc->clips > 0)) { DEBUG(("[crossfade] volume_flow: %d samples clipped!\n", vc->clips)); vc->clips = 0; vc->tv_last = tv; } } void volume_free(volume_context_t *vc) { }