Add new .atomic_get_input_bus_cfgs callback into struct drm_bridge_funcs {}.
This is an extended version of .atomic_get_input_bus_fmts callback which
only returns list of bus formats supported by a bridge and provides no way
to e.g. limit clock frequency required between neighboring bridges.

The new .atomic_get_input_bus_cfgs callback returns list of struct drm_bus_cfg,
which currently contains format and bus flags, but can be extended with other
members, like desired clock frequency, as required.

The .atomic_get_input_bus_cfgs should replace the .atomic_get_input_bus_fmts
once drivers get converted over. The conversion could be done using a script.

Signed-off-by: Marek Vasut <ma...@denx.de>
Cc: Laurent Pinchart <laurent.pinch...@ideasonboard.com>
Cc: Maxime Ripard <max...@cerno.tech>
Cc: Neil Armstrong <narmstr...@baylibre.com>
Cc: Sam Ravnborg <s...@ravnborg.org>
---
 drivers/gpu/drm/drm_bridge.c | 47 ++++++++++++++++++++++++------------
 include/drm/drm_bridge.h     | 42 ++++++++++++++++++++++++++++++++
 2 files changed, 74 insertions(+), 15 deletions(-)

diff --git a/drivers/gpu/drm/drm_bridge.c b/drivers/gpu/drm/drm_bridge.c
index f052d50106758..a069c50cc7d6b 100644
--- a/drivers/gpu/drm/drm_bridge.c
+++ b/drivers/gpu/drm/drm_bridge.c
@@ -841,7 +841,8 @@ static int select_bus_fmt_recursive(struct drm_bridge 
*first_bridge,
         * hope that it can handle this situation gracefully (by providing
         * appropriate default values).
         */
-       if (!cur_bridge->funcs->atomic_get_input_bus_fmts) {
+       if (!cur_bridge->funcs->atomic_get_input_bus_cfgs &&
+           !cur_bridge->funcs->atomic_get_input_bus_fmts) {
                if (cur_bridge != first_bridge) {
                        fixed_bus_cfg.format = MEDIA_BUS_FMT_FIXED;
                        ret = select_bus_fmt_recursive(first_bridge,
@@ -865,32 +866,48 @@ static int select_bus_fmt_recursive(struct drm_bridge 
*first_bridge,
        }
 
        /*
-        * If the driver implements ->atomic_get_input_bus_fmts() it
-        * should also implement the atomic state hooks.
+        * If the driver implements ->atomic_get_input_bus_cfgs()
+        * or legacy ->atomic_get_input_bus_fmts() it should also
+        * implement the atomic state hooks.
         */
        if (WARN_ON(!cur_state))
                return -EINVAL;
 
-       in_bus_fmts = cur_bridge->funcs->atomic_get_input_bus_fmts(cur_bridge,
+       if (cur_bridge->funcs->atomic_get_input_bus_cfgs) {
+               in_bus_cfgs = cur_bridge->funcs->atomic_get_input_bus_cfgs(
+                                                       cur_bridge,
+                                                       cur_state,
+                                                       crtc_state,
+                                                       conn_state,
+                                                       out_bus_cfg,
+                                                       &num_in_bus_fmts);
+               if (!num_in_bus_fmts)
+                       return -ENOTSUPP;
+               else if (!in_bus_cfgs)
+                       return -ENOMEM;
+       } else {
+               in_bus_fmts = cur_bridge->funcs->atomic_get_input_bus_fmts(
+                                                       cur_bridge,
                                                        cur_state,
                                                        crtc_state,
                                                        conn_state,
                                                        out_bus_cfg->format,
                                                        &num_in_bus_fmts);
-       if (!num_in_bus_fmts)
-               return -ENOTSUPP;
-       else if (!in_bus_fmts)
-               return -ENOMEM;
+               if (!num_in_bus_fmts)
+                       return -ENOTSUPP;
+               else if (!in_bus_fmts)
+                       return -ENOMEM;
 
-       /* Transcribe in_bus_fmts to in_bus_cfgs */
-       in_bus_cfgs = kcalloc(num_in_bus_fmts, sizeof(*in_bus_cfgs), 
GFP_KERNEL);
-       if (!in_bus_cfgs)
-               return -ENOMEM;
+               /* Transcribe in_bus_fmts to in_bus_cfgs */
+               in_bus_cfgs = kcalloc(num_in_bus_fmts, sizeof(*in_bus_cfgs), 
GFP_KERNEL);
+               if (!in_bus_cfgs)
+                       return -ENOMEM;
 
-       for (i = 0; i < num_in_bus_fmts; i++)
-               in_bus_cfgs[i].format = in_bus_fmts[i];
+               for (i = 0; i < num_in_bus_fmts; i++)
+                       in_bus_cfgs[i].format = in_bus_fmts[i];
 
-       kfree(in_bus_fmts);
+               kfree(in_bus_fmts);
+       }
 
        if (first_bridge == cur_bridge) {
                cur_state->input_bus_cfg.format = in_bus_cfgs[0].format;
diff --git a/include/drm/drm_bridge.h b/include/drm/drm_bridge.h
index 061d87313fac9..b2fe0ee7294b5 100644
--- a/include/drm/drm_bridge.h
+++ b/include/drm/drm_bridge.h
@@ -439,6 +439,48 @@ struct drm_bridge_funcs {
                                           struct drm_connector_state 
*conn_state,
                                           unsigned int *num_output_fmts);
 
+       /**
+        * @atomic_get_input_bus_cfgs:
+        *
+        * Return the supported bus configurations on the input end of a bridge
+        * for a specific output bus format.
+        *
+        * The returned array must be allocated with kmalloc() and will be
+        * freed by the caller. If the allocation fails, NULL should be
+        * returned. num_output_fmts must be set to the returned array size.
+        * Formats listed in the returned array should be listed in decreasing
+        * preference order (the core will try all formats until it finds one
+        * that works). When the format is not supported NULL should be
+        * returned and num_output_fmts should be set to 0.
+        *
+        * This method is called on all elements of the bridge chain as part of
+        * the bus format negotiation process that happens in
+        * drm_atomic_bridge_chain_select_bus_fmts().
+        * This method is optional. When not implemented, the core will bypass
+        * bus format negotiation on this element of the bridge without
+        * failing, and the previous element in the chain will be passed
+        * MEDIA_BUS_FMT_FIXED as its output bus format.
+        *
+        * Bridge drivers that need to support being linked to bridges that are
+        * not supporting bus format negotiation should handle the
+        * output_fmt == MEDIA_BUS_FMT_FIXED case appropriately, by selecting a
+        * sensible default value or extracting this information from somewhere
+        * else (FW property, &drm_display_mode, &drm_display_info, ...)
+        *
+        * Note: Even if input format selection on the first bridge has no
+        * impact on the negotiation process (bus format negotiation stops once
+        * we reach the first element of the chain), drivers are expected to
+        * return accurate input formats as the input format may be used to
+        * configure the CRTC output appropriately.
+        */
+       struct drm_bus_cfg *(*atomic_get_input_bus_cfgs)
+                                       (struct drm_bridge *bridge,
+                                        struct drm_bridge_state *bridge_state,
+                                        struct drm_crtc_state *crtc_state,
+                                        struct drm_connector_state *conn_state,
+                                        struct drm_bus_cfg *output_cfg,
+                                        unsigned int *num_input_cfgs);
+
        /**
         * @atomic_get_input_bus_fmts:
         *
-- 
2.34.1

Reply via email to