mirror of
https://github.com/mackron/miniaudio.git
synced 2026-05-05 22:54:03 +02:00
15da9bd7ff
A new callback called `onProp` has been added to
`ma_data_source_vtable`. This replaces the following callbacks:
onGetDataFormat
onGetCursor
onGetLength
onSetLooping
This new callback is for retrieving and setting various properties
relating to the data source. It takes a `prop` parameter which is an ID
for the property being handled, and a `void*` pointer for
property-specific data.
Typically onProp implementations would discriminate on the property type
using a switch. The example below shows how to handle the old callbacks:
switch (prop)
{
// Replaces onGetDataFormat (format/channels/rate).
case MA_DATA_SOURCE_GET_DATA_SOURCE:
{
ma_data_source_data_format* pDataFormat =
(ma_data_source_data_format*)pData;
pDataFormat->format = pCustomDataSource->format;
pDataFormat->channels = pCustomDataSource->channels;
pDataFormat->sampleRate = pCustomDataSource->sampleRate;
return MA_SUCCESS;
}
// Replaces onGetDataFormat (channel map)
case MA_DATA_SOURCE_GET_CHANNEL_MAP:
{
ma_channel_map_init_standard(
ma_standard_channel_map_default,
(ma_channel*)pData,
MA_MAX_CHANNELS,
pCustomDataSource->channels);
return MA_SUCCESS;
}
// Replaces onGetCursor
case MA_DATA_SOURCE_GET_CURSOR:
{
*((ma_uint64*)pData) = pCustomDataSource->cursor;
return MA_SUCCESS;
}
// Replaces onGetLength
case MA_DATA_SOURCE_GET_LENGTH:
{
*((ma_uint64*)pData) = pCustomDataSource->length;
return MA_SUCCESS;
}
// Replaces onSetLooping
case MA_DATA_SOURCE_SET_LOOPING:
{
pCustomDataSource->isLooping = *((ma_bool32*)pData);
return MA_SUCCESS;
}
// Mandatory when MA_DATA_SOURCE_SET_LOOPING is implemented.
case MA_DATA_SOURCE_GET_LOOPING:
{
*((ma_bool32*)pData) = pCustomDataSource->isLooping;
return MA_SUCCESS;
}
// Return MA_NOT_IMPLEMENTED for any ignored properties.
default: return MA_NOT_IMPLEMENTED;
}
Note how the format/channels/rate and channel map properties have been
split across two separate properties, `MA_DATA_SOURCE_GET_DATA_SOURCE`
and `MA_DATA_SOURCE_GET_CHANNEL_MAP`. Along with this change, the
channel map parameters have been removed from
`ma_data_source_get_data_format()` and a new function called
`ma_data_source_get_channel_map()` has been added.
New properties have also been added for handling ranges and loop points.
This allows the data source implementation itself to handle it rather
than miniaudio doing it at a higher level. Where this is useful is if
your data source is a wrapper around another data source and you want to
route ranges and loop points to the internal data source.
The reason for this change is that it allows for properties to be added
without having to break the build due to yet another callback being
added. It also hides away the more niche properties that the majority of
data sources do not care about. For example, rarely does a data source
need to handle the `onSetLooping` callback, yet every data source needed
to add a `NULL` entry to their vtables just for this one extremely niche
property.
See documentation for further details.
Tag: release-notes
Tag: api-change
577 lines
17 KiB
C
577 lines
17 KiB
C
#ifndef miniaudio_libvorbis_c
|
|
#define miniaudio_libvorbis_c
|
|
|
|
#include "miniaudio_libvorbis.h"
|
|
|
|
#if !defined(MA_NO_LIBVORBIS)
|
|
#ifndef OV_EXCLUDE_STATIC_CALLBACKS
|
|
#define OV_EXCLUDE_STATIC_CALLBACKS
|
|
#endif
|
|
#include <vorbis/vorbisfile.h>
|
|
#endif
|
|
|
|
#include <string.h> /* For memset(). */
|
|
#include <assert.h>
|
|
|
|
static size_t ma_libvorbis_ds_sizeof(void)
|
|
{
|
|
return sizeof(ma_libvorbis);
|
|
}
|
|
|
|
static void ma_libvorbis_ds_uninit(ma_data_source* pDataSource)
|
|
{
|
|
ma_libvorbis_uninit((ma_libvorbis*)pDataSource);
|
|
}
|
|
|
|
static ma_result ma_libvorbis_ds_read(ma_data_source* pDataSource, void* pFramesOut, ma_uint64 frameCount, ma_uint64* pFramesRead)
|
|
{
|
|
return ma_libvorbis_read_pcm_frames((ma_libvorbis*)pDataSource, pFramesOut, frameCount, pFramesRead);
|
|
}
|
|
|
|
static ma_result ma_libvorbis_ds_seek(ma_data_source* pDataSource, ma_uint64 frameIndex)
|
|
{
|
|
return ma_libvorbis_seek_to_pcm_frame((ma_libvorbis*)pDataSource, frameIndex);
|
|
}
|
|
|
|
static ma_result ma_libvorbis_ds_prop(ma_data_source* pDataSource, int prop, void* pData)
|
|
{
|
|
ma_libvorbis* pVorbis = (ma_libvorbis*)pDataSource;
|
|
|
|
switch (prop)
|
|
{
|
|
case MA_DATA_SOURCE_GET_DATA_FORMAT:
|
|
{
|
|
ma_data_source_data_format* pDataFormat = (ma_data_source_data_format*)pData;
|
|
|
|
return ma_libvorbis_get_data_format(pVorbis, &pDataFormat->format, &pDataFormat->channels, &pDataFormat->sampleRate, NULL, 0);
|
|
}
|
|
|
|
case MA_DATA_SOURCE_GET_CHANNEL_MAP:
|
|
{
|
|
return ma_libvorbis_get_data_format(pVorbis, NULL, NULL, NULL, (ma_channel*)pData, MA_MAX_CHANNELS);
|
|
}
|
|
|
|
case MA_DATA_SOURCE_GET_CURSOR:
|
|
{
|
|
return ma_libvorbis_get_cursor_in_pcm_frames(pVorbis, (ma_uint64*)pData);
|
|
}
|
|
|
|
case MA_DATA_SOURCE_GET_LENGTH:
|
|
{
|
|
return ma_libvorbis_get_length_in_pcm_frames(pVorbis, (ma_uint64*)pData);
|
|
}
|
|
|
|
default: break;
|
|
}
|
|
|
|
return MA_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static ma_data_source_vtable ma_gDataSourceVTable_libvorbis =
|
|
{
|
|
ma_libvorbis_ds_sizeof,
|
|
ma_libvorbis_ds_uninit,
|
|
NULL, /* onCopy. Copying is not supported. */
|
|
ma_libvorbis_ds_read,
|
|
ma_libvorbis_ds_seek,
|
|
ma_libvorbis_ds_prop
|
|
};
|
|
|
|
|
|
#if !defined(MA_NO_LIBVORBIS)
|
|
static size_t ma_libvorbis_vf_callback__read(void* pBufferOut, size_t size, size_t count, void* pUserData)
|
|
{
|
|
ma_libvorbis* pVorbis = (ma_libvorbis*)pUserData;
|
|
ma_result result;
|
|
size_t bytesToRead;
|
|
size_t bytesRead;
|
|
|
|
/* For consistency with fread(). If `size` of `count` is 0, return 0 immediately without changing anything. */
|
|
if (size == 0 || count == 0) {
|
|
return 0;
|
|
}
|
|
|
|
bytesToRead = size * count;
|
|
result = pVorbis->onRead(pVorbis->pReadSeekTellUserData, pBufferOut, bytesToRead, &bytesRead);
|
|
if (result != MA_SUCCESS) {
|
|
/* Not entirely sure what to return here. What if an error occurs, but some data was read and bytesRead is > 0? */
|
|
return 0;
|
|
}
|
|
|
|
return bytesRead / size;
|
|
}
|
|
|
|
static int ma_libvorbis_vf_callback__seek(void* pUserData, ogg_int64_t offset, int whence)
|
|
{
|
|
ma_libvorbis* pVorbis = (ma_libvorbis*)pUserData;
|
|
ma_result result;
|
|
ma_seek_origin origin;
|
|
|
|
if (whence == SEEK_SET) {
|
|
origin = MA_SEEK_SET;
|
|
} else if (whence == SEEK_END) {
|
|
origin = MA_SEEK_END;
|
|
} else {
|
|
origin = MA_SEEK_CUR;
|
|
}
|
|
|
|
result = pVorbis->onSeek(pVorbis->pReadSeekTellUserData, offset, origin);
|
|
if (result != MA_SUCCESS) {
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static long ma_libvorbis_vf_callback__tell(void* pUserData)
|
|
{
|
|
ma_libvorbis* pVorbis = (ma_libvorbis*)pUserData;
|
|
ma_result result;
|
|
ma_int64 cursor;
|
|
|
|
result = pVorbis->onTell(pVorbis->pReadSeekTellUserData, &cursor);
|
|
if (result != MA_SUCCESS) {
|
|
return -1;
|
|
}
|
|
|
|
return (long)cursor;
|
|
}
|
|
#endif
|
|
|
|
static ma_result ma_libvorbis_init_internal(const ma_decoding_backend_config* pConfig, const ma_allocation_callbacks* pAllocationCallbacks, ma_libvorbis* pVorbis)
|
|
{
|
|
if (pVorbis == NULL) {
|
|
return MA_INVALID_ARGS;
|
|
}
|
|
|
|
memset(pVorbis, 0, sizeof(*pVorbis));
|
|
pVorbis->allocationCallbacks = ma_allocation_callbacks_init_copy(pAllocationCallbacks);
|
|
pVorbis->format = ma_format_f32; /* f32 by default. */
|
|
|
|
if (pConfig != NULL && (pConfig->preferredFormat == ma_format_f32 || pConfig->preferredFormat == ma_format_s16)) {
|
|
pVorbis->format = pConfig->preferredFormat;
|
|
} else {
|
|
/* Getting here means something other than f32 and s16 was specified. Just leave this unset to use the default format. */
|
|
}
|
|
|
|
#if !defined(MA_NO_LIBVORBIS)
|
|
{
|
|
ma_result result;
|
|
ma_data_source_config dataSourceConfig;
|
|
|
|
dataSourceConfig = ma_data_source_config_init();
|
|
dataSourceConfig.pVTable = &ma_gDataSourceVTable_libvorbis;
|
|
|
|
result = ma_data_source_base_init(&dataSourceConfig, &pVorbis->ds);
|
|
if (result != MA_SUCCESS) {
|
|
return result; /* Failed to initialize the base data source. */
|
|
}
|
|
|
|
pVorbis->vf = (OggVorbis_File*)ma_malloc(sizeof(OggVorbis_File), pAllocationCallbacks);
|
|
if (pVorbis->vf == NULL) {
|
|
ma_data_source_base_uninit(&pVorbis->ds);
|
|
return MA_OUT_OF_MEMORY;
|
|
}
|
|
|
|
return MA_SUCCESS;
|
|
}
|
|
#else
|
|
{
|
|
/* libvorbis is disabled. */
|
|
(void)ma_gDataSourceVTable_libvorbis;
|
|
(void)pAllocationCallbacks;
|
|
return MA_NOT_IMPLEMENTED;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
MA_API ma_result ma_libvorbis_init(ma_read_proc onRead, ma_seek_proc onSeek, ma_tell_proc onTell, void* pReadSeekTellUserData, const ma_decoding_backend_config* pConfig, const ma_allocation_callbacks* pAllocationCallbacks, ma_libvorbis* pVorbis)
|
|
{
|
|
ma_result result;
|
|
|
|
(void)pAllocationCallbacks; /* Can't seem to find a way to configure memory allocations in libvorbis. */
|
|
|
|
if (onRead == NULL || onSeek == NULL) {
|
|
return MA_INVALID_ARGS; /* onRead and onSeek are mandatory. */
|
|
}
|
|
|
|
result = ma_libvorbis_init_internal(pConfig, pAllocationCallbacks, pVorbis);
|
|
if (result != MA_SUCCESS) {
|
|
return result;
|
|
}
|
|
|
|
pVorbis->onRead = onRead;
|
|
pVorbis->onSeek = onSeek;
|
|
pVorbis->onTell = onTell;
|
|
pVorbis->pReadSeekTellUserData = pReadSeekTellUserData;
|
|
|
|
#if !defined(MA_NO_LIBVORBIS)
|
|
{
|
|
int libvorbisResult;
|
|
ov_callbacks libvorbisCallbacks;
|
|
|
|
/* We can now initialize the vorbis decoder. This must be done after we've set up the callbacks. */
|
|
libvorbisCallbacks.read_func = ma_libvorbis_vf_callback__read;
|
|
libvorbisCallbacks.seek_func = ma_libvorbis_vf_callback__seek;
|
|
libvorbisCallbacks.close_func = NULL;
|
|
libvorbisCallbacks.tell_func = ma_libvorbis_vf_callback__tell;
|
|
|
|
libvorbisResult = ov_open_callbacks(pVorbis, (OggVorbis_File*)pVorbis->vf, NULL, 0, libvorbisCallbacks);
|
|
if (libvorbisResult < 0) {
|
|
ma_data_source_base_uninit(&pVorbis->ds);
|
|
ma_free(pVorbis->vf, pAllocationCallbacks);
|
|
return MA_INVALID_FILE;
|
|
}
|
|
|
|
return MA_SUCCESS;
|
|
}
|
|
#else
|
|
{
|
|
/* libvorbis is disabled. */
|
|
return MA_NOT_IMPLEMENTED;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
MA_API void ma_libvorbis_uninit(ma_libvorbis* pVorbis)
|
|
{
|
|
if (pVorbis == NULL) {
|
|
return;
|
|
}
|
|
|
|
#if !defined(MA_NO_LIBVORBIS)
|
|
{
|
|
ov_clear((OggVorbis_File*)pVorbis->vf);
|
|
}
|
|
#else
|
|
{
|
|
/* libvorbis is disabled. Should never hit this since initialization would have failed. */
|
|
assert(MA_FALSE);
|
|
}
|
|
#endif
|
|
|
|
ma_data_source_base_uninit(&pVorbis->ds);
|
|
ma_free(pVorbis->vf, &pVorbis->allocationCallbacks);
|
|
}
|
|
|
|
MA_API ma_result ma_libvorbis_read_pcm_frames(ma_libvorbis* pVorbis, void* pFramesOut, ma_uint64 frameCount, ma_uint64* pFramesRead)
|
|
{
|
|
if (pFramesRead != NULL) {
|
|
*pFramesRead = 0;
|
|
}
|
|
|
|
if (frameCount == 0) {
|
|
return MA_INVALID_ARGS;
|
|
}
|
|
|
|
if (pVorbis == NULL) {
|
|
return MA_INVALID_ARGS;
|
|
}
|
|
|
|
#if !defined(MA_NO_LIBVORBIS)
|
|
{
|
|
/* We always use floating point format. */
|
|
ma_result result = MA_SUCCESS; /* Must be initialized to MA_SUCCESS. */
|
|
ma_uint64 totalFramesRead;
|
|
ma_format format;
|
|
ma_uint32 channels;
|
|
|
|
ma_libvorbis_get_data_format(pVorbis, &format, &channels, NULL, NULL, 0);
|
|
|
|
totalFramesRead = 0;
|
|
while (totalFramesRead < frameCount) {
|
|
long libvorbisResult;
|
|
ma_uint64 framesToRead;
|
|
ma_uint64 framesRemaining;
|
|
|
|
framesRemaining = (frameCount - totalFramesRead);
|
|
framesToRead = 1024;
|
|
if (framesToRead > framesRemaining) {
|
|
framesToRead = framesRemaining;
|
|
}
|
|
|
|
if (format == ma_format_f32) {
|
|
float** ppFramesF32;
|
|
|
|
libvorbisResult = ov_read_float((OggVorbis_File*)pVorbis->vf, &ppFramesF32, (int)framesToRead, NULL);
|
|
if (libvorbisResult < 0) {
|
|
result = MA_ERROR; /* Error while decoding. */
|
|
break;
|
|
} else {
|
|
/* Frames need to be interleaved. */
|
|
ma_interleave_pcm_frames(format, channels, libvorbisResult, (const void**)ppFramesF32, ma_offset_pcm_frames_ptr(pFramesOut, totalFramesRead, format, channels));
|
|
totalFramesRead += libvorbisResult;
|
|
|
|
if (libvorbisResult == 0) {
|
|
result = MA_AT_END;
|
|
break;
|
|
}
|
|
}
|
|
} else {
|
|
libvorbisResult = ov_read((OggVorbis_File*)pVorbis->vf, (char*)ma_offset_pcm_frames_ptr(pFramesOut, totalFramesRead, format, channels), (int)(framesToRead * ma_get_bytes_per_frame(format, channels)), 0, 2, 1, NULL);
|
|
if (libvorbisResult < 0) {
|
|
result = MA_ERROR; /* Error while decoding. */
|
|
break;
|
|
} else {
|
|
/* Conveniently, there's no need to interleaving when using ov_read(). I'm not sure why ov_read_float() is different in that regard... */
|
|
totalFramesRead += libvorbisResult / ma_get_bytes_per_frame(format, channels);
|
|
|
|
if (libvorbisResult == 0) {
|
|
result = MA_AT_END;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (pFramesRead != NULL) {
|
|
*pFramesRead = totalFramesRead;
|
|
}
|
|
|
|
if (result == MA_SUCCESS && totalFramesRead == 0) {
|
|
result = MA_AT_END;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
#else
|
|
{
|
|
/* libvorbis is disabled. Should never hit this since initialization would have failed. */
|
|
assert(MA_FALSE);
|
|
|
|
(void)pFramesOut;
|
|
(void)frameCount;
|
|
(void)pFramesRead;
|
|
|
|
return MA_NOT_IMPLEMENTED;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
MA_API ma_result ma_libvorbis_seek_to_pcm_frame(ma_libvorbis* pVorbis, ma_uint64 frameIndex)
|
|
{
|
|
if (pVorbis == NULL) {
|
|
return MA_INVALID_ARGS;
|
|
}
|
|
|
|
#if !defined(MA_NO_LIBVORBIS)
|
|
{
|
|
int libvorbisResult = ov_pcm_seek((OggVorbis_File*)pVorbis->vf, (ogg_int64_t)frameIndex);
|
|
if (libvorbisResult != 0) {
|
|
if (libvorbisResult == OV_ENOSEEK) {
|
|
return MA_INVALID_OPERATION; /* Not seekable. */
|
|
} else if (libvorbisResult == OV_EINVAL) {
|
|
return MA_INVALID_ARGS;
|
|
} else {
|
|
return MA_ERROR;
|
|
}
|
|
}
|
|
|
|
return MA_SUCCESS;
|
|
}
|
|
#else
|
|
{
|
|
/* libvorbis is disabled. Should never hit this since initialization would have failed. */
|
|
assert(MA_FALSE);
|
|
|
|
(void)frameIndex;
|
|
|
|
return MA_NOT_IMPLEMENTED;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
MA_API ma_result ma_libvorbis_get_data_format(ma_libvorbis* pVorbis, ma_format* pFormat, ma_uint32* pChannels, ma_uint32* pSampleRate, ma_channel* pChannelMap, size_t channelMapCap)
|
|
{
|
|
/* Defaults for safety. */
|
|
if (pFormat != NULL) {
|
|
*pFormat = ma_format_unknown;
|
|
}
|
|
if (pChannels != NULL) {
|
|
*pChannels = 0;
|
|
}
|
|
if (pSampleRate != NULL) {
|
|
*pSampleRate = 0;
|
|
}
|
|
if (pChannelMap != NULL) {
|
|
memset(pChannelMap, 0, sizeof(*pChannelMap) * channelMapCap);
|
|
}
|
|
|
|
if (pVorbis == NULL) {
|
|
return MA_INVALID_OPERATION;
|
|
}
|
|
|
|
if (pFormat != NULL) {
|
|
*pFormat = pVorbis->format;
|
|
}
|
|
|
|
#if !defined(MA_NO_LIBVORBIS)
|
|
{
|
|
vorbis_info* pInfo = ov_info((OggVorbis_File*)pVorbis->vf, 0);
|
|
if (pInfo == NULL) {
|
|
return MA_INVALID_OPERATION;
|
|
}
|
|
|
|
if (pChannels != NULL) {
|
|
*pChannels = pInfo->channels;
|
|
}
|
|
|
|
if (pSampleRate != NULL) {
|
|
*pSampleRate = pInfo->rate;
|
|
}
|
|
|
|
if (pChannelMap != NULL) {
|
|
ma_channel_map_init_standard(ma_standard_channel_map_vorbis, pChannelMap, channelMapCap, pInfo->channels);
|
|
}
|
|
|
|
return MA_SUCCESS;
|
|
}
|
|
#else
|
|
{
|
|
/* libvorbis is disabled. Should never hit this since initialization would have failed. */
|
|
assert(MA_FALSE);
|
|
return MA_NOT_IMPLEMENTED;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
MA_API ma_result ma_libvorbis_get_cursor_in_pcm_frames(ma_libvorbis* pVorbis, ma_uint64* pCursor)
|
|
{
|
|
if (pCursor == NULL) {
|
|
return MA_INVALID_ARGS;
|
|
}
|
|
|
|
*pCursor = 0; /* Safety. */
|
|
|
|
if (pVorbis == NULL) {
|
|
return MA_INVALID_ARGS;
|
|
}
|
|
|
|
#if !defined(MA_NO_LIBVORBIS)
|
|
{
|
|
ogg_int64_t offset = ov_pcm_tell((OggVorbis_File*)pVorbis->vf);
|
|
if (offset < 0) {
|
|
return MA_INVALID_FILE;
|
|
}
|
|
|
|
*pCursor = (ma_uint64)offset;
|
|
|
|
return MA_SUCCESS;
|
|
}
|
|
#else
|
|
{
|
|
/* libvorbis is disabled. Should never hit this since initialization would have failed. */
|
|
assert(MA_FALSE);
|
|
return MA_NOT_IMPLEMENTED;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
MA_API ma_result ma_libvorbis_get_length_in_pcm_frames(ma_libvorbis* pVorbis, ma_uint64* pLength)
|
|
{
|
|
if (pLength == NULL) {
|
|
return MA_INVALID_ARGS;
|
|
}
|
|
|
|
*pLength = 0; /* Safety. */
|
|
|
|
if (pVorbis == NULL) {
|
|
return MA_INVALID_ARGS;
|
|
}
|
|
|
|
#if !defined(MA_NO_LIBVORBIS)
|
|
{
|
|
/*
|
|
Will work in the supermajority of cases where a file has a single logical bitstream. Concatenated streams
|
|
are much harder to determine the length of since they can have sample rate changes, but they should be
|
|
extremely rare outside of unseekable livestreams anyway.
|
|
*/
|
|
if (ov_streams((OggVorbis_File*)pVorbis->vf) == 1) {
|
|
ogg_int64_t length = ov_pcm_total((OggVorbis_File*)pVorbis->vf, 0);
|
|
if(length != OV_EINVAL) {
|
|
*pLength = (ma_uint64)length;
|
|
} else {
|
|
/* Unseekable. */
|
|
}
|
|
} else {
|
|
/* Concatenated stream. */
|
|
}
|
|
|
|
return MA_SUCCESS;
|
|
}
|
|
#else
|
|
{
|
|
/* libvorbis is disabled. Should never hit this since initialization would have failed. */
|
|
assert(MA_FALSE);
|
|
return MA_NOT_IMPLEMENTED;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
|
|
/*
|
|
The code below defines the vtable that you'll plug into your `ma_decoder_config` object.
|
|
*/
|
|
#if !defined(MA_NO_LIBVORBIS)
|
|
static void ma_decoding_backend_info__libvorbis(void* pUserData, ma_decoding_backend_info* pInfo)
|
|
{
|
|
(void)pUserData;
|
|
|
|
pInfo->pName = "Vorbis";
|
|
pInfo->pLibraryName = "libvorbis";
|
|
pInfo->pVendor = "Xiph.Org";
|
|
pInfo->encodingFormat = ma_encoding_format_vorbis;
|
|
}
|
|
|
|
static ma_result ma_decoding_backend_init__libvorbis(void* pUserData, ma_read_proc onRead, ma_seek_proc onSeek, ma_tell_proc onTell, void* pReadSeekTellUserData, const ma_decoding_backend_config* pConfig, const ma_allocation_callbacks* pAllocationCallbacks, ma_data_source** ppBackend)
|
|
{
|
|
ma_result result;
|
|
ma_libvorbis* pVorbis;
|
|
|
|
(void)pUserData;
|
|
|
|
pVorbis = (ma_libvorbis*)ma_malloc(sizeof(*pVorbis), pAllocationCallbacks);
|
|
if (pVorbis == NULL) {
|
|
return MA_OUT_OF_MEMORY;
|
|
}
|
|
|
|
result = ma_libvorbis_init(onRead, onSeek, onTell, pReadSeekTellUserData, pConfig, pAllocationCallbacks, pVorbis);
|
|
if (result != MA_SUCCESS) {
|
|
ma_free(pVorbis, pAllocationCallbacks);
|
|
return result;
|
|
}
|
|
|
|
*ppBackend = pVorbis;
|
|
|
|
return MA_SUCCESS;
|
|
}
|
|
|
|
static void ma_decoding_backend_uninit__libvorbis(void* pUserData, ma_data_source* pBackend, const ma_allocation_callbacks* pAllocationCallbacks)
|
|
{
|
|
ma_libvorbis* pVorbis = (ma_libvorbis*)pBackend;
|
|
|
|
(void)pUserData;
|
|
|
|
ma_libvorbis_uninit(pVorbis);
|
|
ma_free(pVorbis, pAllocationCallbacks);
|
|
}
|
|
|
|
static ma_decoding_backend_vtable ma_gDecodingBackendVTable_libvorbis =
|
|
{
|
|
ma_decoding_backend_info__libvorbis,
|
|
ma_decoding_backend_init__libvorbis,
|
|
NULL, /* onInitMemory() */
|
|
ma_decoding_backend_uninit__libvorbis
|
|
};
|
|
ma_decoding_backend_vtable* ma_decoding_backend_libvorbis = &ma_gDecodingBackendVTable_libvorbis;
|
|
#else
|
|
ma_decoding_backend_vtable* ma_decoding_backend_libvorbis = NULL;
|
|
#endif
|
|
|
|
MA_API ma_decoding_backend_vtable* ma_libvorbis_get_vtable(void)
|
|
{
|
|
return ma_decoding_backend_libvorbis;
|
|
}
|
|
|
|
#endif /* miniaudio_libvorbis_c */
|