Adds hdmi and hdmi-ddc support for v2 IP.

Signed-off-by: Guillaume Ranquet <granq...@baylibre.com>
---
 drivers/gpu/drm/mediatek/Kconfig            |    2 +
 drivers/gpu/drm/mediatek/Makefile           |    2 +
 drivers/gpu/drm/mediatek/mtk_hdmi_common.c  |   13 +
 drivers/gpu/drm/mediatek/mtk_hdmi_common.h  |    1 +
 drivers/gpu/drm/mediatek/mtk_hdmi_ddc_v2.c  |  362 +++++++++
 drivers/gpu/drm/mediatek/mtk_hdmi_regs_v2.h |  276 +++++++
 drivers/gpu/drm/mediatek/mtk_hdmi_v2.c      | 1105 +++++++++++++++++++++++++++
 drivers/gpu/drm/mediatek/mtk_hdmi_v2.h      |   30 +
 8 files changed, 1791 insertions(+)

diff --git a/drivers/gpu/drm/mediatek/Kconfig b/drivers/gpu/drm/mediatek/Kconfig
index b451dee64d34..242992cd1439 100644
--- a/drivers/gpu/drm/mediatek/Kconfig
+++ b/drivers/gpu/drm/mediatek/Kconfig
@@ -34,5 +34,7 @@ config DRM_MEDIATEK_HDMI
        depends on DRM_MEDIATEK
        select SND_SOC_HDMI_CODEC if SND_SOC
        select PHY_MTK_HDMI
+       select DRM_DISPLAY_HDMI_HELPER
+       select DRM_DISPLAY_HELPER
        help
          DRM/KMS HDMI driver for Mediatek SoCs
diff --git a/drivers/gpu/drm/mediatek/Makefile 
b/drivers/gpu/drm/mediatek/Makefile
index 79bbaa58893e..bb60856b629e 100644
--- a/drivers/gpu/drm/mediatek/Makefile
+++ b/drivers/gpu/drm/mediatek/Makefile
@@ -24,6 +24,8 @@ mediatek-drm-hdmi-objs := mtk_cec.o \
                          mtk_hdmi.o \
                          mtk_hdmi_common.o \
                          mtk_hdmi_ddc.o \
+                         mtk_hdmi_ddc_v2.o \
+                         mtk_hdmi_v2.o \
 
 obj-$(CONFIG_DRM_MEDIATEK_HDMI) += mediatek-drm-hdmi.o
 
diff --git a/drivers/gpu/drm/mediatek/mtk_hdmi_common.c 
b/drivers/gpu/drm/mediatek/mtk_hdmi_common.c
index d7f3d29138d8..97a00102a970 100644
--- a/drivers/gpu/drm/mediatek/mtk_hdmi_common.c
+++ b/drivers/gpu/drm/mediatek/mtk_hdmi_common.c
@@ -341,6 +341,15 @@ static const struct mtk_hdmi_conf mtk_hdmi_conf_mt8173 = {
        .num_clocks = MTK_HDMIV1_CLK_COUNT,
 };
 
