1341 lines
34 KiB
C++
1341 lines
34 KiB
C++
|
|
#include "fmod_settings.h"
|
||
|
|
|
||
|
|
#include "fmod_channel_real.h"
|
||
|
|
#include "fmod_channel_software.h"
|
||
|
|
#include "fmod_channelpool.h"
|
||
|
|
#include "fmod_downmix.h"
|
||
|
|
#include "fmod_dspi.h"
|
||
|
|
#include "fmod_dsp_resampler_linear.h"
|
||
|
|
#include "fmod_localcriticalsection.h"
|
||
|
|
#include "fmod_memory.h"
|
||
|
|
#include "fmod_output_software.h"
|
||
|
|
#include "fmod_outputi.h"
|
||
|
|
#include "fmod_soundi.h"
|
||
|
|
#include "fmod_string.h"
|
||
|
|
|
||
|
|
#ifdef PLATFORM_PS3_SPU
|
||
|
|
#include <cell/dma.h>
|
||
|
|
#include "fmod_systemi_spu.h"
|
||
|
|
#include "fmod_spu_printf.h"
|
||
|
|
#include "fmod_common_spu.h"
|
||
|
|
#else
|
||
|
|
#include "fmod_systemi.h"
|
||
|
|
#endif
|
||
|
|
|
||
|
|
namespace FMOD
|
||
|
|
{
|
||
|
|
|
||
|
|
/*
|
||
|
|
[
|
||
|
|
[DESCRIPTION]
|
||
|
|
|
||
|
|
[PARAMETERS]
|
||
|
|
|
||
|
|
[RETURN_VALUE]
|
||
|
|
FMOD_OK
|
||
|
|
|
||
|
|
[REMARKS]
|
||
|
|
|
||
|
|
[PLATFORMS]
|
||
|
|
|
||
|
|
[SEE_ALSO]
|
||
|
|
]
|
||
|
|
*/
|
||
|
|
Output::Output()
|
||
|
|
{
|
||
|
|
mEnumerated = false;
|
||
|
|
mPolling = false;
|
||
|
|
mSystem = 0;
|
||
|
|
mChannelPool3D = 0;
|
||
|
|
mMixAheadBlocks = 0;
|
||
|
|
mMusicChannelGroup = 0;
|
||
|
|
mNum2DChannelsFromCaps = 0;
|
||
|
|
mNum3DChannelsFromCaps = 0;
|
||
|
|
mTotalChannelsFromCaps = 0;
|
||
|
|
mDSPTick = 1;
|
||
|
|
|
||
|
|
#ifdef FMOD_SUPPORT_RECORDING
|
||
|
|
mRecordEnumerated = false;
|
||
|
|
mRecordNumActive = 0;
|
||
|
|
mRecordInfoForResampler = 0;
|
||
|
|
|
||
|
|
FMOD_OS_CriticalSection_Create(&mRecordInfoCrit);
|
||
|
|
mRecordInfoHead.initNode();
|
||
|
|
#endif
|
||
|
|
}
|
||
|
|
|
||
|
|
|
||
|
|
/*
|
||
|
|
[
|
||
|
|
[DESCRIPTION]
|
||
|
|
|
||
|
|
[PARAMETERS]
|
||
|
|
|
||
|
|
[RETURN_VALUE]
|
||
|
|
FMOD_OK
|
||
|
|
|
||
|
|
[REMARKS]
|
||
|
|
|
||
|
|
[PLATFORMS]
|
||
|
|
|
||
|
|
[SEE_ALSO]
|
||
|
|
]
|
||
|
|
*/
|
||
|
|
FMOD_RESULT Output::release()
|
||
|
|
{
|
||
|
|
if (mDescription.close)
|
||
|
|
{
|
||
|
|
#ifdef FMOD_SUPPORT_SOFTWARE
|
||
|
|
readfrommixer = Output::mixCallback; /* Reset 'read only' variable in FMOD_OUTPUT_STATE in case the user changed it. */
|
||
|
|
#else
|
||
|
|
readfrommixer = 0;
|
||
|
|
#endif
|
||
|
|
|
||
|
|
mDescription.close(this);
|
||
|
|
}
|
||
|
|
|
||
|
|
#ifdef FMOD_SUPPORT_RECORDING
|
||
|
|
FMOD_OS_CriticalSection_Free(mRecordInfoCrit);
|
||
|
|
#endif
|
||
|
|
|
||
|
|
FMOD_Memory_Free(this);
|
||
|
|
return FMOD_OK;
|
||
|
|
}
|
||
|
|
|
||
|
|
|
||
|
|
#ifdef FMOD_SUPPORT_RECORDING
|
||
|
|
|
||
|
|
/*
|
||
|
|
[
|
||
|
|
[DESCRIPTION]
|
||
|
|
|
||
|
|
[PARAMETERS]
|
||
|
|
|
||
|
|
[RETURN_VALUE]
|
||
|
|
|
||
|
|
[REMARKS]
|
||
|
|
|
||
|
|
[SEE_ALSO]
|
||
|
|
]
|
||
|
|
*/
|
||
|
|
FMOD_RESULT Output::recordRead(FMOD_RECORDING_INFO *recordinfo, float *inbuffer, float *outbuffer, unsigned int length, int inchannels, int outchannels)
|
||
|
|
{
|
||
|
|
FMOD_RESULT result = FMOD_OK;
|
||
|
|
char *rptr1 = NULL;
|
||
|
|
char *rptr2 = NULL;
|
||
|
|
float *destptr = outbuffer;
|
||
|
|
unsigned int rlen1 = 0;
|
||
|
|
unsigned int rlen2 = 0;
|
||
|
|
unsigned int roffsetbytes = 0;
|
||
|
|
unsigned int rlenbytes = 0;
|
||
|
|
unsigned int rblockalign = 0;
|
||
|
|
|
||
|
|
SoundI::getBytesFromSamples(recordinfo->mRecordLastCursorPos, &roffsetbytes, outchannels, recordinfo->mRecordFormat);
|
||
|
|
SoundI::getBytesFromSamples(length, &rlenbytes, outchannels, recordinfo->mRecordFormat);
|
||
|
|
SoundI::getBytesFromSamples(1, &rblockalign, outchannels, recordinfo->mRecordFormat);
|
||
|
|
|
||
|
|
result = mDescription.record_lock((FMOD_OUTPUT_STATE *)this, recordinfo, roffsetbytes, rlenbytes, (void **)&rptr1, (void **)&rptr2, &rlen1, &rlen2);
|
||
|
|
if (result != FMOD_OK)
|
||
|
|
{
|
||
|
|
return result;
|
||
|
|
}
|
||
|
|
|
||
|
|
/*
|
||
|
|
PCM8 is recorded in unsigned 8-bit, so convert it to signed 8-bit for FMOD
|
||
|
|
*/
|
||
|
|
if (recordinfo->mRecordFormat == FMOD_SOUND_FORMAT_PCM8)
|
||
|
|
{
|
||
|
|
unsigned char *rptr = NULL;
|
||
|
|
|
||
|
|
if (rptr1 && rlen1)
|
||
|
|
{
|
||
|
|
rptr = (unsigned char *)rptr1;
|
||
|
|
|
||
|
|
for (unsigned int count = 0; count < rlen1; count++)
|
||
|
|
{
|
||
|
|
*rptr++ ^= 128;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
if (rptr2 && rlen2)
|
||
|
|
{
|
||
|
|
rptr = (unsigned char *)rptr2;
|
||
|
|
|
||
|
|
for (unsigned int count = 0; count < rlen2; count++)
|
||
|
|
{
|
||
|
|
*rptr++ ^= 128;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
/*
|
||
|
|
Copy data from hardware and convert to float.
|
||
|
|
*/
|
||
|
|
if (rptr1 && rlen1)
|
||
|
|
{
|
||
|
|
DSPI::convert(destptr, rptr1, FMOD_SOUND_FORMAT_PCMFLOAT, recordinfo->mRecordFormat, (rlen1 / rblockalign) * outchannels, 1, 1, 1.0f);
|
||
|
|
destptr += ((rlen1 / rblockalign) * outchannels);
|
||
|
|
}
|
||
|
|
if (rptr2 && rlen2)
|
||
|
|
{
|
||
|
|
DSPI::convert(destptr, rptr2, FMOD_SOUND_FORMAT_PCMFLOAT, recordinfo->mRecordFormat, (rlen2 / rblockalign) * outchannels, 1, 1, 1.0f);
|
||
|
|
}
|
||
|
|
|
||
|
|
if (mDescription.record_unlock)
|
||
|
|
{
|
||
|
|
result = mDescription.record_unlock((FMOD_OUTPUT_STATE *)this, recordinfo, rptr1, rptr2, rlen1, rlen2);
|
||
|
|
if (result != FMOD_OK)
|
||
|
|
{
|
||
|
|
return result;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
recordinfo->mRecordLastCursorPos += length;
|
||
|
|
if (recordinfo->mRecordLastCursorPos >= recordinfo->mRecordBufferLength)
|
||
|
|
{
|
||
|
|
recordinfo->mRecordLastCursorPos -= recordinfo->mRecordBufferLength;
|
||
|
|
}
|
||
|
|
|
||
|
|
return FMOD_OK;
|
||
|
|
}
|
||
|
|
|
||
|
|
|
||
|
|
/*
|
||
|
|
[
|
||
|
|
[DESCRIPTION]
|
||
|
|
|
||
|
|
[PARAMETERS]
|
||
|
|
|
||
|
|
[RETURN_VALUE]
|
||
|
|
|
||
|
|
[REMARKS]
|
||
|
|
|
||
|
|
[SEE_ALSO]
|
||
|
|
]
|
||
|
|
*/
|
||
|
|
FMOD_RESULT F_CALLBACK Output::recordResamplerReadCallback(FMOD_DSP_STATE *dsp_state, float *inbuffer, float *outbuffer, unsigned int length, int inchannels, int outchannels)
|
||
|
|
{
|
||
|
|
DSPResampler *resampler = (DSPResampler *)dsp_state;
|
||
|
|
Output *output = NULL;
|
||
|
|
|
||
|
|
resampler->getUserData((void **)&output);
|
||
|
|
|
||
|
|
return output->recordRead(output->mRecordInfoForResampler, inbuffer, outbuffer, length, inchannels, outchannels);
|
||
|
|
}
|
||
|
|
|
||
|
|
|
||
|
|
/*
|
||
|
|
[
|
||
|
|
[DESCRIPTION]
|
||
|
|
|
||
|
|
[PARAMETERS]
|
||
|
|
|
||
|
|
[RETURN_VALUE]
|
||
|
|
|
||
|
|
[REMARKS]
|
||
|
|
|
||
|
|
[SEE_ALSO]
|
||
|
|
]
|
||
|
|
*/
|
||
|
|
FMOD_RESULT Output::recordFill(FMOD_RECORDING_INFO *recordinfo, unsigned int length)
|
||
|
|
{
|
||
|
|
FMOD_RESULT result;
|
||
|
|
int channels;
|
||
|
|
char *pptr1 = 0, *pptr2 = 0;
|
||
|
|
unsigned int plen1 = 0, plen2 = 0, poffsetbytes, soundlength, plength, plengthbytes, pblockalign, totalplength;
|
||
|
|
FMOD_SOUND_FORMAT pformat;
|
||
|
|
|
||
|
|
result = recordinfo->mRecordSound->getFormat(0, &pformat, &channels, 0);
|
||
|
|
if (result != FMOD_OK)
|
||
|
|
{
|
||
|
|
return result;
|
||
|
|
}
|
||
|
|
result = recordinfo->mRecordSound->getLength(&soundlength, FMOD_TIMEUNIT_PCM);
|
||
|
|
if (result != FMOD_OK)
|
||
|
|
{
|
||
|
|
return result;
|
||
|
|
}
|
||
|
|
result = SoundI::getBytesFromSamples(1, &pblockalign, channels, pformat);
|
||
|
|
if (result != FMOD_OK)
|
||
|
|
{
|
||
|
|
return result;
|
||
|
|
}
|
||
|
|
|
||
|
|
totalplength = (unsigned int)((length * recordinfo->mRecordSound->mDefaultFrequency) / recordinfo->mRecordRate);
|
||
|
|
while (totalplength)
|
||
|
|
{
|
||
|
|
/*
|
||
|
|
Work in blocks so we dont overflow our record resample buffer.
|
||
|
|
*/
|
||
|
|
plength = totalplength;
|
||
|
|
if (plength > recordinfo->mRecordTempBufferLength)
|
||
|
|
{
|
||
|
|
plength = recordinfo->mRecordTempBufferLength;
|
||
|
|
}
|
||
|
|
|
||
|
|
SoundI::getBytesFromSamples(recordinfo->mRecordOffset, &poffsetbytes, channels, pformat);
|
||
|
|
SoundI::getBytesFromSamples(plength, &plengthbytes, channels, pformat);
|
||
|
|
|
||
|
|
/*
|
||
|
|
Fill one block into the temporary float buffer.
|
||
|
|
*/
|
||
|
|
if (recordinfo->mRecordResamplerDSP)
|
||
|
|
{
|
||
|
|
mRecordInfoForResampler = recordinfo;
|
||
|
|
result = recordinfo->mRecordResamplerDSP->read(&recordinfo->mRecordTempBuffer, &channels, &plength, (FMOD_SPEAKERMODE)0, channels, mDSPTick);
|
||
|
|
if (result != FMOD_OK)
|
||
|
|
{
|
||
|
|
return result;
|
||
|
|
}
|
||
|
|
mDSPTick++;
|
||
|
|
}
|
||
|
|
else
|
||
|
|
{
|
||
|
|
result = recordRead(recordinfo, recordinfo->mRecordTempBuffer, recordinfo->mRecordTempBuffer, plength, channels, channels);
|
||
|
|
if (result != FMOD_OK)
|
||
|
|
{
|
||
|
|
return result;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
/*
|
||
|
|
Now lock the destination sample and write to it.
|
||
|
|
*/
|
||
|
|
result = recordinfo->mRecordSound->lock(poffsetbytes, plengthbytes, (void **)&pptr1, (void **)&pptr2, &plen1, &plen2);
|
||
|
|
if (result != FMOD_OK)
|
||
|
|
{
|
||
|
|
return result;
|
||
|
|
}
|
||
|
|
|
||
|
|
/*
|
||
|
|
Convert from float to the destination sound format, and handle wrapping.
|
||
|
|
*/
|
||
|
|
{
|
||
|
|
float *srcptr = recordinfo->mRecordTempBuffer;
|
||
|
|
|
||
|
|
if (pptr1 && plen1)
|
||
|
|
{
|
||
|
|
DSPI::convert(pptr1, srcptr, pformat, FMOD_SOUND_FORMAT_PCMFLOAT, (plen1 / pblockalign) * channels, 1, 1, 1.0f);
|
||
|
|
srcptr += ((plen1 / pblockalign) * channels);
|
||
|
|
}
|
||
|
|
if (pptr2 && plen2)
|
||
|
|
{
|
||
|
|
DSPI::convert(pptr2, srcptr, pformat, FMOD_SOUND_FORMAT_PCMFLOAT, (plen2 / pblockalign) * channels, 1, 1, 1.0f);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
result = recordinfo->mRecordSound->unlock(pptr1, pptr2, plen1, plen2);
|
||
|
|
if (result != FMOD_OK)
|
||
|
|
{
|
||
|
|
return result;
|
||
|
|
}
|
||
|
|
|
||
|
|
/*
|
||
|
|
Now that the data has been copied, update the destination sound's cursor.
|
||
|
|
*/
|
||
|
|
recordinfo->mRecordOffset += plength;
|
||
|
|
|
||
|
|
/*
|
||
|
|
Stop the recording if it reaches the end of the sample.
|
||
|
|
*/
|
||
|
|
if (plen2 || recordinfo->mRecordOffset >= (int)soundlength)
|
||
|
|
{
|
||
|
|
if (recordinfo->mRecordLoop)
|
||
|
|
{
|
||
|
|
recordinfo->mRecordOffset -= soundlength;
|
||
|
|
if (recordinfo->mRecordOffset < 0)
|
||
|
|
{
|
||
|
|
recordinfo->mRecordOffset = 0;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
else
|
||
|
|
{
|
||
|
|
/*
|
||
|
|
Mark this recording device for stopping in system update
|
||
|
|
*/
|
||
|
|
recordinfo->mRecordFinished = true;
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
totalplength -= plength;
|
||
|
|
}
|
||
|
|
|
||
|
|
return FMOD_OK;
|
||
|
|
}
|
||
|
|
|
||
|
|
/*
|
||
|
|
[
|
||
|
|
[DESCRIPTION]
|
||
|
|
|
||
|
|
[PARAMETERS]
|
||
|
|
|
||
|
|
[RETURN_VALUE]
|
||
|
|
FMOD_OK
|
||
|
|
|
||
|
|
[REMARKS]
|
||
|
|
|
||
|
|
[PLATFORMS]
|
||
|
|
|
||
|
|
[SEE_ALSO]
|
||
|
|
]
|
||
|
|
*/
|
||
|
|
FMOD_RESULT Output::recordUpdate()
|
||
|
|
{
|
||
|
|
FMOD_RESULT result = FMOD_OK;
|
||
|
|
FMOD_RECORDING_INFO *current = NULL;
|
||
|
|
LocalCriticalSection recordInfoCrit(mRecordInfoCrit, true);
|
||
|
|
|
||
|
|
if (!mDescription.record_getposition)
|
||
|
|
{
|
||
|
|
return FMOD_OK;
|
||
|
|
}
|
||
|
|
|
||
|
|
current = SAFE_CAST(FMOD_RECORDING_INFO, mRecordInfoHead.getNext());
|
||
|
|
while (current != &mRecordInfoHead)
|
||
|
|
{
|
||
|
|
unsigned int cursorpos = 0;
|
||
|
|
int length = 0;
|
||
|
|
FMOD_RECORDING_INFO *next = SAFE_CAST(FMOD_RECORDING_INFO, current->getNext());
|
||
|
|
|
||
|
|
if (!current->mRecordBufferLength || current->mRecordFinished)
|
||
|
|
{
|
||
|
|
current = next;
|
||
|
|
continue;
|
||
|
|
}
|
||
|
|
|
||
|
|
result = mDescription.record_getposition((FMOD_OUTPUT_STATE *)this, current, &cursorpos);
|
||
|
|
if (result != FMOD_OK)
|
||
|
|
{
|
||
|
|
return result;
|
||
|
|
}
|
||
|
|
|
||
|
|
length = cursorpos - current->mRecordLastCursorPos;
|
||
|
|
if (length < 0)
|
||
|
|
{
|
||
|
|
length += current->mRecordBufferLength;
|
||
|
|
}
|
||
|
|
if (length < 0 || length > (int)current->mRecordBufferLength) /* Handle bad values coming back from getposition. */
|
||
|
|
{
|
||
|
|
length = 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
if (current->mRecordResamplerDSP)
|
||
|
|
{
|
||
|
|
int blocklength = current->mRecordResamplerDSP->mDescription.mResamplerBlockLength;
|
||
|
|
|
||
|
|
/*
|
||
|
|
We need at least 3 blocks of data. The resampler double buffer + 1 due to
|
||
|
|
the read request at end of first resample buffer.
|
||
|
|
*/
|
||
|
|
if (length < blocklength * 3)
|
||
|
|
{
|
||
|
|
current = next;
|
||
|
|
continue;
|
||
|
|
}
|
||
|
|
|
||
|
|
length /= blocklength; /* Round length to record block size. */
|
||
|
|
length *= blocklength;
|
||
|
|
|
||
|
|
length -= blocklength; /* Fill output record sample with n - 1 blocks, otherwise resampler will pull more data that we have. */
|
||
|
|
}
|
||
|
|
else
|
||
|
|
{
|
||
|
|
/*
|
||
|
|
We need at least some data to record
|
||
|
|
*/
|
||
|
|
if (length <= 0)
|
||
|
|
{
|
||
|
|
current = next;
|
||
|
|
continue;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
result = recordFill(current, length);
|
||
|
|
if (result != FMOD_OK)
|
||
|
|
{
|
||
|
|
return result;
|
||
|
|
}
|
||
|
|
|
||
|
|
current = next;
|
||
|
|
}
|
||
|
|
|
||
|
|
return FMOD_OK;
|
||
|
|
}
|
||
|
|
|
||
|
|
|
||
|
|
/*
|
||
|
|
[
|
||
|
|
[DESCRIPTION]
|
||
|
|
|
||
|
|
[PARAMETERS]
|
||
|
|
|
||
|
|
[RETURN_VALUE]
|
||
|
|
FMOD_OK
|
||
|
|
|
||
|
|
[REMARKS]
|
||
|
|
|
||
|
|
[PLATFORMS]
|
||
|
|
|
||
|
|
[SEE_ALSO]
|
||
|
|
]
|
||
|
|
*/
|
||
|
|
FMOD_RESULT Output::recordGetInfo(int id, FMOD_RECORDING_INFO **info)
|
||
|
|
{
|
||
|
|
FMOD_RESULT result = FMOD_OK;
|
||
|
|
FMOD_RECORDING_INFO *current = NULL;
|
||
|
|
FMOD_GUID guid = {0};
|
||
|
|
|
||
|
|
if (!info)
|
||
|
|
{
|
||
|
|
return FMOD_ERR_INVALID_PARAM;
|
||
|
|
}
|
||
|
|
|
||
|
|
*info = 0;
|
||
|
|
|
||
|
|
result = mSystem->getRecordDriverInfo(id, NULL, 0, &guid);
|
||
|
|
CHECK_RESULT(result);
|
||
|
|
|
||
|
|
current = SAFE_CAST(FMOD_RECORDING_INFO, mRecordInfoHead.getNext());
|
||
|
|
while (current != &mRecordInfoHead)
|
||
|
|
{
|
||
|
|
FMOD_RECORDING_INFO *next = SAFE_CAST(FMOD_RECORDING_INFO, current->getNext());
|
||
|
|
|
||
|
|
if (FMOD_memcmp(&guid, ¤t->mRecordGUID, sizeof(FMOD_GUID)) == 0)
|
||
|
|
{
|
||
|
|
*info = current;
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
current = next;
|
||
|
|
}
|
||
|
|
|
||
|
|
return FMOD_OK;
|
||
|
|
}
|
||
|
|
|
||
|
|
/*
|
||
|
|
[
|
||
|
|
[DESCRIPTION]
|
||
|
|
|
||
|
|
[PARAMETERS]
|
||
|
|
|
||
|
|
[RETURN_VALUE]
|
||
|
|
|
||
|
|
[REMARKS]
|
||
|
|
|
||
|
|
[SEE_ALSO]
|
||
|
|
]
|
||
|
|
*/
|
||
|
|
FMOD_RESULT Output::recordStopAll(bool finishedonly)
|
||
|
|
{
|
||
|
|
FMOD_RECORDING_INFO *currentrecordinfo = NULL;
|
||
|
|
|
||
|
|
currentrecordinfo = SAFE_CAST(FMOD_RECORDING_INFO, mRecordInfoHead.getNext());
|
||
|
|
while (currentrecordinfo != &mRecordInfoHead)
|
||
|
|
{
|
||
|
|
FMOD_RECORDING_INFO *next = SAFE_CAST(FMOD_RECORDING_INFO, currentrecordinfo->getNext());
|
||
|
|
|
||
|
|
if (!finishedonly || currentrecordinfo->mRecordFinished)
|
||
|
|
{
|
||
|
|
recordStop(currentrecordinfo);
|
||
|
|
}
|
||
|
|
|
||
|
|
currentrecordinfo = next;
|
||
|
|
}
|
||
|
|
|
||
|
|
return FMOD_OK;
|
||
|
|
}
|
||
|
|
|
||
|
|
|
||
|
|
/*
|
||
|
|
[
|
||
|
|
[DESCRIPTION]
|
||
|
|
|
||
|
|
[PARAMETERS]
|
||
|
|
|
||
|
|
[RETURN_VALUE]
|
||
|
|
|
||
|
|
[REMARKS]
|
||
|
|
|
||
|
|
[SEE_ALSO]
|
||
|
|
]
|
||
|
|
*/
|
||
|
|
FMOD_RESULT Output::recordStop(FMOD_RECORDING_INFO *recordinfo)
|
||
|
|
{
|
||
|
|
FMOD_RESULT result = FMOD_OK;
|
||
|
|
|
||
|
|
if (!recordinfo)
|
||
|
|
{
|
||
|
|
return FMOD_ERR_INVALID_PARAM;
|
||
|
|
}
|
||
|
|
|
||
|
|
FLOG((FMOD_DEBUG_LEVEL_LOG, __FILE__, __LINE__, "Output::recordStop", "\n"));
|
||
|
|
|
||
|
|
/*
|
||
|
|
Make the record device "dead" by removing it from the list
|
||
|
|
*/
|
||
|
|
FMOD_OS_CriticalSection_Enter(mRecordInfoCrit);
|
||
|
|
{
|
||
|
|
recordinfo->removeNode();
|
||
|
|
mRecordNumActive--;
|
||
|
|
}
|
||
|
|
FMOD_OS_CriticalSection_Leave(mRecordInfoCrit);
|
||
|
|
|
||
|
|
if (mDescription.record_stop)
|
||
|
|
{
|
||
|
|
#ifdef FMOD_SUPPORT_SOFTWARE
|
||
|
|
readfrommixer = Output::mixCallback; /* Reset 'read only' variable in FMOD_OUTPUT_STATE in case the user changed it. */
|
||
|
|
#else
|
||
|
|
readfrommixer = 0;
|
||
|
|
#endif
|
||
|
|
|
||
|
|
result = mDescription.record_stop(this, recordinfo);
|
||
|
|
CHECK_RESULT(result);
|
||
|
|
}
|
||
|
|
|
||
|
|
if (recordinfo->mRecordTempBuffer)
|
||
|
|
{
|
||
|
|
FMOD_Memory_Free(recordinfo->mRecordTempBuffer);
|
||
|
|
recordinfo->mRecordTempBuffer = NULL;
|
||
|
|
recordinfo->mRecordTempBufferLength = 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
if (recordinfo->mRecordResamplerDSP)
|
||
|
|
{
|
||
|
|
if (recordinfo->mRecordResamplerDSP->mResampleBufferMemory)
|
||
|
|
{
|
||
|
|
FMOD_Memory_Free(recordinfo->mRecordResamplerDSP->mResampleBufferMemory);
|
||
|
|
recordinfo->mRecordResamplerDSP->mResampleBufferMemory = NULL;
|
||
|
|
}
|
||
|
|
|
||
|
|
FMOD_Memory_Free(recordinfo->mRecordResamplerDSP);
|
||
|
|
recordinfo->mRecordResamplerDSP = NULL;
|
||
|
|
}
|
||
|
|
|
||
|
|
FMOD_Memory_Free(recordinfo);
|
||
|
|
|
||
|
|
FLOG((FMOD_DEBUG_LEVEL_LOG, __FILE__, __LINE__, "Output::recordStop", "done\n"));
|
||
|
|
|
||
|
|
return FMOD_OK;
|
||
|
|
}
|
||
|
|
|
||
|
|
#endif /* FMOD_SUPPORT_RECORDING */
|
||
|
|
|
||
|
|
#ifdef FMOD_SUPPORT_SOFTWARE
|
||
|
|
/*
|
||
|
|
[
|
||
|
|
[DESCRIPTION]
|
||
|
|
|
||
|
|
[PARAMETERS]
|
||
|
|
|
||
|
|
[RETURN_VALUE]
|
||
|
|
FMOD_OK
|
||
|
|
|
||
|
|
[REMARKS]
|
||
|
|
|
||
|
|
[PLATFORMS]
|
||
|
|
|
||
|
|
[SEE_ALSO]
|
||
|
|
]
|
||
|
|
*/
|
||
|
|
FMOD_RESULT Output::mix(void *buffer, unsigned int numsamples)
|
||
|
|
{
|
||
|
|
FMOD_RESULT result;
|
||
|
|
unsigned int blockalign = 0;
|
||
|
|
unsigned int offset = 0;
|
||
|
|
#ifndef PLATFORM_PS3_SPU
|
||
|
|
LocalCriticalSection criticalsection(mSystem->mDSPCrit);
|
||
|
|
LocalCriticalSection lockdspcriticalsection(mSystem->mDSPLockCrit);
|
||
|
|
#endif
|
||
|
|
FMOD_SOUND_FORMAT outputformat;
|
||
|
|
int outputchannels;
|
||
|
|
int outputchannelsmixer;
|
||
|
|
DSPI *dsphead;
|
||
|
|
|
||
|
|
if (!buffer || !numsamples)
|
||
|
|
{
|
||
|
|
return FMOD_ERR_INVALID_PARAM;
|
||
|
|
}
|
||
|
|
|
||
|
|
#ifdef PLATFORM_PS3_SPU
|
||
|
|
mSystem = &gSystem;
|
||
|
|
|
||
|
|
outputformat = FMOD_SOUND_FORMAT_PCMFLOAT;
|
||
|
|
outputchannelsmixer = 8;
|
||
|
|
#else
|
||
|
|
result = mSystem->getSoftwareFormat(0, &outputformat, &outputchannelsmixer, 0, 0, 0);
|
||
|
|
if (result != FMOD_OK)
|
||
|
|
{
|
||
|
|
return result;
|
||
|
|
}
|
||
|
|
#endif
|
||
|
|
|
||
|
|
outputchannels = outputchannelsmixer;
|
||
|
|
|
||
|
|
#if !defined(PLATFORM_PS3)
|
||
|
|
if (mSystem->mDownmix)
|
||
|
|
{
|
||
|
|
mSystem->mDownmix->getOutputChannels(&outputchannels);
|
||
|
|
}
|
||
|
|
#endif
|
||
|
|
|
||
|
|
#if defined(PLATFORM_PS2)
|
||
|
|
blockalign = 8;
|
||
|
|
#elif defined(PLATFORM_PS3_SPU)
|
||
|
|
blockalign = 32;
|
||
|
|
#else
|
||
|
|
result = SoundI::getBytesFromSamples(1, &blockalign, outputchannels, outputformat);
|
||
|
|
if (result != FMOD_OK)
|
||
|
|
{
|
||
|
|
return result;
|
||
|
|
}
|
||
|
|
#endif
|
||
|
|
|
||
|
|
|
||
|
|
#ifdef PLATFORM_PS3_SPU
|
||
|
|
cellDmaGet((void *)gDMAMemorySoundCard, (uint64_t)gDSPSoundCardMram, gDSPSoundCardSize, TAG1, TID, RID);
|
||
|
|
cellDmaWaitTagStatusAll(1<<TAG1);
|
||
|
|
|
||
|
|
dsphead = (DSPI *)gDMAMemorySoundCard;
|
||
|
|
|
||
|
|
FMOD_PS3_SPU_DSPPointersToLS(dsphead, false);
|
||
|
|
#else
|
||
|
|
|
||
|
|
dsphead = mSystem->mDSPSoundCard;
|
||
|
|
if (!dsphead)
|
||
|
|
{
|
||
|
|
return FMOD_ERR_INVALID_PARAM;
|
||
|
|
}
|
||
|
|
#endif
|
||
|
|
|
||
|
|
#if !defined(PLATFORM_PS3)
|
||
|
|
mSystem->flushDSPConnectionRequests(false);
|
||
|
|
#endif
|
||
|
|
|
||
|
|
#if !defined(PLATFORM_PS2) && !defined(PLATFORM_PS3_SPU)
|
||
|
|
lockdspcriticalsection.enter();
|
||
|
|
criticalsection.enter();
|
||
|
|
#endif
|
||
|
|
|
||
|
|
#ifdef FMOD_SUPPORT_RECORDING
|
||
|
|
if (mRecordNumActive)
|
||
|
|
{
|
||
|
|
recordUpdate();
|
||
|
|
}
|
||
|
|
#endif
|
||
|
|
|
||
|
|
/*
|
||
|
|
================================================================================================
|
||
|
|
Update Mixer
|
||
|
|
================================================================================================
|
||
|
|
*/
|
||
|
|
do
|
||
|
|
{
|
||
|
|
void *outbuffer = (char *)buffer + (offset * blockalign);
|
||
|
|
unsigned int len = numsamples;
|
||
|
|
|
||
|
|
mSystem->mDSPActive = true;
|
||
|
|
{
|
||
|
|
#ifdef FMOD_SUPPORT_MIXER_NONRECURSIVE /* Can't use virtual function on a class DMA'd in from PPU */
|
||
|
|
result = dsphead->run((float **)&outbuffer, &outputchannelsmixer, &len, mSystem->mSpeakerMode, outputchannelsmixer, mDSPTick);
|
||
|
|
|
||
|
|
if (outbuffer != buffer)
|
||
|
|
{
|
||
|
|
FMOD_memcpy(buffer, outbuffer, sizeof(float) * len * outputchannelsmixer);
|
||
|
|
}
|
||
|
|
#else
|
||
|
|
result = dsphead->read(outbuffer, &len, mSystem->mSpeakerMode, outputchannelsmixer, mDSPTick);
|
||
|
|
#endif
|
||
|
|
|
||
|
|
mDSPTick++;
|
||
|
|
}
|
||
|
|
mSystem->mDSPActive = false;
|
||
|
|
|
||
|
|
numsamples -= len;
|
||
|
|
offset += len;
|
||
|
|
}
|
||
|
|
while (numsamples > 0);
|
||
|
|
|
||
|
|
#if !defined(PLATFORM_PS2) && !defined(PLATFORM_PS3_SPU)
|
||
|
|
if (mDescription.postmixcallback)
|
||
|
|
{
|
||
|
|
mDescription.postmixcallback(this);
|
||
|
|
}
|
||
|
|
|
||
|
|
criticalsection.leave();
|
||
|
|
lockdspcriticalsection.leave();
|
||
|
|
#endif
|
||
|
|
|
||
|
|
#if !defined(PLATFORM_PS3)
|
||
|
|
mSystem->mDSPClock.mValue += offset;
|
||
|
|
#endif
|
||
|
|
|
||
|
|
#ifndef PLATFORM_PS3_SPU
|
||
|
|
float delta = (float)mSystem->mDSPBlockSize / (float)mSystem->mOutputRate * 1000.0f; // ms
|
||
|
|
|
||
|
|
FMOD::gGlobal->gDSPClock.mValue += (FMOD_UINT64)(delta * 4294967296.0f);
|
||
|
|
FMOD_OS_Time_GetMs(&FMOD::gGlobal->gDSPClockTimeStamp);
|
||
|
|
#endif
|
||
|
|
|
||
|
|
return FMOD_OK;
|
||
|
|
}
|
||
|
|
#endif
|
||
|
|
|
||
|
|
#ifndef PLATFORM_PS3_SPU
|
||
|
|
/*
|
||
|
|
[
|
||
|
|
[DESCRIPTION]
|
||
|
|
|
||
|
|
[PARAMETERS]
|
||
|
|
|
||
|
|
[RETURN_VALUE]
|
||
|
|
FMOD_OK
|
||
|
|
|
||
|
|
[REMARKS]
|
||
|
|
|
||
|
|
[PLATFORMS]
|
||
|
|
|
||
|
|
[SEE_ALSO]
|
||
|
|
]
|
||
|
|
*/
|
||
|
|
FMOD_RESULT Output::getFreeChannel(FMOD_MODE mode, ChannelReal **realchannel, int numchannels, int numsoundchannels, int *found, bool ignorereserved)
|
||
|
|
{
|
||
|
|
FMOD_RESULT result;
|
||
|
|
|
||
|
|
if (!realchannel)
|
||
|
|
{
|
||
|
|
return FMOD_ERR_INVALID_PARAM;
|
||
|
|
}
|
||
|
|
|
||
|
|
if (mode & FMOD_3D)
|
||
|
|
{
|
||
|
|
if (!mChannelPool3D)
|
||
|
|
{
|
||
|
|
return FMOD_ERR_CHANNEL_ALLOC;
|
||
|
|
}
|
||
|
|
|
||
|
|
result = mChannelPool3D->allocateChannel(realchannel, FMOD_CHANNEL_FREE, numchannels, found, ignorereserved);
|
||
|
|
if (result != FMOD_OK)
|
||
|
|
{
|
||
|
|
return result;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
else
|
||
|
|
{
|
||
|
|
if (!mChannelPool)
|
||
|
|
{
|
||
|
|
return FMOD_ERR_CHANNEL_ALLOC;
|
||
|
|
}
|
||
|
|
|
||
|
|
result = mChannelPool->allocateChannel(realchannel, FMOD_CHANNEL_FREE, numchannels, found, ignorereserved);
|
||
|
|
if (result != FMOD_OK)
|
||
|
|
{
|
||
|
|
return result;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
return FMOD_OK;
|
||
|
|
}
|
||
|
|
#endif
|
||
|
|
|
||
|
|
#ifdef FMOD_SUPPORT_HARDWAREXM
|
||
|
|
/*
|
||
|
|
[API]
|
||
|
|
[
|
||
|
|
[DESCRIPTION]
|
||
|
|
Reserves a physical SPU2 hardware voice so that it is not used by FMOD.
|
||
|
|
|
||
|
|
[PARAMETERS]
|
||
|
|
'voice' Hardware voice number. Values accepted are in the range 0 to 47.
|
||
|
|
'reserved' 1 = reserve voice. 0 = unreserve voice.
|
||
|
|
|
||
|
|
[RETURN_VALUE]
|
||
|
|
FMOD_RESULT
|
||
|
|
|
||
|
|
[REMARKS]
|
||
|
|
|
||
|
|
[PLATFORMS]
|
||
|
|
|
||
|
|
[SEE_ALSO]
|
||
|
|
]
|
||
|
|
*/
|
||
|
|
FMOD_RESULT Output::reserveVoice(int voice, bool reserved)
|
||
|
|
{
|
||
|
|
FMOD_RESULT result;
|
||
|
|
FMOD::ChannelReal *channel;
|
||
|
|
|
||
|
|
result = mChannelPool->getChannel(voice, (FMOD::ChannelReal**)&channel);
|
||
|
|
if (result != FMOD_OK)
|
||
|
|
{
|
||
|
|
return result;
|
||
|
|
}
|
||
|
|
|
||
|
|
return channel->setReserved(reserved);
|
||
|
|
}
|
||
|
|
#endif
|
||
|
|
|
||
|
|
#ifdef FMOD_SUPPORT_SOFTWARE
|
||
|
|
/*
|
||
|
|
[
|
||
|
|
[DESCRIPTION]
|
||
|
|
|
||
|
|
[PARAMETERS]
|
||
|
|
|
||
|
|
[RETURN_VALUE]
|
||
|
|
FMOD_OK
|
||
|
|
|
||
|
|
[REMARKS]
|
||
|
|
|
||
|
|
[PLATFORMS]
|
||
|
|
|
||
|
|
[SEE_ALSO]
|
||
|
|
]
|
||
|
|
*/
|
||
|
|
FMOD_RESULT F_CALLBACK Output::mixCallback(FMOD_OUTPUT_STATE *output, void *buffer, unsigned int length)
|
||
|
|
{
|
||
|
|
Output *out = (Output *)output;
|
||
|
|
|
||
|
|
return out->mix(buffer, length);
|
||
|
|
}
|
||
|
|
#endif
|
||
|
|
|
||
|
|
|
||
|
|
/*
|
||
|
|
[API]
|
||
|
|
[
|
||
|
|
[DESCRIPTION]
|
||
|
|
Called when the user calls System::getNumDrivers.
|
||
|
|
|
||
|
|
[PARAMETERS]
|
||
|
|
'output_state' Pointer to the plugin state. The user can use this variable to access runtime plugin specific variables and plugin writer user data.
|
||
|
|
'numdrivers' Address of a variable to receive the number of output drivers in your plugin.
|
||
|
|
|
||
|
|
[RETURN_VALUE]
|
||
|
|
|
||
|
|
[REMARKS]
|
||
|
|
Remember to return FMOD_OK at the bottom of the function, or an appropriate error code from FMOD_RESULT.<br>
|
||
|
|
Optional. FMOD will assume 0 if this is not specified.
|
||
|
|
|
||
|
|
[PLATFORMS]
|
||
|
|
Win32, Win64, Linux, Linux64, Macintosh, Xbox360, PlayStation 2, PlayStation Portable, PlayStation 3, Wii, Solaris, iPhone
|
||
|
|
|
||
|
|
[SEE_ALSO]
|
||
|
|
System::getNumDrivers
|
||
|
|
System::getDriverInfo
|
||
|
|
FMOD_OUTPUT_GETDRIVERNAMECALLBACK
|
||
|
|
]
|
||
|
|
*/
|
||
|
|
/*
|
||
|
|
FMOD_RESULT F_CALLBACK FMOD_OUTPUT_GETNUMDRIVERSCALLBACK(FMOD_OUTPUT_STATE *output_state, int *numdrivers)
|
||
|
|
{
|
||
|
|
#if 0
|
||
|
|
// here purely for documentation purposes.
|
||
|
|
#endif
|
||
|
|
}
|
||
|
|
*/
|
||
|
|
|
||
|
|
/*
|
||
|
|
[API]
|
||
|
|
[
|
||
|
|
[DESCRIPTION]
|
||
|
|
Called when the user calls System::getDriverInfo.
|
||
|
|
|
||
|
|
[PARAMETERS]
|
||
|
|
'output_state' Pointer to the plugin state. The user can use this variable to access runtime plugin specific variables and plugin writer user data.
|
||
|
|
'id' Index into the total number of outputs possible, provided by the FMOD_OUTPUT_GETNUMDRIVERSCALLBACK callback.
|
||
|
|
'name' Address of a variable to receive the driver name relevant to the index passed in. Fill this in.
|
||
|
|
'namelen' Length of name buffer being passed in by the user.
|
||
|
|
|
||
|
|
[RETURN_VALUE]
|
||
|
|
|
||
|
|
[REMARKS]
|
||
|
|
Remember to return FMOD_OK at the bottom of the function, or an appropriate error code from FMOD_RESULT.
|
||
|
|
|
||
|
|
[PLATFORMS]
|
||
|
|
Win32, Win64, Linux, Linux64, Macintosh, Xbox360, PlayStation 2, PlayStation Portable, PlayStation 3, Wii, Solaris, iPhone
|
||
|
|
|
||
|
|
[SEE_ALSO]
|
||
|
|
System::getDriverInfo
|
||
|
|
System::getNumDrivers
|
||
|
|
FMOD_OUTPUT_GETNUMDRIVERSCALLBACK
|
||
|
|
]
|
||
|
|
*/
|
||
|
|
/*
|
||
|
|
FMOD_RESULT F_CALLBACK FMOD_OUTPUT_GETDRIVERNAMECALLBACK(FMOD_OUTPUT_STATE *output_state, int id, char *name, int namelen)
|
||
|
|
{
|
||
|
|
#if 0
|
||
|
|
// here purely for documentation purposes.
|
||
|
|
#endif
|
||
|
|
}
|
||
|
|
*/
|
||
|
|
|
||
|
|
|
||
|
|
/*
|
||
|
|
[API]
|
||
|
|
[
|
||
|
|
[DESCRIPTION]
|
||
|
|
Called when the user calls System::getDriverCaps.
|
||
|
|
|
||
|
|
[PARAMETERS]
|
||
|
|
'output_state' Pointer to the plugin state. The user can use this variable to access runtime plugin specific variables and plugin writer user data.
|
||
|
|
'id' Index into the total number of outputs possible, provided by the FMOD_OUTPUT_GETNUMDRIVERSCALLBACK callback.
|
||
|
|
'caps' Address of a variable to receive the caps available by this output device. See FMOD_CAPS. Fill this in.
|
||
|
|
|
||
|
|
[RETURN_VALUE]
|
||
|
|
|
||
|
|
[REMARKS]
|
||
|
|
Remember to return FMOD_OK at the bottom of the function, or an appropriate error code from FMOD_RESULT.
|
||
|
|
|
||
|
|
[PLATFORMS]
|
||
|
|
Win32, Win64, Linux, Linux64, Macintosh, Xbox360, PlayStation 2, PlayStation Portable, PlayStation 3, Wii, Solaris, iPhone
|
||
|
|
|
||
|
|
[SEE_ALSO]
|
||
|
|
System::getDriverCaps
|
||
|
|
System::getDriverInfo
|
||
|
|
System::getNumDrivers
|
||
|
|
FMOD_OUTPUT_GETNUMDRIVERSCALLBACK
|
||
|
|
]
|
||
|
|
*/
|
||
|
|
/*
|
||
|
|
FMOD_RESULT F_CALLBACK FMOD_OUTPUT_GETDRIVERCAPSCALLBACK(FMOD_OUTPUT_STATE *output_state, int id, FMOD_CAPS *caps, int *minfrequency, int *maxfrequency, FMOD_SPEAKERMODE *controlpanelspeakermode)
|
||
|
|
{
|
||
|
|
#if 0
|
||
|
|
// here purely for documentation purposes.
|
||
|
|
#endif
|
||
|
|
}
|
||
|
|
*/
|
||
|
|
|
||
|
|
|
||
|
|
/*
|
||
|
|
[API]
|
||
|
|
[
|
||
|
|
[DESCRIPTION]
|
||
|
|
Initialization callback which is called when the user calls System::init.
|
||
|
|
|
||
|
|
[PARAMETERS]
|
||
|
|
'output_state' Pointer to the plugin state. The user can use this variable to access runtime plugin specific variables and plugin writer user data.
|
||
|
|
'selecteddriver' This is the selected driver id that the user chose from calling System::setDriver.
|
||
|
|
'flags' Initialization flags passed in by the user.
|
||
|
|
'outputrate' Output rate selected by the user. If not possible, change the rate to the closest match.
|
||
|
|
'outputchannels' Output channel count selected by the user. For example 1 = mono output. 2 = stereo output.
|
||
|
|
'outputformat' Output format specified by the user. If not possible to support, return FMOD_ERR_FORMAT.
|
||
|
|
'dspbufferlength' Size of the buffer fmod will mix to in one mix update. This value is in PCM samples.
|
||
|
|
'dspnumbuffers' Number of buffers fmod will mix to in a circular fashion. Multiply this by dspbufferlength to get the total size of the output sound buffer to allocate.
|
||
|
|
'extradriverdata' Data passed in by the user specific to this driver. May be used for any purpose.
|
||
|
|
|
||
|
|
[RETURN_VALUE]
|
||
|
|
|
||
|
|
[REMARKS]
|
||
|
|
Remember to return FMOD_OK at the bottom of the function, or an appropriate error code from FMOD_RESULT.
|
||
|
|
|
||
|
|
[PLATFORMS]
|
||
|
|
Win32, Win64, Linux, Linux64, Macintosh, Xbox360, PlayStation 2, PlayStation Portable, PlayStation 3, Wii, Solaris, iPhone
|
||
|
|
|
||
|
|
[SEE_ALSO]
|
||
|
|
FMOD_RESULT
|
||
|
|
System::init
|
||
|
|
System::setDriver
|
||
|
|
]
|
||
|
|
*/
|
||
|
|
/*
|
||
|
|
FMOD_RESULT F_CALLBACK FMOD_OUTPUT_INITCALLBACK(FMOD_OUTPUT_STATE *output_state, int selecteddriver, FMOD_INITFLAGS flags, int *outputrate, int outputchannels, FMOD_SOUND_FORMAT *outputformat, int dspbufferlength, int dspnumbuffers, void *extradriverdata)
|
||
|
|
{
|
||
|
|
#if 0
|
||
|
|
// here purely for documentation purposes.
|
||
|
|
#endif
|
||
|
|
}
|
||
|
|
*/
|
||
|
|
|
||
|
|
|
||
|
|
/*
|
||
|
|
[API]
|
||
|
|
[
|
||
|
|
[DESCRIPTION]
|
||
|
|
Shut down callback which is called when the user calls System::close or System::release. (System::release calls System::close internally)
|
||
|
|
|
||
|
|
[PARAMETERS]
|
||
|
|
'output_state' Pointer to the plugin state. The user can use this variable to access runtime plugin specific variables and plugin writer user data.
|
||
|
|
|
||
|
|
[RETURN_VALUE]
|
||
|
|
|
||
|
|
[REMARKS]
|
||
|
|
Remember to return FMOD_OK at the bottom of the function, or an appropriate error code from FMOD_RESULT.
|
||
|
|
|
||
|
|
[PLATFORMS]
|
||
|
|
Win32, Win64, Linux, Linux64, Macintosh, Xbox360, PlayStation 2, PlayStation Portable, PlayStation 3, Wii, Solaris, iPhone
|
||
|
|
|
||
|
|
[SEE_ALSO]
|
||
|
|
System::release
|
||
|
|
System::close
|
||
|
|
]
|
||
|
|
*/
|
||
|
|
/*
|
||
|
|
FMOD_RESULT F_CALLBACK FMOD_OUTPUT_CLOSECALLBACK(FMOD_OUTPUT_STATE *output_state)
|
||
|
|
{
|
||
|
|
#if 0
|
||
|
|
// here purely for documentation purposes.
|
||
|
|
#endif
|
||
|
|
}
|
||
|
|
*/
|
||
|
|
|
||
|
|
|
||
|
|
/*
|
||
|
|
[API]
|
||
|
|
[
|
||
|
|
[DESCRIPTION]
|
||
|
|
Called when the user calls System::update.
|
||
|
|
|
||
|
|
[PARAMETERS]
|
||
|
|
'output_state' Pointer to the plugin state. The user can use this variable to access runtime plugin specific variables and plugin writer user data.
|
||
|
|
|
||
|
|
[RETURN_VALUE]
|
||
|
|
|
||
|
|
[REMARKS]
|
||
|
|
Remember to return FMOD_OK at the bottom of the function, or an appropriate error code from FMOD_RESULT.
|
||
|
|
|
||
|
|
[PLATFORMS]
|
||
|
|
Win32, Win64, Linux, Linux64, Macintosh, Xbox360, PlayStation 2, PlayStation Portable, PlayStation 3, Wii, Solaris, iPhone
|
||
|
|
|
||
|
|
[SEE_ALSO]
|
||
|
|
]
|
||
|
|
*/
|
||
|
|
/*
|
||
|
|
FMOD_RESULT F_CALLBACK FMOD_OUTPUT_UPDATECALLBACK(FMOD_OUTPUT_STATE *output_state)
|
||
|
|
{
|
||
|
|
#if 0
|
||
|
|
// here purely for documentation purposes.
|
||
|
|
#endif
|
||
|
|
}
|
||
|
|
*/
|
||
|
|
|
||
|
|
|
||
|
|
/*
|
||
|
|
[API]
|
||
|
|
[
|
||
|
|
[DESCRIPTION]
|
||
|
|
Called when the user calls System::getOutputHandle.
|
||
|
|
|
||
|
|
[PARAMETERS]
|
||
|
|
'output_state' Pointer to the plugin state. The user can use this variable to access runtime plugin specific variables and plugin writer user data.
|
||
|
|
'handle' Address of a variable to receieve the current plugin's output 'handle'. This is only if the plugin writer wants to allow the user access to the main handle behind the plugin (for example the file handle in a file writer plugin). The pointer type must be published to the user somehow, as is done in fmod.h.
|
||
|
|
|
||
|
|
[RETURN_VALUE]
|
||
|
|
|
||
|
|
[REMARKS]
|
||
|
|
Remember to return FMOD_OK at the bottom of the function, or an appropriate error code from FMOD_RESULT.
|
||
|
|
|
||
|
|
[PLATFORMS]
|
||
|
|
Win32, Win64, Linux, Linux64, Macintosh, Xbox360, PlayStation 2, PlayStation Portable, PlayStation 3, Wii, Solaris, iPhone
|
||
|
|
|
||
|
|
[SEE_ALSO]
|
||
|
|
]
|
||
|
|
*/
|
||
|
|
/*
|
||
|
|
FMOD_RESULT F_CALLBACK FMOD_OUTPUT_GETHANDLECALLBACK(FMOD_OUTPUT_STATE *output_state, void **handle)
|
||
|
|
{
|
||
|
|
#if 0
|
||
|
|
// here purely for documentation purposes.
|
||
|
|
#endif
|
||
|
|
}
|
||
|
|
*/
|
||
|
|
|
||
|
|
|
||
|
|
/*
|
||
|
|
[API]
|
||
|
|
[
|
||
|
|
[DESCRIPTION]
|
||
|
|
Returns the current PCM offset or playback position for the output stream.<br>
|
||
|
|
Called from the mixer thread, only when the 'polling' member of FMOD_OUTPUT_DESCRIPTION is set to <b>true</b>.<br>
|
||
|
|
The internal FMOD output thread calls this function periodically to determine if it should ask for a block of audio data or not.
|
||
|
|
|
||
|
|
[PARAMETERS]
|
||
|
|
'output_state' Pointer to the plugin state. The user can use this variable to access runtime plugin specific variables and plugin writer user data.
|
||
|
|
'position' Address of a variable that receives the position of the output stream.
|
||
|
|
|
||
|
|
[RETURN_VALUE]
|
||
|
|
|
||
|
|
[REMARKS]
|
||
|
|
|
||
|
|
[PLATFORMS]
|
||
|
|
Win32, Win64, Linux, Linux64, Macintosh, Xbox360, PlayStation 2, PlayStation Portable, PlayStation 3, Wii, Solaris, iPhone
|
||
|
|
|
||
|
|
[SEE_ALSO]
|
||
|
|
FMOD_OUTPUT_DESCRIPTION
|
||
|
|
FMOD_OUTPUT_LOCKCALLBACK
|
||
|
|
FMOD_OUTPUT_UNLOCKCALLBACK
|
||
|
|
]
|
||
|
|
*/
|
||
|
|
/*
|
||
|
|
FMOD_RESULT F_CALLBACK FMOD_OUTPUT_GETPOSITIONCALLBACK(FMOD_OUTPUT_STATE *output_state, unsigned int *pcm)
|
||
|
|
{
|
||
|
|
#if 0
|
||
|
|
// here purely for documentation purposes.
|
||
|
|
#endif
|
||
|
|
}
|
||
|
|
*/
|
||
|
|
|
||
|
|
|
||
|
|
/*
|
||
|
|
[API]
|
||
|
|
[
|
||
|
|
[DESCRIPTION]
|
||
|
|
Called from the mixer thread, only when the 'polling' member of FMOD_OUTPUT_DESCRIPTION is set to true.
|
||
|
|
|
||
|
|
[PARAMETERS]
|
||
|
|
'output_state' Pointer to the plugin state. The user can use this variable to access runtime plugin specific variables and plugin writer user data.
|
||
|
|
'offset' Offset in <i>bytes</i> to the position the caller wants to lock in the sample buffer.
|
||
|
|
'length' Number of <i>bytes</i> the caller want to lock in the sample buffer.
|
||
|
|
'ptr1' Address of a pointer that will point to the first part of the locked data.
|
||
|
|
'ptr2' Address of a pointer that will point to the second part of the locked data. This will be null if the data locked hasn't wrapped at the end of the buffer.
|
||
|
|
'len1' Length of data in <i>bytes</i> that was locked for ptr1
|
||
|
|
'len2' Length of data in <i>bytes</i> that was locked for ptr2. This will be 0 if the data locked hasn't wrapped at the end of the buffer.
|
||
|
|
|
||
|
|
[RETURN_VALUE]
|
||
|
|
|
||
|
|
[REMARKS]
|
||
|
|
|
||
|
|
[PLATFORMS]
|
||
|
|
Win32, Win64, Linux, Linux64, Macintosh, Xbox360, PlayStation 2, PlayStation Portable, PlayStation 3, Wii, Solaris, iPhone
|
||
|
|
|
||
|
|
[SEE_ALSO]
|
||
|
|
FMOD_OUTPUT_DESCRIPTION
|
||
|
|
FMOD_OUTPUT_UNLOCKCALLBACK
|
||
|
|
FMOD_OUTPUT_GETPOSITIONCALLBACK
|
||
|
|
]
|
||
|
|
*/
|
||
|
|
/*
|
||
|
|
FMOD_RESULT F_CALLBACK FMOD_OUTPUT_LOCKCALLBACK(FMOD_OUTPUT_STATE *output_state, unsigned int offset, unsigned int length, void **ptr1, void **ptr2, unsigned int *len1, unsigned int *len2)
|
||
|
|
{
|
||
|
|
#if 0
|
||
|
|
// here purely for documentation purposes.
|
||
|
|
#endif
|
||
|
|
}
|
||
|
|
*/
|
||
|
|
|
||
|
|
|
||
|
|
/*
|
||
|
|
[API]
|
||
|
|
[
|
||
|
|
[DESCRIPTION]
|
||
|
|
Called from the mixer thread, only when the 'polling' member of FMOD_OUTPUT_DESCRIPTION is set to true.
|
||
|
|
|
||
|
|
[PARAMETERS]
|
||
|
|
'output_state' Pointer to the plugin state. The user can use this variable to access runtime plugin specific variables and plugin writer user data.
|
||
|
|
'ptr1' Pointer to the 1st locked portion of sample data, from Sound::lock.
|
||
|
|
'ptr2' Pointer to the 2nd locked portion of sample data, from Sound::lock.
|
||
|
|
'len1' Length of data in <i>bytes</i> that was locked for ptr1
|
||
|
|
'len2' Length of data in <i>bytes</i> that was locked for ptr2. This will be 0 if the data locked hasn't wrapped at the end of the buffer.
|
||
|
|
|
||
|
|
[RETURN_VALUE]
|
||
|
|
|
||
|
|
[REMARKS]
|
||
|
|
This function is normally called after data has been read/written to from Sound::lock. This function will do any post processing nescessary and if needed, send it to sound ram.
|
||
|
|
|
||
|
|
[PLATFORMS]
|
||
|
|
Win32, Win64, Linux, Linux64, Macintosh, Xbox360, PlayStation 2, PlayStation Portable, PlayStation 3, Wii, Solaris, iPhone
|
||
|
|
|
||
|
|
[SEE_ALSO]
|
||
|
|
FMOD_OUTPUT_DESCRIPTION
|
||
|
|
FMOD_OUTPUT_LOCKCALLBACK
|
||
|
|
FMOD_OUTPUT_GETPOSITIONCALLBACK
|
||
|
|
]
|
||
|
|
*/
|
||
|
|
/*
|
||
|
|
FMOD_RESULT F_CALLBACK FMOD_OUTPUT_UNLOCKCALLBACK(FMOD_OUTPUT_STATE *output_state, void *ptr1, void *ptr2, unsigned int len1, unsigned int len2)
|
||
|
|
{
|
||
|
|
#if 0
|
||
|
|
// here purely for documentation purposes.
|
||
|
|
#endif
|
||
|
|
}
|
||
|
|
*/
|
||
|
|
|
||
|
|
|
||
|
|
/*
|
||
|
|
[API]
|
||
|
|
[
|
||
|
|
[DESCRIPTION]
|
||
|
|
Called by the plugin, when the 'polling' member of FMOD_OUTPUT_DESCRIPTION is set to false.<br>
|
||
|
|
Use this function from your own driver irq/timer to read some data from FMOD's DSP engine. All of the resulting output caused by playing sounds and specifying effects by the user will be mixed here and written to the memory provided by the plugin writer.<br>
|
||
|
|
|
||
|
|
[PARAMETERS]
|
||
|
|
'output_state' Pointer to the plugin state. The user can use this variable to access runtime plugin specific variables and plugin writer user data.
|
||
|
|
'buffer' Plugin-writer provided memory for the FMOD Ex mixer to write to.
|
||
|
|
'length' Length of the buffer in samples.
|
||
|
|
|
||
|
|
[RETURN_VALUE]
|
||
|
|
|
||
|
|
[REMARKS]
|
||
|
|
|
||
|
|
[PLATFORMS]
|
||
|
|
Win32, Win64, Linux, Linux64, Macintosh, Xbox360, PlayStation 2, PlayStation Portable, PlayStation 3, Wii, Solaris, iPhone
|
||
|
|
|
||
|
|
[SEE_ALSO]
|
||
|
|
]
|
||
|
|
*/
|
||
|
|
/*
|
||
|
|
FMOD_RESULT F_CALLBACK FMOD_OUTPUT_READFROMMIXER(FMOD_OUTPUT_STATE *output_state, void *buffer, unsigned int length)
|
||
|
|
{
|
||
|
|
#if 0
|
||
|
|
// here purely for documentation purposes.
|
||
|
|
#endif
|
||
|
|
}
|
||
|
|
*/
|
||
|
|
|
||
|
|
/*
|
||
|
|
[
|
||
|
|
[DESCRIPTION]
|
||
|
|
|
||
|
|
[PARAMETERS]
|
||
|
|
|
||
|
|
[RETURN_VALUE]
|
||
|
|
FMOD_OK
|
||
|
|
|
||
|
|
[REMARKS]
|
||
|
|
|
||
|
|
[PLATFORMS]
|
||
|
|
|
||
|
|
[SEE_ALSO]
|
||
|
|
]
|
||
|
|
*/
|
||
|
|
|
||
|
|
#ifdef FMOD_SUPPORT_MEMORYTRACKER
|
||
|
|
|
||
|
|
FMOD_RESULT Output::getMemoryUsedImpl(MemoryTracker *tracker)
|
||
|
|
{
|
||
|
|
if (mChannelPool)
|
||
|
|
{
|
||
|
|
CHECK_RESULT(mChannelPool->getMemoryUsed(tracker));
|
||
|
|
}
|
||
|
|
|
||
|
|
if (mChannelPool3D && (mChannelPool3D != mChannelPool))
|
||
|
|
{
|
||
|
|
CHECK_RESULT(mChannelPool3D->getMemoryUsed(tracker));
|
||
|
|
}
|
||
|
|
|
||
|
|
#ifdef FMOD_SUPPORT_RECORDING
|
||
|
|
FMOD_RECORDING_INFO *current = SAFE_CAST(FMOD_RECORDING_INFO, mRecordInfoHead.getNext());
|
||
|
|
while (current != &mRecordInfoHead)
|
||
|
|
{
|
||
|
|
FMOD_RECORDING_INFO *next = SAFE_CAST(FMOD_RECORDING_INFO, current->getNext());
|
||
|
|
|
||
|
|
if (current->mRecordSound)
|
||
|
|
{
|
||
|
|
CHECK_RESULT(current->mRecordSound->getMemoryUsed(tracker));
|
||
|
|
|
||
|
|
if (current->mRecordTempBuffer)
|
||
|
|
{
|
||
|
|
unsigned int bufferlengthbytes = 0;
|
||
|
|
SoundI::getBytesFromSamples(FMOD_RECORD_TEMPBUFFERSIZE, &bufferlengthbytes, current->mRecordSound->mChannels, FMOD_SOUND_FORMAT_PCMFLOAT);
|
||
|
|
tracker->add(false, FMOD_MEMBITS_RECORDBUFFER, bufferlengthbytes);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
current = next;
|
||
|
|
}
|
||
|
|
#endif
|
||
|
|
|
||
|
|
return FMOD_OK;
|
||
|
|
}
|
||
|
|
|
||
|
|
#endif
|
||
|
|
|
||
|
|
}
|