Hi Andy, Am Mittwoch, 3. September 2025, 13:07:38 Mitteleuropäische Sommerzeit schrieb Andy Yan: > From: Andy Yan <andy....@rock-chips.com> > > Convert it to drm bridge driver, it will be convenient for us to > migrate the connector part to the display driver later. > > Signed-off-by: Andy Yan <andy....@rock-chips.com>
more like a general remark, this essentially conflicts with the big hiword-cleanup [0] that was merged today, as the inno-hdmi driver "lost" its separate HIWORD_UPDATE macro in favor a nicer generic one. I'm not sure what the best way to proceed is, apart from waiting for 6.18-rc1. Heiko [0] https://lore.kernel.org/linux-rockchip/20250825-byeword-update-v3-0-947b841cd...@collabora.com/ > --- > > Changes in v7: > - Rebase on latest drm-misc-next > > Changes in v6: > - Rebase on latest drm-misc-next > - Link to V5: > https://lore.kernel.org/linux-rockchip/20250512124615.2848731-1-andys...@163.com/ > > Changes in v5: > - Split cleanup code to seperate patch > - Switch to devm_drm_bridge_alloc() API > - Link to V4: > https://lore.kernel.org/linux-rockchip/20250422070455.432666-1-andys...@163.com/ > > Changes in v4: > - Do not store colorimetry within inno_hdmi struct > - Link to V3: > https://lore.kernel.org/linux-rockchip/20250402123150.238234-1-andys...@163.com/ > > Changes in v3: > - First included in v3 > - Link to V2: > https://lore.kernel.org/dri-devel/20250325132944.171111-1-andys...@163.com/ > > drivers/gpu/drm/bridge/Kconfig | 7 + > drivers/gpu/drm/bridge/Makefile | 1 + > .../inno_hdmi.c => bridge/inno-hdmi.c} | 502 +++++------------- > drivers/gpu/drm/rockchip/Kconfig | 1 + > drivers/gpu/drm/rockchip/Makefile | 2 +- > drivers/gpu/drm/rockchip/inno_hdmi-rockchip.c | 188 +++++++ > include/drm/bridge/inno_hdmi.h | 33 ++ > 7 files changed, 366 insertions(+), 368 deletions(-) > rename drivers/gpu/drm/{rockchip/inno_hdmi.c => bridge/inno-hdmi.c} (69%) > create mode 100644 drivers/gpu/drm/rockchip/inno_hdmi-rockchip.c > create mode 100644 include/drm/bridge/inno_hdmi.h > > diff --git a/drivers/gpu/drm/bridge/Kconfig b/drivers/gpu/drm/bridge/Kconfig > index 6945029b35929..f3d0503ee4c35 100644 > --- a/drivers/gpu/drm/bridge/Kconfig > +++ b/drivers/gpu/drm/bridge/Kconfig > @@ -100,6 +100,13 @@ config DRM_I2C_NXP_TDA998X > help > Support for NXP Semiconductors TDA998X HDMI encoders. > > +config DRM_INNO_HDMI > + tristate > + select DRM_BRIDGE_CONNECTOR > + select DRM_DISPLAY_HDMI_HELPER > + select DRM_DISPLAY_HELPER > + select DRM_KMS_HELPER > + > 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 c7dc03182e592..909c21cc3acd3 100644 > --- a/drivers/gpu/drm/bridge/Makefile > +++ b/drivers/gpu/drm/bridge/Makefile > @@ -10,6 +10,7 @@ obj-$(CONFIG_DRM_FSL_LDB) += fsl-ldb.o > tda998x-y := tda998x_drv.o > obj-$(CONFIG_DRM_I2C_NXP_TDA998X) += tda998x.o > > +obj-$(CONFIG_DRM_INNO_HDMI) += inno-hdmi.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/rockchip/inno_hdmi.c > b/drivers/gpu/drm/bridge/inno-hdmi.c > similarity index 69% > rename from drivers/gpu/drm/rockchip/inno_hdmi.c > rename to drivers/gpu/drm/bridge/inno-hdmi.c > index 1ab3ad4bde9ea..ab4572eb83950 100644 > --- a/drivers/gpu/drm/rockchip/inno_hdmi.c > +++ b/drivers/gpu/drm/bridge/inno-hdmi.c > @@ -3,12 +3,14 @@ > * Copyright (C) Rockchip Electronics Co., Ltd. > * Zheng Yang <zhengy...@rock-chips.com> > * Yakir Yang <y...@rock-chips.com> > + * Andy Yan <andys...@163.com> > */ > > #include <linux/irq.h> > #include <linux/clk.h> > #include <linux/delay.h> > #include <linux/err.h> > +#include <linux/i2c.h> > #include <linux/hdmi.h> > #include <linux/mfd/syscon.h> > #include <linux/mod_devicetable.h> > @@ -17,18 +19,18 @@ > #include <linux/platform_device.h> > #include <linux/regmap.h> > > +#include <drm/bridge/inno_hdmi.h> > #include <drm/drm_atomic.h> > #include <drm/drm_atomic_helper.h> > #include <drm/drm_edid.h> > #include <drm/drm_of.h> > +#include <drm/drm_print.h> > #include <drm/drm_probe_helper.h> > #include <drm/drm_simple_kms_helper.h> > > #include <drm/display/drm_hdmi_helper.h> > #include <drm/display/drm_hdmi_state_helper.h> > > -#include "rockchip_drm_drv.h" > - > #define INNO_HDMI_MIN_TMDS_CLOCK 25000000U > > #define DDC_SEGMENT_ADDR 0x30 > @@ -382,29 +384,6 @@ enum { > #define HDMI_CEC_BUSFREETIME_H 0xdd > #define HDMI_CEC_LOGICADDR 0xde > > -#define HIWORD_UPDATE(val, mask) ((val) | (mask) << 16) > - > -#define RK3036_GRF_SOC_CON2 0x148 > -#define RK3036_HDMI_PHSYNC BIT(4) > -#define RK3036_HDMI_PVSYNC BIT(5) > - > -enum inno_hdmi_dev_type { > - RK3036_HDMI, > - RK3128_HDMI, > -}; > - > -struct inno_hdmi_phy_config { > - unsigned long pixelclock; > - u8 pre_emphasis; > - u8 voltage_level_control; > -}; > - > -struct inno_hdmi_variant { > - enum inno_hdmi_dev_type dev_type; > - struct inno_hdmi_phy_config *phy_configs; > - struct inno_hdmi_phy_config *default_phy_config; > -}; > - > struct inno_hdmi_i2c { > struct i2c_adapter adap; > > @@ -417,41 +396,17 @@ struct inno_hdmi_i2c { > > struct inno_hdmi { > struct device *dev; > - > + struct drm_bridge bridge; > struct clk *pclk; > struct clk *refclk; > void __iomem *regs; > struct regmap *grf; > > - struct drm_connector connector; > - struct rockchip_encoder encoder; > - > struct inno_hdmi_i2c *i2c; > struct i2c_adapter *ddc; > - > - const struct inno_hdmi_variant *variant; > -}; > - > -struct inno_hdmi_connector_state { > - struct drm_connector_state base; > - unsigned int colorimetry; > + const struct inno_hdmi_plat_data *plat_data; > }; > > -static struct inno_hdmi *encoder_to_inno_hdmi(struct drm_encoder *encoder) > -{ > - struct rockchip_encoder *rkencoder = to_rockchip_encoder(encoder); > - > - return container_of(rkencoder, struct inno_hdmi, encoder); > -} > - > -static struct inno_hdmi *connector_to_inno_hdmi(struct drm_connector > *connector) > -{ > - return container_of(connector, struct inno_hdmi, connector); > -} > - > -#define to_inno_hdmi_conn_state(conn_state) \ > - container_of_const(conn_state, struct inno_hdmi_connector_state, base) > - > enum { > CSC_RGB_0_255_TO_ITU601_16_235_8BIT, > CSC_RGB_0_255_TO_ITU709_16_235_8BIT, > @@ -494,23 +449,15 @@ static const char coeff_csc[][24] = { > }, > }; > > -static struct inno_hdmi_phy_config rk3036_hdmi_phy_configs[] = { > - { 74250000, 0x3f, 0xbb }, > - { 165000000, 0x6f, 0xbb }, > - { ~0UL, 0x00, 0x00 } > -}; > - > -static struct inno_hdmi_phy_config rk3128_hdmi_phy_configs[] = { > - { 74250000, 0x3f, 0xaa }, > - { 165000000, 0x5f, 0xaa }, > - { ~0UL, 0x00, 0x00 } > -}; > +static struct inno_hdmi *bridge_to_inno_hdmi(struct drm_bridge *bridge) > +{ > + return container_of(bridge, struct inno_hdmi, bridge); > +} > > static int inno_hdmi_find_phy_config(struct inno_hdmi *hdmi, > unsigned long pixelclk) > { > - const struct inno_hdmi_phy_config *phy_configs = > - hdmi->variant->phy_configs; > + const struct inno_hdmi_phy_config *phy_configs = > hdmi->plat_data->phy_configs; > int i; > > for (i = 0; phy_configs[i].pixelclock != ~0UL; i++) { > @@ -582,12 +529,12 @@ static void inno_hdmi_power_up(struct inno_hdmi *hdmi, > int ret = inno_hdmi_find_phy_config(hdmi, mpixelclock); > > if (ret < 0) { > - phy_config = hdmi->variant->default_phy_config; > + phy_config = hdmi->plat_data->default_phy_config; > DRM_DEV_ERROR(hdmi->dev, > "Using default phy configuration for TMDS rate > %lu", > mpixelclock); > } else { > - phy_config = &hdmi->variant->phy_configs[ret]; > + phy_config = &hdmi->plat_data->phy_configs[ret]; > } > > inno_hdmi_sys_power(hdmi, false); > @@ -637,14 +584,13 @@ static void inno_hdmi_init_hw(struct inno_hdmi *hdmi) > hdmi_modb(hdmi, HDMI_STATUS, m_MASK_INT_HOTPLUG, v_MASK_INT_HOTPLUG(1)); > } > > -static int inno_hdmi_disable_frame(struct drm_connector *connector, > - enum hdmi_infoframe_type type) > +static int inno_hdmi_bridge_clear_infoframe(struct drm_bridge *bridge, > + enum hdmi_infoframe_type type) > { > - struct inno_hdmi *hdmi = connector_to_inno_hdmi(connector); > + struct inno_hdmi *hdmi = bridge_to_inno_hdmi(bridge); > > if (type != HDMI_INFOFRAME_TYPE_AVI) { > - drm_err(connector->dev, > - "Unsupported infoframe type: %u\n", type); > + drm_err(bridge->dev, "Unsupported infoframe type: %u\n", type); > return 0; > } > > @@ -653,20 +599,19 @@ static int inno_hdmi_disable_frame(struct drm_connector > *connector, > return 0; > } > > -static int inno_hdmi_upload_frame(struct drm_connector *connector, > - enum hdmi_infoframe_type type, > - const u8 *buffer, size_t len) > +static int inno_hdmi_bridge_write_infoframe(struct drm_bridge *bridge, > + enum hdmi_infoframe_type type, > + const u8 *buffer, size_t len) > { > - struct inno_hdmi *hdmi = connector_to_inno_hdmi(connector); > + struct inno_hdmi *hdmi = bridge_to_inno_hdmi(bridge); > ssize_t i; > > if (type != HDMI_INFOFRAME_TYPE_AVI) { > - drm_err(connector->dev, > - "Unsupported infoframe type: %u\n", type); > + drm_err(bridge->dev, "Unsupported infoframe type: %u\n", type); > return 0; > } > > - inno_hdmi_disable_frame(connector, type); > + inno_hdmi_bridge_clear_infoframe(bridge, type); > > for (i = 0; i < len; i++) > hdmi_writeb(hdmi, HDMI_CONTROL_PACKET_ADDR + i, buffer[i]); > @@ -674,23 +619,26 @@ static int inno_hdmi_upload_frame(struct drm_connector > *connector, > return 0; > } > > -static const struct drm_connector_hdmi_funcs inno_hdmi_hdmi_connector_funcs > = { > - .clear_infoframe = inno_hdmi_disable_frame, > - .write_infoframe = inno_hdmi_upload_frame, > -}; > - > -static int inno_hdmi_config_video_csc(struct inno_hdmi *hdmi) > +static int inno_hdmi_config_video_csc(struct inno_hdmi *hdmi, > + struct drm_connector *connector, > + struct drm_display_mode *mode) > { > - struct drm_connector *connector = &hdmi->connector; > struct drm_connector_state *conn_state = connector->state; > - struct inno_hdmi_connector_state *inno_conn_state = > - to_inno_hdmi_conn_state(conn_state); > int c0_c2_change = 0; > int csc_enable = 0; > int csc_mode = 0; > int auto_csc = 0; > int value; > int i; > + int colorimetry; > + u8 vic = drm_match_cea_mode(mode); > + > + if (vic == 6 || vic == 7 || vic == 21 || vic == 22 || > + vic == 2 || vic == 3 || vic == 17 || vic == 18) > + colorimetry = HDMI_COLORIMETRY_ITU_601; > + else > + colorimetry = HDMI_COLORIMETRY_ITU_709; > + > > /* Input video mode is SDR RGB24bit, data enable signal from external */ > hdmi_writeb(hdmi, HDMI_VIDEO_CONTRL1, v_DE_EXTERNAL | > @@ -720,7 +668,7 @@ static int inno_hdmi_config_video_csc(struct inno_hdmi > *hdmi) > return 0; > } > } else { > - if (inno_conn_state->colorimetry == HDMI_COLORIMETRY_ITU_601) { > + if (colorimetry == HDMI_COLORIMETRY_ITU_601) { > if (conn_state->hdmi.output_format == > HDMI_COLORSPACE_YUV444) { > csc_mode = CSC_RGB_0_255_TO_ITU601_16_235_8BIT; > auto_csc = AUTO_CSC_DISABLE; > @@ -738,8 +686,7 @@ static int inno_hdmi_config_video_csc(struct inno_hdmi > *hdmi) > } > > for (i = 0; i < 24; i++) > - hdmi_writeb(hdmi, HDMI_VIDEO_CSC_COEF + i, > - coeff_csc[csc_mode][i]); > + hdmi_writeb(hdmi, HDMI_VIDEO_CSC_COEF + i, > coeff_csc[csc_mode][i]); > > value = v_SOF_DISABLE | csc_enable | v_COLOR_DEPTH_NOT_INDICATED(1); > hdmi_writeb(hdmi, HDMI_VIDEO_CONTRL3, value); > @@ -753,15 +700,11 @@ static int inno_hdmi_config_video_csc(struct inno_hdmi > *hdmi) > static int inno_hdmi_config_video_timing(struct inno_hdmi *hdmi, > struct drm_display_mode *mode) > { > - int value, psync; > - > - if (hdmi->variant->dev_type == RK3036_HDMI) { > - psync = mode->flags & DRM_MODE_FLAG_PHSYNC ? RK3036_HDMI_PHSYNC > : 0; > - value = HIWORD_UPDATE(psync, RK3036_HDMI_PHSYNC); > - psync = mode->flags & DRM_MODE_FLAG_PVSYNC ? RK3036_HDMI_PVSYNC > : 0; > - value |= HIWORD_UPDATE(psync, RK3036_HDMI_PVSYNC); > - regmap_write(hdmi->grf, RK3036_GRF_SOC_CON2, value); > - } > + const struct inno_hdmi_plat_ops *plat_ops = hdmi->plat_data->ops; > + u32 value; > + > + if (plat_ops && plat_ops->enable) > + plat_ops->enable(hdmi->dev, mode); > > /* Set detail external video timing polarity and interlace mode */ > value = v_EXTERANL_VIDEO(1); > @@ -810,14 +753,16 @@ static int inno_hdmi_config_video_timing(struct > inno_hdmi *hdmi, > return 0; > } > > -static int inno_hdmi_setup(struct inno_hdmi *hdmi, > - struct drm_atomic_state *state) > +static int inno_hdmi_setup(struct inno_hdmi *hdmi, struct drm_atomic_state > *state) > { > - struct drm_connector *connector = &hdmi->connector; > - struct drm_display_info *display = &connector->display_info; > + struct drm_bridge *bridge = &hdmi->bridge; > + struct drm_connector *connector; > + struct drm_display_info *info; > struct drm_connector_state *new_conn_state; > struct drm_crtc_state *new_crtc_state; > > + connector = drm_atomic_get_new_connector_for_encoder(state, > bridge->encoder); > + > new_conn_state = drm_atomic_get_new_connector_state(state, connector); > if (WARN_ON(!new_conn_state)) > return -EINVAL; > @@ -826,17 +771,18 @@ static int inno_hdmi_setup(struct inno_hdmi *hdmi, > if (WARN_ON(!new_crtc_state)) > return -EINVAL; > > + info = &connector->display_info; > + > /* Mute video and audio output */ > hdmi_modb(hdmi, HDMI_AV_MUTE, m_AUDIO_MUTE | m_VIDEO_BLACK, > v_AUDIO_MUTE(1) | v_VIDEO_MUTE(1)); > > /* Set HDMI Mode */ > - hdmi_writeb(hdmi, HDMI_HDCP_CTRL, > - v_HDMI_DVI(display->is_hdmi)); > + hdmi_writeb(hdmi, HDMI_HDCP_CTRL, v_HDMI_DVI(info->is_hdmi)); > > inno_hdmi_config_video_timing(hdmi, &new_crtc_state->adjusted_mode); > > - inno_hdmi_config_video_csc(hdmi); > + inno_hdmi_config_video_csc(hdmi, connector, > &new_crtc_state->adjusted_mode); > > drm_atomic_helper_connector_hdmi_update_infoframes(connector, state); > > @@ -857,9 +803,11 @@ static int inno_hdmi_setup(struct inno_hdmi *hdmi, > return 0; > } > > -static enum drm_mode_status inno_hdmi_display_mode_valid(struct inno_hdmi > *hdmi, > - const struct > drm_display_mode *mode) > +static enum drm_mode_status inno_hdmi_bridge_mode_valid(struct drm_bridge > *bridge, > + const struct > drm_display_info *info, > + const struct > drm_display_mode *mode) > { > + struct inno_hdmi *hdmi = bridge_to_inno_hdmi(bridge); > unsigned long mpixelclk, max_tolerance; > long rounded_refclk; > > @@ -889,189 +837,57 @@ static enum drm_mode_status > inno_hdmi_display_mode_valid(struct inno_hdmi *hdmi, > return MODE_OK; > } > > -static void inno_hdmi_encoder_enable(struct drm_encoder *encoder, > - struct drm_atomic_state *state) > -{ > - struct inno_hdmi *hdmi = encoder_to_inno_hdmi(encoder); > - > - inno_hdmi_setup(hdmi, state); > -} > - > -static void inno_hdmi_encoder_disable(struct drm_encoder *encoder, > - struct drm_atomic_state *state) > -{ > - struct inno_hdmi *hdmi = encoder_to_inno_hdmi(encoder); > - > - inno_hdmi_standby(hdmi); > -} > - > -static int > -inno_hdmi_encoder_atomic_check(struct drm_encoder *encoder, > - struct drm_crtc_state *crtc_state, > - struct drm_connector_state *conn_state) > -{ > - struct rockchip_crtc_state *s = to_rockchip_crtc_state(crtc_state); > - struct drm_display_mode *mode = &crtc_state->adjusted_mode; > - u8 vic = drm_match_cea_mode(mode); > - struct inno_hdmi_connector_state *inno_conn_state = > - to_inno_hdmi_conn_state(conn_state); > - > - s->output_mode = ROCKCHIP_OUT_MODE_P888; > - s->output_type = DRM_MODE_CONNECTOR_HDMIA; > - > - if (vic == 6 || vic == 7 || > - vic == 21 || vic == 22 || > - vic == 2 || vic == 3 || > - vic == 17 || vic == 18) > - inno_conn_state->colorimetry = HDMI_COLORIMETRY_ITU_601; > - else > - inno_conn_state->colorimetry = HDMI_COLORIMETRY_ITU_709; > - > - return 0; > -} > - > -static const struct drm_encoder_helper_funcs inno_hdmi_encoder_helper_funcs > = { > - .atomic_check = inno_hdmi_encoder_atomic_check, > - .atomic_enable = inno_hdmi_encoder_enable, > - .atomic_disable = inno_hdmi_encoder_disable, > -}; > - > static enum drm_connector_status > -inno_hdmi_connector_detect(struct drm_connector *connector, bool force) > +inno_hdmi_bridge_detect(struct drm_bridge *bridge, struct drm_connector > *connector) > { > - struct inno_hdmi *hdmi = connector_to_inno_hdmi(connector); > + struct inno_hdmi *hdmi = bridge_to_inno_hdmi(bridge); > > return (hdmi_readb(hdmi, HDMI_STATUS) & m_HOTPLUG) ? > connector_status_connected : connector_status_disconnected; > } > > -static int inno_hdmi_connector_get_modes(struct drm_connector *connector) > +static const struct drm_edid * > +inno_hdmi_bridge_edid_read(struct drm_bridge *bridge, struct drm_connector > *connector) > { > - struct inno_hdmi *hdmi = connector_to_inno_hdmi(connector); > + struct inno_hdmi *hdmi = bridge_to_inno_hdmi(bridge); > const struct drm_edid *drm_edid; > - int ret = 0; > - > - if (!hdmi->ddc) > - return 0; > - > - drm_edid = drm_edid_read_ddc(connector, hdmi->ddc); > - drm_edid_connector_update(connector, drm_edid); > - ret = drm_edid_connector_add_modes(connector); > - drm_edid_free(drm_edid); > - > - return ret; > -} > - > -static enum drm_mode_status > -inno_hdmi_connector_mode_valid(struct drm_connector *connector, > - const struct drm_display_mode *mode) > -{ > - struct inno_hdmi *hdmi = connector_to_inno_hdmi(connector); > - > - return inno_hdmi_display_mode_valid(hdmi, mode); > -} > > -static void > -inno_hdmi_connector_destroy_state(struct drm_connector *connector, > - struct drm_connector_state *state) > -{ > - struct inno_hdmi_connector_state *inno_conn_state = > - to_inno_hdmi_conn_state(state); > + drm_edid = drm_edid_read_ddc(connector, bridge->ddc); > + if (!drm_edid) > + dev_dbg(hdmi->dev, "failed to get edid\n"); > > - __drm_atomic_helper_connector_destroy_state(&inno_conn_state->base); > - kfree(inno_conn_state); > + return drm_edid; > } > > -static void inno_hdmi_connector_reset(struct drm_connector *connector) > +static void inno_hdmi_bridge_atomic_enable(struct drm_bridge *bridge, > + struct drm_atomic_state *state) > { > - struct inno_hdmi_connector_state *inno_conn_state; > - > - if (connector->state) { > - inno_hdmi_connector_destroy_state(connector, connector->state); > - connector->state = NULL; > - } > - > - inno_conn_state = kzalloc(sizeof(*inno_conn_state), GFP_KERNEL); > - if (!inno_conn_state) > - return; > - > - __drm_atomic_helper_connector_reset(connector, &inno_conn_state->base); > - __drm_atomic_helper_connector_hdmi_reset(connector, connector->state); > + struct inno_hdmi *hdmi = bridge_to_inno_hdmi(bridge); > > - inno_conn_state->colorimetry = HDMI_COLORIMETRY_ITU_709; > + inno_hdmi_setup(hdmi, state); > } > > -static struct drm_connector_state * > -inno_hdmi_connector_duplicate_state(struct drm_connector *connector) > +static void inno_hdmi_bridge_atomic_disable(struct drm_bridge *bridge, > + struct drm_atomic_state *state) > { > - struct inno_hdmi_connector_state *inno_conn_state; > - > - if (WARN_ON(!connector->state)) > - return NULL; > - > - inno_conn_state = kmemdup(to_inno_hdmi_conn_state(connector->state), > - sizeof(*inno_conn_state), GFP_KERNEL); > - > - if (!inno_conn_state) > - return NULL; > + struct inno_hdmi *hdmi = bridge_to_inno_hdmi(bridge); > > - __drm_atomic_helper_connector_duplicate_state(connector, > - &inno_conn_state->base); > - > - return &inno_conn_state->base; > + inno_hdmi_standby(hdmi); > } > > -static const struct drm_connector_funcs inno_hdmi_connector_funcs = { > - .fill_modes = drm_helper_probe_single_connector_modes, > - .detect = inno_hdmi_connector_detect, > - .reset = inno_hdmi_connector_reset, > - .atomic_duplicate_state = inno_hdmi_connector_duplicate_state, > - .atomic_destroy_state = inno_hdmi_connector_destroy_state, > -}; > - > -static struct drm_connector_helper_funcs inno_hdmi_connector_helper_funcs = { > - .atomic_check = drm_atomic_helper_connector_hdmi_check, > - .get_modes = inno_hdmi_connector_get_modes, > - .mode_valid = inno_hdmi_connector_mode_valid, > +static const struct drm_bridge_funcs inno_hdmi_bridge_funcs = { > + .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, > + .atomic_enable = inno_hdmi_bridge_atomic_enable, > + .atomic_disable = inno_hdmi_bridge_atomic_disable, > + .detect = inno_hdmi_bridge_detect, > + .edid_read = inno_hdmi_bridge_edid_read, > + .hdmi_clear_infoframe = inno_hdmi_bridge_clear_infoframe, > + .hdmi_write_infoframe = inno_hdmi_bridge_write_infoframe, > + .mode_valid = inno_hdmi_bridge_mode_valid, > }; > > -static int inno_hdmi_register(struct drm_device *drm, struct inno_hdmi *hdmi) > -{ > - struct drm_encoder *encoder = &hdmi->encoder.encoder; > - struct device *dev = hdmi->dev; > - > - encoder->possible_crtcs = drm_of_find_possible_crtcs(drm, dev->of_node); > - > - /* > - * If we failed to find the CRTC(s) which this encoder is > - * supposed to be connected to, it's because the CRTC has > - * not been registered yet. Defer probing, and hope that > - * the required CRTC is added later. > - */ > - if (encoder->possible_crtcs == 0) > - return -EPROBE_DEFER; > - > - drm_encoder_helper_add(encoder, &inno_hdmi_encoder_helper_funcs); > - drm_simple_encoder_init(drm, encoder, DRM_MODE_ENCODER_TMDS); > - > - hdmi->connector.polled = DRM_CONNECTOR_POLL_HPD; > - > - drm_connector_helper_add(&hdmi->connector, > - &inno_hdmi_connector_helper_funcs); > - drmm_connector_hdmi_init(drm, &hdmi->connector, > - "Rockchip", "Inno HDMI", > - &inno_hdmi_connector_funcs, > - &inno_hdmi_hdmi_connector_funcs, > - DRM_MODE_CONNECTOR_HDMIA, > - hdmi->ddc, > - BIT(HDMI_COLORSPACE_RGB), > - 8); > - > - drm_connector_attach_encoder(&hdmi->connector, encoder); > - > - return 0; > -} > - > static irqreturn_t inno_hdmi_i2c_irq(struct inno_hdmi *hdmi) > { > struct inno_hdmi_i2c *i2c = hdmi->i2c; > @@ -1111,7 +927,7 @@ static irqreturn_t inno_hdmi_irq(int irq, void *dev_id) > { > struct inno_hdmi *hdmi = dev_id; > > - drm_helper_hpd_irq_event(hdmi->connector.dev); > + drm_helper_hpd_irq_event(hdmi->bridge.dev); > > return IRQ_HANDLED; > } > @@ -1243,128 +1059,80 @@ static struct i2c_adapter > *inno_hdmi_i2c_adapter(struct inno_hdmi *hdmi) > return adap; > } > > -static int inno_hdmi_bind(struct device *dev, struct device *master, > - void *data) > +struct inno_hdmi *inno_hdmi_bind(struct device *dev, > + struct drm_encoder *encoder, > + const struct inno_hdmi_plat_data *plat_data) > { > struct platform_device *pdev = to_platform_device(dev); > - struct drm_device *drm = data; > struct inno_hdmi *hdmi; > - const struct inno_hdmi_variant *variant; > int irq; > int ret; > > - hdmi = devm_kzalloc(dev, sizeof(*hdmi), GFP_KERNEL); > - if (!hdmi) > - return -ENOMEM; > - > - hdmi->dev = dev; > + if (!plat_data->phy_configs || !plat_data->default_phy_config) { > + dev_err(dev, "Missing platform PHY ops\n"); > + return ERR_PTR(-ENODEV); > + } > > - variant = of_device_get_match_data(hdmi->dev); > - if (!variant) > - return -EINVAL; > + hdmi = devm_drm_bridge_alloc(dev, struct inno_hdmi, bridge, > &inno_hdmi_bridge_funcs); > + if (IS_ERR(hdmi)) > + return ERR_CAST(hdmi); > > - hdmi->variant = variant; > + hdmi->dev = dev; > + hdmi->plat_data = plat_data; > > hdmi->regs = devm_platform_ioremap_resource(pdev, 0); > if (IS_ERR(hdmi->regs)) > - return PTR_ERR(hdmi->regs); > + return ERR_CAST(hdmi->regs); > > hdmi->pclk = devm_clk_get_enabled(hdmi->dev, "pclk"); > - if (IS_ERR(hdmi->pclk)) > - return dev_err_probe(dev, PTR_ERR(hdmi->pclk), "Unable to get > HDMI pclk\n"); > + if (IS_ERR(hdmi->pclk)) { > + dev_err_probe(dev, PTR_ERR(hdmi->pclk), "Unable to get HDMI > pclk\n"); > + return ERR_CAST(hdmi->pclk); > + } > > hdmi->refclk = devm_clk_get_optional_enabled(hdmi->dev, "ref"); > - if (IS_ERR(hdmi->refclk)) > - return dev_err_probe(dev, PTR_ERR(hdmi->refclk), "Unable to get > HDMI refclk\n"); > - > - if (hdmi->variant->dev_type == RK3036_HDMI) { > - hdmi->grf = syscon_regmap_lookup_by_phandle(dev->of_node, > "rockchip,grf"); > - if (IS_ERR(hdmi->grf)) > - return dev_err_probe(dev, > - PTR_ERR(hdmi->grf), "Unable to get > rockchip,grf\n"); > + if (IS_ERR(hdmi->refclk)) { > + dev_err_probe(dev, PTR_ERR(hdmi->refclk), "Unable to get HDMI > refclk\n"); > + return ERR_CAST(hdmi->refclk); > } > > - irq = platform_get_irq(pdev, 0); > - if (irq < 0) > - return irq; > - > inno_hdmi_init_hw(hdmi); > > - hdmi->ddc = inno_hdmi_i2c_adapter(hdmi); > - if (IS_ERR(hdmi->ddc)) > - return PTR_ERR(hdmi->ddc); > - > - ret = inno_hdmi_register(drm, hdmi); > - if (ret) > - return ret; > - > - dev_set_drvdata(dev, hdmi); > + irq = platform_get_irq(pdev, 0); > + if (irq < 0) > + return ERR_PTR(irq); > > ret = devm_request_threaded_irq(dev, irq, inno_hdmi_hardirq, > inno_hdmi_irq, IRQF_SHARED, > dev_name(dev), hdmi); > - if (ret < 0) > - goto err_cleanup_hdmi; > - > - return 0; > -err_cleanup_hdmi: > - hdmi->connector.funcs->destroy(&hdmi->connector); > - hdmi->encoder.encoder.funcs->destroy(&hdmi->encoder.encoder); > - return ret; > -} > - > -static void inno_hdmi_unbind(struct device *dev, struct device *master, > - void *data) > -{ > - struct inno_hdmi *hdmi = dev_get_drvdata(dev); > - > - hdmi->connector.funcs->destroy(&hdmi->connector); > - hdmi->encoder.encoder.funcs->destroy(&hdmi->encoder.encoder); > -} > + if (ret) > + return ERR_PTR(ret); > > -static const struct component_ops inno_hdmi_ops = { > - .bind = inno_hdmi_bind, > - .unbind = inno_hdmi_unbind, > -}; > + hdmi->bridge.driver_private = hdmi; > + hdmi->bridge.ops = DRM_BRIDGE_OP_DETECT | > + DRM_BRIDGE_OP_EDID | > + DRM_BRIDGE_OP_HDMI | > + DRM_BRIDGE_OP_HPD; > + hdmi->bridge.of_node = pdev->dev.of_node; > + hdmi->bridge.type = DRM_MODE_CONNECTOR_HDMIA; > + hdmi->bridge.vendor = "Inno"; > + hdmi->bridge.product = "Inno HDMI"; > + > + hdmi->bridge.ddc = inno_hdmi_i2c_adapter(hdmi); > + if (IS_ERR(hdmi->bridge.ddc)) > + return ERR_CAST(hdmi->bridge.ddc); > + > + ret = devm_drm_bridge_add(dev, &hdmi->bridge); > + if (ret) > + return ERR_PTR(ret); > > -static int inno_hdmi_probe(struct platform_device *pdev) > -{ > - return component_add(&pdev->dev, &inno_hdmi_ops); > -} > + ret = drm_bridge_attach(encoder, &hdmi->bridge, NULL, > DRM_BRIDGE_ATTACH_NO_CONNECTOR); > + if (ret) > + return ERR_PTR(ret); > > -static void inno_hdmi_remove(struct platform_device *pdev) > -{ > - component_del(&pdev->dev, &inno_hdmi_ops); > + return hdmi; > } > - > -static const struct inno_hdmi_variant rk3036_inno_hdmi_variant = { > - .dev_type = RK3036_HDMI, > - .phy_configs = rk3036_hdmi_phy_configs, > - .default_phy_config = &rk3036_hdmi_phy_configs[1], > -}; > - > -static const struct inno_hdmi_variant rk3128_inno_hdmi_variant = { > - .dev_type = RK3128_HDMI, > - .phy_configs = rk3128_hdmi_phy_configs, > - .default_phy_config = &rk3128_hdmi_phy_configs[1], > -}; > - > -static const struct of_device_id inno_hdmi_dt_ids[] = { > - { .compatible = "rockchip,rk3036-inno-hdmi", > - .data = &rk3036_inno_hdmi_variant, > - }, > - { .compatible = "rockchip,rk3128-inno-hdmi", > - .data = &rk3128_inno_hdmi_variant, > - }, > - {}, > -}; > -MODULE_DEVICE_TABLE(of, inno_hdmi_dt_ids); > - > -struct platform_driver inno_hdmi_driver = { > - .probe = inno_hdmi_probe, > - .remove = inno_hdmi_remove, > - .driver = { > - .name = "innohdmi-rockchip", > - .of_match_table = inno_hdmi_dt_ids, > - }, > -}; > +EXPORT_SYMBOL_GPL(inno_hdmi_bind); > +MODULE_AUTHOR("Andy Yan <andys...@163.com>"); > +MODULE_DESCRIPTION("INNOSILICON HDMI transmitter library"); > +MODULE_LICENSE("GPL"); > diff --git a/drivers/gpu/drm/rockchip/Kconfig > b/drivers/gpu/drm/rockchip/Kconfig > index 14ec0281d45a4..e2f355f87ae51 100644 > --- a/drivers/gpu/drm/rockchip/Kconfig > +++ b/drivers/gpu/drm/rockchip/Kconfig > @@ -15,6 +15,7 @@ config DRM_ROCKCHIP > select DRM_DW_HDMI_QP if ROCKCHIP_DW_HDMI_QP > select DRM_DW_MIPI_DSI if ROCKCHIP_DW_MIPI_DSI > select DRM_DW_MIPI_DSI2 if ROCKCHIP_DW_MIPI_DSI2 > + select DRM_INNO_HDMI if ROCKCHIP_INNO_HDMI > select GENERIC_PHY if ROCKCHIP_DW_MIPI_DSI > select GENERIC_PHY_MIPI_DPHY if ROCKCHIP_DW_MIPI_DSI > select SND_SOC_HDMI_CODEC if ROCKCHIP_CDN_DP && SND_SOC > diff --git a/drivers/gpu/drm/rockchip/Makefile > b/drivers/gpu/drm/rockchip/Makefile > index 097f062399c7a..948b0f906d3d8 100644 > --- a/drivers/gpu/drm/rockchip/Makefile > +++ b/drivers/gpu/drm/rockchip/Makefile > @@ -15,7 +15,7 @@ rockchipdrm-$(CONFIG_ROCKCHIP_DW_HDMI_QP) += > dw_hdmi_qp-rockchip.o > rockchipdrm-$(CONFIG_ROCKCHIP_DW_MIPI_DSI) += dw-mipi-dsi-rockchip.o > rockchipdrm-$(CONFIG_ROCKCHIP_DW_MIPI_DSI2) += dw-mipi-dsi2-rockchip.o > rockchipdrm-$(CONFIG_ROCKCHIP_DW_DP) += dw_dp-rockchip.o > -rockchipdrm-$(CONFIG_ROCKCHIP_INNO_HDMI) += inno_hdmi.o > +rockchipdrm-$(CONFIG_ROCKCHIP_INNO_HDMI) += inno_hdmi-rockchip.o > rockchipdrm-$(CONFIG_ROCKCHIP_LVDS) += rockchip_lvds.o > rockchipdrm-$(CONFIG_ROCKCHIP_RGB) += rockchip_rgb.o > rockchipdrm-$(CONFIG_ROCKCHIP_RK3066_HDMI) += rk3066_hdmi.o > diff --git a/drivers/gpu/drm/rockchip/inno_hdmi-rockchip.c > b/drivers/gpu/drm/rockchip/inno_hdmi-rockchip.c > new file mode 100644 > index 0000000000000..31cb2a90308c1 > --- /dev/null > +++ b/drivers/gpu/drm/rockchip/inno_hdmi-rockchip.c > @@ -0,0 +1,188 @@ > +// SPDX-License-Identifier: GPL-2.0-only > +/* > + * Copyright (C) Rockchip Electronics Co., Ltd. > + * Zheng Yang <zhengy...@rock-chips.com> > + * Andy Yan <andy....@rock-chips.com> > + */ > +#include <linux/err.h> > +#include <linux/mfd/syscon.h> > +#include <linux/mod_devicetable.h> > +#include <linux/module.h> > +#include <linux/platform_device.h> > +#include <linux/regmap.h> > + > +#include <drm/bridge/inno_hdmi.h> > +#include <drm/drm_bridge_connector.h> > +#include <drm/drm_of.h> > + > +#include "rockchip_drm_drv.h" > + > +#define HIWORD_UPDATE(val, mask) ((val) | (mask) << 16) > + > +#define RK3036_GRF_SOC_CON2 0x148 > +#define RK3036_HDMI_PHSYNC BIT(4) > +#define RK3036_HDMI_PVSYNC BIT(5) > + > +enum inno_hdmi_dev_type { > + RK3036_HDMI, > + RK3128_HDMI, > +}; > + > +struct inno_hdmi_connector_state { > + struct drm_connector_state base; > + unsigned int colorimetry; > +}; > + > +struct rockchip_inno_hdmi { > + struct inno_hdmi *base; > + struct device *dev; > + struct regmap *grf; > + struct rockchip_encoder encoder; > +}; > + > +static struct inno_hdmi_phy_config rk3036_hdmi_phy_configs[] = { > + { 74250000, 0x3f, 0xbb }, > + { 165000000, 0x6f, 0xbb }, > + { ~0UL, 0x00, 0x00 } > +}; > + > +static struct inno_hdmi_phy_config rk3128_hdmi_phy_configs[] = { > + { 74250000, 0x3f, 0xaa }, > + { 165000000, 0x5f, 0xaa }, > + { ~0UL, 0x00, 0x00 } > +}; > + > +static void inno_hdmi_rk3036_enable(struct device *dev, struct > drm_display_mode *mode) > +{ > + struct rockchip_inno_hdmi *hdmi = dev_get_drvdata(dev); > + int value, psync; > + > + psync = mode->flags & DRM_MODE_FLAG_PHSYNC ? RK3036_HDMI_PHSYNC : 0; > + value = HIWORD_UPDATE(psync, RK3036_HDMI_PHSYNC); > + psync = mode->flags & DRM_MODE_FLAG_PVSYNC ? RK3036_HDMI_PVSYNC : 0; > + value |= HIWORD_UPDATE(psync, RK3036_HDMI_PVSYNC); > + regmap_write(hdmi->grf, RK3036_GRF_SOC_CON2, value); > +} > + > +static int inno_hdmi_encoder_atomic_check(struct drm_encoder *encoder, > + struct drm_crtc_state *crtc_state, > + struct drm_connector_state > *conn_state) > +{ > + struct rockchip_crtc_state *s = to_rockchip_crtc_state(crtc_state); > + > + s->output_mode = ROCKCHIP_OUT_MODE_P888; > + s->output_type = DRM_MODE_CONNECTOR_HDMIA; > + > + return 0; > +} > + > +static const struct drm_encoder_helper_funcs > inno_hdmi_rockchip_encoder_helper_funcs = { > + .atomic_check = inno_hdmi_encoder_atomic_check, > +}; > + > +static int inno_hdmi_rockchip_bind(struct device *dev, struct device > *master, void *data) > +{ > + struct drm_device *drm = data; > + struct drm_connector *connector; > + struct drm_encoder *encoder; > + struct rockchip_inno_hdmi *hdmi; > + const struct inno_hdmi_plat_data *plat_data; > + int ret; > + > + hdmi = devm_kzalloc(dev, sizeof(*hdmi), GFP_KERNEL); > + if (!hdmi) > + return -ENOMEM; > + > + hdmi->dev = dev; > + > + plat_data = of_device_get_match_data(hdmi->dev); > + if (!plat_data) > + return -EINVAL; > + > + if (of_device_is_compatible(dev->of_node, "rockchip,rk3036-inno-hdmi")) > { > + hdmi->grf = syscon_regmap_lookup_by_phandle(dev->of_node, > "rockchip,grf"); > + if (IS_ERR(hdmi->grf)) > + return dev_err_probe(dev, > + PTR_ERR(hdmi->grf), "Unable to get > rockchip,grf\n"); > + } > + > + encoder = &hdmi->encoder.encoder; > + encoder->possible_crtcs = drm_of_find_possible_crtcs(drm, dev->of_node); > + > + /* > + * If we failed to find the CRTC(s) which this encoder is > + * supposed to be connected to, it's because the CRTC has > + * not been registered yet. Defer probing, and hope that > + * the required CRTC is added later. > + */ > + if (encoder->possible_crtcs == 0) > + return -EPROBE_DEFER; > + > + ret = drmm_encoder_init(drm, encoder, NULL, DRM_MODE_ENCODER_TMDS, > NULL); > + if (ret) > + return ret; > + > + drm_encoder_helper_add(encoder, > &inno_hdmi_rockchip_encoder_helper_funcs); > + > + dev_set_drvdata(dev, hdmi); > + > + hdmi->base = inno_hdmi_bind(dev, encoder, plat_data); > + > + connector = drm_bridge_connector_init(drm, encoder); > + if (IS_ERR(connector)) { > + ret = PTR_ERR(connector); > + dev_err(hdmi->dev, "failed to init bridge connector: %d\n", > ret); > + return ret; > + } > + > + return drm_connector_attach_encoder(connector, encoder); > +} > + > +static const struct component_ops inno_hdmi_rockchip_ops = { > + .bind = inno_hdmi_rockchip_bind, > +}; > + > +static int inno_hdmi_rockchip_probe(struct platform_device *pdev) > +{ > + return component_add(&pdev->dev, &inno_hdmi_rockchip_ops); > +} > + > +static void inno_hdmi_rockchip_remove(struct platform_device *pdev) > +{ > + component_del(&pdev->dev, &inno_hdmi_rockchip_ops); > +} > + > +static const struct inno_hdmi_plat_ops rk3036_inno_hdmi_plat_ops = { > + .enable = inno_hdmi_rk3036_enable, > +}; > + > +static const struct inno_hdmi_plat_data rk3036_inno_hdmi_plat_data = { > + .ops = &rk3036_inno_hdmi_plat_ops, > + .phy_configs = rk3036_hdmi_phy_configs, > + .default_phy_config = &rk3036_hdmi_phy_configs[1], > +}; > + > +static const struct inno_hdmi_plat_data rk3128_inno_hdmi_plat_data = { > + .phy_configs = rk3128_hdmi_phy_configs, > + .default_phy_config = &rk3128_hdmi_phy_configs[1], > +}; > + > +static const struct of_device_id inno_hdmi_rockchip_dt_ids[] = { > + { .compatible = "rockchip,rk3036-inno-hdmi", > + .data = &rk3036_inno_hdmi_plat_data, > + }, > + { .compatible = "rockchip,rk3128-inno-hdmi", > + .data = &rk3128_inno_hdmi_plat_data, > + }, > + {}, > +}; > +MODULE_DEVICE_TABLE(of, inno_hdmi_rockchip_dt_ids); > + > +struct platform_driver inno_hdmi_driver = { > + .probe = inno_hdmi_rockchip_probe, > + .remove = inno_hdmi_rockchip_remove, > + .driver = { > + .name = "innohdmi-rockchip", > + .of_match_table = inno_hdmi_rockchip_dt_ids, > + }, > +}; > diff --git a/include/drm/bridge/inno_hdmi.h b/include/drm/bridge/inno_hdmi.h > new file mode 100644 > index 0000000000000..8b39655212e24 > --- /dev/null > +++ b/include/drm/bridge/inno_hdmi.h > @@ -0,0 +1,33 @@ > +/* SPDX-License-Identifier: GPL-2.0-or-later */ > +/* > + * Copyright (c) 2025 Rockchip Electronics Co., Ltd. > + */ > + > +#ifndef __INNO_HDMI__ > +#define __INNO_HDMI__ > + > +struct device; > +struct drm_encoder; > +struct drm_display_mode; > +struct inno_hdmi; > + > +struct inno_hdmi_plat_ops { > + void (*enable)(struct device *pdev, struct drm_display_mode *mode); > +}; > + > +struct inno_hdmi_phy_config { > + unsigned long pixelclock; > + u8 pre_emphasis; > + u8 voltage_level_control; > +}; > + > +struct inno_hdmi_plat_data { > + const struct inno_hdmi_plat_ops *ops; > + struct inno_hdmi_phy_config *phy_configs; > + struct inno_hdmi_phy_config *default_phy_config; > +}; > + > +struct inno_hdmi *inno_hdmi_bind(struct device *pdev, > + struct drm_encoder *encoder, > + const struct inno_hdmi_plat_data *plat_data); > +#endif /* __INNO_HDMI__ */ >