+static const struct mtk_hdmi_conf mtk_hdmi_conf_v2 = {
+       .bridge_funcs = &mtk_v2_hdmi_bridge_funcs,
+       .mtk_hdmi_output_init = mtk_hdmi_output_init_v2,
+       .mtk_hdmi_clk_disable = mtk_hdmi_clk_disable_v2,
+       .mtk_hdmi_clk_enable = mtk_hdmi_clk_enable_v2,
+       .mtk_hdmi_clock_names = mtk_hdmi_clk_names_v2,
+       .num_clocks = MTK_HDMIV2_CLK_COUNT,
+};
+
 static const struct of_device_id mtk_drm_hdmi_of_ids[] = {
        { .compatible = "mediatek,mt2701-hdmi",
          .data = &mtk_hdmi_conf_mt2701,
@@ -351,6 +360,9 @@ static const struct of_device_id mtk_drm_hdmi_of_ids[] = {
        { .compatible = "mediatek,mt8173-hdmi",
          .data = &mtk_hdmi_conf_mt8173,
        },
+       { .compatible = "mediatek,mt8195-hdmi",
+         .data = &mtk_hdmi_conf_v2,
+       },
        {}
 };
 MODULE_DEVICE_TABLE(of, mtk_drm_hdmi_of_ids);
@@ -399,6 +411,7 @@ static struct platform_driver mtk_hdmi_driver = {
 static struct platform_driver * const mtk_hdmi_drivers[] = {
        &mtk_hdmi_ddc_driver,
        &mtk_cec_driver,
+       &mtk_hdmi_ddc_v2_driver,
        &mtk_hdmi_driver,
 };
 
diff --git a/drivers/gpu/drm/mediatek/mtk_hdmi_common.h 
b/drivers/gpu/drm/mediatek/mtk_hdmi_common.h
index 6e4633981018..86de8942727d 100644
--- a/drivers/gpu/drm/mediatek/mtk_hdmi_common.h
+++ b/drivers/gpu/drm/mediatek/mtk_hdmi_common.h
@@ -27,6 +27,7 @@
 
 #include "mtk_cec.h"
 #include "mtk_hdmi.h"
+#include "mtk_hdmi_v2.h"
 
 struct mtk_hdmi_conf {
        bool tz_disabled;
diff --git a/drivers/gpu/drm/mediatek/mtk_hdmi_ddc_v2.c 
b/drivers/gpu/drm/mediatek/mtk_hdmi_ddc_v2.c
new file mode 100644
index 000000000000..c9e8424109f0
--- /dev/null
+++ b/drivers/gpu/drm/mediatek/mtk_hdmi_ddc_v2.c
@@ -0,0 +1,362 @@
+// SPDX-License-Identifier: GPL-2.0
+/*
+ * Copyright (c) 2021 MediaTek Inc.
+ * Copyright (c) 2021 BayLibre, SAS
+ */
+
+#include <linux/clk.h>
+#include <linux/delay.h>
+#include <linux/err.h>
+#include <linux/errno.h>
+#include <linux/i2c.h>
+#include <linux/io.h>
+#include <linux/iopoll.h>
+#include <linux/kernel.h>
+#include <linux/mfd/syscon.h>
+#include <linux/module.h>
+#include <linux/mutex.h>
+#include <linux/of_address.h>
+#include <linux/of_irq.h>
+#include <linux/of_platform.h>
+#include <linux/platform_device.h>
+#include <linux/regmap.h>
+#include <linux/semaphore.h>
+#include <linux/slab.h>
+#include <linux/time.h>
+
+#include <drm/drm_edid.h>
+
+#include "mtk_hdmi_regs_v2.h"
+#include "mtk_hdmi_v2.h"
+
+#define EDID_ID 0x50
+#define DDC2_CLOK 572 /* BIM=208M/(v*4) = 90Khz */
+#define DDC2_CLOK_EDID 832 /* BIM=208M/(v*4) = 62.5Khz */
+
+struct mtk_hdmi_ddc {
+       struct device *dev;
+       /* Serialize read/write operations */
+       struct mutex mtx;
+       struct i2c_adapter adap;
+       struct clk *clk;
+       void __iomem *regs;
+};
+
+enum sif_bit_t_hdmi {
+       SIF_8_BIT_HDMI, /* /< [8 bits data address.] */
+       SIF_16_BIT_HDMI, /* /< [16 bits data address.] */
+};
+
+static void hdmi_ddc_request(struct mtk_hdmi_ddc *ddc)
+{
+       regmap_update_bits(ddc->regs, HDCP2X_POL_CTRL, HDCP2X_DIS_POLL_EN,
+                    HDCP2X_DIS_POLL_EN);
+}
+
+static void mtk_ddc_wr_one(struct mtk_hdmi_ddc *ddc, unsigned int addr_id,
+                      unsigned int offset_id, unsigned char wr_data)
+{
+       u32 val;
+
+       regmap_read(ddc->regs, HDCP2X_DDCM_STATUS, &val);
+
+       if (val & DDC_I2C_BUS_LOW) {
+               regmap_update_bits(ddc->regs, DDC_CTRL, FIELD_PREP(DDC_CMD, 
CLOCK_SCL), DDC_CMD);
+               usleep_range(250, 300);
+       }
+       regmap_update_bits(ddc->regs, HPD_DDC_CTRL, FIELD_PREP(DDC_DELAY_CNT, 
DDC2_CLOK), DDC_DELAY_CNT);
+       regmap_write(ddc->regs, SI2C_CTRL, FIELD_PREP(SI2C_ADDR, 
SI2C_ADDR_READ));
+       regmap_update_bits(ddc->regs, SI2C_CTRL, FIELD_PREP(SI2C_WDATA, 
wr_data), SI2C_WDATA);
+       regmap_update_bits(ddc->regs, SI2C_CTRL, SI2C_WR, SI2C_WR);
+
+       regmap_write(ddc->regs, DDC_CTRL,
+                     FIELD_PREP(DDC_CMD, SEQ_WRITE_REQ_ACK) |
+                     FIELD_PREP(DDC_DIN_CNT, 1) |
+                     FIELD_PREP(DDC_OFFSET, offset_id) |
+                     (addr_id << 1));
+
+       usleep_range(1000, 1250);
+       regmap_read(ddc->regs, HDCP2X_DDCM_STATUS, &val);
+
+       if (val & DDC_I2C_BUS_LOW) {
+               regmap_update_bits(ddc->regs, DDC_CTRL, FIELD_PREP(DDC_CMD, 
CLOCK_SCL), DDC_CMD);
+               usleep_range(250, 300);
+       }
+}
+
+static unsigned char
+ddcm_read_hdmi(struct mtk_hdmi_ddc *ddc,
+              unsigned int u4_clk_div, unsigned char uc_dev, unsigned int 
u4_addr,
+              enum sif_bit_t_hdmi uc_addr_type, unsigned char *puc_value,
+              unsigned int u4_count)
+{
+       u32 val;
+       unsigned int i, temp_length, loop_counter, uc_read_count, uc_idx = 0;
+
+       if (!puc_value || !u4_count || !u4_clk_div)
+               return 0;
+
+       uc_idx = 0;
+       regmap_read(ddc->regs, HDCP2X_DDCM_STATUS, &val);
+       if (val & DDC_I2C_BUS_LOW) {
+               regmap_update_bits(ddc->regs, DDC_CTRL, FIELD_PREP(DDC_CMD, 
CLOCK_SCL), DDC_CMD);
+               usleep_range(250, 300);
+       }
+
+       regmap_update_bits(ddc->regs, DDC_CTRL, FIELD_PREP(DDC_CMD, 
CLEAR_FIFO), DDC_CMD);
+
+       if (u4_count >= 16) {
+               temp_length = 16;
+               loop_counter =
+                       u4_count / 16 + (u4_count % 16 == 0);
+       } else {
+               temp_length = u4_count;
+               loop_counter = 1;
+       }
+       if (uc_dev >= EDID_ID) {
+               if (u4_clk_div < DDC2_CLOK_EDID)
+                       u4_clk_div = DDC2_CLOK_EDID;
+       }
+       regmap_update_bits(ddc->regs, HPD_DDC_CTRL, FIELD_PREP(DDC_DELAY_CNT, 
u4_clk_div), DDC_DELAY_CNT);
+       for (i = 0; i < loop_counter; i++) {
+               if (i == (loop_counter - 1) && i != 0 &&
+                       u4_count % 16)
+                       temp_length = u4_count % 16;
+               if (uc_dev > EDID_ID) {
+                       regmap_update_bits(ddc->regs, SCDC_CTRL, 
FIELD_PREP(DDC_SEGMENT, uc_dev - EDID_ID), DDC_SEGMENT);
+                       regmap_write(ddc->regs, DDC_CTRL, FIELD_PREP(DDC_CMD, 
ENH_READ_NO_ACK) |
+                                         FIELD_PREP(DDC_DIN_CNT, temp_length) |
+                                         FIELD_PREP(DDC_OFFSET, u4_addr + i * 
temp_length) |
+                                         (EDID_ID << 1));
+               } else {
+                       regmap_write(ddc->regs, DDC_CTRL, FIELD_PREP(DDC_CMD, 
SEQ_READ_NO_ACK) |
+                                         FIELD_PREP(DDC_DIN_CNT, temp_length) |
+                                         FIELD_PREP(DDC_OFFSET, u4_addr + i * 
16) |
+                                         (uc_dev << 1));
+               }
+               usleep_range(5000, 5500);
+               if (regmap_read_poll_timeout(ddc->regs, HPD_DDC_STATUS, val,
+                               !(val & DDC_I2C_IN_PROG), 2000, temp_length + 
5)) {
+                       dev_err(ddc->dev, "time out waiting for DDC I2C\n");
+                       return 0;
+               }
+               regmap_read(ddc->regs, HDCP2X_DDCM_STATUS, &val);
+               if ((val & (DDC_I2C_NO_ACK | DDC_I2C_BUS_LOW))) {
+                       regmap_read(ddc->regs, HDCP2X_DDCM_STATUS, &val);
+                       if (val & DDC_I2C_BUS_LOW) {
+                               regmap_update_bits(ddc->regs, DDC_CTRL,
+                                                FIELD_PREP(DDC_CMD, 
CLOCK_SCL), DDC_CMD);
+                               usleep_range(250, 300);
+                       }
+                       return 0;
+               }
+               for (uc_idx = 0; uc_idx < temp_length; uc_idx++) {
+                       regmap_write(ddc->regs, SI2C_CTRL,
+                                         FIELD_PREP(SI2C_ADDR, SI2C_ADDR_READ) 
| SI2C_RD);
+                       regmap_write(ddc->regs, SI2C_CTRL,
+                                         FIELD_PREP(SI2C_ADDR, SI2C_ADDR_READ) 
| SI2C_CONFIRM_READ);
+                       regmap_read(ddc->regs, HPD_DDC_STATUS, &val);
+                       puc_value[i * 16 + uc_idx] = FIELD_GET(DDC_DATA_OUT, 
val);
+                       /*
+                        * when reading edid, if hdmi module been reset,
+                        * ddc will fail and it's speed will be set to 400.
+                        */
+                       regmap_read(ddc->regs, HPD_DDC_CTRL, &val);
+                       if (((val >> 16) & 0xFFFF) < DDC2_CLOK)
+                               return 0;
+
+                       uc_read_count = i * 16 + uc_idx + 1;
+               }
+       }
+       return uc_read_count;
+}
+
+static unsigned char vddc_read(struct mtk_hdmi_ddc *ddc, unsigned int 
u4_clk_div,
+                              unsigned char uc_dev, unsigned int u4_addr,
+                              enum sif_bit_t_hdmi uc_addr_type,
+                              unsigned char *puc_value, unsigned int u4_count)
+{
+       unsigned int u4_read_count = 0;
+       unsigned char uc_return_value = 0;
+
+       if (!puc_value || !u4_count || !u4_clk_div ||
+           uc_addr_type > SIF_16_BIT_HDMI ||
+           (uc_addr_type == SIF_8_BIT_HDMI && u4_addr > 255) ||
+           (uc_addr_type == SIF_16_BIT_HDMI && u4_addr > 65535)) {
+               return 0;
+       }
+
+       if (uc_addr_type == SIF_8_BIT_HDMI)
+               u4_read_count = 255 - u4_addr + 1;
+       else if (uc_addr_type == SIF_16_BIT_HDMI)
+               u4_read_count = 65535 - u4_addr + 1;
+
+       u4_read_count = (u4_read_count > u4_count) ? u4_count : u4_read_count;
+       uc_return_value = ddcm_read_hdmi(ddc, u4_clk_div, uc_dev, u4_addr,
+                                        uc_addr_type, puc_value, 
u4_read_count);
+       return uc_return_value;
+}
+
+static bool fg_ddc_data_read(struct mtk_hdmi_ddc *ddc,
+                                     unsigned char b_dev,
+                                     unsigned char b_data_addr,
+                                     unsigned char b_data_count,
+                                     unsigned char *pr_data)
+{
+       int ret;
+
+       mutex_lock(&ddc->mtx);
+
+       hdmi_ddc_request(ddc);
+       ret = vddc_read(ddc, DDC2_CLOK, b_dev, b_data_addr, SIF_8_BIT_HDMI,
+                     pr_data, b_data_count);
+       mutex_unlock(&ddc->mtx);
+
+       return ret == b_data_count;
+}
+
+static void fg_ddc_data_write(struct mtk_hdmi_ddc *ddc,
+                                      unsigned char b_dev,
+                                      unsigned char b_data_addr,
+                                      unsigned char b_data_count,
+                                      unsigned char *pr_data)
+{
+       unsigned int i;
+
+       mutex_lock(&ddc->mtx);
+
+       hdmi_ddc_request(ddc);
+       for (i = 0; i < b_data_count; i++)
+               mtk_ddc_wr_one(ddc, b_dev, b_data_addr + i, *(pr_data + i));
+
+       mutex_unlock(&ddc->mtx);
+}
+
+static int mtk_hdmi_ddc_xfer(struct i2c_adapter *adapter, struct i2c_msg *msgs,
+                            int num)
+{
+       struct mtk_hdmi_ddc *ddc = adapter->algo_data;
+       struct device *dev = adapter->dev.parent;
+       bool ret;
+       int i;
+       unsigned char offset;
+
+       if (!ddc)
+               return -EINVAL;
+
+       for (i = 0; i < num; i++) {
+               struct i2c_msg *msg = &msgs[i];
+
+               if (msg->flags & I2C_M_RD) {
+                       /* The underlying DDC hardware always issue a write 
request
+                        * that assigns the read offset as part of the read 
operation.
+                        * Therefore we need to use the offset value assigned
+                        * in the previous write request from the drm_edid.c
+                        */
+                       ret = fg_ddc_data_read(ddc, msg->addr,
+                                              offset, /* determined by 
previous write requests */
+                                              (msg->len), &msg->buf[0]);
+                       if (!ret) {
+                               dev_err(dev, "ddc read failed : %d\n", ret);
+                               return ret;
+                       }
+               } else {
+                       fg_ddc_data_write(ddc, msg->addr, msg->buf[0],
+                                               (msg->len - 1), &msg->buf[1]);
+
+                       /* we store the offset value requested by drm_edid 
framework
+                        * to use in subsequent read requests.
+                        */
+                       if (DDC_ADDR == msg->addr && 1 == msg->len)
+                               offset = msg->buf[0];
+               }
+       }
+
+       return i;
+}
+
+static u32 mtk_hdmi_ddc_func(struct i2c_adapter *adapter)
+{
+       return I2C_FUNC_I2C | I2C_FUNC_SMBUS_EMUL;
+}
+
+static const struct i2c_algorithm mtk_hdmi_ddc_algorithm = {
+       .master_xfer = mtk_hdmi_ddc_xfer,
+       .functionality = mtk_hdmi_ddc_func,
+};
+
+static int mtk_hdmi_ddc_probe(struct platform_device *pdev)
+{
+       struct device *dev = &pdev->dev;
+       struct device_node *hdmi;
+       struct mtk_hdmi_ddc *ddc;
+       int ret;
+
+       ddc = devm_kzalloc(dev, sizeof(struct mtk_hdmi_ddc), GFP_KERNEL);
+       if (!ddc)
+               return -ENOMEM;
+
+       ddc->dev = dev;
+       hdmi = of_parse_phandle(dev->of_node, "mediatek,hdmi", 0);
+       if (IS_ERR_OR_NULL(hdmi))
+               return dev_err_probe(dev, -ENODEV, "mediatek hdmi not found");
+
+       ddc->regs = device_node_to_regmap(hdmi);
+       of_node_put(hdmi);
+       if (IS_ERR(ddc->regs))
+               return dev_err_probe(dev, PTR_ERR(ddc->regs), "Unable to get 
hdmi syscon");
+
+       ddc->clk = devm_clk_get_enabled(dev, NULL);
+       if (IS_ERR(ddc->clk))
+               return dev_err_probe(dev, PTR_ERR(ddc->clk), "get ddc_clk 
failed: %p ,\n", ddc->clk);
+
+       mutex_init(&ddc->mtx);
+
+       strscpy(ddc->adap.name, "mediatek-hdmi-ddc", sizeof(ddc->adap.name));
+       ddc->adap.owner = THIS_MODULE;
+       ddc->adap.class = I2C_CLASS_DDC;
+       ddc->adap.algo = &mtk_hdmi_ddc_algorithm;
+       ddc->adap.retries = 3;
+       ddc->adap.dev.of_node = dev->of_node;
+       ddc->adap.algo_data = ddc;
+       ddc->adap.dev.parent = &pdev->dev;
+
+       ret = devm_i2c_add_adapter(dev, &ddc->adap);
+       if (ret < 0) {
+               clk_disable_unprepare(ddc->clk);
+               return dev_err_probe(dev, ret, "failed to add bus to i2c 
core\n");
+       }
+
+       platform_set_drvdata(pdev, ddc);
+       return 0;
+}
+
+static int mtk_hdmi_ddc_remove(struct platform_device *pdev)
+{
+       struct mtk_hdmi_ddc *ddc = platform_get_drvdata(pdev);
+
+       mutex_destroy(&ddc->mtx);
+       clk_disable_unprepare(ddc->clk);
+
+       return 0;
+}
+
+static const struct of_device_id mtk_hdmi_ddc_match[] = {
+       {
+               .compatible = "mediatek,mt8195-hdmi-ddc",
+       },
+       {},
+};
+
+struct platform_driver mtk_hdmi_ddc_v2_driver = {
+       .probe = mtk_hdmi_ddc_probe,
+       .remove = mtk_hdmi_ddc_remove,
+       .driver = {
+               .name = "mediatek-hdmi-mt8195-ddc",
+               .of_match_table = mtk_hdmi_ddc_match,
+       },
+};
+
+MODULE_AUTHOR("Can Zeng <can.z...@mediatek.com>");
+MODULE_DESCRIPTION("MediaTek HDMI DDC Driver");
+MODULE_LICENSE("GPL v2");
diff --git a/drivers/gpu/drm/mediatek/mtk_hdmi_regs_v2.h 
b/drivers/gpu/drm/mediatek/mtk_hdmi_regs_v2.h
new file mode 100644
index 000000000000..7cce175e6b27
--- /dev/null
+++ b/drivers/gpu/drm/mediatek/mtk_hdmi_regs_v2.h
@@ -0,0 +1,276 @@
+/* SPDX-License-Identifier: GPL-2.0 */
+/*
+ * Copyright (c) 2021 MediaTek Inc.
+ * Copyright (c) 2021 BayLibre, SAS
+ */
+
+#ifndef _MTK_HDMI_REGS_H
+#define _MTK_HDMI_REGS_H
+
+#define AIP_CTRL BIT(10)
+#define AIP_CTS_SVAL 0x408
+#define AIP_I2S_CHST0 0x414
+#define AIP_I2S_CHST1 0x418
+#define AIP_I2S_CTRL 0x410
+#define AIP_N_VAL 0x404
+#define AIP_SPDIF_CTRL 0x40C
+#define AIP_TPI_CTRL 0x428
+#define AIP_TXCTRL 0x424
+
+#define AUD_DIS (0x0)
+#define AUD_DIS_WR (0x0)
+#define AUD_EN BIT(2)
+#define AUD_EN_WR BIT(18)
+#define AUD_ERR_THRESH GENMASK(29, 24)
+#define AUD_IN_EN BIT(8)
+#define AUD_MUTE_DIS (0x0)
+#define AUD_MUTE_FIFO_EN BIT(5)
+#define AUD_PACKET_DROP BIT(6)
+#define AUD_RPT_DIS (0x0)
+#define AUD_RPT_EN BIT(2)
+#define AUD_SEL_OWRT BIT(9)
+
+#define AVI_DIS (0)
+#define AVI_DIS_WR (0)
+#define AVI_EN (0x1)
+#define AVI_EN_WR BIT(16)
+#define AVI_HEADER GENMASK(23, 0)
+#define AVI_PKT00 GENMASK(31, 0)
+#define AVI_PKT01 GENMASK(23, 0)
+#define AVI_PKT02 GENMASK(31, 0)
+#define AVI_PKT03 GENMASK(23, 0)
+#define AVI_PKT04 GENMASK(31, 0)
+#define AVI_PKT05 GENMASK(23, 0)
+#define AVI_RPT_DIS (0x0)
+#define AVI_RPT_EN (0x1)
+
+#define C422_C420_CONFIG_BYPASS BIT(5)
+#define C422_C420_CONFIG_ENABLE BIT(4)
+#define C422_C420_CONFIG_OUT_CB_OR_CR BIT(6)
+#define C444_C422_CONFIG_ENABLE (0x1)
+
+#define CBIT_ORDER_SAME BIT(13)
+
+#define CLEAR_FIFO 0x9
+
+#define CLOCK_SCL 0xA
+
+#define CP_CLR_MUTE_EN BIT(1)
+#define CP_EN BIT(5)
+#define CP_EN_WR BIT(21)
+#define CP_RPT_EN BIT(5)
+#define CP_SET_MUTE_DIS (0)
+#define CP_SET_MUTE_EN (1)
+
+#define CTS_CAL_N4 BIT(23)
+#define CTS_REQ_EN BIT(1)
+#define CTS_SW_SEL (1)
+
+#define C_SD0 (0x0)
+#define C_SD1 BIT(4)
+#define C_SD2 BIT(9)
+#define C_SD3 GENMASK(14, 12)
+#define C_SD4 BIT(19)
+#define C_SD5 (0x5 << 20)
+#define C_SD6 GENMASK(26, 25)
+#define C_SD7 GENMASK(30, 28)
+
+#define DATA_DIR_LSB BIT(9)
+#define DATA_DIR_MSB (0)
+
+#define DDC_CMD GENMASK(31, 28)
+#define DDC_CTRL 0xC10
+#define DDC_DATA_OUT GENMASK(23, 16)
+#define DDC_DELAY_CNT GENMASK(31, 16)
+#define DDC_DIN_CNT GENMASK(25, 16)
+#define DDC_I2C_BUS_LOW BIT(11)
+#define DDC_I2C_IN_PROG BIT(13)
+#define DDC_I2C_NO_ACK BIT(10)
+#define DDC_OFFSET GENMASK(15, 8)
+#define DDC_SEGMENT GENMASK(15, 8)
+
+#define DEEPCOLOR_MODE_8BIT (0)
+#define DEEPCOLOR_MODE_MASKBIT GENMASK(10, 8)
+#define DEEPCOLOR_PAT_EN BIT(12)
+#define DEEP_COLOR_ADD BIT(4)
+
+#define DOWNSAMPLE 0x2
+
+#define DSD_EN BIT(15)
+#define DSD_MUTE_DATA BIT(7)
+
+#define ENH_READ_NO_ACK 0x4
+
+#define FIFO0_MAP GENMASK(1, 0)
+#define FIFO1_MAP GENMASK(3, 2)
+#define FIFO2_MAP GENMASK(5, 4)
+#define FIFO3_MAP GENMASK(7, 6)
+
+#define FS_OVERRIDE_WRITE BIT(1)
+#define FS_UNOVERRIDE (0)
+
+#define HBRA_ON BIT(14)
+
+#define HBR_FROM_SPDIF BIT(20)
+
+#define HDCP1X_CTRL 0xCD0
+#define HDCP1X_ENC_EN BIT(6)
+#define HDCP2X_CTRL_0 0xC20
+#define HDCP2X_DDCM_STATUS 0xC68
+#define HDCP2X_DIS_POLL_EN BIT(16)
+#define HDCP2X_EN (0x1)
+#define HDCP2X_ENCRYPT_EN BIT(7)
+#define HDCP2X_HPD_OVR BIT(10)
+#define HDCP2X_HPD_SW BIT(11)
+#define HDCP2X_POL_CTRL 0xC54
+#define HDCP2X_RX_REAUTH_REQ_DDCM_INT_UNMASK BIT(25)
+#define HDCP_TOP_CTRL 0xC00
+
+#define HDMI2_ON BIT(2)
+
+#define HDMITX_CONFIG 0x900
+#define HDMITX_SW_HPD BIT(29)
+#define HDMITX_SW_RSTB BIT(31)
+
+#define HDMI_MODE_DVI (0)
+#define HDMI_MODE_HDMI BIT(3)
+
+#define HDMI_YUV420_MODE BIT(10)
+
+#define HPD_DDC_CTRL 0xC08
+#define HPD_DDC_STATUS 0xC60
+
+#define HPD_PIN_STA BIT(4)
+
+#define HTPLG_F_INT_STA BIT(1)
+#define HTPLG_R_INT_STA BIT(0)
+
+#define I2S2DSD_EN BIT(30)
+#define I2S_1ST_BIT_NOSHIFT BIT(8)
+#define I2S_EN GENMASK(19, 16)
+
+#define JUSTIFY_RIGHT BIT(10)
+
+#define LAYOUT BIT(18)
+#define LAYOUT0 0
+#define LAYOUT1 BIT(4)
+
+#define LFE_CC_SWAP BIT(1)
+
+#define MAP_SD0 0x0
+#define MAP_SD1 0x1
+#define MAP_SD2 0x2
+#define MAP_SD3 0x3
+
+#define MAX_1UI_WRITE GENMASK(15, 8)
+#define MAX_2UI_WRITE GENMASK(23, 16)
+
+#define MCLK_128FS 0x0
+#define MCLK_256FS 0x1
+#define MCLK_384FS 0x2
+#define MCLK_512FS 0x3
+#define MCLK_768FS 0x4
+#define MCLK_CTSGEN_SEL 0
+#define MCLK_EN BIT(2)
+#define NO_MCLK_CTSGEN_SEL BIT(3)
+
+#define NULL_PKT_EN BIT(2)
+#define NULL_PKT_VSYNC_HIGH_EN BIT(3)
+
+#define OUTPUT_FORMAT_DEMUX_420_ENABLE BIT(10)
+
+#define PORD_F_INT_STA BIT(3)
+#define PORD_PIN_STA BIT(5)
+#define PORD_R_INT_STA BIT(2)
+
+#define REG_VMUTE_EN BIT(16)
+#define RST4AUDIO BIT(0)
+#define RST4AUDIO_ACR BIT(2)
+#define RST4AUDIO_FIFO BIT(1)
+
+#define SCDC_CTRL 0xC18
+
+#define SCK_EDGE_RISE BIT(14)
+
+#define SCR_ON BIT(4)
+
+#define SEQ_READ_NO_ACK 0x2
+#define SEQ_WRITE_REQ_ACK 0x7
+
+#define SI2C_ADDR GENMASK(23, 16)
+#define SI2C_ADDR_READ (0xF4)
+#define SI2C_CONFIRM_READ BIT(2)
+#define SI2C_CTRL 0xCAC
+#define SI2C_RD BIT(1)
+#define SI2C_WDATA GENMASK(15, 8)
+#define SI2C_WR BIT(0)
+
+#define SPDIF_EN BIT(13)
+#define SPDIF_HEADER GENMASK(23, 0)
+#define SPDIF_INTERNAL_MODULE BIT(24)
+#define SPDIF_PKT00 GENMASK(31, 0)
+#define SPDIF_PKT01 GENMASK(23, 0)
+#define SPDIF_PKT02 GENMASK(31, 0)
+#define SPDIF_PKT03 GENMASK(23, 0)
+#define SPDIF_PKT04 GENMASK(31, 0)
+#define SPDIF_PKT05 GENMASK(23, 0)
+#define SPDIF_PKT06 GENMASK(31, 0)
+#define SPDIF_PKT07 GENMASK(23, 0)
+
+#define SPD_DIS 0
+#define SPD_DIS_WR 0
+#define SPD_EN BIT(1)
+#define SPD_EN_WR BIT(17)
+#define SPD_RPT_DIS 0
+#define SPD_RPT_EN BIT(1)
+
+#define TOP_AIF_HEADER 0x040
+#define TOP_AIF_PKT00 0x044
+#define TOP_AIF_PKT01 0x048
+#define TOP_AIF_PKT02 0x04C
+#define TOP_AIF_PKT03 0x050
+#define TOP_AUD_MAP 0x00C
+#define TOP_AVI_HEADER 0x024
+#define TOP_AVI_PKT00 0x028
+#define TOP_AVI_PKT01 0x02C
+#define TOP_AVI_PKT02 0x030
+#define TOP_AVI_PKT03 0x034
+#define TOP_AVI_PKT04 0x038
+#define TOP_AVI_PKT05 0x03C
+#define TOP_CFG00 0x000
+#define TOP_CFG01 0x004
+#define TOP_INFO_EN 0x01C
+#define TOP_INFO_RPT 0x020
+#define TOP_INT_CLR00 0x1B8
+#define TOP_INT_CLR01 0x1BC
+#define TOP_INT_MASK00 0x1B0
+#define TOP_INT_MASK01 0x1B4
+#define TOP_INT_STA00 0x1A8
+#define TOP_MISC_CTLR 0x1A4
+#define TOP_SPDIF_HEADER 0x054
+#define TOP_SPDIF_PKT00 0x058
+#define TOP_SPDIF_PKT01 0x05C
+#define TOP_SPDIF_PKT02 0x060
+#define TOP_SPDIF_PKT03 0x064
+#define TOP_SPDIF_PKT04 0x068
+#define TOP_SPDIF_PKT05 0x06C
+#define TOP_SPDIF_PKT06 0x070
+#define TOP_SPDIF_PKT07 0x074
+#define TOP_VMUTE_CFG1 0x1C8
+
+#define TPI_AUDIO_LOOKUP_DIS 0
+#define TPI_AUDIO_LOOKUP_EN BIT(2)
+
+#define VBIT_COM BIT(12)
+#define VBIT_PCM 0
+
+#define VID_DOWNSAMPLE_CONFIG 0x8F0
+#define VID_OUT_FORMAT 0x8FC
+
+#define WR_1UI_LOCK BIT(0)
+#define WR_1UI_UNLOCK 0
+#define WR_2UI_LOCK BIT(2)
+#define WR_2UI_UNLOCK 0
+#define WS_HIGH BIT(11)
+
+#endif /* _MTK_HDMI_REGS_H */
diff --git a/drivers/gpu/drm/mediatek/mtk_hdmi_v2.c 
b/drivers/gpu/drm/mediatek/mtk_hdmi_v2.c
new file mode 100644
index 000000000000..98b1d50ecd72
--- /dev/null
+++ b/drivers/gpu/drm/mediatek/mtk_hdmi_v2.c
@@ -0,0 +1,1105 @@
+// SPDX-License-Identifier: GPL-2.0
+/*
+ * Copyright (c) 2022 MediaTek Inc.
+ * Copyright (c) 2022 BayLibre, SAS
+ */
+
+#include <linux/arm-smccc.h>
+#include <linux/clk-provider.h>
+#include <linux/clk.h>
+#include <linux/debugfs.h>
+#include <linux/delay.h>
+#include <linux/io.h>
+#include <linux/kernel.h>
+#include <linux/mfd/syscon.h>
+#include <linux/of.h>
+#include <linux/of_gpio.h>
+#include <linux/of_graph.h>
+#include <linux/of_irq.h>
+#include <linux/of_platform.h>
+#include <linux/pm_wakeup.h>
+#include <linux/timer.h>
+
+#include <drm/display/drm_scdc_helper.h>
+#include <drm/drm_displayid.h>
+#include <drm/drm_edid.h>
+#include <drm/drm_print.h>
+#include <drm/drm_probe_helper.h>
+
+#include "mtk_drm_crtc.h"
+#include "mtk_hdmi_common.h"
+#include "mtk_hdmi_regs_v2.h"
+#include "mtk_hdmi_v2.h"
+
+#define RGB444_8bit BIT(0)
+#define RGB444_10bit BIT(1)
+#define RGB444_12bit BIT(2)
+#define RGB444_16bit BIT(3)
+
+#define YCBCR444_8bit BIT(4)
+#define YCBCR444_10bit BIT(5)
+#define YCBCR444_12bit BIT(6)
+#define YCBCR444_16bit BIT(7)
+
+#define YCBCR422_8bit_NO_SUPPORT BIT(8)
+#define YCBCR422_10bit_NO_SUPPORT BIT(9)
+#define YCBCR422_12bit BIT(10)
+#define YCBCR422_16bit_NO_SUPPORT BIT(11)
+
+#define YCBCR420_8bit BIT(12)
+#define YCBCR420_10bit BIT(13)
+#define YCBCR420_12bit BIT(14)
+#define YCBCR420_16bit BIT(15)
+
+#define BYTES_TO_UINT32(msb, b1, b2, lsb)                                      
\
+       ((((msb) & 0xff) << 24) + (((b1) & 0xff) << 16) + (((b2) & 0xff) << 8) 
+     \
+        (((lsb) & 0xff)))
+
+const char *const mtk_hdmi_clk_names_v2[MTK_HDMIV2_CLK_COUNT] = {
+    [MTK_HDMIV2_CLK_HDMI_APB_SEL] = "hdmi_apb_sel",
+    [MTK_HDMIV2_HDCP_SEL] = "hdcp_sel",
+    [MTK_HDMIV2_HDCP_24M_SEL] = "hdcp24_sel",
+    [MTK_HDMIV2_VPP_SPLIT_HDMI] = "split_hdmi",
+};
+
+static inline void mtk_hdmi_clr_all_int_status(struct mtk_hdmi *hdmi)
+{
+       /*clear all tx irq*/
+       regmap_write(hdmi->regs, TOP_INT_CLR00, 0xffffffff);
+       regmap_write(hdmi->regs, TOP_INT_CLR00, 0x00000000);
+       regmap_write(hdmi->regs, TOP_INT_CLR01, 0xffffffff);
+       regmap_write(hdmi->regs, TOP_INT_CLR01, 0x00000000);
+}
+
+static inline void mtk_hdmi_disable_all_int(struct mtk_hdmi *hdmi)
+{
+       /*disable all tx irq*/
+       regmap_write(hdmi->regs, TOP_INT_MASK00, 0);
+       regmap_write(hdmi->regs, TOP_INT_MASK01, 0);
+}
+
+static inline void mtk_hdmi_en_hdcp_reauth_int(struct mtk_hdmi *hdmi,
+                                              bool enable)
+{
+       u32 val = 0;
+       if (enable)
+               val = HDCP2X_RX_REAUTH_REQ_DDCM_INT_UNMASK;
+
+       regmap_update_bits(hdmi->regs, TOP_INT_MASK00, 
HDCP2X_RX_REAUTH_REQ_DDCM_INT_UNMASK, val);
+}
+
+static inline void mtk_hdmi_enable_hpd_pord_irq(struct mtk_hdmi *hdmi,
+                                               bool enable)
+{
+       u32 val = 0;
+       if (enable)
+               val = 0xf;
+
+       regmap_update_bits(hdmi->regs, TOP_INT_MASK00, 0xf, val);
+}
+
+static inline void mtk_hdmi_clr_htplg_pord_irq(struct mtk_hdmi *hdmi)
+{
+       regmap_update_bits(hdmi->regs, TOP_INT_CLR00, 0xf, 0xf);
+       regmap_update_bits(hdmi->regs, TOP_INT_CLR00, 0xf, 0);
+}
+
+static inline void mtk_hdmi_set_sw_hpd(struct mtk_hdmi *hdmi, bool high)
+{
+       u32 val = 0;
+       if (high)
+               val = HDMITX_SW_HPD;
+
+       regmap_update_bits(hdmi->regs, HDMITX_CONFIG, HDMITX_SW_HPD, val);
+}
+
+static inline void mtk_hdmi_force_hdcp_hpd(struct mtk_hdmi *hdmi)
+{
+       /* force HDCP HPD to 1*/
+       regmap_update_bits(hdmi->regs, HDCP2X_CTRL_0, HDCP2X_HPD_OVR, 
HDCP2X_HPD_OVR);
+       regmap_update_bits(hdmi->regs, HDCP2X_CTRL_0, HDCP2X_HPD_SW, 
HDCP2X_HPD_SW);
+}
+
+static void mtk_hdmi_disable_hdcp_encrypt(struct mtk_hdmi *hdmi)
+{
+       regmap_update_bits(hdmi->regs, HDCP2X_CTRL_0, HDCP2X_ENCRYPT_EN, 0);
+       regmap_update_bits(hdmi->regs, HDCP1X_CTRL, HDCP1X_ENC_EN, 0);
+}
+
+static inline void mtk_hdmi_enable_scrambling(struct mtk_hdmi *hdmi,
+                                             bool enable)
+{
+       usleep_range(100, 150);
+
+       u32 val = 0;
+       if (enable)
+               val = SCR_ON | HDMI2_ON;
+
+       regmap_update_bits(hdmi->regs, TOP_CFG00, SCR_ON | HDMI2_ON, val);
+
+       drm_scdc_set_scrambling(hdmi->curr_conn, enable);
+       drm_scdc_set_high_tmds_clock_ratio(hdmi->curr_conn, enable);
+}
+
+static void mtk_hdmi_hw_vid_black(struct mtk_hdmi *hdmi, bool black)
+{
+       u32 val = 0;
+       if (black)
+               val = REG_VMUTE_EN;
+
+       regmap_update_bits(hdmi->regs, TOP_VMUTE_CFG1, REG_VMUTE_EN, val);
+}
+
+static void mtk_hdmi_hw_reset(struct mtk_hdmi *hdmi)
+{
+       regmap_update_bits(hdmi->regs, HDMITX_CONFIG, HDMITX_SW_RSTB, 0);
+       udelay(5);
+       regmap_update_bits(hdmi->regs, HDMITX_CONFIG, HDMITX_SW_RSTB, 
HDMITX_SW_RSTB);
+}
+
+static void mtk_hdmi_enable_hdmi_mode(struct mtk_hdmi *hdmi, bool enable)
+{
+       u32 val = HDMI_MODE_DVI;
+       if (enable)
+               val = HDMI_MODE_HDMI;
+
+       regmap_update_bits(hdmi->regs, TOP_CFG00, HDMI_MODE_HDMI, val);
+}
+
+static void mtk_hdmi_hw_audio_infoframe(struct mtk_hdmi *hdmi, u8 *buffer,
+                                       u8 len)
+{
+       enum hdmi_infoframe_type frame_type;
+       u8 frame_ver;
+       u8 frame_len;
+       u8 checksum;
+
+       frame_type = buffer[0];
+       frame_ver = buffer[1];
+       frame_len = buffer[2];
+       checksum = buffer[3];
+
+       regmap_update_bits(hdmi->regs, TOP_INFO_EN, AUD_EN_WR | AUD_EN, 
AUD_DIS_WR | AUD_DIS);
+       regmap_update_bits(hdmi->regs, TOP_INFO_RPT, AUD_RPT_EN, AUD_RPT_DIS);
+
+       regmap_write(hdmi->regs, TOP_AIF_HEADER,
+                      BYTES_TO_UINT32(0, frame_len, frame_ver, frame_type));
+       regmap_write(hdmi->regs, TOP_AIF_PKT00,
+                      BYTES_TO_UINT32(buffer[6], buffer[5], buffer[4],
+                                      buffer[3]));
+       regmap_write(hdmi->regs, TOP_AIF_PKT01,
+                      BYTES_TO_UINT32(0, 0, buffer[8], buffer[7]));
+       regmap_write(hdmi->regs, TOP_AIF_PKT02, 0);
+       regmap_write(hdmi->regs, TOP_AIF_PKT03, 0);
+       regmap_update_bits(hdmi->regs, TOP_INFO_RPT, AUD_RPT_EN, AUD_RPT_EN);
+       regmap_update_bits(hdmi->regs, TOP_INFO_EN, AUD_EN_WR | AUD_EN, 
AUD_EN_WR | AUD_EN);
+}
+
+static void mtk_hdmi_hw_avi_infoframe(struct mtk_hdmi *hdmi, u8 *buffer, u8 
len)
+{
+       regmap_update_bits(hdmi->regs, TOP_INFO_EN, AVI_EN_WR | AVI_EN, 
AVI_DIS_WR | AVI_DIS);
+       regmap_update_bits(hdmi->regs, TOP_INFO_RPT, AVI_RPT_EN, AVI_RPT_DIS);
+
+       regmap_write(hdmi->regs, TOP_AVI_HEADER,
+                      BYTES_TO_UINT32(0, buffer[2], buffer[1], buffer[0]));
+
+       regmap_write(hdmi->regs, TOP_AVI_PKT00,
+                      BYTES_TO_UINT32(buffer[6], buffer[5], buffer[4],
+                                      buffer[3]));
+
+       regmap_write(hdmi->regs, TOP_AVI_PKT01,
+                      BYTES_TO_UINT32(0, buffer[9], buffer[8], buffer[7]));
+
+       regmap_write(hdmi->regs, TOP_AVI_PKT02,
+                      BYTES_TO_UINT32(buffer[13], buffer[12], buffer[11],
+                                      buffer[10]));
+
+       regmap_write(hdmi->regs, TOP_AVI_PKT03,
+                      BYTES_TO_UINT32(0, buffer[16], buffer[15], buffer[14]));
+
+       regmap_write(hdmi->regs, TOP_AVI_PKT04, 0);
+       regmap_write(hdmi->regs, TOP_AVI_PKT05, 0);
+
+       regmap_update_bits(hdmi->regs, TOP_INFO_RPT, AVI_RPT_EN, AVI_RPT_EN);
+       regmap_update_bits(hdmi->regs, TOP_INFO_EN, AVI_EN_WR | AVI_EN, 
AVI_EN_WR | AVI_EN);
+}
+
+static void mtk_hdmi_hw_spd_infoframe(struct mtk_hdmi *hdmi, u8 *buffer, u8 
len)
+{
+       regmap_update_bits(hdmi->regs, TOP_INFO_EN, SPD_EN_WR | SPD_EN, 
SPD_DIS_WR | SPD_DIS);
+       regmap_update_bits(hdmi->regs, TOP_INFO_RPT, SPD_RPT_EN, SPD_RPT_DIS);
+
+       regmap_write(hdmi->regs, TOP_SPDIF_HEADER,
+                      BYTES_TO_UINT32(0, buffer[2], buffer[1], buffer[0]));
+
+       regmap_write(hdmi->regs, TOP_SPDIF_PKT00,
+                      BYTES_TO_UINT32(buffer[6], buffer[5], buffer[4], 
buffer[3]));
+
+       regmap_write(hdmi->regs, TOP_SPDIF_PKT01,
+                      BYTES_TO_UINT32(0, buffer[9], buffer[8], buffer[7]));
+
+       regmap_write(hdmi->regs, TOP_SPDIF_PKT02,
+                      BYTES_TO_UINT32(buffer[13], buffer[12], buffer[11], 
buffer[10]));
+
+       regmap_write(hdmi->regs, TOP_SPDIF_PKT03,
+                      BYTES_TO_UINT32(0, buffer[16], buffer[15], buffer[14]));
+
+       regmap_write(hdmi->regs, TOP_SPDIF_PKT04,
+                      BYTES_TO_UINT32(buffer[20], buffer[19], buffer[18], 
buffer[17]));
+
+       regmap_write(hdmi->regs, TOP_SPDIF_PKT05,
+                      BYTES_TO_UINT32(0, buffer[23], buffer[22], buffer[21]));
+
+       regmap_write(hdmi->regs, TOP_SPDIF_PKT06,
+                      BYTES_TO_UINT32(buffer[27], buffer[26], buffer[25], 
buffer[24]));
+
+       regmap_write(hdmi->regs, TOP_SPDIF_PKT07,
+                      BYTES_TO_UINT32(0, 0, 0, buffer[28]));
+
+       regmap_update_bits(hdmi->regs, TOP_INFO_RPT, SPD_RPT_EN, SPD_RPT_EN);
+       regmap_update_bits(hdmi->regs, TOP_INFO_EN, SPD_EN_WR | SPD_EN, 
SPD_EN_WR | SPD_EN);
+}
+
+static int mtk_hdmi_setup_audio_infoframe(struct mtk_hdmi *hdmi)
+{
+       struct hdmi_codec_params *params = &hdmi->aud_param.codec_params;
+       struct hdmi_audio_infoframe frame;
+       u8 buffer[14];
+       ssize_t err;
+
+       memcpy(&frame, &params->cea, sizeof(struct hdmi_audio_infoframe));
+
+       err = hdmi_audio_infoframe_pack(&frame, buffer, sizeof(buffer));
+       if (err < 0)
+               return err;
+
+       mtk_hdmi_hw_audio_infoframe(hdmi, buffer, sizeof(buffer));
+       return 0;
+}
+
+static void mtk_hdmi_hw_send_aud_packet(struct mtk_hdmi *hdmi, bool enable)
+{
+       if (enable)
+               regmap_update_bits(hdmi->regs, AIP_TXCTRL, AUD_PACKET_DROP, 0);
+       else
+               regmap_update_bits(hdmi->regs, AIP_TXCTRL, AUD_PACKET_DROP, 
AUD_PACKET_DROP);
+}
+
+static inline void mtk_hdmi_hw_send_av_mute(struct mtk_hdmi *hdmi)
+{
+       /*GCP packet */
+       regmap_update_bits(hdmi->regs, TOP_CFG01, CP_CLR_MUTE_EN, 0);
+       regmap_update_bits(hdmi->regs, TOP_CFG01, CP_SET_MUTE_EN, 0);
+       regmap_update_bits(hdmi->regs, TOP_INFO_RPT, CP_RPT_EN, 0);
+       regmap_update_bits(hdmi->regs, TOP_INFO_EN, CP_EN | CP_EN_WR, 0);
+
+       regmap_update_bits(hdmi->regs, TOP_CFG01, CP_CLR_MUTE_EN, 0);
+       regmap_update_bits(hdmi->regs, TOP_CFG01, CP_SET_MUTE_EN, 
CP_SET_MUTE_EN);
+       regmap_update_bits(hdmi->regs, TOP_INFO_RPT, CP_RPT_EN, CP_RPT_EN);
+       regmap_update_bits(hdmi->regs, TOP_INFO_EN, CP_EN | CP_EN_WR, CP_EN | 
CP_EN_WR);
+}
+
+static inline void mtk_hdmi_hw_send_av_unmute(struct mtk_hdmi *hdmi)
+{
+       /*GCP packet */
+       regmap_update_bits(hdmi->regs, TOP_CFG01, CP_CLR_MUTE_EN, 0);
+       regmap_update_bits(hdmi->regs, TOP_CFG01, CP_SET_MUTE_EN, 0);
+       regmap_update_bits(hdmi->regs, TOP_INFO_RPT, CP_RPT_EN, 0);
+       regmap_update_bits(hdmi->regs, TOP_INFO_EN, CP_EN | CP_EN_WR, 0);
+
+       regmap_update_bits(hdmi->regs, TOP_CFG01, CP_CLR_MUTE_EN, 
CP_CLR_MUTE_EN);
+       regmap_update_bits(hdmi->regs, TOP_CFG01, CP_SET_MUTE_DIS, 0);
+       regmap_update_bits(hdmi->regs, TOP_INFO_RPT, CP_RPT_EN, CP_RPT_EN);
+       regmap_update_bits(hdmi->regs, TOP_INFO_EN, CP_EN | CP_EN_WR, CP_EN | 
CP_EN_WR);
+}
+
+static void mtk_hdmi_hw_ncts_enable(struct mtk_hdmi *hdmi, bool enable)
+{
+       unsigned int data;
+
+       regmap_read(hdmi->regs, AIP_CTRL, &data);
+
+       if (enable)
+               data |= CTS_SW_SEL;
+       else
+               data &= ~CTS_SW_SEL;
+
+       regmap_write(hdmi->regs, AIP_CTRL, data);
+}
+
+static void mtk_hdmi_hw_aud_set_channel_status(struct mtk_hdmi *hdmi,
+                                              u8 *channel_status)
+{
+       /* actually, only the first 5 or 7 bytes of Channel Status
+        * contain useful information
+        */
+       regmap_write(hdmi->regs, AIP_I2S_CHST0,
+                      BYTES_TO_UINT32(channel_status[3], channel_status[2],
+                                      channel_status[1], channel_status[0]));
+       regmap_write(hdmi->regs, AIP_I2S_CHST1,
+                      BYTES_TO_UINT32(0, channel_status[6], channel_status[5],
+                                      channel_status[4]));
+}
+
+struct hdmi_acr_n {
+       unsigned int clock;
+       unsigned int n[3];
+};
+
+/* Recommended N values from HDMI specification, tables 7-1 to 7-3 */
+static const struct hdmi_acr_n hdmi_rec_n_table[] = {
+       /* Clock, N: 32kHz 44.1kHz 48kHz */
+       { 25175, { 4576, 7007, 6864 } },
+       { 74176, { 11648, 17836, 11648 } },
+       { 148352, { 11648, 8918, 5824 } },
+       { 296703, { 5824, 4459, 5824 } },
+       { 297000, { 3072, 4704, 5120 } },
+       { 0, { 4096, 6272, 6144 } }, /* all other TMDS clocks */
+};
+
+/**
+ * hdmi_recommended_n() - Return N value recommended by HDMI specification
+ * @freq: audio sample rate in Hz
+ * @clock: rounded TMDS clock in kHz
+ */
+static int hdmi_recommended_n(unsigned int freq, unsigned int clock)
+{
+       const struct hdmi_acr_n *recommended;
+       unsigned int i;
+
+       for (i = 0; i < ARRAY_SIZE(hdmi_rec_n_table) - 1; i++) {
+               if (clock == hdmi_rec_n_table[i].clock)
+                       break;
+       }
+
+       if (i == ARRAY_SIZE(hdmi_rec_n_table))
+               return -EINVAL;
+
+       recommended = hdmi_rec_n_table + i;
+
+       switch (freq) {
+       case 32000:
+               return recommended->n[0];
+       case 44100:
+               return recommended->n[1];
+       case 48000:
+               return recommended->n[2];
+       case 88200:
+               return recommended->n[1] * 2;
+       case 96000:
+               return recommended->n[2] * 2;
+       case 176400:
+               return recommended->n[1] * 4;
+       case 192000:
+               return recommended->n[2] * 4;
+       default:
+               return (128 * freq) / 1000;
+       }
+}
+
+static unsigned int hdmi_mode_clock_to_hz(unsigned int clock)
+{
+       switch (clock) {
+       case 25175:
+               return 25174825; /* 25.2/1.001 MHz */
+       case 74176:
+               return 74175824; /* 74.25/1.001 MHz */
+       case 148352:
+               return 148351648; /* 148.5/1.001 MHz */
+       case 296703:
+               return 296703297; /* 297/1.001 MHz */
+       default:
+               return clock * 1000;
+       }
+}
+
+static unsigned int hdmi_expected_cts(unsigned int audio_sample_rate,
+                                     unsigned int tmds_clock, unsigned int n)
+{
+       return DIV_ROUND_CLOSEST_ULL((u64)hdmi_mode_clock_to_hz(tmds_clock) * n,
+                                    128 * audio_sample_rate);
+}
+
+static void mtk_hdmi_hw_aud_set_ncts(struct mtk_hdmi *hdmi,
+                                    unsigned int sample_rate,
+                                    unsigned int clock)
+{
+       unsigned int ncts;
+       int n;
+
+       n = hdmi_recommended_n(sample_rate, clock);
+
+       if (n == -EINVAL) {
+               DRM_ERROR("Invalid sample rate: %u\n", sample_rate);
+               return;
+       }
+
+       ncts = hdmi_expected_cts(sample_rate, clock, n);
+       regmap_write(hdmi->regs, AIP_N_VAL, n);
+       regmap_write(hdmi->regs, AIP_CTS_SVAL, ncts);
+}
+
+static void mtk_hdmi_aud_enable_packet(struct mtk_hdmi *hdmi, bool enable)
+{
+       mtk_hdmi_hw_send_aud_packet(hdmi, enable);
+}
+
+static void mtk_hdmi_aud_on_off_hw_ncts(struct mtk_hdmi *hdmi, bool on)
+{
+       mtk_hdmi_hw_ncts_enable(hdmi, on);
+}
+
+static void mtk_hdmi_audio_dsd_config(struct mtk_hdmi *hdmi,
+                                     unsigned char chnum, bool dsd_bypass)
+{
+       regmap_update_bits(hdmi->regs, AIP_CTRL, SPDIF_EN | DSD_EN | HBRA_ON, 
DSD_EN);
+       regmap_update_bits(hdmi->regs, AIP_TXCTRL, DSD_MUTE_DATA, 
DSD_MUTE_DATA);
+       if (dsd_bypass)
+               regmap_write(hdmi->regs, TOP_AUD_MAP, 0x75316420);
+       else
+               regmap_write(hdmi->regs, TOP_AUD_MAP, 0x04230150);
+
+       regmap_update_bits(hdmi->regs, AIP_SPDIF_CTRL, I2S2DSD_EN, 0);
+}
+
+static inline void mtk_hdmi_hw_i2s_fifo_map(struct mtk_hdmi *hdmi,
+                                           unsigned int fifo_mapping)
+{
+       regmap_update_bits(hdmi->regs, AIP_I2S_CTRL, FIFO3_MAP | FIFO2_MAP | 
FIFO1_MAP | FIFO0_MAP, fifo_mapping);
+}
+
+static inline void mtk_hdmi_hw_i2s_ch_number(struct mtk_hdmi *hdmi,
+                                            unsigned int chnum)
+{
+       regmap_update_bits(hdmi->regs, AIP_CTRL, I2S_EN, FIELD_PREP(I2S_EN, 
chnum));
+}
+
+static void mtk_hdmi_hw_i2s_ch_mapping(struct mtk_hdmi *hdmi,
+                                      unsigned char chnum,
+                                      unsigned char mapping)
+{
+       unsigned int bdata;
+
+       switch (chnum) {
+       case 2:
+               bdata = 0x1;
+               break;
+       case 3:
+               bdata = 0x3;
+               break;
+       case 6:
+               if (mapping == 0x0E) {
+                       bdata = 0xf;
+                       break;
+               }
+               fallthrough;
+       case 5:
+               bdata = 0x7;
+               break;
+       case 7:
+       case 8:
+               bdata = 0xf;
+               break;
+       default:
+               bdata = 0x1;
+       }
+
+       mtk_hdmi_hw_i2s_fifo_map(hdmi, (MAP_SD3 << 6) | (MAP_SD2 << 4) |
+                                              (MAP_SD1 << 2) | (MAP_SD0 << 0));
+       mtk_hdmi_hw_i2s_ch_number(hdmi, bdata);
+
+       if (chnum == 2)
+               regmap_update_bits(hdmi->regs, AIP_TXCTRL, LAYOUT1, LAYOUT0);
+       else
+               regmap_update_bits(hdmi->regs, AIP_TXCTRL, LAYOUT1, LAYOUT1);
+}
+
+static void mtk_hdmi_i2s_data_fmt(struct mtk_hdmi *hdmi, unsigned char fmt)
+{
+       unsigned int u4Data;
+
+       regmap_read(hdmi->regs, AIP_I2S_CTRL, &u4Data);
+       u4Data &= ~(WS_HIGH | I2S_1ST_BIT_NOSHIFT | JUSTIFY_RIGHT);
+
+       switch (fmt) {
+       case HDMI_I2S_MODE_RJT_24BIT:
+       case HDMI_I2S_MODE_RJT_16BIT:
+               u4Data |= (WS_HIGH | I2S_1ST_BIT_NOSHIFT | JUSTIFY_RIGHT);
+               u4Data |= (WS_HIGH | I2S_1ST_BIT_NOSHIFT | JUSTIFY_RIGHT);
+               break;
+
+       case HDMI_I2S_MODE_LJT_24BIT:
+       case HDMI_I2S_MODE_LJT_16BIT:
+               u4Data |= (WS_HIGH | I2S_1ST_BIT_NOSHIFT);
+               u4Data |= (WS_HIGH | I2S_1ST_BIT_NOSHIFT);
+               break;
+
+       case HDMI_I2S_MODE_I2S_24BIT:
+       case HDMI_I2S_MODE_I2S_16BIT:
+       default:
+               break;
+       }
+       regmap_write(hdmi->regs, AIP_I2S_CTRL, u4Data);
+}
+
+static inline void mtk_hdmi_i2s_sck_edge(struct mtk_hdmi *hdmi,
+                                        unsigned int edge)
+{
+       regmap_update_bits(hdmi->regs, AIP_I2S_CTRL, SCK_EDGE_RISE, edge);
+}
+
+static inline void mtk_hdmi_i2s_cbit_order(struct mtk_hdmi *hdmi,
+                                          unsigned int cbit)
+{
+       regmap_update_bits(hdmi->regs, AIP_I2S_CTRL, CBIT_ORDER_SAME, cbit);
+}
+
+static inline void mtk_hdmi_i2s_vbit(struct mtk_hdmi *hdmi, unsigned int vbit)
+{
+       regmap_update_bits(hdmi->regs, AIP_I2S_CTRL, VBIT_COM, vbit);
+}
+
+static inline void mtk_hdmi_i2s_data_direction(struct mtk_hdmi *hdmi,
+                                              unsigned int data_dir)
+{
+       regmap_update_bits(hdmi->regs, AIP_I2S_CTRL, DATA_DIR_LSB, data_dir);
+}
+
+static inline void mtk_hdmi_hw_audio_type(struct mtk_hdmi *hdmi,
+                                         unsigned int spdif_i2s)
+{
+       regmap_update_bits(hdmi->regs, AIP_CTRL, SPDIF_EN, FIELD_PREP(SPDIF_EN, 
spdif_i2s));
+}
+
+static unsigned char mtk_hdmi_get_i2s_ch_mapping(struct mtk_hdmi *hdmi,
+                                                unsigned char channel_type)
+{
+       unsigned char channelmap = 0x00;
+
+       switch (channel_type) {
+       case HDMI_AUD_CHAN_TYPE_1_1:
+       case HDMI_AUD_CHAN_TYPE_2_1:
+               channelmap = 0x01;
+               break;
+
+       case HDMI_AUD_CHAN_TYPE_3_0:
+               channelmap = 0x02;
+               break;
+
+       case HDMI_AUD_CHAN_TYPE_3_1:
+               channelmap = 0x03;
+               break;
+
+       case HDMI_AUD_CHAN_TYPE_3_0_LRS:
+       case HDMI_AUD_CHAN_TYPE_4_0:
+               channelmap = 0x08;
+               break;
+
+       case HDMI_AUD_CHAN_TYPE_5_1:
+               channelmap = 0x0B;
+               break;
+
+       case HDMI_AUD_CHAN_TYPE_4_1_CLRS:
+       case HDMI_AUD_CHAN_TYPE_6_0:
+       case HDMI_AUD_CHAN_TYPE_6_0_CS:
+       case HDMI_AUD_CHAN_TYPE_6_0_CH:
+       case HDMI_AUD_CHAN_TYPE_6_0_OH:
+       case HDMI_AUD_CHAN_TYPE_6_0_CHR:
+               channelmap = 0x0E;
+               break;
+
+       case HDMI_AUD_CHAN_TYPE_1_0:
+       case HDMI_AUD_CHAN_TYPE_2_0:
+       case HDMI_AUD_CHAN_TYPE_3_1_LRS:
+       case HDMI_AUD_CHAN_TYPE_4_1:
+       case HDMI_AUD_CHAN_TYPE_5_0:
+       case HDMI_AUD_CHAN_TYPE_4_0_CLRS:
+       case HDMI_AUD_CHAN_TYPE_6_1:
+       case HDMI_AUD_CHAN_TYPE_6_1_CS:
+       case HDMI_AUD_CHAN_TYPE_6_1_CH:
+       case HDMI_AUD_CHAN_TYPE_6_1_OH:
+       case HDMI_AUD_CHAN_TYPE_6_1_CHR:
+       case HDMI_AUD_CHAN_TYPE_7_0:
+       case HDMI_AUD_CHAN_TYPE_7_0_LH_RH:
+       case HDMI_AUD_CHAN_TYPE_7_0_LSR_RSR:
+       case HDMI_AUD_CHAN_TYPE_7_0_LC_RC:
+       case HDMI_AUD_CHAN_TYPE_7_0_LW_RW:
+       case HDMI_AUD_CHAN_TYPE_7_0_LSD_RSD:
+       case HDMI_AUD_CHAN_TYPE_7_0_LSS_RSS:
+       case HDMI_AUD_CHAN_TYPE_7_0_LHS_RHS:
+       case HDMI_AUD_CHAN_TYPE_7_0_CS_CH:
+       case HDMI_AUD_CHAN_TYPE_7_0_CS_OH:
+       case HDMI_AUD_CHAN_TYPE_7_0_CS_CHR:
+       case HDMI_AUD_CHAN_TYPE_7_0_CH_OH:
+       case HDMI_AUD_CHAN_TYPE_7_0_CH_CHR:
+       case HDMI_AUD_CHAN_TYPE_7_0_OH_CHR:
+       case HDMI_AUD_CHAN_TYPE_7_0_LSS_RSS_LSR_RSR:
+       case HDMI_AUD_CHAN_TYPE_8_0_LH_RH_CS:
+       case HDMI_AUD_CHAN_TYPE_7_1:
+       case HDMI_AUD_CHAN_TYPE_7_1_LH_RH:
+       case HDMI_AUD_CHAN_TYPE_7_1_LSR_RSR:
+       case HDMI_AUD_CHAN_TYPE_7_1_LC_RC:
+       case HDMI_AUD_CHAN_TYPE_7_1_LW_RW:
+       case HDMI_AUD_CHAN_TYPE_7_1_LSD_RSD:
+       case HDMI_AUD_CHAN_TYPE_7_1_LSS_RSS:
+       case HDMI_AUD_CHAN_TYPE_7_1_LHS_RHS:
+       case HDMI_AUD_CHAN_TYPE_7_1_CS_CH:
+       case HDMI_AUD_CHAN_TYPE_7_1_CS_OH:
+       case HDMI_AUD_CHAN_TYPE_7_1_CS_CHR:
+       case HDMI_AUD_CHAN_TYPE_7_1_CH_OH:
+       case HDMI_AUD_CHAN_TYPE_7_1_CH_CHR:
+       case HDMI_AUD_CHAN_TYPE_7_1_OH_CHR:
+       case HDMI_AUD_CHAN_TYPE_7_1_LSS_RSS_LSR_RSR:
+               channelmap = 0x00;
+               break;
+       }
+
+       return channelmap;
+}
+
+static inline void mtk_hdmi_hw_i2s_ch_swap(struct mtk_hdmi *hdmi,
+                                          unsigned char swapbit)
+{
+       regmap_update_bits(hdmi->regs, AIP_SPDIF_CTRL, 0x0F << 20, swapbit << 
20);
+}
+
+static void mtk_hdmi_hbr_config(struct mtk_hdmi *hdmi, bool dsd_bypass)
+{
+       if (dsd_bypass) {
+               regmap_update_bits(hdmi->regs, AIP_CTRL, SPDIF_EN | DSD_EN | 
HBRA_ON, HBRA_ON);
+               regmap_update_bits(hdmi->regs, AIP_CTRL, I2S_EN, I2S_EN);
+       } else {
+               regmap_update_bits(hdmi->regs, AIP_CTRL, SPDIF_EN | DSD_EN | 
HBRA_ON, SPDIF_EN);
+               regmap_update_bits(hdmi->regs, AIP_CTRL, SPDIF_INTERNAL_MODULE, 
SPDIF_INTERNAL_MODULE);
+               regmap_update_bits(hdmi->regs, AIP_CTRL, HBR_FROM_SPDIF, 
HBR_FROM_SPDIF);
+               regmap_update_bits(hdmi->regs, AIP_CTRL, CTS_CAL_N4, 
CTS_CAL_N4);
+       }
+}
+
+static inline void mtk_hdmi_hw_spdif_config(struct mtk_hdmi *hdmi)
+{
+       regmap_update_bits(hdmi->regs, AIP_SPDIF_CTRL, WR_1UI_LOCK, 
WR_1UI_UNLOCK);
+       regmap_update_bits(hdmi->regs, AIP_SPDIF_CTRL, FS_OVERRIDE_WRITE, 
FS_UNOVERRIDE);
+       regmap_update_bits(hdmi->regs, AIP_SPDIF_CTRL, WR_2UI_LOCK, 
WR_2UI_UNLOCK);
+       regmap_update_bits(hdmi->regs, AIP_SPDIF_CTRL, MAX_1UI_WRITE, 
FIELD_PREP(MAX_1UI_WRITE, 0x4));
+       regmap_update_bits(hdmi->regs, AIP_SPDIF_CTRL, MAX_2UI_WRITE, 
FIELD_PREP(MAX_2UI_WRITE, 0x9));
+       regmap_update_bits(hdmi->regs, AIP_SPDIF_CTRL, AUD_ERR_THRESH, 
FIELD_PREP(AUD_ERR_THRESH, 0x4));
+       regmap_update_bits(hdmi->regs, AIP_SPDIF_CTRL, I2S2DSD_EN, I2S2DSD_EN);
+}
+
+static void mtk_hdmi_aud_set_input(struct mtk_hdmi *hdmi)
+{
+       unsigned char ChMapping;
+
+       regmap_write(hdmi->regs, TOP_AUD_MAP,
+                      C_SD7 + C_SD6 + C_SD5 + C_SD4 + C_SD3 + C_SD2 + C_SD1 +
+                              C_SD0);
+       regmap_update_bits(hdmi->regs, AIP_SPDIF_CTRL, 0x0F << 20, 0);
+       regmap_update_bits(hdmi->regs, AIP_CTRL,
+                     SPDIF_EN | DSD_EN | HBRA_ON | CTS_CAL_N4 |
+                             HBR_FROM_SPDIF | SPDIF_INTERNAL_MODULE, 0);
+       regmap_update_bits(hdmi->regs, AIP_TXCTRL, DSD_MUTE_DATA | LAYOUT1, 0);
+
+       if (hdmi->aud_param.aud_input_type == HDMI_AUD_INPUT_I2S) {
+               if (hdmi->aud_param.aud_codec == HDMI_AUDIO_CODING_TYPE_DSD) {
+                       mtk_hdmi_audio_dsd_config(hdmi, 
hdmi->aud_param.codec_params.channels, 0);
+                       mtk_hdmi_hw_i2s_ch_mapping(hdmi, 
hdmi->aud_param.codec_params.channels, 1);
+               } else {
+                       mtk_hdmi_i2s_data_fmt(hdmi, 
hdmi->aud_param.aud_i2s_fmt);
+                       mtk_hdmi_i2s_sck_edge(hdmi, SCK_EDGE_RISE);
+                       mtk_hdmi_i2s_cbit_order(hdmi, CBIT_ORDER_SAME);
+                       mtk_hdmi_i2s_vbit(hdmi, VBIT_PCM);
+                       mtk_hdmi_i2s_data_direction(hdmi, DATA_DIR_MSB);
+                       mtk_hdmi_hw_audio_type(hdmi, HDMI_AUD_INPUT_I2S);
+                       ChMapping = mtk_hdmi_get_i2s_ch_mapping(hdmi, 
hdmi->aud_param.aud_input_chan_type);
+                       mtk_hdmi_hw_i2s_ch_mapping(hdmi, 
hdmi->aud_param.codec_params.channels, ChMapping);
+                       mtk_hdmi_hw_i2s_ch_swap(hdmi, LFE_CC_SWAP);
+               }
+       } else {
+               if (hdmi->aud_param.aud_input_type == HDMI_AUD_INPUT_SPDIF &&
+                   (hdmi->aud_param.aud_codec ==
+                     HDMI_AUDIO_CODING_TYPE_DTS_HD ||
+                    hdmi->aud_param.aud_codec ==
+                     HDMI_AUDIO_CODING_TYPE_MLP) &&
+                   hdmi->aud_param.codec_params.sample_rate == 768000) {
+                       mtk_hdmi_hbr_config(hdmi, false);
+               } else {
+                       mtk_hdmi_hw_spdif_config(hdmi);
+                       mtk_hdmi_hw_i2s_ch_mapping(hdmi, 2, 0);
+               }
+       }
+}
+
+static void mtk_hdmi_aud_set_sw_ncts(struct mtk_hdmi *hdmi,
+                                    struct drm_display_mode *display_mode)
+{
+       unsigned int sample_rate = hdmi->aud_param.codec_params.sample_rate;
+
+       mtk_hdmi_aud_on_off_hw_ncts(hdmi, false);
+
+       mtk_hdmi_hw_aud_set_ncts(hdmi, sample_rate, display_mode->clock);
+}
+
+static inline void mtk_hdmi_hw_audio_input_enable(struct mtk_hdmi *hdmi,
+                                                 unsigned int enable)
+{
+       if (enable)
+               regmap_update_bits(hdmi->regs, AIP_CTRL, AUD_IN_EN, AUD_IN_EN);
+       else
+               regmap_update_bits(hdmi->regs, AIP_CTRL, AUD_IN_EN, 0);
+}
+
+static void mtk_hdmi_aip_ctrl_init(struct mtk_hdmi *hdmi)
+{
+       regmap_update_bits(hdmi->regs, AIP_CTRL, AUD_SEL_OWRT | 
NO_MCLK_CTSGEN_SEL | MCLK_EN | CTS_REQ_EN, AUD_SEL_OWRT | NO_MCLK_CTSGEN_SEL | 
CTS_REQ_EN);
+       regmap_update_bits(hdmi->regs, AIP_TPI_CTRL, TPI_AUDIO_LOOKUP_EN, 
TPI_AUDIO_LOOKUP_DIS);
+}
+
+static void mtk_hdmi_audio_reset(struct mtk_hdmi *hdmi, bool rst)
+{
+       if (rst)
+               regmap_update_bits(hdmi->regs, AIP_TXCTRL, RST4AUDIO | 
RST4AUDIO_FIFO | RST4AUDIO_ACR, RST4AUDIO | RST4AUDIO_FIFO | RST4AUDIO_ACR);
+       else
+               regmap_update_bits(hdmi->regs, AIP_TXCTRL, RST4AUDIO | 
RST4AUDIO_FIFO | RST4AUDIO_ACR, 0);
+}
+
+static void mtk_hdmi_aud_output_config(struct mtk_hdmi *hdmi,
+                                      struct drm_display_mode *display_mode)
+{
+       mtk_hdmi_aud_enable_packet(hdmi, false);
+       mtk_hdmi_audio_reset(hdmi, true);
+       mtk_hdmi_aip_ctrl_init(hdmi);
+       mtk_hdmi_aud_set_input(hdmi);
+       mtk_hdmi_hw_aud_set_channel_status(hdmi, 
hdmi->aud_param.codec_params.iec.status);
+       mtk_hdmi_setup_audio_infoframe(hdmi);
+       mtk_hdmi_hw_audio_input_enable(hdmi, true);
+       mtk_hdmi_audio_reset(hdmi, false);
+       mtk_hdmi_aud_set_sw_ncts(hdmi, display_mode);
+       usleep_range(25, 50);
+       mtk_hdmi_aud_on_off_hw_ncts(hdmi, true);
+       mtk_hdmi_aud_enable_packet(hdmi, true);
+}
+
+void mtk_hdmi_output_init_v2(struct mtk_hdmi *hdmi)
+{
+       struct hdmi_audio_param *aud_param = &hdmi->aud_param;
+
+       aud_param->aud_codec = HDMI_AUDIO_CODING_TYPE_PCM;
+       aud_param->aud_sampe_size = HDMI_AUDIO_SAMPLE_SIZE_16;
+       aud_param->aud_input_type = HDMI_AUD_INPUT_I2S;
+       aud_param->aud_i2s_fmt = HDMI_I2S_MODE_I2S_24BIT;
+       aud_param->aud_mclk = HDMI_AUD_MCLK_128FS;
+       aud_param->aud_input_chan_type = HDMI_AUD_CHAN_TYPE_2_0;
+
+       hdmi->hpd = HDMI_PLUG_OUT;
+}
+
+static void mtk_hdmi_change_video_resolution(struct mtk_hdmi *hdmi)
+{
+       mtk_hdmi_hw_reset(hdmi);
+       mtk_hdmi_set_sw_hpd(hdmi, true);
+       usleep_range(2, 5);
+
+       regmap_write(hdmi->regs, HDCP_TOP_CTRL, 0x0);
+       mtk_hdmi_en_hdcp_reauth_int(hdmi, true);
+       mtk_hdmi_enable_hpd_pord_irq(hdmi, true);
+       mtk_hdmi_force_hdcp_hpd(hdmi);
+
+       regmap_update_bits(hdmi->regs, TOP_CFG00, DEEPCOLOR_MODE_MASKBIT, 
DEEPCOLOR_MODE_8BIT);
+       regmap_update_bits(hdmi->regs, TOP_CFG00, DEEPCOLOR_PAT_EN, 0);
+       regmap_update_bits(hdmi->regs, TOP_MISC_CTLR, DEEP_COLOR_ADD, 0);
+       mtk_hdmi_enable_hdmi_mode(hdmi, !hdmi->dvi_mode);
+
+       usleep_range(5, 10);
+       mtk_hdmi_hw_vid_black(hdmi, true);
+       mtk_hdmi_hw_send_av_unmute(hdmi);
+
+       regmap_update_bits(hdmi->regs, TOP_CFG01, NULL_PKT_VSYNC_HIGH_EN | 
NULL_PKT_EN, NULL_PKT_VSYNC_HIGH_EN);
+
+       /* enable scrambling if tmds clock is over 340Mhz */
+       mtk_hdmi_enable_scrambling(hdmi, hdmi->mode.clock >= 340 * KILO);
+
+       regmap_update_bits(hdmi->regs, HDMITX_CONFIG, HDMI_YUV420_MODE | 
HDMITX_SW_HPD, 0 | HDMITX_SW_HPD);
+       regmap_update_bits(hdmi->regs, VID_DOWNSAMPLE_CONFIG, 
C444_C422_CONFIG_ENABLE, 0);
+       regmap_update_bits(hdmi->regs, VID_DOWNSAMPLE_CONFIG, 
C422_C420_CONFIG_ENABLE, 0);
+       regmap_update_bits(hdmi->regs, VID_DOWNSAMPLE_CONFIG, 
C422_C420_CONFIG_BYPASS, C422_C420_CONFIG_BYPASS);
+       regmap_update_bits(hdmi->regs, VID_DOWNSAMPLE_CONFIG, 
C422_C420_CONFIG_OUT_CB_OR_CR, 0);
+       regmap_update_bits(hdmi->regs, VID_OUT_FORMAT, 
OUTPUT_FORMAT_DEMUX_420_ENABLE, 0);
+}
+
+static void mtk_hdmi_output_set_display_mode(struct mtk_hdmi *hdmi,
+                                            struct drm_display_mode *mode)
+{
+       int ret;
+       union phy_configure_opts opts = {
+               .dp = { .link_rate = hdmi->mode.clock * KILO}
+       };
+
+       ret = phy_configure(hdmi->phy, &opts);
+       if (ret)
+               dev_err(hdmi->dev, "Setting clock=%d failed: %d", mode->clock, 
ret);
+
+       mtk_hdmi_change_video_resolution(hdmi);
+       mtk_hdmi_aud_output_config(hdmi, mode);
+}
+
+int mtk_hdmi_clk_enable_v2(struct mtk_hdmi *hdmi)
+{
+       int ret;
+
+       ret = clk_prepare_enable(hdmi->clk[MTK_HDMIV2_HDCP_SEL]);
+       if (ret)
+               return ret;
+       ret = clk_prepare_enable(hdmi->clk[MTK_HDMIV2_HDCP_24M_SEL]);
+       if (ret)
+               return ret;
+       ret = clk_prepare_enable(hdmi->clk[MTK_HDMIV2_CLK_HDMI_APB_SEL]);
+       if (ret)
+               return ret;
+       ret = clk_prepare_enable(hdmi->clk[MTK_HDMIV2_VPP_SPLIT_HDMI]);
+
+       return ret;
+}
+
+void mtk_hdmi_clk_disable_v2(struct mtk_hdmi *hdmi)
+{
+       clk_disable_unprepare(hdmi->clk[MTK_HDMIV2_HDCP_SEL]);
+       clk_disable_unprepare(hdmi->clk[MTK_HDMIV2_HDCP_24M_SEL]);
+       clk_disable_unprepare(hdmi->clk[MTK_HDMIV2_VPP_SPLIT_HDMI]);
+}
+
+static void mtk_hdmi_hpd_event(enum hdmi_hpd_state hpd, struct device *dev)
+{
+       struct mtk_hdmi *hdmi = dev_get_drvdata(dev);
+
+       if (hdmi && hdmi->bridge.encoder && hdmi->bridge.encoder->dev)
+               drm_helper_hpd_irq_event(hdmi->bridge.encoder->dev);
+}
+
+static enum hdmi_hpd_state mtk_hdmi_hpd_pord_status(struct mtk_hdmi *hdmi)
+{
+
+       u32 hpd_status, pin_sta, pord_pin_sta;
+
+       regmap_read(hdmi->regs, HPD_DDC_STATUS, &hpd_status);
+       pin_sta = FIELD_GET(HPD_PIN_STA, hpd_status);
+       pord_pin_sta = FIELD_GET(PORD_PIN_STA, hpd_status);
+
+       if (pin_sta && pord_pin_sta)
+               return HDMI_PLUG_IN_AND_SINK_POWER_ON;
+       else if (pord_pin_sta)
+               return HDMI_PLUG_IN_ONLY;
+       else
+               return HDMI_PLUG_OUT;
+}
+
+static irqreturn_t mtk_hdmi_isr(int irq, void *arg)
+{
+       struct mtk_hdmi *hdmi = arg;
+       unsigned int int_status;
+       int ret = IRQ_HANDLED;
+
+       regmap_read(hdmi->regs, TOP_INT_STA00, &int_status);
+
+       /* handle hpd interrupt */
+       if (int_status & (PORD_F_INT_STA | PORD_R_INT_STA | HTPLG_F_INT_STA |
+                         HTPLG_R_INT_STA)) {
+               mtk_hdmi_enable_hpd_pord_irq(hdmi, false);
+               mtk_hdmi_clr_htplg_pord_irq(hdmi);
+               ret = IRQ_WAKE_THREAD;
+       }
+
+       /*clear all tx irq*/
+       mtk_hdmi_clr_all_int_status(hdmi);
+
+       return ret;
+}
+
+static irqreturn_t mtk_hdmi_hpd_work_handle(int irq, void *arg)
+{
+       struct mtk_hdmi *hdmi = arg;
+       enum hdmi_hpd_state hpd;
+
+       /* Debounce HDMI monitor HPD status.
+        * 30ms is the result of empirical testing */
+       msleep(30);
+
+       hpd = mtk_hdmi_hpd_pord_status(hdmi);
+       if (hpd != hdmi->hpd) {
+               hdmi->hpd = hpd;
+               mtk_hdmi_hpd_event(hpd, hdmi->dev);
+       }
+
+       mtk_hdmi_enable_hpd_pord_irq(hdmi, true);
+       return IRQ_HANDLED;
+}
+
+static int mtk_hdmi_enable_disable(struct mtk_hdmi *hdmi, bool enable)
+{
+       int ret;
+
+       if (enable && !hdmi->hdmi_enabled) {
+               if (!hdmi->power_clk_enabled) {
+                       /* power domain on */
+                       ret = pm_runtime_get_sync(hdmi->dev);
+
+                       /* clk on */
+                       mtk_hdmi_clk_enable_v2(hdmi);
+                       hdmi->power_clk_enabled = true;
+               }
+
+               if (!hdmi->irq_registered) {
+                       /* disable all tx interrupts */
+                       mtk_hdmi_disable_all_int(hdmi);
+                       /* request irq */
+                       hdmi->hdmi_irq =
+                               irq_of_parse_and_map(hdmi->dev->of_node, 0);
+                       ret = request_threaded_irq(hdmi->hdmi_irq, mtk_hdmi_isr,
+                                                  mtk_hdmi_hpd_work_handle,
+                                                  IRQF_TRIGGER_HIGH, "hdmiirq",
+                                                  hdmi);
+                       hdmi->irq_registered = true;
+                       /* enable hpd interrupt */
+                       mtk_hdmi_set_sw_hpd(hdmi, true);
+                       mtk_hdmi_enable_hpd_pord_irq(hdmi, true);
+               }
+
+       } else if (!enable && hdmi->hdmi_enabled) {
+               if (hdmi->irq_registered) {
+                       /* free irq */
+                       free_irq(hdmi->hdmi_irq, NULL);
+                       hdmi->irq_registered = false;
+               }
+
+               if (hdmi->power_clk_enabled) {
+                       /* clk disable */
+                       mtk_hdmi_clk_disable_v2(hdmi);
+                       /* power domain off */
+                       ret = pm_runtime_put_sync(hdmi->dev);
+                       hdmi->power_clk_enabled = false;
+               }
+       }
+
+       hdmi->hdmi_enabled = enable;
+
+       return 0;
+}
+
+/*
+ * Bridge callbacks
+ */
+
+static int mtk_hdmi_bridge_attach(struct drm_bridge *bridge,
+                                 enum drm_bridge_attach_flags flags)
+{
+       struct mtk_hdmi *hdmi = hdmi_ctx_from_bridge(bridge);
+       int ret;
+
+       if (!(flags & DRM_BRIDGE_ATTACH_NO_CONNECTOR)) {
+               DRM_ERROR("The flag DRM_BRIDGE_ATTACH_NO_CONNECTOR must be 
supplied\n");
+               return -EINVAL;
+       }
+       if (hdmi->next_bridge) {
+               ret = drm_bridge_attach(bridge->encoder, hdmi->next_bridge, 
bridge, flags);
+               if (ret)
+                       return ret;
+       }
+
+       pm_runtime_enable(hdmi->dev);
+       mtk_hdmi_enable_disable(hdmi, true);
+
+       return 0;
+}
+
+static void mtk_hdmi_bridge_disable(struct drm_bridge *bridge,
+                                   struct drm_bridge_state *old_bridge_state)
+{
+       struct mtk_hdmi *hdmi = hdmi_ctx_from_bridge(bridge);
+
+       if (!hdmi->enabled)
+               return;
+
+       mtk_hdmi_hw_send_av_mute(hdmi);
+       usleep_range(50000, 50050);
+       mtk_hdmi_hw_vid_black(hdmi, true);
+       mtk_hdmi_disable_hdcp_encrypt(hdmi);
+       usleep_range(50000, 50050);
+
+       phy_power_off(hdmi->phy);
+
+       hdmi->enabled = false;
+}
+
+static void mtk_hdmi_bridge_post_disable(struct drm_bridge *bridge,
+                                        struct drm_bridge_state *old_state)
+{
+       struct mtk_hdmi *hdmi = hdmi_ctx_from_bridge(bridge);
+
+       if (!hdmi->powered)
+               return;
+
+       phy_power_off(hdmi->phy);
+
+       hdmi->powered = false;
+}
+
+static void mtk_hdmi_bridge_pre_enable(struct drm_bridge *bridge,
+                                      struct drm_bridge_state *old_state)
+{
+       struct mtk_hdmi *hdmi = hdmi_ctx_from_bridge(bridge);
+       struct drm_atomic_state *state = old_state->base.state;
+       union phy_configure_opts opts = {
+               .dp = { .link_rate = hdmi->mode.clock * KILO}
+       };
+
+       /* Retrieve the connector through the atomic state. */
+       hdmi->curr_conn = drm_atomic_get_new_connector_for_encoder(state,
+                                                                  
bridge->encoder);
+
+       mtk_hdmi_output_set_display_mode(hdmi, &hdmi->mode);
+       /* configuring phy clock link with appropriate rate */
+       phy_configure(hdmi->phy, &opts);
+       phy_power_on(hdmi->phy);
+       hdmi->powered = true;
+}
+
+static void mtk_hdmi_bridge_enable(struct drm_bridge *bridge,
+                                  struct drm_bridge_state *old_state)
+{
+       struct mtk_hdmi *hdmi = hdmi_ctx_from_bridge(bridge);
+       u8 buffer_spd[HDMI_INFOFRAME_SIZE(SPD)];
+       u8 buffer_avi[HDMI_INFOFRAME_SIZE(AVI)];
+
+       phy_power_on(hdmi->phy);
+       mtk_hdmi_send_infoframe(hdmi, buffer_spd, sizeof(buffer_spd),
+                               buffer_avi, sizeof(buffer_avi), &hdmi->mode);
+       mtk_hdmi_hw_spd_infoframe(hdmi, buffer_spd, sizeof(buffer_spd));
+       mtk_hdmi_hw_avi_infoframe(hdmi, buffer_avi, sizeof(buffer_avi));
+
+       mtk_hdmi_hw_vid_black(hdmi, false);
+
+       hdmi->enabled = true;
+}
+
+static enum drm_connector_status mtk_hdmi_bridge_detect(struct drm_bridge 
*bridge)
+{
+       struct mtk_hdmi *hdmi = hdmi_ctx_from_bridge(bridge);
+
+       return (hdmi->hpd != HDMI_PLUG_OUT) ? connector_status_connected :
+                                                   
connector_status_disconnected;
+}
+
+const struct drm_bridge_funcs mtk_v2_hdmi_bridge_funcs = {
+       .attach = mtk_hdmi_bridge_attach,
+       .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,
+       .mode_fixup = mtk_hdmi_bridge_mode_fixup,
+       .atomic_disable = mtk_hdmi_bridge_disable,
+       .atomic_post_disable = mtk_hdmi_bridge_post_disable,
+       .mode_set = mtk_hdmi_bridge_mode_set,
+       .atomic_pre_enable = mtk_hdmi_bridge_pre_enable,
+       .atomic_enable = mtk_hdmi_bridge_enable,
+       .get_edid = mtk_hdmi_bridge_get_edid,
+       .detect = mtk_hdmi_bridge_detect,
+};
diff --git a/drivers/gpu/drm/mediatek/mtk_hdmi_v2.h 
b/drivers/gpu/drm/mediatek/mtk_hdmi_v2.h
new file mode 100644
index 000000000000..15d06d82eb8e
--- /dev/null
+++ b/drivers/gpu/drm/mediatek/mtk_hdmi_v2.h
@@ -0,0 +1,30 @@
+/* SPDX-License-Identifier: GPL-2.0 */
+/*
+ * Copyright (c) 2022 MediaTek Inc.
+ * Copyright (c) 2022 BayLibre, SAS
+ */
+
+#ifndef _MTK_HDMI_V2_CTRL_H
+#define _MTK_HDMI_V2_CTRL_H
+
+#include <linux/hdmi.h>
+#include <drm/drm_bridge.h>
+
+struct mtk_hdmi;
+
+extern struct platform_driver mtk_hdmi_ddc_v2_driver;
+extern const struct drm_bridge_funcs mtk_v2_hdmi_bridge_funcs;
+void mtk_hdmi_output_init_v2(struct mtk_hdmi *hdmi);
+int mtk_hdmi_clk_enable_v2(struct mtk_hdmi *hdmi);
+void mtk_hdmi_clk_disable_v2(struct mtk_hdmi *hdmi);
+
+enum mtk_hdmi_clk_id_v2 {
+       MTK_HDMIV2_HDCP_SEL,
+       MTK_HDMIV2_HDCP_24M_SEL,
+       MTK_HDMIV2_VPP_SPLIT_HDMI,
+       MTK_HDMIV2_CLK_HDMI_APB_SEL,
+       MTK_HDMIV2_CLK_COUNT,
+};
+
+extern const char *const mtk_hdmi_clk_names_v2[MTK_HDMIV2_CLK_COUNT];
+#endif /* _MTK_HDMI_V2_CTRL_H */

-- 
2.40.0

Reply via email to