2014-10-22 20:11:03 +00:00
|
|
|
#include "movieaudiofactory.hpp"
|
|
|
|
|
2015-04-19 18:14:06 +00:00
|
|
|
#include <extern/osg-ffmpeg-videoplayer/audiodecoder.hpp>
|
|
|
|
#include <extern/osg-ffmpeg-videoplayer/videostate.hpp>
|
2014-10-22 20:11:03 +00:00
|
|
|
|
|
|
|
#include "../mwbase/environment.hpp"
|
|
|
|
#include "../mwbase/soundmanager.hpp"
|
|
|
|
|
|
|
|
#include "sound_decoder.hpp"
|
|
|
|
#include "sound.hpp"
|
|
|
|
|
|
|
|
namespace MWSound
|
|
|
|
{
|
|
|
|
|
|
|
|
class MovieAudioDecoder;
|
2017-12-08 15:00:04 +00:00
|
|
|
class MWSoundDecoderBridge final : public Sound_Decoder
|
2014-10-22 20:11:03 +00:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
MWSoundDecoderBridge(MWSound::MovieAudioDecoder* decoder)
|
2018-10-09 06:21:12 +00:00
|
|
|
: Sound_Decoder(nullptr)
|
2015-04-19 18:07:18 +00:00
|
|
|
, mDecoder(decoder)
|
2014-10-22 20:11:03 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
MWSound::MovieAudioDecoder* mDecoder;
|
|
|
|
|
2017-12-09 19:00:56 +00:00
|
|
|
void open(const std::string &fname) override;
|
2017-12-08 15:00:04 +00:00
|
|
|
void close() override;
|
|
|
|
std::string getName() override;
|
2017-12-09 19:00:56 +00:00
|
|
|
void getInfo(int *samplerate, ChannelConfig *chans, SampleType *type) override;
|
2017-12-08 15:00:04 +00:00
|
|
|
size_t read(char *buffer, size_t bytes) override;
|
|
|
|
size_t getSampleOffset() override;
|
2014-10-22 20:11:03 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
class MovieAudioDecoder : public Video::MovieAudioDecoder
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
MovieAudioDecoder(Video::VideoState *videoState)
|
2017-09-12 04:33:18 +00:00
|
|
|
: Video::MovieAudioDecoder(videoState), mAudioTrack(nullptr)
|
2014-10-22 20:11:03 +00:00
|
|
|
{
|
|
|
|
mDecoderBridge.reset(new MWSoundDecoderBridge(this));
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t getSampleOffset()
|
|
|
|
{
|
2014-10-24 19:19:17 +00:00
|
|
|
ssize_t clock_delay = (mFrameSize-mFramePos) / av_get_channel_layout_nb_channels(mOutputChannelLayout) /
|
|
|
|
av_get_bytes_per_sample(mOutputSampleFormat);
|
2018-10-20 07:09:52 +00:00
|
|
|
return (size_t)(mAudioClock*mAudioContext->sample_rate) - clock_delay;
|
2014-10-22 20:11:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
std::string getStreamName()
|
|
|
|
{
|
2015-04-19 18:07:18 +00:00
|
|
|
return std::string();
|
2014-10-22 20:11:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
// MovieAudioDecoder overrides
|
|
|
|
|
2020-10-16 18:18:54 +00:00
|
|
|
double getAudioClock() override
|
2014-10-22 20:11:03 +00:00
|
|
|
{
|
2018-10-20 07:09:52 +00:00
|
|
|
return (double)getSampleOffset()/(double)mAudioContext->sample_rate -
|
2015-11-30 13:42:51 +00:00
|
|
|
MWBase::Environment::get().getSoundManager()->getTrackTimeDelay(mAudioTrack);
|
2014-10-22 20:11:03 +00:00
|
|
|
}
|
|
|
|
|
2020-10-16 18:18:54 +00:00
|
|
|
void adjustAudioSettings(AVSampleFormat& sampleFormat, uint64_t& channelLayout, int& sampleRate) override
|
2014-10-22 20:11:03 +00:00
|
|
|
{
|
2014-10-25 15:17:57 +00:00
|
|
|
if (sampleFormat == AV_SAMPLE_FMT_U8P || sampleFormat == AV_SAMPLE_FMT_U8)
|
2014-10-22 20:11:03 +00:00
|
|
|
sampleFormat = AV_SAMPLE_FMT_U8;
|
2014-10-25 15:17:57 +00:00
|
|
|
else if (sampleFormat == AV_SAMPLE_FMT_S16P || sampleFormat == AV_SAMPLE_FMT_S16)
|
2014-10-22 20:11:03 +00:00
|
|
|
sampleFormat = AV_SAMPLE_FMT_S16;
|
2014-10-25 18:50:41 +00:00
|
|
|
else if (sampleFormat == AV_SAMPLE_FMT_FLTP || sampleFormat == AV_SAMPLE_FMT_FLT)
|
|
|
|
sampleFormat = AV_SAMPLE_FMT_S16; // FIXME: check for AL_EXT_FLOAT32 support
|
2014-10-22 20:11:03 +00:00
|
|
|
else
|
2014-10-25 15:17:57 +00:00
|
|
|
sampleFormat = AV_SAMPLE_FMT_S16;
|
2014-10-22 20:11:03 +00:00
|
|
|
|
2014-10-25 18:50:41 +00:00
|
|
|
if (channelLayout == AV_CH_LAYOUT_5POINT1 || channelLayout == AV_CH_LAYOUT_7POINT1
|
|
|
|
|| channelLayout == AV_CH_LAYOUT_QUAD) // FIXME: check for AL_EXT_MCFORMATS support
|
2014-10-25 15:17:57 +00:00
|
|
|
channelLayout = AV_CH_LAYOUT_STEREO;
|
|
|
|
else if (channelLayout != AV_CH_LAYOUT_MONO
|
|
|
|
&& channelLayout != AV_CH_LAYOUT_STEREO)
|
2014-10-22 20:11:03 +00:00
|
|
|
channelLayout = AV_CH_LAYOUT_STEREO;
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
~MovieAudioDecoder()
|
|
|
|
{
|
2017-09-12 04:33:18 +00:00
|
|
|
if(mAudioTrack)
|
2015-11-30 13:42:51 +00:00
|
|
|
MWBase::Environment::get().getSoundManager()->stopTrack(mAudioTrack);
|
2017-09-12 04:33:18 +00:00
|
|
|
mAudioTrack = nullptr;
|
2014-10-22 20:11:03 +00:00
|
|
|
mDecoderBridge.reset();
|
|
|
|
}
|
|
|
|
|
2017-09-12 04:33:18 +00:00
|
|
|
MWBase::SoundStream *mAudioTrack;
|
2017-05-05 17:26:09 +00:00
|
|
|
std::shared_ptr<MWSoundDecoderBridge> mDecoderBridge;
|
2014-10-22 20:11:03 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2017-12-09 19:00:56 +00:00
|
|
|
void MWSoundDecoderBridge::open(const std::string &fname)
|
2014-10-22 20:11:03 +00:00
|
|
|
{
|
2017-12-09 19:00:56 +00:00
|
|
|
throw std::runtime_error("Method not implemented");
|
2014-10-22 20:11:03 +00:00
|
|
|
}
|
|
|
|
void MWSoundDecoderBridge::close() {}
|
|
|
|
|
|
|
|
std::string MWSoundDecoderBridge::getName()
|
|
|
|
{
|
|
|
|
return mDecoder->getStreamName();
|
|
|
|
}
|
|
|
|
|
2017-12-09 19:00:56 +00:00
|
|
|
void MWSoundDecoderBridge::getInfo(int *samplerate, ChannelConfig *chans, SampleType *type)
|
2014-10-22 20:11:03 +00:00
|
|
|
{
|
|
|
|
*samplerate = mDecoder->getOutputSampleRate();
|
|
|
|
|
|
|
|
uint64_t outputChannelLayout = mDecoder->getOutputChannelLayout();
|
|
|
|
if (outputChannelLayout == AV_CH_LAYOUT_MONO)
|
|
|
|
*chans = ChannelConfig_Mono;
|
|
|
|
else if (outputChannelLayout == AV_CH_LAYOUT_5POINT1)
|
|
|
|
*chans = ChannelConfig_5point1;
|
|
|
|
else if (outputChannelLayout == AV_CH_LAYOUT_7POINT1)
|
|
|
|
*chans = ChannelConfig_7point1;
|
|
|
|
else if (outputChannelLayout == AV_CH_LAYOUT_STEREO)
|
|
|
|
*chans = ChannelConfig_Stereo;
|
|
|
|
else if (outputChannelLayout == AV_CH_LAYOUT_QUAD)
|
|
|
|
*chans = ChannelConfig_Quad;
|
|
|
|
else
|
2017-12-09 19:00:56 +00:00
|
|
|
throw std::runtime_error("Unsupported channel layout: "+
|
|
|
|
std::to_string(outputChannelLayout));
|
2014-10-22 20:11:03 +00:00
|
|
|
|
|
|
|
AVSampleFormat outputSampleFormat = mDecoder->getOutputSampleFormat();
|
|
|
|
if (outputSampleFormat == AV_SAMPLE_FMT_U8)
|
|
|
|
*type = SampleType_UInt8;
|
|
|
|
else if (outputSampleFormat == AV_SAMPLE_FMT_FLT)
|
|
|
|
*type = SampleType_Float32;
|
|
|
|
else if (outputSampleFormat == AV_SAMPLE_FMT_S16)
|
|
|
|
*type = SampleType_Int16;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
char str[1024];
|
|
|
|
av_get_sample_fmt_string(str, sizeof(str), outputSampleFormat);
|
2017-12-09 19:00:56 +00:00
|
|
|
throw std::runtime_error(std::string("Unsupported sample format: ")+str);
|
2014-10-22 20:11:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t MWSoundDecoderBridge::read(char *buffer, size_t bytes)
|
|
|
|
{
|
|
|
|
return mDecoder->read(buffer, bytes);
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t MWSoundDecoderBridge::getSampleOffset()
|
|
|
|
{
|
|
|
|
return mDecoder->getSampleOffset();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2017-05-05 17:26:09 +00:00
|
|
|
std::shared_ptr<Video::MovieAudioDecoder> MovieAudioFactory::createDecoder(Video::VideoState* videoState)
|
2014-10-22 20:11:03 +00:00
|
|
|
{
|
2017-05-05 17:26:09 +00:00
|
|
|
std::shared_ptr<MWSound::MovieAudioDecoder> decoder(new MWSound::MovieAudioDecoder(videoState));
|
2014-10-22 20:11:03 +00:00
|
|
|
decoder->setupFormat();
|
|
|
|
|
2015-11-30 15:32:42 +00:00
|
|
|
MWBase::SoundManager *sndMgr = MWBase::Environment::get().getSoundManager();
|
2017-09-15 08:03:41 +00:00
|
|
|
MWBase::SoundStream *sound = sndMgr->playTrack(decoder->mDecoderBridge, MWSound::Type::Movie);
|
2017-09-12 04:33:18 +00:00
|
|
|
if (!sound)
|
2014-10-22 20:11:03 +00:00
|
|
|
{
|
|
|
|
decoder.reset();
|
|
|
|
return decoder;
|
|
|
|
}
|
|
|
|
|
|
|
|
decoder->mAudioTrack = sound;
|
|
|
|
return decoder;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|