changeset 8528:9d143176d95f

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
author arpi
date Sun, 22 Dec 2002 21:01:01 +0000
parents 157667ae4d85
children ffbe67e968e2
files Makefile configure libmpdemux/Makefile libmpdemux/demux_xmms.c libmpdemux/demux_xmms_plugin.h libmpdemux/demuxer.c libmpdemux/demuxer.h libmpdemux/open.c
diffstat 8 files changed, 578 insertions(+), 3 deletions(-) [+]
line wrap: on
line diff
--- a/Makefile	Sun Dec 22 20:16:52 2002 +0000
+++ b/Makefile	Sun Dec 22 21:01:01 2002 +0000
@@ -35,7 +35,7 @@
 
 VO_LIBS = $(AA_LIB) $(X_LIB) $(SDL_LIB) $(GGI_LIB) $(MP1E_LIB) $(MLIB_LIB) $(SVGA_LIB) $(DIRECTFB_LIB) $(GIF_LIB) 
 AO_LIBS = $(ARTS_LIB) $(NAS_LIB) $(SGIAUDIO_LIB)
-CODEC_LIBS = $(AV_LIB) $(FAME_LIB) $(MAD_LIB) $(VORBIS_LIB) $(FAAD_LIB) $(LIBLZO_LIB) $(XVID_LIB) $(DECORE_LIB) $(PNG_LIB) $(Z_LIB) $(JPEG_LIB) $(ALSA_LIB) 
+CODEC_LIBS = $(AV_LIB) $(FAME_LIB) $(MAD_LIB) $(VORBIS_LIB) $(FAAD_LIB) $(LIBLZO_LIB) $(XVID_LIB) $(DECORE_LIB) $(PNG_LIB) $(Z_LIB) $(JPEG_LIB) $(ALSA_LIB) $(XMMS_LIB)
 COMMON_LIBS = libmpcodecs/libmpcodecs.a mp3lib/libMP3.a liba52/liba52.a libmpeg2/libmpeg2.a $(W32_LIB) $(DS_LIB) libaf/libaf.a libmpdemux/libmpdemux.a input/libinput.a $(PP_LIB) postproc/libswscale.a linux/libosdep.a $(CSS_LIB) $(CODEC_LIBS) $(FREETYPE_LIB) $(TERMCAP_LIB) $(CDPARANOIA_LIB) $(STREAMING_LIB) $(WIN32_LIB)
 
 CFLAGS = $(OPTFLAGS) -Ilibmpdemux -Iloader -Ilibvo $(FREETYPE_INC) $(EXTRA_INC) $(CDPARANOIA_INC) $(SDL_INC) # -Wall
--- a/configure	Sun Dec 22 20:16:52 2002 +0000
+++ b/configure	Sun Dec 22 21:01:01 2002 +0000
@@ -175,6 +175,7 @@
   --enable-faad          build with FAAD2 (MP4/AAC) support [autodetect]
   --disable-libdv        disable libdv 0.9.5 en/decoding support [autodetect]
   --disable-mad          disable libmad (mpeg audio) support [autodetect]
+  --enable-xmms          build with XMMS inputplugin support [disabled]
   
 Video output:
   --disable-vidix        disable VIDIX stuff [enable on x86 *nix]
@@ -268,6 +269,8 @@
   --with-glib-config=PATH  path to glib*-config (e.g. /opt/bin/glib-config)
   --with-dvdnav-config=PATH  path to dvdnav-config
   --with-livelibdir=DIR    path to LIVE.COM Streaming Media libraries
+  --with-xmmsplugindir=DIR path to xmms plugins
+  --with-xmmslibdir=DIR    path to libxmms.so.1
 
 EOF
     exit 0
@@ -976,6 +979,7 @@
 _vorbis=auto
 _tremor=no
 _faad=auto
+_xmms=no
 _css=auto
 _dvdnav=yes
 _dvdread=auto
@@ -1116,6 +1120,7 @@
   --disable-tremor)	_tremor=no	;;
   --enable-faad)	_faad=yes	;;
   --disable-faad)	_faad=no	;;
+  --enable-xmms)	_xmms=yes	;;
   --enable-css)		_css=yes	;;
   --disable-css)	_css=no		;;
   --enable-dvdread)	_dvdread=yes	;;
@@ -1259,6 +1264,14 @@
     _mlib=yes
     ;;
 
+  --with-xmmslibdir=*)
+    _xmmslibdir=`echo $ac_option | cut -d '=' -f 2`
+    ;;
+    
+  --with-xmmsplugindir=*)
+    _xmmsplugindir=`echo $ac_option | cut -d '=' -f 2`
+    ;;
+    
   --disable-profile)
     _profile=
     ;;
