Still has some odd problem where dsound thinks we have buffer underruns and plays audio way too fast, anyone with ideas? Otherwise this should be good for a first submission. * documentation/samples/config, dlls/Makefile.in, configure.ac, DEVELOPERS-HINTS, dlls/winmm/winejack/Makefile.in, audio.c, jack.c, jack.h, winejack.drv.spec: Chris Morgan <cmorgan@alum.wpi.edu> Add support for Jack audio server, http://jackit.sourceforge.net.
Index: documentation/samples/config =================================================================== RCS file: /home/wine/wine/documentation/samples/config,v retrieving revision 1.36 diff -u -r1.36 config --- documentation/samples/config 15 Nov 2002 01:01:48 -0000 1.36 +++ documentation/samples/config 10 Dec 2002 00:20:17 -0000 @@ -253,8 +253,9 @@ "4" = "/usr/X11R6/lib/X11/fonts/Type1" [WinMM] -"Drivers" = "wineoss.drv" #"Drivers" = "winearts.drv" +#"Drivers" = "winejack.drv" +"Drivers" = "wineoss.drv" "WaveMapper" = "msacm.drv" "MidiMapper" = "midimap.drv" Index: dlls/Makefile.in =================================================================== RCS file: /home/wine/wine/dlls/Makefile.in,v retrieving revision 1.160 diff -u -r1.160 Makefile.in --- dlls/Makefile.in 19 Nov 2002 00:47:13 -0000 1.160 +++ dlls/Makefile.in 10 Dec 2002 00:20:18 -0000 @@ -98,6 +98,7 @@ winmm/wavemap \ winmm/winealsa \ winmm/winearts \ + winmm/winejack \ winmm/wineaudioio \ winmm/winenas \ winmm/wineoss \ @@ -254,6 +255,7 @@ winearts.drv$(DLLEXT) \ wineaudioio.drv$(DLLEXT) \ winedos.dll$(DLLEXT) \ + winejack.drv$(DLLEXT) \ winemp3.acm$(DLLEXT) \ winenas.drv$(DLLEXT) \ wineoss.drv$(DLLEXT) \ @@ -525,6 +527,9 @@ winedos.dll$(DLLEXT) winedos16.dll$(DLLEXT): winedos/winedos.dll$(DLLEXT) $(RM) $@ && $(LN_S) winedos/winedos.dll$(DLLEXT) $@ +winejack.drv$(DLLEXT): winmm/winejack/winejack.drv$(DLLEXT) + $(RM) $@ && $(LN_S) winmm/winejack/winejack.drv$(DLLEXT) $@ + winemp3.acm$(DLLEXT): msacm/winemp3/winemp3.acm$(DLLEXT) $(RM) $@ && $(LN_S) msacm/winemp3/winemp3.acm$(DLLEXT) $@ @@ -652,6 +657,7 @@ winmm/winearts/winearts.drv$(DLLEXT): winmm/winearts winmm/wineaudioio/wineaudioio.drv$(DLLEXT): winmm/wineaudioio winedos/winedos.dll$(DLLEXT): winedos +winmm/winejack/winejack.drv$(DLLEXT): winmm/winejack msacm/winemp3/winemp3.acm$(DLLEXT): msacm/winemp3 winmm/winenas/winenas.drv$(DLLEXT): winmm/winenas winmm/wineoss/wineoss.drv$(DLLEXT): winmm/wineoss @@ -752,6 +758,7 @@ winmm/winearts/__install__: winearts.drv$(DLLEXT) winmm/wineaudioio/__install__: wineaudioio.drv$(DLLEXT) winedos/__install__: winedos.dll$(DLLEXT) +winmm/winejack/__install__: winejack.drv$(DLLEXT) msacm/winemp3/__install__: winemp3.acm$(DLLEXT) winmm/winenas/__install__: winenas.drv$(DLLEXT) winmm/wineoss/__install__: wineoss.drv$(DLLEXT) @@ -865,6 +872,7 @@ winmm/wavemap: msacm32.dll$(DLLEXT) winmm.dll$(DLLEXT) user32.dll$(DLLEXT) kernel32.dll$(DLLEXT) winmm/winealsa: winmm.dll$(DLLEXT) user32.dll$(DLLEXT) kernel32.dll$(DLLEXT) ntdll.dll$(DLLEXT) winmm/winearts: winmm.dll$(DLLEXT) user32.dll$(DLLEXT) kernel32.dll$(DLLEXT) +winmm/winejack: winmm.dll$(DLLEXT) user32.dll$(DLLEXT) kernel32.dll$(DLLEXT) winmm/wineaudioio: winmm.dll$(DLLEXT) user32.dll$(DLLEXT) kernel32.dll$(DLLEXT) ntdll.dll$(DLLEXT) winmm/winenas: winmm.dll$(DLLEXT) user32.dll$(DLLEXT) kernel32.dll$(DLLEXT) winmm/wineoss: winmm.dll$(DLLEXT) user32.dll$(DLLEXT) kernel32.dll$(DLLEXT) Index: configure.ac =================================================================== RCS file: /home/wine/wine/configure.ac,v retrieving revision 1.102 diff -u -r1.102 configure.ac --- configure.ac 2 Dec 2002 21:17:05 -0000 1.102 +++ configure.ac 10 Dec 2002 00:20:18 -0000 @@ -602,6 +602,29 @@ AC_DEFINE(HAVE_ARTS, 1, [Define if you have ARTS sound server]) fi +dnl **** Check for Jack Sound Server **** +AC_CACHE_CHECK([for Jack Sound server], + ac_cv_c_jackserver, + JACK_LIBS="-lpthread -ljack" + jack_prefix="/usr/local" + oldcflags=$CFLAGS + CFLAGS="$CFLAGS $JACK_LIBS -ldl -I$jack_prefix/include" + AC_CHECK_LIB(jack,jack_activate, + [ + ac_cv_c_jackserver=yes + ], + [ + ac_cv_c_jackserver=no + ]) + CFLAGS=$oldcflags +) + +if test "$ac_cv_c_jackserver" = "yes" +then + AC_SUBST(JACKLIBS, $JACK_LIBS) + AC_DEFINE(HAVE_JACK, 1, [Define if you have Jack sound server]) +fi + dnl **** Check for ALSA **** AC_SUBST(ALSALIBS,"") AC_CHECK_HEADERS(alsa/asoundlib.h sys/asoundlib.h, break) @@ -858,6 +881,7 @@ WINE_GET_SONAME(freetype,FT_Init_FreeType,[$X_LIBS]) WINE_GET_SONAME(GL,glXQueryExtension,[$X_LIBS $X_EXTRA_LIBS]) WINE_GET_SONAME(cups,cupsGetDefault) + WINE_GET_SONAME(jack,jack_client_new) fi @@ -1502,6 +1526,7 @@ dlls/winmm/winearts/Makefile dlls/winmm/wineaudioio/Makefile dlls/winmm/winenas/Makefile +dlls/winmm/winejack/Makefile dlls/winmm/wineoss/Makefile dlls/winnls/Makefile dlls/winsock/Makefile Index: DEVELOPERS-HINTS =================================================================== RCS file: /home/wine/wine/DEVELOPERS-HINTS,v retrieving revision 1.14 diff -u -r1.14 DEVELOPERS-HINTS --- DEVELOPERS-HINTS 18 Sep 2002 23:11:19 -0000 1.14 +++ DEVELOPERS-HINTS 10 Dec 2002 00:20:18 -0000 @@ -89,6 +89,7 @@ midimap/- midi mapper wavemap/- audio mapper winearts/ - ARTS audio driver + winejack/ - jack audio server driver wineoss/- MM driver for OSS systems winnls/ - National Language Support winsock/ --- /dev/null 2002-07-06 21:24:44.000000000 -0500 +++ dlls/winmm/winejack/Makefile.in 2002-11-25 22:21:16.000000000 -0500 @@ -0,0 +1,17 @@ +TOPSRCDIR = @top_srcdir@ +TOPOBJDIR = ../../.. +SRCDIR = @srcdir@ +VPATH = @srcdir@ +MODULE = winejack.drv +IMPORTS = winmm user32 kernel32 +LDDLLFLAGS = @LDDLLFLAGS@ +SYMBOLFILE = $(MODULE).tmp.o + + +C_SRCS = \ + audio.c \ + jack.c + +@MAKE_DLL_RULES@ + +### Dependencies: --- /dev/null 2002-07-06 21:24:44.000000000 -0500 +++ dlls/winmm/winejack/audio.c 2002-12-09 20:14:15.000000000 -0500 @@ -0,0 +1,1611 @@ +/* -*- tab-width: 8; c-basic-offset: 4 -*- */ +/* + * Wine Driver for jack Sound Server + * http://jackit.sourceforge.net + * + * Copyright 1994 Martin Ayotte + * 1999 Eric Pouech (async playing in waveOut/waveIn) + * 2000 Eric Pouech (loops in waveOut) + * 2002 Chris Morgan (jack version of this file) + */ +/* + * TODO: + * implement audio stream resampling for any arbitrary frequenty + * right now we use the winmm layer to do resampling although it would + * be nice to have a full set of algorithms to choose from based on cpu + * time + * implement wave-in support with jack + * + * FIXME: + * pause in waveOut during loop is not handled correctly + */ + +#include "config.h" + +#include <stdlib.h> +#include <stdio.h> +#include <string.h> +#include <unistd.h> +#include <errno.h> +#include <fcntl.h> +#include "windef.h" +#include "wingdi.h" +#include "winerror.h" +#include "wine/winuser16.h" +#include "mmddk.h" +#include "dsound.h" +#include "dsdriver.h" +#include "jack.h" +#include "wine/debug.h" + +#ifdef HAVE_JACK +#include <jack/jack.h> +#endif + + +WINE_DEFAULT_DEBUG_CHANNEL(wave); + +#ifdef HAVE_JACK + +#define MAKE_FUNCPTR(f) static typeof(f) * fp_##f = NULL; + +/* Function pointers for dynamic loading of libjack */ +/* these are prefixed with "fp_", ie. "fp_jack_client_new" */ +MAKE_FUNCPTR(jack_activate); +MAKE_FUNCPTR(jack_connect); +MAKE_FUNCPTR(jack_client_new); +MAKE_FUNCPTR(jack_client_close); +MAKE_FUNCPTR(jack_deactivate); +MAKE_FUNCPTR(jack_set_process_callback); +MAKE_FUNCPTR(jack_set_buffer_size_callback); +MAKE_FUNCPTR(jack_set_sample_rate_callback); +MAKE_FUNCPTR(jack_on_shutdown); +MAKE_FUNCPTR(jack_get_sample_rate); +MAKE_FUNCPTR(jack_port_register); +MAKE_FUNCPTR(jack_port_get_buffer); +MAKE_FUNCPTR(jack_get_ports); +MAKE_FUNCPTR(jack_port_name); +#undef MAKE_FUNCPTR + +/* define the below to work around a bug in jack where closing a port */ +/* takes a very long time, so to get around this we actually don't */ +/* close the port when the device is closed but instead mark the */ +/* corresponding device as unused */ +#define JACK_CLOSE_HACK 1 + +typedef jack_default_audio_sample_t sample_t; +typedef jack_nframes_t nframes_t; + +/* only allow 10 output devices through this driver, this ought to be adequate */ +#define MAX_WAVEOUTDRV (10) +#define MAX_WAVEINDRV (1) + +/* state diagram for waveOut writing: + * + * +---------+-------------+---------------+---------------------------------+ + * | state | function | event | new state | + * +---------+-------------+---------------+---------------------------------+ + * | | open() | | STOPPED | + * | PAUSED | write() | | PAUSED | + * | STOPPED | write() | <thrd create> | PLAYING | + * | PLAYING | write() | HEADER | PLAYING | + * | (other) | write() | <error> | | + * | (any) | pause() | PAUSING | PAUSED | + * | PAUSED | restart() | RESTARTING | PLAYING (if no thrd => STOPPED) | + * | (any) | reset() | RESETTING | STOPPED | + * | (any) | close() | CLOSING | CLOSED | + * +---------+-------------+---------------+---------------------------------+ + */ + +/* states of the playing device */ +#define WINE_WS_PLAYING 0 +#define WINE_WS_PAUSED 1 +#define WINE_WS_STOPPED 2 +#define WINE_WS_CLOSED 3 + +typedef struct { + volatile int state; /* one of the WINE_WS_ manifest constants */ + WAVEOPENDESC waveDesc; + WORD wFlags; + PCMWAVEFORMAT format; + WAVEOUTCAPSA caps; + WORD wDevID; + + jack_port_t* out_port_l; /* ports for left and right channels */ + jack_port_t* out_port_r; + jack_client_t* client; + long sample_rate; /* jack server sample rate */ + +#if JACK_CLOSE_HACK + BOOL in_use; /* TRUE if this device is in use */ +#endif + + char* sound_buffer; + unsigned long buffer_size; + + DWORD volume_left; + DWORD volume_right; + + LPWAVEHDR lpQueuePtr; /* start of queued WAVEHDRs (waiting to be notified) */ + LPWAVEHDR lpPlayPtr; /* start of not yet fully played buffers */ + DWORD dwPartialOffset; /* Offset of not yet written bytes in lpPlayPtr */ + + LPWAVEHDR lpLoopPtr; /* pointer of first buffer in loop, if any */ + DWORD dwLoops; /* private copy of loop counter */ + + DWORD dwPlayedTotal; /* number of bytes actually played since opening */ + DWORD dwWrittenTotal; /* number of bytes written to jack since opening */ + + DWORD bytesInJack; /* bytes that we wrote during the previous JACK_Callback() */ + DWORD tickCountMS; /* time in MS of last JACK_Callback() */ + + /* synchronization stuff */ + CRITICAL_SECTION access_crst; +} WINE_WAVEOUT; + +typedef struct { + volatile int state; + WAVEOPENDESC waveDesc; + WORD wFlags; + PCMWAVEFORMAT format; + LPWAVEHDR lpQueuePtr; + DWORD dwTotalRecorded; + WAVEINCAPSA caps; + BOOL bTriggerSupport; + + /* synchronization stuff */ + CRITICAL_SECTION access_crst; +} WINE_WAVEIN; + +static WINE_WAVEOUT WOutDev [MAX_WAVEOUTDRV]; +static WINE_WAVEIN WInDev [MAX_WAVEINDRV ]; + +static DWORD wodDsCreate(UINT wDevID, PIDSDRIVER* drv); +static LPWAVEHDR wodHelper_PlayPtrNext(WINE_WAVEOUT* wwo); +static DWORD wodHelper_NotifyCompletions(WINE_WAVEOUT* wwo, BOOL force); + +static int JACK_OpenDevice(WINE_WAVEOUT* wwo); + +#if JACK_CLOSE_HACK +static void JACK_CloseDevice(WINE_WAVEOUT* wwo, BOOL close_client); +#else +static void JACK_CloseDevice(WINE_WAVEOUT* wwo); +#endif + + +/*======================================================================* + * Low level WAVE implementation * + *======================================================================*/ + +#define SAMPLE_MAX_16BIT 32767.0f + +/* Alsaplayer function that applies volume changes to a buffer */ +/* (C) Andy Lo A Foe */ +/* Length is in terms of 32 bit samples */ +void volume_effect32(void *buffer, int length, int left, int right) +{ + short *data = (short *)buffer; + int i, v; + + if (right == -1) right = left; + + for(i = 0; i < length; i++) { + v = (int) ((*(data) * left) / 100); + *(data++) = (v>32767) ? 32767 : ((v<-32768) ? -32768 : v); + v = (int) ((*(data) * right) / 100); + *(data++) = (v>32767) ? 32767 : ((v<-32768) ? -32768 : v); + } +} + +/* move 16 bit mono/stereo to 16 bit stereo */ +void sample_move_d16_d16(short *dst, short *src, + unsigned long nsamples, int nChannels) +{ + while(nsamples--) + { + *dst = *src; + dst++; + + if(nChannels == 2) src++; + + *dst = *src; + dst++; + + src++; + } +} + +/* convert from 16 bit to floating point */ +/* allow for copying of stereo data with alternating left/right */ +/* channels to a buffer that will hold a single channel stream */ +/* nsamples is in terms of 16bit samples */ +/* src_skip is in terms of 16bit samples */ +void sample_move_d16_s16 (sample_t *dst, short *src, + unsigned long nsamples, unsigned long src_skip) +{ + /* ALERT: signed sign-extension portability !!! */ + while (nsamples--) + { + *dst = (*src) / SAMPLE_MAX_16BIT; + dst++; + src += src_skip; + } +} + +/* fill dst buffer with nsamples worth of silence */ +void sample_silence_dS (sample_t *dst, unsigned long nsamples) +{ + /* ALERT: signed sign-extension portability !!! */ + while (nsamples--) + { + *dst = 0; + dst++; + } +} + +/****************************************************************** + * JACK_callback + */ +/* everytime the jack server wants something from us it calls this +function, so we either deliver it some sound to play or deliver it nothing +to play */ +int JACK_callback (nframes_t nframes, void *arg) +{ + sample_t* out_l; + sample_t* out_r; + WINE_WAVEOUT* wwo = (WINE_WAVEOUT*)arg; + + TRACE("wDevID: %d, nframes %ld\n", wwo->wDevID, nframes); + + if(!wwo->client) + ERR("client is closed, this is weird...\n"); + + out_l = (sample_t *) fp_jack_port_get_buffer(wwo->out_port_l, + nframes); + out_r = (sample_t *) fp_jack_port_get_buffer(wwo->out_port_r, + nframes); + + EnterCriticalSection(&wwo->access_crst); + + if(wwo->state == WINE_WS_PLAYING) + { + DWORD jackBytesAvailableThisCallback = sizeof(sample_t) * nframes; + DWORD jackBytesLeft = sizeof(sample_t) * nframes; + + DWORD inputBytesAvailable; /* number of bytes we have from the app, after conversion to 16bit stereo */ + DWORD jackBytesToWrite; /* number of bytes we are going to write out, after conversion */ + + DWORD bytesInput; /* the number of bytes from the app */ + DWORD appBytesToWrite; /* number of bytes from the app we are going to write */ + + long written = 0; + char* buffer; + +#if JACK_CLOSE_HACK + if(wwo->in_use == FALSE) + { + /* output silence if nothing is being outputted */ + sample_silence_dS(out_l, nframes); + sample_silence_dS(out_r, nframes); + + return 0; + } +#endif + + TRACE("wwo.state == WINE_WS_PLAYING\n"); + + /* see if our buffer is large enough for the data we are writing */ + /* ie. Buffer_size < (bytes we already wrote + bytes we are going to write in this loop) */ + if(wwo->buffer_size < jackBytesAvailableThisCallback) + { + ERR("for some reason JACK_BufSize() didn't allocate enough memory\n"); + ERR("allocated %ld bytes, need %ld bytes\n", wwo->buffer_size, + jackBytesAvailableThisCallback); + LeaveCriticalSection(&wwo->access_crst); + return 0; + } + + /* while we have jackBytesLeft and a wave header to be played */ + while(jackBytesLeft && wwo->lpPlayPtr) + { + /* find the amount of audio to be played at this time */ + bytesInput = wwo->lpPlayPtr->dwBufferLength - wwo->dwPartialOffset; + inputBytesAvailable = bytesInput; + + /* calculate inputBytesAvailable based on audio format conversion */ + if(wwo->format.wf.nChannels == 1) + inputBytesAvailable<<=1; /* multiply by two for mono->stereo conversion */ + + /* find the minimum of the inputBytesAvailable and the space available */ + jackBytesToWrite = min(jackBytesLeft, inputBytesAvailable); + + /* calculate appBytesToWrite based on audio format conversion */ + appBytesToWrite = jackBytesToWrite; + if(wwo->format.wf.nChannels == 1) + appBytesToWrite>>=1; /* divide by two for stereo->mono conversion */ + + TRACE("jackBytesToWrite == %ld, appBytesToWrite == %ld\n", jackBytesToWrite, appBytesToWrite); + + buffer = wwo->lpPlayPtr->lpData + wwo->dwPartialOffset; + + /* convert from mono to stereo if necessary */ + /* otherwise just memcpy to the output buffer */ + if(wwo->format.wf.nChannels == 1) + { + sample_move_d16_d16((short*)wwo->sound_buffer +((jackBytesAvailableThisCallback - jackBytesLeft) / sizeof(short)), + (short*)buffer, jackBytesToWrite, wwo->format.wf.nChannels); + } else /* just copy the memory over */ + { + memcpy(wwo->sound_buffer + (jackBytesAvailableThisCallback - jackBytesLeft), + buffer, jackBytesToWrite); + } + + /* advance to the next wave header if possible, or advance pointer */ + /* inside of the current header if we haven't completed it */ + if(appBytesToWrite == bytesInput) + { + wodHelper_PlayPtrNext(wwo); /* we wrote the whole waveheader, skip to the next one*/ + } + else + { + wwo->dwPartialOffset+=appBytesToWrite; /* else advance by the bytes we took in to write */ + } + + written+=appBytesToWrite; /* add on what we wrote */ + jackBytesLeft-=jackBytesToWrite; /* take away what was written in terms of output bytes */ + } + + wwo->tickCountMS = GetTickCount(); /* record the current time */ + wwo->dwWrittenTotal+=written; /* update states on wave device */ + wwo->dwPlayedTotal+=wwo->bytesInJack; /* we must have finished with the last bytes or we wouldn't be back inside of this callback again... */ + wwo->bytesInJack = written; /* record the bytes inside of jack */ + + /* Now that we have finished filling the buffer either until it is full or until */ + /* we have run out of application sound data to process, apply volume and output */ + /* the audio to the jack server */ + + /* apply volume to the buffer */ + /* NOTE: buffer_size >> 2 to convert from bytes to 16 bit stereo(32bit) samples */ + volume_effect32(wwo->sound_buffer, (jackBytesAvailableThisCallback - jackBytesLeft)>>2, wwo->volume_left, + wwo->volume_right); + + /* convert from stereo 16 bit to single channel 32 bit float */ + /* for each jack server channel */ + /* NOTE: we skip over two sample since we want to only get either the left or right channel */ + sample_move_d16_s16(out_l, (short*)wwo->sound_buffer, (jackBytesAvailableThisCallback - jackBytesLeft)>>2, 2); + sample_move_d16_s16(out_r, (short*)wwo->sound_buffer + 1, + (jackBytesAvailableThisCallback - jackBytesLeft)>>2, 2); + + /* see if we still have jackBytesLeft here, if we do that means that we + ran out of wave data to play and had a buffer underrun, fill in + the rest of the space with zero bytes */ + if(jackBytesLeft) + { + ERR("buffer underrun of %ld bytes\n", jackBytesLeft); + sample_silence_dS(out_l + ((jackBytesAvailableThisCallback - jackBytesLeft) / sizeof(sample_t)), jackBytesLeft / sizeof(sample_t)); + sample_silence_dS(out_r + ((jackBytesAvailableThisCallback - jackBytesLeft) / sizeof(sample_t)), jackBytesLeft / sizeof(sample_t)); + } + } + else if(wwo->state == WINE_WS_PAUSED || + wwo->state == WINE_WS_STOPPED || + wwo->state == WINE_WS_CLOSED) + { + /* output silence if nothing is being outputted */ + sample_silence_dS(out_l, nframes); + sample_silence_dS(out_r, nframes); + } + + /* notify the client of completed wave headers */ + wodHelper_NotifyCompletions(wwo, FALSE); + + LeaveCriticalSection(&wwo->access_crst); + + TRACE("ending\n"); + + return 0; +} + +/****************************************************************** + * JACK_bufsize + * + * Called whenever the jack server changes the the max number + * of frames passed to JACK_callback + */ +int JACK_bufsize (nframes_t nframes, void *arg) +{ + WINE_WAVEOUT* wwo = (WINE_WAVEOUT*)arg; + DWORD buffer_required; + TRACE("the maximum buffer size is now %lu frames\n", nframes); + + /* make sure the callback routine has adequate memory */ + /* see if our buffer is large enough for the data we are writing */ + /* ie. Buffer_size < (bytes we already wrote + bytes we are going to write in this loop) */ + EnterCriticalSection(&wwo->access_crst); + + buffer_required = sizeof(sample_t) * nframes; + if(wwo->buffer_size < buffer_required) + { + TRACE("expanding buffer from wwo->buffer_size == %ld, to %ld\n", + wwo->buffer_size, buffer_required); + TRACE("GetProcessHeap() == %p\n", GetProcessHeap()); + wwo->buffer_size = buffer_required; + wwo->sound_buffer = HeapReAlloc(GetProcessHeap(), 0, wwo->sound_buffer, wwo->buffer_size); + + /* if we don't have a buffer then error out */ + if(!wwo->sound_buffer) + { + ERR("error allocating sound_buffer memory\n"); + LeaveCriticalSection(&wwo->access_crst); + return 0; + } + } + + LeaveCriticalSection(&wwo->access_crst); + + TRACE("called\n"); + + return 0; +} + +/****************************************************************** + * JACK_srate + */ +int JACK_srate (nframes_t nframes, void *arg) +{ + TRACE("the sample rate is now %lu/sec\n", nframes); + return 0; +} + + +/****************************************************************** + * JACK_shutdown + */ +/* if this is called then jack shut down... handle this appropriately */ +void JACK_shutdown(void* arg) +{ + WINE_WAVEOUT* wwo = (WINE_WAVEOUT*)arg; + + wwo->client = 0; /* reset client */ + + TRACE("trying to reconnect after sleeping for a short while...\n"); + + /* lets see if we can't reestablish the connection */ + Sleep(750); /* pause for a short period of time */ + if(!JACK_OpenDevice(wwo)) + { + ERR("unable to reconnect with jack...\n"); + } +} + + +/****************************************************************** + * JACK_OpenDevice + */ +static int JACK_OpenDevice(WINE_WAVEOUT* wwo) +{ + const char** ports; + int i; + char client_name[64]; + jack_port_t* out_port_l; + jack_port_t* out_port_r; + jack_client_t* client; + int failed = 0; + + TRACE("creating jack client and setting up callbacks\n"); + +#if JACK_CLOSE_HACK + /* see if this device is already open */ + if(wwo->client) + { + /* if this device is already in use then it is bad for us to be in here */ + if(wwo->in_use) + return 0; + + TRACE("using existing client\n"); + wwo->in_use = TRUE; + return 1; + } +#endif + + /* zero out the buffer pointer and the size of the buffer */ + wwo->sound_buffer = 0; + wwo->buffer_size = 0; + + /* try to become a client of the JACK server */ + snprintf(client_name, sizeof(client_name), "wine_jack_client %d", wwo->wDevID); + TRACE("client name '%s'\n", client_name); + if ((client = fp_jack_client_new (client_name)) == 0) + { + /* jack has problems with shutting down clients, so lets */ + /* wait a short while and try once more before we give up */ + Sleep(250); + if ((client = fp_jack_client_new (client_name)) == 0) + { + ERR("jack server not running?\n"); + return 0; + } + } + + /* tell the JACK server to call `JACK_callback()' whenever + there is work to be done. */ + fp_jack_set_process_callback (client, JACK_callback, wwo); + + /* tell the JACK server to call `JACK_bufsize()' whenever + the maximum number of frames that will be passed + to `JACK_Callback()' changes */ + fp_jack_set_buffer_size_callback (client, JACK_bufsize, wwo); + + /* tell the JACK server to call `srate()' whenever + the sample rate of the system changes. */ + fp_jack_set_sample_rate_callback (client, JACK_srate, wwo); + + /* tell the JACK server to call `jack_shutdown()' if + it ever shuts down, either entirely, or if it + just decides to stop calling us. */ + fp_jack_on_shutdown (client, JACK_shutdown, wwo); + + /* display the current sample rate. once the client is activated + (see below), you should rely on your own sample rate + callback (see above) for this value. */ + wwo->sample_rate = fp_jack_get_sample_rate(client); + TRACE("engine sample rate: %lu\n", wwo->sample_rate); + + /* create the left and right channel output ports */ + /* jack's ports are all mono so for stereo you need two */ + out_port_l = fp_jack_port_register (client, "out_l", + JACK_DEFAULT_AUDIO_TYPE, JackPortIsOutput, 0); + + out_port_r = fp_jack_port_register (client, "out_r", + JACK_DEFAULT_AUDIO_TYPE, JackPortIsOutput, 0); + + /* save away important values to the WINE_WAVEOUT struct */ + wwo->client = client; + wwo->out_port_l = out_port_l; + wwo->out_port_r = out_port_r; + +#if JACK_CLOSE_HACK + wwo->in_use = TRUE; /* mark this device as in use since it now is ;-) */ +#endif + + /* tell the JACK server that we are ready to roll */ + if (fp_jack_activate (client)) + { + ERR( "cannot activate client\n"); + return 0; + } + + /* figure out what the ports that we want to output on are */ + /* NOTE: we do this instead of using stuff like "alsa_pcm:playback_X" because */ + /* this way works if names are changed */ + ports = fp_jack_get_ports(client, NULL, NULL, JackPortIsPhysical|JackPortIsInput); + + /* display a trace of the output ports we found */ + for(i = 0; ports[i]; i++) + { + TRACE("ports[%d] = '%s'\n", i, ports[i]); + } + + if(!ports) + { + ERR("jack_get_ports() failed to find 'JackPortIsPhysical|JackPortIsInput'\n"); + } + + /* connect the ports. Note: you can't do this before + the client is activated (this may change in the future). + */ + /* we want to connect to two ports so we have stereo output ;-) */ + + if(fp_jack_connect(client, fp_jack_port_name(out_port_l), ports[0])) + { + ERR ("cannot connect to output port %d('%s')\n", 0, ports[0]); + failed = 1; + } + + if(fp_jack_connect(client, fp_jack_port_name(out_port_r), ports[1])) + { + ERR ("cannot connect to output port %d('%s')\n", 1, ports[1]); + failed = 1; + } + + free(ports); /* free the returned array of ports */ + + /* if something failed we need to shut the client down and return 0 */ + if(failed) + { + JACK_CloseDevice(wwo, TRUE); + return 0; + } + + return 1; /* return success */ +} + +/****************************************************************** + * JACK_CloseDevice + * + * Close the connection to the server cleanly. + * If close_client is TRUE we close the client for this device instead of + * just marking the device as in_use(JACK_CLOSE_HACK only) + */ +#if JACK_CLOSE_HACK +static void JACK_CloseDevice(WINE_WAVEOUT* wwo, BOOL close_client) +#else +static void JACK_CloseDevice(WINE_WAVEOUT* wwo) +#endif +{ +#if JACK_CLOSE_HACK + TRACE("wDevID: %d, close_client: %d\n", wwo->wDevID, close_client); +#else + TRACE("wDevID: %d\n", wwo->wDevID); +#endif + +#if JACK_CLOSE_HACK + if(close_client) + { +#endif + fp_jack_deactivate(wwo->client); /* supposed to help the jack_client_close() to succeed */ + fp_jack_client_close (wwo->client); + + EnterCriticalSection(&wwo->access_crst); + wwo->client = 0; /* reset client */ + HeapFree(GetProcessHeap(), 0, wwo->sound_buffer); /* free buffer memory */ + wwo->sound_buffer = 0; + wwo->buffer_size = 0; /* zero out size of the buffer */ + LeaveCriticalSection(&wwo->access_crst); +#if JACK_CLOSE_HACK + } else + { + EnterCriticalSection(&wwo->access_crst); + TRACE("setting in_use to FALSE\n"); + wwo->in_use = FALSE; + LeaveCriticalSection(&wwo->access_crst); + } +#endif +} + +/****************************************************************** + * JACK_WaveRelease + * + * + */ +LONG JACK_WaveRelease(void) +{ + int iDevice; + + TRACE("closing all open devices\n"); + + /* close all open devices */ + for(iDevice = 0; iDevice < MAX_WAVEOUTDRV; iDevice++) + { + TRACE("iDevice == %d\n", iDevice); + if(WOutDev[iDevice].client) + { +#if JACK_CLOSE_HACK + JACK_CloseDevice(&WOutDev[iDevice], TRUE); /* close the device, FORCE the client to close */ +#else + JACK_CloseDevice(&WOutDev[iDevice]); /* close the device, FORCE the client to close */ +#endif + DeleteCriticalSection(&(WOutDev[iDevice].access_crst)); /* delete the critical section */ + } + } + + TRACE("returning 1\n"); + + return 1; +} + +/****************************************************************** + * JACK_WaveInit + * + * Initialize internal structures from JACK server info + */ +LONG JACK_WaveInit(void) +{ + int i; + + TRACE("called\n"); + + /* setup function pointers */ +#define LOAD_FUNCPTR(f) if((fp_##f = wine_dlsym(jackhandle, #f, NULL, 0)) == NULL) goto sym_not_found; + LOAD_FUNCPTR(jack_activate); + LOAD_FUNCPTR(jack_connect); + LOAD_FUNCPTR(jack_client_new); + LOAD_FUNCPTR(jack_client_close); + LOAD_FUNCPTR(jack_deactivate); + LOAD_FUNCPTR(jack_set_process_callback); + LOAD_FUNCPTR(jack_set_buffer_size_callback); + LOAD_FUNCPTR(jack_set_sample_rate_callback); + LOAD_FUNCPTR(jack_on_shutdown); + LOAD_FUNCPTR(jack_get_sample_rate); + LOAD_FUNCPTR(jack_port_register); + LOAD_FUNCPTR(jack_port_get_buffer); + LOAD_FUNCPTR(jack_get_ports); + LOAD_FUNCPTR(jack_port_name); +#undef LOAD_FUNCPTR + + /* start with output device */ + + for (i = 0; i < MAX_WAVEOUTDRV; ++i) + { + WOutDev[i].client = 0; /* initialize the client to 0 */ + +#if JACK_CLOSE_HACK + WOutDev[i].in_use = FALSE; +#endif + + memset(&WOutDev[i].caps, 0, sizeof(WOutDev[i].caps)); + + /* FIXME: some programs compare this string against the content of the registry + * for MM drivers. The names have to match in order for the program to work + * (e.g. MS win9x mplayer.exe) + */ +#ifdef EMULATE_SB16 + WOutDev[i].caps.wMid = 0x0002; + WOutDev[i].caps.wPid = 0x0104; + strcpy(WOutDev[i].caps.szPname, "SB16 Wave Out"); +#else + WOutDev[i].caps.wMid = 0x00FF; /* Manufac ID */ + WOutDev[i].caps.wPid = 0x0001; /* Product ID */ + /* strcpy(WOutDev[i].caps.szPname, "OpenSoundSystem WAVOUT Driver");*/ + strcpy(WOutDev[i].caps.szPname, "CS4236/37/38"); +#endif + WOutDev[i].caps.vDriverVersion = 0x0100; + WOutDev[i].caps.dwFormats = 0x00000000; + WOutDev[i].caps.dwSupport = WAVECAPS_VOLUME; + + WOutDev[i].caps.wChannels = 2; + WOutDev[i].caps.dwSupport |= WAVECAPS_LRVOLUME; + +/* NOTE: we don't support any 8 bit modes so note that */ +/* WOutDev[i].caps.dwFormats |= WAVE_FORMAT_4M08; + WOutDev[i].caps.dwFormats |= WAVE_FORMAT_4S08; */ + WOutDev[i].caps.dwFormats |= WAVE_FORMAT_4S16; + WOutDev[i].caps.dwFormats |= WAVE_FORMAT_4M16; +/* WOutDev[i].caps.dwFormats |= WAVE_FORMAT_2M08; + WOutDev[i].caps.dwFormats |= WAVE_FORMAT_2S08; */ + WOutDev[i].caps.dwFormats |= WAVE_FORMAT_2M16; + WOutDev[i].caps.dwFormats |= WAVE_FORMAT_2S16; +/* WOutDev[i].caps.dwFormats |= WAVE_FORMAT_1M08; + WOutDev[i].caps.dwFormats |= WAVE_FORMAT_1S08;*/ + WOutDev[i].caps.dwFormats |= WAVE_FORMAT_1M16; + WOutDev[i].caps.dwFormats |= WAVE_FORMAT_1S16; + } + + /* then do input device */ + for (i = 0; i < MAX_WAVEINDRV; ++i) + { + /* TODO: we should initialize read stuff here */ + memset(&WInDev[0].caps, 0, sizeof(WInDev[0].caps)); + } + + return 1; /* return success */ + +/* error path for function pointer loading errors */ +sym_not_found: + WINE_MESSAGE( + "Wine cannot find certain functions that it needs inside the jack" + "library. To enable Wine to use the jack audio server please " + "install libjack\n"); + wine_dlclose(jackhandle, NULL, 0); + jackhandle = NULL; + return FALSE; +} + +/*======================================================================* + * Low level WAVE OUT implementation * + *======================================================================*/ + +/************************************************************************** + * wodNotifyClient [internal] + */ +static DWORD wodNotifyClient(WINE_WAVEOUT* wwo, WORD wMsg, DWORD dwParam1, DWORD dwParam2) +{ + TRACE("wMsg = 0x%04x dwParm1 = %04lX dwParam2 = %04lX\n", wMsg, dwParam1, dwParam2); + + switch (wMsg) { + case WOM_OPEN: + case WOM_CLOSE: + case WOM_DONE: + if (wwo->wFlags != DCB_NULL && + !DriverCallback(wwo->waveDesc.dwCallback, wwo->wFlags, + (HDRVR)wwo->waveDesc.hWave, wMsg, wwo->waveDesc.dwInstance, + dwParam1, dwParam2)) + { + WARN("can't notify client !\n"); + return MMSYSERR_ERROR; + } + break; + default: + FIXME("Unknown callback message %u\n", wMsg); + return MMSYSERR_INVALPARAM; + } + return MMSYSERR_NOERROR; +} + +/************************************************************************** + * wodHelper_BeginWaveHdr [internal] + * + * Makes the specified lpWaveHdr the currently playing wave header. + * If the specified wave header is a begin loop and we're not already in + * a loop, setup the loop. + */ +static void wodHelper_BeginWaveHdr(WINE_WAVEOUT* wwo, LPWAVEHDR lpWaveHdr) +{ + EnterCriticalSection(&wwo->access_crst); + + wwo->lpPlayPtr = lpWaveHdr; + + if (!lpWaveHdr) + { + LeaveCriticalSection(&wwo->access_crst); + return; + } + + if (lpWaveHdr->dwFlags & WHDR_BEGINLOOP) + { + if (wwo->lpLoopPtr) + { + WARN("Already in a loop. Discarding loop on this header (%p)\n", lpWaveHdr); + TRACE("Already in a loop. Discarding loop on this header (%p)\n", lpWaveHdr); + } else + { + TRACE("Starting loop (%ldx) with %p\n", lpWaveHdr->dwLoops, lpWaveHdr); + wwo->lpLoopPtr = lpWaveHdr; + /* Windows does not touch WAVEHDR.dwLoops, + * so we need to make an internal copy */ + wwo->dwLoops = lpWaveHdr->dwLoops; + } + } + wwo->dwPartialOffset = 0; + + LeaveCriticalSection(&wwo->access_crst); +} + + +/************************************************************************** + * wodHelper_PlayPtrNext [internal] + * + * Advance the play pointer to the next waveheader, looping if required. + */ +static LPWAVEHDR wodHelper_PlayPtrNext(WINE_WAVEOUT* wwo) +{ + LPWAVEHDR lpWaveHdr; + + EnterCriticalSection(&wwo->access_crst); + + lpWaveHdr = wwo->lpPlayPtr; + + wwo->dwPartialOffset = 0; + if ((lpWaveHdr->dwFlags & WHDR_ENDLOOP) && wwo->lpLoopPtr) + { + /* We're at the end of a loop, loop if required */ + if (--wwo->dwLoops > 0) + { + wwo->lpPlayPtr = wwo->lpLoopPtr; + } else + { + /* Handle overlapping loops correctly */ + if (wwo->lpLoopPtr != lpWaveHdr && (lpWaveHdr->dwFlags & WHDR_BEGINLOOP)) { + FIXME("Correctly handled case ? (ending loop buffer also starts a new loop)\n"); + /* shall we consider the END flag for the closing loop or for + * the opening one or for both ??? + * code assumes for closing loop only + */ + } else + { + lpWaveHdr = lpWaveHdr->lpNext; + } + wwo->lpLoopPtr = NULL; + wodHelper_BeginWaveHdr(wwo, lpWaveHdr); + } + } else + { + /* We're not in a loop. Advance to the next wave header */ + TRACE("not inside of a loop, advancing to next wave header\n"); + wodHelper_BeginWaveHdr(wwo, lpWaveHdr = lpWaveHdr->lpNext); + } + + LeaveCriticalSection(&wwo->access_crst); + + return lpWaveHdr; +} + +/* if force is TRUE then notify the client that all the headers were completed */ +static DWORD wodHelper_NotifyCompletions(WINE_WAVEOUT* wwo, BOOL force) +{ + LPWAVEHDR lpWaveHdr; + DWORD retval; + + TRACE("called\n"); + + EnterCriticalSection(&wwo->access_crst); + + /* Start from lpQueuePtr and keep notifying until: + * - we hit an unwritten wavehdr + * - we hit the beginning of a running loop + * - we hit a wavehdr which hasn't finished playing + */ + while ((lpWaveHdr = wwo->lpQueuePtr) && + (force || + (lpWaveHdr != wwo->lpPlayPtr && + lpWaveHdr != wwo->lpLoopPtr))) + { + wwo->lpQueuePtr = lpWaveHdr->lpNext; + + lpWaveHdr->dwFlags &= ~WHDR_INQUEUE; + lpWaveHdr->dwFlags |= WHDR_DONE; + TRACE("calling notify client\n"); + + wodNotifyClient(wwo, WOM_DONE, (DWORD)lpWaveHdr, 0); + } + + retval = (lpWaveHdr && lpWaveHdr != wwo->lpPlayPtr && lpWaveHdr != + wwo->lpLoopPtr) ? 0 : INFINITE; + + LeaveCriticalSection(&wwo->access_crst); + + return retval; +} + +/************************************************************************** + * wodHelper_Reset [internal] + * + * Resets current output stream. + */ +static void wodHelper_Reset(WINE_WAVEOUT* wwo, BOOL reset) +{ + EnterCriticalSection(&wwo->access_crst); + + /* updates current notify list */ + wodHelper_NotifyCompletions(wwo, FALSE); + + if (reset) + { + /* remove all wave headers and notify client that all headers were completed */ + wodHelper_NotifyCompletions(wwo, TRUE); + + wwo->lpPlayPtr = wwo->lpQueuePtr = wwo->lpLoopPtr = NULL; + wwo->state = WINE_WS_STOPPED; + wwo->dwPlayedTotal = wwo->dwWrittenTotal = wwo->bytesInJack = 0; + + wwo->dwPartialOffset = 0; /* Clear partial wavehdr */ + } else + { + if (wwo->lpLoopPtr) + { + /* complicated case, not handled yet (could imply modifying the loop counter) */ + FIXME("Pausing while in loop isn't correctly handled yet, except strange results\n"); + wwo->lpPlayPtr = wwo->lpLoopPtr; + wwo->dwPartialOffset = 0; + wwo->dwWrittenTotal = wwo->dwPlayedTotal; /* this is wrong !!! */ + } else + { + LPWAVEHDR ptr; + DWORD sz = wwo->dwPartialOffset; + + /* reset all the data as if we had written only up to lpPlayedTotal bytes */ + /* compute the max size playable from lpQueuePtr */ + for (ptr = wwo->lpQueuePtr; ptr != wwo->lpPlayPtr; ptr = ptr->lpNext) + { + sz += ptr->dwBufferLength; + } + + /* because the reset lpPlayPtr will be lpQueuePtr */ + if (wwo->dwWrittenTotal > wwo->dwPlayedTotal + sz) ERR("doh\n"); + wwo->dwPartialOffset = sz - (wwo->dwWrittenTotal - wwo->dwPlayedTotal); + wwo->dwWrittenTotal = wwo->dwPlayedTotal; + wwo->lpPlayPtr = wwo->lpQueuePtr; + } + + wwo->state = WINE_WS_PAUSED; + } + + LeaveCriticalSection(&wwo->access_crst); +} + +/************************************************************************** + * wodGetDevCaps [internal] + */ +static DWORD wodGetDevCaps(WORD wDevID, LPWAVEOUTCAPSA lpCaps, DWORD dwSize) +{ + TRACE("(%u, %p, %lu);\n", wDevID, lpCaps, dwSize); + + if (lpCaps == NULL) return MMSYSERR_NOTENABLED; + + if (wDevID >= MAX_WAVEOUTDRV) + { + TRACE("MAX_WAVOUTDRV reached !\n"); + return MMSYSERR_BADDEVICEID; + } + + memcpy(lpCaps, &WOutDev[wDevID].caps, min(dwSize, sizeof(*lpCaps))); + return MMSYSERR_NOERROR; +} + +/************************************************************************** + * wodOpen [internal] + */ +static DWORD wodOpen(WORD wDevID, LPWAVEOPENDESC lpDesc, DWORD dwFlags) +{ + WINE_WAVEOUT* wwo; + DWORD retval; + + TRACE("(%u, %p, %08lX);\n", wDevID, lpDesc, dwFlags); + if (lpDesc == NULL) + { + WARN("Invalid Parameter !\n"); + return MMSYSERR_INVALPARAM; + } + if (wDevID >= MAX_WAVEOUTDRV) { + TRACE("MAX_WAVOUTDRV reached !\n"); + return MMSYSERR_BADDEVICEID; + } + +#if JACK_CLOSE_HACK + if(WOutDev[wDevID].client && WOutDev[wDevID].in_use) +#else + if(WOutDev[wDevID].client) +#endif + { + TRACE("device %d already allocated\n", wDevID); + return MMSYSERR_ALLOCATED; + } + + /* make sure we aren't being opened in 8 bit mode */ + if(lpDesc->lpFormat->wBitsPerSample == 8) + { + TRACE("8bits per sample unsupported, returning WAVERR_BADFORMAT\n"); + return WAVERR_BADFORMAT; + } + + wwo = &WOutDev[wDevID]; + wwo->wDevID = wDevID; + + /* Set things up before we call JACK_OpenDevice because */ + /* we will start getting callbacks before JACK_OpenDevice */ + /* even returns and we want to be initialized before then */ + wwo->state = WINE_WS_STOPPED; /* start in a stopped state */ + wwo->dwPlayedTotal = 0; /* zero out these totals */ + wwo->dwWrittenTotal = 0; + wwo->bytesInJack = 0; + wwo->tickCountMS = 0; + + InitializeCriticalSection(&wwo->access_crst); /* initialize the critical section */ + + /* open up jack ports for this device */ + if (!JACK_OpenDevice(&WOutDev[wDevID])) + { + ERR("JACK_OpenDevice(%d) failed\n", wDevID); + return MMSYSERR_ERROR; /* return unspecified error */ + } + + /* only PCM format is supported so far... */ + if (lpDesc->lpFormat->wFormatTag != WAVE_FORMAT_PCM || + lpDesc->lpFormat->nChannels == 0 || + lpDesc->lpFormat->nSamplesPerSec == 0) + { + WARN("Bad format: tag=%04X nChannels=%d nSamplesPerSec=%ld !\n", + lpDesc->lpFormat->wFormatTag, lpDesc->lpFormat->nChannels, + lpDesc->lpFormat->nSamplesPerSec); + return WAVERR_BADFORMAT; + } + + if (dwFlags & WAVE_FORMAT_QUERY) + { + TRACE("Query format: tag=%04X nChannels=%d nSamplesPerSec=%ld !\n", + lpDesc->lpFormat->wFormatTag, lpDesc->lpFormat->nChannels, + lpDesc->lpFormat->nSamplesPerSec); + return MMSYSERR_NOERROR; + } + + dwFlags &= ~WAVE_DIRECTSOUND; /* direct sound not supported, ignore the flag */ + + EnterCriticalSection(&wwo->access_crst); + + wwo->wFlags = HIWORD(dwFlags & CALLBACK_TYPEMASK); + + memcpy(&wwo->waveDesc, lpDesc, sizeof(WAVEOPENDESC)); + memcpy(&wwo->format, lpDesc->lpFormat, sizeof(PCMWAVEFORMAT)); + + LeaveCriticalSection(&wwo->access_crst); + + /* display the current wave format */ + TRACE("wBitsPerSample=%u, nAvgBytesPerSec=%lu, nSamplesPerSec=%lu, nChannels=%u nBlockAlign=%u!\n", + wwo->format.wBitsPerSample, wwo->format.wf.nAvgBytesPerSec, + wwo->format.wf.nSamplesPerSec, wwo->format.wf.nChannels, + wwo->format.wf.nBlockAlign); + + /* make sure that we have the same sample rate in our audio stream */ + /* as we do in the jack server */ + if(wwo->format.wf.nSamplesPerSec != wwo->sample_rate) + { + TRACE("error: jack server sample rate is '%ld', wave sample rate is '%ld'\n", + wwo->sample_rate, wwo->format.wf.nSamplesPerSec); + +#if JACK_CLOSE_HACK + JACK_CloseDevice(wwo, FALSE); /* close this device, don't force the client to close */ +#else + JACK_CloseDevice(wwo); /* close this device */ +#endif + return WAVERR_BADFORMAT; + } + + /* check for an invalid number of bits per sample */ + if (wwo->format.wBitsPerSample == 0) + { + WARN("Resetting zeroed wBitsPerSample to 16\n"); + wwo->format.wBitsPerSample = 16 * + (wwo->format.wf.nAvgBytesPerSec / + wwo->format.wf.nSamplesPerSec) / + wwo->format.wf.nChannels; + } + + EnterCriticalSection(&wwo->access_crst); + retval = wodNotifyClient(wwo, WOM_OPEN, 0L, 0L); + LeaveCriticalSection(&wwo->access_crst); + + return retval; +} + +/************************************************************************** + * wodClose [internal] + */ +static DWORD wodClose(WORD wDevID) +{ + DWORD ret = MMSYSERR_NOERROR; + WINE_WAVEOUT* wwo; + + TRACE("(%u);\n", wDevID); + + if (wDevID >= MAX_WAVEOUTDRV || !WOutDev[wDevID].client) + { + WARN("bad device ID !\n"); + return MMSYSERR_BADDEVICEID; + } + + wwo = &WOutDev[wDevID]; + if (wwo->lpQueuePtr) + { + WARN("buffers still playing !\n"); + ret = WAVERR_STILLPLAYING; + } else + { + /* sanity check: this should not happen since the device must have been reset before */ + if (wwo->lpQueuePtr || wwo->lpPlayPtr) ERR("out of sync\n"); + + wwo->state = WINE_WS_CLOSED; /* mark the device as closed */ + +#if JACK_CLOSE_HACK + JACK_CloseDevice(wwo, FALSE); /* close the jack device, DO NOT force the client to close */ +#else + JACK_CloseDevice(wwo); /* close the jack device */ + DeleteCriticalSection(&wwo->access_crst); /* delete the critical section so we can initialize it again from wodOpen() */ +#endif + + ret = wodNotifyClient(wwo, WOM_CLOSE, 0L, 0L); + } + + return ret; +} + +/************************************************************************** + * wodWrite [internal] + * + */ +static DWORD wodWrite(WORD wDevID, LPWAVEHDR lpWaveHdr, DWORD dwSize) +{ + LPWAVEHDR*wh; + WINE_WAVEOUT *wwo; + + TRACE("(%u, %p, %08lX);\n", wDevID, lpWaveHdr, dwSize); + + /* first, do the sanity checks... */ + if (wDevID >= MAX_WAVEOUTDRV || !WOutDev[wDevID].client) + { + WARN("bad dev ID !\n"); + return MMSYSERR_BADDEVICEID; + } + + wwo = &WOutDev[wDevID]; + + if (lpWaveHdr->lpData == NULL || !(lpWaveHdr->dwFlags & WHDR_PREPARED)) + { + TRACE("unprepared\n"); + return WAVERR_UNPREPARED; + } + + if (lpWaveHdr->dwFlags & WHDR_INQUEUE) + { + TRACE("still playing\n"); + return WAVERR_STILLPLAYING; + } + + lpWaveHdr->dwFlags &= ~WHDR_DONE; + lpWaveHdr->dwFlags |= WHDR_INQUEUE; + lpWaveHdr->lpNext = 0; + + EnterCriticalSection(&wwo->access_crst); + + /* insert buffer at the end of queue */ + for (wh = &(wwo->lpQueuePtr); *wh; wh = &((*wh)->lpNext)); + *wh = lpWaveHdr; + + LeaveCriticalSection(&wwo->access_crst); + + EnterCriticalSection(&wwo->access_crst); + if (!wwo->lpPlayPtr) + wodHelper_BeginWaveHdr(wwo,lpWaveHdr); + if (wwo->state == WINE_WS_STOPPED) + wwo->state = WINE_WS_PLAYING; + LeaveCriticalSection(&wwo->access_crst); + + return MMSYSERR_NOERROR; +} + +/************************************************************************** + * wodPrepare [internal] + */ +static DWORD wodPrepare(WORD wDevID, LPWAVEHDR lpWaveHdr, DWORD dwSize) +{ + TRACE("(%u, %p, %08lX);\n", wDevID, lpWaveHdr, dwSize); + + if (wDevID >= MAX_WAVEOUTDRV) + { + WARN("bad device ID !\n"); + return MMSYSERR_BADDEVICEID; + } + + if (lpWaveHdr->dwFlags & WHDR_INQUEUE) + return WAVERR_STILLPLAYING; + + lpWaveHdr->dwFlags |= WHDR_PREPARED; + lpWaveHdr->dwFlags &= ~WHDR_DONE; + return MMSYSERR_NOERROR; +} + +/************************************************************************** + * wodUnprepare [internal] + */ +static DWORD wodUnprepare(WORD wDevID, LPWAVEHDR lpWaveHdr, DWORD dwSize) +{ + TRACE("(%u, %p, %08lX);\n", wDevID, lpWaveHdr, dwSize); + + if (wDevID >= MAX_WAVEOUTDRV) + { + WARN("bad device ID !\n"); + return MMSYSERR_BADDEVICEID; + } + + if (lpWaveHdr->dwFlags & WHDR_INQUEUE) + return WAVERR_STILLPLAYING; + + lpWaveHdr->dwFlags &= ~WHDR_PREPARED; + lpWaveHdr->dwFlags |= WHDR_DONE; + + return MMSYSERR_NOERROR; +} + +/************************************************************************** + * wodPause [internal] + */ +static DWORD wodPause(WORD wDevID) +{ + TRACE("(%u);!\n", wDevID); + + if (wDevID >= MAX_WAVEOUTDRV || !WOutDev[wDevID].client) + { + WARN("bad device ID !\n"); + return MMSYSERR_BADDEVICEID; + } + + TRACE("[3-PAUSING]\n"); + + EnterCriticalSection(&(WOutDev[wDevID].access_crst)); + wodHelper_Reset(&WOutDev[wDevID], FALSE); + LeaveCriticalSection(&(WOutDev[wDevID].access_crst)); + + return MMSYSERR_NOERROR; +} + +/************************************************************************** + * wodRestart [internal] + */ +static DWORD wodRestart(WORD wDevID) +{ + TRACE("(%u);\n", wDevID); + + if (wDevID >= MAX_WAVEOUTDRV || !WOutDev[wDevID].client) + { + WARN("bad device ID !\n"); + return MMSYSERR_BADDEVICEID; + } + + if (WOutDev[wDevID].state == WINE_WS_PAUSED) + { + EnterCriticalSection(&(WOutDev[wDevID].access_crst)); + WOutDev[wDevID].state = WINE_WS_PLAYING; + LeaveCriticalSection(&(WOutDev[wDevID].access_crst)); + } + + return MMSYSERR_NOERROR; +} + +/************************************************************************** + * wodReset [internal] + */ +static DWORD wodReset(WORD wDevID) +{ + TRACE("(%u);\n", wDevID); + + if (wDevID >= MAX_WAVEOUTDRV || !WOutDev[wDevID].client) + { + WARN("bad device ID !\n"); + return MMSYSERR_BADDEVICEID; + } + + EnterCriticalSection(&(WOutDev[wDevID].access_crst)); + wodHelper_Reset(&WOutDev[wDevID], TRUE); + LeaveCriticalSection(&(WOutDev[wDevID].access_crst)); + + return MMSYSERR_NOERROR; +} + +/************************************************************************** + * wodGetPosition [internal] + */ +static DWORD wodGetPosition(WORD wDevID, LPMMTIME lpTime, DWORD uSize) +{ + int time; + DWORD val; + WINE_WAVEOUT* wwo; + DWORD elapsedMS; + + TRACE("(%u, %p, %lu);\n", wDevID, lpTime, uSize); + + if (wDevID >= MAX_WAVEOUTDRV || !WOutDev[wDevID].client) + { + WARN("bad device ID !\n"); + return MMSYSERR_BADDEVICEID; + } + + /* if null pointer to time structure return error */ + if (lpTime == NULL) return MMSYSERR_INVALPARAM; + + wwo = &WOutDev[wDevID]; + + EnterCriticalSection(&(WOutDev[wDevID].access_crst)); + val = wwo->dwPlayedTotal; + elapsedMS = GetTickCount() - wwo->tickCountMS; + LeaveCriticalSection(&(WOutDev[wDevID].access_crst)); + + /* account for the bytes played since the last JACK_Callback() */ + val+=((elapsedMS * wwo->format.wf.nAvgBytesPerSec) / 1000); + + TRACE("wType=%04X wBitsPerSample=%u nSamplesPerSec=%lu nChannels=%u nAvgBytesPerSec=%lu\n", + lpTime->wType, wwo->format.wBitsPerSample, + wwo->format.wf.nSamplesPerSec, wwo->format.wf.nChannels, + wwo->format.wf.nAvgBytesPerSec); + TRACE("dwPlayedTotal=%lu\n", val); + + switch (lpTime->wType) { + case TIME_BYTES: + lpTime->u.cb = val; + TRACE("TIME_BYTES=%lu\n", lpTime->u.cb); + break; + case TIME_SAMPLES: + lpTime->u.sample = val * 8 / wwo->format.wBitsPerSample /wwo->format.wf.nChannels; + TRACE("TIME_SAMPLES=%lu\n", lpTime->u.sample); + break; + case TIME_SMPTE: + time = val / (wwo->format.wf.nAvgBytesPerSec / 1000); + lpTime->u.smpte.hour = time / 108000; + time -= lpTime->u.smpte.hour * 108000; + lpTime->u.smpte.min = time / 1800; + time -= lpTime->u.smpte.min * 1800; + lpTime->u.smpte.sec = time / 30; + time -= lpTime->u.smpte.sec * 30; + lpTime->u.smpte.frame = time; + lpTime->u.smpte.fps = 30; + TRACE("TIME_SMPTE=%02u:%02u:%02u:%02u\n", + lpTime->u.smpte.hour, lpTime->u.smpte.min, + lpTime->u.smpte.sec, lpTime->u.smpte.frame); + break; + default: + FIXME("Format %d not supported ! use TIME_MS !\n", lpTime->wType); + lpTime->wType = TIME_MS; + case TIME_MS: + lpTime->u.ms = val / (wwo->format.wf.nAvgBytesPerSec / 1000); + TRACE("TIME_MS=%lu\n", lpTime->u.ms); + break; + } + return MMSYSERR_NOERROR; +} + +/************************************************************************** + * wodBreakLoop [internal] + */ +static DWORD wodBreakLoop(WORD wDevID) +{ + TRACE("(%u);\n", wDevID); + + if (wDevID >= MAX_WAVEOUTDRV || !WOutDev[wDevID].client) + { + WARN("bad device ID !\n"); + return MMSYSERR_BADDEVICEID; + } + + EnterCriticalSection(&(WOutDev[wDevID].access_crst)); + + if (WOutDev[wDevID].state == WINE_WS_PLAYING && WOutDev[wDevID].lpLoopPtr != NULL) + { + /* ensure exit at end of current loop */ + WOutDev[wDevID].dwLoops = 1; + } + + LeaveCriticalSection(&(WOutDev[wDevID].access_crst)); + + return MMSYSERR_NOERROR; +} + +/************************************************************************** + * wodGetVolume [internal] + */ +static DWORD wodGetVolume(WORD wDevID, LPDWORD lpdwVol) +{ + DWORD left, right; + + left = WOutDev[wDevID].volume_left; + right = WOutDev[wDevID].volume_right; + + TRACE("(%u, %p);\n", wDevID, lpdwVol); + + *lpdwVol = ((left * 0xFFFFl) / 100) + (((right * 0xFFFFl) / 100) << + 16); + + return MMSYSERR_NOERROR; +} + +/************************************************************************** + * wodSetVolume [internal] + */ +static DWORD wodSetVolume(WORD wDevID, DWORD dwParam) +{ + DWORD left, right; + + left = (LOWORD(dwParam) * 100) / 0xFFFFl; + right = (HIWORD(dwParam) * 100) / 0xFFFFl; + + TRACE("(%u, %08lX);\n", wDevID, dwParam); + + EnterCriticalSection(&(WOutDev[wDevID].access_crst)); + + WOutDev[wDevID].volume_left = left; + WOutDev[wDevID].volume_right = right; + + LeaveCriticalSection(&(WOutDev[wDevID].access_crst)); + + return MMSYSERR_NOERROR; +} + +/************************************************************************** + * wodGetNumDevs [internal] + */ +static DWORD wodGetNumDevs(void) +{ + return MAX_WAVEOUTDRV; +} + +/************************************************************************** + * wodMessage (WINEJACK.7) + */ +DWORD WINAPI JACK_wodMessage(UINT wDevID, UINT wMsg, DWORD dwUser, + DWORD dwParam1, DWORD dwParam2) +{ + TRACE("(%u, %04X, %08lX, %08lX, %08lX);\n", + wDevID, wMsg, dwUser, dwParam1, dwParam2); + + switch (wMsg) { + case DRVM_INIT: + TRACE("DRVM_INIT\n"); + return JACK_WaveInit(); + case DRVM_EXIT: + TRACE("DRVM_EXIT\n"); + return JACK_WaveRelease(); + case DRVM_ENABLE: + /* FIXME: Pretend this is supported */ + TRACE("DRVM_ENABLE\n"); + return 0; + case DRVM_DISABLE: + /* FIXME: Pretend this is supported */ + TRACE("DRVM_DISABLE\n"); + return 0; + case WODM_OPEN: return wodOpen(wDevID, (LPWAVEOPENDESC)dwParam1, dwParam2); + case WODM_CLOSE: return wodClose(wDevID); + case WODM_WRITE: return wodWrite(wDevID, (LPWAVEHDR)dwParam1, dwParam2); + case WODM_PAUSE: return wodPause(wDevID); + case WODM_GETPOS: return wodGetPosition(wDevID, (LPMMTIME)dwParam1, dwParam2); + case WODM_BREAKLOOP: return wodBreakLoop(wDevID); + case WODM_PREPARE: return wodPrepare(wDevID, (LPWAVEHDR)dwParam1, dwParam2); + case WODM_UNPREPARE: return wodUnprepare(wDevID, (LPWAVEHDR)dwParam1, dwParam2); + case WODM_GETDEVCAPS: return wodGetDevCaps(wDevID, (LPWAVEOUTCAPSA)dwParam1, dwParam2); + case WODM_GETNUMDEVS: return wodGetNumDevs(); + case WODM_GETPITCH: return MMSYSERR_NOTSUPPORTED; + case WODM_SETPITCH: return MMSYSERR_NOTSUPPORTED; + case WODM_GETPLAYBACKRATE: return MMSYSERR_NOTSUPPORTED; + case WODM_SETPLAYBACKRATE: return MMSYSERR_NOTSUPPORTED; + case WODM_GETVOLUME: return wodGetVolume(wDevID, (LPDWORD)dwParam1); + case WODM_SETVOLUME: return wodSetVolume(wDevID, dwParam1); + case WODM_RESTART: return wodRestart(wDevID); + case WODM_RESET: return wodReset(wDevID); + + case DRV_QUERYDSOUNDIFACE: return wodDsCreate(wDevID, (PIDSDRIVER*)dwParam1); + default: + FIXME("unknown message %d!\n", wMsg); + } + return MMSYSERR_NOTSUPPORTED; +} + +/*======================================================================* + * Low level DSOUND implementation * + *======================================================================*/ + +typedef struct IDsDriverImpl IDsDriverImpl; +typedef struct IDsDriverBufferImpl IDsDriverBufferImpl; + +struct IDsDriverImpl +{ + /* IUnknown fields */ + ICOM_VFIELD(IDsDriver); + DWORD ref; + /* IDsDriverImpl fields */ + UINT wDevID; + IDsDriverBufferImpl*primary; +}; + +struct IDsDriverBufferImpl +{ + /* IUnknown fields */ + ICOM_VFIELD(IDsDriverBuffer); + DWORD ref; + /* IDsDriverBufferImpl fields */ + IDsDriverImpl* drv; + DWORD buflen; +}; + +static DWORD wodDsCreate(UINT wDevID, PIDSDRIVER* drv) +{ + /* we can't perform memory mapping as we don't have a file stream + interface with jack like we do with oss */ + MESSAGE("This sound card's driver does not support direct access\n"); + MESSAGE("The (slower) DirectSound HEL mode will be used instead.\n"); + return MMSYSERR_NOTSUPPORTED; +} + +/*======================================================================* + * Low level WAVE IN implementation * + *======================================================================*/ + +/************************************************************************** + * widMessage (WINEJACK.6) + */ +DWORD WINAPI JACK_widMessage(WORD wDevID, WORD wMsg, DWORD dwUser, + DWORD dwParam1, DWORD dwParam2) +{ + TRACE("(%u, %04X, %08lX, %08lX, %08lX);\n", + wDevID, wMsg, dwUser, dwParam1, dwParam2); + + return MMSYSERR_NOTSUPPORTED; +} + +#else /* !HAVE_JACK */ + +/************************************************************************** + * wodMessage (WINEJACK.7) + */ +DWORD WINAPI JACK_wodMessage(WORD wDevID, WORD wMsg, DWORD dwUser, + DWORD dwParam1, DWORD dwParam2) +{ + FIXME("(%u, %04X, %08lX, %08lX, %08lX):jack support not compiled into wine\n", wDevID, wMsg, dwUser, dwParam1, dwParam2); + return MMSYSERR_NOTENABLED; +} + +#endif /* HAVE_JACK */ --- /dev/null 2002-07-06 21:24:44.000000000 -0500 +++ dlls/winmm/winejack/jack.c 2002-12-08 11:43:30.000000000 -0500 @@ -0,0 +1,184 @@ +/* -*- tab-width: 8; c-basic-offset: 4 -*- */ +/* + * Wine Driver for jack Sound Server + * http://jackit.sourceforge.net + * + * Copyright 2002 Chris Morgan<cmorgan@alum.wpi.edu> + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 2.1 of the License, or (at your option) any later version. + * + * This library 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 + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 2.1 of the License, or (at your option) any later version. + * + * This library 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 + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ + +#include "config.h" +#include "wine/port.h" +#include "wine/library.h" + +#include "windef.h" +#include "winbase.h" +#include "wingdi.h" +#include "winuser.h" +#include "mmddk.h" +#include "jack.h" + +#ifdef HAVE_JACK +static int jack = 0; + +/* set this to zero or one to enable or disable tracing in here */ +#define TRACING 0 + +#if TRACING +#define PRINTF(...) printf(...) +#else +#define PRINTF(...) +#endif + +void *jackhandle = NULL; + +/************************************************************************** + * JACK_drvLoad [internal] + */ +static DWORD JACK_drvLoad(void) +{ + PRINTF("JACK_drvLoad()\n"); + + /* dynamically load the jack library if not already loaded */ + if(!jackhandle) + { + jackhandle = wine_dlopen(JACK_SONAME, RTLD_NOW, NULL, 0); + PRINTF("JACK_drvLoad: JACK_SONAME == %s\n", JACK_SONAME); + PRINTF("JACK_drvLoad: jackhandle == 0x%x\n", jackhandle); + if(!jackhandle) + { + PRINTF("JACK_drvLoad: error loading the jack library %s, please install this library to use jack\n", JACK_SONAME); + jackhandle = (void*)-1; + return 0; + } + } + + return 1; +} + +/************************************************************************** + * JACK_drvFree [internal] + */ +/* unload the jack library on driver free */ +static DWORD JACK_drvFree(void) +{ + PRINTF("JACK_drvFree()\n"); + + if(jackhandle && (jackhandle != (void*)-1)) + { + PRINTF("JACK_drvFree: calling wine_dlclose() on jackhandle\n"); + wine_dlclose(jackhandle, NULL, 0); + jackhandle = NULL; + } + + return 1; +} + +/************************************************************************** + * JACK_drvOpen [internal] + */ +static DWORD JACK_drvOpen(LPSTR str) +{ + /* if we were unable to load the jack library then fail the */ + /* driver open */ + if(!jackhandle) + { + PRINTF("JACK_drvOpen: unable to open the jack library, returning 0\n"); + return 0; + } + + if (jack) + { + PRINTF("JACK_drvOpen: jack != 0 (already open), returning 0\n"); + return 0; + } + + /* I know, this is ugly, but who cares... */ + PRINTF("JACK_drvOpen: opened jack(set jack = 1), returning 1\n"); + jack = 1; + return 1; +} + +/************************************************************************** + * JACK_drvClose [internal] + */ +static DWORD JACK_drvClose(DWORD dwDevID) +{ + if (jack) + { + PRINTF("JACK_drvClose: jack is nonzero, setting jack to 0 and returning 1\n"); + jack = 0; + return 1; + } + + PRINTF("JACK_drvClose: jack is zero(closed), returning 0\n"); + return 0; +} +#endif /* #ifdef HAVE_JACK */ + + +/************************************************************************** + * DriverProc (WINEJACK.1) + */ +LONG CALLBACK JACK_DriverProc(DWORD dwDevID, HDRVR hDriv, DWORD wMsg, + DWORD dwParam1, DWORD dwParam2) +{ +/* EPP TRACE("(%08lX, %04X, %08lX, %08lX, %08lX)\n", */ +/* EPP dwDevID, hDriv, wMsg, dwParam1, dwParam2); */ + + switch(wMsg) { +#ifdef HAVE_JACK + case DRV_LOAD: + PRINTF("JACK_DriverProc: DRV_LOAD\n"); + return JACK_drvLoad(); + case DRV_FREE: + PRINTF("JACK_DriverProc: DRV_FREE\n"); + return JACK_drvFree(); + case DRV_OPEN: + PRINTF("JACK_DriverProc: DRV_OPEN\n"); + return JACK_drvOpen((LPSTR)dwParam1); + case DRV_CLOSE: + PRINTF("JACK_DriverProc: DRV_CLOSE\n"); + return JACK_drvClose(dwDevID); + case DRV_ENABLE: + PRINTF("JACK_DriverProc: DRV_ENABLE\n"); + return 1; + case DRV_DISABLE: + PRINTF("JACK_DriverProc: DRV_DISABLE\n"); + return 1; + case DRV_QUERYCONFIGURE: return 1; + case DRV_CONFIGURE: MessageBoxA(0, "jack audio driver!", "jack driver", MB_OK); return 1; + case DRV_INSTALL: + PRINTF("JACK_DriverProc: DRV_INSTALL\n"); + return DRVCNF_RESTART; + case DRV_REMOVE: + PRINTF("JACK_DriverProc: DRV_REMOVE\n"); + return DRVCNF_RESTART; +#endif + default: + return DefDriverProc(dwDevID, hDriv, wMsg, dwParam1, dwParam2); + } +} + + --- /dev/null 2002-07-06 21:24:44.000000000 -0500 +++ dlls/winmm/winejack/jack.h 2002-11-25 21:48:10.000000000 -0500 @@ -0,0 +1,40 @@ +/* Definition for jack driver : wine multimedia system + * + * Copyright 2002 Chris Morgan<cmorgan@alum.wpi.edu> + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 2.1 of the License, or (at your option) any later version. + * + * This library 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 + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ + +#ifndef __WINE_JACK_H +#define __WINE_JACK_H + +#ifndef __WINE_CONFIG_H +# error You must include config.h to use this header +#endif + +#ifdef HAVE_SYS_ERRNO_H +#include <sys/errno.h> +#endif + +#ifndef JACK_SONAME +#define JACK_SONAME "libjack.so" +#endif + +extern void *jackhandle; /* pointer to the libjack library */ + +extern LONG JACK_WaveInit(void); +extern LONG JACK_WaveRelease(void); + +#endif /* __WINE_JACK_H */ --- /dev/null 2002-07-06 21:24:44.000000000 -0500 +++ dlls/winmm/winejack/winejack.drv.spec 2002-09-12 16:19:58.000000000 -0500 @@ -0,0 +1,2 @@ +@ stdcall DriverProc(long long long long long) JACK_DriverProc +@ stdcall wodMessage(long long long long long) JACK_wodMessage