<div dir="ltr">Hi Laurent<div><br></div><div>Whoops, bit of a slip in the little "optimisation" in the v2 version. Thanks for spotting it, I'll send a small patch.</div><div><br></div><div>David</div></div><br><div class="gmail_quote"><div dir="ltr" class="gmail_attr">On Mon, 6 Feb 2023 at 17:56, Laurent Pinchart <<a href="mailto:laurent.pinchart@ideasonboard.com">laurent.pinchart@ideasonboard.com</a>> wrote:<br></div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex">Hi David,<br>
<br>
Thank you for the patch.<br>
<br>
On Fri, Jan 20, 2023 at 04:22:58PM +0000, David Plowman via libcamera-devel wrote:<br>
> We implement a custom validateColorSpaces method that forces all<br>
> (non-raw) streams to same colour space, whilst distinguishing RGB<br>
> streams from YUV ones, as the former must have the YCbCr encoding and<br>
> range over-written.<br>
> <br>
> When we apply the colour space, we always send the full YUV version as<br>
> that gets converted correctly to what our hardware drivers expect. It<br>
> is also careful to check what comes back as the YCbCr information gets<br>
> overwritten again on the way back.<br>
> <br>
> Signed-off-by: David Plowman <<a href="mailto:david.plowman@raspberrypi.com" target="_blank">david.plowman@raspberrypi.com</a>><br>
> Reviewed-by: Naushir Patuck <<a href="mailto:naush@raspberrypi.com" target="_blank">naush@raspberrypi.com</a>><br>
> ---<br>
> .../pipeline/raspberrypi/raspberrypi.cpp | 115 +++++++++++++++++-<br>
> 1 file changed, 114 insertions(+), 1 deletion(-)<br>
> <br>
> diff --git a/src/libcamera/pipeline/raspberrypi/raspberrypi.cpp b/src/libcamera/pipeline/raspberrypi/raspberrypi.cpp<br>
> index 8569df17..f768dced 100644<br>
> --- a/src/libcamera/pipeline/raspberrypi/raspberrypi.cpp<br>
> +++ b/src/libcamera/pipeline/raspberrypi/raspberrypi.cpp<br>
> @@ -310,6 +310,7 @@ class RPiCameraConfiguration : public CameraConfiguration<br>
> public:<br>
> RPiCameraConfiguration(const RPiCameraData *data);<br>
> <br>
> + CameraConfiguration::Status validateColorSpaces(ColorSpaceFlags flags);<br>
> Status validate() override;<br>
> <br>
> /* Cache the combinedTransform_ that will be applied to the sensor */<br>
> @@ -317,6 +318,13 @@ public:<br>
> <br>
> private:<br>
> const RPiCameraData *data_;<br>
> +<br>
> + /*<br>
> + * Store the colour spaces that all our streams will have. RGB format streams<br>
> + * will be the same but need the YCbCr fields cleared.<br>
> + */<br>
> + std::optional<ColorSpace> yuvColorSpace_;<br>
> + std::optional<ColorSpace> rgbColorSpace_;<br>
> };<br>
> <br>
> class PipelineHandlerRPi : public PipelineHandler<br>
> @@ -357,6 +365,105 @@ RPiCameraConfiguration::RPiCameraConfiguration(const RPiCameraData *data)<br>
> {<br>
> }<br>
> <br>
> +static const std::vector<ColorSpace> validColorSpaces = {<br>
> + ColorSpace::Sycc,<br>
> + ColorSpace::Smpte170m,<br>
> + ColorSpace::Rec709<br>
> +};<br>
> +<br>
> +static std::optional<ColorSpace> findValidColorSpace(const ColorSpace &colourSpace)<br>
> +{<br>
> + for (auto cs : validColorSpaces) {<br>
> + if (colourSpace.primaries == cs.primaries &&<br>
> + colourSpace.transferFunction == cs.transferFunction)<br>
> + return cs;<br>
> + }<br>
> +<br>
> + return std::nullopt;<br>
> +}<br>
> +<br>
> +static bool isRgb(const PixelFormat &pixFmt)<br>
> +{<br>
> + const PixelFormatInfo &info = PixelFormatInfo::info(pixFmt);<br>
> + return info.colourEncoding == PixelFormatInfo::ColourEncodingRGB;<br>
> +}<br>
> +<br>
> +static bool isYuv(const PixelFormat &pixFmt)<br>
> +{<br>
> + /* The code below would return true for raw mono streams, so weed those out first. */<br>
> + if (isRaw(pixFmt))<br>
> + return false;<br>
> +<br>
> + const PixelFormatInfo &info = PixelFormatInfo::info(pixFmt);<br>
> + return info.colourEncoding == PixelFormatInfo::ColourEncodingYUV;<br>
> +}<br>
> +<br>
> +/*<br>
> + * Raspberry Pi drivers expect the following colour spaces:<br>
> + * - V4L2_COLORSPACE_RAW for raw streams.<br>
> + * - One of V4L2_COLORSPACE_JPEG, V4L2_COLORSPACE_SMPTE170M, V4L2_COLORSPACE_REC709 for<br>
> + * non-raw streams. Other fields such as transfer function, YCbCr encoding and<br>
> + * quantisation are not used.<br>
> + *<br>
> + * The libcamera colour spaces that we wish to use corresponding to these are therefore:<br>
> + * - ColorSpace::Raw for V4L2_COLORSPACE_RAW<br>
> + * - ColorSpace::Sycc for V4L2_COLORSPACE_JPEG<br>
> + * - ColorSpace::Smpte170m for V4L2_COLORSPACE_SMPTE170M<br>
> + * - ColorSpace::Rec709 for V4L2_COLORSPACE_REC709<br>
> + */<br>
> +<br>
> +CameraConfiguration::Status RPiCameraConfiguration::validateColorSpaces([[maybe_unused]] ColorSpaceFlags flags)<br>
> +{<br>
> + Status status = Valid;<br>
> + yuvColorSpace_.reset();<br>
> +<br>
> + for (auto cfg : config_) {<br>
> + /* First fix up raw streams to have the "raw" colour space. */<br>
> + if (isRaw(cfg.pixelFormat)) {<br>
> + /* If there was no value here, that doesn't count as "adjusted". */<br>
> + if (cfg.colorSpace && cfg.colorSpace != ColorSpace::Raw) {<br>
> + status = Adjusted;<br>
> + cfg.colorSpace = ColorSpace::Raw;<br>
<br>
Agreed for the Adjusted flag, but shouldn't you set cfg.colorSpace to<br>
raw even if !cfg.colorSpace ?<br>
<br>
> + }<br>
> + continue;<br>
> + }<br>
> +<br>
> + /* Next we need to find our shared colour space. The first valid one will do. */<br>
> + if (cfg.colorSpace && !yuvColorSpace_)<br>
> + yuvColorSpace_ = findValidColorSpace(cfg.colorSpace.value());<br>
> + }<br>
> +<br>
> + /* If no colour space was given anywhere, choose sYCC. */<br>
> + if (!yuvColorSpace_)<br>
> + yuvColorSpace_ = ColorSpace::Sycc;<br>
> +<br>
> + /* Note the version of this that any RGB streams will have to use. */<br>
> + rgbColorSpace_ = yuvColorSpace_;<br>
> + rgbColorSpace_->ycbcrEncoding = ColorSpace::YcbcrEncoding::None;<br>
> + rgbColorSpace_->range = ColorSpace::Range::Full;<br>
> +<br>
> + /* Go through the streams again and force everyone to the same colour space. */<br>
> + for (auto cfg : config_) {<br>
> + if (cfg.colorSpace == ColorSpace::Raw)<br>
> + continue;<br>
> +<br>
> + if (isYuv(cfg.pixelFormat) && cfg.colorSpace != yuvColorSpace_) {<br>
> + /* Again, no value means "not adjusted". */<br>
> + if (cfg.colorSpace)<br>
> + status = Adjusted;<br>
> + cfg.colorSpace = yuvColorSpace_;<br>
> + }<br>
> + if (isRgb(cfg.pixelFormat) && cfg.colorSpace != rgbColorSpace_) {<br>
> + /* Be nice, and let the YUV version count as non-adjusted too. */<br>
> + if (cfg.colorSpace && cfg.colorSpace != yuvColorSpace_)<br>
> + status = Adjusted;<br>
> + cfg.colorSpace = rgbColorSpace_;<br>
> + }<br>
> + }<br>
> +<br>
> + return status;<br>
> +}<br>
> +<br>
> CameraConfiguration::Status RPiCameraConfiguration::validate()<br>
> {<br>
> Status status = Valid;<br>
> @@ -533,7 +640,8 @@ CameraConfiguration::Status RPiCameraConfiguration::validate()<br>
> V4L2DeviceFormat format;<br>
> format.fourcc = dev->toV4L2PixelFormat(cfg.pixelFormat);<br>
> format.size = cfg.size;<br>
> - format.colorSpace = cfg.colorSpace;<br>
> + /* We want to send the associated YCbCr info through to the driver. */<br>
> + format.colorSpace = yuvColorSpace_;<br>
> <br>
> LOG(RPI, Debug)<br>
> << "Try color space " << ColorSpace::toString(cfg.colorSpace);<br>
> @@ -542,6 +650,11 @@ CameraConfiguration::Status RPiCameraConfiguration::validate()<br>
> if (ret)<br>
> return Invalid;<br>
> <br>
> + /*<br>
> + * But for RGB streams, the YCbCr info gets overwritten on the way back<br>
> + * so we must check against what the stream cfg says, not what we actually<br>
> + * requested (which carefully included the YCbCr info)!<br>
> + */<br>
> if (cfg.colorSpace != format.colorSpace) {<br>
> status = Adjusted;<br>
> LOG(RPI, Debug)<br>
<br>
-- <br>
Regards,<br>
<br>
Laurent Pinchart<br>
</blockquote></div>