@@ -4346,6 +4359,33 @@
 fi
 echores "$_sortsub"
 
+
+echocheck "XMMS inputplugin support"
+if test "$_xmms" = yes ; then
+
+  if ( xmms-config --version ) >/dev/null 2>&1 ; then
+    if test -z "$_xmmsplugindir" ; then
+      _xmmsplugindir=`xmms-config --input-plugin-dir`
+    fi
+    if test -z "$_xmmslibdir" ; then
+      _xmmslibdir=`xmms-config --exec-prefix`/lib
+    fi
+  else
+    if test -z "$_xmmsplugindir" ; then
+      _xmmsplugindir=/usr/lib/xmms/Input
+    fi
+    if test -z "$_xmmslibdir" ; then
+      _xmmslibdir=/usr/lib
+    fi
+  fi
+
+  _def_xmms='#define HAVE_XMMS 1'
+  _xmms_lib="${_xmmslibdir}/libxmms.so.1 -export-dynamic"
+else
+  _def_xmms='#undef HAVE_XMMS'
+fi
+echores "$_xmms"
+
 # --------------- GUI specific tests begin -------------------
 echocheck "GUI"
 echo "$_gui"
@@ -4695,6 +4735,8 @@
 MAD_LIB = $_ld_mad
 VORBIS_LIB = $_ld_vorbis $_ld_libdv
 FAAD_LIB = $_ld_faad
+XMMS_PLUGINS = $_xmms
+XMMS_LIB = $_xmms_lib
 
 # --- Some stuff for autoconfigure ----
 $_target_arch
@@ -5101,6 +5143,10 @@
 /* enables / disables subtitles sorting */
 $_def_sortsub
 
+/* XMMS input plugin support */
+$_def_xmms
+#define XMMS_INPUT_PLUGIN_DIR "$_xmmsplugindir"
+
 /* Extension defines */
 $_def_3dnow	// only define if you have 3DNOW (AMD k6-2, AMD Athlon, iDT WinChip, etc.)
 $_def_3dnowex	// only define if you have 3DNOWEX (AMD Athlon, etc.)
--- a/libmpdemux/Makefile	Sun Dec 22 20:16:52 2002 +0000
+++ b/libmpdemux/Makefile	Sun Dec 22 21:01:01 2002 +0000
@@ -4,6 +4,9 @@
 include ../config.mak
 
 SRCS = mp3_hdr.c video.c mpeg_hdr.c cache2.c asfheader.c aviheader.c aviprint.c aviwrite.c demux_asf.c demux_avi.c demux_mov.c parse_mp4.c demux_mpg.c demux_pva.c demux_viv.c demuxer.c dvdauth.c dvdnav_stream.c open.c parse_es.c stream.c tv.c tvi_dummy.c tvi_v4l.c tvi_bsdbt848.c frequencies.c demux_fli.c demux_real.c demux_y4m.c yuv4mpeg.c yuv4mpeg_ratio.c demux_nuv.c demux_film.c demux_roq.c mf.c demux_mf.c demux_audio.c demux_demuxers.c demux_ogg.c demux_bmp.c cdda.c demux_rawaudio.c cddb.c cdinfo.c demux_rawdv.c ai_alsa.c ai_oss.c audio_in.c demux_smjpeg.c
+ifeq ($(XMMS_PLUGINS),yes)
+SRCS += demux_xmms.c
+endif 
 ifeq ($(STREAMING),yes)
 SRCS += asf_streaming.c url.c http.c network.c asf_mmst_streaming.c
 ifeq ($(STREAMING_LIVE_DOT_COM),yes)
@@ -20,7 +23,7 @@
 OBJS	= $(SRCS:.c=.o)
 OBJS   += $(CPLUSPLUSSRCS:.cpp=.o)
 INCLUDE = -I../loader $(CSS_INC) $(EXTRA_INC)
