You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
216 lines
6.7 KiB
216 lines
6.7 KiB
/* Copyright (c) 2002, John Edwards |
|
|
|
Redistribution and use in source and binary forms, with or without |
|
modification, are permitted provided that the following conditions |
|
are met: |
|
|
|
- Redistributions of source code must retain the above copyright |
|
notice, this list of conditions and the following disclaimer. |
|
|
|
- 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. |
|
|
|
- Neither the name of the Xiph.org Foundation nor the names of its |
|
contributors may be used to endorse or promote products derived from |
|
this software without specific prior written permission. |
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS 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 FOUNDATION 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. |
|
*/ |
|
|
|
/* Set TABS = 4 */ |
|
/******************************************************************** |
|
|
|
function: To provide playback of 16 bit PCM wave data in Win32 |
|
environments from decoded compressed files. |
|
|
|
********************************************************************/ |
|
|
|
#if defined WIN32 || defined _WIN32 |
|
|
|
#include <string.h> |
|
#include <errno.h> |
|
#include "wave_out.h" |
|
|
|
#define MAXWAVESIZE 4294967040LU |
|
#define MAX_WAVEBLOCKS 32 |
|
|
|
// This is modified for USE_WIN_AUDIO - ONLY 2002-02-27 |
|
|
|
|
|
static CRITICAL_SECTION cs; |
|
static HWAVEOUT dev = NULL; |
|
static int ScheduledBlocks = 0; |
|
static int PlayedWaveHeadersCount = 0; // free index |
|
static WAVEHDR* PlayedWaveHeaders [MAX_WAVEBLOCKS]; |
|
|
|
static int |
|
Box ( const char* msg ) |
|
{ |
|
MessageBox ( NULL, msg, " "VERSION_STRING": Error Message . . .", MB_OK | MB_ICONEXCLAMATION ); |
|
return -1; |
|
} |
|
|
|
|
|
/* |
|
* This function registers already played WAVE chunks. Freeing is done by free_memory(), |
|
*/ |
|
|
|
static void CALLBACK |
|
wave_callback ( HWAVE hWave, UINT uMsg, DWORD dwInstance, DWORD dwParam1, DWORD dwParam2 ) |
|
{ |
|
if ( uMsg == WOM_DONE ) { |
|
EnterCriticalSection ( &cs ); |
|
PlayedWaveHeaders [PlayedWaveHeadersCount++] = (WAVEHDR*) dwParam1; |
|
LeaveCriticalSection ( &cs ); |
|
} |
|
} |
|
|
|
|
|
static void |
|
free_memory ( void ) |
|
{ |
|
WAVEHDR* wh; |
|
HGLOBAL hg; |
|
|
|
EnterCriticalSection ( &cs ); |
|
wh = PlayedWaveHeaders [--PlayedWaveHeadersCount]; |
|
ScheduledBlocks--; // decrease the number of USED blocks |
|
LeaveCriticalSection ( &cs ); |
|
|
|
waveOutUnprepareHeader ( dev, wh, sizeof (WAVEHDR) ); |
|
|
|
hg = GlobalHandle ( wh -> lpData ); // Deallocate the buffer memory |
|
GlobalUnlock (hg); |
|
GlobalFree (hg); |
|
|
|
hg = GlobalHandle ( wh ); // Deallocate the header memory |
|
GlobalUnlock (hg); |
|
GlobalFree (hg); |
|
} |
|
|
|
|
|
Int |
|
Set_WIN_Params ( FILE_T dummyFile , |
|
Ldouble SampleFreq, |
|
Uint BitsPerSample, |
|
Uint Channels ) |
|
{ |
|
WAVEFORMATEX outFormat; |
|
UINT deviceID = WAVE_MAPPER; |
|
|
|
(void) dummyFile; |
|
|
|
if ( waveOutGetNumDevs () == 0 ) |
|
return Box ( "No audio device present." ); |
|
|
|
outFormat.wFormatTag = WAVE_FORMAT_PCM; |
|
outFormat.wBitsPerSample = BitsPerSample; |
|
outFormat.nChannels = Channels; |
|
outFormat.nSamplesPerSec = (unsigned long)(SampleFreq + 0.5); |
|
outFormat.nBlockAlign = (outFormat.wBitsPerSample + 7) / 8 * outFormat.nChannels; |
|
outFormat.nAvgBytesPerSec = outFormat.nSamplesPerSec * outFormat.nBlockAlign; |
|
|
|
switch ( waveOutOpen ( &dev, deviceID, &outFormat, (DWORD)wave_callback, 0, CALLBACK_FUNCTION ) ) |
|
{ |
|
case MMSYSERR_ALLOCATED: return Box ( "Device is already open." ); |
|
case MMSYSERR_BADDEVICEID: return Box ( "The specified device is out of range." ); |
|
case MMSYSERR_NODRIVER: return Box ( "There is no audio driver in this system." ); |
|
case MMSYSERR_NOMEM: return Box ( "Unable to allocate sound memory." ); |
|
case WAVERR_BADFORMAT: return Box ( "This audio format is not supported." ); |
|
case WAVERR_SYNC: return Box ( "The device is synchronous." ); |
|
default: return Box ( "Unknown media error." ); |
|
case MMSYSERR_NOERROR: break; |
|
} |
|
|
|
waveOutReset ( dev ); |
|
InitializeCriticalSection ( &cs ); |
|
SetPriorityClass ( GetCurrentProcess (), HIGH_PRIORITY_CLASS ); |
|
return 0; |
|
} |
|
|
|
|
|
int |
|
WIN_Play_Samples ( const void* data, size_t len ) |
|
{ |
|
HGLOBAL hg; |
|
HGLOBAL hg2; |
|
LPWAVEHDR wh; |
|
void* allocptr; |
|
|
|
do { |
|
while ( PlayedWaveHeadersCount > 0 ) // free used blocks ... |
|
free_memory (); |
|
|
|
if ( ScheduledBlocks < sizeof(PlayedWaveHeaders)/sizeof(*PlayedWaveHeaders) ) // wait for a free block ... |
|
break; |
|
Sleep (26); |
|
} while (1); |
|
|
|
if ( (hg2 = GlobalAlloc ( GMEM_MOVEABLE, len )) == NULL ) // allocate some memory for a copy of the buffer |
|
return Box ( "GlobalAlloc failed." ); |
|
|
|
allocptr = GlobalLock (hg2); |
|
CopyMemory ( allocptr, data, len ); // Here we can call any modification output functions we want.... |
|
|
|
if ( (hg = GlobalAlloc (GMEM_MOVEABLE | GMEM_ZEROINIT, sizeof (WAVEHDR))) == NULL ) // now make a header and WRITE IT! |
|
return -1; |
|
|
|
wh = GlobalLock (hg); |
|
wh -> dwBufferLength = len; |
|
wh -> lpData = allocptr; |
|
|
|
if ( waveOutPrepareHeader ( dev, wh, sizeof (WAVEHDR)) != MMSYSERR_NOERROR ) { |
|
GlobalUnlock (hg); |
|
GlobalFree (hg); |
|
return -1; |
|
} |
|
|
|
if ( waveOutWrite ( dev, wh, sizeof (WAVEHDR)) != MMSYSERR_NOERROR ) { |
|
GlobalUnlock (hg); |
|
GlobalFree (hg); |
|
return -1; |
|
} |
|
|
|
EnterCriticalSection ( &cs ); |
|
ScheduledBlocks++; |
|
LeaveCriticalSection ( &cs ); |
|
|
|
return len; |
|
} |
|
|
|
|
|
int |
|
WIN_Audio_close ( void ) |
|
{ |
|
if ( dev != NULL ) { |
|
|
|
while ( ScheduledBlocks > 0 ) { |
|
Sleep (ScheduledBlocks); |
|
while ( PlayedWaveHeadersCount > 0 ) // free used blocks ... |
|
free_memory (); |
|
} |
|
|
|
waveOutReset (dev); // reset the device |
|
waveOutClose (dev); // close the device |
|
dev = NULL; |
|
} |
|
|
|
DeleteCriticalSection ( &cs ); |
|
ScheduledBlocks = 0; |
|
return 0; |
|
} |
|
|
|
#endif |
|
|
|
/* end of wave_out.c */
|
|
|