diff --git a/qtox.pro b/qtox.pro index 0517166f3..3c288ae0c 100644 --- a/qtox.pro +++ b/qtox.pro @@ -130,7 +130,7 @@ contains(DEFINES, QTOX_PLATFORM_EXT) { win32 { RC_FILE = windows/qtox.rc LIBS += -L$$PWD/libs/lib -ltoxav -ltoxcore -ltoxencryptsave -ltoxdns -lsodium -lvpx -lpthread - LIBS += -L$$PWD/libs/lib -lopencv_core249 -lopencv_highgui249 -lopencv_imgproc249 -lOpenAL32 -lopus + LIBS += -L$$PWD/libs/lib -lavformat -lavdevice -lavcodec -lavutil -lswscale -lOpenAL32 -lopus LIBS += -lopengl32 -lole32 -loleaut32 -luuid -lvfw32 -lws2_32 -liphlpapi -lz LIBS += -lqrencode contains(DEFINES, QTOX_FILTER_AUDIO) { @@ -146,27 +146,26 @@ win32 { ICON = img/icons/qtox.icns QMAKE_INFO_PLIST = osx/info.plist QMAKE_MACOSX_DEPLOYMENT_TARGET = 10.7 - LIBS += -L$$PWD/libs/lib/ -ltoxcore -ltoxav -ltoxencryptsave -ltoxdns -lsodium -lvpx -lopus -framework OpenAL -lopencv_core -lopencv_highgui -mmacosx-version-min=10.7 + LIBS += -L$$PWD/libs/lib/ -ltoxcore -ltoxav -ltoxencryptsave -ltoxdns -lsodium -lvpx -lopus -framework OpenAL -lavformat -lavdevice -lavcodec -lavutil -lswscale -mmacosx-version-min=10.7 LIBS += -lqrencode contains(DEFINES, QTOX_PLATFORM_EXT) { LIBS += -framework IOKit -framework CoreFoundation } contains(DEFINES, QTOX_FILTER_AUDIO) { LIBS += -lfilteraudio } } else { android { LIBS += -ltoxcore -ltoxav -ltoxencryptsave -ltoxdns - LIBS += -lopencv_videoio -lopencv_imgcodecs -lopencv_highgui -lopencv_imgproc -lopencv_androidcamera - LIBS += -llibjpeg -llibwebp -llibpng -llibtiff -llibjasper -lIlmImf -lopencv_core + LIBS += -llibjpeg -llibwebp -llibpng -llibtiff -llibjasper -lIlmImf LIBS += -lopus -lvpx -lsodium -lopenal } else { # If we're building a package, static link libtox[core,av] and libsodium, since they are not provided by any package contains(STATICPKG, YES) { target.path = /usr/bin INSTALLS += target - LIBS += -L$$PWD/libs/lib/ -lopus -lvpx -lopenal -Wl,-Bstatic -ltoxcore -ltoxav -ltoxencryptsave -ltoxdns -lsodium -lopencv_highgui -lopencv_imgproc -lopencv_core -lz -Wl,-Bdynamic + LIBS += -L$$PWD/libs/lib/ -lopus -lvpx -lopenal -Wl,-Bstatic -ltoxcore -ltoxav -ltoxencryptsave -ltoxdns -lsodium -lavformat -lavdevice -lavcodec -lavutil -lswscale -lz -Wl,-Bdynamic LIBS += -Wl,-Bstatic -ljpeg -ltiff -lpng -ljasper -lIlmImf -lIlmThread -lIex -ldc1394 -lraw1394 -lHalf -lz -llzma -ljbig LIBS += -Wl,-Bdynamic -lv4l1 -lv4l2 -lavformat -lavcodec -lavutil -lswscale -lusb-1.0 LIBS += -lqrencode } else { - LIBS += -L$$PWD/libs/lib/ -ltoxcore -ltoxav -ltoxencryptsave -ltoxdns -lvpx -lsodium -lopenal -lopencv_core -lopencv_highgui -lopencv_imgproc + LIBS += -L$$PWD/libs/lib/ -ltoxcore -ltoxav -ltoxencryptsave -ltoxdns -lvpx -lsodium -lopenal -lavformat -lavdevice -lavcodec -lavutil -lswscale LIBS += -lqrencode } @@ -183,7 +182,7 @@ win32 { } contains(JENKINS, YES) { - LIBS = ./libs/lib/libtoxav.a ./libs/lib/libvpx.a ./libs/lib/libopus.a ./libs/lib/libtoxdns.a ./libs/lib/libtoxencryptsave.a ./libs/lib/libtoxcore.a ./libs/lib/libopenal.a ./libs/lib/libsodium.a ./libs/lib/libfilteraudio.a /usr/lib/libopencv_core.so /usr/lib/libopencv_highgui.so /usr/lib/libopencv_imgproc.so -lX11 -lXss -lqrencode + LIBS = ./libs/lib/libtoxav.a ./libs/lib/libvpx.a ./libs/lib/libopus.a ./libs/lib/libtoxdns.a ./libs/lib/libtoxencryptsave.a ./libs/lib/libtoxcore.a ./libs/lib/libopenal.a ./libs/lib/libsodium.a ./libs/lib/libfilteraudio.a -lX11 -lXss -lqrencode contains(ENABLE_SYSTRAY_UNITY_BACKEND, YES) { LIBS += -lgobject-2.0 -lappindicator -lgtk-x11-2.0 } @@ -430,9 +429,6 @@ SOURCES += \ src/misc/db/genericddinterface.cpp \ src/misc/db/plaindb.cpp \ src/misc/db/encrypteddb.cpp \ - src/video/camera.cpp \ - src/video/cameraworker.cpp \ - src/video/netvideosource.cpp \ src/video/videoframe.cpp \ src/widget/gui.cpp \ src/toxme.cpp \ @@ -449,6 +445,9 @@ SOURCES += \ src/widget/tool/toolboxgraphicsitem.cpp \ src/widget/tool/flyoutoverlaywidget.cpp \ src/widget/form/settings/verticalonlyscroller.cpp \ + src/video/cameradevice.cpp \ + src/video/camerasource.cpp \ + src/video/corevideosource.cpp \ src/core/toxid.cpp @@ -467,11 +466,8 @@ HEADERS += \ src/misc/db/genericddinterface.h \ src/misc/db/plaindb.h \ src/misc/db/encrypteddb.h \ - src/video/camera.h \ - src/video/cameraworker.h \ src/video/videoframe.h \ src/video/videosource.h \ - src/video/netvideosource.h \ src/widget/gui.h \ src/toxme.h \ src/profilelocker.h \ @@ -482,4 +478,7 @@ HEADERS += \ src/widget/tool/toolboxgraphicsitem.h \ src/widget/tool/flyoutoverlaywidget.h \ src/widget/form/settings/verticalonlyscroller.h \ + src/video/cameradevice.h \ + src/video/camerasource.h \ + src/video/corevideosource.h \ src/core/toxid.h diff --git a/src/core/core.cpp b/src/core/core.cpp index 05726542b..ea747635b 100644 --- a/src/core/core.cpp +++ b/src/core/core.cpp @@ -52,8 +52,8 @@ QThread* Core::coreThread{nullptr}; #define MAX_GROUP_MESSAGE_LEN 1024 -Core::Core(Camera* cam, QThread *CoreThread, QString loadPath) : - tox(nullptr), toxav(nullptr), camera(cam), loadPath(loadPath), ready{false} +Core::Core(QThread *CoreThread, QString loadPath) : + tox(nullptr), toxav(nullptr), loadPath(loadPath), ready{false} { qDebug() << "loading Tox from" << loadPath; @@ -76,10 +76,7 @@ Core::Core(Camera* cam, QThread *CoreThread, QString loadPath) : calls[i].active = false; calls[i].alSource = 0; calls[i].sendAudioTimer = new QTimer(); - calls[i].sendVideoTimer = new QTimer(); calls[i].sendAudioTimer->moveToThread(coreThread); - calls[i].sendVideoTimer->moveToThread(coreThread); - connect(calls[i].sendVideoTimer, &QTimer::timeout, [this,i](){sendCallVideo(i);}); } // OpenAL init diff --git a/src/core/core.h b/src/core/core.h index ed9965ae7..c1eacaf72 100644 --- a/src/core/core.h +++ b/src/core/core.h @@ -30,11 +30,11 @@ #include "toxid.h" template class QList; -class Camera; class QTimer; class QString; class CString; class VideoSource; +class VideoFrame; #ifdef QTOX_FILTER_AUDIO class AudioFilterer; #endif @@ -45,7 +45,7 @@ class Core : public QObject public: enum PasswordType {ptMain = 0, ptHistory, ptCounter}; - explicit Core(Camera* cam, QThread* coreThread, QString initialLoadPath); + explicit Core(QThread* coreThread, QString initialLoadPath); static Core* getInstance(); ///< Returns the global widget's Core instance ~Core(); @@ -274,7 +274,7 @@ private: static void playAudioBuffer(ALuint alSource, const int16_t *data, int samples, unsigned channels, int sampleRate); static void playCallVideo(void *toxav, int32_t callId, const vpx_image_t* img, void *user_data); - void sendCallVideo(int callId); + static void sendCallVideo(int callId, ToxAv* toxav, std::shared_ptr frame); bool checkConnection(); @@ -292,7 +292,6 @@ private: Tox* tox; ToxAv* toxav; QTimer *toxTimer, *fileTimer; //, *saveTimer; - Camera* camera; QString loadPath; // meaningless after start() is called int dhtServerId; static ToxCall calls[TOXAV_MAX_CALLS]; diff --git a/src/core/coreav.cpp b/src/core/coreav.cpp index 6de3028df..2db076942 100644 --- a/src/core/coreav.cpp +++ b/src/core/coreav.cpp @@ -15,7 +15,9 @@ */ #include "core.h" -#include "src/video/camera.h" +#include "src/video/camerasource.h" +#include "src/video/corevideosource.h" +#include "src/video/videoframe.h" #include "src/audio.h" #ifdef QTOX_FILTER_AUDIO #include "src/audiofilterer.h" @@ -71,12 +73,13 @@ void Core::prepareCall(uint32_t friendId, int32_t callId, ToxAv* toxav, bool vid calls[callId].sendAudioTimer->setSingleShot(true); connect(calls[callId].sendAudioTimer, &QTimer::timeout, [=](){sendCallAudio(callId,toxav);}); calls[callId].sendAudioTimer->start(); - calls[callId].sendVideoTimer->setInterval(50); - calls[callId].sendVideoTimer->setSingleShot(true); if (calls[callId].videoEnabled) { - calls[callId].sendVideoTimer->start(); - Camera::getInstance()->subscribe(); + calls[callId].videoSource = new CoreVideoSource; + calls[callId].camera = new CameraSource; + calls[callId].camera->subscribe(); + connect(calls[callId].camera, &VideoSource::frameAvailable, + [=](std::shared_ptr frame){sendCallVideo(callId,toxav,frame);}); } #ifdef QTOX_FILTER_AUDIO @@ -109,17 +112,20 @@ void Core::onAvMediaChange(void* toxav, int32_t callId, void* core) if (cap == (av_VideoEncoding|av_VideoDecoding)) // Video call { - Camera::getInstance()->subscribe(); - calls[callId].videoEnabled = true; - calls[callId].sendVideoTimer->start(); emit static_cast(core)->avMediaChange(friendId, callId, true); + calls[callId].videoSource = new CoreVideoSource; + calls[callId].camera = new CameraSource; + calls[callId].camera->subscribe(); + calls[callId].videoEnabled = true; } else // Audio call { - calls[callId].videoEnabled = false; - calls[callId].sendVideoTimer->stop(); - Camera::getInstance()->unsubscribe(); emit static_cast(core)->avMediaChange(friendId, callId, false); + calls[callId].videoEnabled = false; + delete calls[callId].camera; + calls[callId].camera = nullptr; + calls[callId].videoSource->setDeleteOnClose(true); + calls[callId].videoSource = nullptr; } return; @@ -226,9 +232,13 @@ void Core::cleanupCall(int32_t callId) calls[callId].active = false; disconnect(calls[callId].sendAudioTimer,0,0,0); calls[callId].sendAudioTimer->stop(); - calls[callId].sendVideoTimer->stop(); if (calls[callId].videoEnabled) - Camera::getInstance()->unsubscribe(); + { + delete calls[callId].camera; + calls[callId].camera = nullptr; + calls[callId].videoSource->setDeleteOnClose(true); + calls[callId].videoSource = nullptr; + } Audio::unsuscribeInput(); toxav_kill_transmission(Core::getInstance()->toxav, callId); @@ -314,37 +324,35 @@ void Core::playCallVideo(void*, int32_t callId, const vpx_image_t* img, void *us if (!calls[callId].active || !calls[callId].videoEnabled) return; - calls[callId].videoSource.pushVPXFrame(img); + calls[callId].videoSource->pushFrame(img); } -void Core::sendCallVideo(int32_t callId) +void Core::sendCallVideo(int32_t callId, ToxAv* toxav, std::shared_ptr vframe) { if (!calls[callId].active || !calls[callId].videoEnabled) return; - vpx_image frame = camera->getLastFrame().createVpxImage(); - if (frame.w && frame.h) + // This frame shares vframe's buffers, we don't call vpx_img_free but just delete it + vpx_image* frame = vframe->toVpxImage(); + if (frame->fmt == VPX_IMG_FMT_NONE) { - int result; - if ((result = toxav_prepare_video_frame(toxav, callId, videobuf, videobufsize, &frame)) < 0) - { - qDebug() << QString("toxav_prepare_video_frame: error %1").arg(result); - vpx_img_free(&frame); - calls[callId].sendVideoTimer->start(); - return; - } - - if ((result = toxav_send_video(toxav, callId, (uint8_t*)videobuf, result)) < 0) - qDebug() << QString("toxav_send_video error: %1").arg(result); - - vpx_img_free(&frame); - } - else - { - qDebug("sendCallVideo: Invalid frame (bad camera ?)"); + qWarning() << "Invalid frame"; + delete frame; + return; } - calls[callId].sendVideoTimer->start(); + int result; + if ((result = toxav_prepare_video_frame(toxav, callId, videobuf, videobufsize, frame)) < 0) + { + qDebug() << QString("toxav_prepare_video_frame: error %1").arg(result); + delete frame; + return; + } + + if ((result = toxav_send_video(toxav, callId, (uint8_t*)videobuf, result)) < 0) + qDebug() << QString("toxav_send_video error: %1").arg(result); + + delete frame; } void Core::micMuteToggle(int32_t callId) @@ -412,9 +420,9 @@ void Core::onAvEnd(void* _toxav, int32_t call_index, void* core) } qDebug() << QString("AV end from %1").arg(friendId); - cleanupCall(call_index); - emit static_cast(core)->avEnd(friendId, call_index); + + cleanupCall(call_index); } void Core::onAvRinging(void* _toxav, int32_t call_index, void* core) @@ -452,9 +460,9 @@ void Core::onAvRequestTimeout(void* _toxav, int32_t call_index, void* core) } qDebug() << QString("AV request timeout with %1").arg(friendId); - cleanupCall(call_index); - emit static_cast(core)->avRequestTimeout(friendId, call_index); + + cleanupCall(call_index); } void Core::onAvPeerTimeout(void* _toxav, int32_t call_index, void* core) @@ -469,9 +477,9 @@ void Core::onAvPeerTimeout(void* _toxav, int32_t call_index, void* core) } qDebug() << QString("AV peer timeout with %1").arg(friendId); - cleanupCall(call_index); - emit static_cast(core)->avPeerTimeout(friendId, call_index); + + cleanupCall(call_index); } @@ -593,7 +601,7 @@ void Core::playAudioBuffer(ALuint alSource, const int16_t *data, int samples, un VideoSource *Core::getVideoSourceFromCall(int callNumber) { - return &calls[callNumber].videoSource; + return calls[callNumber].videoSource; } void Core::joinGroupCall(int groupId) diff --git a/src/core/coreav.h b/src/core/coreav.h index 6b81b29f1..b4b057471 100644 --- a/src/core/coreav.h +++ b/src/core/coreav.h @@ -3,7 +3,6 @@ #include #include -#include "src/video/netvideosource.h" #if defined(__APPLE__) && defined(__MACH__) #include @@ -14,11 +13,13 @@ #endif class QTimer; +class CoreVideoSource; +class CameraSource; struct ToxCall { ToxAvCSettings codecSettings; - QTimer *sendAudioTimer, *sendVideoTimer; + QTimer *sendAudioTimer; int32_t callId; uint32_t friendId; bool videoEnabled; @@ -26,7 +27,8 @@ struct ToxCall bool muteMic; bool muteVol; ALuint alSource; - NetVideoSource videoSource; + CoreVideoSource* videoSource; + CameraSource* camera; }; struct ToxGroupCall diff --git a/src/core/corefile.h b/src/core/corefile.h index 40539e8d3..e94481fd8 100644 --- a/src/core/corefile.h +++ b/src/core/corefile.h @@ -55,7 +55,6 @@ private: private: static QMutex fileSendMutex; static QHash fileMap; - /// TODO: Replace the two queues by a hash map uint64_t -> unique_ptr }; #endif // COREFILE_H diff --git a/src/main.cpp b/src/main.cpp index 81a023d67..4c24ff7f7 100644 --- a/src/main.cpp +++ b/src/main.cpp @@ -29,7 +29,6 @@ #include #include #include -#include #include diff --git a/src/misc/settings.cpp b/src/misc/settings.cpp index 6cc1bcad5..3b724bb01 100644 --- a/src/misc/settings.cpp +++ b/src/misc/settings.cpp @@ -314,6 +314,7 @@ void Settings::load() s.endGroup(); s.beginGroup("Video"); + videoDev = s.value("videoDev", "").toString(); camVideoRes = s.value("camVideoRes",QSize()).toSize(); s.endGroup(); @@ -479,6 +480,7 @@ void Settings::saveGlobal(QString path) s.endGroup(); s.beginGroup("Video"); + s.setValue("videoDev", videoDev); s.setValue("camVideoRes",camVideoRes); s.endGroup(); } @@ -1124,6 +1126,16 @@ void Settings::setInDev(const QString& deviceSpecifier) inDev = deviceSpecifier; } +QString Settings::getVideoDev() const +{ + return videoDev; +} + +void Settings::setVideoDev(const QString& deviceSpecifier) +{ + videoDev = deviceSpecifier; +} + QString Settings::getOutDev() const { return outDev; diff --git a/src/misc/settings.h b/src/misc/settings.h index 3b01b1bc7..a56aa2506 100644 --- a/src/misc/settings.h +++ b/src/misc/settings.h @@ -149,6 +149,9 @@ public: bool getFilterAudio() const; void setFilterAudio(bool newValue); + QString getVideoDev() const; + void setVideoDev(const QString& deviceSpecifier); + QSize getCamVideoRes() const; void setCamVideoRes(QSize newValue); @@ -347,6 +350,7 @@ private: bool filterAudio; // Video + QString videoDev; QSize camVideoRes; struct friendProp diff --git a/src/nexus.cpp b/src/nexus.cpp index f129e9b6f..bb31d627d 100644 --- a/src/nexus.cpp +++ b/src/nexus.cpp @@ -1,7 +1,7 @@ #include "nexus.h" #include "src/core/core.h" #include "misc/settings.h" -#include "video/camera.h" +#include "video/camerasource.h" #include "widget/gui.h" #include #include @@ -57,6 +57,7 @@ void Nexus::start() qRegisterMetaType("ToxFile"); qRegisterMetaType("ToxFile::FileDirection"); qRegisterMetaType("Core::PasswordType"); + qRegisterMetaType>("std::shared_ptr"); // Create GUI #ifndef Q_OS_ANDROID @@ -67,7 +68,7 @@ void Nexus::start() QString profilePath = Settings::getInstance().detectProfile(); coreThread = new QThread(this); coreThread->setObjectName("qTox Core"); - core = new Core(Camera::getInstance(), coreThread, profilePath); + core = new Core(coreThread, profilePath); core->moveToThread(coreThread); connect(coreThread, &QThread::started, core, &Core::start); diff --git a/src/video/camera.cpp b/src/video/camera.cpp deleted file mode 100644 index e7df04834..000000000 --- a/src/video/camera.cpp +++ /dev/null @@ -1,113 +0,0 @@ -/* - This file is part of qTox, a Qt-based graphical interface for Tox. - - This program 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. - 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 COPYING file for more details. -*/ - -#include "camera.h" -#include "src/video/cameraworker.h" -#include -#include -#include - -Camera* Camera::getInstance() -{ - static Camera instance; - - return &instance; -} - -Camera::Camera() - : refcount(0) - , workerThread(nullptr) - , worker(nullptr) -{ - worker = new CameraWorker(0); - workerThread = new QThread(); - - worker->moveToThread(workerThread); - - connect(workerThread, &QThread::started, worker, &CameraWorker::onStart); - connect(workerThread, &QThread::finished, worker, &CameraWorker::deleteLater); - connect(worker, &CameraWorker::newFrameAvailable, this, &Camera::onNewFrameAvailable); - - connect(worker, &CameraWorker::resProbingFinished, this, &Camera::resolutionProbingFinished); - connect(worker, &CameraWorker::propProbingFinished, this, [=](int prop, double val) { emit propProbingFinished(Prop(prop), val); } ); - - workerThread->start(); -} - -Camera::~Camera() -{ - workerThread->exit(); - workerThread->deleteLater(); -} - -void Camera::subscribe() -{ - if (refcount++ <= 0) - worker->resume(); -} - -void Camera::unsubscribe() -{ - if (--refcount <= 0) - { - worker->suspend(); - refcount = 0; - } -} - -void Camera::probeProp(Camera::Prop prop) -{ - worker->probeProp(int(prop)); -} - -void Camera::probeResolutions() -{ - worker->probeResolutions(); -} - -void Camera::setResolution(QSize res) -{ - worker->setProp(CV_CAP_PROP_FRAME_WIDTH, res.width()); - worker->setProp(CV_CAP_PROP_FRAME_HEIGHT, res.height()); -} - -QSize Camera::getCurrentResolution() -{ - return QSize(worker->getProp(CV_CAP_PROP_FRAME_WIDTH), worker->getProp(CV_CAP_PROP_FRAME_HEIGHT)); -} - -void Camera::setProp(Camera::Prop prop, double val) -{ - worker->setProp(int(prop), val); -} - -double Camera::getProp(Camera::Prop prop) -{ - return worker->getProp(int(prop)); -} - -void Camera::onNewFrameAvailable(const VideoFrame &frame) -{ - emit frameAvailable(frame); - - mutex.lock(); - currFrame = frame; - mutex.unlock(); -} - -VideoFrame Camera::getLastFrame() -{ - QMutexLocker lock(&mutex); - return currFrame; -} diff --git a/src/video/camera.h b/src/video/camera.h deleted file mode 100644 index 138b38ca8..000000000 --- a/src/video/camera.h +++ /dev/null @@ -1,84 +0,0 @@ -/* - This file is part of qTox, a Qt-based graphical interface for Tox. - - This program 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. - 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 COPYING file for more details. -*/ - -#ifndef CAMERA_H -#define CAMERA_H - -#include -#include -#include -#include "vpx/vpx_image.h" -#include "opencv2/highgui/highgui.hpp" -#include "src/video/videosource.h" - -class CameraWorker; - -/** - * 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 only when needed, and giving access to the last frames - **/ - -class Camera : public VideoSource -{ - Q_OBJECT -public: - enum Prop : int { - BRIGHTNESS = CV_CAP_PROP_BRIGHTNESS, - SATURATION = CV_CAP_PROP_SATURATION, - CONTRAST = CV_CAP_PROP_CONTRAST, - HUE = CV_CAP_PROP_HUE, - WIDTH = CV_CAP_PROP_FRAME_WIDTH, - HEIGHT = CV_CAP_PROP_FRAME_HEIGHT, - }; - - ~Camera(); - - static Camera* getInstance(); ///< Returns the global widget's Camera instance - VideoFrame getLastFrame(); - - void setResolution(QSize res); - QSize getCurrentResolution(); - - void setProp(Prop prop, double val); - double getProp(Prop prop); - - void probeProp(Prop prop); - void probeResolutions(); - - // VideoSource interface - virtual void subscribe(); - virtual void unsubscribe(); - -signals: - void resolutionProbingFinished(QList res); - void propProbingFinished(Prop prop, double val); - -protected: - Camera(); - -private: - int refcount; ///< Number of users suscribed to the camera - VideoFrame currFrame; - QMutex mutex; - - QThread* workerThread; - CameraWorker* worker; - -private slots: - void onNewFrameAvailable(const VideoFrame& frame); - -}; - -#endif // CAMERA_H diff --git a/src/video/cameradevice.cpp b/src/video/cameradevice.cpp new file mode 100644 index 000000000..713698f4a --- /dev/null +++ b/src/video/cameradevice.cpp @@ -0,0 +1,140 @@ +#include +extern "C" { +#include +#include +} +#include "cameradevice.h" +#include "src/misc/settings.h" + +QHash CameraDevice::openDevices; +QMutex CameraDevice::openDeviceLock, CameraDevice::iformatLock; +AVInputFormat* CameraDevice::iformat{nullptr}; + +CameraDevice::CameraDevice(const QString devName, AVFormatContext *context) + : devName{devName}, context{context}, refcount{1} +{ + +} + +CameraDevice* CameraDevice::open(QString devName) +{ + openDeviceLock.lock(); + AVFormatContext* fctx = nullptr; + CameraDevice* dev = openDevices.value(devName); + if (dev) + goto out; + + if (avformat_open_input(&fctx, devName.toStdString().c_str(), nullptr, nullptr)<0) + goto out; + + if (avformat_find_stream_info(fctx, NULL) < 0) + { + avformat_close_input(&fctx); + goto out; + } + + dev = new CameraDevice{devName, fctx}; + openDevices[devName] = dev; + +out: + openDeviceLock.unlock(); + return dev; +} + +void CameraDevice::open() +{ + ++refcount; +} + +bool CameraDevice::close() +{ + if (--refcount <= 0) + { + openDeviceLock.lock(); + openDevices.remove(devName); + openDeviceLock.unlock(); + avformat_close_input(&context); + delete this; + return true; + } + else + { + return false; + } +} + +QVector> CameraDevice::getDeviceList() +{ + QVector> devices; + + if (!getDefaultInputFormat()) + return devices; + + AVDeviceInfoList* devlist; + avdevice_list_input_sources(iformat, nullptr, nullptr, &devlist); + + devices.resize(devlist->nb_devices); + for (int i=0; inb_devices; i++) + { + AVDeviceInfo* dev = devlist->devices[i]; + devices[i].first = dev->device_name; + devices[i].second = dev->device_description; + } + + return devices; +} + +QString CameraDevice::getDefaultDeviceName() +{ + QString device = Settings::getInstance().getVideoDev(); + + if (!getDefaultInputFormat()) + return device; + + bool actuallyExists = false; + AVDeviceInfoList* devlist; + avdevice_list_input_sources(iformat, nullptr, nullptr, &devlist); + + for (int i=0; inb_devices; i++) + { + if (device == devlist->devices[i]->device_name) + { + actuallyExists = true; + break; + } + } + if (actuallyExists) + return device; + + if (!devlist->nb_devices) + return QString(); + + int defaultDev = devlist->default_device == -1 ? 0 : devlist->default_device; + return QString(devlist->devices[defaultDev]->device_name); +} + +bool CameraDevice::getDefaultInputFormat() +{ + QMutexLocker locker(&iformatLock); + if (iformat) + return true; + + avdevice_register_all(); + + // Linux + if ((iformat = av_find_input_format("v4l2"))) + return true; + + // Windows + if ((iformat = av_find_input_format("dshow"))) + return true; + + // Mac + if ((iformat = av_find_input_format("avfoundation"))) + return true; + if ((iformat = av_find_input_format("qtkit"))) + return true; + + qWarning() << "No valid input format found"; + return false; +} diff --git a/src/video/cameradevice.h b/src/video/cameradevice.h new file mode 100644 index 000000000..39d647bec --- /dev/null +++ b/src/video/cameradevice.h @@ -0,0 +1,51 @@ +#ifndef CAMERADEVICE_H +#define CAMERADEVICE_H + +#include +#include +#include +#include +#include + +struct AVFormatContext; +struct AVInputFormat; + +/// Maintains an FFmpeg context for open camera devices, +/// takes care of sharing the context accross users +/// and closing the camera device when not in use. +/// The device can be opened recursively, +/// and must then be closed recursively +class CameraDevice +{ +public: + /// Opens a device, creating a new one if needed + /// Returns a nullptr if the device couldn't be opened + static CameraDevice* open(QString devName); + void open(); ///< Opens the device again. Never fails + bool close(); ///< Closes the device. Never fails. If returns true, "this" becomes invalid + + /// Returns a list of device names and descriptions + /// The names are the first part of the pair and can be passed to open(QString) + static QVector > getDeviceList(); + + /// Returns the short name of the default defice + /// This is either the device in the settings + /// or the system default. + static QString getDefaultDeviceName(); + +private: + CameraDevice(const QString devName, AVFormatContext *context); + static bool getDefaultInputFormat(); ///< Sets CameraDevice::iformat, returns success/failure + +public: + const QString devName; ///< Short name of the device + AVFormatContext* context; ///< Context of the open device, must always be valid + +private: + std::atomic_int refcount; ///< Number of times the device was opened + static QHash openDevices; + static QMutex openDeviceLock, iformatLock; + static AVInputFormat* iformat; +}; + +#endif // CAMERADEVICE_H diff --git a/src/video/camerasource.cpp b/src/video/camerasource.cpp new file mode 100644 index 000000000..51d8ff8a2 --- /dev/null +++ b/src/video/camerasource.cpp @@ -0,0 +1,273 @@ +/* + This file is part of qTox, a Qt-based graphical interface for Tox. + + This program 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. + 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 COPYING file for more details. +*/ + +extern "C" { +#include +#include +#include +#include +} +#include +#include +#include +#include +#include +#include "camerasource.h" +#include "cameradevice.h" +#include "videoframe.h" + +CameraSource::CameraSource() + : CameraSource{CameraDevice::getDefaultDeviceName()} +{ +} + +CameraSource::CameraSource(const QString deviceName) + : deviceName{deviceName}, device{nullptr}, + cctx{nullptr}, videoStreamIndex{-1}, + biglock{false}, freelistLock{false}, subscriptions{0} +{ + av_register_all(); + avdevice_register_all(); +} + +CameraSource::~CameraSource() +{ + // Fast lock, in case our stream thread is running + { + bool expected = false; + while (!biglock.compare_exchange_weak(expected, true)) + expected = false; + } + + if (cctx) + avcodec_free_context(&cctx); + + for (int i=subscriptions; i; --i) + device->close(); + device = nullptr; + biglock=false; + + // Synchronize with our stream thread + while (streamFuture.isRunning()) + QThread::yieldCurrentThread(); +} + +bool CameraSource::subscribe() +{ + // Fast lock + { + bool expected = false; + while (!biglock.compare_exchange_weak(expected, true)) + expected = false; + } + + if (device) + { + device->open(); + ++subscriptions; + biglock = false; + return true; + } + + // We need to create a new CameraDevice + AVCodec* codec; + device = CameraDevice::open(deviceName); + if (!device) + { + biglock = false; + return false; + } + + // Find the first video stream + for (unsigned i=0; icontext->nb_streams; i++) + { + if(device->context->streams[i]->codec->codec_type==AVMEDIA_TYPE_VIDEO) + { + videoStreamIndex=i; + break; + } + } + if (videoStreamIndex == -1) + goto fail; + + // Get a pointer to the codec context for the video stream + cctxOrig=device->context->streams[videoStreamIndex]->codec; + codec=avcodec_find_decoder(cctxOrig->codec_id); + if(!codec) + goto fail; + + // Copy context, since we apparently aren't allowed to use the original + cctx = avcodec_alloc_context3(codec); + if(avcodec_copy_context(cctx, cctxOrig) != 0) + goto fail; + cctx->refcounted_frames = 1; + + // Open codec + if(avcodec_open2(cctx, codec, nullptr)<0) + { + avcodec_free_context(&cctx); + goto fail; + } + + if (streamFuture.isRunning()) + qCritical() << "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(); + + ++subscriptions; + biglock = false; + return true; + +fail: + while (!device->close()) {} + biglock = false; + return false; +} + +void CameraSource::unsubscribe() +{ + // Fast lock + { + bool expected = false; + while (!biglock.compare_exchange_weak(expected, true)) + expected = false; + } + + if (!device) + { + qWarning() << "Unsubscribing with zero subscriber"; + biglock = false; + return; + } + + if (--subscriptions == 0) + { + // Free all remaining VideoFrame + // Locking must be done precisely this way to avoid races + for (int i=0; i vframe = freelist[i].lock(); + if (!vframe) + continue; + vframe->releaseFrame(); + } + + // Free our resources and close the device + videoStreamIndex = -1; + avcodec_free_context(&cctx); + avcodec_close(cctxOrig); + cctxOrig = nullptr; + device->close(); + device = nullptr; + + biglock = false; + + // Synchronize with our stream thread + while (streamFuture.isRunning()) + QThread::yieldCurrentThread(); + } + else + { + device->close(); + biglock = false; + } +} + +void CameraSource::stream() +{ + auto streamLoop = [=]() + { + AVFrame* frame = av_frame_alloc(); + if (!frame) + return; + frame->opaque = nullptr; + + AVPacket packet; + if (av_read_frame(device->context, &packet)<0) + 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; + + // Broadcast a new VideoFrame, it takes ownership of the AVFrame + { + bool expected = false; + while (!freelistLock.compare_exchange_weak(expected, true)) + expected = false; + } + int freeFreelistSlot = getFreelistSlotLockless(); + auto frameFreeCb = std::bind(&CameraSource::freelistCallback, this, freeFreelistSlot); + std::shared_ptr vframe = std::make_shared(frame, frameFreeCb); + freelist.append(vframe); + freelistLock = false; + emit frameAvailable(vframe); + } + + // Free the packet that was allocated by av_read_frame + av_free_packet(&packet); + }; + + forever { + // Fast lock + { + bool expected = false; + while (!biglock.compare_exchange_weak(expected, true)) + expected = false; + } + + if (!device) + { + biglock = false; + return; + } + + streamLoop(); + + // Give a chance to other functions to pick up the lock if needed + biglock = false; + QThread::yieldCurrentThread(); + } +} + +void CameraSource::freelistCallback(int freelistIndex) +{ + // Fast lock + { + bool expected = false; + while (!freelistLock.compare_exchange_weak(expected, true)) + expected = false; + } + freelist[freelistIndex].reset(); + freelistLock = false; +} + +int CameraSource::getFreelistSlotLockless() +{ + int size = freelist.size(); + for (int i=0; i>1)+4); // Arbitrary growth strategy, should work well + return size; +} diff --git a/src/video/camerasource.h b/src/video/camerasource.h new file mode 100644 index 000000000..397c7f5ca --- /dev/null +++ b/src/video/camerasource.h @@ -0,0 +1,72 @@ +/* + This file is part of qTox, a Qt-based graphical interface for Tox. + + This program 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. + 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 COPYING file for more details. +*/ + +#ifndef CAMERA_H +#define CAMERA_H + +#include +#include +#include +#include +#include +#include "src/video/videosource.h" + +class CameraDevice; +struct AVCodecContext; + +/** + * 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. + **/ + +class CameraSource : public VideoSource +{ + Q_OBJECT +public: + CameraSource(); ///< Opens the camera device in the settings, or the system default + CameraSource(const QString deviceName); + ~CameraSource(); + + // VideoSource interface + virtual bool subscribe() override; + virtual void unsubscribe() override; + +private: + /// Blocking. Decodes video stream and emits new frames. + /// Designed to run in its own thread. + void stream(); + /// All VideoFrames must be deleted or released before we can close the device + /// or the device will forcibly free them, and then ~VideoFrame() will double free. + /// In theory very careful coding from our users could ensure all VideoFrames + /// die before unsubscribing, even the ones currently in flight in the metatype system. + /// But that's just asking for trouble and mysterious crashes, so we'll just + /// maintain a freelist and have all VideoFrames tell us when they die so we can forget them. + void freelistCallback(int freelistIndex); + /// Get the index of a free slot in the freelist + /// Callers must hold the freelistLock + int getFreelistSlotLockless(); + +private: + QVector> freelist; ///< Frames that need freeing before we can safely close the device + QFuture streamFuture; ///< Future of the streaming thread + const QString deviceName; ///< Short name of the device for CameraDevice's open(QString) + CameraDevice* device; ///< Non-owning pointer to an open CameraDevice, or nullptr + AVCodecContext* cctx, *cctxOrig; ///< Codec context of the camera's selected video stream + int videoStreamIndex; ///< A camera can have multiple streams, this is the one we're decoding + std::atomic_bool biglock, freelistLock; ///< True when locked. Faster than mutexes for video decoding. + std::atomic_int subscriptions; ///< Remember how many times we subscribed for RAII +}; + +#endif // CAMERA_H diff --git a/src/video/cameraworker.cpp b/src/video/cameraworker.cpp deleted file mode 100644 index 2e78a3d4d..000000000 --- a/src/video/cameraworker.cpp +++ /dev/null @@ -1,240 +0,0 @@ -/* - This file is part of qTox, a Qt-based graphical interface for Tox. - - This program 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. - 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 COPYING file for more details. -*/ - -#include "cameraworker.h" - -#include -#include -#include - -CameraWorker::CameraWorker(int index) - : clock(nullptr) - , camIndex(index) - , refCount(0) -{ - qRegisterMetaType(); - qRegisterMetaType>(); -} - -CameraWorker::~CameraWorker() -{ - if (clock) - delete clock; -} - -void CameraWorker::onStart() -{ - if (!clock) - { - clock = new QTimer(this); - clock->setSingleShot(false); - clock->setInterval(1000/60); - - connect(clock, &QTimer::timeout, this, &CameraWorker::doWork); - } - emit started(); -} - -void CameraWorker::_suspend() -{ - qDebug() << "Suspend"; - clock->stop(); - unsubscribe(); -} - -void CameraWorker::_resume() -{ - qDebug() << "Resume"; - subscribe(); - clock->start(); -} - -void CameraWorker::_setProp(int prop, double val) -{ - props[prop] = val; - - if (cam.isOpened()) - cam.set(prop, val); -} - -double CameraWorker::_getProp(int prop) -{ - if (!props.contains(prop)) - { - subscribe(); - props[prop] = cam.get(prop); - emit propProbingFinished(prop, props[prop]); - unsubscribe(); - } - - return props.value(prop); -} - -void CameraWorker::_probeResolutions() -{ - if (resolutions.isEmpty()) - { - subscribe(); - - // probe resolutions - QList propbeRes = { - QSize( 160, 120), // QQVGA - QSize( 320, 240), // HVGA - QSize( 432, 240), // WQVGA - QSize( 640, 360), // nHD - QSize( 640, 480), - QSize( 800, 600), - QSize( 960, 640), - QSize(1024, 768), // XGA - QSize(1280, 720), - QSize(1280, 1024), - QSize(1360, 768), - QSize(1366, 768), - QSize(1400, 1050), - QSize(1440, 900), - QSize(1600, 1200), - QSize(1680, 1050), - QSize(1920, 1200), - }; - - for (QSize res : propbeRes) - { - cam.set(CV_CAP_PROP_FRAME_WIDTH, res.width()); - cam.set(CV_CAP_PROP_FRAME_HEIGHT, res.height()); - - double w = cam.get(CV_CAP_PROP_FRAME_WIDTH); - double h = cam.get(CV_CAP_PROP_FRAME_HEIGHT); - - //qDebug() << "PROBING:" << res << " got " << w << h; - - if (w>0 && h>0 && !resolutions.contains(QSize(w,h))) - resolutions.append(QSize(w,h)); - } - - unsubscribe(); - - qDebug() << "Resolutions" <clock->stop(); // prevent log spamming - qDebug() << "stopped clock"; - } - - if (!bSuccess) - { - qDebug() << "Cannot read frame"; - return; - } - - QByteArray frameData = QByteArray::fromRawData(reinterpret_cast(frame.data), frame.total() * frame.channels()); - - emit newFrameAvailable(VideoFrame{frameData, QSize(frame.cols, frame.rows), VideoFrame::BGR}); -} - -void CameraWorker::suspend() -{ - QMetaObject::invokeMethod(this, "_suspend"); -} - -void CameraWorker::resume() -{ - QMetaObject::invokeMethod(this, "_resume"); -} - -void CameraWorker::setProp(int prop, double val) -{ - QMetaObject::invokeMethod(this, "_setProp", Q_ARG(int, prop), Q_ARG(double, val)); -} - -void CameraWorker::probeProp(int prop) -{ - QMetaObject::invokeMethod(this, "_getProp", Q_ARG(int, prop)); -} - -void CameraWorker::probeResolutions() -{ - QMetaObject::invokeMethod(this, "_probeResolutions"); -} - -double CameraWorker::getProp(int prop) -{ - double ret = 0.0; - QMetaObject::invokeMethod(this, "_getProp", Qt::BlockingQueuedConnection, Q_RETURN_ARG(double, ret), Q_ARG(int, prop)); - - return ret; -} diff --git a/src/video/cameraworker.h b/src/video/cameraworker.h deleted file mode 100644 index dbdc1d95b..000000000 --- a/src/video/cameraworker.h +++ /dev/null @@ -1,78 +0,0 @@ -/* - This file is part of qTox, a Qt-based graphical interface for Tox. - - This program 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. - 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 COPYING file for more details. -*/ - -#ifndef CAMERAWORKER_H -#define CAMERAWORKER_H - -#include -#include -#include -#include -#include -#include - -#include "opencv2/highgui/highgui.hpp" -#include "videosource.h" - -class QTimer; - -class CameraWorker : public QObject -{ - Q_OBJECT -public: - CameraWorker(int index); - ~CameraWorker(); - void doWork(); - - void suspend(); - void resume(); - void setProp(int prop, double val); - double getProp(int prop); // blocking call! - -public slots: - void onStart(); - void probeProp(int prop); - void probeResolutions(); - -signals: - void started(); - void newFrameAvailable(const VideoFrame& frame); - void resProbingFinished(QList res); - void propProbingFinished(int prop, double val); - -private slots: - void _suspend(); - void _resume(); - void _setProp(int prop, double val); - double _getProp(int prop); - void _probeResolutions(); - -private: - void applyProps(); - void subscribe(); - void unsubscribe(); - -private: - QMutex mutex; - QQueue queue; - QTimer* clock; - cv::VideoCapture cam; - cv::Mat3b frame; - int camIndex; - QMap props; - QList resolutions; - int refCount; -}; - -#endif // CAMERAWORKER_H diff --git a/src/video/corevideosource.cpp b/src/video/corevideosource.cpp new file mode 100644 index 000000000..7febe4541 --- /dev/null +++ b/src/video/corevideosource.cpp @@ -0,0 +1,108 @@ +extern "C" { +#include +} +#include "corevideosource.h" +#include "videoframe.h" + +CoreVideoSource::CoreVideoSource() + : subscribers{0}, deleteOnClose{false}, + biglock{false} +{ +} + +void CoreVideoSource::pushFrame(const vpx_image_t* vpxframe) +{ + // Fast lock + { + bool expected = false; + while (!biglock.compare_exchange_weak(expected, true)) + expected = false; + } + + std::shared_ptr vframe; + AVFrame* avframe; + uint8_t* buf; + int width = vpxframe->d_w, height = vpxframe->d_h; + int dstStride, srcStride, minStride; + + if (subscribers <= 0) + goto end; + + avframe = av_frame_alloc(); + if (!avframe) + goto end; + avframe->width = width; + avframe->height = height; + avframe->format = AV_PIX_FMT_YUV420P; + + buf = (uint8_t*)av_malloc(avpicture_get_size(AV_PIX_FMT_YUV420P, width, height)); + if (!buf) + { + av_frame_free(&avframe); + goto end; + } + avframe->opaque = buf; + + avpicture_fill((AVPicture*)avframe, buf, AV_PIX_FMT_YUV420P, width, height); + + dstStride=avframe->linesize[0], srcStride=vpxframe->stride[0], minStride=std::min(dstStride, srcStride); + for (int i=0; idata[0]+dstStride*i, vpxframe->planes[0]+srcStride*i, minStride); + dstStride=avframe->linesize[1], srcStride=vpxframe->stride[1], minStride=std::min(dstStride, srcStride); + for (int i=0; idata[1]+dstStride*i, vpxframe->planes[1]+srcStride*i, minStride); + dstStride=avframe->linesize[2], srcStride=vpxframe->stride[2], minStride=std::min(dstStride, srcStride); + for (int i=0; idata[2]+dstStride*i, vpxframe->planes[2]+srcStride*i, minStride); + + vframe = std::make_shared(avframe); + emit frameAvailable(vframe); + +end: + biglock = false; +} + +bool CoreVideoSource::subscribe() +{ + // Fast lock + { + bool expected = false; + while (!biglock.compare_exchange_weak(expected, true)) + expected = false; + } + ++subscribers; + biglock = false; + return true; +} + +void CoreVideoSource::unsubscribe() +{ + // Fast lock + { + bool expected = false; + while (!biglock.compare_exchange_weak(expected, true)) + expected = false; + } + if (--subscribers == 0) + { + if (deleteOnClose) + { + biglock = false; + delete this; + return; + } + } + biglock = false; +} + +void CoreVideoSource::setDeleteOnClose(bool newstate) +{ + // Fast lock + { + bool expected = false; + while (!biglock.compare_exchange_weak(expected, true)) + expected = false; + } + deleteOnClose = newstate; + biglock = false; +} diff --git a/src/video/corevideosource.h b/src/video/corevideosource.h new file mode 100644 index 000000000..91f595c32 --- /dev/null +++ b/src/video/corevideosource.h @@ -0,0 +1,35 @@ +#ifndef COREVIDEOSOURCE_H +#define COREVIDEOSOURCE_H + +#include +#include +#include "videosource.h" + +/// A VideoSource that emits frames received by Core +class CoreVideoSource : public VideoSource +{ + Q_OBJECT +public: + // VideoSource interface + virtual bool subscribe() override; + virtual void unsubscribe() override; + +private: + // Only Core should create a CoreVideoSource since + // only Core can push images to it + CoreVideoSource(); + + /// Makes a copy of the vpx_image_t and emits it as a new VideoFrame + void pushFrame(const vpx_image_t *frame); + /// If true, self-delete after the last suscriber is gone + void setDeleteOnClose(bool newstate); + +private: + std::atomic_int subscribers; ///< Number of suscribers + std::atomic_bool deleteOnClose; ///< If true, self-delete after the last suscriber is gone + std::atomic_bool biglock; ///< Fast lock + +friend class Core; +}; + +#endif // COREVIDEOSOURCE_H diff --git a/src/video/netvideosource.cpp b/src/video/netvideosource.cpp deleted file mode 100644 index 3b411721b..000000000 --- a/src/video/netvideosource.cpp +++ /dev/null @@ -1,62 +0,0 @@ -/* - This file is part of qTox, a Qt-based graphical interface for Tox. - - This program 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. - 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 COPYING file for more details. -*/ - -#include "netvideosource.h" - -#include -#include - -NetVideoSource::NetVideoSource() -{ -} - -void NetVideoSource::pushFrame(VideoFrame frame) -{ - emit frameAvailable(frame); -} - -void NetVideoSource::pushVPXFrame(const vpx_image *image) -{ - const int dw = image->d_w; - const int dh = image->d_h; - - const int bpl = image->stride[VPX_PLANE_Y]; - const int cxbpl = image->stride[VPX_PLANE_V]; - - VideoFrame frame; - frame.frameData.resize(dw * dh * 3); //YUV 24bit - frame.resolution = QSize(dw, dh); - frame.format = VideoFrame::YUV; - - const uint8_t* yData = image->planes[VPX_PLANE_Y]; - const uint8_t* uData = image->planes[VPX_PLANE_U]; - const uint8_t* vData = image->planes[VPX_PLANE_V]; - - // convert from planar to packed - for (int y = 0; y < dh; ++y) - { - for (int x = 0; x < dw; ++x) - { - uint8_t Y = yData[x + y * bpl]; - uint8_t U = uData[x/(1 << image->x_chroma_shift) + y/(1 << image->y_chroma_shift)*cxbpl]; - uint8_t V = vData[x/(1 << image->x_chroma_shift) + y/(1 << image->y_chroma_shift)*cxbpl]; - - frame.frameData.data()[dw * 3 * y + x * 3 + 0] = Y; - frame.frameData.data()[dw * 3 * y + x * 3 + 1] = U; - frame.frameData.data()[dw * 3 * y + x * 3 + 2] = V; - } - } - - pushFrame(frame); -} diff --git a/src/video/netvideosource.h b/src/video/netvideosource.h deleted file mode 100644 index 0d1dad258..000000000 --- a/src/video/netvideosource.h +++ /dev/null @@ -1,34 +0,0 @@ -/* - This file is part of qTox, a Qt-based graphical interface for Tox. - - This program 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. - 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 COPYING file for more details. -*/ - -#ifndef NETVIDEOSOURCE_H -#define NETVIDEOSOURCE_H - -#include "videosource.h" - -struct vpx_image; - -class NetVideoSource : public VideoSource -{ -public: - NetVideoSource(); - - void pushFrame(VideoFrame frame); - void pushVPXFrame(const vpx_image *image); - - virtual void subscribe() {} - virtual void unsubscribe() {} -}; - -#endif // NETVIDEOSOURCE_H diff --git a/src/video/videoframe.cpp b/src/video/videoframe.cpp index 7e11e0b57..541d7bafd 100644 --- a/src/video/videoframe.cpp +++ b/src/video/videoframe.cpp @@ -12,43 +12,206 @@ See the COPYING file for more details. */ +#include +#include +#include +extern "C" { +#include +#include +} #include "videoframe.h" -vpx_image_t VideoFrame::createVpxImage() const +VideoFrame::VideoFrame(AVFrame* frame, int w, int h, int fmt, std::function freelistCallback) + : freelistCallback{freelistCallback}, + frameOther{nullptr}, frameYUV420{nullptr}, frameRGB24{nullptr}, + width{w}, height{h}, pixFmt{fmt} { - vpx_image img; - img.w = img.h = img.d_w = img.d_h = 0; + if (pixFmt == AV_PIX_FMT_YUV420P) + frameYUV420 = frame; + else if (pixFmt == AV_PIX_FMT_RGB24) + frameRGB24 = frame; + else + frameOther = frame; +} - if (!isValid()) +VideoFrame::VideoFrame(AVFrame* frame, std::function freelistCallback) + : VideoFrame{frame, frame->width, frame->height, frame->format, freelistCallback} +{ +} + +VideoFrame::VideoFrame(AVFrame* frame) + : VideoFrame{frame, frame->width, frame->height, frame->format, nullptr} +{ +} + +VideoFrame::~VideoFrame() +{ + if (freelistCallback) + freelistCallback(); + + releaseFrameLockless(); +} + +QImage VideoFrame::toQImage() +{ + if (!convertToRGB24()) + return QImage(); + + QMutexLocker locker(&biglock); + + return QImage(*frameRGB24->data, width, height, *frameRGB24->linesize, QImage::Format_RGB888); +} + +vpx_image *VideoFrame::toVpxImage() +{ + // libvpx doesn't provide a clean way to reuse an existing external buffer + // so we'll manually fill-in the vpx_image fields and hope for the best. + vpx_image* img = new vpx_image; + memset(img, 0, sizeof(vpx_image)); + + if (!convertToYUV420()) return img; - const int w = resolution.width(); - const int h = resolution.height(); - - // I420 "It comprises an NxM Y plane followed by (N/2)x(M/2) V and U planes." - // http://fourcc.org/yuv.php#IYUV - vpx_img_alloc(&img, VPX_IMG_FMT_VPXI420, w, h, 1); - - for (int y = 0; y < h; ++y) - { - for (int x = 0; x < w; ++x) - { - uint8_t b = frameData.data()[(x + y * w) * 3 + 0]; - uint8_t g = frameData.data()[(x + y * w) * 3 + 1]; - uint8_t r = frameData.data()[(x + y * w) * 3 + 2]; - - img.planes[VPX_PLANE_Y][x + y * img.stride[VPX_PLANE_Y]] = ((66 * r + 129 * g + 25 * b) >> 8) + 16; - - if (!(x % (1 << img.x_chroma_shift)) && !(y % (1 << img.y_chroma_shift))) - { - const int i = x / (1 << img.x_chroma_shift); - const int j = y / (1 << img.y_chroma_shift); - - img.planes[VPX_PLANE_V][i + j * img.stride[VPX_PLANE_V]] = ((112 * r + -94 * g + -18 * b) >> 8) + 128; - img.planes[VPX_PLANE_U][i + j * img.stride[VPX_PLANE_U]] = ((-38 * r + -74 * g + 112 * b) >> 8) + 128; - } - } - } - + img->w = img->d_w = width; + img->h = img->d_h = height; + img->fmt = VPX_IMG_FMT_I420; + img->planes[0] = frameYUV420->data[0]; + img->planes[1] = frameYUV420->data[1]; + img->planes[2] = frameYUV420->data[2]; + img->planes[3] = nullptr; + img->stride[0] = frameYUV420->linesize[0]; + img->stride[1] = frameYUV420->linesize[1]; + img->stride[2] = frameYUV420->linesize[2]; + img->stride[3] = frameYUV420->linesize[3]; return img; } + +bool VideoFrame::convertToRGB24() +{ + QMutexLocker locker(&biglock); + + if (frameRGB24) + return true; + + AVFrame* sourceFrame; + if (frameOther) + { + sourceFrame = frameOther; + } + else if (frameYUV420) + { + sourceFrame = frameYUV420; + } + else + { + qCritical() << "None of the frames are valid! Did someone release us?"; + return false; + } + + frameRGB24=av_frame_alloc(); + if (!frameRGB24) + { + qCritical() << "av_frame_alloc failed"; + return false; + } + + uint8_t* buf = (uint8_t*)av_malloc(avpicture_get_size(AV_PIX_FMT_RGB24, width, height)); + if (!buf) + { + qCritical() << "av_malloc failed"; + av_frame_free(&frameRGB24); + return false; + } + frameRGB24->opaque = buf; + + avpicture_fill((AVPicture*)frameRGB24, buf, AV_PIX_FMT_RGB24, width, height); + + SwsContext *swsCtx = sws_getContext(width, height, (AVPixelFormat)pixFmt, + width, height, AV_PIX_FMT_RGB24, + SWS_BILINEAR, nullptr, nullptr, nullptr); + sws_scale(swsCtx, (uint8_t const * const *)sourceFrame->data, + sourceFrame->linesize, 0, height, + frameRGB24->data, frameRGB24->linesize); + sws_freeContext(swsCtx); + + return true; +} + +bool VideoFrame::convertToYUV420() +{ + QMutexLocker locker(&biglock); + + if (frameYUV420) + return true; + + AVFrame* sourceFrame; + if (frameOther) + { + sourceFrame = frameOther; + } + else if (frameRGB24) + { + sourceFrame = frameOther; + } + else + { + qCritical() << "None of the frames are valid! Did someone release us?"; + return false; + } + + frameYUV420=av_frame_alloc(); + if (!frameYUV420) + { + qCritical() << "av_frame_alloc failed"; + return false; + } + + uint8_t* buf = (uint8_t*)av_malloc(avpicture_get_size(AV_PIX_FMT_RGB24, width, height)); + if (!buf) + { + qCritical() << "av_malloc failed"; + av_frame_free(&frameYUV420); + return false; + } + frameYUV420->opaque = buf; + + avpicture_fill((AVPicture*)frameYUV420, buf, AV_PIX_FMT_YUV420P, width, height); + + SwsContext *swsCtx = sws_getContext(width, height, (AVPixelFormat)pixFmt, + width, height, AV_PIX_FMT_YUV420P, + SWS_BILINEAR, nullptr, nullptr, nullptr); + sws_scale(swsCtx, (uint8_t const * const *)sourceFrame->data, + sourceFrame->linesize, 0, height, + frameYUV420->data, frameYUV420->linesize); + sws_freeContext(swsCtx); + + return true; +} + +void VideoFrame::releaseFrame() +{ + QMutexLocker locker(&biglock); + releaseFrameLockless(); +} + +void VideoFrame::releaseFrameLockless() +{ + if (frameOther) + { + av_free(frameOther->opaque); + av_frame_unref(frameOther); + av_frame_free(&frameOther); + } + if (frameYUV420) + { + av_free(frameYUV420->opaque); + av_frame_unref(frameYUV420); + av_frame_free(&frameYUV420); + } + if (frameRGB24) + { + av_free(frameRGB24->opaque); + av_frame_unref(frameRGB24); + av_frame_free(&frameRGB24); + } +} diff --git a/src/video/videoframe.h b/src/video/videoframe.h index b8c1e79f5..28183ae8e 100644 --- a/src/video/videoframe.h +++ b/src/video/videoframe.h @@ -15,42 +15,53 @@ #ifndef VIDEOFRAME_H #define VIDEOFRAME_H -#include -#include -#include +#include +#include +#include -#include "vpx/vpx_image.h" +struct AVFrame; +struct AVCodecContext; +struct vpx_image; -struct VideoFrame +/// VideoFrame takes ownership of an AVFrame* and allows fast conversions to other formats +/// Ownership of all video frame buffers is kept by the VideoFrame, even after conversion +/// All references to the frame data become invalid when the VideoFrame is deleted +/// We try to avoid pixel format conversions as much as possible, at the cost of some memory +/// All methods are thread-safe. If provided freelistCallback will be called by the destructor. +class VideoFrame { - enum ColorFormat - { - NONE, - BGR, - YUV, - }; +public: + VideoFrame(AVFrame* frame); + VideoFrame(AVFrame* frame, std::function freelistCallback); + VideoFrame(AVFrame* frame, int w, int h, int fmt, std::function freelistCallback); + ~VideoFrame(); - QByteArray frameData; - QSize resolution; - ColorFormat format; + /// Frees all internal buffers and frame data + /// This makes all converted objects that shares our internal buffers invalid + void releaseFrame(); - VideoFrame() : format(NONE) {} - VideoFrame(QByteArray d, QSize r, ColorFormat f) : frameData(d), resolution(r), format(f) {} + /// Converts the VideoFrame to a QImage that shares our internal video buffer + QImage toQImage(); + /// Converts the VideoFrame to a vpx_image_t that shares our internal video buffer + /// Free it with operator delete, NOT vpx_img_free + vpx_image* toVpxImage(); - void invalidate() - { - frameData = QByteArray(); - resolution = QSize(-1,-1); - } +protected: + bool convertToRGB24(); + bool convertToYUV420(); + void releaseFrameLockless(); - bool isValid() const - { - return !frameData.isEmpty() && resolution.isValid() && format != NONE; - } +private: + // Disable copy. Use a shared_ptr if you need copies. + VideoFrame(const VideoFrame& other)=delete; + VideoFrame& operator=(const VideoFrame& other)=delete; - vpx_image_t createVpxImage() const; +private: + std::function freelistCallback; + QMutex biglock; + AVFrame* frameOther, *frameYUV420, *frameRGB24; + int width, height; + int pixFmt; }; -Q_DECLARE_METATYPE(VideoFrame) - #endif // VIDEOFRAME_H diff --git a/src/video/videosource.h b/src/video/videosource.h index 98abf567a..878e18701 100644 --- a/src/video/videosource.h +++ b/src/video/videosource.h @@ -16,22 +16,25 @@ #define VIDEOSOURCE_H #include -#include -#include +#include -#include "videoframe.h" +class VideoFrame; +/// An abstract source of video frames +/// When it has at least one subscriber the source will emit new video frames +/// Subscribing is recursive, multiple users can subscribe to the same VideoSource class VideoSource : public QObject { Q_OBJECT public: - virtual void subscribe() = 0; + /// If subscribe sucessfully opens the source, it will start emitting frameAvailable signals + virtual bool subscribe() = 0; + /// Stop emitting frameAvailable signals, and free associated resources if necessary virtual void unsubscribe() = 0; signals: - void frameAvailable(const VideoFrame& frame); - + void frameAvailable(std::shared_ptr frame); }; #endif // VIDEOSOURCE_H diff --git a/src/widget/form/chatform.cpp b/src/widget/form/chatform.cpp index 958843e1b..78e4e94df 100644 --- a/src/widget/form/chatform.cpp +++ b/src/widget/form/chatform.cpp @@ -408,6 +408,8 @@ void ChatForm::onAvStarting(uint32_t FriendId, int, bool video) qDebug() << "onAvStarting"; + callId = CallId; + callButton->disconnect(); videoButton->disconnect(); if (video) diff --git a/src/widget/form/settings/avform.cpp b/src/widget/form/settings/avform.cpp index 21409de8b..1552e20b2 100644 --- a/src/widget/form/settings/avform.cpp +++ b/src/widget/form/settings/avform.cpp @@ -16,6 +16,9 @@ #include "ui_avsettings.h" #include "src/misc/settings.h" #include "src/audio.h" +#include "src/video/camerasource.h" +#include "src/video/cameradevice.h" +#include "src/widget/videosurface.h" #if defined(__APPLE__) && defined(__MACH__) #include @@ -25,13 +28,15 @@ #include #endif +#include + #ifndef ALC_ALL_DEVICES_SPECIFIER #define ALC_ALL_DEVICES_SPECIFIER ALC_DEVICE_SPECIFIER #endif AVForm::AVForm() : GenericForm(tr("Audio/Video"), QPixmap(":/img/settings/av.png")), - CamVideoSurface{nullptr} + camVideoSurface{nullptr}, camera{nullptr} { bodyUI = new Ui::AVSettings; bodyUI->setupUi(this); @@ -42,12 +47,11 @@ AVForm::AVForm() : bodyUI->filterAudio->setDisabled(true); #endif - connect(Camera::getInstance(), &Camera::propProbingFinished, this, &AVForm::onPropProbingFinished); - connect(Camera::getInstance(), &Camera::resolutionProbingFinished, this, &AVForm::onResProbingFinished); - - auto qcomboboxIndexChanged = (void(QComboBox::*)(const QString&)) &QComboBox::currentIndexChanged; - connect(bodyUI->inDevCombobox, qcomboboxIndexChanged, this, &AVForm::onInDevChanged); - connect(bodyUI->outDevCombobox, qcomboboxIndexChanged, this, &AVForm::onOutDevChanged); + auto qcbxIndexChangedStr = (void(QComboBox::*)(const QString&)) &QComboBox::currentIndexChanged; + auto qcbxIndexChangedInt = (void(QComboBox::*)(int)) &QComboBox::currentIndexChanged; + connect(bodyUI->inDevCombobox, qcbxIndexChangedStr, this, &AVForm::onInDevChanged); + connect(bodyUI->outDevCombobox, qcbxIndexChangedStr, this, &AVForm::onOutDevChanged); + connect(bodyUI->videoDevCombobox, qcbxIndexChangedInt, this, &AVForm::onVideoDevChanged); connect(bodyUI->filterAudio, &QCheckBox::toggled, this, &AVForm::onFilterAudioToggled); connect(bodyUI->rescanButton, &QPushButton::clicked, this, [=](){getAudioInDevices(); getAudioOutDevices();}); bodyUI->playbackSlider->setValue(100); @@ -63,6 +67,11 @@ AVForm::AVForm() : AVForm::~AVForm() { delete bodyUI; + if (camera) + { + delete camera; + camera = nullptr; + } } void AVForm::present() @@ -71,65 +80,30 @@ void AVForm::present() getAudioInDevices(); createVideoSurface(); - CamVideoSurface->setSource(Camera::getInstance()); - - Camera::getInstance()->probeProp(Camera::SATURATION); - Camera::getInstance()->probeProp(Camera::CONTRAST); - Camera::getInstance()->probeProp(Camera::BRIGHTNESS); - Camera::getInstance()->probeProp(Camera::HUE); - Camera::getInstance()->probeResolutions(); bodyUI->videoModescomboBox->blockSignals(true); bodyUI->videoModescomboBox->addItem(tr("Initializing Camera...")); bodyUI->videoModescomboBox->blockSignals(false); } -void AVForm::on_ContrastSlider_sliderMoved(int position) -{ - Camera::getInstance()->setProp(Camera::CONTRAST, position / 100.0); -} - -void AVForm::on_SaturationSlider_sliderMoved(int position) -{ - Camera::getInstance()->setProp(Camera::SATURATION, position / 100.0); -} - -void AVForm::on_BrightnessSlider_sliderMoved(int position) -{ - Camera::getInstance()->setProp(Camera::BRIGHTNESS, position / 100.0); -} - -void AVForm::on_HueSlider_sliderMoved(int position) -{ - Camera::getInstance()->setProp(Camera::HUE, position / 100.0); -} - void AVForm::on_videoModescomboBox_currentIndexChanged(int index) { QSize res = bodyUI->videoModescomboBox->itemData(index).toSize(); Settings::getInstance().setCamVideoRes(res); - Camera::getInstance()->setResolution(res); } -void AVForm::onPropProbingFinished(Camera::Prop prop, double val) +void AVForm::onVideoDevChanged(int index) { - switch (prop) + if (index<0 || index>=videoDeviceList.size()) { - case Camera::BRIGHTNESS: - bodyUI->BrightnessSlider->setValue(val*100); - break; - case Camera::CONTRAST: - bodyUI->ContrastSlider->setValue(val*100); - break; - case Camera::SATURATION: - bodyUI->SaturationSlider->setValue(val*100); - break; - case Camera::HUE: - bodyUI->HueSlider->setValue(val*100); - break; - default: - break; + qWarning() << "Invalid index"; + return; } + camVideoSurface->setSource(nullptr); + if (camera) + delete camera; + camera = new CameraSource(videoDeviceList[index].first); + camVideoSurface->setSource(camera); } void AVForm::onResProbingFinished(QList res) @@ -157,17 +131,43 @@ void AVForm::onResProbingFinished(QList res) void AVForm::hideEvent(QHideEvent *) { - if (CamVideoSurface) + if (camVideoSurface) { - CamVideoSurface->setSource(nullptr); + camVideoSurface->setSource(nullptr); killVideoSurface(); } + if (camera) + { + delete camera; + camera = nullptr; + } + videoDeviceList.clear(); } void AVForm::showEvent(QShowEvent *) { createVideoSurface(); - CamVideoSurface->setSource(Camera::getInstance()); + getVideoDevices(); +} + +void AVForm::getVideoDevices() +{ + QString settingsInDev = Settings::getInstance().getVideoDev(); + int videoDevIndex = 0; + videoDeviceList = CameraDevice::getDeviceList(); + bodyUI->videoDevCombobox->clear(); + //prevent currentIndexChanged to be fired while adding items + bodyUI->videoDevCombobox->blockSignals(true); + for (QPair device : videoDeviceList) + { + bodyUI->videoDevCombobox->addItem(device.second); + if (device.first == settingsInDev) + videoDevIndex = bodyUI->videoDevCombobox->count()-1; + } + //addItem changes currentIndex -> reset + bodyUI->videoDevCombobox->setCurrentIndex(-1); + bodyUI->videoDevCombobox->blockSignals(false); + bodyUI->videoDevCombobox->setCurrentIndex(videoDevIndex); } void AVForm::getAudioInDevices() @@ -190,9 +190,7 @@ void AVForm::getAudioInDevices() #endif bodyUI->inDevCombobox->addItem(inDev); if (settingsInDev == inDev) - { inDevIndex = bodyUI->inDevCombobox->count()-1; - } pDeviceList += len+1; } //addItem changes currentIndex -> reset @@ -255,30 +253,6 @@ void AVForm::onFilterAudioToggled(bool filterAudio) Settings::getInstance().setFilterAudio(filterAudio); } -void AVForm::on_HueSlider_valueChanged(int value) -{ - Camera::getInstance()->setProp(Camera::HUE, value / 100.0); - bodyUI->hueMax->setText(QString::number(value)); -} - -void AVForm::on_BrightnessSlider_valueChanged(int value) -{ - Camera::getInstance()->setProp(Camera::BRIGHTNESS, value / 100.0); - bodyUI->brightnessMax->setText(QString::number(value)); -} - -void AVForm::on_SaturationSlider_valueChanged(int value) -{ - Camera::getInstance()->setProp(Camera::SATURATION, value / 100.0); - bodyUI->saturationMax->setText(QString::number(value)); -} - -void AVForm::on_ContrastSlider_valueChanged(int value) -{ - Camera::getInstance()->setProp(Camera::CONTRAST, value / 100.0); - bodyUI->contrastMax->setText(QString::number(value)); -} - void AVForm::on_playbackSlider_valueChanged(int value) { Audio::setOutputVolume(value / 100.0); @@ -304,22 +278,22 @@ bool AVForm::eventFilter(QObject *o, QEvent *e) void AVForm::createVideoSurface() { - if (CamVideoSurface) + if (camVideoSurface) return; - CamVideoSurface = new VideoSurface(bodyUI->CamFrame); - CamVideoSurface->setObjectName(QStringLiteral("CamVideoSurface")); - CamVideoSurface->setMinimumSize(QSize(160, 120)); - bodyUI->gridLayout->addWidget(CamVideoSurface, 0, 0, 1, 1); + camVideoSurface = new VideoSurface(bodyUI->CamFrame); + camVideoSurface->setObjectName(QStringLiteral("CamVideoSurface")); + camVideoSurface->setMinimumSize(QSize(160, 120)); + bodyUI->gridLayout->addWidget(camVideoSurface, 0, 0, 1, 1); } void AVForm::killVideoSurface() { - if (!CamVideoSurface) + if (!camVideoSurface) return; QLayoutItem *child; while ((child = bodyUI->gridLayout->takeAt(0)) != 0) delete child; - delete CamVideoSurface; - CamVideoSurface = nullptr; + delete camVideoSurface; + camVideoSurface = nullptr; } diff --git a/src/widget/form/settings/avform.h b/src/widget/form/settings/avform.h index a62dc9901..1a07654f8 100644 --- a/src/widget/form/settings/avform.h +++ b/src/widget/form/settings/avform.h @@ -15,18 +15,16 @@ #ifndef AVFORM_H #define AVFORM_H +#include +#include #include "genericsettings.h" -#include "src/widget/videosurface.h" -#include "src/video/camera.h" -#include -#include -#include namespace Ui { class AVSettings; } -class Camera; +class CameraSource; +class VideoSurface; class AVForm : public GenericForm { @@ -39,15 +37,12 @@ public: private: void getAudioInDevices(); void getAudioOutDevices(); + void getVideoDevices(); void createVideoSurface(); void killVideoSurface(); private slots: - void on_ContrastSlider_sliderMoved(int position); - void on_SaturationSlider_sliderMoved(int position); - void on_BrightnessSlider_sliderMoved(int position); - void on_HueSlider_sliderMoved(int position); void on_videoModescomboBox_currentIndexChanged(int index); // audio @@ -58,23 +53,20 @@ private slots: void on_microphoneSlider_valueChanged(int value); // camera - void onPropProbingFinished(Camera::Prop prop, double val); + void onVideoDevChanged(int index); void onResProbingFinished(QList res); virtual void hideEvent(QHideEvent*); virtual void showEvent(QShowEvent*); - - void on_HueSlider_valueChanged(int value); - void on_BrightnessSlider_valueChanged(int value); - void on_SaturationSlider_valueChanged(int value); - void on_ContrastSlider_valueChanged(int value); protected: bool eventFilter(QObject *o, QEvent *e); private: Ui::AVSettings *bodyUI; - VideoSurface* CamVideoSurface; + VideoSurface* camVideoSurface; + CameraSource* camera; + QVector> videoDeviceList; }; #endif diff --git a/src/widget/form/settings/avsettings.ui b/src/widget/form/settings/avsettings.ui index 81295e706..4971c2ef5 100644 --- a/src/widget/form/settings/avsettings.ui +++ b/src/widget/form/settings/avsettings.ui @@ -41,10 +41,36 @@ Audio Settings - - + + - 100 + 0 + + + + + + + Filter sound from your microphone, so that people hearing you would get better sound. + + + Filter audio + + + + + + + + + + Use slider to set volume of your speakers. + + + 100 + + + Qt::Horizontal @@ -55,17 +81,24 @@ - - + + - Playback + 100 - - + + - Capture device + Rescan audio devices + + + + + + + 100 @@ -83,47 +116,10 @@ WARNING: slider is not supposed to work yet. - - - - Use slider to set volume of your speakers. - - - 100 - - - Qt::Horizontal - - - - - - - Filter sound from your microphone, so that people hearing you would get better sound. - + + - Filter audio - - - - - - - Playback device - - - - - - - 0 - - - - - - - Rescan audio devices + Capture device @@ -134,18 +130,22 @@ WARNING: slider is not supposed to work yet. - - + + - 100 + Playback - - + + + + Playback device + + @@ -158,21 +158,133 @@ WARNING: slider is not supposed to work yet. - - + + + + Set resolution of your camera. +The higher values, the better video quality your friends may get. +Note though that with better video quality there is needed better internet connection. +Sometimes your connection may not be good enough to handle higher video quality, +which may lead to problems with video calls. + + + Resolution + + + + + + + 100 + + + + + + + 100 + + + + + + + Saturation + + + + + Qt::Horizontal - + Qt::Horizontal - + + + + Contrast + + + + + + + 0 + + + + + + + Qt::Horizontal + + + + + + + 100 + + + + + + + 0 + + + + + + + Brightness + + + + + + + 0 + + + + + + + Hue + + + + + + + 0 + + + + + + + Qt::Horizontal + + + + + + + 100 + + + + @@ -189,117 +301,15 @@ which may lead to problems with video calls. - - - - 0 - - - - - - - Contrast - - - - - - - 0 - - - - - - - Qt::Horizontal - - - - - - - Hue - - - - - - Set resolution of your camera. -The higher values, the better video quality your friends may get. -Note though that with better video quality there is needed better internet connection. -Sometimes your connection may not be good enough to handle higher video quality, -which may lead to problems with video calls. - + - Resolution + Video device - - - - 0 - - - - - - - Qt::Horizontal - - - - - - - 0 - - - - - - - Saturation - - - - - - - Brightness - - - - - - - 100 - - - - - - - 100 - - - - - - - 100 - - - - - - - 100 - - + + @@ -308,7 +318,7 @@ which may lead to problems with video calls. 1 - 99 + 150 diff --git a/src/widget/form/settings/generalform.h b/src/widget/form/settings/generalform.h index 59a75ef2a..c1b9bce04 100644 --- a/src/widget/form/settings/generalform.h +++ b/src/widget/form/settings/generalform.h @@ -21,6 +21,8 @@ namespace Ui { class GeneralSettings; } +class SettingsWidget; + class GeneralForm : public GenericForm { Q_OBJECT diff --git a/src/widget/form/settings/genericsettings.h b/src/widget/form/settings/genericsettings.h index cc0920d4a..e3d395dfd 100644 --- a/src/widget/form/settings/genericsettings.h +++ b/src/widget/form/settings/genericsettings.h @@ -16,7 +16,6 @@ #define GENERICFORM_H #include -#include "src/widget/form/settingswidget.h" class GenericForm : public QWidget { diff --git a/src/widget/form/settingswidget.cpp b/src/widget/form/settingswidget.cpp index 47d8fb1bc..e41144870 100644 --- a/src/widget/form/settingswidget.cpp +++ b/src/widget/form/settingswidget.cpp @@ -15,7 +15,7 @@ #include "settingswidget.h" #include "src/widget/widget.h" #include "ui_mainwindow.h" -#include "src/video/camera.h" +#include "src/video/camerasource.h" #include "src/widget/form/settings/generalform.h" #include "src/widget/form/settings/privacyform.h" #include "src/widget/form/settings/avform.h" diff --git a/src/widget/videosurface.cpp b/src/widget/videosurface.cpp index 08bbb0d2a..710a4c0c9 100644 --- a/src/widget/videosurface.cpp +++ b/src/widget/videosurface.cpp @@ -13,22 +13,14 @@ */ #include "videosurface.h" -#include "src/video/camera.h" -#include -#include -#include -#include +#include "src/video/videoframe.h" +#include VideoSurface::VideoSurface(QWidget* parent) - : QGLWidget(QGLFormat(QGL::SingleBuffer), parent) + : QWidget{parent} , source{nullptr} - , pbo{nullptr, nullptr} - , bgrProgramm{nullptr} - , yuvProgramm{nullptr} - , textureId{0} - , pboAllocSize{0} + , frameLock{false} , hasSubscribed{false} - , pboIndex{0} { } @@ -41,18 +33,6 @@ VideoSurface::VideoSurface(VideoSource *source, QWidget* parent) VideoSurface::~VideoSurface() { - if (pbo[0]) - { - delete pbo[0]; - delete pbo[1]; - } - - if (textureId != 0) - glDeleteTextures(1, &textureId); - - delete bgrProgramm; - delete yuvProgramm; - unsubscribe(); } @@ -66,182 +46,6 @@ void VideoSurface::setSource(VideoSource *src) subscribe(); } -void VideoSurface::initializeGL() -{ - QGLWidget::initializeGL(); - qDebug() << "Init"; - // pbo - pbo[0] = new QOpenGLBuffer(QOpenGLBuffer::PixelUnpackBuffer); - pbo[0]->setUsagePattern(QOpenGLBuffer::StreamDraw); - pbo[0]->create(); - - pbo[1] = new QOpenGLBuffer(QOpenGLBuffer::PixelUnpackBuffer); - pbo[1]->setUsagePattern(QOpenGLBuffer::StreamDraw); - pbo[1]->create(); - - // shaders - bgrProgramm = new QOpenGLShaderProgram; - bgrProgramm->addShaderFromSourceCode(QOpenGLShader::Vertex, - "attribute vec4 vertices;" - "varying vec2 coords;" - "void main() {" - " gl_Position = vec4(vertices.xy, 0.0, 1.0);" - " coords = vertices.xy*vec2(0.5, 0.5) + vec2(0.5, 0.5);" - "}"); - - // brg frag-shader - bgrProgramm->addShaderFromSourceCode(QOpenGLShader::Fragment, - "uniform sampler2D texture0;" - "varying vec2 coords;" - "void main() {" - " vec4 color = texture2D(texture0,coords*vec2(1.0, -1.0));" - " gl_FragColor = vec4(color.bgr, 1.0);" - "}"); - - bgrProgramm->bindAttributeLocation("vertices", 0); - bgrProgramm->link(); - - // shaders - yuvProgramm = new QOpenGLShaderProgram; - yuvProgramm->addShaderFromSourceCode(QOpenGLShader::Vertex, - "attribute vec4 vertices;" - "varying vec2 coords;" - "void main() {" - " gl_Position = vec4(vertices.xy, 0.0, 1.0);" - " coords = vertices.xy*vec2(0.5, 0.5) + vec2(0.5, 0.5);" - "}"); - - // yuv frag-shader - yuvProgramm->addShaderFromSourceCode(QOpenGLShader::Fragment, - "uniform sampler2D texture0;" - "varying vec2 coords;" - "void main() {" - " vec3 yuv = texture2D(texture0,coords*vec2(1.0, -1.0)).rgb - vec3(0.0, 0.5, 0.5);" - " vec3 rgb = mat3(1.0, 1.0, 1.0, 0.0, -0.21482, 2.12798, 1.28033, -0.38059, 0.0)*yuv;" - " gl_FragColor = vec4(rgb, 1.0);" - "}"); - - yuvProgramm->bindAttributeLocation("vertices", 0); - yuvProgramm->link(); -} - -void VideoSurface::paintGL() -{ - mutex.lock(); - VideoFrame currFrame = frame; - frame.invalidate(); - mutex.unlock(); - - if (currFrame.isValid() && res != currFrame.resolution) - { - res = currFrame.resolution; - - // delete old texture - if (textureId != 0) - glDeleteTextures(1, &textureId); - - // a texture used to render the pbo (has the match the pixelformat of the source) - glGenTextures(1,&textureId); - glBindTexture(GL_TEXTURE_2D, textureId); - glTexImage2D(GL_TEXTURE_2D,0, GL_RGB, res.width(), res.height(), 0, GL_RGB, GL_UNSIGNED_BYTE, 0); - glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR); - } - - if (currFrame.isValid()) - { - pboIndex = (pboIndex + 1) % 2; - int nextPboIndex = (pboIndex + 1) % 2; - - if (pboAllocSize != currFrame.frameData.size()) - { - qDebug() << "Resize pbo " << currFrame.frameData.size() << "(" << currFrame.resolution << ")" << "bytes (before" << pboAllocSize << ")"; - - pbo[0]->bind(); - pbo[0]->allocate(currFrame.frameData.size()); - pbo[0]->release(); - - pbo[1]->bind(); - pbo[1]->allocate(currFrame.frameData.size()); - pbo[1]->release(); - - pboAllocSize = currFrame.frameData.size(); - } - - - pbo[pboIndex]->bind(); - glBindTexture(GL_TEXTURE_2D, textureId); - glTexSubImage2D(GL_TEXTURE_2D,0,0,0, res.width(), res.height(), GL_RGB, GL_UNSIGNED_BYTE, 0); - pbo[pboIndex]->unmap(); - pbo[pboIndex]->release(); - - // transfer data - pbo[nextPboIndex]->bind(); - void* ptr = pbo[nextPboIndex]->map(QOpenGLBuffer::WriteOnly); - if (ptr) - memcpy(ptr, currFrame.frameData.data(), currFrame.frameData.size()); - pbo[nextPboIndex]->unmap(); - pbo[nextPboIndex]->release(); - } - - // background - glClearColor(0, 0, 0, 1); - glClear(GL_COLOR_BUFFER_BIT); - - // keep aspect ratio - float aspectRatio = float(res.width()) / float(res.height()); - if (width() < float(height()) * aspectRatio) - { - float h = float(width()) / aspectRatio; - glViewport(0, (height() - h)*0.5f, width(), h); - } - else - { - float w = float(height()) * float(aspectRatio); - glViewport((width() - w)*0.5f, 0, w, height()); - } - - QOpenGLShaderProgram* programm = nullptr; - switch (frame.format) - { - case VideoFrame::YUV: - programm = yuvProgramm; - break; - case VideoFrame::BGR: - programm = bgrProgramm; - break; - default: - break; - } - - if (programm) - { - // render pbo - static float values[] = { - -1, -1, - 1, -1, - -1, 1, - 1, 1 - }; - - programm->bind(); - programm->setAttributeArray(0, GL_FLOAT, values, 2); - programm->enableAttributeArray(0); - } - - glBindTexture(GL_TEXTURE_2D, textureId); - - //draw fullscreen quad - glDrawArrays(GL_TRIANGLE_STRIP, 0, 4); - - glBindTexture(GL_TEXTURE_2D, 0); - - if (programm) - { - programm->disableAttributeArray(0); - programm->release(); - } -} - void VideoSurface::subscribe() { if (source && !hasSubscribed) @@ -254,22 +58,56 @@ void VideoSurface::subscribe() void VideoSurface::unsubscribe() { - if (source && hasSubscribed) + if (!source || !hasSubscribed) + return; + + // Fast lock { - source->unsubscribe(); - hasSubscribed = false; - disconnect(source, &VideoSource::frameAvailable, this, &VideoSurface::onNewFrameAvailable); + bool expected = false; + while (!frameLock.compare_exchange_weak(expected, true)) + expected = false; } + lastFrame.reset(); + frameLock = false; + + source->unsubscribe(); + hasSubscribed = false; + disconnect(source, &VideoSource::frameAvailable, this, &VideoSurface::onNewFrameAvailable); } -void VideoSurface::onNewFrameAvailable(const VideoFrame& newFrame) +void VideoSurface::onNewFrameAvailable(std::shared_ptr newFrame) { - mutex.lock(); - frame = newFrame; - mutex.unlock(); + // Fast lock + { + bool expected = false; + while (!frameLock.compare_exchange_weak(expected, true)) + expected = false; + } - updateGL(); + lastFrame = newFrame; + frameLock = false; + update(); } +void VideoSurface::paintEvent(QPaintEvent*) +{ + // Fast lock + { + bool expected = false; + while (!frameLock.compare_exchange_weak(expected, true)) + expected = false; + } - + QPainter painter(this); + painter.fillRect(painter.viewport(), Qt::black); + if (lastFrame) + { + QRect rect = painter.viewport(); + QImage frame = lastFrame->toQImage(); + int width = frame.width()*rect.height()/frame.height(); + rect.setLeft((rect.width()-width)/2); + rect.setWidth(width); + painter.drawImage(rect, frame); + } + frameLock = false; +} diff --git a/src/widget/videosurface.h b/src/widget/videosurface.h index fb2f66c43..f8bcbb242 100644 --- a/src/widget/videosurface.h +++ b/src/widget/videosurface.h @@ -15,14 +15,12 @@ #ifndef SELFCAMVIEW_H #define SELFCAMVIEW_H -#include -#include +#include +#include +#include #include "src/video/videosource.h" -class QOpenGLBuffer; -class QOpenGLShaderProgram; - -class VideoSurface : public QGLWidget +class VideoSurface : public QWidget { Q_OBJECT @@ -33,30 +31,20 @@ public: void setSource(VideoSource* src); //NULL is a valid option - // QGLWidget interface protected: - virtual void initializeGL(); - virtual void paintGL(); - void subscribe(); void unsubscribe(); + virtual void paintEvent(QPaintEvent * event) override; + private slots: - void onNewFrameAvailable(const VideoFrame &newFrame); + void onNewFrameAvailable(std::shared_ptr newFrame); private: VideoSource* source; - QOpenGLBuffer* pbo[2]; - QOpenGLShaderProgram* bgrProgramm; - QOpenGLShaderProgram* yuvProgramm; - GLuint textureId; - int pboAllocSize; - QSize res; + std::shared_ptr lastFrame; + std::atomic_bool frameLock; ///< Fast lock for lastFrame bool hasSubscribed; - - QMutex mutex; - VideoFrame frame; - int pboIndex; }; #endif // SELFCAMVIEW_H diff --git a/src/widget/widget.cpp b/src/widget/widget.cpp index 0613016eb..3d18d764e 100644 --- a/src/widget/widget.cpp +++ b/src/widget/widget.cpp @@ -26,7 +26,6 @@ #include "form/groupchatform.h" #include "src/misc/style.h" #include "friendlistwidget.h" -#include "src/video/camera.h" #include "form/chatform.h" #include "maskablepixmapwidget.h" #include "src/historykeeper.h"