summaryrefslogtreecommitdiffstats
path: root/libmpdemux/demux_xmms.c
diff options
context:
space:
mode:
authorarpi <arpi@b3059339-0415-0410-9bf9-f77b7e298cf2>2002-12-22 21:01:01 +0000
committerarpi <arpi@b3059339-0415-0410-9bf9-f77b7e298cf2>2002-12-22 21:01:01 +0000
commit61d797672169a892c7b0d45e687fb9a76925c0cd (patch)
tree11a80a8ff74984e59bd24b2bae0bdfa7713f827d /libmpdemux/demux_xmms.c
parent6da94a060f342b3c7f6e838846e922811c1fb683 (diff)
downloadmpv-61d797672169a892c7b0d45e687fb9a76925c0cd.tar.bz2
mpv-61d797672169a892c7b0d45e687fb9a76925c0cd.tar.xz
XMMS Input plugin support
based on patches by Balatoni Denes <pnis@coder.hu> changes by me: glib dependency removed, files merged, code simplified, some bugfixes git-svn-id: svn://svn.mplayerhq.hu/mplayer/trunk@8529 b3059339-0415-0410-9bf9-f77b7e298cf2
Diffstat (limited to 'libmpdemux/demux_xmms.c')
-rw-r--r--libmpdemux/demux_xmms.c336
1 files changed, 336 insertions, 0 deletions
diff --git a/libmpdemux/demux_xmms.c b/libmpdemux/demux_xmms.c
new file mode 100644
index 0000000000..2a03a0be3c
--- /dev/null
+++ b/libmpdemux/demux_xmms.c
@@ -0,0 +1,336 @@
+// This is not reentrant (because of global variables shared with demux_xmms_output.c)
+// and the plugins are not reentrant either perhaps
+#include "config.h"
+
+#include <stdlib.h>
+#include <stdio.h>
+#include <unistd.h>
+#include <pthread.h>
+#include <dlfcn.h>
+#include <dirent.h>
+#include <inttypes.h>
+
+#include "../cfgparser.h"
+#include "../libao2/afmt.h"
+#include "stream.h"
+#include "demuxer.h"
+#include "stheader.h"
+
+#include <dirent.h>
+#include <string.h>
+#include <sys/stat.h>
+//#include <glib.h>
+
+#define XMMS_PACKETSIZE 65536 // some plugins won't play if this is too small
+
+#include "demux_xmms_plugin.h"
+//#include "demux_xmms_pluginenum.h"
+//#include "demux_xmms_input.h"
+
+//extern OutputPlugin xmms_output_plugin;
+
+static pthread_mutex_t xmms_mutex;
+static int format = 0x1; // Raw PCM
+static char xmms_audiobuffer[XMMS_PACKETSIZE];
+static uint32_t xmms_channels;
+static uint32_t xmms_samplerate;
+static uint32_t xmms_afmt;
+static int xmms_length;
+static char *xmms_title=NULL;
+static uint32_t xmms_audiopos=0;
+
+static uint64_t written = 0;
+
+static void disk_close(void) {}
+static void disk_flush(int time) {}
+static void disk_pause(short p) {}
+static void disk_init(void) {}
+
+static int disk_free(void) { // vqf plugin sends more than it should
+ return (XMMS_PACKETSIZE-xmms_audiopos<XMMS_PACKETSIZE/4 ? 0:XMMS_PACKETSIZE-xmms_audiopos-XMMS_PACKETSIZE/4);
+}
+
+static int disk_playing(void) {
+ return 0; //??
+}
+
+static int disk_get_output_time(void) {
+ return 10;
+}
+
+static int disk_open(AFormat fmt, int rate, int nch) {
+ switch (fmt) {
+ case FMT_U8:
+ xmms_afmt=AFMT_U8;
+ break;
+ case FMT_S8:
+ xmms_afmt=AFMT_S8;
+ break;
+ case FMT_U16_LE:
+ xmms_afmt=AFMT_U16_LE;
+ break;
+ case FMT_U16_NE:
+#if WORDS_BIGENDIAN
+ xmms_afmt=AFMT_U16_BE;
+#else
+ xmms_afmt=AFMT_U16_LE;
+#endif
+ break;
+ case FMT_U16_BE:
+ xmms_afmt=AFMT_U16_BE;
+ break;
+ case FMT_S16_NE:
+ xmms_afmt=AFMT_S16_NE;
+ break;
+ case FMT_S16_LE:
+ xmms_afmt=AFMT_S16_LE;
+ break;
+ case FMT_S16_BE:
+ xmms_afmt=AFMT_S16_BE;
+ break;
+ }
+ xmms_samplerate=rate;
+ xmms_channels=nch;
+ return 1;
+}
+
+static void disk_write(void *ptr, int length) {
+ pthread_mutex_lock(&xmms_mutex);
+ written += length;
+ memcpy(&xmms_audiobuffer[xmms_audiopos],ptr,length);
+ xmms_audiopos+=length;
+ pthread_mutex_unlock(&xmms_mutex);
+}
+
+static OutputPlugin xmms_output_plugin =
+{
+ NULL,
+ NULL,
+ "Mplayer output interface Plugin ", /* Description */
+ disk_init,
+ NULL, /* about */
+ NULL, /* configure */
+ NULL, /* get_volume */
+ NULL, /* set_volume */
+ disk_open,
+ disk_write,
+ disk_close,
+ disk_flush,
+ disk_pause,
+ disk_free,
+ disk_playing,
+ disk_get_output_time,
+ disk_get_output_time
+};
+
+
+
+typedef struct {
+ uint32_t spos;
+ InputPlugin* ip;
+} xmms_priv_t;
+
+static InputPlugin* input_plugins[100];
+static int no_plugins=0;
+
+/* Dummy functions */
+static int input_get_vis_type(){return 0;}
+//static void input_add_vis(int time, unsigned char *s, InputVisType type){}
+static void input_add_vis_pcm(int time, AFormat fmt, int nch, int length, void *ptr){}
+//static void input_update_vis(gint time){}
+//static gchar *input_get_info_text(void){return NULL;}
+static void input_set_info_text(char * text){}
+static void input_set_info(char* ha,int a, int b, int c, int d){};
+/* Dummy functions END*/
+
+static void init_plugins(){
+ DIR *dir;
+ struct dirent *ent;
+
+ no_plugins=0;
+
+ dir = opendir(XMMS_INPUT_PLUGIN_DIR);
+ if (!dir) return;
+
+ while ((ent = readdir(dir)) != NULL){
+ char filename[strlen(XMMS_INPUT_PLUGIN_DIR)+strlen(ent->d_name)+4];
+ void* handle;
+ sprintf(filename,XMMS_INPUT_PLUGIN_DIR "/%s",ent->d_name);
+ handle=dlopen(filename, RTLD_NOW);
+ if(handle){
+ void *(*gpi) (void);
+ gpi=dlsym(handle, "get_iplugin_info");
+ if(gpi){
+ InputPlugin *p=gpi();
+ printf("XMMS: found plugin: %s (%s)\n",ent->d_name,p->description);
+ p->handle = handle;
+ p->filename = strdup(filename);
+ p->get_vis_type = input_get_vis_type;
+ p->add_vis_pcm = input_add_vis_pcm;
+ p->set_info = input_set_info;
+ p->set_info_text = input_set_info_text;
+ if(p->init) p->init();
+ input_plugins[no_plugins++]=p;
+ } else
+ dlclose(handle);
+ }
+ }
+ closedir(dir);
+}
+
+static void cleanup_plugins(){
+ while(no_plugins>0){
+ --no_plugins;
+ printf("XMMS: Closing plugin %s\n",input_plugins[no_plugins]->filename);
+ if(input_plugins[no_plugins]->cleanup)
+ input_plugins[no_plugins]->cleanup();
+ dlclose(input_plugins[no_plugins]->handle);
+ }
+}
+
+extern void resync_audio_stream(sh_audio_t *sh_audio);
+
+int demux_xmms_open(demuxer_t* demuxer) {
+ InputPlugin* ip = NULL;
+ sh_audio_t* sh_audio;
+ WAVEFORMATEX* w;
+ xmms_priv_t *priv;
+ int i;
+
+ init_plugins();
+ for(i=0;i<no_plugins;i++){
+ if (input_plugins[i]->is_our_file(demuxer->stream->url)){
+ ip=input_plugins[i]; break;
+ }
+ }
+ if(!ip) return 0; // no plugin to handle this...
+
+ pthread_mutex_init(&xmms_mutex,NULL);
+
+ priv=(xmms_priv_t *)malloc(sizeof(xmms_priv_t));
+ memset(priv,0,sizeof(xmms_priv_t));
+ priv->ip=ip;
+
+ memset(xmms_audiobuffer,0,XMMS_PACKETSIZE);
+
+ xmms_channels=0;
+ sh_audio = new_sh_audio(demuxer,0);
+ sh_audio->wf = w = (WAVEFORMATEX*)malloc(sizeof(WAVEFORMATEX));
+ w->wFormatTag = sh_audio->format = format;
+
+ demuxer->movi_start = 0;
+ demuxer->movi_end = 100;
+ demuxer->audio->id = 0;
+ demuxer->audio->sh = sh_audio;
+ demuxer->priv=priv;
+ sh_audio->ds = demuxer->audio;
+
+ xmms_output_plugin.init();
+ ip->get_song_info(demuxer->stream->url,&xmms_title,&xmms_length);
+ if (xmms_length<=0) demuxer->seekable=0;
+// printf("XMMS song title='%s' length=%d\n",xmms_title,xmms_length);
+// input_play(demuxer->stream->url);
+ ip->output = &xmms_output_plugin;
+ ip->play_file(demuxer->stream->url);
+
+ mp_msg(MSGT_DEMUX,MSGL_INFO,"Waiting for the XMMS plugin to start playback of '%s'...\n",demuxer->stream->url);
+ while (xmms_channels==0) {
+ usleep(10000);
+ if(ip->get_time()<0) return 0;
+ }
+ sh_audio->sample_format= xmms_afmt;
+ switch (xmms_afmt) {
+ case AFMT_S16_LE:
+ case AFMT_S16_BE:
+ case AFMT_U16_LE:
+ case AFMT_U16_BE:
+ sh_audio->samplesize = 2;
+ break;
+ default:
+ sh_audio->samplesize = 1;
+ }
+ w->wBitsPerSample = sh_audio->samplesize*8;
+ w->nChannels = sh_audio->channels = xmms_channels;
+ w->nSamplesPerSec = sh_audio->samplerate = xmms_samplerate;
+ w->nAvgBytesPerSec = xmms_samplerate*sh_audio->channels*sh_audio->samplesize;
+ w->nBlockAlign = sh_audio->samplesize*sh_audio->channels;
+ w->cbSize = 0;
+
+ return 1;
+}
+
+int demux_xmms_fill_buffer(demuxer_t* demuxer, demux_stream_t *ds) {
+ sh_audio_t *sh_audio = demuxer->audio->sh;
+ xmms_priv_t *priv=demuxer->priv;
+ demux_packet_t* dp;
+
+ while (xmms_audiopos<XMMS_PACKETSIZE/2) {
+ if(priv->ip->get_time()<0) return 0;
+ usleep(1000);
+ }
+
+ pthread_mutex_lock(&xmms_mutex);
+ dp = new_demux_packet(XMMS_PACKETSIZE/2);
+ ds->pts = priv->spos / (float)(sh_audio->wf->nAvgBytesPerSec);
+ ds->pos = priv->spos;
+ priv->spos+=XMMS_PACKETSIZE/2;
+ memcpy(dp->buffer,xmms_audiobuffer,XMMS_PACKETSIZE/2);
+ memcpy(xmms_audiobuffer,&xmms_audiobuffer[XMMS_PACKETSIZE/2],xmms_audiopos-XMMS_PACKETSIZE/2);
+ xmms_audiopos-=XMMS_PACKETSIZE/2;
+ pthread_mutex_unlock(&xmms_mutex);
+
+ ds_add_packet(ds,dp);
+
+ return 1;
+}
+
+void demux_xmms_seek(demuxer_t *demuxer,float rel_seek_secs,int flags){
+ stream_t* s = demuxer->stream;
+ sh_audio_t* sh_audio = demuxer->audio->sh;
+ xmms_priv_t *priv=demuxer->priv;
+ float pos;
+
+ if(priv->ip->get_time()<0) return;
+
+ pos = (flags & 1) ? 0 : priv->spos / (float)(sh_audio->wf->nAvgBytesPerSec);
+ if (flags & 2)
+ pos+= rel_seek_secs*xmms_length;
+ else
+ pos+= rel_seek_secs;
+
+ if (pos<0) pos=0;
+ if (pos>=xmms_length) pos=xmms_length-1;
+
+ priv->ip->seek((pos<0)?0:(int)pos);
+ priv->spos=pos * (float)(sh_audio->wf->nAvgBytesPerSec);;
+}
+
+int demux_close_xmms(demuxer_t* demuxer) {
+ xmms_priv_t *priv=demuxer->priv;
+ priv->ip->stop();
+ free(priv); demuxer->priv=NULL;
+ cleanup_plugins();
+ return 1;
+}
+
+int demux_xmms_control(demuxer_t *demuxer,int cmd, void *arg){
+ demux_stream_t *d_video=demuxer->video;
+ sh_audio_t *sh_audio=demuxer->audio->sh;
+ xmms_priv_t *priv=demuxer->priv;
+
+ switch(cmd) {
+ case DEMUXER_CTRL_GET_TIME_LENGTH:
+ if (xmms_length<=0) return DEMUXER_CTRL_DONTKNOW;
+ *((unsigned long *)arg)=(unsigned long)xmms_length/1000;
+ return DEMUXER_CTRL_GUESS;
+
+ case DEMUXER_CTRL_GET_PERCENT_POS:
+ if (xmms_length<=0)
+ return DEMUXER_CTRL_DONTKNOW;
+ *((int *)arg)=(int)( priv->spos / (float)(sh_audio->wf->nAvgBytesPerSec) / xmms_length );
+ return DEMUXER_CTRL_OK;
+
+ default:
+ return DEMUXER_CTRL_NOTIMPL;
+ }
+}