tdesktop/Telegram/SourceFiles/media/media_audio_ffmpeg_loader.h
2018-01-03 13:23:14 +03:00

160 lines
3.7 KiB
C++

/*
This file is part of Telegram Desktop,
the official desktop application for the Telegram messaging service.
For license and copyright information please follow this link:
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
*/
#pragma once
#include "media/media_audio.h"
#include "media/media_audio_loader.h"
extern "C" {
#include <libavcodec/avcodec.h>
#include <libavformat/avformat.h>
#include <libavutil/opt.h>
#include <libswresample/swresample.h>
} // extern "C"
#include <AL/al.h>
class AbstractFFMpegLoader : public AudioPlayerLoader {
public:
AbstractFFMpegLoader(
const FileLocation &file,
const QByteArray &data,
base::byte_vector &&bytes)
: AudioPlayerLoader(file, data, std::move(bytes)) {
}
bool open(TimeMs positionMs) override;
int64 samplesCount() override {
return _samplesCount;
}
int samplesFrequency() override {
return _samplesFrequency;
}
static uint64_t ComputeChannelLayout(
uint64_t channel_layout,
int channels);
~AbstractFFMpegLoader();
protected:
static int64 Mul(int64 value, AVRational rational);
int _samplesFrequency = Media::Player::kDefaultFrequency;
int64 _samplesCount = 0;
uchar *ioBuffer = nullptr;
AVIOContext *ioContext = nullptr;
AVFormatContext *fmtContext = nullptr;
AVCodec *codec = nullptr;
int32 streamId = 0;
bool _opened = false;
private:
static int _read_data(void *opaque, uint8_t *buf, int buf_size);
static int64_t _seek_data(void *opaque, int64_t offset, int whence);
static int _read_bytes(void *opaque, uint8_t *buf, int buf_size);
static int64_t _seek_bytes(void *opaque, int64_t offset, int whence);
static int _read_file(void *opaque, uint8_t *buf, int buf_size);
static int64_t _seek_file(void *opaque, int64_t offset, int whence);
};
class AbstractAudioFFMpegLoader : public AbstractFFMpegLoader {
public:
AbstractAudioFFMpegLoader(
const FileLocation &file,
const QByteArray &data,
base::byte_vector &&bytes);
int64 samplesCount() override {
return _outputSamplesCount;
}
int samplesFrequency() override {
return _swrDstRate;
}
int format() override {
return _outputFormat;
}
~AbstractAudioFFMpegLoader();
protected:
bool initUsingContext(
not_null<AVCodecContext*> context,
int64 initialCount,
int initialFrequency);
ReadResult readFromReadyContext(
not_null<AVCodecContext*> context,
QByteArray &result,
int64 &samplesAdded);
int sampleSize() const {
return _outputSampleSize;
}
private:
ReadResult readFromReadyFrame(QByteArray &result, int64 &samplesAdded);
bool frameHasDesiredFormat() const;
bool initResampleForFrame();
bool initResampleUsingFormat();
bool ensureResampleSpaceAvailable(int samples);
void appendSamples(
QByteArray &result,
int64 &samplesAdded,
uint8_t **data,
int count) const;
AVFrame *_frame = nullptr;
int _outputFormat = AL_FORMAT_STEREO16;
int _outputChannels = 2;
int _outputSampleSize = 2 * sizeof(uint16);
int64 _outputSamplesCount = 0;
SwrContext *_swrContext = nullptr;
int _swrSrcRate = 0;
AVSampleFormat _swrSrcSampleFormat = AV_SAMPLE_FMT_NONE;
uint64_t _swrSrcChannelLayout = 0;
const int _swrDstRate = Media::Player::kDefaultFrequency;
AVSampleFormat _swrDstSampleFormat = AV_SAMPLE_FMT_S16;
uint64_t _swrDstChannelLayout = AV_CH_LAYOUT_STEREO;
uint8_t **_swrDstData = nullptr;
int _swrDstDataCapacity = 0;
};
class FFMpegLoader : public AbstractAudioFFMpegLoader {
public:
FFMpegLoader(
const FileLocation &file,
const QByteArray &data,
base::byte_vector &&bytes);
bool open(TimeMs positionMs) override;
ReadResult readMore(QByteArray &result, int64 &samplesAdded) override;
~FFMpegLoader();
private:
bool openCodecContext();
bool seekTo(TimeMs positionMs);
AVCodecContext *_codecContext = nullptr;
AVPacket _packet;
};