mirror of
https://gitlab.com/OpenMW/openmw.git
synced 2025-01-17 19:10:24 +00:00
b39d69e98c
- Fix rindex overflow - Fix audio sample size bugs (was using sample_fmt and channel count of the decoder, instead of the resampled settings). We didn't notice this bug before, because the OpenAL MovieAudioFactory tries to resample to a format of the same byte size. - Add support for play/pause and seeking controls (not used by cutscenes in OpenMW) - Closing the video when arriving at the stream end is now handled by the user (we may also want to keep the video open and seek back) The video player now has a standalone demo, at https://github.com/scrawl/ogre-ffmpeg-videoplayer
174 lines
5.5 KiB
C++
174 lines
5.5 KiB
C++
#include "movieaudiofactory.hpp"
|
|
|
|
#include <extern/ogre-ffmpeg-videoplayer/audiodecoder.hpp>
|
|
#include <extern/ogre-ffmpeg-videoplayer/videostate.hpp>
|
|
|
|
#include "../mwbase/environment.hpp"
|
|
#include "../mwbase/soundmanager.hpp"
|
|
|
|
#include "sound_decoder.hpp"
|
|
#include "sound.hpp"
|
|
|
|
namespace MWSound
|
|
{
|
|
|
|
class MovieAudioDecoder;
|
|
class MWSoundDecoderBridge : public Sound_Decoder
|
|
{
|
|
public:
|
|
MWSoundDecoderBridge(MWSound::MovieAudioDecoder* decoder)
|
|
: mDecoder(decoder)
|
|
{
|
|
}
|
|
|
|
private:
|
|
MWSound::MovieAudioDecoder* mDecoder;
|
|
|
|
virtual void open(const std::string &fname);
|
|
virtual void close();
|
|
virtual void rewind();
|
|
virtual std::string getName();
|
|
virtual void getInfo(int *samplerate, ChannelConfig *chans, SampleType *type);
|
|
virtual size_t read(char *buffer, size_t bytes);
|
|
virtual size_t getSampleOffset();
|
|
};
|
|
|
|
class MovieAudioDecoder : public Video::MovieAudioDecoder
|
|
{
|
|
public:
|
|
MovieAudioDecoder(Video::VideoState *videoState)
|
|
: Video::MovieAudioDecoder(videoState)
|
|
{
|
|
mDecoderBridge.reset(new MWSoundDecoderBridge(this));
|
|
}
|
|
|
|
size_t getSampleOffset()
|
|
{
|
|
ssize_t clock_delay = (mFrameSize-mFramePos) / av_get_channel_layout_nb_channels(mOutputChannelLayout) /
|
|
av_get_bytes_per_sample(mOutputSampleFormat);
|
|
return (size_t)(mAudioClock*mAVStream->codec->sample_rate) - clock_delay;
|
|
}
|
|
|
|
std::string getStreamName()
|
|
{
|
|
return mVideoState->stream->getName();
|
|
}
|
|
|
|
private:
|
|
// MovieAudioDecoder overrides
|
|
|
|
virtual double getAudioClock()
|
|
{
|
|
return mAudioTrack->getTimeOffset();
|
|
}
|
|
|
|
virtual void adjustAudioSettings(AVSampleFormat& sampleFormat, uint64_t& channelLayout, int& sampleRate)
|
|
{
|
|
if (sampleFormat == AV_SAMPLE_FMT_U8P)
|
|
sampleFormat = AV_SAMPLE_FMT_U8;
|
|
else if (sampleFormat == AV_SAMPLE_FMT_S16P)
|
|
sampleFormat = AV_SAMPLE_FMT_S16;
|
|
else if (sampleFormat == AV_SAMPLE_FMT_FLTP)
|
|
sampleFormat = AV_SAMPLE_FMT_FLT;
|
|
else
|
|
sampleFormat = AV_SAMPLE_FMT_FLT;
|
|
|
|
if (channelLayout != AV_CH_LAYOUT_MONO
|
|
&& channelLayout != AV_CH_LAYOUT_5POINT1
|
|
&& channelLayout != AV_CH_LAYOUT_7POINT1
|
|
&& channelLayout != AV_CH_LAYOUT_STEREO
|
|
&& channelLayout != AV_CH_LAYOUT_QUAD)
|
|
channelLayout = AV_CH_LAYOUT_STEREO;
|
|
}
|
|
|
|
public:
|
|
~MovieAudioDecoder()
|
|
{
|
|
mAudioTrack.reset();
|
|
mDecoderBridge.reset();
|
|
}
|
|
|
|
MWBase::SoundPtr mAudioTrack;
|
|
boost::shared_ptr<MWSoundDecoderBridge> mDecoderBridge;
|
|
};
|
|
|
|
|
|
void MWSoundDecoderBridge::open(const std::string &fname)
|
|
{
|
|
throw std::runtime_error("unimplemented");
|
|
}
|
|
void MWSoundDecoderBridge::close() {}
|
|
void MWSoundDecoderBridge::rewind() {}
|
|
|
|
std::string MWSoundDecoderBridge::getName()
|
|
{
|
|
return mDecoder->getStreamName();
|
|
}
|
|
|
|
void MWSoundDecoderBridge::getInfo(int *samplerate, ChannelConfig *chans, SampleType *type)
|
|
{
|
|
*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
|
|
{
|
|
std::stringstream error;
|
|
error << "Unsupported channel layout: " << outputChannelLayout;
|
|
throw std::runtime_error(error.str());
|
|
}
|
|
|
|
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);
|
|
throw std::runtime_error(std::string("Unsupported sample format: ") + str);
|
|
}
|
|
}
|
|
|
|
size_t MWSoundDecoderBridge::read(char *buffer, size_t bytes)
|
|
{
|
|
return mDecoder->read(buffer, bytes);
|
|
}
|
|
|
|
size_t MWSoundDecoderBridge::getSampleOffset()
|
|
{
|
|
return mDecoder->getSampleOffset();
|
|
}
|
|
|
|
|
|
|
|
boost::shared_ptr<Video::MovieAudioDecoder> MovieAudioFactory::createDecoder(Video::VideoState* videoState)
|
|
{
|
|
boost::shared_ptr<MWSound::MovieAudioDecoder> decoder(new MWSound::MovieAudioDecoder(videoState));
|
|
decoder->setupFormat();
|
|
|
|
MWBase::SoundPtr sound = MWBase::Environment::get().getSoundManager()->playTrack(decoder->mDecoderBridge, MWBase::SoundManager::Play_TypeMovie);
|
|
if (!sound.get())
|
|
{
|
|
decoder.reset();
|
|
return decoder;
|
|
}
|
|
|
|
decoder->mAudioTrack = sound;
|
|
return decoder;
|
|
}
|
|
|
|
}
|