diff --git a/src/platform/camera/v4l2.cpp b/src/platform/camera/v4l2.cpp index 28f0ee35d..a442afad4 100644 --- a/src/platform/camera/v4l2.cpp +++ b/src/platform/camera/v4l2.cpp @@ -28,6 +28,7 @@ #include #include #include +#include /** * Most of this file is adapted from libavdevice's v4l2.c, @@ -35,6 +36,26 @@ * stdout and is not part of the public API for some reason. */ +static std::map createPixFmtToQuality() +{ + std::map m; + m[V4L2_PIX_FMT_H264] = 3; + m[V4L2_PIX_FMT_MJPEG] = 2; + m[V4L2_PIX_FMT_YUYV] = 1; + return m; +} +const std::map pixFmtToQuality = createPixFmtToQuality(); + +static std::map createPixFmtToName() +{ + std::map m; + m[V4L2_PIX_FMT_H264] = QString("h264"); + m[V4L2_PIX_FMT_MJPEG] = QString("mjpeg"); + m[V4L2_PIX_FMT_YUYV] = QString("yuyv422"); + return m; +} +const std::map pixFmtToName = createPixFmtToName(); + static int deviceOpen(QString devName) { struct v4l2_capability cap; @@ -173,20 +194,24 @@ QVector> v4l2::getDeviceList() QString v4l2::getPixelFormatString(uint32_t pixel_format) { - if (pixel_format == V4L2_PIX_FMT_H264) + if (pixFmtToName.find(pixel_format) == pixFmtToName.end()) { - return QString("h264"); - } - else if (pixel_format == V4L2_PIX_FMT_MJPEG) - { - return QString("mjpeg"); - } - else if (pixel_format == V4L2_PIX_FMT_YUYV) - { - return QString("yuyv422"); - } - else { + printf("BAD!\n"); return QString("unknown"); } + return pixFmtToName.at(pixel_format); +} + +bool v4l2::betterPixelFormat(uint32_t a, uint32_t b) +{ + if (pixFmtToQuality.find(a) == pixFmtToQuality.end()) + { + return false; + } + else if (pixFmtToQuality.find(b) == pixFmtToQuality.end()) + { + return true; + } + return pixFmtToQuality.at(a) > pixFmtToQuality.at(b); } diff --git a/src/platform/camera/v4l2.h b/src/platform/camera/v4l2.h index 8d82de9cd..c9556c84c 100644 --- a/src/platform/camera/v4l2.h +++ b/src/platform/camera/v4l2.h @@ -33,6 +33,7 @@ namespace v4l2 QVector getDeviceModes(QString devName); QVector> getDeviceList(); QString getPixelFormatString(uint32_t pixel_format); + bool betterPixelFormat(uint32_t a, uint32_t b); } #endif // V4L2_H diff --git a/src/video/cameradevice.cpp b/src/video/cameradevice.cpp index 398eaf9da..113f26e96 100644 --- a/src/video/cameradevice.cpp +++ b/src/video/cameradevice.cpp @@ -372,6 +372,15 @@ QString CameraDevice::getPixelFormatString(uint32_t pixel_format) #endif } +bool CameraDevice::betterPixelFormat(uint32_t a, uint32_t b) +{ +#ifdef Q_OS_LINUX + return v4l2::betterPixelFormat(a, b); +#else + return false; +#endif +} + bool CameraDevice::getDefaultInputFormat() { QMutexLocker locker(&iformatLock); diff --git a/src/video/cameradevice.h b/src/video/cameradevice.h index ced3d4e14..e6f304e35 100644 --- a/src/video/cameradevice.h +++ b/src/video/cameradevice.h @@ -61,6 +61,8 @@ public: static QVector getVideoModes(QString devName); /// Get the name of the pixel format of a video mode static QString getPixelFormatString(uint32_t pixel_format); + /// Returns true if we prefer format a to b, false otherwise (such as if there's no preference) + static bool betterPixelFormat(uint32_t a, uint32_t b); /// Returns the short name of the default defice /// This is either the device in the settings diff --git a/src/video/videoframe.cpp b/src/video/videoframe.cpp index 427cfa8c8..c686b5fde 100644 --- a/src/video/videoframe.cpp +++ b/src/video/videoframe.cpp @@ -17,6 +17,8 @@ along with qTox. If not, see . */ +#include + #include #include #include @@ -48,12 +50,13 @@ VideoFrame::VideoFrame(AVFrame* frame, int w, int h, int fmt, std::functioncolor_range = AVCOL_RANGE_UNSPECIFIED; - if (pixFmt == AV_PIX_FMT_YUV420P) + if (pixFmt == AV_PIX_FMT_YUV420P) { frameYUV420 = frame; - else if (pixFmt == AV_PIX_FMT_RGB24) + } else if (pixFmt == AV_PIX_FMT_RGB24) { frameRGB24 = frame; - else + } else { frameOther = frame; + } } VideoFrame::VideoFrame(AVFrame* frame, std::function freelistCallback) @@ -126,6 +129,7 @@ bool VideoFrame::convertToRGB24(QSize size) qWarning() << "None of the frames are valid! Did someone release us?"; return false; } + //std::cout << "converting to RGB24" << std::endl; if (size.isEmpty()) { @@ -198,6 +202,7 @@ bool VideoFrame::convertToYUV420() qCritical() << "None of the frames are valid! Did someone release us?"; return false; } + //std::cout << "converting to YUV420" << std::endl; frameYUV420=av_frame_alloc(); if (!frameYUV420) diff --git a/src/video/videomode.h b/src/video/videomode.h index c1628f439..b3560409c 100644 --- a/src/video/videomode.h +++ b/src/video/videomode.h @@ -41,6 +41,11 @@ struct VideoMode && FPS == other.FPS && pixel_format == other.pixel_format; } + + uint32_t norm(const VideoMode& other) const + { + return std::abs(this->width-other.width) + std::abs(this->height-other.height); + } }; #endif // VIDEOMODE_H diff --git a/src/widget/form/settings/avform.cpp b/src/widget/form/settings/avform.cpp index 71534df05..3d1b55e36 100644 --- a/src/widget/form/settings/avform.cpp +++ b/src/widget/form/settings/avform.cpp @@ -30,6 +30,7 @@ #include #include +#include #ifndef ALC_ALL_DEVICES_SPECIFIER #define ALC_ALL_DEVICES_SPECIFIER ALC_DEVICE_SPECIFIER @@ -172,23 +173,65 @@ void AVForm::updateVideoModes(int curIndex) a.FPS>b.FPS;}); bool previouslyBlocked = bodyUI->videoModescomboBox->blockSignals(true); bodyUI->videoModescomboBox->clear(); + + // Identify the best resolutions available for the supposed XXXXp resolutions. + std::map idealModes; + idealModes[240] = {460,240,0,0}; idealModes[360] = {640,360,0,0}; + idealModes[480] = {854,480,0,0}; idealModes[720] = {1280,720,0,0}; + idealModes[1080] = {1920,1080,0,0}; + std::map bestModeInds; + for (int i=0; ifirst; + VideoMode idealMode = iter->second; + // don't take approximately correct resolutions unless they really + // are close + if (mode.norm(idealMode) > 300) continue; + if (bestModeInds.find(res) == bestModeInds.end()) + { + bestModeInds[res] = i; + continue; + } + int ind = bestModeInds[res]; + if (mode.norm(idealMode) < videoModes[ind].norm(idealMode)) + { + bestModeInds[res] = i; + } + else if (mode.norm(idealMode) == videoModes[ind].norm(idealMode)) + { + // prefer higher FPS and "better" pixel formats + if (mode.FPS > videoModes[ind].FPS) { + bestModeInds[res] = i; + } + else if (mode.FPS == videoModes[ind].FPS && + CameraDevice::betterPixelFormat(mode.pixel_format, videoModes[ind].pixel_format)) + { + bestModeInds[res] = i; + } + } + } + } + printf("=====\n"); int prefResIndex = -1; QSize prefRes = Settings::getInstance().getCamVideoRes(); unsigned short prefFPS = Settings::getInstance().getCamVideoFPS(); - for (int i=0; isecond; VideoMode mode = videoModes[i]; if (mode.width==prefRes.width() && mode.height==prefRes.height() && mode.FPS == prefFPS && prefResIndex==-1) prefResIndex = i; QString str; + printf("width: %d, height: %d, FPS: %f, pixel format: %s\n", mode.width, mode.height, mode.FPS, CameraDevice::getPixelFormatString(mode.pixel_format).toStdString().c_str()); if (mode.height && mode.width) - str += tr("%1x%2").arg(mode.width).arg(mode.height); + str += tr("%1p").arg(iter->first); else str += tr("Default resolution"); - if (mode.FPS) - str += tr(" at %1 FPS").arg(mode.FPS); - if (mode.pixel_format) - str += tr(" using %1").arg(CameraDevice::getPixelFormatString(mode.pixel_format)); bodyUI->videoModescomboBox->addItem(str); } if (videoModes.isEmpty())