|
|
|
/*==LICENSE==*
|
|
|
|
|
|
|
|
CyanWorlds.com Engine - MMOG client, server and tools
|
|
|
|
Copyright (C) 2011 Cyan Worlds, Inc.
|
|
|
|
|
|
|
|
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 3 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, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
You can contact Cyan Worlds, Inc. by email legal@cyan.com
|
|
|
|
or by snail mail at:
|
|
|
|
Cyan Worlds, Inc.
|
|
|
|
14617 N Newport Hwy
|
|
|
|
Mead, WA 99021
|
|
|
|
|
|
|
|
*==LICENSE==*/
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
// //
|
|
|
|
// plOGGCodec - Plasma codec support for the OGG/Vorbis file format. //
|
|
|
|
// //
|
|
|
|
//// Notes ///////////////////////////////////////////////////////////////////
|
|
|
|
// //
|
|
|
|
// 2.7.2003 - Created by mcn. If only life were really this simple... //
|
|
|
|
// //
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
#include <cstdlib>
|
|
|
|
#include <cstdio>
|
|
|
|
#include <cstring>
|
|
|
|
|
|
|
|
#include <math.h>
|
|
|
|
#include <vorbis/codec.h>
|
|
|
|
#include <vorbis/vorbisfile.h>
|
|
|
|
|
|
|
|
#include "hsTypes.h"
|
|
|
|
#include "plOGGCodec.h"
|
|
|
|
|
|
|
|
#include "hsTimer.h"
|
|
|
|
#include "pnNetCommon/plNetApp.h"
|
|
|
|
|
|
|
|
plOGGCodec::DecodeFormat plOGGCodec::fDecodeFormat = plOGGCodec::k16bitSigned;
|
|
|
|
UInt8 plOGGCodec::fDecodeFlags = 0;
|
|
|
|
|
|
|
|
//// Constructor/Destructor //////////////////////////////////////////////////
|
|
|
|
|
|
|
|
plOGGCodec::plOGGCodec( const char *path, plAudioCore::ChannelSelect whichChan ) : fFileHandle( nil )
|
|
|
|
{
|
|
|
|
fOggFile = nil;
|
|
|
|
IOpen( path, whichChan );
|
|
|
|
fCurHeaderPos = 0;
|
|
|
|
fHeadBuf = nil;
|
|
|
|
}
|
|
|
|
|
|
|
|
void plOGGCodec::BuildActualWaveHeader()
|
|
|
|
{
|
|
|
|
// Build an actual WAVE header for this ogg
|
|
|
|
int fmtSize = 16;
|
|
|
|
short fmt = 1;
|
|
|
|
int factsize = 4;
|
|
|
|
int factdata = 0;
|
|
|
|
int size = fDataSize+48; // size of data with header except for first four bytes
|
|
|
|
|
|
|
|
fHeadBuf = (UInt8 *) ALLOC(56);
|
|
|
|
memcpy(fHeadBuf, "RIFF", 4);
|
|
|
|
memcpy(fHeadBuf+4, &size, 4);
|
|
|
|
memcpy(fHeadBuf+8, "WAVE", 4);
|
|
|
|
memcpy(fHeadBuf+12, "fmt ", 4);
|
|
|
|
memcpy(fHeadBuf+16, &fmtSize, 4);
|
|
|
|
memcpy(fHeadBuf+20, &fmt, 2); /* format */
|
|
|
|
memcpy(fHeadBuf+22, &fHeader.fNumChannels, 2);
|
|
|
|
memcpy(fHeadBuf+24, &fHeader.fNumSamplesPerSec, 4);
|
|
|
|
memcpy(fHeadBuf+28, &fHeader.fAvgBytesPerSec, 4);
|
|
|
|
memcpy(fHeadBuf+32, &fHeader.fBlockAlign, 4);
|
|
|
|
memcpy(fHeadBuf+34, &fHeader.fBitsPerSample, 2);
|
|
|
|
memcpy(fHeadBuf+36, "fact", 4);
|
|
|
|
memcpy(fHeadBuf+40, &factsize, 4);
|
|
|
|
memcpy(fHeadBuf+44, &factdata, 4);
|
|
|
|
memcpy(fHeadBuf+48, "data", 4);
|
|
|
|
memcpy(fHeadBuf+52, &fDataSize, 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool plOGGCodec::ReadFromHeader(int numBytes, void *data)
|
|
|
|
{
|
|
|
|
if(fCurHeaderPos < 56)
|
|
|
|
{
|
|
|
|
memcpy(data, fHeadBuf+fCurHeaderPos, numBytes);
|
|
|
|
fCurHeaderPos += numBytes;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void plOGGCodec::IOpen( const char *path, plAudioCore::ChannelSelect whichChan )
|
|
|
|
{
|
|
|
|
hsAssert( path != nil, "Invalid path specified in plOGGCodec reader" );
|
|
|
|
|
|
|
|
// plNetClientApp::StaticDebugMsg("Ogg Open %s, t=%f, start", path, hsTimer::GetSeconds());
|
|
|
|
|
|
|
|
strncpy( fFilename, path, sizeof( fFilename ) );
|
|
|
|
fWhichChannel = whichChan;
|
|
|
|
|
|
|
|
/// Open the file as a plain binary stream
|
|
|
|
fFileHandle = fopen( path, "rb" );
|
|
|
|
if( fFileHandle != nil )
|
|
|
|
{
|
|
|
|
/// Create the OGG data struct
|
|
|
|
fOggFile = TRACKED_NEW OggVorbis_File;
|
|
|
|
|
|
|
|
/// Open the OGG decompressor
|
|
|
|
if( ov_open( fFileHandle, fOggFile, NULL, 0 ) < 0 )
|
|
|
|
{
|
|
|
|
IError( "Unable to open OGG source file" );
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Construct some header info from the ogg info
|
|
|
|
vorbis_info *vInfo = ov_info( fOggFile, -1 );
|
|
|
|
|
|
|
|
fHeader.fFormatTag = 1;
|
|
|
|
fHeader.fNumChannels = vInfo->channels;
|
|
|
|
fHeader.fNumSamplesPerSec = vInfo->rate;
|
|
|
|
|
|
|
|
// Funny thing about the bits per sample: we get to CHOOSE. Go figure!
|
|
|
|
fHeader.fBitsPerSample = ( fDecodeFormat == k8bitUnsigned ) ? 8 : 16;
|
|
|
|
|
|
|
|
// Why WAV files hold this info when it can be calculated is beyond me...
|
|
|
|
fHeader.fBlockAlign = ( fHeader.fBitsPerSample * fHeader.fNumChannels ) >> 3;
|
|
|
|
fHeader.fAvgBytesPerSec = fHeader.fNumSamplesPerSec * fHeader.fBlockAlign;
|
|
|
|
|
|
|
|
|
|
|
|
/// The size in bytes of our PCM data stream
|
|
|
|
/// Note: OGG sometimes seems to be off by 1 sample, which causes our reads to suck
|
|
|
|
/// because we end up waiting for 1 more sample than we actually have. So, on the
|
|
|
|
/// assumption that OGG is just slightly wrong sometimes, we just subtract 1 sample
|
|
|
|
/// from what it tells us. As Brice put it, who's going to miss 1/40,000'ths of a second?
|
|
|
|
fDataSize = (UInt32)(( ov_pcm_total( fOggFile, -1 ) - 1 ) * fHeader.fBlockAlign);
|
|
|
|
|
|
|
|
/// Channel select
|
|
|
|
if( fWhichChannel != plAudioCore::kAll )
|
|
|
|
{
|
|
|
|
fChannelAdjust = 2;
|
|
|
|
fChannelOffset = ( fWhichChannel == plAudioCore::kLeft ) ? 0 : 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fChannelAdjust = 1;
|
|
|
|
fChannelOffset = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// Construct our fake header for channel adjustment
|
|
|
|
fFakeHeader = fHeader;
|
|
|
|
fFakeHeader.fAvgBytesPerSec /= fChannelAdjust;
|
|
|
|
fFakeHeader.fNumChannels /= (UInt16)fChannelAdjust;
|
|
|
|
fFakeHeader.fBlockAlign /= (UInt16)fChannelAdjust;
|
|
|
|
|
|
|
|
SetPosition( 0 );
|
|
|
|
}
|
|
|
|
// plNetClientApp::StaticDebugMsg("Ogg Open %s, t=%f, end", path, hsTimer::GetSeconds());
|
|
|
|
}
|
|
|
|
|
|
|
|
plOGGCodec::~plOGGCodec()
|
|
|
|
{
|
|
|
|
Close();
|
|
|
|
}
|
|
|
|
|
|
|
|
void plOGGCodec::Close( void )
|
|
|
|
{
|
|
|
|
// plNetClientApp::StaticDebugMsg("Ogg Close, t=%f, start", hsTimer::GetSeconds());
|
|
|
|
FREE(fHeadBuf);
|
|
|
|
fHeadBuf = nil;
|
|
|
|
if( fOggFile != nil )
|
|
|
|
{
|
|
|
|
ov_clear( fOggFile );
|
|
|
|
DEL(fOggFile);
|
|
|
|
fOggFile = nil;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( fFileHandle != nil )
|
|
|
|
{
|
|
|
|
fclose( fFileHandle );
|
|
|
|
fFileHandle = nil;
|
|
|
|
}
|
|
|
|
// plNetClientApp::StaticDebugMsg("Ogg Close, t=%f, end", hsTimer::GetSeconds());
|
|
|
|
}
|
|
|
|
|
|
|
|
void plOGGCodec::IError( const char *msg )
|
|
|
|
{
|
|
|
|
hsAssert( false, msg );
|
|
|
|
Close();
|
|
|
|
}
|
|
|
|
|
|
|
|
plWAVHeader &plOGGCodec::GetHeader( void )
|
|
|
|
{
|
|
|
|
hsAssert( IsValid(), "GetHeader() called on an invalid OGG file" );
|
|
|
|
|
|
|
|
return fFakeHeader;
|
|
|
|
}
|
|
|
|
|
|
|
|
float plOGGCodec::GetLengthInSecs( void )
|
|
|
|
{
|
|
|
|
hsAssert( IsValid(), "GetLengthInSecs() called on an invalid OGG file" );
|
|
|
|
|
|
|
|
// Just query ogg directly...starting to see how cool ogg is yet?
|
|
|
|
return (float)ov_time_total( fOggFile, -1 );
|
|
|
|
}
|
|
|
|
|
|
|
|
hsBool plOGGCodec::SetPosition( UInt32 numBytes )
|
|
|
|
{
|
|
|
|
hsAssert( IsValid(), "GetHeader() called on an invalid OGG file" );
|
|
|
|
|
|
|
|
|
|
|
|
if( !ov_seekable( fOggFile ) )
|
|
|
|
{
|
|
|
|
hsAssert( false, "Trying to set position on an unseekable OGG stream!" );
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// The numBytes position is in uncompressed space and should be sample-aligned anyway,
|
|
|
|
// so this should be just fine here.
|
|
|
|
ogg_int64_t newSample = ( numBytes / (fFakeHeader.fBlockAlign * fChannelAdjust) );
|
|
|
|
|
|
|
|
// Now please note how freaking easy it is here to do accurate or fast seeking...
|
|
|
|
// Also note that if we're doing our channel extraction, we MUST do it the accurate way
|
|
|
|
if( ( fDecodeFlags & kFastSeeking ) && fChannelAdjust == 1 )
|
|
|
|
{
|
|
|
|
if( ov_pcm_seek_page( fOggFile, newSample ) != 0 )
|
|
|
|
{
|
|
|
|
IError( "Unable to seek OGG stream" );
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if( ov_pcm_seek( fOggFile, newSample ) != 0 )
|
|
|
|
{
|
|
|
|
IError( "Unable to seek OGG stream" );
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
hsBool plOGGCodec::Read( UInt32 numBytes, void *buffer )
|
|
|
|
{
|
|
|
|
hsAssert( IsValid(), "GetHeader() called on an invalid OGG file" );
|
|
|
|
// plNetClientApp::StaticDebugMsg("Ogg Read, t=%f, start", hsTimer::GetSeconds());
|
|
|
|
|
|
|
|
int bytesPerSample = ( fDecodeFormat == k16bitSigned ) ? 2 : 1;
|
|
|
|
int isSigned = ( fDecodeFormat == k16bitSigned ) ? 1 : 0;
|
|
|
|
int currSection;
|
|
|
|
|
|
|
|
if( fWhichChannel == plAudioCore::kAll )
|
|
|
|
{
|
|
|
|
// Easy, just a straight read
|
|
|
|
char *uBuffer = (char *)buffer;
|
|
|
|
|
|
|
|
while( numBytes > 0 )
|
|
|
|
{
|
|
|
|
// Supposedly we should pay attention to currSection in case of bitrate changes,
|
|
|
|
// but hopefully we'll never have those....
|
|
|
|
|
|
|
|
long bytesRead = ov_read( fOggFile, uBuffer, numBytes, 0, bytesPerSample, isSigned, &currSection );
|
|
|
|
|
|
|
|
// Since our job is so simple, do some extra error checking
|
|
|
|
if( bytesRead == OV_HOLE )
|
|
|
|
{
|
|
|
|
IError( "Unable to read from OGG file: missing data" );
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else if( bytesRead == OV_EBADLINK )
|
|
|
|
{
|
|
|
|
IError( "Unable to read from OGG file: corrupt link" );
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else if( bytesRead == 0 )
|
|
|
|
{
|
|
|
|
IError( "Unable to finish reading from OGG file: end of stream" );
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else if( bytesRead < 0 )
|
|
|
|
{
|
|
|
|
IError( "Unable to read from OGG file: unknown error" );
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
numBytes -= bytesRead;
|
|
|
|
uBuffer += bytesRead;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/// Read in 4k chunks and extract
|
|
|
|
static char trashBuffer[ 4096 ];
|
|
|
|
|
|
|
|
long toRead, i, thisRead, sampleSize = fFakeHeader.fBlockAlign;
|
|
|
|
|
|
|
|
for( ; numBytes > 0; )
|
|
|
|
{
|
|
|
|
/// Read 4k worth of samples
|
|
|
|
toRead = ( sizeof( trashBuffer ) < numBytes * fChannelAdjust ) ? sizeof( trashBuffer ) : numBytes * fChannelAdjust;
|
|
|
|
|
|
|
|
|
|
|
|
thisRead = ov_read( fOggFile, (char *)trashBuffer, toRead, 0, bytesPerSample, isSigned, &currSection );
|
|
|
|
if( thisRead < 0 )
|
|
|
|
return false;
|
|
|
|
|
|
|
|
/// Copy every other sample out
|
|
|
|
int sampleOffset = (fChannelOffset == 1) ? sampleSize : 0;
|
|
|
|
for (i = 0; i < thisRead; i += sampleSize * 2)
|
|
|
|
{
|
|
|
|
memcpy(buffer, &trashBuffer[i + sampleOffset], sampleSize);
|
|
|
|
buffer = (void*)((UInt8*)buffer + sampleSize);
|
|
|
|
|
|
|
|
numBytes -= sampleSize;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// plNetClientApp::StaticDebugMsg("Ogg Read, t=%f, end", hsTimer::GetSeconds());
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
UInt32 plOGGCodec::NumBytesLeft( void )
|
|
|
|
{
|
|
|
|
if(!IsValid())
|
|
|
|
{
|
|
|
|
hsAssert( false, "GetHeader() called on an invalid OGG file" );
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (UInt32)(( fDataSize - ( ov_pcm_tell( fOggFile ) * fHeader.fBlockAlign ) ) / fChannelAdjust);
|
|
|
|
}
|