15852d539SSimon Glass /*
25852d539SSimon Glass * Copyright (c) 2015 Google, Inc
35852d539SSimon Glass * Copyright 2014 Rockchip Inc.
45852d539SSimon Glass *
55852d539SSimon Glass * SPDX-License-Identifier: GPL-2.0+
65852d539SSimon Glass */
75852d539SSimon Glass
85852d539SSimon Glass #include <common.h>
95852d539SSimon Glass #include <clk.h>
105852d539SSimon Glass #include <display.h>
115852d539SSimon Glass #include <dm.h>
125852d539SSimon Glass #include <edid.h>
135852d539SSimon Glass #include <panel.h>
145852d539SSimon Glass #include <regmap.h>
155852d539SSimon Glass #include <syscon.h>
165852d539SSimon Glass #include <asm/gpio.h>
175852d539SSimon Glass #include <asm/io.h>
185852d539SSimon Glass #include <asm/arch/clock.h>
195852d539SSimon Glass #include <asm/arch/edp_rk3288.h>
205852d539SSimon Glass #include <asm/arch/grf_rk3288.h>
215852d539SSimon Glass #include <dt-bindings/clock/rk3288-cru.h>
225852d539SSimon Glass
235852d539SSimon Glass DECLARE_GLOBAL_DATA_PTR;
245852d539SSimon Glass
255852d539SSimon Glass #define MAX_CR_LOOP 5
265852d539SSimon Glass #define MAX_EQ_LOOP 5
275852d539SSimon Glass #define DP_LINK_STATUS_SIZE 6
285852d539SSimon Glass
295852d539SSimon Glass static const char * const voltage_names[] = {
305852d539SSimon Glass "0.4V", "0.6V", "0.8V", "1.2V"
315852d539SSimon Glass };
325852d539SSimon Glass static const char * const pre_emph_names[] = {
335852d539SSimon Glass "0dB", "3.5dB", "6dB", "9.5dB"
345852d539SSimon Glass };
355852d539SSimon Glass
365852d539SSimon Glass #define DP_VOLTAGE_MAX DP_TRAIN_VOLTAGE_SWING_1200
375852d539SSimon Glass #define DP_PRE_EMPHASIS_MAX DP_TRAIN_PRE_EMPHASIS_9_5
385852d539SSimon Glass
395852d539SSimon Glass struct rk_edp_priv {
405852d539SSimon Glass struct rk3288_edp *regs;
415852d539SSimon Glass struct rk3288_grf *grf;
425852d539SSimon Glass struct udevice *panel;
435852d539SSimon Glass struct link_train link_train;
445852d539SSimon Glass u8 train_set[4];
455852d539SSimon Glass };
465852d539SSimon Glass
rk_edp_init_refclk(struct rk3288_edp * regs)475852d539SSimon Glass static void rk_edp_init_refclk(struct rk3288_edp *regs)
485852d539SSimon Glass {
495852d539SSimon Glass writel(SEL_24M, ®s->analog_ctl_2);
505852d539SSimon Glass writel(REF_CLK_24M, ®s->pll_reg_1);
515852d539SSimon Glass
525852d539SSimon Glass writel(LDO_OUTPUT_V_SEL_145 | KVCO_DEFALUT | CHG_PUMP_CUR_SEL_5US |
535852d539SSimon Glass V2L_CUR_SEL_1MA, ®s->pll_reg_2);
545852d539SSimon Glass
555852d539SSimon Glass writel(LOCK_DET_CNT_SEL_256 | LOOP_FILTER_RESET | PALL_SSC_RESET |
565852d539SSimon Glass LOCK_DET_BYPASS | PLL_LOCK_DET_MODE | PLL_LOCK_DET_FORCE,
575852d539SSimon Glass ®s->pll_reg_3);
585852d539SSimon Glass
595852d539SSimon Glass writel(REGULATOR_V_SEL_950MV | STANDBY_CUR_SEL |
605852d539SSimon Glass CHG_PUMP_INOUT_CTRL_1200MV | CHG_PUMP_INPUT_CTRL_OP,
615852d539SSimon Glass ®s->pll_reg_5);
625852d539SSimon Glass
635852d539SSimon Glass writel(SSC_OFFSET | SSC_MODE | SSC_DEPTH, ®s->ssc_reg);
645852d539SSimon Glass
655852d539SSimon Glass writel(TX_SWING_PRE_EMP_MODE | PRE_DRIVER_PW_CTRL1 |
665852d539SSimon Glass LP_MODE_CLK_REGULATOR | RESISTOR_MSB_CTRL | RESISTOR_CTRL,
675852d539SSimon Glass ®s->tx_common);
685852d539SSimon Glass
695852d539SSimon Glass writel(DP_AUX_COMMON_MODE | DP_AUX_EN | AUX_TERM_50OHM,
705852d539SSimon Glass ®s->dp_aux);
715852d539SSimon Glass
725852d539SSimon Glass writel(DP_BG_OUT_SEL | DP_DB_CUR_CTRL | DP_BG_SEL | DP_RESISTOR_TUNE_BG,
735852d539SSimon Glass ®s->dp_bias);
745852d539SSimon Glass
755852d539SSimon Glass writel(CH1_CH3_SWING_EMP_CTRL | CH0_CH2_SWING_EMP_CTRL,
765852d539SSimon Glass ®s->dp_reserv2);
775852d539SSimon Glass }
785852d539SSimon Glass
rk_edp_init_interrupt(struct rk3288_edp * regs)795852d539SSimon Glass static void rk_edp_init_interrupt(struct rk3288_edp *regs)
805852d539SSimon Glass {
815852d539SSimon Glass /* Set interrupt pin assertion polarity as high */
825852d539SSimon Glass writel(INT_POL, ®s->int_ctl);
835852d539SSimon Glass
845852d539SSimon Glass /* Clear pending registers */
855852d539SSimon Glass writel(0xff, ®s->common_int_sta_1);
865852d539SSimon Glass writel(0x4f, ®s->common_int_sta_2);
875852d539SSimon Glass writel(0xff, ®s->common_int_sta_3);
885852d539SSimon Glass writel(0x27, ®s->common_int_sta_4);
895852d539SSimon Glass writel(0x7f, ®s->dp_int_sta);
905852d539SSimon Glass
915852d539SSimon Glass /* 0:mask,1: unmask */
925852d539SSimon Glass writel(0x00, ®s->common_int_mask_1);
935852d539SSimon Glass writel(0x00, ®s->common_int_mask_2);
945852d539SSimon Glass writel(0x00, ®s->common_int_mask_3);
955852d539SSimon Glass writel(0x00, ®s->common_int_mask_4);
965852d539SSimon Glass writel(0x00, ®s->int_sta_mask);
975852d539SSimon Glass }
985852d539SSimon Glass
rk_edp_enable_sw_function(struct rk3288_edp * regs)995852d539SSimon Glass static void rk_edp_enable_sw_function(struct rk3288_edp *regs)
1005852d539SSimon Glass {
1015852d539SSimon Glass clrbits_le32(®s->func_en_1, SW_FUNC_EN_N);
1025852d539SSimon Glass }
1035852d539SSimon Glass
rk_edp_get_pll_locked(struct rk3288_edp * regs)1045852d539SSimon Glass static bool rk_edp_get_pll_locked(struct rk3288_edp *regs)
1055852d539SSimon Glass {
1065852d539SSimon Glass u32 val;
1075852d539SSimon Glass
1085852d539SSimon Glass val = readl(®s->dp_debug_ctl);
1095852d539SSimon Glass
1105852d539SSimon Glass return val & PLL_LOCK;
1115852d539SSimon Glass }
1125852d539SSimon Glass
rk_edp_init_analog_func(struct rk3288_edp * regs)1135852d539SSimon Glass static int rk_edp_init_analog_func(struct rk3288_edp *regs)
1145852d539SSimon Glass {
1155852d539SSimon Glass ulong start;
1165852d539SSimon Glass
1175852d539SSimon Glass writel(0x00, ®s->dp_pd);
1185852d539SSimon Glass writel(PLL_LOCK_CHG, ®s->common_int_sta_1);
1195852d539SSimon Glass
1205852d539SSimon Glass clrbits_le32(®s->dp_debug_ctl, F_PLL_LOCK | PLL_LOCK_CTRL);
1215852d539SSimon Glass
1225852d539SSimon Glass start = get_timer(0);
1235852d539SSimon Glass while (!rk_edp_get_pll_locked(regs)) {
1245852d539SSimon Glass if (get_timer(start) > PLL_LOCK_TIMEOUT) {
1255852d539SSimon Glass printf("%s: PLL is not locked\n", __func__);
1265852d539SSimon Glass return -ETIMEDOUT;
1275852d539SSimon Glass }
1285852d539SSimon Glass }
1295852d539SSimon Glass
1305852d539SSimon Glass /* Enable Serdes FIFO function and Link symbol clock domain module */
1315852d539SSimon Glass clrbits_le32(®s->func_en_2, SERDES_FIFO_FUNC_EN_N |
1325852d539SSimon Glass LS_CLK_DOMAIN_FUNC_EN_N | AUX_FUNC_EN_N |
1335852d539SSimon Glass SSC_FUNC_EN_N);
1345852d539SSimon Glass
1355852d539SSimon Glass return 0;
1365852d539SSimon Glass }
1375852d539SSimon Glass
rk_edp_init_aux(struct rk3288_edp * regs)1385852d539SSimon Glass static void rk_edp_init_aux(struct rk3288_edp *regs)
1395852d539SSimon Glass {
1405852d539SSimon Glass /* Clear inerrupts related to AUX channel */
1415852d539SSimon Glass writel(AUX_FUNC_EN_N, ®s->dp_int_sta);
1425852d539SSimon Glass
1435852d539SSimon Glass /* Disable AUX channel module */
1445852d539SSimon Glass setbits_le32(®s->func_en_2, AUX_FUNC_EN_N);
1455852d539SSimon Glass
1465852d539SSimon Glass /* Receive AUX Channel DEFER commands equal to DEFFER_COUNT*64 */
1475852d539SSimon Glass writel(DEFER_CTRL_EN | DEFER_COUNT(1), ®s->aux_ch_defer_dtl);
1485852d539SSimon Glass
1495852d539SSimon Glass /* Enable AUX channel module */
1505852d539SSimon Glass clrbits_le32(®s->func_en_2, AUX_FUNC_EN_N);
1515852d539SSimon Glass }
1525852d539SSimon Glass
rk_edp_aux_enable(struct rk3288_edp * regs)1535852d539SSimon Glass static int rk_edp_aux_enable(struct rk3288_edp *regs)
1545852d539SSimon Glass {
1555852d539SSimon Glass ulong start;
1565852d539SSimon Glass
1575852d539SSimon Glass setbits_le32(®s->aux_ch_ctl_2, AUX_EN);
1585852d539SSimon Glass start = get_timer(0);
1595852d539SSimon Glass do {
1605852d539SSimon Glass if (!(readl(®s->aux_ch_ctl_2) & AUX_EN))
1615852d539SSimon Glass return 0;
1625852d539SSimon Glass } while (get_timer(start) < 20);
1635852d539SSimon Glass
1645852d539SSimon Glass return -ETIMEDOUT;
1655852d539SSimon Glass }
1665852d539SSimon Glass
rk_edp_is_aux_reply(struct rk3288_edp * regs)1675852d539SSimon Glass static int rk_edp_is_aux_reply(struct rk3288_edp *regs)
1685852d539SSimon Glass {
1695852d539SSimon Glass ulong start;
1705852d539SSimon Glass
1715852d539SSimon Glass start = get_timer(0);
1725852d539SSimon Glass while (!(readl(®s->dp_int_sta) & RPLY_RECEIV)) {
1735852d539SSimon Glass if (get_timer(start) > 10)
1745852d539SSimon Glass return -ETIMEDOUT;
1755852d539SSimon Glass }
1765852d539SSimon Glass
1775852d539SSimon Glass writel(RPLY_RECEIV, ®s->dp_int_sta);
1785852d539SSimon Glass
1795852d539SSimon Glass return 0;
1805852d539SSimon Glass }
1815852d539SSimon Glass
rk_edp_start_aux_transaction(struct rk3288_edp * regs)1825852d539SSimon Glass static int rk_edp_start_aux_transaction(struct rk3288_edp *regs)
1835852d539SSimon Glass {
1845852d539SSimon Glass int val, ret;
1855852d539SSimon Glass
1865852d539SSimon Glass /* Enable AUX CH operation */
1875852d539SSimon Glass ret = rk_edp_aux_enable(regs);
1885852d539SSimon Glass if (ret) {
1895852d539SSimon Glass debug("AUX CH enable timeout!\n");
1905852d539SSimon Glass return ret;
1915852d539SSimon Glass }
1925852d539SSimon Glass
1935852d539SSimon Glass /* Is AUX CH command reply received? */
1945852d539SSimon Glass if (rk_edp_is_aux_reply(regs)) {
1955852d539SSimon Glass debug("AUX CH command reply failed!\n");
1965852d539SSimon Glass return ret;
1975852d539SSimon Glass }
1985852d539SSimon Glass
1995852d539SSimon Glass /* Clear interrupt source for AUX CH access error */
2005852d539SSimon Glass val = readl(®s->dp_int_sta);
2015852d539SSimon Glass if (val & AUX_ERR) {
2025852d539SSimon Glass writel(AUX_ERR, ®s->dp_int_sta);
2035852d539SSimon Glass return -EIO;
2045852d539SSimon Glass }
2055852d539SSimon Glass
2065852d539SSimon Glass /* Check AUX CH error access status */
2075852d539SSimon Glass val = readl(®s->dp_int_sta);
2085852d539SSimon Glass if (val & AUX_STATUS_MASK) {
2095852d539SSimon Glass debug("AUX CH error happens: %d\n\n", val & AUX_STATUS_MASK);
2105852d539SSimon Glass return -EIO;
2115852d539SSimon Glass }
2125852d539SSimon Glass
2135852d539SSimon Glass return 0;
2145852d539SSimon Glass }
2155852d539SSimon Glass
rk_edp_dpcd_transfer(struct rk3288_edp * regs,unsigned int val_addr,u8 * in_data,unsigned int length,enum dpcd_request request)2165852d539SSimon Glass static int rk_edp_dpcd_transfer(struct rk3288_edp *regs,
2175852d539SSimon Glass unsigned int val_addr, u8 *in_data,
2185852d539SSimon Glass unsigned int length,
2195852d539SSimon Glass enum dpcd_request request)
2205852d539SSimon Glass {
2215852d539SSimon Glass int val;
2225852d539SSimon Glass int i, try_times;
2235852d539SSimon Glass u8 *data;
2245852d539SSimon Glass int ret = 0;
2255852d539SSimon Glass u32 len = 0;
2265852d539SSimon Glass
2275852d539SSimon Glass while (length) {
2285852d539SSimon Glass len = min(length, 16U);
2295852d539SSimon Glass for (try_times = 0; try_times < 10; try_times++) {
2305852d539SSimon Glass data = in_data;
2315852d539SSimon Glass /* Clear AUX CH data buffer */
2325852d539SSimon Glass writel(BUF_CLR, ®s->buf_data_ctl);
2335852d539SSimon Glass
2345852d539SSimon Glass /* Select DPCD device address */
2355852d539SSimon Glass writel(AUX_ADDR_7_0(val_addr), ®s->aux_addr_7_0);
2365852d539SSimon Glass writel(AUX_ADDR_15_8(val_addr), ®s->aux_addr_15_8);
2375852d539SSimon Glass writel(AUX_ADDR_19_16(val_addr), ®s->aux_addr_19_16);
2385852d539SSimon Glass
2395852d539SSimon Glass /*
2405852d539SSimon Glass * Set DisplayPort transaction and read 1 byte
2415852d539SSimon Glass * If bit 3 is 1, DisplayPort transaction.
2425852d539SSimon Glass * If Bit 3 is 0, I2C transaction.
2435852d539SSimon Glass */
2445852d539SSimon Glass if (request == DPCD_WRITE) {
2455852d539SSimon Glass val = AUX_LENGTH(len) |
2465852d539SSimon Glass AUX_TX_COMM_DP_TRANSACTION |
2475852d539SSimon Glass AUX_TX_COMM_WRITE;
2485852d539SSimon Glass for (i = 0; i < len; i++)
2495852d539SSimon Glass writel(*data++, ®s->buf_data[i]);
2505852d539SSimon Glass } else
2515852d539SSimon Glass val = AUX_LENGTH(len) |
2525852d539SSimon Glass AUX_TX_COMM_DP_TRANSACTION |
2535852d539SSimon Glass AUX_TX_COMM_READ;
2545852d539SSimon Glass
2555852d539SSimon Glass writel(val, ®s->aux_ch_ctl_1);
2565852d539SSimon Glass
2575852d539SSimon Glass /* Start AUX transaction */
2585852d539SSimon Glass ret = rk_edp_start_aux_transaction(regs);
2595852d539SSimon Glass if (ret == 0)
2605852d539SSimon Glass break;
2615852d539SSimon Glass else
2625852d539SSimon Glass printf("read dpcd Aux Transaction fail!\n");
2635852d539SSimon Glass }
2645852d539SSimon Glass
2655852d539SSimon Glass if (ret)
2665852d539SSimon Glass return ret;
2675852d539SSimon Glass
2685852d539SSimon Glass if (request == DPCD_READ) {
2695852d539SSimon Glass for (i = 0; i < len; i++)
2705852d539SSimon Glass *data++ = (u8)readl(®s->buf_data[i]);
2715852d539SSimon Glass }
2725852d539SSimon Glass
2735852d539SSimon Glass length -= len;
2745852d539SSimon Glass val_addr += len;
2755852d539SSimon Glass in_data += len;
2765852d539SSimon Glass }
2775852d539SSimon Glass
2785852d539SSimon Glass return 0;
2795852d539SSimon Glass }
2805852d539SSimon Glass
rk_edp_dpcd_read(struct rk3288_edp * regs,u32 addr,u8 * values,size_t size)2815852d539SSimon Glass static int rk_edp_dpcd_read(struct rk3288_edp *regs, u32 addr, u8 *values,
2825852d539SSimon Glass size_t size)
2835852d539SSimon Glass {
2845852d539SSimon Glass return rk_edp_dpcd_transfer(regs, addr, values, size, DPCD_READ);
2855852d539SSimon Glass }
2865852d539SSimon Glass
rk_edp_dpcd_write(struct rk3288_edp * regs,u32 addr,u8 * values,size_t size)2875852d539SSimon Glass static int rk_edp_dpcd_write(struct rk3288_edp *regs, u32 addr, u8 *values,
2885852d539SSimon Glass size_t size)
2895852d539SSimon Glass {
2905852d539SSimon Glass return rk_edp_dpcd_transfer(regs, addr, values, size, DPCD_WRITE);
2915852d539SSimon Glass }
2925852d539SSimon Glass
2935852d539SSimon Glass
rk_edp_link_power_up(struct rk_edp_priv * edp)2945852d539SSimon Glass static int rk_edp_link_power_up(struct rk_edp_priv *edp)
2955852d539SSimon Glass {
2965852d539SSimon Glass u8 value;
2975852d539SSimon Glass int ret;
2985852d539SSimon Glass
2995852d539SSimon Glass /* DP_SET_POWER register is only available on DPCD v1.1 and later */
3005852d539SSimon Glass if (edp->link_train.revision < 0x11)
3015852d539SSimon Glass return 0;
3025852d539SSimon Glass
3035852d539SSimon Glass ret = rk_edp_dpcd_read(edp->regs, DPCD_LINK_POWER_STATE, &value, 1);
3045852d539SSimon Glass if (ret)
3055852d539SSimon Glass return ret;
3065852d539SSimon Glass
3075852d539SSimon Glass value &= ~DP_SET_POWER_MASK;
3085852d539SSimon Glass value |= DP_SET_POWER_D0;
3095852d539SSimon Glass
3105852d539SSimon Glass ret = rk_edp_dpcd_write(edp->regs, DPCD_LINK_POWER_STATE, &value, 1);
3115852d539SSimon Glass if (ret)
3125852d539SSimon Glass return ret;
3135852d539SSimon Glass
3145852d539SSimon Glass /*
3155852d539SSimon Glass * According to the DP 1.1 specification, a "Sink Device must exit the
3165852d539SSimon Glass * power saving state within 1 ms" (Section 2.5.3.1, Table 5-52, "Sink
3175852d539SSimon Glass * Control Field" (register 0x600).
3185852d539SSimon Glass */
3195852d539SSimon Glass mdelay(1);
3205852d539SSimon Glass
3215852d539SSimon Glass return 0;
3225852d539SSimon Glass }
3235852d539SSimon Glass
rk_edp_link_configure(struct rk_edp_priv * edp)3245852d539SSimon Glass static int rk_edp_link_configure(struct rk_edp_priv *edp)
3255852d539SSimon Glass {
3265852d539SSimon Glass u8 values[2];
3275852d539SSimon Glass
3285852d539SSimon Glass values[0] = edp->link_train.link_rate;
3295852d539SSimon Glass values[1] = edp->link_train.lane_count;
3305852d539SSimon Glass
3315852d539SSimon Glass return rk_edp_dpcd_write(edp->regs, DPCD_LINK_BW_SET, values,
3325852d539SSimon Glass sizeof(values));
3335852d539SSimon Glass }
3345852d539SSimon Glass
rk_edp_set_link_training(struct rk_edp_priv * edp,const u8 * training_values)3355852d539SSimon Glass static void rk_edp_set_link_training(struct rk_edp_priv *edp,
3365852d539SSimon Glass const u8 *training_values)
3375852d539SSimon Glass {
3385852d539SSimon Glass int i;
3395852d539SSimon Glass
3405852d539SSimon Glass for (i = 0; i < edp->link_train.lane_count; i++)
3415852d539SSimon Glass writel(training_values[i], &edp->regs->ln_link_trn_ctl[i]);
3425852d539SSimon Glass }
3435852d539SSimon Glass
edp_link_status(const u8 * link_status,int r)3445852d539SSimon Glass static u8 edp_link_status(const u8 *link_status, int r)
3455852d539SSimon Glass {
3465852d539SSimon Glass return link_status[r - DPCD_LANE0_1_STATUS];
3475852d539SSimon Glass }
3485852d539SSimon Glass
rk_edp_dpcd_read_link_status(struct rk_edp_priv * edp,u8 * link_status)3495852d539SSimon Glass static int rk_edp_dpcd_read_link_status(struct rk_edp_priv *edp,
3505852d539SSimon Glass u8 *link_status)
3515852d539SSimon Glass {
3525852d539SSimon Glass return rk_edp_dpcd_read(edp->regs, DPCD_LANE0_1_STATUS, link_status,
3535852d539SSimon Glass DP_LINK_STATUS_SIZE);
3545852d539SSimon Glass }
3555852d539SSimon Glass
edp_get_lane_status(const u8 * link_status,int lane)3565852d539SSimon Glass static u8 edp_get_lane_status(const u8 *link_status, int lane)
3575852d539SSimon Glass {
3585852d539SSimon Glass int i = DPCD_LANE0_1_STATUS + (lane >> 1);
3595852d539SSimon Glass int s = (lane & 1) * 4;
3605852d539SSimon Glass u8 l = edp_link_status(link_status, i);
3615852d539SSimon Glass
3625852d539SSimon Glass return (l >> s) & 0xf;
3635852d539SSimon Glass }
3645852d539SSimon Glass
rk_edp_clock_recovery(const u8 * link_status,int lane_count)3655852d539SSimon Glass static int rk_edp_clock_recovery(const u8 *link_status, int lane_count)
3665852d539SSimon Glass {
3675852d539SSimon Glass int lane;
3685852d539SSimon Glass u8 lane_status;
3695852d539SSimon Glass
3705852d539SSimon Glass for (lane = 0; lane < lane_count; lane++) {
3715852d539SSimon Glass lane_status = edp_get_lane_status(link_status, lane);
3725852d539SSimon Glass if ((lane_status & DP_LANE_CR_DONE) == 0)
3735852d539SSimon Glass return -EIO;
3745852d539SSimon Glass }
3755852d539SSimon Glass
3765852d539SSimon Glass return 0;
3775852d539SSimon Glass }
3785852d539SSimon Glass
rk_edp_channel_eq(const u8 * link_status,int lane_count)3795852d539SSimon Glass static int rk_edp_channel_eq(const u8 *link_status, int lane_count)
3805852d539SSimon Glass {
3815852d539SSimon Glass u8 lane_align;
3825852d539SSimon Glass u8 lane_status;
3835852d539SSimon Glass int lane;
3845852d539SSimon Glass
3855852d539SSimon Glass lane_align = edp_link_status(link_status,
3865852d539SSimon Glass DPCD_LANE_ALIGN_STATUS_UPDATED);
3875852d539SSimon Glass if (!(lane_align & DP_INTERLANE_ALIGN_DONE))
3885852d539SSimon Glass return -EIO;
3895852d539SSimon Glass for (lane = 0; lane < lane_count; lane++) {
3905852d539SSimon Glass lane_status = edp_get_lane_status(link_status, lane);
3915852d539SSimon Glass if ((lane_status & DP_CHANNEL_EQ_BITS) != DP_CHANNEL_EQ_BITS)
3925852d539SSimon Glass return -EIO;
3935852d539SSimon Glass }
3945852d539SSimon Glass
3955852d539SSimon Glass return 0;
3965852d539SSimon Glass }
3975852d539SSimon Glass
rk_edp_get_adjust_request_voltage(const u8 * link_status,int lane)3985852d539SSimon Glass static uint rk_edp_get_adjust_request_voltage(const u8 *link_status, int lane)
3995852d539SSimon Glass {
4005852d539SSimon Glass int i = DPCD_ADJUST_REQUEST_LANE0_1 + (lane >> 1);
4015852d539SSimon Glass int s = ((lane & 1) ?
4025852d539SSimon Glass DP_ADJUST_VOLTAGE_SWING_LANE1_SHIFT :
4035852d539SSimon Glass DP_ADJUST_VOLTAGE_SWING_LANE0_SHIFT);
4045852d539SSimon Glass u8 l = edp_link_status(link_status, i);
4055852d539SSimon Glass
4065852d539SSimon Glass return ((l >> s) & 0x3) << DP_TRAIN_VOLTAGE_SWING_SHIFT;
4075852d539SSimon Glass }
4085852d539SSimon Glass
rk_edp_get_adjust_request_pre_emphasis(const u8 * link_status,int lane)4095852d539SSimon Glass static uint rk_edp_get_adjust_request_pre_emphasis(const u8 *link_status,
4105852d539SSimon Glass int lane)
4115852d539SSimon Glass {
4125852d539SSimon Glass int i = DPCD_ADJUST_REQUEST_LANE0_1 + (lane >> 1);
4135852d539SSimon Glass int s = ((lane & 1) ?
4145852d539SSimon Glass DP_ADJUST_PRE_EMPHASIS_LANE1_SHIFT :
4155852d539SSimon Glass DP_ADJUST_PRE_EMPHASIS_LANE0_SHIFT);
4165852d539SSimon Glass u8 l = edp_link_status(link_status, i);
4175852d539SSimon Glass
4185852d539SSimon Glass return ((l >> s) & 0x3) << DP_TRAIN_PRE_EMPHASIS_SHIFT;
4195852d539SSimon Glass }
4205852d539SSimon Glass
edp_get_adjust_train(const u8 * link_status,int lane_count,u8 train_set[])4215852d539SSimon Glass static void edp_get_adjust_train(const u8 *link_status, int lane_count,
4225852d539SSimon Glass u8 train_set[])
4235852d539SSimon Glass {
4245852d539SSimon Glass uint v = 0;
4255852d539SSimon Glass uint p = 0;
4265852d539SSimon Glass int lane;
4275852d539SSimon Glass
4285852d539SSimon Glass for (lane = 0; lane < lane_count; lane++) {
4295852d539SSimon Glass uint this_v, this_p;
4305852d539SSimon Glass
4315852d539SSimon Glass this_v = rk_edp_get_adjust_request_voltage(link_status, lane);
4325852d539SSimon Glass this_p = rk_edp_get_adjust_request_pre_emphasis(link_status,
4335852d539SSimon Glass lane);
4345852d539SSimon Glass
4355852d539SSimon Glass debug("requested signal parameters: lane %d voltage %s pre_emph %s\n",
4365852d539SSimon Glass lane,
4375852d539SSimon Glass voltage_names[this_v >> DP_TRAIN_VOLTAGE_SWING_SHIFT],
4385852d539SSimon Glass pre_emph_names[this_p >> DP_TRAIN_PRE_EMPHASIS_SHIFT]);
4395852d539SSimon Glass
4405852d539SSimon Glass if (this_v > v)
4415852d539SSimon Glass v = this_v;
4425852d539SSimon Glass if (this_p > p)
4435852d539SSimon Glass p = this_p;
4445852d539SSimon Glass }
4455852d539SSimon Glass
4465852d539SSimon Glass if (v >= DP_VOLTAGE_MAX)
4475852d539SSimon Glass v |= DP_TRAIN_MAX_SWING_REACHED;
4485852d539SSimon Glass
4495852d539SSimon Glass if (p >= DP_PRE_EMPHASIS_MAX)
4505852d539SSimon Glass p |= DP_TRAIN_MAX_PRE_EMPHASIS_REACHED;
4515852d539SSimon Glass
4525852d539SSimon Glass debug("using signal parameters: voltage %s pre_emph %s\n",
4535852d539SSimon Glass voltage_names[(v & DP_TRAIN_VOLTAGE_SWING_MASK)
4545852d539SSimon Glass >> DP_TRAIN_VOLTAGE_SWING_SHIFT],
4555852d539SSimon Glass pre_emph_names[(p & DP_TRAIN_PRE_EMPHASIS_MASK)
4565852d539SSimon Glass >> DP_TRAIN_PRE_EMPHASIS_SHIFT]);
4575852d539SSimon Glass
4585852d539SSimon Glass for (lane = 0; lane < 4; lane++)
4595852d539SSimon Glass train_set[lane] = v | p;
4605852d539SSimon Glass }
4615852d539SSimon Glass
rk_edp_link_train_cr(struct rk_edp_priv * edp)4625852d539SSimon Glass static int rk_edp_link_train_cr(struct rk_edp_priv *edp)
4635852d539SSimon Glass {
4645852d539SSimon Glass struct rk3288_edp *regs = edp->regs;
4655852d539SSimon Glass int clock_recovery;
4665852d539SSimon Glass uint voltage, tries = 0;
4675852d539SSimon Glass u8 status[DP_LINK_STATUS_SIZE];
4685852d539SSimon Glass int i, ret;
4695852d539SSimon Glass u8 value;
4705852d539SSimon Glass
4715852d539SSimon Glass value = DP_TRAINING_PATTERN_1;
4725852d539SSimon Glass writel(value, ®s->dp_training_ptn_set);
4735852d539SSimon Glass ret = rk_edp_dpcd_write(regs, DPCD_TRAINING_PATTERN_SET, &value, 1);
4745852d539SSimon Glass if (ret)
4755852d539SSimon Glass return ret;
4765852d539SSimon Glass memset(edp->train_set, '\0', sizeof(edp->train_set));
4775852d539SSimon Glass
4785852d539SSimon Glass /* clock recovery loop */
4795852d539SSimon Glass clock_recovery = 0;
4805852d539SSimon Glass tries = 0;
4815852d539SSimon Glass voltage = 0xff;
4825852d539SSimon Glass
4835852d539SSimon Glass while (1) {
4845852d539SSimon Glass rk_edp_set_link_training(edp, edp->train_set);
4855852d539SSimon Glass ret = rk_edp_dpcd_write(regs, DPCD_TRAINING_LANE0_SET,
4865852d539SSimon Glass edp->train_set,
4875852d539SSimon Glass edp->link_train.lane_count);
4885852d539SSimon Glass if (ret)
4895852d539SSimon Glass return ret;
4905852d539SSimon Glass
4915852d539SSimon Glass mdelay(1);
4925852d539SSimon Glass
4935852d539SSimon Glass ret = rk_edp_dpcd_read_link_status(edp, status);
4945852d539SSimon Glass if (ret) {
4955852d539SSimon Glass printf("displayport link status failed, ret=%d\n", ret);
4965852d539SSimon Glass break;
4975852d539SSimon Glass }
4985852d539SSimon Glass
4995852d539SSimon Glass clock_recovery = rk_edp_clock_recovery(status,
5005852d539SSimon Glass edp->link_train.lane_count);
5015852d539SSimon Glass if (!clock_recovery)
5025852d539SSimon Glass break;
5035852d539SSimon Glass
5045852d539SSimon Glass for (i = 0; i < edp->link_train.lane_count; i++) {
5055852d539SSimon Glass if ((edp->train_set[i] &
5065852d539SSimon Glass DP_TRAIN_MAX_SWING_REACHED) == 0)
5075852d539SSimon Glass break;
5085852d539SSimon Glass }
5095852d539SSimon Glass if (i == edp->link_train.lane_count) {
5105852d539SSimon Glass printf("clock recovery reached max voltage\n");
5115852d539SSimon Glass break;
5125852d539SSimon Glass }
5135852d539SSimon Glass
5145852d539SSimon Glass if ((edp->train_set[0] & DP_TRAIN_VOLTAGE_SWING_MASK) ==
5155852d539SSimon Glass voltage) {
5165852d539SSimon Glass if (++tries == MAX_CR_LOOP) {
5175852d539SSimon Glass printf("clock recovery tried 5 times\n");
5185852d539SSimon Glass break;
5195852d539SSimon Glass }
5205852d539SSimon Glass } else {
5215852d539SSimon Glass tries = 0;
5225852d539SSimon Glass }
5235852d539SSimon Glass
5245852d539SSimon Glass voltage = edp->train_set[0] & DP_TRAIN_VOLTAGE_SWING_MASK;
5255852d539SSimon Glass
5265852d539SSimon Glass /* Compute new train_set as requested by sink */
5275852d539SSimon Glass edp_get_adjust_train(status, edp->link_train.lane_count,
5285852d539SSimon Glass edp->train_set);
5295852d539SSimon Glass }
5305852d539SSimon Glass if (clock_recovery) {
5315852d539SSimon Glass printf("clock recovery failed: %d\n", clock_recovery);
5325852d539SSimon Glass return clock_recovery;
5335852d539SSimon Glass } else {
5345852d539SSimon Glass debug("clock recovery at voltage %d pre-emphasis %d\n",
5355852d539SSimon Glass edp->train_set[0] & DP_TRAIN_VOLTAGE_SWING_MASK,
5365852d539SSimon Glass (edp->train_set[0] & DP_TRAIN_PRE_EMPHASIS_MASK) >>
5375852d539SSimon Glass DP_TRAIN_PRE_EMPHASIS_SHIFT);
5385852d539SSimon Glass return 0;
5395852d539SSimon Glass }
5405852d539SSimon Glass }
5415852d539SSimon Glass
rk_edp_link_train_ce(struct rk_edp_priv * edp)5425852d539SSimon Glass static int rk_edp_link_train_ce(struct rk_edp_priv *edp)
5435852d539SSimon Glass {
5445852d539SSimon Glass struct rk3288_edp *regs = edp->regs;
5455852d539SSimon Glass int channel_eq;
5465852d539SSimon Glass u8 value;
5475852d539SSimon Glass int tries;
5485852d539SSimon Glass u8 status[DP_LINK_STATUS_SIZE];
5495852d539SSimon Glass int ret;
5505852d539SSimon Glass
5515852d539SSimon Glass value = DP_TRAINING_PATTERN_2;
5525852d539SSimon Glass writel(value, ®s->dp_training_ptn_set);
5535852d539SSimon Glass ret = rk_edp_dpcd_write(regs, DPCD_TRAINING_PATTERN_SET, &value, 1);
5545852d539SSimon Glass if (ret)
5555852d539SSimon Glass return ret;
5565852d539SSimon Glass
5575852d539SSimon Glass /* channel equalization loop */
5585852d539SSimon Glass channel_eq = 0;
5595852d539SSimon Glass for (tries = 0; tries < 5; tries++) {
5605852d539SSimon Glass rk_edp_set_link_training(edp, edp->train_set);
5615852d539SSimon Glass udelay(400);
5625852d539SSimon Glass
5635852d539SSimon Glass if (rk_edp_dpcd_read_link_status(edp, status) < 0) {
5645852d539SSimon Glass printf("displayport link status failed\n");
5655852d539SSimon Glass return -1;
5665852d539SSimon Glass }
5675852d539SSimon Glass
5685852d539SSimon Glass channel_eq = rk_edp_channel_eq(status,
5695852d539SSimon Glass edp->link_train.lane_count);
5705852d539SSimon Glass if (!channel_eq)
5715852d539SSimon Glass break;
5725852d539SSimon Glass edp_get_adjust_train(status, edp->link_train.lane_count,
5735852d539SSimon Glass edp->train_set);
5745852d539SSimon Glass }
5755852d539SSimon Glass
5765852d539SSimon Glass if (channel_eq) {
5775852d539SSimon Glass printf("channel eq failed, ret=%d\n", channel_eq);
5785852d539SSimon Glass return channel_eq;
5795852d539SSimon Glass }
5805852d539SSimon Glass
5815852d539SSimon Glass debug("channel eq at voltage %d pre-emphasis %d\n",
5825852d539SSimon Glass edp->train_set[0] & DP_TRAIN_VOLTAGE_SWING_MASK,
5835852d539SSimon Glass (edp->train_set[0] & DP_TRAIN_PRE_EMPHASIS_MASK)
5845852d539SSimon Glass >> DP_TRAIN_PRE_EMPHASIS_SHIFT);
5855852d539SSimon Glass
5865852d539SSimon Glass return 0;
5875852d539SSimon Glass }
5885852d539SSimon Glass
rk_edp_init_training(struct rk_edp_priv * edp)5895852d539SSimon Glass static int rk_edp_init_training(struct rk_edp_priv *edp)
5905852d539SSimon Glass {
5915852d539SSimon Glass u8 values[3];
5925852d539SSimon Glass int ret;
5935852d539SSimon Glass
5945852d539SSimon Glass ret = rk_edp_dpcd_read(edp->regs, DPCD_DPCD_REV, values,
5955852d539SSimon Glass sizeof(values));
5965852d539SSimon Glass if (ret < 0)
5975852d539SSimon Glass return ret;
5985852d539SSimon Glass
5995852d539SSimon Glass edp->link_train.revision = values[0];
6005852d539SSimon Glass edp->link_train.link_rate = values[1];
6015852d539SSimon Glass edp->link_train.lane_count = values[2] & DP_MAX_LANE_COUNT_MASK;
6025852d539SSimon Glass
6035852d539SSimon Glass debug("max link rate:%d.%dGps max number of lanes:%d\n",
6045852d539SSimon Glass edp->link_train.link_rate * 27 / 100,
6055852d539SSimon Glass edp->link_train.link_rate * 27 % 100,
6065852d539SSimon Glass edp->link_train.lane_count);
6075852d539SSimon Glass
6085852d539SSimon Glass if ((edp->link_train.link_rate != LINK_RATE_1_62GBPS) &&
6095852d539SSimon Glass (edp->link_train.link_rate != LINK_RATE_2_70GBPS)) {
6105852d539SSimon Glass debug("Rx Max Link Rate is abnormal :%x\n",
6115852d539SSimon Glass edp->link_train.link_rate);
6125852d539SSimon Glass return -EPERM;
6135852d539SSimon Glass }
6145852d539SSimon Glass
6155852d539SSimon Glass if (edp->link_train.lane_count == 0) {
6165852d539SSimon Glass debug("Rx Max Lane count is abnormal :%x\n",
6175852d539SSimon Glass edp->link_train.lane_count);
6185852d539SSimon Glass return -EPERM;
6195852d539SSimon Glass }
6205852d539SSimon Glass
6215852d539SSimon Glass ret = rk_edp_link_power_up(edp);
6225852d539SSimon Glass if (ret)
6235852d539SSimon Glass return ret;
6245852d539SSimon Glass
6255852d539SSimon Glass return rk_edp_link_configure(edp);
6265852d539SSimon Glass }
6275852d539SSimon Glass
rk_edp_hw_link_training(struct rk_edp_priv * edp)6285852d539SSimon Glass static int rk_edp_hw_link_training(struct rk_edp_priv *edp)
6295852d539SSimon Glass {
6305852d539SSimon Glass ulong start;
6315852d539SSimon Glass u32 val;
6325852d539SSimon Glass int ret;
6335852d539SSimon Glass
6345852d539SSimon Glass /* Set link rate and count as you want to establish */
6355852d539SSimon Glass writel(edp->link_train.link_rate, &edp->regs->link_bw_set);
6365852d539SSimon Glass writel(edp->link_train.lane_count, &edp->regs->lane_count_set);
6375852d539SSimon Glass
6385852d539SSimon Glass ret = rk_edp_link_train_cr(edp);
6395852d539SSimon Glass if (ret)
6405852d539SSimon Glass return ret;
6415852d539SSimon Glass ret = rk_edp_link_train_ce(edp);
6425852d539SSimon Glass if (ret)
6435852d539SSimon Glass return ret;
6445852d539SSimon Glass
6455852d539SSimon Glass writel(HW_LT_EN, &edp->regs->dp_hw_link_training);
6465852d539SSimon Glass start = get_timer(0);
6475852d539SSimon Glass do {
6485852d539SSimon Glass val = readl(&edp->regs->dp_hw_link_training);
6495852d539SSimon Glass if (!(val & HW_LT_EN))
6505852d539SSimon Glass break;
6515852d539SSimon Glass } while (get_timer(start) < 10);
6525852d539SSimon Glass
6535852d539SSimon Glass if (val & HW_LT_ERR_CODE_MASK) {
6545852d539SSimon Glass printf("edp hw link training error: %d\n",
6555852d539SSimon Glass val >> HW_LT_ERR_CODE_SHIFT);
6565852d539SSimon Glass return -EIO;
6575852d539SSimon Glass }
6585852d539SSimon Glass
6595852d539SSimon Glass return 0;
6605852d539SSimon Glass }
6615852d539SSimon Glass
rk_edp_select_i2c_device(struct rk3288_edp * regs,unsigned int device_addr,unsigned int val_addr)6625852d539SSimon Glass static int rk_edp_select_i2c_device(struct rk3288_edp *regs,
6635852d539SSimon Glass unsigned int device_addr,
6645852d539SSimon Glass unsigned int val_addr)
6655852d539SSimon Glass {
6665852d539SSimon Glass int ret;
6675852d539SSimon Glass
6685852d539SSimon Glass /* Set EDID device address */
6695852d539SSimon Glass writel(device_addr, ®s->aux_addr_7_0);
6705852d539SSimon Glass writel(0x0, ®s->aux_addr_15_8);
6715852d539SSimon Glass writel(0x0, ®s->aux_addr_19_16);
6725852d539SSimon Glass
6735852d539SSimon Glass /* Set offset from base address of EDID device */
6745852d539SSimon Glass writel(val_addr, ®s->buf_data[0]);
6755852d539SSimon Glass
6765852d539SSimon Glass /*
6775852d539SSimon Glass * Set I2C transaction and write address
6785852d539SSimon Glass * If bit 3 is 1, DisplayPort transaction.
6795852d539SSimon Glass * If Bit 3 is 0, I2C transaction.
6805852d539SSimon Glass */
6815852d539SSimon Glass writel(AUX_TX_COMM_I2C_TRANSACTION | AUX_TX_COMM_MOT |
6825852d539SSimon Glass AUX_TX_COMM_WRITE, ®s->aux_ch_ctl_1);
6835852d539SSimon Glass
6845852d539SSimon Glass /* Start AUX transaction */
6855852d539SSimon Glass ret = rk_edp_start_aux_transaction(regs);
6865852d539SSimon Glass if (ret != 0) {
6875852d539SSimon Glass debug("select_i2c_device Aux Transaction fail!\n");
6885852d539SSimon Glass return ret;
6895852d539SSimon Glass }
6905852d539SSimon Glass
6915852d539SSimon Glass return 0;
6925852d539SSimon Glass }
6935852d539SSimon Glass
rk_edp_i2c_read(struct rk3288_edp * regs,unsigned int device_addr,unsigned int val_addr,unsigned int count,u8 edid[])6945852d539SSimon Glass static int rk_edp_i2c_read(struct rk3288_edp *regs, unsigned int device_addr,
6955852d539SSimon Glass unsigned int val_addr, unsigned int count, u8 edid[])
6965852d539SSimon Glass {
6975852d539SSimon Glass u32 val;
6985852d539SSimon Glass unsigned int i, j;
6995852d539SSimon Glass unsigned int cur_data_idx;
7005852d539SSimon Glass unsigned int defer = 0;
7015852d539SSimon Glass int ret = 0;
7025852d539SSimon Glass
7035852d539SSimon Glass for (i = 0; i < count; i += 16) {
7045852d539SSimon Glass for (j = 0; j < 10; j++) { /* try 10 times */
7055852d539SSimon Glass /* Clear AUX CH data buffer */
7065852d539SSimon Glass writel(BUF_CLR, ®s->buf_data_ctl);
7075852d539SSimon Glass
7085852d539SSimon Glass /* Set normal AUX CH command */
7095852d539SSimon Glass clrbits_le32(®s->aux_ch_ctl_2, ADDR_ONLY);
7105852d539SSimon Glass
7115852d539SSimon Glass /*
7125852d539SSimon Glass * If Rx sends defer, Tx sends only reads
7135852d539SSimon Glass * request without sending addres
7145852d539SSimon Glass */
7155852d539SSimon Glass if (!defer) {
7165852d539SSimon Glass ret = rk_edp_select_i2c_device(regs,
7175852d539SSimon Glass device_addr,
7185852d539SSimon Glass val_addr + i);
7195852d539SSimon Glass } else {
7205852d539SSimon Glass defer = 0;
7215852d539SSimon Glass }
7225852d539SSimon Glass
7235852d539SSimon Glass /*
7245852d539SSimon Glass * Set I2C transaction and write data
7255852d539SSimon Glass * If bit 3 is 1, DisplayPort transaction.
7265852d539SSimon Glass * If Bit 3 is 0, I2C transaction.
7275852d539SSimon Glass */
7285852d539SSimon Glass writel(AUX_LENGTH(16) | AUX_TX_COMM_I2C_TRANSACTION |
7295852d539SSimon Glass AUX_TX_COMM_READ, ®s->aux_ch_ctl_1);
7305852d539SSimon Glass
7315852d539SSimon Glass /* Start AUX transaction */
7325852d539SSimon Glass ret = rk_edp_start_aux_transaction(regs);
7335852d539SSimon Glass if (ret == 0) {
7345852d539SSimon Glass break;
7355852d539SSimon Glass } else {
7365852d539SSimon Glass debug("Aux Transaction fail!\n");
7375852d539SSimon Glass continue;
7385852d539SSimon Glass }
7395852d539SSimon Glass
7405852d539SSimon Glass /* Check if Rx sends defer */
7415852d539SSimon Glass val = readl(®s->aux_rx_comm);
7425852d539SSimon Glass if (val == AUX_RX_COMM_AUX_DEFER ||
7435852d539SSimon Glass val == AUX_RX_COMM_I2C_DEFER) {
7445852d539SSimon Glass debug("Defer: %d\n\n", val);
7455852d539SSimon Glass defer = 1;
7465852d539SSimon Glass }
7475852d539SSimon Glass }
7485852d539SSimon Glass
7495852d539SSimon Glass if (ret)
7505852d539SSimon Glass return ret;
7515852d539SSimon Glass
7525852d539SSimon Glass for (cur_data_idx = 0; cur_data_idx < 16; cur_data_idx++) {
7535852d539SSimon Glass val = readl(®s->buf_data[cur_data_idx]);
7545852d539SSimon Glass edid[i + cur_data_idx] = (u8)val;
7555852d539SSimon Glass }
7565852d539SSimon Glass }
7575852d539SSimon Glass
7585852d539SSimon Glass return 0;
7595852d539SSimon Glass }
7605852d539SSimon Glass
rk_edp_set_link_train(struct rk_edp_priv * edp)7615852d539SSimon Glass static int rk_edp_set_link_train(struct rk_edp_priv *edp)
7625852d539SSimon Glass {
7635852d539SSimon Glass int ret;
7645852d539SSimon Glass
7655852d539SSimon Glass ret = rk_edp_init_training(edp);
7665852d539SSimon Glass if (ret) {
7675852d539SSimon Glass printf("DP LT init failed!\n");
7685852d539SSimon Glass return ret;
7695852d539SSimon Glass }
7705852d539SSimon Glass
7715852d539SSimon Glass ret = rk_edp_hw_link_training(edp);
7725852d539SSimon Glass if (ret)
7735852d539SSimon Glass return ret;
7745852d539SSimon Glass
7755852d539SSimon Glass return 0;
7765852d539SSimon Glass }
7775852d539SSimon Glass
rk_edp_init_video(struct rk3288_edp * regs)7785852d539SSimon Glass static void rk_edp_init_video(struct rk3288_edp *regs)
7795852d539SSimon Glass {
7805852d539SSimon Glass writel(VSYNC_DET | VID_FORMAT_CHG | VID_CLK_CHG,
7815852d539SSimon Glass ®s->common_int_sta_1);
7825852d539SSimon Glass writel(CHA_CRI(4) | CHA_CTRL, ®s->sys_ctl_2);
7835852d539SSimon Glass writel(VID_HRES_TH(2) | VID_VRES_TH(0), ®s->video_ctl_8);
7845852d539SSimon Glass }
7855852d539SSimon Glass
rk_edp_config_video_slave_mode(struct rk3288_edp * regs)7865852d539SSimon Glass static void rk_edp_config_video_slave_mode(struct rk3288_edp *regs)
7875852d539SSimon Glass {
7885852d539SSimon Glass clrbits_le32(®s->func_en_1, VID_FIFO_FUNC_EN_N | VID_CAP_FUNC_EN_N);
7895852d539SSimon Glass }
7905852d539SSimon Glass
rk_edp_set_video_cr_mn(struct rk3288_edp * regs,enum clock_recovery_m_value_type type,u32 m_value,u32 n_value)7915852d539SSimon Glass static void rk_edp_set_video_cr_mn(struct rk3288_edp *regs,
7925852d539SSimon Glass enum clock_recovery_m_value_type type,
7935852d539SSimon Glass u32 m_value,
7945852d539SSimon Glass u32 n_value)
7955852d539SSimon Glass {
7965852d539SSimon Glass if (type == REGISTER_M) {
7975852d539SSimon Glass setbits_le32(®s->sys_ctl_4, FIX_M_VID);
7985852d539SSimon Glass writel(m_value & 0xff, ®s->m_vid_0);
7995852d539SSimon Glass writel((m_value >> 8) & 0xff, ®s->m_vid_1);
8005852d539SSimon Glass writel((m_value >> 16) & 0xff, ®s->m_vid_2);
8015852d539SSimon Glass
8025852d539SSimon Glass writel(n_value & 0xf, ®s->n_vid_0);
8035852d539SSimon Glass writel((n_value >> 8) & 0xff, ®s->n_vid_1);
8045852d539SSimon Glass writel((n_value >> 16) & 0xff, ®s->n_vid_2);
8055852d539SSimon Glass } else {
8065852d539SSimon Glass clrbits_le32(®s->sys_ctl_4, FIX_M_VID);
8075852d539SSimon Glass
8085852d539SSimon Glass writel(0x00, ®s->n_vid_0);
8095852d539SSimon Glass writel(0x80, ®s->n_vid_1);
8105852d539SSimon Glass writel(0x00, ®s->n_vid_2);
8115852d539SSimon Glass }
8125852d539SSimon Glass }
8135852d539SSimon Glass
rk_edp_is_video_stream_clock_on(struct rk3288_edp * regs)8145852d539SSimon Glass static int rk_edp_is_video_stream_clock_on(struct rk3288_edp *regs)
8155852d539SSimon Glass {
8165852d539SSimon Glass ulong start;
8175852d539SSimon Glass u32 val;
8185852d539SSimon Glass
8195852d539SSimon Glass start = get_timer(0);
8205852d539SSimon Glass do {
8215852d539SSimon Glass val = readl(®s->sys_ctl_1);
8225852d539SSimon Glass
8235852d539SSimon Glass /* must write value to update DET_STA bit status */
8245852d539SSimon Glass writel(val, ®s->sys_ctl_1);
8255852d539SSimon Glass val = readl(®s->sys_ctl_1);
8265852d539SSimon Glass if (!(val & DET_STA))
8275852d539SSimon Glass continue;
8285852d539SSimon Glass
8295852d539SSimon Glass val = readl(®s->sys_ctl_2);
8305852d539SSimon Glass
8315852d539SSimon Glass /* must write value to update CHA_STA bit status */
8325852d539SSimon Glass writel(val, ®s->sys_ctl_2);
8335852d539SSimon Glass val = readl(®s->sys_ctl_2);
8345852d539SSimon Glass if (!(val & CHA_STA))
8355852d539SSimon Glass return 0;
8365852d539SSimon Glass
8375852d539SSimon Glass } while (get_timer(start) < 100);
8385852d539SSimon Glass
8395852d539SSimon Glass return -ETIMEDOUT;
8405852d539SSimon Glass }
8415852d539SSimon Glass
rk_edp_is_video_stream_on(struct rk_edp_priv * edp)8425852d539SSimon Glass static int rk_edp_is_video_stream_on(struct rk_edp_priv *edp)
8435852d539SSimon Glass {
8445852d539SSimon Glass ulong start;
8455852d539SSimon Glass u32 val;
8465852d539SSimon Glass
8475852d539SSimon Glass start = get_timer(0);
8485852d539SSimon Glass do {
8495852d539SSimon Glass val = readl(&edp->regs->sys_ctl_3);
8505852d539SSimon Glass
8515852d539SSimon Glass /* must write value to update STRM_VALID bit status */
8525852d539SSimon Glass writel(val, &edp->regs->sys_ctl_3);
8535852d539SSimon Glass
8545852d539SSimon Glass val = readl(&edp->regs->sys_ctl_3);
8555852d539SSimon Glass if (!(val & STRM_VALID))
8565852d539SSimon Glass return 0;
8575852d539SSimon Glass } while (get_timer(start) < 100);
8585852d539SSimon Glass
8595852d539SSimon Glass return -ETIMEDOUT;
8605852d539SSimon Glass }
8615852d539SSimon Glass
rk_edp_config_video(struct rk_edp_priv * edp)8625852d539SSimon Glass static int rk_edp_config_video(struct rk_edp_priv *edp)
8635852d539SSimon Glass {
8645852d539SSimon Glass int ret;
8655852d539SSimon Glass
8665852d539SSimon Glass rk_edp_config_video_slave_mode(edp->regs);
8675852d539SSimon Glass
8685852d539SSimon Glass if (!rk_edp_get_pll_locked(edp->regs)) {
8695852d539SSimon Glass debug("PLL is not locked yet.\n");
8705852d539SSimon Glass return -ETIMEDOUT;
8715852d539SSimon Glass }
8725852d539SSimon Glass
8735852d539SSimon Glass ret = rk_edp_is_video_stream_clock_on(edp->regs);
8745852d539SSimon Glass if (ret)
8755852d539SSimon Glass return ret;
8765852d539SSimon Glass
8775852d539SSimon Glass /* Set to use the register calculated M/N video */
8785852d539SSimon Glass rk_edp_set_video_cr_mn(edp->regs, CALCULATED_M, 0, 0);
8795852d539SSimon Glass
8805852d539SSimon Glass /* For video bist, Video timing must be generated by register */
8815852d539SSimon Glass clrbits_le32(&edp->regs->video_ctl_10, F_SEL);
8825852d539SSimon Glass
8835852d539SSimon Glass /* Disable video mute */
8845852d539SSimon Glass clrbits_le32(&edp->regs->video_ctl_1, VIDEO_MUTE);
8855852d539SSimon Glass
8865852d539SSimon Glass /* Enable video at next frame */
8875852d539SSimon Glass setbits_le32(&edp->regs->video_ctl_1, VIDEO_EN);
8885852d539SSimon Glass
8895852d539SSimon Glass return rk_edp_is_video_stream_on(edp);
8905852d539SSimon Glass }
8915852d539SSimon Glass
rockchip_edp_force_hpd(struct rk_edp_priv * edp)8925852d539SSimon Glass static void rockchip_edp_force_hpd(struct rk_edp_priv *edp)
8935852d539SSimon Glass {
8945852d539SSimon Glass setbits_le32(&edp->regs->sys_ctl_3, F_HPD | HPD_CTRL);
8955852d539SSimon Glass }
8965852d539SSimon Glass
rockchip_edp_get_plug_in_status(struct rk_edp_priv * edp)8975852d539SSimon Glass static int rockchip_edp_get_plug_in_status(struct rk_edp_priv *edp)
8985852d539SSimon Glass {
8995852d539SSimon Glass u32 val;
9005852d539SSimon Glass
9015852d539SSimon Glass val = readl(&edp->regs->sys_ctl_3);
9025852d539SSimon Glass if (val & HPD_STATUS)
9035852d539SSimon Glass return 1;
9045852d539SSimon Glass
9055852d539SSimon Glass return 0;
9065852d539SSimon Glass }
9075852d539SSimon Glass
9085852d539SSimon Glass /*
9095852d539SSimon Glass * support edp HPD function
9105852d539SSimon Glass * some hardware version do not support edp hdp,
9115852d539SSimon Glass * we use 200ms to try to get the hpd single now,
9125852d539SSimon Glass * if we can not get edp hpd single, it will delay 200ms,
9135852d539SSimon Glass * also meet the edp power timing request, to compatible
9145852d539SSimon Glass * all of the hardware version
9155852d539SSimon Glass */
rockchip_edp_wait_hpd(struct rk_edp_priv * edp)9165852d539SSimon Glass static void rockchip_edp_wait_hpd(struct rk_edp_priv *edp)
9175852d539SSimon Glass {
9185852d539SSimon Glass ulong start;
9195852d539SSimon Glass
9205852d539SSimon Glass start = get_timer(0);
9215852d539SSimon Glass do {
9225852d539SSimon Glass if (rockchip_edp_get_plug_in_status(edp))
9235852d539SSimon Glass return;
9245852d539SSimon Glass udelay(100);
9255852d539SSimon Glass } while (get_timer(start) < 200);
9265852d539SSimon Glass
9275852d539SSimon Glass debug("do not get hpd single, force hpd\n");
9285852d539SSimon Glass rockchip_edp_force_hpd(edp);
9295852d539SSimon Glass }
9305852d539SSimon Glass
rk_edp_enable(struct udevice * dev,int panel_bpp,const struct display_timing * edid)9315852d539SSimon Glass static int rk_edp_enable(struct udevice *dev, int panel_bpp,
9325852d539SSimon Glass const struct display_timing *edid)
9335852d539SSimon Glass {
9345852d539SSimon Glass struct rk_edp_priv *priv = dev_get_priv(dev);
9355852d539SSimon Glass int ret = 0;
9365852d539SSimon Glass
9375852d539SSimon Glass ret = rk_edp_set_link_train(priv);
9385852d539SSimon Glass if (ret) {
9395852d539SSimon Glass printf("link train failed!\n");
9405852d539SSimon Glass return ret;
9415852d539SSimon Glass }
9425852d539SSimon Glass
9435852d539SSimon Glass rk_edp_init_video(priv->regs);
9445852d539SSimon Glass ret = rk_edp_config_video(priv);
9455852d539SSimon Glass if (ret) {
9465852d539SSimon Glass printf("config video failed\n");
9475852d539SSimon Glass return ret;
9485852d539SSimon Glass }
9495852d539SSimon Glass ret = panel_enable_backlight(priv->panel);
9505852d539SSimon Glass if (ret) {
9515852d539SSimon Glass debug("%s: backlight error: %d\n", __func__, ret);
9525852d539SSimon Glass return ret;
9535852d539SSimon Glass }
9545852d539SSimon Glass
9555852d539SSimon Glass return 0;
9565852d539SSimon Glass }
9575852d539SSimon Glass
rk_edp_read_edid(struct udevice * dev,u8 * buf,int buf_size)9585852d539SSimon Glass static int rk_edp_read_edid(struct udevice *dev, u8 *buf, int buf_size)
9595852d539SSimon Glass {
9605852d539SSimon Glass struct rk_edp_priv *priv = dev_get_priv(dev);
9615852d539SSimon Glass u32 edid_size = EDID_LENGTH;
9625852d539SSimon Glass int ret;
9635852d539SSimon Glass int i;
9645852d539SSimon Glass
9655852d539SSimon Glass for (i = 0; i < 3; i++) {
9665852d539SSimon Glass ret = rk_edp_i2c_read(priv->regs, EDID_ADDR, EDID_HEADER,
9675852d539SSimon Glass EDID_LENGTH, &buf[EDID_HEADER]);
9685852d539SSimon Glass if (ret) {
9695852d539SSimon Glass debug("EDID read failed\n");
9705852d539SSimon Glass continue;
9715852d539SSimon Glass }
9725852d539SSimon Glass
9735852d539SSimon Glass /*
9745852d539SSimon Glass * check if the EDID has an extension flag, and read additional
9755852d539SSimon Glass * EDID data if needed
9765852d539SSimon Glass */
9775852d539SSimon Glass if (buf[EDID_EXTENSION_FLAG]) {
9785852d539SSimon Glass edid_size += EDID_LENGTH;
9795852d539SSimon Glass ret = rk_edp_i2c_read(priv->regs, EDID_ADDR,
9805852d539SSimon Glass EDID_LENGTH, EDID_LENGTH,
9815852d539SSimon Glass &buf[EDID_LENGTH]);
9825852d539SSimon Glass if (ret) {
9835852d539SSimon Glass debug("EDID Read failed!\n");
9845852d539SSimon Glass continue;
9855852d539SSimon Glass }
9865852d539SSimon Glass }
9875852d539SSimon Glass goto done;
9885852d539SSimon Glass }
9895852d539SSimon Glass
9905852d539SSimon Glass /* After 3 attempts, give up */
9915852d539SSimon Glass return ret;
9925852d539SSimon Glass
9935852d539SSimon Glass done:
9945852d539SSimon Glass return edid_size;
9955852d539SSimon Glass }
9965852d539SSimon Glass
rk_edp_ofdata_to_platdata(struct udevice * dev)9975852d539SSimon Glass static int rk_edp_ofdata_to_platdata(struct udevice *dev)
9985852d539SSimon Glass {
9995852d539SSimon Glass struct rk_edp_priv *priv = dev_get_priv(dev);
10005852d539SSimon Glass
1001a821c4afSSimon Glass priv->regs = (struct rk3288_edp *)devfdt_get_addr(dev);
10025852d539SSimon Glass priv->grf = syscon_get_first_range(ROCKCHIP_SYSCON_GRF);
10035852d539SSimon Glass
10045852d539SSimon Glass return 0;
10055852d539SSimon Glass }
10065852d539SSimon Glass
rk_edp_remove(struct udevice * dev)1007*f418676eSSimon Glass static int rk_edp_remove(struct udevice *dev)
1008*f418676eSSimon Glass {
1009*f418676eSSimon Glass struct rk_edp_priv *priv = dev_get_priv(dev);
1010*f418676eSSimon Glass struct rk3288_edp *regs = priv->regs;
1011*f418676eSSimon Glass
1012*f418676eSSimon Glass setbits_le32(®s->video_ctl_1, VIDEO_MUTE);
1013*f418676eSSimon Glass clrbits_le32(®s->video_ctl_1, VIDEO_EN);
1014*f418676eSSimon Glass clrbits_le32(®s->sys_ctl_3, F_HPD | HPD_CTRL);
1015*f418676eSSimon Glass setbits_le32(®s->func_en_1, SW_FUNC_EN_N);
1016*f418676eSSimon Glass
1017*f418676eSSimon Glass return 0;
1018*f418676eSSimon Glass }
1019*f418676eSSimon Glass
rk_edp_probe(struct udevice * dev)1020*f418676eSSimon Glass static int rk_edp_probe(struct udevice *dev)
10215852d539SSimon Glass {
10225852d539SSimon Glass struct display_plat *uc_plat = dev_get_uclass_platdata(dev);
10235852d539SSimon Glass struct rk_edp_priv *priv = dev_get_priv(dev);
10245852d539SSimon Glass struct rk3288_edp *regs = priv->regs;
1025135aa950SStephen Warren struct clk clk;
10265852d539SSimon Glass int ret;
10275852d539SSimon Glass
10285852d539SSimon Glass ret = uclass_get_device_by_phandle(UCLASS_PANEL, dev, "rockchip,panel",
10295852d539SSimon Glass &priv->panel);
10305852d539SSimon Glass if (ret) {
10315852d539SSimon Glass debug("%s: Cannot find panel for '%s' (ret=%d)\n", __func__,
10325852d539SSimon Glass dev->name, ret);
10335852d539SSimon Glass return ret;
10345852d539SSimon Glass }
10355852d539SSimon Glass
10365852d539SSimon Glass int vop_id = uc_plat->source_id;
10375852d539SSimon Glass debug("%s, uc_plat=%p, vop_id=%u\n", __func__, uc_plat, vop_id);
10385852d539SSimon Glass
10395852d539SSimon Glass ret = clk_get_by_index(dev, 1, &clk);
10405852d539SSimon Glass if (ret >= 0) {
1041135aa950SStephen Warren ret = clk_set_rate(&clk, 0);
1042135aa950SStephen Warren clk_free(&clk);
10435852d539SSimon Glass }
10445852d539SSimon Glass if (ret) {
10455852d539SSimon Glass debug("%s: Failed to set EDP clock: ret=%d\n", __func__, ret);
10465852d539SSimon Glass return ret;
10475852d539SSimon Glass }
10485852d539SSimon Glass
10495852d539SSimon Glass ret = clk_get_by_index(uc_plat->src_dev, 0, &clk);
10505852d539SSimon Glass if (ret >= 0) {
1051135aa950SStephen Warren ret = clk_set_rate(&clk, 192000000);
1052135aa950SStephen Warren clk_free(&clk);
10535852d539SSimon Glass }
10545852d539SSimon Glass if (ret < 0) {
10555852d539SSimon Glass debug("%s: Failed to set clock in source device '%s': ret=%d\n",
10565852d539SSimon Glass __func__, uc_plat->src_dev->name, ret);
10575852d539SSimon Glass return ret;
10585852d539SSimon Glass }
10595852d539SSimon Glass
10605852d539SSimon Glass /* grf_edp_ref_clk_sel: from internal 24MHz or 27MHz clock */
10615852d539SSimon Glass rk_setreg(&priv->grf->soc_con12, 1 << 4);
10625852d539SSimon Glass
10635852d539SSimon Glass /* select epd signal from vop0 or vop1 */
10645852d539SSimon Glass rk_setreg(&priv->grf->soc_con6, (vop_id == 1) ? (1 << 5) : (1 << 5));
10655852d539SSimon Glass
10665852d539SSimon Glass rockchip_edp_wait_hpd(priv);
10675852d539SSimon Glass
10685852d539SSimon Glass rk_edp_init_refclk(regs);
10695852d539SSimon Glass rk_edp_init_interrupt(regs);
10705852d539SSimon Glass rk_edp_enable_sw_function(regs);
10715852d539SSimon Glass ret = rk_edp_init_analog_func(regs);
10725852d539SSimon Glass if (ret)
10735852d539SSimon Glass return ret;
10745852d539SSimon Glass rk_edp_init_aux(regs);
10755852d539SSimon Glass
10765852d539SSimon Glass return 0;
10775852d539SSimon Glass }
10785852d539SSimon Glass
10795852d539SSimon Glass static const struct dm_display_ops dp_rockchip_ops = {
10805852d539SSimon Glass .read_edid = rk_edp_read_edid,
10815852d539SSimon Glass .enable = rk_edp_enable,
10825852d539SSimon Glass };
10835852d539SSimon Glass
10845852d539SSimon Glass static const struct udevice_id rockchip_dp_ids[] = {
10855852d539SSimon Glass { .compatible = "rockchip,rk3288-edp" },
10865852d539SSimon Glass { }
10875852d539SSimon Glass };
10885852d539SSimon Glass
10895852d539SSimon Glass U_BOOT_DRIVER(dp_rockchip) = {
10905852d539SSimon Glass .name = "edp_rockchip",
10915852d539SSimon Glass .id = UCLASS_DISPLAY,
10925852d539SSimon Glass .of_match = rockchip_dp_ids,
10935852d539SSimon Glass .ops = &dp_rockchip_ops,
10945852d539SSimon Glass .ofdata_to_platdata = rk_edp_ofdata_to_platdata,
10955852d539SSimon Glass .probe = rk_edp_probe,
1096*f418676eSSimon Glass .remove = rk_edp_remove,
10975852d539SSimon Glass .priv_auto_alloc_size = sizeof(struct rk_edp_priv),
10985852d539SSimon Glass };
1099