-CFLAGS  = $(OPTFLAGS) $(INCLUDE)
+CFLAGS  = $(OPTFLAGS) $(INCLUDE) $(XMMS_CFLAGS)
 CPLUSPLUSFLAGS  = $(CFLAGS) $(CPLUSPLUSINCLUDE)
 CPLUSPLUS = $(CC)
 
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/libmpdemux/demux_xmms.c	Sun Dec 22 21:01:01 2002 +0000
@@ -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;
+    }
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/libmpdemux/demux_xmms_plugin.h	Sun Dec 22 21:01:01 2002 +0000
@@ -0,0 +1,155 @@
+/*  XMMS - Cross-platform multimedia player
+ *  Copyright (C) 1998-2000  Peter Alm, Mikael Alm, Olle Hallnas, Thomas Nilsson and 4Front Technologies
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met: 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer. 2.
+ * Redistributions in binary form must reproduce the above copyright notice,
+ * this list of conditions and the following disclaimer in the documentation
+ * and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE FOR
+ * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
+ * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
+ * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ */
+#ifndef PLUGIN_H
+#define PLUGIN_H
+
+typedef enum
+{
+	FMT_U8, FMT_S8, FMT_U16_LE, FMT_U16_BE, FMT_U16_NE, FMT_S16_LE, FMT_S16_BE, FMT_S16_NE
+}
+AFormat;
+
+typedef struct
+{
+	void *handle;		/* Filled in by xmms */
+	char *filename;		/* Filled in by xmms */
+	char *description;	/* The description that is shown in the preferences box */
+	void (*init) (void);
+	void (*about) (void);	/* Show the about box */
+	void (*configure) (void);	/* Show the configuration dialog */
+	void (*get_volume) (int *l, int *r);
+	void (*set_volume) (int l, int r);	/* Set the volume */
+	int (*open_audio) (AFormat fmt, int rate, int nch);	/* Open the device, if the device can't handle the given 
+								   parameters the plugin is responsible for downmixing
+								   the data to the right format before outputting it */
+	void (*write_audio) (void *ptr, int length);	/* The input plugin calls this to write data to the output 
+							   buffer */
+	void (*close_audio) (void);	/* No comment... */
+	void (*flush) (int time);	/* Flush the buffer and set the plugins internal timers to time */
+	void (*pause) (short paused);	/* Pause or unpause the output */
+	int (*buffer_free) (void);	/* Return the amount of data that can be written to the buffer,
+					   two calls to this without a call to write_audio should make
+					   the plugin output audio directly */
+	int (*buffer_playing) (void);	/* Returns TRUE if the plugin currently is playing some audio,
+					   otherwise return FALSE */
+	int (*output_time) (void);	/* Return the current playing time */
+	int (*written_time) (void);	/* Return the length of all the data that has been written to
+					   the buffer */
+}
+OutputPlugin;
+
+typedef struct
+{
+	void *handle;		/* Filled in by xmms */
+	char *filename;		/* Filled in by xmms */
+	char *description;	/* The description that is shown in the preferences box */
+	void (*init) (void);	/* Called when the plugin is loaded */
+	void (*cleanup) (void);	/* Called when the plugin is unloaded */
+	void (*about) (void);	/* Show the about box */
+	void (*configure) (void);	/* Show the configure box */
+	int (*mod_samples) (void *data, int length, AFormat fmt, int srate, int nch);	/* Modify samples */
+	void (*query_format) (AFormat *fmt,int *rate, int *nch);
+}
+EffectPlugin;
+
+typedef enum
+{
+	INPUT_VIS_ANALYZER, INPUT_VIS_SCOPE, INPUT_VIS_VU, INPUT_VIS_OFF
+}
+InputVisType;
+
+typedef struct
+{
+	void *handle;		/* Filled in by xmms */
+	char *filename;		/* Filled in by xmms */
+	char *description;	/* The description that is shown in the preferences box */
+	void (*init) (void);	/* Called when the plugin is loaded */
+	void (*about) (void);	/* Show the about box */
+	void (*configure) (void);
+	int (*is_our_file) (char *filename);	/* Return 1 if the plugin can handle the file */
+	void *(*scan_dir) (char *dirname);	/* Look in Input/cdaudio/cdaudio.c to see how */
+	/* to use this */
+	void (*play_file) (char *filename);	/* Guess what... */
+	void (*stop) (void);	/* Tricky one */
+	void (*pause) (short paused);	/* Pause or unpause */
+	void (*seek) (int time);	/* Seek to the specified time */
+	void (*set_eq) (int on, float preamp, float *bands);	/* Set the equalizer, most plugins won't be able to do this */
+	int (*get_time) (void);	/* Get the time, usually returns the output plugins output time */
+	void (*get_volume) (int *l, int *r);	/* Input-plugin specific volume functions, just provide a NULL if */
+	void (*set_volume) (int l, int r);	/*  you want the output plugin to handle it */
+	void (*cleanup) (void);			/* Called when xmms exit */
+	InputVisType (*get_vis_type) (void); /* OBSOLETE, DO NOT USE! */
+	void (*add_vis_pcm) (int time, AFormat fmt, int nch, int length, void *ptr); /* Send data to the visualization plugins 
+											Preferably 512 samples/block */
+	void (*set_info) (char *title, int length, int rate, int freq, int nch);	/* Fill in the stuff that is shown in the player window
+											   set length to -1 if it's unknown. Filled in by xmms */
+	void (*set_info_text) (char *text);	/* Show some text in the song title box in the main window,
+						   call it with NULL as argument to reset it to the song title.
+						   Filled in by xmms */
+	void (*get_song_info) (char *filename, char **title, int *length);	/* Function to grab the title string */
+	void (*file_info_box) (char *filename);		/* Bring up an info window for the filename passed in */
+	OutputPlugin *output;	/* Handle to the current output plugin. Filled in by xmms */
+}
+InputPlugin;
+
+/* So that input plugins can get the title formatting information */
+char *xmms_get_gentitle_format(void);
+
+/* So that output plugins can communicate with effect plugins */
+EffectPlugin *get_current_effect_plugin(void);
+int effects_enabled(void);
+
+typedef struct
+{
+	void *handle;		/* Filled in by xmms */
+	char *filename;		/* Filled in by xmms */
+	int xmms_session;	/* The session ID for attaching to the control socket */
+	char *description;	/* The description that is shown in the preferences box */
+	void (*init) (void);	/* Called when the plugin is enabled */
+	void (*about) (void);	/* Show the about box */
+	void (*configure) (void);
+	void (*cleanup) (void);	/* Called when the plugin is disabled or when xmms exits */
+}
+GeneralPlugin;
+
+typedef struct _VisPlugin
+{
+	void *handle; 	/* Filled in by xmms */
+	char *filename; /* Filled in by xmms */
+	int xmms_session; /* The session ID for attaching to the control socket */
+	char *description; /* The description that is shown in the preferences box */
+	int num_pcm_chs_wanted; /* Numbers of PCM channels wanted in the call to render_pcm */
+	int num_freq_chs_wanted; /* Numbers of freq channels wanted in the call to render_freq */
+	void (*init)(void); /* Called when the plugin is enabled */
+	void (*cleanup)(void); /* Called when the plugin is disabled */
+	void (*about)(void); /* Show the about box */
+	void (*configure)(void); /* Show the configure box */
+	void (*disable_plugin)(struct _VisPlugin *); /* Call this with a pointer to your plugin to disable the plugin */
+	void (*playback_start)(void); /* Called when playback starts */
+	void (*playback_stop)(void); /* Called when playback stops */
+	void (*render_pcm)(short pcm_data[2][512]); /* Render the PCM data, don't do anything time consuming in here */
+	void (*render_freq)(short freq_data[2][256]); /* Render the freq data, don't do anything time consuming in here */
+} VisPlugin;
+
+#endif
--- a/libmpdemux/demuxer.c	Sun Dec 22 20:16:52 2002 +0000
+++ b/libmpdemux/demuxer.c	Sun Dec 22 21:01:01 2002 +0000
@@ -132,6 +132,7 @@
 extern void demux_close_rawdv(demuxer_t* demuxer);
 extern void demux_close_pva(demuxer_t* demuxer);
 extern void demux_close_smjpeg(demuxer_t* demuxer);
