Mercurial > audlegacy-plugins
view src/ffmpeg/ffmpeg.c @ 867:d9c7b1d413c9 trunk
[svn] - last commit wasn't obviously related wit changes here; revert stuff that is still being tested
author | giacomo |
---|---|
date | Fri, 16 Mar 2007 18:58:10 -0700 |
parents | 80e7162bd968 |
children |
line wrap: on
line source
/* * Audacious FFmpeg Plugin * (C) 2007 William Pitcock <nenolod@sacredspiral.co.uk> * * Based on: * Audacious WMA input plugin * (C) 2005-2007 Tony Vroon * * xmms-wma - WMA player for BMP * Copyright (C) 2004,2005 McMCC <mcmcc@mail.ru> * bmp-wma - WMA player for BMP * Copyright (C) 2004 Roman Bogorodskiy <bogorodskiy@inbox.ru> * * 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. */ #define _XOPEN_SOURCE 600 #include <stdlib.h> #include <unistd.h> #include <math.h> #include <stdbool.h> #include <stdio.h> #include <string.h> #include <strings.h> #include <glib.h> #include <audacious/plugin.h> #include <audacious/output.h> #include <audacious/util.h> #include <audacious/titlestring.h> #include <audacious/vfs.h> #include <audacious/strings.h> #include <audacious/i18n.h> #include "config.h" #include "avcodec.h" #include "avformat.h" #include "avutil.h" #define ST_BUFF 1024 static int ffmpeg_decode = 0; static gboolean ffmpeg_pause = 0; static int ffmpeg_seekpos = -1; static int ffmpeg_st_buff, ffmpeg_idx, ffmpeg_idx2; static GThread *ffmpeg_decode_thread; GStaticMutex ffmpeg_mutex = G_STATIC_MUTEX_INIT; static AVCodecContext *c = NULL; static AVFormatContext *ic = NULL; static AVCodecContext *c2 = NULL; static AVFormatContext *ic2 = NULL; static uint8_t *ffmpeg_outbuf, *ffmpeg_s_outbuf; char description[64]; static void ffmpeg_init(void); static int ffmpeg_is_our_file(char *filename); static int ffmpeg_is_our_fd(char *filename, VFSFile *fd); static void ffmpeg_play_file(InputPlayback *data); static void ffmpeg_stop(InputPlayback *data); static void ffmpeg_seek(InputPlayback *data, int time); static void ffmpeg_do_pause(InputPlayback *data, short p); static int ffmpeg_get_time(InputPlayback *data); static void ffmpeg_get_song_info(char *filename, char **title, int *length); static TitleInput *ffmpeg_get_song_tuple(char *filename); static char *wsong_title; static int wsong_time; static GtkWidget *dialog1, *button1, *label1; InputPlugin *get_iplugin_info(void); gchar *ffmpeg_fmts[] = { "wma", "shn", NULL }; InputPlugin ffmpeg_ip = { NULL, // Filled in by xmms NULL, // Filled in by xmms description, // The description that is shown in the preferences box ffmpeg_init, // Called when the plugin is loaded NULL, // Show the about box NULL, // Show the configure box ffmpeg_is_our_file, // Return 1 if the plugin can handle the file NULL, // Scan dir ffmpeg_play_file, // Play file ffmpeg_stop, // Stop ffmpeg_do_pause, // Pause ffmpeg_seek, // Seek NULL, // Set the equalizer, most plugins won't be able to do this ffmpeg_get_time, // Get the time, usually returns the output plugins output time NULL, // Get volume NULL, // Set volume NULL, // OBSOLETE! NULL, // OBSOLETE! NULL, // Send data to the visualization plugins NULL, // Fill in the stuff that is shown in the player window NULL, // Show some text in the song title box. Filled in by xmms ffmpeg_get_song_info, // Function to grab the title string NULL, // Bring up an info window for the filename passed in NULL, // Handle to the current output plugin. Filled in by xmms ffmpeg_get_song_tuple, // Tuple builder NULL, NULL, ffmpeg_is_our_fd, // vfs ffmpeg_fmts }; InputPlugin *get_iplugin_info(void) { memset(description, 0, 64); ffmpeg_ip.description = g_strdup(_("FFmpeg Audio Plugin")); return &ffmpeg_ip; } static gchar *str_twenty_to_space(gchar * str) { gchar *match, *match_end; g_return_val_if_fail(str != NULL, NULL); while ((match = strstr(str, "%20"))) { match_end = match + 3; *match++ = ' '; while (*match_end) *match++ = *match_end++; *match = 0; } return str; } static void ffmpeg_init(void) { avcodec_init(); avcodec_register_all(); av_register_all(); } static int ffmpeg_is_our_file(char *filename) { AVCodec *codec2; if(av_open_input_file(&ic2, str_twenty_to_space(filename), NULL, 0, NULL) < 0) return 0; for(ffmpeg_idx2 = 0; ffmpeg_idx2 < ic2->nb_streams; ffmpeg_idx2++) { c2 = ic2->streams[ffmpeg_idx2]->codec; if(c2->codec_type == CODEC_TYPE_AUDIO) break; } av_find_stream_info(ic2); codec2 = avcodec_find_decoder(c2->codec_id); if(!codec2) { av_close_input_file(ic2); return 0; } av_close_input_file(ic2); return 1; } static int ffmpeg_is_our_fd(char *filename, VFSFile *fd) { AVCodec *codec2; if(av_open_input_vfsfile(&ic2, filename, fd, NULL, 0, NULL) < 0) return 0; for(ffmpeg_idx2 = 0; ffmpeg_idx2 < ic2->nb_streams; ffmpeg_idx2++) { c2 = ic2->streams[ffmpeg_idx2]->codec; if(c2->codec_type == CODEC_TYPE_AUDIO) break; } av_find_stream_info(ic2); codec2 = avcodec_find_decoder(c2->codec_id); if(!codec2) return 0; return 1; } static void ffmpeg_do_pause(InputPlayback *playback, short p) { ffmpeg_pause = p; playback->output->pause(ffmpeg_pause); } static void ffmpeg_seek(InputPlayback *playback, int time) { ffmpeg_seekpos = time; if(ffmpeg_pause) playback->output->pause(0); while(ffmpeg_decode && ffmpeg_seekpos!=-1) xmms_usleep(10000); if(ffmpeg_pause) playback->output->pause(1); } static int ffmpeg_get_time(InputPlayback *playback) { playback->output->buffer_free(); if(ffmpeg_decode) return playback->output->output_time(); return -1; } static gchar *extname(const char *filename) { gchar *ext = strrchr(filename, '.'); if(ext != NULL) ++ext; return ext; } static char* w_getstr(char* str) { if(str && strlen(str) > 0) return g_strdup(str); return NULL; } static TitleInput *ffmpeg_get_song_tuple(gchar * filename) { TitleInput *tuple = NULL; AVFormatContext *in = NULL; gchar *filename_proxy = g_strdup(filename); if(av_open_input_file(&in, str_twenty_to_space(filename), NULL, 0, NULL) < 0) return NULL; tuple = bmp_title_input_new(); tuple->file_name = g_path_get_basename(filename_proxy); tuple->file_path = g_path_get_dirname(filename_proxy); tuple->file_ext = extname(filename_proxy); av_find_stream_info(in); if((in->title[0] != '\0') || (in->author[0] != '\0') || (in->album[0] != '\0') || (in->comment[0] != '\0') || (in->genre[0] != '\0') || (in->year != 0) || (in->track != 0)) { tuple->performer = str_to_utf8(w_getstr(in->author)); tuple->album_name = str_to_utf8(w_getstr(in->album)); tuple->track_name = str_to_utf8(w_getstr(in->title)); tuple->year = in->year; tuple->track_number = in->track; tuple->genre = str_to_utf8(w_getstr(in->genre)); tuple->comment = str_to_utf8(w_getstr(in->comment)); } if (in->duration) tuple->length = in->duration / 1000; av_close_input_file(in); return tuple; } static gchar *get_song_title(AVFormatContext *in, gchar * filename) { gchar *ret = NULL; TitleInput *input; input = bmp_title_input_new(); if((in->title[0] != '\0') || (in->author[0] != '\0') || (in->album[0] != '\0') || (in->comment[0] != '\0') || (in->genre[0] != '\0') || (in->year != 0) || (in->track != 0)) { input->performer = w_getstr(in->author); input->album_name = w_getstr(in->album); input->track_name = w_getstr(in->title); input->year = in->year; input->track_number = in->track; input->genre = w_getstr(in->genre); input->comment = w_getstr(in->comment); } input->file_name = g_path_get_basename(filename); input->file_path = g_path_get_dirname(filename); input->file_ext = extname(filename); ret = xmms_get_titlestring(xmms_get_gentitle_format(), input); if(input) g_free(input); if(!ret) { ret = g_strdup(input->file_name); if (extname(ret) != NULL) *(extname(ret) - 1) = '\0'; } return ret; } static guint get_song_time(AVFormatContext *in) { if(in->duration) return in->duration/1000; else return 0; } static void ffmpeg_get_song_info(char *filename, char **title_real, int *len_real) { TitleInput *tuple = ffmpeg_get_song_tuple(filename); if (tuple == NULL) return; (*len_real) = tuple->length; (*title_real) = xmms_get_titlestring(xmms_get_gentitle_format(), tuple); } static void ffmpeg_playbuff(InputPlayback *playback, int out_size) { AVFifoBuffer f; int sst_buff; av_fifo_init(&f, out_size*2); av_fifo_write(&f, ffmpeg_outbuf, out_size); while(!av_fifo_read(&f, ffmpeg_s_outbuf, ffmpeg_st_buff) && ffmpeg_decode) { sst_buff = ffmpeg_st_buff; if(ffmpeg_pause) memset(ffmpeg_s_outbuf, 0, sst_buff); while(playback->output->buffer_free() < ffmpeg_st_buff) xmms_usleep(20000); produce_audio(playback->output->written_time(), FMT_S16_NE, c->channels, sst_buff, (short *)ffmpeg_s_outbuf, NULL); memset(ffmpeg_s_outbuf, 0, sst_buff); } av_fifo_free(&f); return; } static void *ffmpeg_play_loop(void *arg) { InputPlayback *playback = arg; uint8_t *inbuf_ptr; int out_size, size, len; AVPacket pkt; g_static_mutex_lock(&ffmpeg_mutex); while(ffmpeg_decode){ if(ffmpeg_seekpos != -1) { av_seek_frame(ic, ffmpeg_idx, ffmpeg_seekpos * AV_TIME_BASE, 0); playback->output->flush(ffmpeg_seekpos * 1000); ffmpeg_seekpos = -1; } if(av_read_frame(ic, &pkt) < 0) break; size = pkt.size; inbuf_ptr = pkt.data; if(size == 0) break; while(size > 0){ len = avcodec_decode_audio(c, (short *)ffmpeg_outbuf, &out_size, inbuf_ptr, size); if(len < 0) break; if(out_size <= 0) continue; ffmpeg_playbuff(playback, out_size); size -= len; inbuf_ptr += len; if(pkt.data) av_free_packet(&pkt); } } while(ffmpeg_decode && playback->output->buffer_playing()) xmms_usleep(30000); ffmpeg_decode = 0; if(ffmpeg_s_outbuf) g_free(ffmpeg_s_outbuf); if(ffmpeg_outbuf) g_free(ffmpeg_outbuf); if(pkt.data) av_free_packet(&pkt); if(c) avcodec_close(c); if(ic) av_close_input_file(ic); g_static_mutex_unlock(&ffmpeg_mutex); g_thread_exit(NULL); return(NULL); } static void ffmpeg_play_file(InputPlayback *playback) { char *filename = playback->filename; AVCodec *codec; if(av_open_input_file(&ic, str_twenty_to_space(filename), NULL, 0, NULL) < 0) return; for(ffmpeg_idx = 0; ffmpeg_idx < ic->nb_streams; ffmpeg_idx++) { c = ic->streams[ffmpeg_idx]->codec; if(c->codec_type == CODEC_TYPE_AUDIO) break; } av_find_stream_info(ic); codec = avcodec_find_decoder(c->codec_id); if(!codec) return; if(avcodec_open(c, codec) < 0) return; wsong_title = get_song_title(ic, filename); wsong_time = get_song_time(ic); if(playback->output->open_audio(FMT_S16_NE, c->sample_rate, c->channels) <= 0) return; ffmpeg_st_buff = ST_BUFF; ffmpeg_ip.set_info(wsong_title, wsong_time, c->bit_rate, c->sample_rate, c->channels); /* av_malloc() will wrap posix_memalign() if necessary -nenolod */ ffmpeg_s_outbuf = av_malloc(ffmpeg_st_buff); ffmpeg_outbuf = av_malloc(AVCODEC_MAX_AUDIO_FRAME_SIZE); ffmpeg_seekpos = -1; ffmpeg_decode = 1; ffmpeg_decode_thread = g_thread_create((GThreadFunc)ffmpeg_play_loop, playback, TRUE, NULL); } static void ffmpeg_stop(InputPlayback *playback) { ffmpeg_decode = 0; if(ffmpeg_pause) ffmpeg_do_pause(playback, 0); g_thread_join(ffmpeg_decode_thread); playback->output->close_audio(); }