1
0
mirror of https://github.com/qTox/qTox.git synced 2024-03-22 14:00:36 +08:00
qTox/src/video/camerasource.cpp
sudden6 a2927de27d
fix(video): use float framerates also for V4L2
also make -1 the default value for the framerate
2017-12-22 21:33:35 +01:00

463 lines
12 KiB
C++

/*
Copyright © 2015 by The qTox Project Contributors
This file is part of qTox, a Qt-based graphical interface for Tox.
qTox is libre 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, either version 3 of the License, or
(at your option) any later version.
qTox 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 for more details.
You should have received a copy of the GNU General Public License
along with qTox. If not, see <http://www.gnu.org/licenses/>.
*/
extern "C" {
#include <libavcodec/avcodec.h>
#include <libavdevice/avdevice.h>
#include <libavformat/avformat.h>
#include <libswscale/swscale.h>
}
#include "cameradevice.h"
#include "camerasource.h"
#include "videoframe.h"
#include "src/persistence/settings.h"
#include <QDebug>
#include <QReadLocker>
#include <QWriteLocker>
#include <QtConcurrent/QtConcurrentRun>
#include <functional>
#include <memory>
/**
* @class CameraSource
* @brief This class is a wrapper to share a camera's captured video frames
*
* It allows objects to suscribe and unsuscribe to the stream, starting
* the camera and streaming new video frames only when needed.
* This is a singleton, since we can only capture from one
* camera at the same time without thread-safety issues.
* The source is lazy in the sense that it will only keep the video
* device open as long as there are subscribers, the source can be
* open but the device closed if there are zero subscribers.
*/
/**
* @var QVector<std::weak_ptr<VideoFrame>> CameraSource::freelist
* @brief Frames that need freeing before we can safely close the device
*
* @var QFuture<void> CameraSource::streamFuture
* @brief Future of the streaming thread
*
* @var QString CameraSource::deviceName
* @brief Short name of the device for CameraDevice's open(QString)
*
* @var CameraDevice* CameraSource::device
* @brief Non-owning pointer to an open CameraDevice, or nullptr. Not atomic, synced with memfences
* when becomes null.
*
* @var VideoMode CameraSource::mode
* @brief What mode we tried to open the device in, all zeros means default mode
*
* @var AVCodecContext* CameraSource::cctx
* @brief Codec context of the camera's selected video stream
*
* @var AVCodecContext* CameraSource::cctxOrig
* @brief Codec context of the camera's selected video stream
* @deprecated
*
* @var int CameraSource::videoStreamIndex
* @brief A camera can have multiple streams, this is the one we're decoding
*
* @var QMutex CameraSource::biglock
* @brief True when locked. Faster than mutexes for video decoding.
*
* @var QMutex CameraSource::freelistLock
* @brief True when locked. Faster than mutexes for video decoding.
*
* @var std::atomic_bool CameraSource::streamBlocker
* @brief Holds the streaming thread still when true
*
* @var std::atomic_int CameraSource::subscriptions
* @brief Remember how many times we subscribed for RAII
*/
CameraSource* CameraSource::instance{nullptr};
CameraSource::CameraSource()
: deviceThread{new QThread}
, deviceName{"none"}
, device{nullptr}
, mode(VideoMode())
// clang-format off
, cctx{nullptr}
#if LIBAVCODEC_VERSION_INT < 3747941
, cctxOrig{nullptr}
#endif
, videoStreamIndex{-1}
, _isNone{true}
, subscriptions{0}
{
qRegisterMetaType<VideoMode>("VideoMode");
deviceThread->setObjectName("Device thread");
deviceThread->start();
moveToThread(deviceThread);
subscriptions = 0;
av_register_all();
avdevice_register_all();
}
// clang-format on
/**
* @brief Returns the singleton instance.
*/
CameraSource& CameraSource::getInstance()
{
if (!instance)
instance = new CameraSource();
return *instance;
}
void CameraSource::destroyInstance()
{
if (instance) {
delete instance;
instance = nullptr;
}
}
/**
* @brief Setup default device
* @note If a device is already open, the source will seamlessly switch to the new device.
*/
void CameraSource::setupDefault()
{
QString deviceName = CameraDevice::getDefaultDeviceName();
bool isScreen = CameraDevice::isScreen(deviceName);
VideoMode mode = VideoMode(Settings::getInstance().getScreenRegion());
if (!isScreen) {
mode = VideoMode(Settings::getInstance().getCamVideoRes());
mode.FPS = Settings::getInstance().getCamVideoFPS();
}
setupDevice(deviceName, mode);
}
/**
* @brief Change the device and mode.
* @note If a device is already open, the source will seamlessly switch to the new device.
*/
void CameraSource::setupDevice(const QString& DeviceName, const VideoMode& Mode)
{
if (QThread::currentThread() != deviceThread) {
QMetaObject::invokeMethod(this, "setupDevice", Q_ARG(const QString&, DeviceName),
Q_ARG(const VideoMode&, Mode));
return;
}
QWriteLocker locker{&deviceMutex};
if (DeviceName == deviceName && Mode == mode) {
return;
}
if (subscriptions) {
// To force close, ignoring optimization
int subs = subscriptions;
subscriptions = 0;
closeDevice();
subscriptions = subs;
}
deviceName = DeviceName;
mode = Mode;
_isNone = (deviceName == "none");
if (subscriptions && !_isNone) {
openDevice();
}
}
bool CameraSource::isNone() const
{
return _isNone;
}
CameraSource::~CameraSource()
{
QWriteLocker locker{&streamMutex};
QWriteLocker locker2{&deviceMutex};
// Stop the device thread
deviceThread->exit(0);
deviceThread->wait();
delete deviceThread;
if (_isNone) {
return;
}
// Free all remaining VideoFrame
VideoFrame::untrackFrames(id, true);
if (cctx) {
avcodec_free_context(&cctx);
}
#if LIBAVCODEC_VERSION_INT < 3747941
if (cctxOrig) {
avcodec_close(cctxOrig);
}
#endif
if (device) {
for (int i = 0; i < subscriptions; ++i)
device->close();
device = nullptr;
}
locker.unlock();
// Synchronize with our stream thread
while (streamFuture.isRunning())
QThread::yieldCurrentThread();
}
void CameraSource::subscribe()
{
QWriteLocker locker{&deviceMutex};
++subscriptions;
openDevice();
}
void CameraSource::unsubscribe()
{
QWriteLocker locker{&deviceMutex};
--subscriptions;
if (subscriptions == 0) {
closeDevice();
}
}
/**
* @brief Opens the video device and starts streaming.
* @note Callers must own the biglock.
*/
void CameraSource::openDevice()
{
if (QThread::currentThread() != deviceThread) {
QMetaObject::invokeMethod(this, "openDevice");
return;
}
QWriteLocker locker{&streamMutex};
if (subscriptions == 0) {
return;
}
qDebug() << "Opening device " << deviceName;
if (device) {
device->open();
emit openFailed();
return;
}
// We need to create a new CameraDevice
AVCodec* codec;
device = CameraDevice::open(deviceName, mode);
if (!device) {
qWarning() << "Failed to open device!";
emit openFailed();
return;
}
// We need to open the device as many time as we already have subscribers,
// otherwise the device could get closed while we still have subscribers
for (int i = 0; i < subscriptions; ++i) {
device->open();
}
// Find the first video stream, if any
for (unsigned i = 0; i < device->context->nb_streams; ++i) {
AVMediaType type;
#if LIBAVCODEC_VERSION_INT < 3747941
type = device->context->streams[i]->codec->codec_type;
#else
type = device->context->streams[i]->codecpar->codec_type;
#endif
if (type == AVMEDIA_TYPE_VIDEO) {
videoStreamIndex = i;
break;
}
}
if (videoStreamIndex == -1) {
qWarning() << "Video stream not found";
emit openFailed();
return;
}
AVCodecID codecId;
#if LIBAVCODEC_VERSION_INT < 3747941
cctxOrig = device->context->streams[videoStreamIndex]->codec;
codecId = cctxOrig->codec_id;
#else
// Get the stream's codec's parameters and find a matching decoder
AVCodecParameters* cparams = device->context->streams[videoStreamIndex]->codecpar;
codecId = cparams->codec_id;
#endif
codec = avcodec_find_decoder(codecId);
if (!codec) {
qWarning() << "Codec not found";
emit openFailed();
return;
}
#if LIBAVCODEC_VERSION_INT < 3747941
// Copy context, since we apparently aren't allowed to use the original
cctx = avcodec_alloc_context3(codec);
if (avcodec_copy_context(cctx, cctxOrig) != 0) {
qWarning() << "Can't copy context";
emit openFailed();
return;
}
cctx->refcounted_frames = 1;
#else
// Create a context for our codec, using the existing parameters
cctx = avcodec_alloc_context3(codec);
if (avcodec_parameters_to_context(cctx, cparams) < 0) {
qWarning() << "Can't create AV context from parameters";
emit openFailed();
return;
}
#endif
// Open codec
if (avcodec_open2(cctx, codec, nullptr) < 0) {
qWarning() << "Can't open codec";
avcodec_free_context(&cctx);
emit openFailed();
return;
}
if (streamFuture.isRunning())
qDebug() << "The stream thread is already running! Keeping the current one open.";
else
streamFuture = QtConcurrent::run(std::bind(&CameraSource::stream, this));
// Synchronize with our stream thread
while (!streamFuture.isRunning())
QThread::yieldCurrentThread();
emit deviceOpened();
}
/**
* @brief Closes the video device and stops streaming.
* @note Callers must own the biglock.
*/
void CameraSource::closeDevice()
{
if (QThread::currentThread() != deviceThread) {
QMetaObject::invokeMethod(this, "closeDevice");
return;
}
QWriteLocker locker{&streamMutex};
if (subscriptions != 0) {
return;
}
qDebug() << "Closing device " << deviceName;
// Free all remaining VideoFrame
VideoFrame::untrackFrames(id, true);
// Free our resources and close the device
videoStreamIndex = -1;
avcodec_free_context(&cctx);
#if LIBAVCODEC_VERSION_INT < 3747941
avcodec_close(cctxOrig);
cctxOrig = nullptr;
#endif
while (device && !device->close()) {
}
device = nullptr;
}
/**
* @brief Blocking. Decodes video stream and emits new frames.
* @note Designed to run in its own thread.
*/
void CameraSource::stream()
{
auto streamLoop = [=]() {
AVPacket packet;
if (av_read_frame(device->context, &packet) != 0) {
return;
}
#if LIBAVCODEC_VERSION_INT < 3747941
AVFrame* frame = av_frame_alloc();
if (!frame) {
return;
}
// Only keep packets from the right stream;
if (packet.stream_index == videoStreamIndex) {
// Decode video frame
int frameFinished;
avcodec_decode_video2(cctx, frame, &frameFinished, &packet);
if (!frameFinished) {
return;
}
VideoFrame* vframe = new VideoFrame(id, frame);
emit frameAvailable(vframe->trackFrame());
}
#else
// Forward packets to the decoder and grab the decoded frame
bool isVideo = packet.stream_index == videoStreamIndex;
bool readyToRecive = isVideo && !avcodec_send_packet(cctx, &packet);
if (readyToRecive) {
AVFrame* frame = av_frame_alloc();
if (frame && !avcodec_receive_frame(cctx, frame)) {
VideoFrame* vframe = new VideoFrame(id, frame);
emit frameAvailable(vframe->trackFrame());
} else {
av_frame_free(&frame);
}
}
#endif
av_packet_unref(&packet);
};
forever
{
QReadLocker locker{&streamMutex};
// Exit if device is no longer valid
if (!device) {
break;
}
streamLoop();
}
}