mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-12 02:15:31 +00:00
v4l2object: improve colorspace handling for JPEG sources
gstjpegdec sets 1:4:0:0 colorimetry (full range BT.601 YCbCr encoding with unknown primaries and unknown transfer function). This currently gets translated to bt601 or bt709 depending on resolution. Both cases result in a negotiation failure: ERROR: from element /GstPipeline:pipeline0/v4l2video0convert:v4l2video0convert0: Device '/dev/video0' does not support 1:4:0:0 colorimetry Improve the guessing game by selecting JPEG colorimetry (JPEG colorspace with sRGB transfer function) under these specific conditions, and loosen the matching so that 1:4:0:0 input gets accepted if the device is actually configured to 1:4:7:1 (V4L2_PIX_FMT_JPEG default). https://bugzilla.gnome.org/show_bug.cgi?id=796940
This commit is contained in:
parent
9595fd08e3
commit
02d3f9748b
1 changed files with 41 additions and 9 deletions
|
@ -3197,6 +3197,32 @@ gst_v4l2_object_extrapolate_stride (const GstVideoFormatInfo * finfo,
|
||||||
return estride;
|
return estride;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
gst_v4l2_video_colorimetry_matches (const GstVideoColorimetry * cinfo,
|
||||||
|
const gchar * color)
|
||||||
|
{
|
||||||
|
GstVideoColorimetry ci;
|
||||||
|
static const GstVideoColorimetry ci_likely_jpeg = {
|
||||||
|
GST_VIDEO_COLOR_RANGE_0_255, GST_VIDEO_COLOR_MATRIX_BT601,
|
||||||
|
GST_VIDEO_TRANSFER_UNKNOWN, GST_VIDEO_COLOR_PRIMARIES_UNKNOWN };
|
||||||
|
static const GstVideoColorimetry ci_jpeg = {
|
||||||
|
GST_VIDEO_COLOR_RANGE_0_255, GST_VIDEO_COLOR_MATRIX_BT601,
|
||||||
|
GST_VIDEO_TRANSFER_SRGB, GST_VIDEO_COLOR_PRIMARIES_BT709 };
|
||||||
|
|
||||||
|
if (!gst_video_colorimetry_from_string (&ci, color))
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
if (gst_video_colorimetry_is_equal (&ci, cinfo))
|
||||||
|
return TRUE;
|
||||||
|
|
||||||
|
/* Allow 1:4:0:0 (produced by jpegdec) if the device expects 1:4:7:1 */
|
||||||
|
if (gst_video_colorimetry_is_equal (&ci, &ci_likely_jpeg)
|
||||||
|
&& gst_video_colorimetry_is_equal (cinfo, &ci_jpeg))
|
||||||
|
return TRUE;
|
||||||
|
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
gst_v4l2_object_set_format_full (GstV4l2Object * v4l2object, GstCaps * caps,
|
gst_v4l2_object_set_format_full (GstV4l2Object * v4l2object, GstCaps * caps,
|
||||||
gboolean try_only, GstV4l2Error * error)
|
gboolean try_only, GstV4l2Error * error)
|
||||||
|
@ -3376,11 +3402,19 @@ gst_v4l2_object_set_format_full (GstV4l2Object * v4l2object, GstCaps * caps,
|
||||||
if (colorspace == 0) {
|
if (colorspace == 0) {
|
||||||
/* Try to guess colorspace according to pixelformat and size */
|
/* Try to guess colorspace according to pixelformat and size */
|
||||||
if (GST_VIDEO_INFO_IS_YUV (&info)) {
|
if (GST_VIDEO_INFO_IS_YUV (&info)) {
|
||||||
|
if (range == V4L2_QUANTIZATION_FULL_RANGE
|
||||||
|
&& matrix == V4L2_YCBCR_ENC_601 && transfer == 0) {
|
||||||
|
/* Full range BT.601 YCbCr encoding with unknown primaries and transfer
|
||||||
|
* function most likely is JPEG */
|
||||||
|
colorspace = V4L2_COLORSPACE_JPEG;
|
||||||
|
transfer = V4L2_XFER_FUNC_SRGB;
|
||||||
|
} else {
|
||||||
/* SD streams likely use SMPTE170M and HD streams REC709 */
|
/* SD streams likely use SMPTE170M and HD streams REC709 */
|
||||||
if (width <= 720 && height <= 576)
|
if (width <= 720 && height <= 576)
|
||||||
colorspace = V4L2_COLORSPACE_SMPTE170M;
|
colorspace = V4L2_COLORSPACE_SMPTE170M;
|
||||||
else
|
else
|
||||||
colorspace = V4L2_COLORSPACE_REC709;
|
colorspace = V4L2_COLORSPACE_REC709;
|
||||||
|
}
|
||||||
} else if (GST_VIDEO_INFO_IS_RGB (&info)) {
|
} else if (GST_VIDEO_INFO_IS_RGB (&info)) {
|
||||||
colorspace = V4L2_COLORSPACE_SRGB;
|
colorspace = V4L2_COLORSPACE_SRGB;
|
||||||
transfer = V4L2_XFER_FUNC_NONE;
|
transfer = V4L2_XFER_FUNC_NONE;
|
||||||
|
@ -3527,10 +3561,8 @@ gst_v4l2_object_set_format_full (GstV4l2Object * v4l2object, GstCaps * caps,
|
||||||
|
|
||||||
s = gst_caps_get_structure (caps, 0);
|
s = gst_caps_get_structure (caps, 0);
|
||||||
if (gst_structure_has_field (s, "colorimetry")) {
|
if (gst_structure_has_field (s, "colorimetry")) {
|
||||||
GstVideoColorimetry ci;
|
if (!gst_v4l2_video_colorimetry_matches (&info.colorimetry,
|
||||||
if (!gst_video_colorimetry_from_string (&ci,
|
gst_structure_get_string (s, "colorimetry")))
|
||||||
gst_structure_get_string (s, "colorimetry"))
|
|
||||||
|| !gst_video_colorimetry_is_equal (&ci, &info.colorimetry))
|
|
||||||
goto invalid_colorimetry;
|
goto invalid_colorimetry;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue