Mercurial > audlegacy-plugins
view src/crossfade/convert.c @ 3147:a6a57fe6a75c
Fix last second(s) of playback getting lost, by Hans de Goede.
author | Tony Vroon <chainsaw@gentoo.org> |
---|---|
date | Sun, 10 May 2009 21:15:29 +0100 |
parents | 2e241e90494a |
children |
line wrap: on
line source
/* * XMMS Crossfade Plugin * Copyright (C) 2000-2007 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. */ /* * Convert to standard (16bit-le stereo) */ #ifdef HAVE_CONFIG_H # include "config.h" #endif #include <string.h> #include "convert.h" void convert_init(convert_context_t * cc) { memset(cc, 0, sizeof(*cc)); } gint convert_flow(convert_context_t * cc, gpointer * buffer, gint length, format_t * format) { gpointer data; gint size; gint sample_size, sample_count; gint element_count; gint16 *out, s; if (!cc) return 0; if (length <= 0) return 0; /* calculate sample count */ sample_size = format->is_8bit ? 1 : 2; sample_count = length / sample_size; if (sample_count == 0) return 0; /* calculate buffer size */ size = sample_count * 2; if (format->nch == 1) size *= 2; /* resize buffer if necessary */ if (!cc->data || (size > cc->size)) { if (!(data = g_realloc(cc->data, size))) { DEBUG(("[crossfade] convert: g_realloc(%d) failed!\n", size)); return 0; } cc->data = data; cc->size = size; } /* calculate number of stereo samples */ element_count = sample_count; if (format->nch == 2) element_count /= 2; #define CONVERT(x) \ if (format->nch == 1) { \ while (sample_count--) { s = x; *out++ = s; *out++ = s; } \ } \ else { \ while (sample_count--) *out++ = x; \ } out = cc->data; if (format->is_8bit) { if (format->is_unsigned) { guint8 *in = *buffer; CONVERT((gint16) (*in++ ^ 128) << 8); } else { gint8 *in = *buffer; CONVERT((gint16) * in++ << 8); } } else { if (format->is_unsigned) { guint16 *in = *buffer; if (format->is_swapped) { CONVERT((gint16) (((*in & 0x00ff) << 8) | (*in >> 8)) ^ 32768); in++; } else { CONVERT((gint16) * in++ ^ 32768); } } else { gint16 *in = *buffer; if (format->is_swapped) { CONVERT(((*in & 0x00ff) << 8) | (*in >> 8)); in++; } else { if (format->nch == 1) { CONVERT(*in++); } else memcpy(out, in, size); } } } *buffer = cc->data; return size; } void convert_free(convert_context_t * cc) { if (cc->data) { g_free(cc->data); cc->data = NULL; } }