/* Copyright (c) 2002-2012 Croteam Ltd. All rights reserved. */ #include "stdh.h" #include <Engine/Sound/SoundDecoder.h> #include <Engine/Base/Stream.h> #include <Engine/Base/Console.h> #include <Engine/Base/ErrorReporting.h> #include <Engine/Base/FileName.h> #include <Engine/Base/Unzip.h> #include <Engine/Base/Translation.h> #include <Engine/Math/Functions.h> // generic function called if a dll function is not found static void FailFunction_t(const char *strName) { ThrowF_t(TRANS("Function %s not found."), strName); } // ------------------------------------ AMP11 // amp11lib vars extern BOOL _bAMP11Enabled = FALSE; static HINSTANCE _hAmp11lib = NULL; // amp11lib types typedef signed char ALsint8; typedef unsigned char ALuint8; typedef signed short ALsint16; typedef unsigned short ALuint16; typedef signed int ALsint32; typedef unsigned int ALuint32; typedef signed int ALsize; typedef int ALbool; typedef float ALfloat; #define ALtrue 1 #define ALfalse 0 typedef ALsint32 ALhandle; // define amp11lib function pointers #define DLLFUNCTION(dll, output, name, inputs, params, required) \ output (__stdcall *p##name) inputs = NULL; #include "al_functions.h" #undef DLLFUNCTION static void AMP11_SetFunctionPointers_t(void) { const char *strName; // get amp11lib function pointers #define DLLFUNCTION(dll, output, name, inputs, params, required) \ strName = "_" #name "@" #params; \ p##name = (output (__stdcall*) inputs) GetProcAddress( _hAmp11lib, strName); \ if(p##name == NULL) FailFunction_t(strName); #include "al_functions.h" #undef DLLFUNCTION } static void AMP11_ClearFunctionPointers(void) { // clear amp11lib function pointers #define DLLFUNCTION(dll, output, name, inputs, params, required) p##name = NULL; #include "al_functions.h" #undef DLLFUNCTION } class CDecodeData_MPEG { public: ALhandle mpeg_hMainFile; // mainfile handle if using subfile ALhandle mpeg_hFile; // file handle ALhandle mpeg_hDecoder; // the decoder handle FLOAT mpeg_fSecondsLen; // length of sound in seconds WAVEFORMATEX mpeg_wfeFormat; // format of sound }; // ------------------------------------ Ogg Vorbis //#include <vorbis\vorbisfile.h> // we define needed stuff ourselves, and ignore the rest // vorbis vars extern BOOL _bOVEnabled = FALSE; static HINSTANCE _hOV = NULL; #define OV_FALSE -1 #define OV_EOF -2 #define OV_HOLE -3 #define OV_EREAD -128 #define OV_EFAULT -129 #define OV_EIMPL -130 #define OV_EINVAL -131 #define OV_ENOTVORBIS -132 #define OV_EBADHEADER -133 #define OV_EVERSION -134 #define OV_ENOTAUDIO -135 #define OV_EBADPACKET -136 #define OV_EBADLINK -137 #define OV_ENOSEEK -138 // vorbis types typedef __int64 ogg_int64_t; typedef struct { size_t (*read_func) (void *ptr, size_t size, size_t nmemb, void *datasource); int (*seek_func) (void *datasource, ogg_int64_t offset, int whence); int (*close_func) (void *datasource); long (*tell_func) (void *datasource); } ov_callbacks; struct OggVorbis_File { // don't wanna know whats inside UBYTE dummy[2048]; // last time checked, the actual size was 720 (ogg vorbis version 1.0rc1) }; struct vorbis_info { int version; int channels; long rate; long bitrate_upper; long bitrate_nominal; long bitrate_lower; // don't want to know the rest... //.................. }; // define vorbis function pointers #define DLLFUNCTION(dll, output, name, inputs, params, required) \ output (__cdecl *p##name) inputs = NULL; #include "ov_functions.h" #undef DLLFUNCTION static void OV_SetFunctionPointers_t(void) { const char *strName; // get vo function pointers #define DLLFUNCTION(dll, output, name, inputs, params, required) \ strName = #name ; \ p##name = (output (__cdecl *) inputs) GetProcAddress( _hOV, strName); \ if(p##name == NULL) FailFunction_t(strName); #include "ov_functions.h" #undef DLLFUNCTION } static void OV_ClearFunctionPointers(void) { // clear vo function pointers #define DLLFUNCTION(dll, output, name, inputs, params, required) p##name = NULL; #include "ov_functions.h" #undef DLLFUNCTION } class CDecodeData_OGG { public: FILE *ogg_fFile; // the stdio file that ogg is in SLONG ogg_slOffset; // offset where the ogg starts in the file (!=0 for oggs in zip) SLONG ogg_slSize; // size of ogg in the file (!=filesize for oggs in zip) OggVorbis_File *ogg_vfVorbisFile; // the decoder file WAVEFORMATEX ogg_wfeFormat; // format of sound }; // ogg file reading callbacks // static size_t ogg_read_func (void *ptr, size_t size, size_t nmemb, void *datasource) { CDecodeData_OGG *pogg = (CDecodeData_OGG *)datasource; // calculate how much can be read at most SLONG slToRead = size*nmemb; SLONG slCurrentPos = ftell(pogg->ogg_fFile)-pogg->ogg_slOffset; SLONG slSizeLeft = ClampDn(pogg->ogg_slSize-slCurrentPos, 0L); slToRead = ClampUp(slToRead, slSizeLeft); // rounded down to the block size slToRead/=size; slToRead*=size; // if there is nothing to read if (slToRead<=0) { return 0; } return fread(ptr, size, slToRead/size, pogg->ogg_fFile); } static int ogg_seek_func (void *datasource, ogg_int64_t offset, int whence) { return -1; /* !!!! seeking is evil with vorbisfile 1.0RC2 CDecodeData_OGG *pogg = (CDecodeData_OGG *)datasource; SLONG slCurrentPos = ftell(pogg->ogg_fFile)-pogg->ogg_slOffset; if (whence==SEEK_CUR) { return fseek(pogg->ogg_fFile, offset, SEEK_CUR); } else if (whence==SEEK_END) { return fseek(pogg->ogg_fFile, pogg->ogg_slOffset+pogg->ogg_slSize-offset, SEEK_SET); } else { ASSERT(whence==SEEK_SET); return fseek(pogg->ogg_fFile, pogg->ogg_slOffset+offset, SEEK_SET); } */ } static int ogg_close_func (void *datasource) { return 0; /* !!!! closing is evil with vorbisfile 1.0RC2 CDecodeData_OGG *pogg = (CDecodeData_OGG *)datasource; fclose(pogg->ogg_fFile); */ } static long ogg_tell_func (void *datasource) { return -1; /* !!!! seeking is evil with vorbisfile 1.0RC2 CDecodeData_OGG *pogg = (CDecodeData_OGG *)datasource; ftell(pogg->ogg_fFile)-pogg->ogg_slOffset; */ } static ov_callbacks ovcCallbacks = { ogg_read_func, ogg_seek_func, ogg_close_func, ogg_tell_func, }; // initialize/end the decoding support engine(s) void CSoundDecoder::InitPlugins(void) { try { // load vorbis if (_hOV==NULL) { #ifndef NDEBUG #define VORBISLIB "vorbisfile_d.dll" #else #define VORBISLIB "vorbisfile.dll" #endif _hOV = ::LoadLibraryA(VORBISLIB); } if( _hOV == NULL) { ThrowF_t(TRANS("Cannot load vorbisfile.dll.")); } // prepare function pointers OV_SetFunctionPointers_t(); // if all successful, enable mpx playing _bOVEnabled = TRUE; CPrintF(TRANS(" vorbisfile.dll loaded, ogg playing enabled\n")); } catch (char *strError) { CPrintF(TRANS("OGG playing disabled: %s\n"), strError); } try { // load amp11lib if (_hAmp11lib==NULL) { _hAmp11lib = ::LoadLibraryA( "amp11lib.dll"); } if( _hAmp11lib == NULL) { ThrowF_t(TRANS("Cannot load amp11lib.dll.")); } // prepare function pointers AMP11_SetFunctionPointers_t(); // initialize amp11lib before calling any of its functions palInitLibrary(); // if all successful, enable mpx playing _bAMP11Enabled = TRUE; CPrintF(TRANS(" amp11lib.dll loaded, mpx playing enabled\n")); } catch (char *strError) { CPrintF(TRANS("MPX playing disabled: %s\n"), strError); } } void CSoundDecoder::EndPlugins(void) { // cleanup amp11lib when not needed anymore if (_bAMP11Enabled) { palEndLibrary(); AMP11_ClearFunctionPointers(); FreeLibrary(_hAmp11lib); _hAmp11lib = NULL; _bAMP11Enabled = FALSE; } // cleanup vorbis when not needed anymore if (_bOVEnabled) { OV_ClearFunctionPointers(); FreeLibrary(_hOV); _hOV = NULL; _bOVEnabled = FALSE; } } // decoder that streams from file CSoundDecoder::CSoundDecoder(const CTFileName &fnm) { sdc_pogg = NULL; sdc_pmpeg = NULL; CTFileName fnmExpanded; INDEX iFileType = ExpandFilePath(EFP_READ, fnm, fnmExpanded); // if ogg if (fnmExpanded.FileExt()==".ogg") { if (!_bOVEnabled) { return; } sdc_pogg = new CDecodeData_OGG; sdc_pogg->ogg_fFile = NULL; sdc_pogg->ogg_vfVorbisFile = NULL; sdc_pogg->ogg_slOffset = 0; sdc_pogg->ogg_slSize = 0; INDEX iZipHandle = 0; try { // if in zip if (iFileType==EFP_BASEZIP || iFileType==EFP_MODZIP) { // open it iZipHandle = UNZIPOpen_t(fnmExpanded); CTFileName fnmZip; SLONG slOffset; SLONG slSizeCompressed; SLONG slSizeUncompressed; BOOL bCompressed; UNZIPGetFileInfo(iZipHandle, fnmZip, slOffset, slSizeCompressed, slSizeUncompressed, bCompressed); // if compressed if (bCompressed) { ThrowF_t(TRANS("encoded audio in archives must not be compressed!\n")); } // open ogg file sdc_pogg->ogg_fFile = fopen(fnmZip, "rb"); // if error if (sdc_pogg->ogg_fFile==0) { ThrowF_t(TRANS("cannot open archive '%s'"), (const char*)fnmZip); } // remember offset and size sdc_pogg->ogg_slOffset = slOffset; sdc_pogg->ogg_slSize = slSizeUncompressed; fseek(sdc_pogg->ogg_fFile, slOffset, SEEK_SET); // if not in zip } else if (iFileType==EFP_FILE) { // open ogg file sdc_pogg->ogg_fFile = fopen(fnmExpanded, "rb"); // if error if (sdc_pogg->ogg_fFile==0) { ThrowF_t(TRANS("cannot open encoded audio file")); } // remember offset and size sdc_pogg->ogg_slOffset = 0; fseek(sdc_pogg->ogg_fFile, 0, SEEK_END); sdc_pogg->ogg_slSize = ftell(sdc_pogg->ogg_fFile); fseek(sdc_pogg->ogg_fFile, 0, SEEK_SET); // if not found } else { ThrowF_t(TRANS("file not found")); } // initialize decoder sdc_pogg->ogg_vfVorbisFile = new OggVorbis_File; int iRes = pov_open_callbacks(sdc_pogg, sdc_pogg->ogg_vfVorbisFile, NULL, 0, ovcCallbacks); // if error if (iRes!=0) { ThrowF_t(TRANS("cannot open ogg decoder")); } // get info on the file vorbis_info *pvi = pov_info(sdc_pogg->ogg_vfVorbisFile, -1); // remember it's format WAVEFORMATEX form; form.wFormatTag=WAVE_FORMAT_PCM; form.nChannels=pvi->channels; form.nSamplesPerSec=pvi->rate; form.wBitsPerSample=16; form.nBlockAlign=form.nChannels*form.wBitsPerSample/8; form.nAvgBytesPerSec=form.nSamplesPerSec*form.nBlockAlign; form.cbSize=0; // check for stereo if (pvi->channels!=2) { ThrowF_t(TRANS("not stereo")); } sdc_pogg->ogg_wfeFormat = form; } catch (char*strError) { CPrintF(TRANS("Cannot open encoded audio '%s' for streaming: %s\n"), (const char*)fnm, (const char*)strError); if (sdc_pogg->ogg_vfVorbisFile!=NULL) { delete sdc_pogg->ogg_vfVorbisFile; sdc_pogg->ogg_vfVorbisFile = NULL; } if (sdc_pogg->ogg_fFile!=NULL) { fclose(sdc_pogg->ogg_fFile); sdc_pogg->ogg_fFile = NULL; } if (iZipHandle!=0) { UNZIPClose(iZipHandle); } Clear(); return; } if (iZipHandle!=0) { UNZIPClose(iZipHandle); } // if mp3 } else if (fnmExpanded.FileExt()==".mp3") { if (!_bAMP11Enabled) { return; } sdc_pmpeg = new CDecodeData_MPEG; sdc_pmpeg->mpeg_hMainFile = 0; sdc_pmpeg->mpeg_hFile = 0; sdc_pmpeg->mpeg_hDecoder = 0; INDEX iZipHandle = 0; try { // if in zip if (iFileType==EFP_BASEZIP || iFileType==EFP_MODZIP) { // open it iZipHandle = UNZIPOpen_t(fnmExpanded); CTFileName fnmZip; SLONG slOffset; SLONG slSizeCompressed; SLONG slSizeUncompressed; BOOL bCompressed; UNZIPGetFileInfo(iZipHandle, fnmZip, slOffset, slSizeCompressed, slSizeUncompressed, bCompressed); // if compressed if (bCompressed) { ThrowF_t(TRANS("encoded audio in archives must not be compressed!\n")); } // open the zip file sdc_pmpeg->mpeg_hMainFile = palOpenInputFile(fnmZip); // if error if (sdc_pmpeg->mpeg_hMainFile==0) { ThrowF_t(TRANS("cannot open archive '%s'"), (const char*)fnmZip); } // open the subfile sdc_pmpeg->mpeg_hFile = palOpenSubFile(sdc_pmpeg->mpeg_hMainFile, slOffset, slSizeUncompressed); // if error if (sdc_pmpeg->mpeg_hFile==0) { ThrowF_t(TRANS("cannot open encoded audio file")); } // if not in zip } else if (iFileType==EFP_FILE) { // open mpx file sdc_pmpeg->mpeg_hFile = palOpenInputFile(fnmExpanded); // if error if (sdc_pmpeg->mpeg_hFile==0) { ThrowF_t(TRANS("cannot open mpx file")); } // if not found } else { ThrowF_t(TRANS("file not found")); } // get info on the file int layer, ver, freq, stereo, rate; if (!palGetMPXHeader(sdc_pmpeg->mpeg_hFile, &layer, &ver, &freq, &stereo, &rate)) { ThrowF_t(TRANS("not a valid mpeg audio file.")); } // remember it's format WAVEFORMATEX form; form.wFormatTag=WAVE_FORMAT_PCM; form.nChannels=stereo?2:1; form.nSamplesPerSec=freq; form.wBitsPerSample=16; form.nBlockAlign=form.nChannels*form.wBitsPerSample/8; form.nAvgBytesPerSec=form.nSamplesPerSec*form.nBlockAlign; form.cbSize=0; // check for stereo if (!stereo) { ThrowF_t(TRANS("not stereo")); } sdc_pmpeg->mpeg_wfeFormat = form; // initialize decoder sdc_pmpeg->mpeg_hDecoder = palOpenDecoder(sdc_pmpeg->mpeg_hFile); // if error if (sdc_pmpeg->mpeg_hDecoder==0) { ThrowF_t(TRANS("cannot open mpx decoder")); } } catch (char*strError) { CPrintF(TRANS("Cannot open mpx '%s' for streaming: %s\n"), (const char*)fnm, (const char*)strError); if (iZipHandle!=0) { UNZIPClose(iZipHandle); } Clear(); return; } if (iZipHandle!=0) { UNZIPClose(iZipHandle); } sdc_pmpeg->mpeg_fSecondsLen = palDecGetLen(sdc_pmpeg->mpeg_hDecoder); } } CSoundDecoder::~CSoundDecoder(void) { Clear(); } void CSoundDecoder::Clear(void) { if (sdc_pmpeg!=NULL) { if (sdc_pmpeg->mpeg_hDecoder!=0) palClose(sdc_pmpeg->mpeg_hDecoder); if (sdc_pmpeg->mpeg_hFile!=0) palClose(sdc_pmpeg->mpeg_hFile); if (sdc_pmpeg->mpeg_hMainFile!=0) palClose(sdc_pmpeg->mpeg_hMainFile); sdc_pmpeg->mpeg_hMainFile = 0; sdc_pmpeg->mpeg_hFile = 0; sdc_pmpeg->mpeg_hDecoder = 0; delete sdc_pmpeg; sdc_pmpeg = NULL; } else if (sdc_pogg!=NULL) { if (sdc_pogg->ogg_vfVorbisFile!=NULL) { pov_clear(sdc_pogg->ogg_vfVorbisFile); delete sdc_pogg->ogg_vfVorbisFile; sdc_pogg->ogg_vfVorbisFile = NULL; } if (sdc_pogg->ogg_fFile!=NULL) { fclose(sdc_pogg->ogg_fFile); sdc_pogg->ogg_fFile = NULL; } delete sdc_pogg; sdc_pogg = NULL; } } // reset decoder to start of sample void CSoundDecoder::Reset(void) { if (sdc_pmpeg!=NULL) { palDecSeekAbs(sdc_pmpeg->mpeg_hDecoder, 0.0f); } else if (sdc_pogg!=NULL) { /* !!!! seeking is evil with vorbisfile 1.0RC2 pov_time_seek(sdc_pogg->ogg_vfVorbisFile, 0.0f); */ // so instead, we reinit pov_clear(sdc_pogg->ogg_vfVorbisFile); fseek(sdc_pogg->ogg_fFile, sdc_pogg->ogg_slOffset, SEEK_SET); pov_open_callbacks(sdc_pogg, sdc_pogg->ogg_vfVorbisFile, NULL, 0, ovcCallbacks); } } BOOL CSoundDecoder::IsOpen(void) { if (sdc_pmpeg!=NULL && sdc_pmpeg->mpeg_hDecoder!=0) { return TRUE; } else if (sdc_pogg!=NULL && sdc_pogg->ogg_vfVorbisFile!=0) { return TRUE; } else { return FALSE; } } void CSoundDecoder::GetFormat(WAVEFORMATEX &wfe) { if (sdc_pmpeg!=NULL) { wfe = sdc_pmpeg->mpeg_wfeFormat; } else if (sdc_pogg!=NULL) { wfe = sdc_pogg->ogg_wfeFormat; } else { NOTHING; } } // decode a block of bytes INDEX CSoundDecoder::Decode(void *pvDestBuffer, INDEX ctBytesToDecode) { // if ogg if (sdc_pogg!=NULL && sdc_pogg->ogg_vfVorbisFile!=0) { // decode ogg static int iCurrrentSection = -1; // we don't care about this char *pch = (char *)pvDestBuffer; INDEX ctDecoded = 0; while (ctDecoded<ctBytesToDecode) { long iRes = pov_read(sdc_pogg->ogg_vfVorbisFile, pch, ctBytesToDecode-ctDecoded, 0, 2, 1, &iCurrrentSection); if (iRes<=0) { return ctDecoded; } ctDecoded+=iRes; pch+=iRes; } return ctDecoded; // if mpeg } else if (sdc_pmpeg!=NULL && sdc_pmpeg->mpeg_hDecoder!=0) { // decode mpeg return palRead(sdc_pmpeg->mpeg_hDecoder, pvDestBuffer, ctBytesToDecode); // if no decoder } else { // play all zeroes memset(pvDestBuffer, 0, ctBytesToDecode); return ctBytesToDecode; } }