2009-07-27 05:47:50 +00:00
|
|
|
/*
|
|
|
|
Copyright (C) 1994-1995 Apogee Software, Ltd.
|
2015-07-08 03:33:56 +00:00
|
|
|
Copyright (C) 2015 EDuke32 developers
|
|
|
|
Copyright (C) 2015 Voidpoint, LLC
|
2009-07-27 05:47:50 +00:00
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or
|
|
|
|
modify it under the terms of the GNU General Public License
|
|
|
|
as published by the Free Software Foundation; either version 2
|
|
|
|
of the License, or (at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
|
|
|
|
|
|
See the GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program; if not, write to the Free Software
|
2014-07-20 08:55:56 +00:00
|
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
2009-07-27 05:47:50 +00:00
|
|
|
|
|
|
|
*/
|
|
|
|
/**********************************************************************
|
|
|
|
module: FX_MAN.C
|
|
|
|
|
|
|
|
author: James R. Dose
|
|
|
|
date: March 17, 1994
|
|
|
|
|
|
|
|
Device independant sound effect routines.
|
|
|
|
|
|
|
|
(c) Copyright 1994 James R. Dose. All Rights Reserved.
|
|
|
|
**********************************************************************/
|
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
#include "compat.h"
|
2009-07-27 05:47:50 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
2009-07-27 10:46:42 +00:00
|
|
|
#include <string.h>
|
2009-07-27 05:47:50 +00:00
|
|
|
#include "drivers.h"
|
|
|
|
#include "multivoc.h"
|
|
|
|
#include "fx_man.h"
|
|
|
|
|
|
|
|
int32_t FX_ErrorCode = FX_Ok;
|
|
|
|
int32_t FX_Installed = FALSE;
|
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
#define FX_SetErrorCode(status) FX_ErrorCode = (status);
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
const char *FX_ErrorString(int32_t ErrorNumber)
|
2009-07-27 05:47:50 +00:00
|
|
|
{
|
|
|
|
const char *ErrorString;
|
|
|
|
|
|
|
|
switch (ErrorNumber)
|
|
|
|
{
|
2015-07-08 03:33:56 +00:00
|
|
|
case FX_Warning:
|
|
|
|
case FX_Error: ErrorString = FX_ErrorString(FX_ErrorCode); break;
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
case FX_Ok: ErrorString = "Fx ok."; break;
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
case FX_InvalidCard: ErrorString = "Invalid Sound Fx device."; break;
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
case FX_MultiVocError: ErrorString = MV_ErrorString(MV_Error); break;
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
default: ErrorString = "Unknown Fx error code."; break;
|
|
|
|
}
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
return ErrorString;
|
|
|
|
}
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
static inline int32_t FX_CheckMVErr(int32_t status)
|
2009-07-27 05:47:50 +00:00
|
|
|
{
|
2015-07-08 03:33:56 +00:00
|
|
|
if (status != MV_Ok)
|
|
|
|
{
|
|
|
|
FX_SetErrorCode(FX_MultiVocError);
|
|
|
|
status = FX_Warning;
|
|
|
|
}
|
|
|
|
|
|
|
|
return status;
|
|
|
|
}
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:34:09 +00:00
|
|
|
int32_t FX_Init(int32_t SoundCard, int32_t numvoices, int32_t numchannels, unsigned mixrate, void *initdata)
|
2015-07-08 03:33:56 +00:00
|
|
|
{
|
2009-07-27 05:47:50 +00:00
|
|
|
if (FX_Installed)
|
|
|
|
FX_Shutdown();
|
|
|
|
|
|
|
|
if (SoundCard == ASS_AutoDetect)
|
|
|
|
{
|
2016-06-15 07:08:45 +00:00
|
|
|
#if defined MIXERTYPEWIN
|
2009-07-27 05:47:50 +00:00
|
|
|
SoundCard = ASS_DirectSound;
|
2016-06-15 07:08:45 +00:00
|
|
|
#elif defined MIXERTYPESDL
|
2009-07-27 05:47:50 +00:00
|
|
|
SoundCard = ASS_SDL;
|
|
|
|
#else
|
2012-12-29 05:22:55 +00:00
|
|
|
#warning No sound driver selected!
|
2009-07-27 05:47:50 +00:00
|
|
|
SoundCard = ASS_NoSound;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
if (SoundCard < 0 || SoundCard >= ASS_NumSoundCards)
|
|
|
|
{
|
|
|
|
FX_SetErrorCode(FX_InvalidCard);
|
2015-07-08 03:33:56 +00:00
|
|
|
return FX_Error;
|
2009-07-27 05:47:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (SoundDriver_IsSupported(SoundCard) == 0)
|
|
|
|
{
|
|
|
|
// unsupported cards fall back to no sound
|
|
|
|
SoundCard = ASS_NoSound;
|
|
|
|
}
|
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
int status = FX_Ok;
|
|
|
|
|
2015-07-08 03:34:09 +00:00
|
|
|
if (MV_Init(SoundCard, mixrate, numvoices, numchannels, initdata) != MV_Ok)
|
2009-07-27 05:47:50 +00:00
|
|
|
{
|
|
|
|
FX_SetErrorCode(FX_MultiVocError);
|
|
|
|
status = FX_Error;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (status == FX_Ok)
|
|
|
|
FX_Installed = TRUE;
|
|
|
|
|
2012-12-29 10:59:21 +00:00
|
|
|
return status;
|
2009-07-27 05:47:50 +00:00
|
|
|
}
|
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
int32_t FX_Shutdown(void)
|
2009-07-27 05:47:50 +00:00
|
|
|
{
|
|
|
|
if (!FX_Installed)
|
2012-12-29 10:59:21 +00:00
|
|
|
return FX_Ok;
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
int status = MV_Shutdown();
|
|
|
|
|
2009-07-27 05:47:50 +00:00
|
|
|
if (status != MV_Ok)
|
|
|
|
{
|
|
|
|
FX_SetErrorCode(FX_MultiVocError);
|
|
|
|
status = FX_Error;
|
|
|
|
}
|
|
|
|
|
|
|
|
FX_Installed = FALSE;
|
|
|
|
|
2012-12-29 10:59:21 +00:00
|
|
|
return status;
|
2009-07-27 05:47:50 +00:00
|
|
|
}
|
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
void FX_SetCallBack(void (*function)(uint32_t)) { MV_SetCallBack(function); }
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
void FX_SetVolume(int32_t volume) { MV_SetVolume(volume); }
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
int32_t FX_GetVolume(void) { return MV_GetVolume(); }
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
void FX_SetReverseStereo(int32_t setting) { MV_SetReverseStereo(setting); }
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
int32_t FX_GetReverseStereo(void) { return MV_GetReverseStereo(); }
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
void FX_SetReverb(int32_t reverb) { MV_SetReverb(reverb); }
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
int32_t FX_GetMaxReverbDelay(void) { return MV_GetMaxReverbDelay(); }
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
int32_t FX_GetReverbDelay(void) { return MV_GetReverbDelay(); }
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
void FX_SetReverbDelay(int32_t delay) { MV_SetReverbDelay(delay); }
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
int32_t FX_VoiceAvailable(int32_t priority) { return MV_VoiceAvailable(priority); }
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
int32_t FX_PauseVoice(int32_t handle, int32_t pause) { return FX_CheckMVErr(MV_PauseVoice(handle, pause)); }
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
int32_t FX_GetPosition(int32_t handle, int32_t *position) { return FX_CheckMVErr(MV_GetPosition(handle, position)); }
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
int32_t FX_SetPosition(int32_t handle, int32_t position) { return FX_CheckMVErr(MV_SetPosition(handle, position)); }
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
int32_t FX_EndLooping(int32_t handle) { return FX_CheckMVErr(MV_EndLooping(handle)); }
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
int32_t FX_SetPan(int32_t handle, int32_t vol, int32_t left, int32_t right)
|
2009-07-27 05:47:50 +00:00
|
|
|
{
|
2015-07-08 03:33:56 +00:00
|
|
|
return FX_CheckMVErr(MV_SetPan(handle, vol, left, right));
|
2009-07-27 05:47:50 +00:00
|
|
|
}
|
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
int32_t FX_SetPitch(int32_t handle, int32_t pitchoffset) { return FX_CheckMVErr(MV_SetPitch(handle, pitchoffset)); }
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
int32_t FX_SetFrequency(int32_t handle, int32_t frequency) { return FX_CheckMVErr(MV_SetFrequency(handle, frequency)); }
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
int32_t FX_Pan3D(int32_t handle, int32_t angle, int32_t distance)
|
2009-07-27 05:47:50 +00:00
|
|
|
{
|
2015-07-08 03:33:56 +00:00
|
|
|
return FX_CheckMVErr(MV_Pan3D(handle, angle, distance));
|
2009-07-27 05:47:50 +00:00
|
|
|
}
|
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
int32_t FX_SoundActive(int32_t handle) { return MV_VoicePlaying(handle); }
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
int32_t FX_SoundsPlaying(void) { return MV_VoicesPlaying(); }
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
int32_t FX_StopSound(int32_t handle) { return FX_CheckMVErr(MV_Kill(handle)); }
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
int32_t FX_StopAllSounds(void) { return FX_CheckMVErr(MV_KillAllVoices()); }
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
static wavefmt_t FX_AutoDetectFormat(const char *ptr, uint32_t length)
|
2009-07-27 05:47:50 +00:00
|
|
|
{
|
2015-07-08 03:33:56 +00:00
|
|
|
wavefmt_t fmt = FMT_UNKNOWN;
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2014-01-24 21:38:59 +00:00
|
|
|
if (length < 12)
|
2015-07-08 03:33:56 +00:00
|
|
|
return fmt;
|
2014-01-24 21:38:59 +00:00
|
|
|
|
2016-01-11 05:05:38 +00:00
|
|
|
switch (LITTLE32(*(int32_t const *)ptr))
|
2009-07-27 05:47:50 +00:00
|
|
|
{
|
2015-07-08 03:33:56 +00:00
|
|
|
case 'C' + ('r' << 8) + ('e' << 16) + ('a' << 24): // Crea
|
|
|
|
fmt = FMT_VOC;
|
2014-01-27 10:30:00 +00:00
|
|
|
break;
|
2015-07-08 03:33:56 +00:00
|
|
|
case 'O' + ('g' << 8) + ('g' << 16) + ('S' << 24): // OggS
|
|
|
|
fmt = FMT_VORBIS;
|
|
|
|
break;
|
|
|
|
case 'R' + ('I' << 8) + ('F' << 16) + ('F' << 24): // RIFF
|
2016-01-11 05:05:38 +00:00
|
|
|
switch (LITTLE32(*(int32_t const *)(ptr + 8)))
|
2015-07-08 03:33:56 +00:00
|
|
|
{
|
|
|
|
case 'C' + ('D' << 8) + ('X' << 16) + ('A' << 24): // CDXA
|
|
|
|
fmt = FMT_XA;
|
|
|
|
break;
|
|
|
|
default: fmt = FMT_WAV; break;
|
|
|
|
}
|
2014-01-27 10:30:00 +00:00
|
|
|
break;
|
2015-07-08 03:33:56 +00:00
|
|
|
case 'f' + ('L' << 8) + ('a' << 16) + ('C' << 24): // fLaC
|
|
|
|
fmt = FMT_FLAC;
|
|
|
|
break;
|
|
|
|
default:
|
2016-01-11 05:05:38 +00:00
|
|
|
switch (LITTLE32(*(int32_t const *)(ptr + 8)))
|
2015-07-08 03:33:56 +00:00
|
|
|
{
|
|
|
|
case 'W' + ('A' << 8) + ('V' << 16) + ('E' << 24): // WAVE
|
|
|
|
fmt = FMT_WAV;
|
|
|
|
break;
|
|
|
|
}
|
2009-08-06 10:12:13 +00:00
|
|
|
break;
|
2009-07-27 05:47:50 +00:00
|
|
|
}
|
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
return fmt;
|
2009-07-27 05:47:50 +00:00
|
|
|
}
|
|
|
|
|
2012-12-29 10:59:21 +00:00
|
|
|
int32_t FX_PlayAuto(char *ptr, uint32_t length, int32_t pitchoffset, int32_t vol,
|
|
|
|
int32_t left, int32_t right, int32_t priority, uint32_t callbackval)
|
|
|
|
{
|
2014-02-01 16:19:57 +00:00
|
|
|
return FX_PlayLoopedAuto(ptr, length, -1, -1, pitchoffset, vol, left, right, priority, callbackval);
|
2010-06-22 21:50:01 +00:00
|
|
|
}
|
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
int32_t FX_PlayLoopedAuto(char *ptr, uint32_t length, int32_t loopstart, int32_t loopend, int32_t pitchoffset,
|
|
|
|
int32_t vol, int32_t left, int32_t right, int32_t priority, uint32_t callbackval)
|
2009-07-27 05:47:50 +00:00
|
|
|
{
|
|
|
|
int32_t handle = -1;
|
2012-12-29 10:59:21 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
EDUKE32_STATIC_ASSERT(FMT_MAX == 7);
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
static int32_t(*const func[FMT_MAX])(char *, uint32_t, int32_t, int32_t, int32_t, int32_t, int32_t, int32_t, int32_t, uint32_t) =
|
|
|
|
{ NULL, NULL, MV_PlayVOC, MV_PlayWAV, MV_PlayVorbis, MV_PlayFLAC, MV_PlayXA };
|
|
|
|
|
|
|
|
wavefmt_t const fmt = FX_AutoDetectFormat(ptr, length);
|
|
|
|
|
|
|
|
if (func[fmt])
|
|
|
|
handle = func[fmt](ptr, length, loopstart, loopend, pitchoffset, vol, left, right, priority, callbackval);
|
2009-08-06 10:12:13 +00:00
|
|
|
|
2010-06-22 21:50:01 +00:00
|
|
|
if (handle <= MV_Ok)
|
2009-07-27 05:47:50 +00:00
|
|
|
{
|
|
|
|
FX_SetErrorCode(FX_MultiVocError);
|
|
|
|
handle = FX_Warning;
|
|
|
|
}
|
|
|
|
|
|
|
|
return handle;
|
|
|
|
}
|
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
int32_t FX_PlayAuto3D(char *ptr, uint32_t length, int32_t loophow, int32_t pitchoffset, int32_t angle, int32_t distance,
|
|
|
|
int32_t priority, uint32_t callbackval)
|
2009-07-27 05:47:50 +00:00
|
|
|
{
|
|
|
|
int32_t handle = -1;
|
|
|
|
|
2015-07-08 03:33:56 +00:00
|
|
|
EDUKE32_STATIC_ASSERT(FMT_MAX == 7);
|
|
|
|
|
|
|
|
static int32_t (*const func[FMT_MAX])(char *, uint32_t, int32_t, int32_t, int32_t, int32_t, int32_t, uint32_t) =
|
|
|
|
{ NULL, NULL, MV_PlayVOC3D, MV_PlayWAV3D, MV_PlayVorbis3D, MV_PlayFLAC3D, MV_PlayXA3D };
|
|
|
|
|
|
|
|
wavefmt_t const fmt = FX_AutoDetectFormat(ptr, length);
|
|
|
|
|
|
|
|
if (func[fmt])
|
|
|
|
handle = func[fmt](ptr, length, loophow, pitchoffset, angle, distance, priority, callbackval);
|
2009-07-27 05:47:50 +00:00
|
|
|
|
2010-06-22 21:50:01 +00:00
|
|
|
if (handle <= MV_Ok)
|
2009-07-27 05:47:50 +00:00
|
|
|
{
|
|
|
|
FX_SetErrorCode(FX_MultiVocError);
|
|
|
|
handle = FX_Warning;
|
|
|
|
}
|
|
|
|
|
|
|
|
return handle;
|
|
|
|
}
|
2010-06-22 21:50:01 +00:00
|
|
|
|
|
|
|
int32_t FX_SetVoiceCallback(int32_t handle, uint32_t callbackval)
|
|
|
|
{
|
2015-07-08 03:33:56 +00:00
|
|
|
int32_t status = MV_SetVoiceCallback(handle, callbackval);
|
2010-06-22 21:50:01 +00:00
|
|
|
|
|
|
|
if (status != MV_Ok)
|
|
|
|
{
|
|
|
|
FX_SetErrorCode(FX_MultiVocError);
|
2012-12-29 10:59:21 +00:00
|
|
|
return FX_Warning;
|
2010-06-22 21:50:01 +00:00
|
|
|
}
|
|
|
|
|
2012-12-29 10:59:21 +00:00
|
|
|
return FX_Ok;
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t FX_SetPrintf(void (*function)(const char *, ...))
|
|
|
|
{
|
|
|
|
MV_SetPrintf(function);
|
|
|
|
|
|
|
|
return FX_Ok;
|
2010-06-22 21:50:01 +00:00
|
|
|
}
|