+extern void demux_close_xmms(demuxer_t* demuxer);
 
 #ifdef USE_TV
 #include "tv.h"
@@ -194,6 +195,11 @@
     case DEMUXER_TYPE_AVI_NI:
     case DEMUXER_TYPE_AVI_NINI:
       demux_close_avi(demuxer); return;
+#ifdef HAVE_XMMS
+    case DEMUXER_TYPE_XMMS:
+      demux_close_xmms(demuxer); break;
+#endif
+
     }
     // free streams:
     for(i=0;i<256;i++){
@@ -269,6 +275,7 @@
 int demux_y4m_fill_buffer(demuxer_t *demux);
 int demux_audio_fill_buffer(demux_stream_t *ds);
 int demux_pva_fill_buffer(demuxer_t *demux);
+int demux_xmms_fill_buffer(demuxer_t *demux,demux_stream_t *ds);
 
 extern int demux_demuxers_fill_buffer(demuxer_t *demux,demux_stream_t *ds);
 extern int demux_ogg_fill_buffer(demuxer_t *d);
@@ -303,6 +310,9 @@
 #endif
     case DEMUXER_TYPE_Y4M: return demux_y4m_fill_buffer(demux);
     case DEMUXER_TYPE_AUDIO: return demux_audio_fill_buffer(ds);
+#ifdef HAVE_XMMS
+    case DEMUXER_TYPE_XMMS: return demux_xmms_fill_buffer(demux,ds);
+#endif
     case DEMUXER_TYPE_DEMUXERS: return demux_demuxers_fill_buffer(demux,ds);
 #ifdef HAVE_OGGVORBIS
     case DEMUXER_TYPE_OGG: return demux_ogg_fill_buffer(demux);
@@ -533,6 +543,7 @@
 extern int smjpeg_check_file(demuxer_t *demuxer);
 extern int demux_open_smjpeg(demuxer_t* demuxer);
 extern int bmp_check_file(demuxer_t *demuxer);
+extern int demux_xmms_open(demuxer_t* demuxer);
 
 extern demuxer_t* init_avi_with_ogg(demuxer_t* demuxer);
 
@@ -850,6 +861,19 @@
     demuxer = NULL;
   }
 }
