mirror of
git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-09-04 20:19:47 +08:00

The drm_bridge structure contains an encoder pointer that is widely used by bridge drivers. This pattern is largely documented as deprecated in other KMS entities for atomic drivers. However, one of the main use of that pointer is done in attach to just call drm_bridge_attach on the next bridge to add it to the bridge list. While this dereferences the bridge->encoder pointer, it's effectively the same encoder the bridge was being attached to. We can make it more explicit by adding the encoder the bridge is attached to to the list of attach parameters. This also removes the need to dereference bridge->encoder in most drivers. Reviewed-by: Dmitry Baryshkov <dmitry.baryshkov@linaro.org> Reviewed-by: Douglas Anderson <dianders@chromium.org> Tested-by: Douglas Anderson <dianders@chromium.org> Tested-by: Luca Ceresoli <luca.ceresoli@bootlin.com> Reviewed-by: Luca Ceresoli <luca.ceresoli@bootlin.com> Link: https://patchwork.freedesktop.org/patch/msgid/20250313-bridge-connector-v6-1-511c54a604fb@kernel.org Signed-off-by: Maxime Ripard <mripard@kernel.org>
132 lines
3.7 KiB
C
132 lines
3.7 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/*
|
|
* Copyright (C) 2018 Texas Instruments Incorporated - https://www.ti.com/
|
|
* Author: Tomi Valkeinen <tomi.valkeinen@ti.com>
|
|
*/
|
|
|
|
#include <linux/export.h>
|
|
|
|
#include <drm/drm_atomic_helper.h>
|
|
#include <drm/drm_bridge.h>
|
|
#include <drm/drm_bridge_connector.h>
|
|
#include <drm/drm_crtc.h>
|
|
#include <drm/drm_modeset_helper_vtables.h>
|
|
#include <drm/drm_panel.h>
|
|
#include <drm/drm_of.h>
|
|
#include <drm/drm_simple_kms_helper.h>
|
|
|
|
#include "tidss_crtc.h"
|
|
#include "tidss_drv.h"
|
|
#include "tidss_encoder.h"
|
|
|
|
struct tidss_encoder {
|
|
struct drm_bridge bridge;
|
|
struct drm_encoder encoder;
|
|
struct drm_connector *connector;
|
|
struct drm_bridge *next_bridge;
|
|
struct tidss_device *tidss;
|
|
};
|
|
|
|
static inline struct tidss_encoder
|
|
*bridge_to_tidss_encoder(struct drm_bridge *b)
|
|
{
|
|
return container_of(b, struct tidss_encoder, bridge);
|
|
}
|
|
|
|
static int tidss_bridge_attach(struct drm_bridge *bridge,
|
|
struct drm_encoder *encoder,
|
|
enum drm_bridge_attach_flags flags)
|
|
{
|
|
struct tidss_encoder *t_enc = bridge_to_tidss_encoder(bridge);
|
|
|
|
return drm_bridge_attach(encoder, t_enc->next_bridge,
|
|
bridge, flags);
|
|
}
|
|
|
|
static int tidss_bridge_atomic_check(struct drm_bridge *bridge,
|
|
struct drm_bridge_state *bridge_state,
|
|
struct drm_crtc_state *crtc_state,
|
|
struct drm_connector_state *conn_state)
|
|
{
|
|
struct tidss_encoder *t_enc = bridge_to_tidss_encoder(bridge);
|
|
struct tidss_device *tidss = t_enc->tidss;
|
|
struct tidss_crtc_state *tcrtc_state = to_tidss_crtc_state(crtc_state);
|
|
struct drm_display_info *di = &conn_state->connector->display_info;
|
|
struct drm_bridge_state *next_bridge_state = NULL;
|
|
|
|
if (t_enc->next_bridge)
|
|
next_bridge_state = drm_atomic_get_new_bridge_state(crtc_state->state,
|
|
t_enc->next_bridge);
|
|
|
|
if (next_bridge_state) {
|
|
tcrtc_state->bus_flags = next_bridge_state->input_bus_cfg.flags;
|
|
tcrtc_state->bus_format = next_bridge_state->input_bus_cfg.format;
|
|
} else if (di->num_bus_formats) {
|
|
tcrtc_state->bus_format = di->bus_formats[0];
|
|
tcrtc_state->bus_flags = di->bus_flags;
|
|
} else {
|
|
dev_err(tidss->dev, "%s: No bus_formats in connected display\n",
|
|
__func__);
|
|
return -EINVAL;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct drm_bridge_funcs tidss_bridge_funcs = {
|
|
.attach = tidss_bridge_attach,
|
|
.atomic_check = tidss_bridge_atomic_check,
|
|
.atomic_reset = drm_atomic_helper_bridge_reset,
|
|
.atomic_duplicate_state = drm_atomic_helper_bridge_duplicate_state,
|
|
.atomic_destroy_state = drm_atomic_helper_bridge_destroy_state,
|
|
};
|
|
|
|
int tidss_encoder_create(struct tidss_device *tidss,
|
|
struct drm_bridge *next_bridge,
|
|
u32 encoder_type, u32 possible_crtcs)
|
|
{
|
|
struct tidss_encoder *t_enc;
|
|
struct drm_encoder *enc;
|
|
struct drm_connector *connector;
|
|
int ret;
|
|
|
|
t_enc = drmm_simple_encoder_alloc(&tidss->ddev, struct tidss_encoder,
|
|
encoder, encoder_type);
|
|
if (IS_ERR(t_enc))
|
|
return PTR_ERR(t_enc);
|
|
|
|
t_enc->tidss = tidss;
|
|
t_enc->next_bridge = next_bridge;
|
|
t_enc->bridge.funcs = &tidss_bridge_funcs;
|
|
|
|
enc = &t_enc->encoder;
|
|
enc->possible_crtcs = possible_crtcs;
|
|
|
|
/* Attaching first bridge to the encoder */
|
|
ret = drm_bridge_attach(enc, &t_enc->bridge, NULL,
|
|
DRM_BRIDGE_ATTACH_NO_CONNECTOR);
|
|
if (ret) {
|
|
dev_err(tidss->dev, "bridge attach failed: %d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
/* Initializing the connector at the end of bridge-chain */
|
|
connector = drm_bridge_connector_init(&tidss->ddev, enc);
|
|
if (IS_ERR(connector)) {
|
|
dev_err(tidss->dev, "bridge_connector create failed\n");
|
|
return PTR_ERR(connector);
|
|
}
|
|
|
|
ret = drm_connector_attach_encoder(connector, enc);
|
|
if (ret) {
|
|
dev_err(tidss->dev, "attaching encoder to connector failed\n");
|
|
return ret;
|
|
}
|
|
|
|
t_enc->connector = connector;
|
|
|
|
dev_dbg(tidss->dev, "Encoder create done\n");
|
|
|
|
return ret;
|
|
}
|