mirror of
https://github.com/f4exb/sdrangel.git
synced 2024-12-23 10:05:46 -05:00
740 lines
23 KiB
C++
740 lines
23 KiB
C++
///////////////////////////////////////////////////////////////////////////////////
|
|
// Copyright (C) 2018-2021 Edouard Griffiths, F4EXB <f4exb06@gmail.com> //
|
|
// Copyright (C) 2022 Jon Beniston, M7RCE <jon@beniston.com> //
|
|
// Copyright (C) 2023 Daniele Forsi <iu5hkx@gmail.com> //
|
|
// Copyright (C) 2018 F4HKW //
|
|
// for F4EXB / SDRAngel //
|
|
// //
|
|
// This program is free software; you can redistribute it and/or modify //
|
|
// it under the terms of the GNU General Public License as published by //
|
|
// the Free Software Foundation as version 3 of the License, or //
|
|
// (at your option) any later version. //
|
|
// //
|
|
// This program is distributed in the hope that it will be useful, //
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of //
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the //
|
|
// GNU General Public License V3 for more details. //
|
|
// //
|
|
// You should have received a copy of the GNU General Public License //
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>. //
|
|
///////////////////////////////////////////////////////////////////////////////////
|
|
|
|
#include <math.h>
|
|
#include <algorithm>
|
|
|
|
#include <QLayout>
|
|
|
|
extern "C"
|
|
{
|
|
#include <libswresample/swresample.h>
|
|
}
|
|
|
|
#include "audio/audiofifo.h"
|
|
#include "datvideorender.h"
|
|
|
|
DATVideoRender::DATVideoRender(QWidget *parent) :
|
|
TVScreen(true, parent),
|
|
m_parentWidget(parent)
|
|
{
|
|
installEventFilter(this);
|
|
m_isFullScreen = false;
|
|
m_isOpen = false;
|
|
m_formatCtx = nullptr;
|
|
m_videoDecoderCtx = nullptr;
|
|
m_audioDecoderCtx = nullptr;
|
|
m_swsCtx = nullptr;
|
|
m_audioFifo = nullptr;
|
|
m_audioSWR = nullptr;
|
|
m_audioSampleRate = 48000;
|
|
m_audioFifoBufferIndex = 0;
|
|
m_videoStreamIndex = -1;
|
|
m_audioStreamIndex = -1;
|
|
m_audioMute = false;
|
|
m_videoMute = false;
|
|
m_audioVolume = 0;
|
|
|
|
m_currentRenderWidth = -1;
|
|
m_currentRenderHeight = -1;
|
|
|
|
m_frame = nullptr;
|
|
m_frameCount = -1;
|
|
|
|
m_audioDecodeOK = false;
|
|
m_videoDecodeOK = false;
|
|
|
|
av_log_set_level(AV_LOG_FATAL);
|
|
}
|
|
|
|
DATVideoRender::~DATVideoRender()
|
|
{
|
|
qDebug("DATVideoRender::~DATVideoRender");
|
|
|
|
if (m_audioSWR) {
|
|
swr_free(&m_audioSWR);
|
|
}
|
|
}
|
|
|
|
bool DATVideoRender::eventFilter(QObject *obj, QEvent *event)
|
|
{
|
|
if (event->type() == QEvent::MouseButtonRelease)
|
|
{
|
|
setFullScreen(false);
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
// standard event processing
|
|
return QObject::eventFilter(obj, event);
|
|
}
|
|
}
|
|
|
|
void DATVideoRender::setFullScreen(bool fullScreen)
|
|
{
|
|
if (m_isFullScreen == fullScreen)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (fullScreen == true)
|
|
{
|
|
qDebug("DATVideoRender::setFullScreen: go to fullscreen");
|
|
// m_originalWindowFlags = this->windowFlags();
|
|
// m_originalSize = this->size();
|
|
// m_parentWidget->layout()->removeWidget(this);
|
|
// //this->setParent(0);
|
|
// this->setWindowFlags( Qt::FramelessWindowHint|Qt::WindowStaysOnTopHint);
|
|
// m_parentWidget->show();
|
|
setWindowFlags(Qt::Window);
|
|
setWindowState(Qt::WindowFullScreen);
|
|
show();
|
|
m_isFullScreen = true;
|
|
}
|
|
else
|
|
{
|
|
qDebug("DATVideoRender::setFullScreen: come back from fullscreen");
|
|
// //this->setParent(m_parentWidget);
|
|
// this->resize(m_originalSize);
|
|
// this->overrideWindowFlags(m_originalWindowFlags);
|
|
// setWindowState(Qt::WindowNoState);
|
|
// m_parentWidget->layout()->addWidget(this);
|
|
// m_parentWidget->show();
|
|
setWindowFlags(Qt::Widget);
|
|
setWindowState(Qt::WindowNoState);
|
|
show();
|
|
m_isFullScreen = false;
|
|
}
|
|
}
|
|
|
|
int DATVideoRender::ReadFunction(void *opaque, uint8_t *buf, int buf_size)
|
|
{
|
|
DATVideostream *stream = reinterpret_cast<DATVideostream *>(opaque);
|
|
int nbBytes = stream->read((char *)buf, buf_size);
|
|
return nbBytes;
|
|
}
|
|
|
|
int64_t DATVideoRender::SeekFunction(void *opaque, int64_t offset, int whence)
|
|
{
|
|
DATVideostream *stream = reinterpret_cast<DATVideostream *>(opaque);
|
|
|
|
if (whence == AVSEEK_SIZE)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
if (stream->isSequential())
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
if (stream->seek(offset) == false)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
return stream->pos();
|
|
}
|
|
|
|
void DATVideoRender::resetMetaData()
|
|
{
|
|
m_metaData.reset();
|
|
emit onMetaDataChanged(new DataTSMetaData2(m_metaData));
|
|
}
|
|
|
|
bool DATVideoRender::preprocessStream()
|
|
{
|
|
AVDictionary *opts = nullptr;
|
|
const AVCodec *videoCodec = nullptr;
|
|
const AVCodec *audioCodec = nullptr;
|
|
|
|
int intRet = -1;
|
|
char *buffer = nullptr;
|
|
|
|
resetMetaData();
|
|
|
|
//Identify stream
|
|
|
|
if (avformat_find_stream_info(m_formatCtx, nullptr) < 0)
|
|
{
|
|
avformat_close_input(&m_formatCtx);
|
|
qDebug() << "DATVideoRender::preprocessStream cannot find stream info";
|
|
return false;
|
|
}
|
|
|
|
//Find video stream
|
|
intRet = av_find_best_stream(m_formatCtx, AVMEDIA_TYPE_VIDEO, -1, -1, nullptr, 0);
|
|
|
|
if (intRet < 0)
|
|
{
|
|
avformat_close_input(&m_formatCtx);
|
|
qDebug() << "DATVideoRender::preprocessStream cannot find video stream";
|
|
return false;
|
|
}
|
|
|
|
m_videoStreamIndex = intRet;
|
|
|
|
//Find audio stream
|
|
intRet = av_find_best_stream(m_formatCtx, AVMEDIA_TYPE_AUDIO, -1, -1, nullptr, 0);
|
|
|
|
if (intRet < 0) {
|
|
qDebug() << "DATVideoRender::preprocessStream cannot find audio stream";
|
|
}
|
|
|
|
m_audioStreamIndex = intRet;
|
|
|
|
// Prepare Video Codec and extract meta data
|
|
|
|
AVCodecParameters *parms = m_formatCtx->streams[m_videoStreamIndex]->codecpar;
|
|
|
|
if (m_videoDecoderCtx) {
|
|
avcodec_free_context(&m_videoDecoderCtx);
|
|
}
|
|
|
|
m_videoDecoderCtx = avcodec_alloc_context3(nullptr);
|
|
avcodec_parameters_to_context(m_videoDecoderCtx, parms);
|
|
|
|
//Meta Data
|
|
|
|
m_metaData.PID = m_formatCtx->streams[m_videoStreamIndex]->id;
|
|
m_metaData.CodecID = m_videoDecoderCtx->codec_id;
|
|
m_metaData.OK_TransportStream = true;
|
|
m_metaData.Program = "";
|
|
m_metaData.Stream = "";
|
|
|
|
if (m_formatCtx->programs && m_formatCtx->programs[m_videoStreamIndex])
|
|
{
|
|
buffer = nullptr;
|
|
av_dict_get_string(m_formatCtx->programs[m_videoStreamIndex]->metadata, &buffer, ':', '\n');
|
|
|
|
if (buffer != nullptr)
|
|
{
|
|
m_metaData.Program = QString("%1").arg(buffer);
|
|
}
|
|
}
|
|
|
|
buffer = nullptr;
|
|
|
|
av_dict_get_string(m_formatCtx->streams[m_videoStreamIndex]->metadata, &buffer, ':', '\n');
|
|
|
|
if (buffer != nullptr)
|
|
{
|
|
m_metaData.Stream = QString("%1").arg(buffer);
|
|
}
|
|
|
|
emit onMetaDataChanged(new DataTSMetaData2(m_metaData));
|
|
|
|
//Decoder
|
|
videoCodec = avcodec_find_decoder(m_videoDecoderCtx->codec_id);
|
|
|
|
if (videoCodec == nullptr)
|
|
{
|
|
avformat_close_input(&m_formatCtx);
|
|
qDebug() << "DATVideoRender::preprocessStream cannot find associated video CODEC";
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
qDebug() << "DATVideoRender::preprocessStream: video CODEC found: " << videoCodec->name;
|
|
}
|
|
|
|
av_dict_set(&opts, "refcounted_frames", "1", 0);
|
|
|
|
if (avcodec_open2(m_videoDecoderCtx, videoCodec, &opts) < 0)
|
|
{
|
|
avformat_close_input(&m_formatCtx);
|
|
qDebug() << "DATVideoRender::preprocessStream cannot open associated video CODEC";
|
|
return false;
|
|
}
|
|
|
|
//Allocate Frame
|
|
m_frame = av_frame_alloc();
|
|
|
|
if (!m_frame)
|
|
{
|
|
avformat_close_input(&m_formatCtx);
|
|
qDebug() << "DATVideoRender::preprocessStream cannot allocate frame";
|
|
return false;
|
|
}
|
|
|
|
m_frameCount = 0;
|
|
m_metaData.Width = m_videoDecoderCtx->width;
|
|
m_metaData.Height = m_videoDecoderCtx->height;
|
|
m_metaData.BitRate = m_videoDecoderCtx->bit_rate;
|
|
m_metaData.Channels = m_videoDecoderCtx->channels;
|
|
m_metaData.CodecDescription = QString("%1").arg(videoCodec->long_name);
|
|
m_metaData.OK_VideoStream = true;
|
|
|
|
QString metaStr;
|
|
m_metaData.formatString(metaStr);
|
|
qDebug() << "DATVideoRender::preprocessStream: video: " << metaStr;
|
|
|
|
emit onMetaDataChanged(new DataTSMetaData2(m_metaData));
|
|
|
|
// Prepare Audio Codec
|
|
|
|
if (m_audioStreamIndex >= 0)
|
|
{
|
|
AVCodecParameters *parms = m_formatCtx->streams[m_audioStreamIndex]->codecpar;
|
|
|
|
if (m_audioDecoderCtx) {
|
|
avcodec_free_context(&m_audioDecoderCtx);
|
|
}
|
|
|
|
m_audioDecoderCtx = avcodec_alloc_context3(nullptr);
|
|
avcodec_parameters_to_context(m_audioDecoderCtx, parms);
|
|
|
|
//m_audioDecoderCtx = m_formatCtx->streams[m_audioStreamIndex]->codec; // old style
|
|
|
|
qDebug() << "DATVideoRender::preprocessStream: audio: "
|
|
<< " channels: " << m_audioDecoderCtx->channels
|
|
<< " channel_layout: " << m_audioDecoderCtx->channel_layout
|
|
<< " sample_rate: " << m_audioDecoderCtx->sample_rate
|
|
<< " sample_fmt: " << m_audioDecoderCtx->sample_fmt
|
|
<< " codec_id: "<< m_audioDecoderCtx->codec_id;
|
|
|
|
audioCodec = avcodec_find_decoder(m_audioDecoderCtx->codec_id);
|
|
|
|
if (audioCodec == nullptr)
|
|
{
|
|
qDebug() << "DATVideoRender::preprocessStream cannot find associated audio CODEC";
|
|
m_audioStreamIndex = -1; // invalidate audio
|
|
}
|
|
else
|
|
{
|
|
qDebug() << "DATVideoRender::preprocessStream: audio CODEC found: " << audioCodec->name;
|
|
|
|
if (avcodec_open2(m_audioDecoderCtx, audioCodec, nullptr) < 0)
|
|
{
|
|
qDebug() << "DATVideoRender::preprocessStream cannot open associated audio CODEC";
|
|
m_audioStreamIndex = -1; // invalidate audio
|
|
}
|
|
else
|
|
{
|
|
setResampler();
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool DATVideoRender::openStream(DATVideostream *device)
|
|
{
|
|
int ioBufferSize = DATVideostream::m_defaultMemoryLimit;
|
|
unsigned char *ptrIOBuffer = nullptr;
|
|
AVIOContext *ioCtx = nullptr;
|
|
|
|
if (device == nullptr)
|
|
{
|
|
qDebug() << "DATVideoRender::openStream QIODevice is nullptr";
|
|
return false;
|
|
}
|
|
|
|
if (m_isOpen)
|
|
{
|
|
qDebug() << "DATVideoRender::openStream already open";
|
|
return false;
|
|
}
|
|
|
|
m_isOpen = false;
|
|
|
|
if (device->bytesAvailable() <= 0)
|
|
{
|
|
qDebug() << "DATVideoRender::openStream no data available";
|
|
m_metaData.OK_Data = false;
|
|
emit onMetaDataChanged(new DataTSMetaData2(m_metaData));
|
|
return false;
|
|
}
|
|
|
|
m_metaData.OK_Data = true;
|
|
emit onMetaDataChanged(new DataTSMetaData2(m_metaData));
|
|
|
|
if (!device->open(QIODevice::ReadOnly))
|
|
{
|
|
qDebug() << "DATVideoRender::openStream cannot open QIODevice";
|
|
return false;
|
|
}
|
|
|
|
//Connect QIODevice to FFMPEG Reader
|
|
|
|
m_formatCtx = avformat_alloc_context();
|
|
|
|
if (m_formatCtx == nullptr)
|
|
{
|
|
qDebug() << "DATVideoRender::openStream cannot alloc format FFMPEG context";
|
|
return false;
|
|
}
|
|
|
|
ptrIOBuffer = (unsigned char *)av_malloc(ioBufferSize + AV_INPUT_BUFFER_PADDING_SIZE);
|
|
|
|
ioCtx = avio_alloc_context(
|
|
ptrIOBuffer,
|
|
ioBufferSize,
|
|
0,
|
|
reinterpret_cast<void *>(device),
|
|
&DATVideoRender::ReadFunction,
|
|
nullptr,
|
|
&DATVideoRender::SeekFunction
|
|
);
|
|
|
|
m_formatCtx->pb = ioCtx;
|
|
m_formatCtx->flags |= AVFMT_FLAG_CUSTOM_IO;
|
|
|
|
if (avformat_open_input(&m_formatCtx, nullptr, nullptr, nullptr) < 0)
|
|
{
|
|
qDebug() << "DATVideoRender::openStream cannot open stream";
|
|
return false;
|
|
}
|
|
|
|
if (!preprocessStream())
|
|
{
|
|
return false;
|
|
}
|
|
|
|
qDebug("DATVideoRender::openStream: successful");
|
|
m_isOpen = true;
|
|
|
|
return true;
|
|
}
|
|
|
|
bool DATVideoRender::renderStream()
|
|
{
|
|
AVPacket packet;
|
|
int gotFrame;
|
|
bool needRenderingSetup;
|
|
|
|
if (!m_isOpen)
|
|
{
|
|
qDebug() << "DATVideoRender::renderStream Stream not open";
|
|
return false;
|
|
}
|
|
|
|
//********** Rendering **********
|
|
if (av_read_frame(m_formatCtx, &packet) < 0)
|
|
{
|
|
qDebug() << "DATVideoRender::renderStream reading packet error";
|
|
return false;
|
|
}
|
|
|
|
if (packet.size == 0)
|
|
{
|
|
qDebug() << "DATVideoRender::renderStream packet empty";
|
|
av_packet_unref(&packet);
|
|
return true;
|
|
}
|
|
|
|
//Video channel
|
|
if ((packet.stream_index == m_videoStreamIndex) && (!m_videoMute))
|
|
{
|
|
// memset(m_frame, 0, sizeof(AVFrame));
|
|
av_frame_unref(m_frame);
|
|
|
|
gotFrame = 0;
|
|
|
|
if (newDecode(m_videoDecoderCtx, m_frame, &gotFrame, &packet) >= 0)
|
|
{
|
|
m_videoDecodeOK = true;
|
|
|
|
if (gotFrame)
|
|
{
|
|
//Rendering and RGB Converter setup
|
|
needRenderingSetup = (m_frameCount == 0);
|
|
needRenderingSetup |= (m_swsCtx == nullptr);
|
|
|
|
if ((m_currentRenderWidth != m_frame->width) || (m_currentRenderHeight != m_frame->height))
|
|
{
|
|
needRenderingSetup = true;
|
|
}
|
|
|
|
if (needRenderingSetup)
|
|
{
|
|
if (m_swsCtx != nullptr)
|
|
{
|
|
sws_freeContext(m_swsCtx);
|
|
m_swsCtx = nullptr;
|
|
}
|
|
|
|
//Convertisseur YUV -> RGB
|
|
m_swsCtx = sws_alloc_context();
|
|
|
|
av_opt_set_int(m_swsCtx, "srcw", m_frame->width, 0);
|
|
av_opt_set_int(m_swsCtx, "srch", m_frame->height, 0);
|
|
av_opt_set_int(m_swsCtx, "src_format", m_frame->format, 0);
|
|
|
|
av_opt_set_int(m_swsCtx, "dstw", m_frame->width, 0);
|
|
av_opt_set_int(m_swsCtx, "dsth", m_frame->height, 0);
|
|
av_opt_set_int(m_swsCtx, "dst_format", AV_PIX_FMT_RGB24, 0);
|
|
|
|
av_opt_set_int(m_swsCtx, "sws_flag", SWS_FAST_BILINEAR /* SWS_BICUBIC*/, 0);
|
|
|
|
if (sws_init_context(m_swsCtx, nullptr, nullptr) < 0)
|
|
{
|
|
qDebug() << "DATVideoRender::renderStream cannot init video data converter";
|
|
m_swsCtx = nullptr;
|
|
av_packet_unref(&packet);
|
|
return false;
|
|
}
|
|
|
|
if ((m_currentRenderHeight > 0) && (m_currentRenderWidth > 0))
|
|
{
|
|
//av_freep(&m_decodedData[0]);
|
|
//av_freep(&m_decodedLineSize[0]);
|
|
}
|
|
|
|
if (av_image_alloc(m_decodedData, m_decodedLineSize, m_frame->width, m_frame->height, AV_PIX_FMT_RGB24, 1) < 0)
|
|
{
|
|
qDebug() << "DATVideoRender::renderStream cannot init video image buffer";
|
|
sws_freeContext(m_swsCtx);
|
|
m_swsCtx = nullptr;
|
|
av_packet_unref(&packet);
|
|
return false;
|
|
}
|
|
|
|
//Rendering device setup
|
|
|
|
resizeTVScreen(m_frame->width, m_frame->height);
|
|
update();
|
|
resetImage();
|
|
|
|
m_currentRenderWidth = m_frame->width;
|
|
m_currentRenderHeight = m_frame->height;
|
|
|
|
m_metaData.Width = m_frame->width;
|
|
m_metaData.Height = m_frame->height;
|
|
m_metaData.OK_Decoding = true;
|
|
emit onMetaDataChanged(new DataTSMetaData2(m_metaData));
|
|
}
|
|
|
|
//Frame rendering
|
|
|
|
if (sws_scale(m_swsCtx, m_frame->data, m_frame->linesize, 0, m_frame->height, m_decodedData, m_decodedLineSize) < 0)
|
|
{
|
|
qDebug() << "DATVideoRender::renderStream error converting video frame to RGB";
|
|
av_packet_unref(&packet);
|
|
return false;
|
|
}
|
|
|
|
renderImage(m_decodedData[0]);
|
|
av_frame_unref(m_frame);
|
|
m_frameCount++;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_videoDecodeOK = false;
|
|
// qDebug() << "DATVideoRender::renderStream video decode error";
|
|
}
|
|
}
|
|
// Audio channel
|
|
else if ((packet.stream_index == m_audioStreamIndex) && (m_audioFifo) && (swr_is_initialized(m_audioSWR)) && (!m_audioMute))
|
|
{
|
|
// memset(m_frame, 0, sizeof(AVFrame));
|
|
av_frame_unref(m_frame);
|
|
gotFrame = 0;
|
|
|
|
if (newDecode(m_audioDecoderCtx, m_frame, &gotFrame, &packet) >= 0)
|
|
{
|
|
m_audioDecodeOK = true;
|
|
|
|
if (gotFrame)
|
|
{
|
|
int in_samplerate = m_audioDecoderCtx->sample_rate;
|
|
int out_num_samples = av_rescale_rnd(swr_get_delay(m_audioSWR, in_samplerate) + m_frame->nb_samples, m_audioSampleRate, in_samplerate, AV_ROUND_UP);
|
|
int16_t *audioBuffer = nullptr;
|
|
av_samples_alloc((uint8_t**) &audioBuffer, nullptr, 2, out_num_samples, AV_SAMPLE_FMT_S16, 1);
|
|
int samples_per_channel = swr_convert(m_audioSWR, (uint8_t**) &audioBuffer, out_num_samples, (const uint8_t**) m_frame->data, m_frame->nb_samples);
|
|
if (samples_per_channel < m_frame->nb_samples) {
|
|
qDebug("DATVideoRender::renderStream: converted samples missing %d/%d returned", samples_per_channel, m_frame->nb_samples);
|
|
}
|
|
|
|
// direct writing:
|
|
std::for_each(audioBuffer, audioBuffer + 2*samples_per_channel, [this](int16_t &x) {
|
|
x *= m_audioVolume;
|
|
});
|
|
int ret = m_audioFifo->write((const quint8*) &audioBuffer[0], samples_per_channel);
|
|
if (ret < samples_per_channel) {
|
|
// qDebug("DATVideoRender::renderStream: audio samples missing %d/%d written", ret, samples_per_channel);
|
|
}
|
|
|
|
// buffered writing:
|
|
// if (m_audioFifoBufferIndex + samples_per_channel < m_audioFifoBufferSize)
|
|
// {
|
|
// std::copy(&audioBuffer[0], &audioBuffer[2*samples_per_channel], &m_audioFifoBuffer[2*m_audioFifoBufferIndex]);
|
|
// m_audioFifoBufferIndex += samples_per_channel;
|
|
// }
|
|
// else
|
|
// {
|
|
// int remainder = m_audioFifoBufferSize - m_audioFifoBufferIndex;
|
|
// std::copy(&audioBuffer[0], &audioBuffer[2*remainder], &m_audioFifoBuffer[2*m_audioFifoBufferIndex]);
|
|
// std::for_each(m_audioFifoBuffer, m_audioFifoBuffer+2*m_audioFifoBufferSize, [this](int16_t &x) {
|
|
// x *= m_audioVolume;
|
|
// });
|
|
// int ret = m_audioFifo->write((const quint8*) &m_audioFifoBuffer[0], m_audioFifoBufferSize);
|
|
// if (ret < m_audioFifoBufferSize) {
|
|
// qDebug("DATVideoRender::renderStream: audio samples missing %d/%d written", ret, m_audioFifoBufferSize);
|
|
// }
|
|
// std::copy(&audioBuffer[2*remainder], &audioBuffer[2*samples_per_channel], &m_audioFifoBuffer[0]);
|
|
// m_audioFifoBufferIndex = samples_per_channel - remainder;
|
|
// }
|
|
|
|
av_freep(&audioBuffer);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_audioDecodeOK = false;
|
|
// qDebug("DATVideoRender::renderStream: audio decode error");
|
|
}
|
|
}
|
|
|
|
av_packet_unref(&packet);
|
|
//********** Rendering **********
|
|
|
|
return true;
|
|
}
|
|
|
|
void DATVideoRender::setAudioVolume(int audioVolume)
|
|
{
|
|
int audioVolumeConstrained = audioVolume < 0 ? 0 : audioVolume > 100 ? 100 : audioVolume;
|
|
m_audioVolume = pow(10.0, audioVolumeConstrained*0.02 - 2.0); // .01 -> 1 log
|
|
}
|
|
|
|
void DATVideoRender::setResampler()
|
|
{
|
|
if (m_audioSWR) {
|
|
swr_free(&m_audioSWR);
|
|
}
|
|
|
|
m_audioSWR = swr_alloc();
|
|
av_opt_set_int(m_audioSWR, "in_channel_count", m_audioDecoderCtx->channels, 0);
|
|
av_opt_set_int(m_audioSWR, "out_channel_count", 2, 0);
|
|
av_opt_set_int(m_audioSWR, "in_channel_layout", m_audioDecoderCtx->channel_layout, 0);
|
|
av_opt_set_int(m_audioSWR, "out_channel_layout", AV_CH_LAYOUT_STEREO, 0);
|
|
av_opt_set_int(m_audioSWR, "in_sample_rate", m_audioDecoderCtx->sample_rate, 0);
|
|
av_opt_set_int(m_audioSWR, "out_sample_rate", m_audioSampleRate, 0);
|
|
av_opt_set_sample_fmt(m_audioSWR, "in_sample_fmt", m_audioDecoderCtx->sample_fmt, 0);
|
|
av_opt_set_sample_fmt(m_audioSWR, "out_sample_fmt", AV_SAMPLE_FMT_S16, 0);
|
|
|
|
swr_init(m_audioSWR);
|
|
|
|
qDebug() << "DATVideoRender::setResampler: "
|
|
<< " in_channel_count: " << m_audioDecoderCtx->channels
|
|
<< " out_channel_count: " << 2
|
|
<< " in_channel_layout: " << m_audioDecoderCtx->channel_layout
|
|
<< " out_channel_layout: " << AV_CH_LAYOUT_STEREO
|
|
<< " in_sample_rate: " << m_audioDecoderCtx->sample_rate
|
|
<< " out_sample_rate: " << m_audioSampleRate
|
|
<< " in_sample_fmt: " << m_audioDecoderCtx->sample_fmt
|
|
<< " out_sample_fmt: " << AV_SAMPLE_FMT_S16;
|
|
}
|
|
|
|
bool DATVideoRender::closeStream(QIODevice *device)
|
|
{
|
|
qDebug("DATVideoRender::closeStream");
|
|
|
|
if (!device)
|
|
{
|
|
qDebug() << "DATVideoRender::closeStream QIODevice is nullptr";
|
|
return false;
|
|
}
|
|
|
|
if (!m_isOpen)
|
|
{
|
|
qDebug() << "DATVideoRender::closeStream Stream not open";
|
|
return false;
|
|
}
|
|
|
|
if (!m_formatCtx)
|
|
{
|
|
qDebug() << "DATVideoRender::closeStream FFMEG Context is not initialized";
|
|
return false;
|
|
}
|
|
|
|
avformat_close_input(&m_formatCtx);
|
|
|
|
if (m_videoDecoderCtx) {
|
|
avcodec_free_context(&m_videoDecoderCtx);
|
|
}
|
|
|
|
if (m_audioDecoderCtx) {
|
|
avcodec_free_context(&m_audioDecoderCtx);
|
|
}
|
|
|
|
if (m_audioSWR) {
|
|
swr_free(&m_audioSWR);
|
|
}
|
|
|
|
if (m_frame)
|
|
{
|
|
av_frame_unref(m_frame);
|
|
av_frame_free(&m_frame);
|
|
}
|
|
|
|
if (m_swsCtx != nullptr)
|
|
{
|
|
sws_freeContext(m_swsCtx);
|
|
m_swsCtx = nullptr;
|
|
}
|
|
|
|
device->close();
|
|
m_isOpen = false;
|
|
m_currentRenderWidth = -1;
|
|
m_currentRenderHeight = -1;
|
|
|
|
resetMetaData();
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Replacement of deprecated avcodec_decode_video2 with the same signature
|
|
* https://blogs.gentoo.org/lu_zero/2016/03/29/new-avcodec-api/
|
|
*/
|
|
int DATVideoRender::newDecode(AVCodecContext *avctx, AVFrame *frame, int *got_frame, AVPacket *pkt)
|
|
{
|
|
int ret;
|
|
|
|
*got_frame = 0;
|
|
|
|
if (pkt)
|
|
{
|
|
ret = avcodec_send_packet(avctx, pkt);
|
|
// In particular, we don't expect AVERROR(EAGAIN), because we read all
|
|
// decoded frames with avcodec_receive_frame() until done.
|
|
if (ret < 0) {
|
|
return ret == AVERROR_EOF ? 0 : ret;
|
|
}
|
|
}
|
|
|
|
ret = avcodec_receive_frame(avctx, frame);
|
|
|
|
if (ret < 0 && ret != AVERROR(EAGAIN) && ret != AVERROR_EOF) {
|
|
return ret;
|
|
}
|
|
|
|
if (ret >= 0) {
|
|
*got_frame = 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|