|
@@ -21,6 +21,7 @@
|
|
|
#include "VorbisDecoder.h"
|
|
|
|
|
|
#include <string.h>
|
|
|
+#include <common/config.h>
|
|
|
#include <common/Exception.h>
|
|
|
|
|
|
namespace love
|
|
@@ -32,41 +33,41 @@ namespace lullaby
|
|
|
/**
|
|
|
* CALLBACK FUNCTIONS
|
|
|
**/
|
|
|
- int vorbisClose(void * datasource /* ptr to the data that the vorbis files need*/)
|
|
|
- {
|
|
|
- // Does nothing (handled elsewhere)
|
|
|
- return 1;
|
|
|
+ int vorbisClose(void * datasource /* ptr to the data that the vorbis files need*/)
|
|
|
+ {
|
|
|
+ // Does nothing (handled elsewhere)
|
|
|
+ return 1;
|
|
|
}
|
|
|
|
|
|
- size_t vorbisRead(void * ptr /* ptr to the data that the vorbis files need*/,
|
|
|
- size_t byteSize /* how big a byte is*/,
|
|
|
- size_t sizeToRead /* How much we can read*/,
|
|
|
+ size_t vorbisRead(void * ptr /* ptr to the data that the vorbis files need*/,
|
|
|
+ size_t byteSize /* how big a byte is*/,
|
|
|
+ size_t sizeToRead /* How much we can read*/,
|
|
|
void * datasource /* this is a pointer to the data we passed into ov_open_callbacks (our SOggFile struct*/)
|
|
|
{
|
|
|
- size_t spaceToEOF; // How much more we can read till we hit the EOF marker
|
|
|
- size_t actualSizeToRead; // How much data we are actually going to read from memory
|
|
|
- SOggFile* vorbisData; // Our vorbis data, for the typecast
|
|
|
-
|
|
|
- // Get the data in the right format
|
|
|
- vorbisData = (SOggFile*)datasource;
|
|
|
-
|
|
|
- // Calculate how much we need to read. This can be sizeToRead*byteSize or less depending on how near the EOF marker we are
|
|
|
- spaceToEOF = vorbisData->dataSize - vorbisData->dataRead;
|
|
|
- if ((sizeToRead*byteSize) < spaceToEOF)
|
|
|
- actualSizeToRead = (sizeToRead*byteSize);
|
|
|
- else
|
|
|
- actualSizeToRead = spaceToEOF;
|
|
|
-
|
|
|
- // A simple copy of the data from memory to the datastruct that the vorbis libs will use
|
|
|
- if (actualSizeToRead)
|
|
|
- {
|
|
|
- // Copy the data from the start of the file PLUS how much we have already read in
|
|
|
- memcpy(ptr, (char*)vorbisData->dataPtr + vorbisData->dataRead, actualSizeToRead);
|
|
|
- // Increase by how much we have read by
|
|
|
- vorbisData->dataRead += (actualSizeToRead);
|
|
|
- }
|
|
|
-
|
|
|
- // Return how much we read (in the same way fread would)
|
|
|
+ size_t spaceToEOF; // How much more we can read till we hit the EOF marker
|
|
|
+ size_t actualSizeToRead; // How much data we are actually going to read from memory
|
|
|
+ SOggFile* vorbisData; // Our vorbis data, for the typecast
|
|
|
+
|
|
|
+ // Get the data in the right format
|
|
|
+ vorbisData = (SOggFile*)datasource;
|
|
|
+
|
|
|
+ // Calculate how much we need to read. This can be sizeToRead*byteSize or less depending on how near the EOF marker we are
|
|
|
+ spaceToEOF = vorbisData->dataSize - vorbisData->dataRead;
|
|
|
+ if ((sizeToRead*byteSize) < spaceToEOF)
|
|
|
+ actualSizeToRead = (sizeToRead*byteSize);
|
|
|
+ else
|
|
|
+ actualSizeToRead = spaceToEOF;
|
|
|
+
|
|
|
+ // A simple copy of the data from memory to the datastruct that the vorbis libs will use
|
|
|
+ if (actualSizeToRead)
|
|
|
+ {
|
|
|
+ // Copy the data from the start of the file PLUS how much we have already read in
|
|
|
+ memcpy(ptr, (char*)vorbisData->dataPtr + vorbisData->dataRead, actualSizeToRead);
|
|
|
+ // Increase by how much we have read by
|
|
|
+ vorbisData->dataRead += (actualSizeToRead);
|
|
|
+ }
|
|
|
+
|
|
|
+ // Return how much we read (in the same way fread would)
|
|
|
return actualSizeToRead;
|
|
|
}
|
|
|
|
|
@@ -74,43 +75,43 @@ namespace lullaby
|
|
|
ogg_int64_t offset /*offset from the point we wish to seek to*/,
|
|
|
int whence /*where we want to seek to*/)
|
|
|
{
|
|
|
- size_t spaceToEOF; // How much more we can read till we hit the EOF marker
|
|
|
- ogg_int64_t actualOffset; // How much we can actually offset it by
|
|
|
- SOggFile* vorbisData; // The data we passed in (for the typecast)
|
|
|
-
|
|
|
- // Get the data in the right format
|
|
|
- vorbisData = (SOggFile*) datasource;
|
|
|
-
|
|
|
- // Goto where we wish to seek to
|
|
|
- switch (whence)
|
|
|
- {
|
|
|
- case SEEK_SET: // Seek to the start of the data file
|
|
|
- // Make sure we are not going to the end of the file
|
|
|
- if (vorbisData->dataSize >= offset)
|
|
|
- actualOffset = offset;
|
|
|
- else
|
|
|
- actualOffset = vorbisData->dataSize;
|
|
|
- // Set where we now are
|
|
|
- vorbisData->dataRead = (int)actualOffset;
|
|
|
- break;
|
|
|
- case SEEK_CUR: // Seek from where we are
|
|
|
- // Make sure we dont go past the end
|
|
|
- spaceToEOF = vorbisData->dataSize - vorbisData->dataRead;
|
|
|
- if (offset < spaceToEOF)
|
|
|
- actualOffset = (offset);
|
|
|
- else
|
|
|
- actualOffset = spaceToEOF;
|
|
|
- // Seek from our currrent location
|
|
|
- vorbisData->dataRead += (int)actualOffset;
|
|
|
- break;
|
|
|
- case SEEK_END: // Seek from the end of the file
|
|
|
- vorbisData->dataRead = vorbisData->dataSize+1;
|
|
|
- break;
|
|
|
- default:
|
|
|
- throw love::Exception("Unknown seek command in vorbisSeek\n");
|
|
|
- break;
|
|
|
- };
|
|
|
-
|
|
|
+ size_t spaceToEOF; // How much more we can read till we hit the EOF marker
|
|
|
+ ogg_int64_t actualOffset; // How much we can actually offset it by
|
|
|
+ SOggFile* vorbisData; // The data we passed in (for the typecast)
|
|
|
+
|
|
|
+ // Get the data in the right format
|
|
|
+ vorbisData = (SOggFile*) datasource;
|
|
|
+
|
|
|
+ // Goto where we wish to seek to
|
|
|
+ switch (whence)
|
|
|
+ {
|
|
|
+ case SEEK_SET: // Seek to the start of the data file
|
|
|
+ // Make sure we are not going to the end of the file
|
|
|
+ if (vorbisData->dataSize >= offset)
|
|
|
+ actualOffset = offset;
|
|
|
+ else
|
|
|
+ actualOffset = vorbisData->dataSize;
|
|
|
+ // Set where we now are
|
|
|
+ vorbisData->dataRead = (int)actualOffset;
|
|
|
+ break;
|
|
|
+ case SEEK_CUR: // Seek from where we are
|
|
|
+ // Make sure we dont go past the end
|
|
|
+ spaceToEOF = vorbisData->dataSize - vorbisData->dataRead;
|
|
|
+ if (offset < spaceToEOF)
|
|
|
+ actualOffset = (offset);
|
|
|
+ else
|
|
|
+ actualOffset = spaceToEOF;
|
|
|
+ // Seek from our currrent location
|
|
|
+ vorbisData->dataRead += (int)actualOffset;
|
|
|
+ break;
|
|
|
+ case SEEK_END: // Seek from the end of the file
|
|
|
+ vorbisData->dataRead = vorbisData->dataSize+1;
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ throw love::Exception("Unknown seek command in vorbisSeek\n");
|
|
|
+ break;
|
|
|
+ };
|
|
|
+
|
|
|
return 0;
|
|
|
}
|
|
|
|
|
@@ -127,11 +128,18 @@ namespace lullaby
|
|
|
VorbisDecoder::VorbisDecoder(Data * data, const std::string & ext, int bufferSize, int sampleRate)
|
|
|
: Decoder(data, ext, bufferSize, sampleRate)
|
|
|
{
|
|
|
- // Initialize callbacks
|
|
|
- vorbisCallbacks.close_func = vorbisClose;
|
|
|
- vorbisCallbacks.seek_func = vorbisSeek;
|
|
|
- vorbisCallbacks.read_func = vorbisRead;
|
|
|
+ // Initialize callbacks
|
|
|
+ vorbisCallbacks.close_func = vorbisClose;
|
|
|
+ vorbisCallbacks.seek_func = vorbisSeek;
|
|
|
+ vorbisCallbacks.read_func = vorbisRead;
|
|
|
vorbisCallbacks.tell_func = vorbisTell;
|
|
|
+
|
|
|
+ // Check endianness
|
|
|
+#ifdef LOVE_BIG_ENDIAN
|
|
|
+ endian = 1;
|
|
|
+#else
|
|
|
+ endian = 0;
|
|
|
+#endif
|
|
|
|
|
|
// Initialize OGG file
|
|
|
oggFile.dataPtr = (char *) data->getData();
|
|
@@ -143,7 +151,7 @@ namespace lullaby
|
|
|
throw love::Exception("Could not read Ogg bitstream");
|
|
|
|
|
|
// Get info and comments
|
|
|
- vorbisInfo = ov_info(&handle, -1);
|
|
|
+ vorbisInfo = ov_info(&handle, -1);
|
|
|
vorbisComment = ov_comment(&handle, -1);
|
|
|
}
|
|
|
|
|
@@ -176,7 +184,7 @@ namespace lullaby
|
|
|
{
|
|
|
int bits = this->getBits();
|
|
|
int size = 0;
|
|
|
- int section;
|
|
|
+ int section;
|
|
|
int result;
|
|
|
|
|
|
if(bits == 16)
|
|
@@ -186,7 +194,7 @@ namespace lullaby
|
|
|
|
|
|
while(size < bufferSize)
|
|
|
{
|
|
|
- result = ov_read(&handle, (char*) buffer + size, bufferSize - size, 0, bits, 1, §ion);
|
|
|
+ result = ov_read(&handle, (char*) buffer + size, bufferSize - size, endian, bits, 1, §ion);
|
|
|
|
|
|
if(result > 0)
|
|
|
size += result;
|
|
@@ -194,17 +202,17 @@ namespace lullaby
|
|
|
{
|
|
|
switch(result)
|
|
|
{
|
|
|
- case OV_EREAD:
|
|
|
- throw love::Exception("Vorbis read: Read from media");
|
|
|
- case OV_ENOTVORBIS:
|
|
|
- throw love::Exception("Vorbis read: Not Vorbis data");
|
|
|
- case OV_EVERSION:
|
|
|
- throw love::Exception("Vorbis read: Vorbis version mismatch");
|
|
|
- case OV_EBADHEADER:
|
|
|
- throw love::Exception("Vorbis read: Invalid Vorbis header");
|
|
|
- case OV_EFAULT:
|
|
|
- throw love::Exception("Vorbis read: Internal logic fault (bug or heap/stack corruption)");
|
|
|
- default:
|
|
|
+ case OV_EREAD:
|
|
|
+ throw love::Exception("Vorbis read: Read from media");
|
|
|
+ case OV_ENOTVORBIS:
|
|
|
+ throw love::Exception("Vorbis read: Not Vorbis data");
|
|
|
+ case OV_EVERSION:
|
|
|
+ throw love::Exception("Vorbis read: Vorbis version mismatch");
|
|
|
+ case OV_EBADHEADER:
|
|
|
+ throw love::Exception("Vorbis read: Invalid Vorbis header");
|
|
|
+ case OV_EFAULT:
|
|
|
+ throw love::Exception("Vorbis read: Internal logic fault (bug or heap/stack corruption)");
|
|
|
+ default:
|
|
|
throw love::Exception("Vorbis read: Unknown error");
|
|
|
}
|
|
|
}
|
|
@@ -257,20 +265,20 @@ namespace lullaby
|
|
|
return (result != 0);
|
|
|
}
|
|
|
|
|
|
- int VorbisDecoder::getChannels() const
|
|
|
- {
|
|
|
- return vorbisInfo->channels;
|
|
|
- }
|
|
|
-
|
|
|
- int VorbisDecoder::getBits() const
|
|
|
- {
|
|
|
+ int VorbisDecoder::getChannels() const
|
|
|
+ {
|
|
|
+ return vorbisInfo->channels;
|
|
|
+ }
|
|
|
+
|
|
|
+ int VorbisDecoder::getBits() const
|
|
|
+ {
|
|
|
//return vorbisInfo->bitrate_nominal / 10000;
|
|
|
- return 16;
|
|
|
+ return 16;
|
|
|
}
|
|
|
|
|
|
- int VorbisDecoder::getSampleRate() const
|
|
|
- {
|
|
|
- return vorbisInfo->rate;
|
|
|
+ int VorbisDecoder::getSampleRate() const
|
|
|
+ {
|
|
|
+ return vorbisInfo->rate;
|
|
|
}
|
|
|
|
|
|
} // lullaby
|