2006-02-24 04:48:15 +00:00
|
|
|
#ifdef _WIN32
|
|
|
|
#define WIN32_LEAN_AND_MEAN
|
2006-09-14 00:02:31 +00:00
|
|
|
#define USE_WINDOWS_DWORD
|
2006-02-24 04:48:15 +00:00
|
|
|
#if defined(_WIN32_WINNT) && _WIN32_WINNT < 0x0400
|
|
|
|
#undef _WIN32_WINNT
|
|
|
|
#endif
|
|
|
|
#ifndef _WIN32_WINNT
|
|
|
|
#define _WIN32_WINNT 0x0400
|
|
|
|
#endif
|
|
|
|
#include <windows.h>
|
|
|
|
#include <mmsystem.h>
|
|
|
|
#else
|
|
|
|
#include <SDL.h>
|
|
|
|
#define FALSE 0
|
|
|
|
#define TRUE 1
|
|
|
|
#endif
|
|
|
|
#include <fmod.h>
|
|
|
|
#include "tempfiles.h"
|
|
|
|
#include "oplsynth/opl_mus_player.h"
|
|
|
|
#include "c_cvars.h"
|
|
|
|
#include "mus2midi.h"
|
|
|
|
#include "i_sound.h"
|
|
|
|
|
|
|
|
void I_InitMusicWin32 ();
|
|
|
|
void I_ShutdownMusicWin32 ();
|
|
|
|
|
|
|
|
extern float relative_volume;
|
|
|
|
|
|
|
|
// The base music class. Everything is derived from this --------------------
|
|
|
|
|
|
|
|
class MusInfo
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
MusInfo () : m_Status(STATE_Stopped) {}
|
|
|
|
virtual ~MusInfo ();
|
|
|
|
virtual void SetVolume (float volume) = 0;
|
|
|
|
virtual void Play (bool looping) = 0;
|
|
|
|
virtual void Pause () = 0;
|
|
|
|
virtual void Resume () = 0;
|
|
|
|
virtual void Stop () = 0;
|
|
|
|
virtual bool IsPlaying () = 0;
|
|
|
|
virtual bool IsMIDI () const = 0;
|
|
|
|
virtual bool IsValid () const = 0;
|
|
|
|
virtual bool SetPosition (int order);
|
|
|
|
|
|
|
|
enum EState
|
|
|
|
{
|
|
|
|
STATE_Stopped,
|
|
|
|
STATE_Playing,
|
|
|
|
STATE_Paused
|
|
|
|
} m_Status;
|
|
|
|
bool m_Looping;
|
|
|
|
};
|
|
|
|
|
|
|
|
// MUS file played with MIDI output messages --------------------------------
|
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
class MUSSong2 : public MusInfo
|
|
|
|
{
|
|
|
|
public:
|
2006-04-14 12:58:52 +00:00
|
|
|
MUSSong2 (FILE *file, char * musiccache, int length);
|
2006-02-24 04:48:15 +00:00
|
|
|
~MUSSong2 ();
|
|
|
|
|
|
|
|
void SetVolume (float volume);
|
|
|
|
void Play (bool looping);
|
|
|
|
void Pause ();
|
|
|
|
void Resume ();
|
|
|
|
void Stop ();
|
|
|
|
bool IsPlaying ();
|
|
|
|
bool IsMIDI () const;
|
|
|
|
bool IsValid () const;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
static DWORD WINAPI PlayerProc (LPVOID lpParameter);
|
|
|
|
void OutputVolume (DWORD volume);
|
|
|
|
int SendCommand ();
|
|
|
|
|
2006-09-14 00:02:31 +00:00
|
|
|
enum
|
|
|
|
{
|
|
|
|
SEND_DONE,
|
|
|
|
SEND_WAIT
|
2006-02-24 04:48:15 +00:00
|
|
|
};
|
|
|
|
|
2006-09-14 00:02:31 +00:00
|
|
|
HMIDIOUT MidiOut;
|
|
|
|
HANDLE PlayerThread;
|
|
|
|
HANDLE PauseEvent;
|
|
|
|
HANDLE ExitEvent;
|
|
|
|
HANDLE VolumeChangeEvent;
|
|
|
|
DWORD SavedVolume;
|
|
|
|
bool VolumeWorks;
|
|
|
|
|
2006-02-24 04:48:15 +00:00
|
|
|
BYTE *MusBuffer;
|
|
|
|
MUSHeader *MusHeader;
|
|
|
|
BYTE LastVelocity[16];
|
|
|
|
BYTE ChannelVolumes[16];
|
|
|
|
size_t MusP, MaxMusP;
|
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// MIDI file played with MIDI output messages -------------------------------
|
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
class MIDISong2 : public MusInfo
|
|
|
|
{
|
|
|
|
public:
|
2006-04-14 12:58:52 +00:00
|
|
|
MIDISong2 (FILE *file, char * musiccache, int length);
|
2006-02-24 04:48:15 +00:00
|
|
|
~MIDISong2 ();
|
|
|
|
|
|
|
|
void SetVolume (float volume);
|
|
|
|
void Play (bool looping);
|
|
|
|
void Pause ();
|
|
|
|
void Resume ();
|
|
|
|
void Stop ();
|
|
|
|
bool IsPlaying ();
|
|
|
|
bool IsMIDI () const;
|
|
|
|
bool IsValid () const;
|
|
|
|
|
|
|
|
protected:
|
2006-09-14 00:02:31 +00:00
|
|
|
struct TrackInfo;
|
|
|
|
|
2006-02-24 04:48:15 +00:00
|
|
|
static DWORD WINAPI PlayerProc (LPVOID lpParameter);
|
|
|
|
void OutputVolume (DWORD volume);
|
|
|
|
void ProcessInitialMetaEvents ();
|
|
|
|
DWORD SendCommands ();
|
|
|
|
void SendCommand (TrackInfo *track);
|
|
|
|
TrackInfo *FindNextDue ();
|
|
|
|
|
2006-09-14 00:02:31 +00:00
|
|
|
HMIDIOUT MidiOut;
|
|
|
|
HANDLE PlayerThread;
|
|
|
|
HANDLE PauseEvent;
|
|
|
|
HANDLE ExitEvent;
|
|
|
|
HANDLE VolumeChangeEvent;
|
|
|
|
DWORD SavedVolume;
|
|
|
|
bool VolumeWorks;
|
|
|
|
|
2006-02-24 04:48:15 +00:00
|
|
|
BYTE *MusHeader;
|
|
|
|
BYTE ChannelVolumes[16];
|
|
|
|
TrackInfo *Tracks;
|
|
|
|
TrackInfo *TrackDue;
|
|
|
|
int NumTracks;
|
|
|
|
int Format;
|
|
|
|
int Division;
|
|
|
|
int Tempo;
|
|
|
|
WORD DesignationMask;
|
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// MOD file played with FMOD ------------------------------------------------
|
|
|
|
|
|
|
|
class MODSong : public MusInfo
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
MODSong (const char *file, int offset, int length);
|
|
|
|
~MODSong ();
|
|
|
|
void SetVolume (float volume);
|
|
|
|
void Play (bool looping);
|
|
|
|
void Pause ();
|
|
|
|
void Resume ();
|
|
|
|
void Stop ();
|
|
|
|
bool IsPlaying ();
|
|
|
|
bool IsMIDI () const { return false; }
|
|
|
|
bool IsValid () const { return m_Module != NULL; }
|
|
|
|
bool SetPosition (int order);
|
|
|
|
|
|
|
|
protected:
|
|
|
|
MODSong () {}
|
|
|
|
|
|
|
|
SoundTrackerModule *m_Module;
|
|
|
|
};
|
|
|
|
|
|
|
|
// OGG/MP3/WAV/other format streamed through FMOD ---------------------------
|
|
|
|
|
|
|
|
class StreamSong : public MusInfo
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
StreamSong (const char *file, int offset, int length);
|
|
|
|
~StreamSong ();
|
|
|
|
void SetVolume (float volume);
|
|
|
|
void Play (bool looping);
|
|
|
|
void Pause ();
|
|
|
|
void Resume ();
|
|
|
|
void Stop ();
|
|
|
|
bool IsPlaying ();
|
|
|
|
bool IsMIDI () const { return false; }
|
|
|
|
bool IsValid () const { return m_Stream != NULL; }
|
|
|
|
|
|
|
|
protected:
|
|
|
|
StreamSong () : m_Stream(NULL), m_LastPos(0) {}
|
|
|
|
|
|
|
|
SoundStream *m_Stream;
|
|
|
|
int m_LastPos;
|
|
|
|
};
|
|
|
|
|
|
|
|
// SPC file, rendered with SNESAPU.DLL and streamed through FMOD ------------
|
|
|
|
|
|
|
|
typedef void (__stdcall *SNESAPUInfo_TYPE) (DWORD*, DWORD*, DWORD*);
|
|
|
|
typedef void (__stdcall *GetAPUData_TYPE) (void**, BYTE**, BYTE**, DWORD**, void**, void**, DWORD**, DWORD**);
|
|
|
|
typedef void (__stdcall *LoadSPCFile_TYPE) (void*);
|
|
|
|
typedef void (__stdcall *ResetAPU_TYPE) (DWORD);
|
|
|
|
typedef void (__stdcall *SetDSPAmp_TYPE) (DWORD);
|
|
|
|
typedef void (__stdcall *FixAPU_TYPE) (WORD, BYTE, BYTE, BYTE, BYTE, BYTE);
|
|
|
|
typedef void (__stdcall *SetAPUOpt_TYPE) (DWORD, DWORD, DWORD, DWORD, DWORD, DWORD);
|
|
|
|
typedef void *(__stdcall *EmuAPU_TYPE) (void *, DWORD, BYTE);
|
|
|
|
|
|
|
|
class SPCSong : public StreamSong
|
|
|
|
{
|
|
|
|
public:
|
2006-04-14 12:58:52 +00:00
|
|
|
SPCSong (FILE *file, char * musiccache, int length);
|
2006-02-24 04:48:15 +00:00
|
|
|
~SPCSong ();
|
|
|
|
void Play (bool looping);
|
|
|
|
bool IsPlaying ();
|
|
|
|
bool IsValid () const;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
bool LoadEmu ();
|
|
|
|
void CloseEmu ();
|
|
|
|
|
|
|
|
static bool FillStream (SoundStream *stream, void *buff, int len, void *userdata);
|
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
HINSTANCE HandleAPU;
|
|
|
|
#else
|
|
|
|
void *HandleAPU;
|
|
|
|
#endif
|
|
|
|
int APUVersion;
|
|
|
|
|
|
|
|
bool Stereo;
|
|
|
|
bool Is8Bit;
|
|
|
|
|
|
|
|
SNESAPUInfo_TYPE SNESAPUInfo;
|
|
|
|
GetAPUData_TYPE GetAPUData;
|
|
|
|
LoadSPCFile_TYPE LoadSPCFile;
|
|
|
|
ResetAPU_TYPE ResetAPU;
|
|
|
|
SetDSPAmp_TYPE SetDSPAmp;
|
|
|
|
FixAPU_TYPE FixAPU;
|
|
|
|
SetAPUOpt_TYPE SetAPUOpt;
|
|
|
|
EmuAPU_TYPE EmuAPU;
|
|
|
|
};
|
|
|
|
|
|
|
|
// MIDI file played with Timidity and possibly streamed through FMOD --------
|
|
|
|
|
|
|
|
class TimiditySong : public StreamSong
|
|
|
|
{
|
|
|
|
public:
|
2006-04-14 12:58:52 +00:00
|
|
|
TimiditySong (FILE *file, char * musiccache, int length);
|
2006-02-24 04:48:15 +00:00
|
|
|
~TimiditySong ();
|
|
|
|
void Play (bool looping);
|
|
|
|
void Stop ();
|
|
|
|
bool IsPlaying ();
|
2006-04-09 19:34:35 +00:00
|
|
|
bool IsValid () const { return CommandLine.Len() > 0; }
|
2008-01-27 18:09:06 +00:00
|
|
|
void SetVolume (float volume);
|
2006-02-24 04:48:15 +00:00
|
|
|
|
|
|
|
protected:
|
|
|
|
void PrepTimidity ();
|
|
|
|
bool LaunchTimidity ();
|
|
|
|
|
|
|
|
FTempFileName DiskName;
|
|
|
|
#ifdef _WIN32
|
|
|
|
HANDLE ReadWavePipe;
|
|
|
|
HANDLE WriteWavePipe;
|
|
|
|
HANDLE KillerEvent;
|
|
|
|
HANDLE ChildProcess;
|
|
|
|
bool Validated;
|
|
|
|
bool ValidateTimidity ();
|
|
|
|
#else // _WIN32
|
|
|
|
int WavePipe[2];
|
|
|
|
pid_t ChildProcess;
|
|
|
|
#endif
|
2006-05-03 22:45:01 +00:00
|
|
|
FString CommandLine;
|
2006-04-09 19:34:35 +00:00
|
|
|
size_t LoopPos;
|
2006-02-24 04:48:15 +00:00
|
|
|
|
|
|
|
static bool FillStream (SoundStream *stream, void *buff, int len, void *userdata);
|
|
|
|
#ifdef _WIN32
|
|
|
|
static const char EventName[];
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
|
|
|
// MUS file played by a software OPL2 synth and streamed through FMOD -------
|
|
|
|
|
|
|
|
class OPLMUSSong : public StreamSong
|
|
|
|
{
|
|
|
|
public:
|
2006-04-14 12:58:52 +00:00
|
|
|
OPLMUSSong (FILE *file, char * musiccache, int length);
|
2006-02-24 04:48:15 +00:00
|
|
|
~OPLMUSSong ();
|
|
|
|
void Play (bool looping);
|
|
|
|
bool IsPlaying ();
|
|
|
|
bool IsValid () const;
|
|
|
|
void ResetChips ();
|
|
|
|
|
|
|
|
protected:
|
|
|
|
static bool FillStream (SoundStream *stream, void *buff, int len, void *userdata);
|
|
|
|
|
|
|
|
OPLmusicBlock *Music;
|
|
|
|
};
|
|
|
|
|
|
|
|
// FLAC file streamed through FMOD (You should probably use Vorbis instead) -
|
|
|
|
|
|
|
|
class FLACSong : public StreamSong
|
|
|
|
{
|
|
|
|
public:
|
2006-04-14 12:58:52 +00:00
|
|
|
FLACSong (FILE *file, char * musiccache, int length);
|
2006-02-24 04:48:15 +00:00
|
|
|
~FLACSong ();
|
|
|
|
void Play (bool looping);
|
|
|
|
bool IsPlaying ();
|
|
|
|
bool IsValid () const;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
static bool FillStream (SoundStream *stream, void *buff, int len, void *userdata);
|
|
|
|
|
|
|
|
class FLACStreamer;
|
|
|
|
|
|
|
|
FLACStreamer *State;
|
|
|
|
};
|
|
|
|
|
|
|
|
// CD track/disk played through the multimedia system -----------------------
|
|
|
|
|
|
|
|
class CDSong : public MusInfo
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
CDSong (int track, int id);
|
|
|
|
~CDSong ();
|
|
|
|
void SetVolume (float volume) {}
|
|
|
|
void Play (bool looping);
|
|
|
|
void Pause ();
|
|
|
|
void Resume ();
|
|
|
|
void Stop ();
|
|
|
|
bool IsPlaying ();
|
|
|
|
bool IsMIDI () const { return false; }
|
|
|
|
bool IsValid () const { return m_Inited; }
|
|
|
|
|
|
|
|
protected:
|
|
|
|
CDSong () : m_Inited(false) {}
|
|
|
|
|
|
|
|
int m_Track;
|
|
|
|
bool m_Inited;
|
|
|
|
};
|
|
|
|
|
|
|
|
// CD track on a specific disk played through the multimedia system ---------
|
|
|
|
|
|
|
|
class CDDAFile : public CDSong
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
CDDAFile (FILE *file, int length);
|
|
|
|
};
|
|
|
|
|
|
|
|
// --------------------------------------------------------------------------
|
|
|
|
|
|
|
|
extern MusInfo *currSong;
|
|
|
|
extern int nomusic;
|
|
|
|
|
|
|
|
EXTERN_CVAR (Float, snd_musicvolume)
|
|
|
|
EXTERN_CVAR (Bool, opl_enable)
|