Mercurial > audlegacy-plugins
view src/wav/wav-sndfile.c @ 493:31d21ad70903 trunk
[svn] flac 113 plugin: stability fixes and a new option to disable bitrate update in player window during playback (saves cpu, this is the solution for most people playing flac and reporting high cpu usage)
author | giacomo |
---|---|
date | Sun, 21 Jan 2007 16:08:19 -0800 |
parents | 3da1b8942b8b |
children | d124034ebea3 |
line wrap: on
line source
/* Audacious - Cross-platform multimedia player * Copyright (C) 2005 Audacious development team. * * Based on the xmms_sndfile input plugin: * Copyright (C) 2000, 2002 Erik de Castro Lopo * * 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include <glib.h> #include <glib/gi18n.h> #include <string.h> #include <math.h> #include <stdio.h> #include <audacious/util.h> #include <audacious/titlestring.h> #include "audacious/output.h" #include "wav-sndfile.h" #include <sndfile.h> static SNDFILE *sndfile = NULL; static SF_INFO sfinfo; static int song_length; static int bit_rate = 0; static int decoding; static int seek_time = -1; static GThread *decode_thread; GStaticMutex decode_mutex = G_STATIC_MUTEX_INIT; InputPlugin wav_ip = { NULL, NULL, NULL, plugin_init, wav_about, NULL, is_our_file, NULL, play_start, play_stop, NULL, /* Could call do_pause here, but it will cause auto-stop anyway */ file_seek, NULL, get_time, NULL, NULL, NULL, NULL, NULL, NULL, NULL, get_song_info, NULL, NULL }; int get_song_length (char *filename) { SNDFILE *tmp_sndfile; SF_INFO tmp_sfinfo; if (! (tmp_sndfile = sf_open (filename, SFM_READ, &tmp_sfinfo))) return 0; sf_close (tmp_sndfile); tmp_sndfile = NULL; if (tmp_sfinfo.samplerate <= 0) return 0; return (int) ceil (1000.0 * tmp_sfinfo.frames / tmp_sfinfo.samplerate); } /* get_song_length */ static gchar *get_title(char *filename) { gchar *title; title = g_path_get_basename(filename); return title; } static void plugin_init (void) { decoding = FALSE; seek_time = -1; } /* plugin_int */ static int is_our_file (char *filename) { SNDFILE *tmp_sndfile; SF_INFO tmp_sfinfo; /* Have to open the file to see if libsndfile can handle it. */ if (! (tmp_sndfile = sf_open (filename, SFM_READ, &tmp_sfinfo))) return FALSE; /* It can so close file and return TRUE. */ sf_close (tmp_sndfile); tmp_sndfile = NULL; return TRUE; } /* is_our_file */ static void* play_loop (void *arg) { static short buffer [BUFFER_SIZE]; int samples; g_static_mutex_lock(&decode_mutex); decoding = TRUE; while (decoding) { /* sf_read_short will return 0 for all reads at EOF. */ samples = sf_read_short (sndfile, buffer, BUFFER_SIZE); if (samples > 0 && decoding) { while ((wav_ip.output->buffer_free () < (samples * sizeof (short))) && decoding) xmms_usleep (10000); produce_audio (wav_ip.output->written_time (), FMT_S16_NE, sfinfo.channels, samples * sizeof (short), buffer, &decoding); } else xmms_usleep (10000); /* Do seek if seek_time is valid. */ if (seek_time > 0) { sf_seek (sndfile, seek_time * sfinfo.samplerate, SEEK_SET); wav_ip.output->flush (seek_time * 1000); seek_time = -1; }; }; /* while (decoding) */ g_static_mutex_unlock(&decode_mutex); g_thread_exit (NULL); return NULL; } /* play_loop */ static void play_start (char *filename) { int pcmbitwidth; gchar *song_title; if (sndfile) return; pcmbitwidth = 32; song_title = get_title(filename); if (! (sndfile = sf_open (filename, SFM_READ, &sfinfo))) return; bit_rate = sfinfo.samplerate * pcmbitwidth * sfinfo.channels; if (sfinfo.samplerate > 0) song_length = (int) ceil (1000.0 * sfinfo.frames / sfinfo.samplerate); else song_length = 0; if (! wav_ip.output->open_audio (FMT_S16_NE, sfinfo.samplerate, sfinfo.channels)) { sf_close (sndfile); sndfile = NULL; return; }; wav_ip.set_info (song_title, song_length, bit_rate, sfinfo.samplerate, sfinfo.channels); g_free (song_title); decode_thread = g_thread_create ((GThreadFunc)play_loop, NULL, TRUE, NULL); xmms_usleep (40000); } /* play_start */ static void play_stop (void) { if (decode_thread == NULL) return; decoding = FALSE; g_thread_join (decode_thread); wav_ip.output->close_audio (); sf_close (sndfile); sndfile = NULL; decode_thread = NULL; seek_time = -1; } /* play_stop */ static void file_seek (int time) { if (! sfinfo.seekable) return; seek_time = time; while (seek_time != -1) xmms_usleep (80000); } /* file_seek */ static int get_time (void) { if ( ! (wav_ip.output->buffer_playing () && decoding)) return -1; return wav_ip.output->output_time (); } /* get_time */ static void get_song_info (char *filename, char **title, int *length) { (*length) = get_song_length(filename); (*title) = get_title(filename); } /* get_song_info */ static void wav_about(void) { static GtkWidget *box; if (!box) { box = xmms_show_message( _("About sndfile WAV support"), _("Adapted for Audacious usage by Tony Vroon <chainsaw@gentoo.org>\n" "from the xmms_sndfile plugin which is:\n" "Copyright (C) 2000, 2002 Erik de Castro Lopo\n\n" "This program is free software ; you can redistribute it and/or modify \n" "it under the terms of the GNU General Public License as published by \n" "the Free Software Foundation ; either version 2 of the License, or \n" "(at your option) any later version. \n \n" "This program is distributed in the hope that it will be useful, \n" "but WITHOUT ANY WARRANTY ; without even the implied warranty of \n" "MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. \n" "See the GNU General Public License for more details. \n\n" "You should have received a copy of the GNU General Public \n" "License along with this program ; if not, write to \n" "the Free Software Foundation, Inc., \n" "51 Franklin Street, Fifth Floor, \n" "Boston, MA 02110-1301 USA"), _("Ok"), FALSE, NULL, NULL); g_signal_connect(G_OBJECT(box), "destroy", (GCallback)gtk_widget_destroyed, &box); } } InputPlugin *get_iplugin_info(void) { wav_ip.description = g_strdup_printf(_("sndfile WAV plugin")); return &wav_ip; }