+#ifdef HAVE_XMMS
+//=============== Try to open as XMMS file: =================
+if(file_format==DEMUXER_TYPE_UNKNOWN){
+  demuxer=new_demuxer(stream,DEMUXER_TYPE_XMMS,audio_id,video_id,dvdsub_id);
+  if(demux_xmms_open(demuxer)){
+    mp_msg(MSGT_DEMUXER,MSGL_INFO,MSGTR_DetectedAudiofile);
+    file_format=DEMUXER_TYPE_XMMS;
+  } else {
+    free_demuxer(demuxer);
+    demuxer = NULL;
+  }
+}
+#endif
 //=============== Try to open as a RTP stream: ===========
  if(file_format==DEMUXER_TYPE_RTP) {
    demuxer=new_demuxer(stream,DEMUXER_TYPE_RTP,audio_id,video_id,dvdsub_id);
@@ -1107,6 +1131,7 @@
 extern void demux_demuxers_seek(demuxer_t *demuxer,float rel_seek_secs,int flags);
 extern void demux_ogg_seek(demuxer_t *demuxer,float rel_seek_secs,int flags);
 extern void demux_rawaudio_seek(demuxer_t *demuxer,float rel_seek_secs,int flags);
+extern void demux_xmms_seek(demuxer_t *demuxer,float rel_seek_secs,int flags);
 
 int demux_seek(demuxer_t *demuxer,float rel_seek_secs,int flags){
     demux_stream_t *d_audio=demuxer->audio;
@@ -1188,6 +1213,10 @@
 #endif
  case DEMUXER_TYPE_RAWAUDIO:
       demux_rawaudio_seek(demuxer,rel_seek_secs,flags);  break;
+#ifdef HAVE_XMMS
+ case DEMUXER_TYPE_XMMS:
+      demux_xmms_seek(demuxer,rel_seek_secs,flags); break;
+#endif
 
 
 } // switch(demuxer->file_format)
@@ -1248,6 +1277,7 @@
 extern int demux_mpg_control(demuxer_t *demuxer, int cmd, void *arg);
 extern int demux_asf_control(demuxer_t *demuxer, int cmd, void *arg);
 extern int demux_avi_control(demuxer_t *demuxer, int cmd, void *arg);
+extern int demux_xmms_control(demuxer_t *demuxer, int cmd, void *arg);
 
 int demux_control(demuxer_t *demuxer, int cmd, void *arg) {
     switch(demuxer->type) {
@@ -1260,7 +1290,10 @@
 	case DEMUXER_TYPE_AVI_NI:
 	case DEMUXER_TYPE_AVI_NINI:
 	    return demux_avi_control(demuxer,cmd,arg);
-
+#ifdef HAVE_XMMS
+	case DEMUXER_TYPE_XMMS:
+	    return demux_xmms_control(demuxer,cmd,arg);
+#endif
 	default:
 	    return DEMUXER_CTRL_NOTIMPL;
     }
--- a/libmpdemux/demuxer.h	Sun Dec 22 20:16:52 2002 +0000
+++ b/libmpdemux/demuxer.h	Sun Dec 22 21:01:01 2002 +0000
@@ -33,6 +33,7 @@
 #define DEMUXER_TYPE_RAWDV 22
 #define DEMUXER_TYPE_PVA 23
 #define DEMUXER_TYPE_SMJPEG 24
+#define DEMUXER_TYPE_XMMS 25
 
 // This should always match the higest demuxer type number.
 // Unless you want to disallow users to force the demuxer to some types
--- a/libmpdemux/open.c	Sun Dec 22 20:16:52 2002 +0000
+++ b/libmpdemux/open.c	Sun Dec 22 21:01:01 2002 +0000
@@ -526,6 +526,7 @@
 
        stream=new_stream(f,STREAMTYPE_FILE);
        stream->end_pos=len;
+       stream->url=filename?strdup(filename):NULL;
        return stream;
 }