Message ID | 20241231-hotplug-drm-bridge-v5-10-173065a1ece1@bootlin.com (mailing list archive) |
---|---|
State | New |
Headers | show |
Series | Add support for hot-pluggable DRM bridges | expand |
On Tue, Dec 31, 2024 at 11:40:04AM +0100, Luca Ceresoli wrote: > This driver implements the point of a DRM pipeline where a connector allows > removal of all the following bridges up to the panel. > > The DRM subsystem currently allows hotplug of the monitor but not preceding > components. However there are embedded devices where the "tail" of the DRM > pipeline, including one or more bridges, can be physically removed: > > .------------------------. > | DISPLAY CONTROLLER | > | .---------. .------. | > | | ENCODER |<--| CRTC | | > | '---------' '------' | > '------|-----------------' > | > | HOTPLUG > V CONNECTOR > .---------. .--. .-. .---------. .-------. > | 0 to N | | _| _| | | 1 to N | | | > | BRIDGES |--DSI-->||_ |_ |--DSI-->| BRIDGES |--LVDS-->| PANEL | > | | | | | | | | | | > '---------' '--' '-' '---------' '-------' > > [--- fixed components --] [----------- removable add-on -----------] > > This driver supports such a device, where the final segment of a MIPI DSI > bus, including one or more bridges, can be physically disconnected and > reconnected at runtime, possibly with a different model. > > The add-on supported by this driver has a MIPI DSI bus traversing the > hotplug connector and a DSI to LVDS bridge and an LVDS panel on the add-on. > Hovever this driver is designed to be as far as possible generic and > extendable to other busses that have no native hotplug and model ID > discovery. > > This driver does not itself add and remove the bridges or panel on the > add-on: this needs to be done by other means, e.g. device tree overlay > runtime insertion and removal. The hotplug-bridge gets notified by the DRM > bridge core after a removable bridge gets added or before it is removed. > > The hotplug-bridge role is to implement the "hot-pluggable connector" in > the bridge chain. In this position, what the hotplug-bridge should ideally > do is: > > * communicate with the previous component (bridge or encoder) so that it > believes it always has a connected bridge following it and the DRM card > is always present > * be notified of the addition and removal of the following bridge and > attach/detach to/from it > * communicate with the following bridge so that it will attach and detach > using the normal procedure (as if the entire pipeline were being created > or destroyed, not only the tail) > * instantiate two DRM connectors (similarly to what the DisplayPort MST > code does): > - a DSI connector representing the video lines of the hotplug connector; > the status is always "disconnected" (no panel is ever attached > directly to it) > - an LSVD connector representing the classic connection to the panel; > this gets added/removed whenever the add-on gets > connected/disconnected; the status is always "connected" as the panel > is always connected to the preceding bridge I'd rather have just a single connector. MST connectors can be added and gone as there is fit, so should be your LVDS panel-related connector. I think with the bridge refcounting and with proper notifications there should be a perfect way to do so. On bridge removal there should be a call to the callback from drm_encoder (or drm_connector), which removes corresponding connector (they are dynamic!), removes a part of the bridge chain, notifying all parties while we are doing it. Yes, this needs one major change to the existing drivers: we have been pushing towards drm_bridge_connector to be created during DRM card creation time. It might be a good time to actually make a step and make drm_bridge_connector to be created by the drm_bridge_attach() if there is none at the end of the drm_bridge chain (or if a special flag is being passed to it). Then make hotplug bridge signal (e.g. via the type) that the drm_bridge_connector should not be created by default. However once the full bridge chain is initialized (how do we identify that, BTW?) the hotplug bridge (or the core?) can create the drm_bride_connector. > However some aspects make it a bit more complex than that. Most notably: > > * the next bridge can be probed and removed at any moment and all probing > sequences need to be handled > * the DSI host/device registration process, which adds to the DRM bridge > attach process, makes the initial card registration tricky > * the need to register and deregister the following bridges at runtime > without tearing down the whole DRM card prevents using some of the > functions that are normally recommended > * the automatic mechanism to call the appropriate .get_modes operation > (typically provided by the panel bridge) cannot work as the panel can > disappear and reappear as a different model, so an ad-hoc lookup is > needed > > The code handling these and other tricky aspects is accurately documented > by comments in the code. > > The userspace representation when the add-on is not connected is: > > # modetest -c | grep -i '^[a-z0-9]' > Connectors: > id encoder status name size (mm) modes encoders > 38 0 disconnected DSI-1 0x0 0 37 I'm really sorry if I missed that, but the cover letter doesn't explain, why do you need this connector at all. I think it might confuse userspace and users. > > And when it is connected a new connector appears: > > # modetest -c | grep -i '^[a-z0-9]' > Connectors: > id encoder status name size (mm) modes encoders > 38 0 disconnected DSI-1 0x0 0 37 > 39 0 connected LVDS-1 344x194 1 37 > > Co-developed-by: Paul Kocialkowski <paul.kocialkowski@bootlin.com> > Signed-off-by: Paul Kocialkowski <paul.kocialkowski@bootlin.com> > Signed-off-by: Luca Ceresoli <luca.ceresoli@bootlin.com> > > --- > > Changed in v5: > - use drm_bridge dynamic lifetime management > - refcount bridge_modes and next_bridge via drm_bridge_get()/drm_bridge_put() > - fix dynconn removal by using drm_connector_put() and making the .destroy > callback work > - add hotplug_bridge_grab() in hotplug_bridge_attach() to handle the case > where the encoder driver is probed last > - migrate platform driver from .remove_new to .remove > > Changed in v4: > - convert from generic notifiers to the new bridge_event_notify bridge > func > - improved and updated commit message, adding 'modetest' output > with/without add-on > - select DRM_DISPLAY_HELPER and DRM_BRIDGE_CONNECTOR, required after > commit 9da7ec9b19d8 ("drm/bridge-connector: move to DRM_DISPLAY_HELPER > module") > > Changed in v3: > - dynamically add/remove the LVDS connector on hot(un)plug > - take the firmware node normally via dev->of_node instead of using > device_set_node(); this makes code more self-contained and generic > - minor rewordings and cleanups > > Changed in v2: > - change to be a platform device instantiated from the connector driver > instead of a self-standing OF driver > - add missing error handling for devm_drm_bridge_add() > - various cleanups and style improvements > - fix typo in comment > --- > MAINTAINERS | 5 + > drivers/gpu/drm/bridge/Kconfig | 17 + > drivers/gpu/drm/bridge/Makefile | 1 + > drivers/gpu/drm/bridge/hotplug-bridge.c | 695 ++++++++++++++++++++++++++++++++ > 4 files changed, 718 insertions(+) > > diff --git a/MAINTAINERS b/MAINTAINERS > index 1675e44799e55d72711a6251f692a4a14bc3a84a..ab255f5696880258deee55d99ff0a7cde85efb7c 100644 > --- a/MAINTAINERS > +++ b/MAINTAINERS > @@ -7182,6 +7182,11 @@ T: git https://gitlab.freedesktop.org/drm/misc/kernel.git > F: Documentation/devicetree/bindings/display/panel/himax,hx8394.yaml > F: drivers/gpu/drm/panel/panel-himax-hx8394.c > > +DRM DRIVER FOR HOTPLUG VIDEO CONNECTOR BRIDGE > +M: Luca Ceresoli <luca.ceresoli@bootlin.com> > +S: Maintained > +F: drivers/gpu/drm/bridge/hotplug-bridge.c > + > DRM DRIVER FOR HX8357D PANELS > S: Orphan > T: git https://gitlab.freedesktop.org/drm/misc/kernel.git > diff --git a/drivers/gpu/drm/bridge/Kconfig b/drivers/gpu/drm/bridge/Kconfig > index 6b4664d91faa80f096ac6a0548ed342e802ae68b..f01971638d6818e33b32217922e165a8c18d51ee 100644 > --- a/drivers/gpu/drm/bridge/Kconfig > +++ b/drivers/gpu/drm/bridge/Kconfig > @@ -90,6 +90,23 @@ config DRM_FSL_LDB > help > Support for i.MX8MP DPI-to-LVDS on-SoC encoder. > > +config DRM_HOTPLUG_BRIDGE > + tristate "Hotplug DRM bridge support" > + depends on OF > + select DRM_PANEL_BRIDGE > + select DRM_MIPI_DSI > + select DRM_KMS_HELPER > + select DRM_DISPLAY_HELPER > + select DRM_BRIDGE_CONNECTOR > + help > + Driver for a DRM bridge representing a physical connector that > + splits a DRM pipeline into a fixed part and a physically > + removable part. The fixed part includes up to the encoder and > + zero or more bridges. The removable part includes any following > + bridges up to the connector and panel and can be physically > + removed and connected at runtime, possibly with different > + components. > + > config DRM_ITE_IT6263 > tristate "ITE IT6263 LVDS/HDMI bridge" > depends on OF > diff --git a/drivers/gpu/drm/bridge/Makefile b/drivers/gpu/drm/bridge/Makefile > index 97304b429a530c108dcbff906965cda091b0a7a2..2f6ae1a97d15045316ee191c04dbc65650162bab 100644 > --- a/drivers/gpu/drm/bridge/Makefile > +++ b/drivers/gpu/drm/bridge/Makefile > @@ -6,6 +6,7 @@ obj-$(CONFIG_DRM_CHRONTEL_CH7033) += chrontel-ch7033.o > obj-$(CONFIG_DRM_CROS_EC_ANX7688) += cros-ec-anx7688.o > obj-$(CONFIG_DRM_DISPLAY_CONNECTOR) += display-connector.o > obj-$(CONFIG_DRM_FSL_LDB) += fsl-ldb.o > +obj-$(CONFIG_DRM_HOTPLUG_BRIDGE) += hotplug-bridge.o > obj-$(CONFIG_DRM_ITE_IT6263) += ite-it6263.o > obj-$(CONFIG_DRM_ITE_IT6505) += ite-it6505.o > obj-$(CONFIG_DRM_LONTIUM_LT8912B) += lontium-lt8912b.o > diff --git a/drivers/gpu/drm/bridge/hotplug-bridge.c b/drivers/gpu/drm/bridge/hotplug-bridge.c > new file mode 100644 > index 0000000000000000000000000000000000000000..f717206287fc598cf7a3c5ac5bf9e1be4c8540d9 > --- /dev/null > +++ b/drivers/gpu/drm/bridge/hotplug-bridge.c > @@ -0,0 +1,695 @@ > +// SPDX-License-Identifier: GPL-2.0 > +/* > + * A DRM bridge representing the split point between a fixed part of the > + * DRM pipeline and a physically removable part. The fixed part includes up > + * to the encoder and zero or more bridges. Insertion and removal of the > + * "downstream" components happens via device driver probe/removal. > + * > + * Copyright (C) 2024, GE HealthCare > + * > + * Authors: > + * Luca Ceresoli <luca.ceresoli@bootlin.com> > + * Paul Kocialkowski <paul.kocialkowski@bootlin.com> > + */ > + > +#include <linux/mutex.h> > +#include <linux/of.h> > +#include <linux/of_graph.h> > +#include <linux/platform_device.h> > + > +#include <drm/drm_atomic_helper.h> > +#include <drm/drm_bridge.h> > +#include <drm/drm_bridge_connector.h> > +#include <drm/drm_mipi_dsi.h> > +#include <drm/drm_of.h> > +#include <drm/drm_probe_helper.h> > + > +/* > + * Internal hotplug-bridge data. > + * > + * We have two 'struct drm_connector' here: > + * - fixconn represents the DSI video lines on the hotplug connector where > + * the removable part attaches. It is thus always instantiated. > + * - dynconn represents the LVDS video lines where the panel is attached. > + * It is part of the removable part of the video pipeline and as such is > + * added and removed dynamically based on when the downstream devices > + * appear and disappear. > + */ > +struct hotplug_bridge { > + struct device *dev; > + > + /* Local bridge */ > + struct drm_bridge bridge; > + > + /* Always-present connector (where the removal part will connect to) */ > + struct drm_connector *fixconn; > + > + /* Downstream bridge (next in the chain) */ > + struct drm_bridge *next_bridge; > + /* Protect next_bridge */ > + struct mutex next_bridge_mutex; > + > + /* Pointer to the last bridge exposing OP_MODES */ > + struct drm_bridge *bridge_modes; > + > + /* The "tail" connector that gets added/removed at runtime */ > + struct drm_connector dynconn; > + > + /* Local DSI host, for the downstream DSI device to attach to */ > + struct mipi_dsi_host dsi_host; > + /* Local DSI device, attached to the upstream DSI host */ > + struct mipi_dsi_device *dsi_dev; > + /* Upstream DSI host (the actual DSI controller) */ > + struct mipi_dsi_host *prev_dsi_host; > + > + struct work_struct hpd_work; > +}; > + > +static struct hotplug_bridge *hotplug_bridge_from_drm_bridge(struct drm_bridge *bridge) > +{ > + return container_of(bridge, struct hotplug_bridge, bridge); > +} > + > +/* -------------------------------------------------------------------------- > + * dynconn implementation > + */ > +static struct hotplug_bridge *hotplug_bridge_from_dynconn(struct drm_connector *conn) > +{ > + return container_of(conn, struct hotplug_bridge, dynconn); > +} > + > +static int hotplug_bridge_dynconn_get_modes(struct drm_connector *connector) > +{ > + struct hotplug_bridge *hpb = hotplug_bridge_from_dynconn(connector); > + > + if (hpb->bridge_modes) > + return hpb->bridge_modes->funcs->get_modes(hpb->bridge_modes, connector); > + > + return 0; > +} > + > +static const struct drm_connector_helper_funcs hotplug_bridge_dynconn_connector_helper_funcs = { > + .get_modes = hotplug_bridge_dynconn_get_modes, > +}; > + > +static void hotplug_bridge_dynconn_destroy(struct drm_connector *connector) > +{ > + drm_connector_unregister(connector); > + drm_connector_cleanup(connector); > +} > + > +static const struct drm_connector_funcs dynconn_funcs = { > + .destroy = hotplug_bridge_dynconn_destroy, > + .atomic_duplicate_state = drm_atomic_helper_connector_duplicate_state, > + .atomic_destroy_state = drm_atomic_helper_connector_destroy_state, > + .fill_modes = drm_helper_probe_single_connector_modes, > +}; > + > +/* > + * In non-removable pipelines using a "bridge connector", > + * drm_bridge_connector_init() stores in the bridge_connector a pointer to > + * the last bridge having OP_MODES (typically the panel bridge), so the > + * .get_modes op will automatically be called on that bridge (it also takes > + * pointers to other bridges which we don't care about). The "bridge > + * connector" is too restrictive for our use case so we cannot use it. But > + * we need a pointer to the modes-providing bridge, so we need to replicate > + * that bit of its logic. > + * > + * If no modes bridge is found, nothing is done. This is allowed. > + * > + * Also get the modes bridge to tie its lifetime to ours. > + */ > +static void hotplug_bridge_dynconn_bridge_modes_get(struct hotplug_bridge *hpb) > +{ > + struct drm_bridge *bridge; > + > + if (WARN_ON(!hpb->next_bridge || !hpb->bridge.encoder)) > + return; > + > + drm_for_each_bridge_in_chain(hpb->bridge.encoder, bridge) > + if (bridge->ops & DRM_BRIDGE_OP_MODES) { > + drm_bridge_get(bridge); > + hpb->bridge_modes = bridge; > + } > +} > + > +static void hotplug_bridge_dynconn_bridge_modes_put(struct hotplug_bridge *hpb) > +{ > + if (hpb->bridge_modes) > + drm_bridge_put_and_clear(hpb->bridge_modes); > +} > + > +static int hotplug_bridge_dynconn_add(struct hotplug_bridge *hpb) > +{ > + int err; > + > + err = drm_connector_init(hpb->bridge.dev, &hpb->dynconn, &dynconn_funcs, > + DRM_MODE_CONNECTOR_LVDS); > + if (err) > + return err; > + > + drm_atomic_helper_connector_reset(&hpb->dynconn); > + > + drm_connector_helper_add(&hpb->dynconn, > + &hotplug_bridge_dynconn_connector_helper_funcs); > + > + drm_connector_attach_encoder(&hpb->dynconn, hpb->bridge.encoder); > + if (err) > + goto err_cleanup; > + > + hotplug_bridge_dynconn_bridge_modes_get(hpb); > + > + err = drm_connector_register(&hpb->dynconn); > + if (err) > + goto err_cleanup; > + > + return 0; > + > +err_cleanup: > + drm_connector_cleanup(&hpb->dynconn); > + hotplug_bridge_dynconn_bridge_modes_put(hpb); > + return err; > +} > + > +/* ----------------------------------------------------------------------- */ > + > +/* > + * Attach the remote bridge to the encoder and to the next bridge in the > + * chain, if possible. For this to succeed, we need to know: > + * > + * - the encoder, which is set at the first drm_bridge_attach() time > + * - the next bridge, which is obtained via a notifier whenever the next > + * bridge is (re)probed, or at probe time in case it was probed before us > + * > + * In order to handle different execution sequences, this function can be > + * called from multiple places and needs to check all the prerequisites > + * every time, and it will act only if both are met. > + * > + * Must be called with hpb->next_bridge_mutex held. > + * > + * Returns 0 if the encoder was attached successfully, -ENODEV if any of > + * the two prerequisites above is not met (no encoder or no next bridge), > + * the error returned by drm_bridge_attach() otherwise. > + */ > +static int hotplug_bridge_attach_to_encoder_chain(struct hotplug_bridge *hpb) > +{ > + int ret; > + > + if (!hpb->next_bridge || !hpb->bridge.encoder) > + return -ENODEV; > + > + ret = drm_bridge_attach(hpb->bridge.encoder, hpb->next_bridge, &hpb->bridge, > + DRM_BRIDGE_ATTACH_NO_CONNECTOR); > + if (ret) > + return dev_err_probe(hpb->dev, ret, "drm_bridge_attach failed\n"); > + > + dev_dbg(hpb->dev, "attached to encoder chain\n"); > + > + return 0; > +} > + > +/* > + * Stop the video pipeline and detach next_bridge. > + * > + * Must be called with hpb->next_bridge_mutex held. > + */ > +static void hotplug_bridge_detach_from_encoder_chain(struct hotplug_bridge *hpb) > +{ > + WARN_ON_ONCE(!hpb->next_bridge); > + > + dev_dbg(hpb->dev, "detaching from encoder chain\n"); > + > + drm_atomic_helper_shutdown(hpb->bridge.dev); > + > + drm_encoder_cleanup_from(hpb->bridge.encoder, hpb->next_bridge); > +} > + > +static void hotplug_bridge_grab(struct hotplug_bridge *hpb) > +{ > + struct device *dev = hpb->dev; > + struct drm_bridge *bridge; > + struct drm_panel *panel; > + int err; > + > + mutex_lock(&hpb->next_bridge_mutex); > + > + if (hpb->next_bridge) > + goto out_unlock; > + > + /* > + * This is supposed to be replaced by devm_drm_of_get_bridge(), but > + * that is a devm_, and we need to remove the panel bridge also on > + * next_bridge disconnect. > + */ > + err = drm_of_find_panel_or_bridge(dev->of_node, 1, 0, &panel, &bridge); > + if (err) > + goto out_unlock; > + > + /* Convert the remote panel to a bridge */ > + if (panel) > + bridge = drm_panel_bridge_add(panel); > + if (IS_ERR(bridge)) > + goto out_unlock; > + > + drm_bridge_get(bridge); > + hpb->next_bridge = bridge; > + > + dev_dbg(dev, "grabbed next bridge (%pOFn)\n", hpb->next_bridge->of_node); > + > + hpb->bridge.pre_enable_prev_first = hpb->next_bridge->pre_enable_prev_first; > + > + err = hotplug_bridge_attach_to_encoder_chain(hpb); > + if (err) > + goto err_panel_bridge_remove; > + > + err = hotplug_bridge_dynconn_add(hpb); > + if (err) > + goto err_detach_from_encoder_chain; > + > + queue_work(system_wq, &hpb->hpd_work); > + goto out_unlock; > + > +err_detach_from_encoder_chain: > + hotplug_bridge_detach_from_encoder_chain(hpb); > +err_panel_bridge_remove: > + drm_panel_bridge_remove(hpb->next_bridge); > + drm_bridge_put_and_clear(hpb->next_bridge); > +out_unlock: > + mutex_unlock(&hpb->next_bridge_mutex); > +} > + > +/* > + * Detach from the next bridge and remove the panel bridge, either on > + * release or when the downstream bridge is being removed. > + * > + * Can be called in these ways: > + * > + * - bridge_being_removed is NULL: detach unconditionally > + * (this is useful on .remove() to teardown everything) > + * - bridge_being_removed == hpb->next_bridge: detach > + * (the downstream bridge is being removed) > + * - bridge_being_removed != hpb->next_bridge: do nothing > + * (the bridge being removed is not the downstream bridge) > + * > + * In all cases, does nothing when there is no downstream bridge. > + */ > +static void hotplug_bridge_release(struct hotplug_bridge *hpb, > + struct drm_bridge *bridge_being_removed) > +{ > + mutex_lock(&hpb->next_bridge_mutex); > + > + if (!hpb->next_bridge) > + goto out; > + > + if (bridge_being_removed && bridge_being_removed != hpb->next_bridge) > + goto out; > + > + if (hpb->bridge_modes) > + hotplug_bridge_dynconn_bridge_modes_put(hpb); > + > + dev_dbg(hpb->dev, "releasing next bridge (%pOFn)\n", hpb->next_bridge->of_node); > + hotplug_bridge_detach_from_encoder_chain(hpb); > + > + dev_dbg(hpb->dev, "removing %s connector\n", hpb->dynconn.name); > + drm_connector_put(&hpb->dynconn); > + > + /* > + * This will check that the bridge actually belongs to panel-bridge > + * before doing anything with it, so we can safely always call it. > + */ > + drm_panel_bridge_remove(hpb->next_bridge); > + drm_bridge_put_and_clear(hpb->next_bridge); > + > + queue_work(system_wq, &hpb->hpd_work); > + > +out: > + mutex_unlock(&hpb->next_bridge_mutex); > +} > + > +static void hotplug_bridge_bridge_event_notify(struct drm_bridge *bridge, > + enum drm_bridge_event_type event, > + struct drm_bridge *event_bridge) > +{ > + struct hotplug_bridge *hpb = container_of(bridge, struct hotplug_bridge, bridge); > + > + switch (event) { > + case DRM_EVENT_BRIDGE_ADD: > + hotplug_bridge_grab(hpb); > + break; > + case DRM_EVENT_BRIDGE_REMOVE: > + hotplug_bridge_release(hpb, event_bridge); > + break; > + } > +} > + > +static int hotplug_bridge_attach(struct drm_bridge *bridge, > + enum drm_bridge_attach_flags flags) > +{ > + struct hotplug_bridge *hpb = hotplug_bridge_from_drm_bridge(bridge); > + struct device *dev = hpb->dev; > + struct drm_connector *connector; > + struct drm_encoder *encoder = hpb->bridge.encoder; > + int err; > + > + /* Encoder was not yet provided to our bridge */ > + if (!encoder) > + return -ENODEV; > + > + /* Connector was already created */ > + if (hpb->fixconn) > + return dev_err_probe(dev, -EBUSY, "connector already created\n"); > + > + connector = drm_bridge_connector_init(bridge->dev, encoder); > + if (IS_ERR(connector)) > + return dev_err_probe(dev, PTR_ERR(connector), "failed to initialize connector\n"); > + > + drm_connector_attach_encoder(connector, encoder); > + > + hpb->fixconn = connector; > + > + drm_connector_register(connector); > + > + mutex_lock(&hpb->next_bridge_mutex); > + err = hotplug_bridge_attach_to_encoder_chain(hpb); > + mutex_unlock(&hpb->next_bridge_mutex); > + > + /* -ENODEV is acceptable, in case next_bridge is not yet known */ > + if (err == -ENODEV) > + err = 0; > + > + /* > + * If the encoder driver is probed last, the > + * hotplug_bridge_attach_to_encoder_chain() call in > + * hotplug_bridge_grab() fails because hpb->bridge.encoder is still > + * NULL, and hotplug_bridge_grab() will not have another chance to > + * execute. So call it now, at the end of the encoder attach > + * process. > + */ > + hotplug_bridge_grab(hpb); > + > + return err; > +} > + > +static void hotplug_bridge_detach(struct drm_bridge *bridge) > +{ > + struct hotplug_bridge *hpb = hotplug_bridge_from_drm_bridge(bridge); > + > + mutex_lock(&hpb->next_bridge_mutex); > + hotplug_bridge_detach_from_encoder_chain(hpb); > + mutex_unlock(&hpb->next_bridge_mutex); > + > + if (hpb->fixconn) { > + drm_connector_unregister(hpb->fixconn); > + drm_connector_cleanup(hpb->fixconn); > + hpb->fixconn = NULL; > + } > +} > + > +static void hotplug_bridge_destroy(struct drm_bridge *bridge) > +{ > + struct hotplug_bridge *hpb = hotplug_bridge_from_drm_bridge(bridge); > + > + kfree(hpb); > +} > + > +/* > + * The fixed connector is never attached to a panel, so it should always be > + * reported as disconnected. > + */ > +static enum drm_connector_status hotplug_bridge_detect(struct drm_bridge *bridge) > +{ > + return connector_status_disconnected; > +} > + > +static void hotplug_bridge_hpd_work_func(struct work_struct *work) > +{ > + struct hotplug_bridge *hpb = container_of(work, struct hotplug_bridge, hpd_work); > + > + if (hpb->bridge.dev) > + drm_helper_hpd_irq_event(hpb->bridge.dev); > +} > + > +static const struct drm_bridge_funcs hotplug_bridge_funcs = { > + .attach = hotplug_bridge_attach, > + .detach = hotplug_bridge_detach, > + .destroy = hotplug_bridge_destroy, > + .detect = hotplug_bridge_detect, > + .bridge_event_notify = hotplug_bridge_bridge_event_notify, > +}; > + > +static int hotplug_bridge_dsi_detach(struct mipi_dsi_host *host, > + struct mipi_dsi_device *device_remote) > +{ > + struct hotplug_bridge *hpb = dev_get_drvdata(host->dev); > + > + if (!hpb->dsi_dev) > + return -ENODEV; > + > + mipi_dsi_detach(hpb->dsi_dev); > + mipi_dsi_device_unregister(hpb->dsi_dev); > + hpb->dsi_dev = NULL; > + > + return 0; > +} > + > +/* > + * Attach the local DSI device to the upstream DSI host, possibly with a > + * "null" format. > + * > + * In "normal" bridges this function should be _only_ used as the .attach > + * callback of hotplug_bridge_dsi_ops. But "normal" bridges have their > + * downstream DSI device always connected, which we don't. When booting > + * without anything connected downstream, our upstream bridge could be not > + * even calling drm_bridge_add() until we do attach ourselves as a DSI > + * device, preventing the whole DRM card from being instantiated. > + * > + * In order to always have a DRM card after boot, we do call this same > + * function while probing in order to attach as a DSI device to the DSI > + * master. However during probe we don't know the bus format yet. It would > + * be nice to be able to update the format afterwards when a downstream DSI > + * device is attaching to our local host, but there is no callback for > + * that. To overcome this limitation, this function can be called in two > + * ways: > + * > + * - during probe, to make the upstream bridge happy, when there is no > + * next_dsi_dev yet and thus the lanes/format/etc are unknown > + * - as the mipi_dsi_host_ops.attach callback proper, as soon as the > + * next_dsi_dev is known > + * > + * The resulting call sequence is: > + * > + * 1. hotplug_bridge_dsi_attach() called by hotplug_bridge_probe() with > + * next_dsi_dev == NULL: we attach to the host but with a fake format > + * so the DRM card can be populated. hpb->dsi_dev becomes non-NULL. > + * 2. hotplug_bridge_dsi_attach() called as .attach callback from a > + * downstream device when it becomes available: we need to detach in > + * order to re-attach with the format of the device. hpb->dsi_dev > + * is found non-NULL, then reused so it will be non-NULL again. > + * 3. hotplug_bridge_dsi_detach() called as the .detach callback by a > + * downstream device: cleans up everything normally. hpb->dsi_dev goes > + * from non-NULL to NULL. > + * 4. hotplug_bridge_dsi_attach() called by a downstream device: attaches > + * normally to the upstream DSI host. hpb->dsi_dev goes from NULL to > + * non-NULL. > + * > + * Steps 3 and 4 are the "normal" attach/detach steps as on "normal" > + * bridges. > + * > + * Steps 1 and 2 happen only the first time, steps 3 and 4 will happen > + * every time the downstream bridge disconnects and reconnects. > + */ > +static int hotplug_bridge_dsi_attach(struct mipi_dsi_host *host, > + struct mipi_dsi_device *next_dsi_dev) > +{ > + struct device *dev = host->dev; > + struct hotplug_bridge *hpb = dev_get_drvdata(dev); > + struct mipi_dsi_device *dsi_dev; > + const struct mipi_dsi_device_info dsi_info = { > + .type = "hotplug-bridge", > + .channel = 0, > + .node = NULL, > + }; > + int err; > + > + /* > + * Step 2 only (first time we are called for an actual device > + * attaching): clean up the fake attach done at step 1 > + */ > + if (hpb->dsi_dev) > + hotplug_bridge_dsi_detach(&hpb->dsi_host, NULL); > + > + /* Register a local DSI device with the remote DSI host */ > + dsi_dev = mipi_dsi_device_register_full(hpb->prev_dsi_host, > + &dsi_info); > + if (IS_ERR(dsi_dev)) > + return PTR_ERR(dsi_dev); > + > + /* At step 1 we have no downstream device to get the format from */ > + if (next_dsi_dev) { > + dsi_dev->channel = next_dsi_dev->channel; > + dsi_dev->lanes = next_dsi_dev->lanes; > + dsi_dev->format = next_dsi_dev->format; > + dsi_dev->mode_flags = next_dsi_dev->mode_flags; > + } > + > + /* Attach our local DSI device to the remote DSI host */ > + err = mipi_dsi_attach(dsi_dev); > + if (err) { > + mipi_dsi_device_unregister(dsi_dev); > + return dev_err_probe(dev, err, "failed to attach hotplug dsi device to host\n"); > + } > + > + hpb->dsi_dev = dsi_dev; > + > + return 0; > +} > + > +/* > + * Propagate mipi_dsi_device_transfer() to the upstream DSI host. > + * > + * Reimplements identically the minimal needed part of > + * mipi_dsi_device_transfer(), including the -ENOSYS return value. > + */ > +static ssize_t hotplug_bridge_dsi_transfer(struct mipi_dsi_host *host, > + const struct mipi_dsi_msg *msg) > +{ > + struct hotplug_bridge *hpb = dev_get_drvdata(host->dev); > + const struct mipi_dsi_host_ops *ops; > + > + if (!hpb->dsi_dev) > + return -ENODEV; > + > + ops = hpb->dsi_dev->host->ops; > + > + if (!ops || !ops->transfer) > + return -ENOSYS; > + > + return ops->transfer(hpb->dsi_dev->host, msg); > +} > + > +static const struct mipi_dsi_host_ops hotplug_bridge_dsi_ops = { > + .attach = hotplug_bridge_dsi_attach, > + .detach = hotplug_bridge_dsi_detach, > + .transfer = hotplug_bridge_dsi_transfer, > +}; > + > +/* > + * Find the upstream DSI host and register our downstream-facing DSI host. > + */ > +static int hotplug_bridge_dsi_setup(struct hotplug_bridge *hpb) > +{ > + struct device *dev = hpb->dev; > + struct device_node *endpoint; > + struct device_node *node; > + > + endpoint = of_graph_get_endpoint_by_regs(dev->of_node, 0, -1); > + node = of_graph_get_remote_port_parent(endpoint); > + > + hpb->prev_dsi_host = of_find_mipi_dsi_host_by_node(node); > + > + of_node_put(node); > + of_node_put(endpoint); > + > + if (!hpb->prev_dsi_host) > + return -EPROBE_DEFER; > + > + hpb->dsi_host.dev = dev; > + hpb->dsi_host.ops = &hotplug_bridge_dsi_ops; > + > + return mipi_dsi_host_register(&hpb->dsi_host); > +} > + > +static void hotplug_bridge_dsi_cleanup(struct hotplug_bridge *hpb) > +{ > + mipi_dsi_host_unregister(&hpb->dsi_host); > +} > + > +static int hotplug_bridge_probe(struct platform_device *pdev) > +{ > + struct device *dev = &pdev->dev; > + struct hotplug_bridge *hpb; > + struct drm_bridge *bridge; > + int err; > + > + hpb = kzalloc(sizeof(*hpb), GFP_KERNEL); > + if (!hpb) > + return -ENOMEM; > + > + err = drm_bridge_init(dev, &hpb->bridge, &hotplug_bridge_funcs); > + if (err) > + return err; > + > + hpb->dev = dev; > + > + mutex_init(&hpb->next_bridge_mutex); > + INIT_WORK(&hpb->hpd_work, hotplug_bridge_hpd_work_func); > + > + err = hotplug_bridge_dsi_setup(hpb); > + if (err) > + return dev_err_probe(dev, err, "failed to setup DSI\n"); > + > + bridge = &hpb->bridge; > + bridge->of_node = dev->of_node; > + bridge->type = DRM_MODE_CONNECTOR_DSI; > + bridge->ops |= DRM_BRIDGE_OP_DETECT | DRM_BRIDGE_OP_HPD; > + > + platform_set_drvdata(pdev, hpb); > + > + err = devm_drm_bridge_add(dev, bridge); > + if (err) { > + dev_err_probe(dev, err, "failed adding bridge\n"); > + goto err_dsi_cleanup; > + } > + > + err = hotplug_bridge_dsi_attach(&hpb->dsi_host, NULL); > + if (err) { > + dev_err_probe(dev, err, "failed first attach to upstream DSI host\n"); > + goto err_dsi_cleanup; > + } > + > + /* > + * Since devm_drm_bridge_add() we can be notified of any bridges > + * appearing, but also check now, in case the next bridge was > + * probed earlier > + */ > + hotplug_bridge_grab(hpb); > + > + return 0; > + > +err_dsi_cleanup: > + hotplug_bridge_dsi_cleanup(hpb); > + return err; > +} > + > +static void hotplug_bridge_remove(struct platform_device *pdev) > +{ > + struct hotplug_bridge *hpb = platform_get_drvdata(pdev); > + > + cancel_work_sync(&hpb->hpd_work); > + > + hotplug_bridge_release(hpb, NULL); > + > + hotplug_bridge_dsi_cleanup(hpb); > +} > + > +static const struct platform_device_id hotplug_bridge_platform_ids[] = { > + { .name = "hotplug-dsi-bridge" }, > + {}, > +}; > +MODULE_DEVICE_TABLE(platform, hotplug_bridge_platform_ids); > + > +static struct platform_driver hotplug_bridge_driver = { > + .probe = hotplug_bridge_probe, > + .remove = hotplug_bridge_remove, > + .id_table = hotplug_bridge_platform_ids, > + .driver = { > + .name = "hotplug-drm-bridge", > + }, > +}; > + > +module_platform_driver(hotplug_bridge_driver); > + > +MODULE_AUTHOR("Luca Ceresoli <luca.ceresoli@bootlin.com>"); > +MODULE_AUTHOR("Paul Kocialkowski <paul.kocialkowski@bootlin.com>"); > +MODULE_DESCRIPTION("Hotplug DRM Bridge"); > +MODULE_LICENSE("GPL"); > > -- > 2.34.1 >
Hi Dmitry, On Tue, 31 Dec 2024 17:29:52 +0200 Dmitry Baryshkov <dmitry.baryshkov@linaro.org> wrote: > On Tue, Dec 31, 2024 at 11:40:04AM +0100, Luca Ceresoli wrote: > > This driver implements the point of a DRM pipeline where a connector allows > > removal of all the following bridges up to the panel. > > > > The DRM subsystem currently allows hotplug of the monitor but not preceding > > components. However there are embedded devices where the "tail" of the DRM > > pipeline, including one or more bridges, can be physically removed: > > > > .------------------------. > > | DISPLAY CONTROLLER | > > | .---------. .------. | > > | | ENCODER |<--| CRTC | | > > | '---------' '------' | > > '------|-----------------' > > | > > | HOTPLUG > > V CONNECTOR > > .---------. .--. .-. .---------. .-------. > > | 0 to N | | _| _| | | 1 to N | | | > > | BRIDGES |--DSI-->||_ |_ |--DSI-->| BRIDGES |--LVDS-->| PANEL | > > | | | | | | | | | | > > '---------' '--' '-' '---------' '-------' > > > > [--- fixed components --] [----------- removable add-on -----------] > > > > This driver supports such a device, where the final segment of a MIPI DSI > > bus, including one or more bridges, can be physically disconnected and > > reconnected at runtime, possibly with a different model. > > > > The add-on supported by this driver has a MIPI DSI bus traversing the > > hotplug connector and a DSI to LVDS bridge and an LVDS panel on the add-on. > > Hovever this driver is designed to be as far as possible generic and > > extendable to other busses that have no native hotplug and model ID > > discovery. > > > > This driver does not itself add and remove the bridges or panel on the > > add-on: this needs to be done by other means, e.g. device tree overlay > > runtime insertion and removal. The hotplug-bridge gets notified by the DRM > > bridge core after a removable bridge gets added or before it is removed. > > > > The hotplug-bridge role is to implement the "hot-pluggable connector" in > > the bridge chain. In this position, what the hotplug-bridge should ideally > > do is: > > > > * communicate with the previous component (bridge or encoder) so that it > > believes it always has a connected bridge following it and the DRM card > > is always present > > * be notified of the addition and removal of the following bridge and > > attach/detach to/from it > > * communicate with the following bridge so that it will attach and detach > > using the normal procedure (as if the entire pipeline were being created > > or destroyed, not only the tail) > > * instantiate two DRM connectors (similarly to what the DisplayPort MST > > code does): > > - a DSI connector representing the video lines of the hotplug connector; > > the status is always "disconnected" (no panel is ever attached > > directly to it) > > - an LSVD connector representing the classic connection to the panel; > > this gets added/removed whenever the add-on gets > > connected/disconnected; the status is always "connected" as the panel > > is always connected to the preceding bridge > > I'd rather have just a single connector. MST connectors can be added and > gone as there is fit, so should be your LVDS panel-related connector. The plan we discussed at LPC 2024 is to eventually get rid of the first connector (see "Roadmap and current status" in the cover letter), so you can consider this legacy code. However the current implementation won't work without this connector, so it is still there for the time being. Pointing this out in a note in the commit message of this patch would probably be useful to avoid future misunderstanding, so I'm adding one for v6. Thanks for reviewing! Luca
diff --git a/MAINTAINERS b/MAINTAINERS index 1675e44799e55d72711a6251f692a4a14bc3a84a..ab255f5696880258deee55d99ff0a7cde85efb7c 100644 --- a/MAINTAINERS +++ b/MAINTAINERS @@ -7182,6 +7182,11 @@ T: git https://gitlab.freedesktop.org/drm/misc/kernel.git F: Documentation/devicetree/bindings/display/panel/himax,hx8394.yaml F: drivers/gpu/drm/panel/panel-himax-hx8394.c +DRM DRIVER FOR HOTPLUG VIDEO CONNECTOR BRIDGE +M: Luca Ceresoli <luca.ceresoli@bootlin.com> +S: Maintained +F: drivers/gpu/drm/bridge/hotplug-bridge.c + DRM DRIVER FOR HX8357D PANELS S: Orphan T: git https://gitlab.freedesktop.org/drm/misc/kernel.git diff --git a/drivers/gpu/drm/bridge/Kconfig b/drivers/gpu/drm/bridge/Kconfig index 6b4664d91faa80f096ac6a0548ed342e802ae68b..f01971638d6818e33b32217922e165a8c18d51ee 100644 --- a/drivers/gpu/drm/bridge/Kconfig +++ b/drivers/gpu/drm/bridge/Kconfig @@ -90,6 +90,23 @@ config DRM_FSL_LDB help Support for i.MX8MP DPI-to-LVDS on-SoC encoder. +config DRM_HOTPLUG_BRIDGE + tristate "Hotplug DRM bridge support" + depends on OF + select DRM_PANEL_BRIDGE + select DRM_MIPI_DSI + select DRM_KMS_HELPER + select DRM_DISPLAY_HELPER + select DRM_BRIDGE_CONNECTOR + help + Driver for a DRM bridge representing a physical connector that + splits a DRM pipeline into a fixed part and a physically + removable part. The fixed part includes up to the encoder and + zero or more bridges. The removable part includes any following + bridges up to the connector and panel and can be physically + removed and connected at runtime, possibly with different + components. + config DRM_ITE_IT6263 tristate "ITE IT6263 LVDS/HDMI bridge" depends on OF diff --git a/drivers/gpu/drm/bridge/Makefile b/drivers/gpu/drm/bridge/Makefile index 97304b429a530c108dcbff906965cda091b0a7a2..2f6ae1a97d15045316ee191c04dbc65650162bab 100644 --- a/drivers/gpu/drm/bridge/Makefile +++ b/drivers/gpu/drm/bridge/Makefile @@ -6,6 +6,7 @@ obj-$(CONFIG_DRM_CHRONTEL_CH7033) += chrontel-ch7033.o obj-$(CONFIG_DRM_CROS_EC_ANX7688) += cros-ec-anx7688.o obj-$(CONFIG_DRM_DISPLAY_CONNECTOR) += display-connector.o obj-$(CONFIG_DRM_FSL_LDB) += fsl-ldb.o +obj-$(CONFIG_DRM_HOTPLUG_BRIDGE) += hotplug-bridge.o obj-$(CONFIG_DRM_ITE_IT6263) += ite-it6263.o obj-$(CONFIG_DRM_ITE_IT6505) += ite-it6505.o obj-$(CONFIG_DRM_LONTIUM_LT8912B) += lontium-lt8912b.o diff --git a/drivers/gpu/drm/bridge/hotplug-bridge.c b/drivers/gpu/drm/bridge/hotplug-bridge.c new file mode 100644 index 0000000000000000000000000000000000000000..f717206287fc598cf7a3c5ac5bf9e1be4c8540d9 --- /dev/null +++ b/drivers/gpu/drm/bridge/hotplug-bridge.c @@ -0,0 +1,695 @@ +// SPDX-License-Identifier: GPL-2.0 +/* + * A DRM bridge representing the split point between a fixed part of the + * DRM pipeline and a physically removable part. The fixed part includes up + * to the encoder and zero or more bridges. Insertion and removal of the + * "downstream" components happens via device driver probe/removal. + * + * Copyright (C) 2024, GE HealthCare + * + * Authors: + * Luca Ceresoli <luca.ceresoli@bootlin.com> + * Paul Kocialkowski <paul.kocialkowski@bootlin.com> + */ + +#include <linux/mutex.h> +#include <linux/of.h> +#include <linux/of_graph.h> +#include <linux/platform_device.h> + +#include <drm/drm_atomic_helper.h> +#include <drm/drm_bridge.h> +#include <drm/drm_bridge_connector.h> +#include <drm/drm_mipi_dsi.h> +#include <drm/drm_of.h> +#include <drm/drm_probe_helper.h> + +/* + * Internal hotplug-bridge data. + * + * We have two 'struct drm_connector' here: + * - fixconn represents the DSI video lines on the hotplug connector where + * the removable part attaches. It is thus always instantiated. + * - dynconn represents the LVDS video lines where the panel is attached. + * It is part of the removable part of the video pipeline and as such is + * added and removed dynamically based on when the downstream devices + * appear and disappear. + */ +struct hotplug_bridge { + struct device *dev; + + /* Local bridge */ + struct drm_bridge bridge; + + /* Always-present connector (where the removal part will connect to) */ + struct drm_connector *fixconn; + + /* Downstream bridge (next in the chain) */ + struct drm_bridge *next_bridge; + /* Protect next_bridge */ + struct mutex next_bridge_mutex; + + /* Pointer to the last bridge exposing OP_MODES */ + struct drm_bridge *bridge_modes; + + /* The "tail" connector that gets added/removed at runtime */ + struct drm_connector dynconn; + + /* Local DSI host, for the downstream DSI device to attach to */ + struct mipi_dsi_host dsi_host; + /* Local DSI device, attached to the upstream DSI host */ + struct mipi_dsi_device *dsi_dev; + /* Upstream DSI host (the actual DSI controller) */ + struct mipi_dsi_host *prev_dsi_host; + + struct work_struct hpd_work; +}; + +static struct hotplug_bridge *hotplug_bridge_from_drm_bridge(struct drm_bridge *bridge) +{ + return container_of(bridge, struct hotplug_bridge, bridge); +} + +/* -------------------------------------------------------------------------- + * dynconn implementation + */ +static struct hotplug_bridge *hotplug_bridge_from_dynconn(struct drm_connector *conn) +{ + return container_of(conn, struct hotplug_bridge, dynconn); +} + +static int hotplug_bridge_dynconn_get_modes(struct drm_connector *connector) +{ + struct hotplug_bridge *hpb = hotplug_bridge_from_dynconn(connector); + + if (hpb->bridge_modes) + return hpb->bridge_modes->funcs->get_modes(hpb->bridge_modes, connector); + + return 0; +} + +static const struct drm_connector_helper_funcs hotplug_bridge_dynconn_connector_helper_funcs = { + .get_modes = hotplug_bridge_dynconn_get_modes, +}; + +static void hotplug_bridge_dynconn_destroy(struct drm_connector *connector) +{ + drm_connector_unregister(connector); + drm_connector_cleanup(connector); +} + +static const struct drm_connector_funcs dynconn_funcs = { + .destroy = hotplug_bridge_dynconn_destroy, + .atomic_duplicate_state = drm_atomic_helper_connector_duplicate_state, + .atomic_destroy_state = drm_atomic_helper_connector_destroy_state, + .fill_modes = drm_helper_probe_single_connector_modes, +}; + +/* + * In non-removable pipelines using a "bridge connector", + * drm_bridge_connector_init() stores in the bridge_connector a pointer to + * the last bridge having OP_MODES (typically the panel bridge), so the + * .get_modes op will automatically be called on that bridge (it also takes + * pointers to other bridges which we don't care about). The "bridge + * connector" is too restrictive for our use case so we cannot use it. But + * we need a pointer to the modes-providing bridge, so we need to replicate + * that bit of its logic. + * + * If no modes bridge is found, nothing is done. This is allowed. + * + * Also get the modes bridge to tie its lifetime to ours. + */ +static void hotplug_bridge_dynconn_bridge_modes_get(struct hotplug_bridge *hpb) +{ + struct drm_bridge *bridge; + + if (WARN_ON(!hpb->next_bridge || !hpb->bridge.encoder)) + return; + + drm_for_each_bridge_in_chain(hpb->bridge.encoder, bridge) + if (bridge->ops & DRM_BRIDGE_OP_MODES) { + drm_bridge_get(bridge); + hpb->bridge_modes = bridge; + } +} + +static void hotplug_bridge_dynconn_bridge_modes_put(struct hotplug_bridge *hpb) +{ + if (hpb->bridge_modes) + drm_bridge_put_and_clear(hpb->bridge_modes); +} + +static int hotplug_bridge_dynconn_add(struct hotplug_bridge *hpb) +{ + int err; + + err = drm_connector_init(hpb->bridge.dev, &hpb->dynconn, &dynconn_funcs, + DRM_MODE_CONNECTOR_LVDS); + if (err) + return err; + + drm_atomic_helper_connector_reset(&hpb->dynconn); + + drm_connector_helper_add(&hpb->dynconn, + &hotplug_bridge_dynconn_connector_helper_funcs); + + drm_connector_attach_encoder(&hpb->dynconn, hpb->bridge.encoder); + if (err) + goto err_cleanup; + + hotplug_bridge_dynconn_bridge_modes_get(hpb); + + err = drm_connector_register(&hpb->dynconn); + if (err) + goto err_cleanup; + + return 0; + +err_cleanup: + drm_connector_cleanup(&hpb->dynconn); + hotplug_bridge_dynconn_bridge_modes_put(hpb); + return err; +} + +/* ----------------------------------------------------------------------- */ + +/* + * Attach the remote bridge to the encoder and to the next bridge in the + * chain, if possible. For this to succeed, we need to know: + * + * - the encoder, which is set at the first drm_bridge_attach() time + * - the next bridge, which is obtained via a notifier whenever the next + * bridge is (re)probed, or at probe time in case it was probed before us + * + * In order to handle different execution sequences, this function can be + * called from multiple places and needs to check all the prerequisites + * every time, and it will act only if both are met. + * + * Must be called with hpb->next_bridge_mutex held. + * + * Returns 0 if the encoder was attached successfully, -ENODEV if any of + * the two prerequisites above is not met (no encoder or no next bridge), + * the error returned by drm_bridge_attach() otherwise. + */ +static int hotplug_bridge_attach_to_encoder_chain(struct hotplug_bridge *hpb) +{ + int ret; + + if (!hpb->next_bridge || !hpb->bridge.encoder) + return -ENODEV; + + ret = drm_bridge_attach(hpb->bridge.encoder, hpb->next_bridge, &hpb->bridge, + DRM_BRIDGE_ATTACH_NO_CONNECTOR); + if (ret) + return dev_err_probe(hpb->dev, ret, "drm_bridge_attach failed\n"); + + dev_dbg(hpb->dev, "attached to encoder chain\n"); + + return 0; +} + +/* + * Stop the video pipeline and detach next_bridge. + * + * Must be called with hpb->next_bridge_mutex held. + */ +static void hotplug_bridge_detach_from_encoder_chain(struct hotplug_bridge *hpb) +{ + WARN_ON_ONCE(!hpb->next_bridge); + + dev_dbg(hpb->dev, "detaching from encoder chain\n"); + + drm_atomic_helper_shutdown(hpb->bridge.dev); + + drm_encoder_cleanup_from(hpb->bridge.encoder, hpb->next_bridge); +} + +static void hotplug_bridge_grab(struct hotplug_bridge *hpb) +{ + struct device *dev = hpb->dev; + struct drm_bridge *bridge; + struct drm_panel *panel; + int err; + + mutex_lock(&hpb->next_bridge_mutex); + + if (hpb->next_bridge) + goto out_unlock; + + /* + * This is supposed to be replaced by devm_drm_of_get_bridge(), but + * that is a devm_, and we need to remove the panel bridge also on + * next_bridge disconnect. + */ + err = drm_of_find_panel_or_bridge(dev->of_node, 1, 0, &panel, &bridge); + if (err) + goto out_unlock; + + /* Convert the remote panel to a bridge */ + if (panel) + bridge = drm_panel_bridge_add(panel); + if (IS_ERR(bridge)) + goto out_unlock; + + drm_bridge_get(bridge); + hpb->next_bridge = bridge; + + dev_dbg(dev, "grabbed next bridge (%pOFn)\n", hpb->next_bridge->of_node); + + hpb->bridge.pre_enable_prev_first = hpb->next_bridge->pre_enable_prev_first; + + err = hotplug_bridge_attach_to_encoder_chain(hpb); + if (err) + goto err_panel_bridge_remove; + + err = hotplug_bridge_dynconn_add(hpb); + if (err) + goto err_detach_from_encoder_chain; + + queue_work(system_wq, &hpb->hpd_work); + goto out_unlock; + +err_detach_from_encoder_chain: + hotplug_bridge_detach_from_encoder_chain(hpb); +err_panel_bridge_remove: + drm_panel_bridge_remove(hpb->next_bridge); + drm_bridge_put_and_clear(hpb->next_bridge); +out_unlock: + mutex_unlock(&hpb->next_bridge_mutex); +} + +/* + * Detach from the next bridge and remove the panel bridge, either on + * release or when the downstream bridge is being removed. + * + * Can be called in these ways: + * + * - bridge_being_removed is NULL: detach unconditionally + * (this is useful on .remove() to teardown everything) + * - bridge_being_removed == hpb->next_bridge: detach + * (the downstream bridge is being removed) + * - bridge_being_removed != hpb->next_bridge: do nothing + * (the bridge being removed is not the downstream bridge) + * + * In all cases, does nothing when there is no downstream bridge. + */ +static void hotplug_bridge_release(struct hotplug_bridge *hpb, + struct drm_bridge *bridge_being_removed) +{ + mutex_lock(&hpb->next_bridge_mutex); + + if (!hpb->next_bridge) + goto out; + + if (bridge_being_removed && bridge_being_removed != hpb->next_bridge) + goto out; + + if (hpb->bridge_modes) + hotplug_bridge_dynconn_bridge_modes_put(hpb); + + dev_dbg(hpb->dev, "releasing next bridge (%pOFn)\n", hpb->next_bridge->of_node); + hotplug_bridge_detach_from_encoder_chain(hpb); + + dev_dbg(hpb->dev, "removing %s connector\n", hpb->dynconn.name); + drm_connector_put(&hpb->dynconn); + + /* + * This will check that the bridge actually belongs to panel-bridge + * before doing anything with it, so we can safely always call it. + */ + drm_panel_bridge_remove(hpb->next_bridge); + drm_bridge_put_and_clear(hpb->next_bridge); + + queue_work(system_wq, &hpb->hpd_work); + +out: + mutex_unlock(&hpb->next_bridge_mutex); +} + +static void hotplug_bridge_bridge_event_notify(struct drm_bridge *bridge, + enum drm_bridge_event_type event, + struct drm_bridge *event_bridge) +{ + struct hotplug_bridge *hpb = container_of(bridge, struct hotplug_bridge, bridge); + + switch (event) { + case DRM_EVENT_BRIDGE_ADD: + hotplug_bridge_grab(hpb); + break; + case DRM_EVENT_BRIDGE_REMOVE: + hotplug_bridge_release(hpb, event_bridge); + break; + } +} + +static int hotplug_bridge_attach(struct drm_bridge *bridge, + enum drm_bridge_attach_flags flags) +{ + struct hotplug_bridge *hpb = hotplug_bridge_from_drm_bridge(bridge); + struct device *dev = hpb->dev; + struct drm_connector *connector; + struct drm_encoder *encoder = hpb->bridge.encoder; + int err; + + /* Encoder was not yet provided to our bridge */ + if (!encoder) + return -ENODEV; + + /* Connector was already created */ + if (hpb->fixconn) + return dev_err_probe(dev, -EBUSY, "connector already created\n"); + + connector = drm_bridge_connector_init(bridge->dev, encoder); + if (IS_ERR(connector)) + return dev_err_probe(dev, PTR_ERR(connector), "failed to initialize connector\n"); + + drm_connector_attach_encoder(connector, encoder); + + hpb->fixconn = connector; + + drm_connector_register(connector); + + mutex_lock(&hpb->next_bridge_mutex); + err = hotplug_bridge_attach_to_encoder_chain(hpb); + mutex_unlock(&hpb->next_bridge_mutex); + + /* -ENODEV is acceptable, in case next_bridge is not yet known */ + if (err == -ENODEV) + err = 0; + + /* + * If the encoder driver is probed last, the + * hotplug_bridge_attach_to_encoder_chain() call in + * hotplug_bridge_grab() fails because hpb->bridge.encoder is still + * NULL, and hotplug_bridge_grab() will not have another chance to + * execute. So call it now, at the end of the encoder attach + * process. + */ + hotplug_bridge_grab(hpb); + + return err; +} + +static void hotplug_bridge_detach(struct drm_bridge *bridge) +{ + struct hotplug_bridge *hpb = hotplug_bridge_from_drm_bridge(bridge); + + mutex_lock(&hpb->next_bridge_mutex); + hotplug_bridge_detach_from_encoder_chain(hpb); + mutex_unlock(&hpb->next_bridge_mutex); + + if (hpb->fixconn) { + drm_connector_unregister(hpb->fixconn); + drm_connector_cleanup(hpb->fixconn); + hpb->fixconn = NULL; + } +} + +static void hotplug_bridge_destroy(struct drm_bridge *bridge) +{ + struct hotplug_bridge *hpb = hotplug_bridge_from_drm_bridge(bridge); + + kfree(hpb); +} + +/* + * The fixed connector is never attached to a panel, so it should always be + * reported as disconnected. + */ +static enum drm_connector_status hotplug_bridge_detect(struct drm_bridge *bridge) +{ + return connector_status_disconnected; +} + +static void hotplug_bridge_hpd_work_func(struct work_struct *work) +{ + struct hotplug_bridge *hpb = container_of(work, struct hotplug_bridge, hpd_work); + + if (hpb->bridge.dev) + drm_helper_hpd_irq_event(hpb->bridge.dev); +} + +static const struct drm_bridge_funcs hotplug_bridge_funcs = { + .attach = hotplug_bridge_attach, + .detach = hotplug_bridge_detach, + .destroy = hotplug_bridge_destroy, + .detect = hotplug_bridge_detect, + .bridge_event_notify = hotplug_bridge_bridge_event_notify, +}; + +static int hotplug_bridge_dsi_detach(struct mipi_dsi_host *host, + struct mipi_dsi_device *device_remote) +{ + struct hotplug_bridge *hpb = dev_get_drvdata(host->dev); + + if (!hpb->dsi_dev) + return -ENODEV; + + mipi_dsi_detach(hpb->dsi_dev); + mipi_dsi_device_unregister(hpb->dsi_dev); + hpb->dsi_dev = NULL; + + return 0; +} + +/* + * Attach the local DSI device to the upstream DSI host, possibly with a + * "null" format. + * + * In "normal" bridges this function should be _only_ used as the .attach + * callback of hotplug_bridge_dsi_ops. But "normal" bridges have their + * downstream DSI device always connected, which we don't. When booting + * without anything connected downstream, our upstream bridge could be not + * even calling drm_bridge_add() until we do attach ourselves as a DSI + * device, preventing the whole DRM card from being instantiated. + * + * In order to always have a DRM card after boot, we do call this same + * function while probing in order to attach as a DSI device to the DSI + * master. However during probe we don't know the bus format yet. It would + * be nice to be able to update the format afterwards when a downstream DSI + * device is attaching to our local host, but there is no callback for + * that. To overcome this limitation, this function can be called in two + * ways: + * + * - during probe, to make the upstream bridge happy, when there is no + * next_dsi_dev yet and thus the lanes/format/etc are unknown + * - as the mipi_dsi_host_ops.attach callback proper, as soon as the + * next_dsi_dev is known + * + * The resulting call sequence is: + * + * 1. hotplug_bridge_dsi_attach() called by hotplug_bridge_probe() with + * next_dsi_dev == NULL: we attach to the host but with a fake format + * so the DRM card can be populated. hpb->dsi_dev becomes non-NULL. + * 2. hotplug_bridge_dsi_attach() called as .attach callback from a + * downstream device when it becomes available: we need to detach in + * order to re-attach with the format of the device. hpb->dsi_dev + * is found non-NULL, then reused so it will be non-NULL again. + * 3. hotplug_bridge_dsi_detach() called as the .detach callback by a + * downstream device: cleans up everything normally. hpb->dsi_dev goes + * from non-NULL to NULL. + * 4. hotplug_bridge_dsi_attach() called by a downstream device: attaches + * normally to the upstream DSI host. hpb->dsi_dev goes from NULL to + * non-NULL. + * + * Steps 3 and 4 are the "normal" attach/detach steps as on "normal" + * bridges. + * + * Steps 1 and 2 happen only the first time, steps 3 and 4 will happen + * every time the downstream bridge disconnects and reconnects. + */ +static int hotplug_bridge_dsi_attach(struct mipi_dsi_host *host, + struct mipi_dsi_device *next_dsi_dev) +{ + struct device *dev = host->dev; + struct hotplug_bridge *hpb = dev_get_drvdata(dev); + struct mipi_dsi_device *dsi_dev; + const struct mipi_dsi_device_info dsi_info = { + .type = "hotplug-bridge", + .channel = 0, + .node = NULL, + }; + int err; + + /* + * Step 2 only (first time we are called for an actual device + * attaching): clean up the fake attach done at step 1 + */ + if (hpb->dsi_dev) + hotplug_bridge_dsi_detach(&hpb->dsi_host, NULL); + + /* Register a local DSI device with the remote DSI host */ + dsi_dev = mipi_dsi_device_register_full(hpb->prev_dsi_host, + &dsi_info); + if (IS_ERR(dsi_dev)) + return PTR_ERR(dsi_dev); + + /* At step 1 we have no downstream device to get the format from */ + if (next_dsi_dev) { + dsi_dev->channel = next_dsi_dev->channel; + dsi_dev->lanes = next_dsi_dev->lanes; + dsi_dev->format = next_dsi_dev->format; + dsi_dev->mode_flags = next_dsi_dev->mode_flags; + } + + /* Attach our local DSI device to the remote DSI host */ + err = mipi_dsi_attach(dsi_dev); + if (err) { + mipi_dsi_device_unregister(dsi_dev); + return dev_err_probe(dev, err, "failed to attach hotplug dsi device to host\n"); + } + + hpb->dsi_dev = dsi_dev; + + return 0; +} + +/* + * Propagate mipi_dsi_device_transfer() to the upstream DSI host. + * + * Reimplements identically the minimal needed part of + * mipi_dsi_device_transfer(), including the -ENOSYS return value. + */ +static ssize_t hotplug_bridge_dsi_transfer(struct mipi_dsi_host *host, + const struct mipi_dsi_msg *msg) +{ + struct hotplug_bridge *hpb = dev_get_drvdata(host->dev); + const struct mipi_dsi_host_ops *ops; + + if (!hpb->dsi_dev) + return -ENODEV; + + ops = hpb->dsi_dev->host->ops; + + if (!ops || !ops->transfer) + return -ENOSYS; + + return ops->transfer(hpb->dsi_dev->host, msg); +} + +static const struct mipi_dsi_host_ops hotplug_bridge_dsi_ops = { + .attach = hotplug_bridge_dsi_attach, + .detach = hotplug_bridge_dsi_detach, + .transfer = hotplug_bridge_dsi_transfer, +}; + +/* + * Find the upstream DSI host and register our downstream-facing DSI host. + */ +static int hotplug_bridge_dsi_setup(struct hotplug_bridge *hpb) +{ + struct device *dev = hpb->dev; + struct device_node *endpoint; + struct device_node *node; + + endpoint = of_graph_get_endpoint_by_regs(dev->of_node, 0, -1); + node = of_graph_get_remote_port_parent(endpoint); + + hpb->prev_dsi_host = of_find_mipi_dsi_host_by_node(node); + + of_node_put(node); + of_node_put(endpoint); + + if (!hpb->prev_dsi_host) + return -EPROBE_DEFER; + + hpb->dsi_host.dev = dev; + hpb->dsi_host.ops = &hotplug_bridge_dsi_ops; + + return mipi_dsi_host_register(&hpb->dsi_host); +} + +static void hotplug_bridge_dsi_cleanup(struct hotplug_bridge *hpb) +{ + mipi_dsi_host_unregister(&hpb->dsi_host); +} + +static int hotplug_bridge_probe(struct platform_device *pdev) +{ + struct device *dev = &pdev->dev; + struct hotplug_bridge *hpb; + struct drm_bridge *bridge; + int err; + + hpb = kzalloc(sizeof(*hpb), GFP_KERNEL); + if (!hpb) + return -ENOMEM; + + err = drm_bridge_init(dev, &hpb->bridge, &hotplug_bridge_funcs); + if (err) + return err; + + hpb->dev = dev; + + mutex_init(&hpb->next_bridge_mutex); + INIT_WORK(&hpb->hpd_work, hotplug_bridge_hpd_work_func); + + err = hotplug_bridge_dsi_setup(hpb); + if (err) + return dev_err_probe(dev, err, "failed to setup DSI\n"); + + bridge = &hpb->bridge; + bridge->of_node = dev->of_node; + bridge->type = DRM_MODE_CONNECTOR_DSI; + bridge->ops |= DRM_BRIDGE_OP_DETECT | DRM_BRIDGE_OP_HPD; + + platform_set_drvdata(pdev, hpb); + + err = devm_drm_bridge_add(dev, bridge); + if (err) { + dev_err_probe(dev, err, "failed adding bridge\n"); + goto err_dsi_cleanup; + } + + err = hotplug_bridge_dsi_attach(&hpb->dsi_host, NULL); + if (err) { + dev_err_probe(dev, err, "failed first attach to upstream DSI host\n"); + goto err_dsi_cleanup; + } + + /* + * Since devm_drm_bridge_add() we can be notified of any bridges + * appearing, but also check now, in case the next bridge was + * probed earlier + */ + hotplug_bridge_grab(hpb); + + return 0; + +err_dsi_cleanup: + hotplug_bridge_dsi_cleanup(hpb); + return err; +} + +static void hotplug_bridge_remove(struct platform_device *pdev) +{ + struct hotplug_bridge *hpb = platform_get_drvdata(pdev); + + cancel_work_sync(&hpb->hpd_work); + + hotplug_bridge_release(hpb, NULL); + + hotplug_bridge_dsi_cleanup(hpb); +} + +static const struct platform_device_id hotplug_bridge_platform_ids[] = { + { .name = "hotplug-dsi-bridge" }, + {}, +}; +MODULE_DEVICE_TABLE(platform, hotplug_bridge_platform_ids); + +static struct platform_driver hotplug_bridge_driver = { + .probe = hotplug_bridge_probe, + .remove = hotplug_bridge_remove, + .id_table = hotplug_bridge_platform_ids, + .driver = { + .name = "hotplug-drm-bridge", + }, +}; + +module_platform_driver(hotplug_bridge_driver); + +MODULE_AUTHOR("Luca Ceresoli <luca.ceresoli@bootlin.com>"); +MODULE_AUTHOR("Paul Kocialkowski <paul.kocialkowski@bootlin.com>"); +MODULE_DESCRIPTION("Hotplug DRM Bridge"); +MODULE_LICENSE("GPL");