On Fri, Feb 27, 2026 at 08:20:17PM +0100, Nicolas Frattaroli wrote:
> The atomic_get_output_bus_fmts callback is used by the DRM bridge layer
> to recursively select a suitable output format in a bridge chain.
>
> As a bridge that outputs to HDMI, dw-hdmi-qp will have its output
> formats determined by which formats the platform-specific integration of
> the hardware supports, and the chosen HDMI output bit depth.
>
> Implement this callback. The returned u32* buffer is supposed to be
> freed by the caller of this callback, as specified by the callback's
> documentation.
>
> Signed-off-by: Nicolas Frattaroli <[email protected]>
> ---
> drivers/gpu/drm/bridge/synopsys/dw-hdmi-qp.c | 64
> ++++++++++++++++++++++++++++
> 1 file changed, 64 insertions(+)
>
> diff --git a/drivers/gpu/drm/bridge/synopsys/dw-hdmi-qp.c
> b/drivers/gpu/drm/bridge/synopsys/dw-hdmi-qp.c
> index d649a1cf07f5..4556494f9bb1 100644
> --- a/drivers/gpu/drm/bridge/synopsys/dw-hdmi-qp.c
> +++ b/drivers/gpu/drm/bridge/synopsys/dw-hdmi-qp.c
> @@ -11,6 +11,7 @@
> #include <linux/export.h>
> #include <linux/i2c.h>
> #include <linux/irq.h>
> +#include <linux/media-bus-format.h>
> #include <linux/minmax.h>
> #include <linux/module.h>
> #include <linux/mutex.h>
> @@ -749,6 +750,68 @@ static struct i2c_adapter *dw_hdmi_qp_i2c_adapter(struct
> dw_hdmi_qp *hdmi)
> return adap;
> }
>
> +static u32*
> +dw_hdmi_qp_bridge_get_output_bus_fmts(struct drm_bridge *bridge,
> + struct drm_bridge_state *bridge_state,
> + struct drm_crtc_state *crtc_state,
> + struct drm_connector_state *conn_state,
> + unsigned int *num_output_fmts)
> +{
> + unsigned int num_fmts = 0;
> + u32 *out_fmts;
> +
> + /*
> + * bridge->supported_formats is a bit field of the HDMI_COLORSPACE_*
> enums.
> + * These enums are defined by the HDMI standard, and currently top out
> at
> + * 7. Consequently, BIT(7) is the highest bit that will be set here,
> unless
> + * the standard runs out of reserved pixel formats. Therefore,
> hweight8()
> + * will give an accurate count of how many bus formats we'll output.
> + */
> + out_fmts = kmalloc_array(hweight8(bridge->supported_formats),
> sizeof(u32),
> + GFP_KERNEL);
> + if (!out_fmts) {
> + *num_output_fmts = 0;
> + return NULL;
> + }
> +
> + switch (conn_state->hdmi.output_bpc) {
> + case 12:
> + if (bridge->supported_formats &
> BIT(DRM_OUTPUT_COLOR_FORMAT_RGB444))
> + out_fmts[num_fmts++] = MEDIA_BUS_FMT_RGB121212_1X36;
> + if (bridge->supported_formats &
> BIT(DRM_OUTPUT_COLOR_FORMAT_YCBCR444))
> + out_fmts[num_fmts++] = MEDIA_BUS_FMT_YUV12_1X36;
> + if (bridge->supported_formats &
> BIT(DRM_OUTPUT_COLOR_FORMAT_YCBCR422))
> + out_fmts[num_fmts++] = MEDIA_BUS_FMT_UYVY12_1X24;
> + if (bridge->supported_formats &
> BIT(DRM_OUTPUT_COLOR_FORMAT_YCBCR420))
> + out_fmts[num_fmts++] = MEDIA_BUS_FMT_UYYVYY12_0_5X36;
This looks like very generic code. It uses bridge->supported_formats
only. Can we move this to the helpers library?
> + break;
> + case 10:
> + if (bridge->supported_formats &
> BIT(DRM_OUTPUT_COLOR_FORMAT_RGB444))
> + out_fmts[num_fmts++] = MEDIA_BUS_FMT_RGB101010_1X30;
> + if (bridge->supported_formats &
> BIT(DRM_OUTPUT_COLOR_FORMAT_YCBCR444))
> + out_fmts[num_fmts++] = MEDIA_BUS_FMT_YUV10_1X30;
> + if (bridge->supported_formats &
> BIT(DRM_OUTPUT_COLOR_FORMAT_YCBCR422))
> + out_fmts[num_fmts++] = MEDIA_BUS_FMT_UYVY10_1X20;
> + if (bridge->supported_formats &
> BIT(DRM_OUTPUT_COLOR_FORMAT_YCBCR420))
> + out_fmts[num_fmts++] = MEDIA_BUS_FMT_UYYVYY10_0_5X30;
> + break;
> + default:
> + if (bridge->supported_formats &
> BIT(DRM_OUTPUT_COLOR_FORMAT_RGB444))
> + out_fmts[num_fmts++] = MEDIA_BUS_FMT_RGB888_1X24;
> + if (bridge->supported_formats &
> BIT(DRM_OUTPUT_COLOR_FORMAT_YCBCR444))
> + out_fmts[num_fmts++] = MEDIA_BUS_FMT_YUV8_1X24;
> + if (bridge->supported_formats &
> BIT(DRM_OUTPUT_COLOR_FORMAT_YCBCR422))
> + out_fmts[num_fmts++] = MEDIA_BUS_FMT_UYVY8_1X16;
> + if (bridge->supported_formats &
> BIT(DRM_OUTPUT_COLOR_FORMAT_YCBCR420))
> + out_fmts[num_fmts++] = MEDIA_BUS_FMT_UYYVYY8_0_5X24;
> + break;
> + }
> +
> + *num_output_fmts = num_fmts;
> +
> + return out_fmts;
> +}
> +
> static void dw_hdmi_qp_bridge_atomic_enable(struct drm_bridge *bridge,
> struct drm_atomic_state *state)
> {
> @@ -1192,6 +1255,7 @@ static int dw_hdmi_qp_cec_transmit(struct drm_bridge
> *bridge, u8 attempts,
> #endif /* CONFIG_DRM_DW_HDMI_QP_CEC */
>
> static const struct drm_bridge_funcs dw_hdmi_qp_bridge_funcs = {
> + .atomic_get_output_bus_fmts = dw_hdmi_qp_bridge_get_output_bus_fmts,
> .atomic_duplicate_state = drm_atomic_helper_bridge_duplicate_state,
> .atomic_destroy_state = drm_atomic_helper_bridge_destroy_state,
> .atomic_reset = drm_atomic_helper_bridge_reset,
>
> --
> 2.53.0
>
--
With best wishes
Dmitry