mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/chenhuacai/linux-loongson
synced 2025-09-04 18:49:41 +00:00

In some application scenarios, we hope to get the corresponding connector when the bridge's detect hook is invoked. In most cases, we can get the connector by drm_atomic_get_connector_for_encoder if the encoder attached to the bridge is enabled, however there will still be some scenarios where the detect hook of the bridge is called but the corresponding encoder has not been enabled yet. For instance, this occurs when the device is hot plug in for the first time. Since the call to bridge's detect is initiated by the connector, passing down the corresponding connector directly will make things simpler. Signed-off-by: Andy Yan <andy.yan@rock-chips.com> Reviewed-by: Dmitry Baryshkov <dmitry.baryshkov@oss.qualcomm.com> Link: https://lore.kernel.org/r/20250703125027.311109-3-andyshrk@163.com [DB: added the chunk to the cdn-dp driver] Signed-off-by: Dmitry Baryshkov <dmitry.baryshkov@oss.qualcomm.com>
2639 lines
67 KiB
C
2639 lines
67 KiB
C
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
/*
|
|
* TC358767/TC358867/TC9595 DSI/DPI-to-DPI/(e)DP bridge driver
|
|
*
|
|
* The TC358767/TC358867/TC9595 can operate in multiple modes.
|
|
* All modes are supported -- DPI->(e)DP / DSI->DPI / DSI->(e)DP .
|
|
*
|
|
* Copyright (C) 2016 CogentEmbedded Inc
|
|
* Author: Andrey Gusakov <andrey.gusakov@cogentembedded.com>
|
|
*
|
|
* Copyright (C) 2016 Pengutronix, Philipp Zabel <p.zabel@pengutronix.de>
|
|
*
|
|
* Copyright (C) 2016 Zodiac Inflight Innovations
|
|
*
|
|
* Initially based on: drivers/gpu/drm/i2c/tda998x_drv.c
|
|
*
|
|
* Copyright (C) 2012 Texas Instruments
|
|
* Author: Rob Clark <robdclark@gmail.com>
|
|
*/
|
|
|
|
#include <linux/bitfield.h>
|
|
#include <linux/clk.h>
|
|
#include <linux/device.h>
|
|
#include <linux/gpio/consumer.h>
|
|
#include <linux/i2c.h>
|
|
#include <linux/kernel.h>
|
|
#include <linux/media-bus-format.h>
|
|
#include <linux/module.h>
|
|
#include <linux/regmap.h>
|
|
#include <linux/slab.h>
|
|
|
|
#include <drm/display/drm_dp_helper.h>
|
|
#include <drm/drm_atomic_helper.h>
|
|
#include <drm/drm_bridge.h>
|
|
#include <drm/drm_edid.h>
|
|
#include <drm/drm_mipi_dsi.h>
|
|
#include <drm/drm_of.h>
|
|
#include <drm/drm_panel.h>
|
|
#include <drm/drm_print.h>
|
|
#include <drm/drm_probe_helper.h>
|
|
|
|
/* Registers */
|
|
|
|
/* DSI D-PHY Layer registers */
|
|
#define D0W_DPHYCONTTX 0x0004
|
|
#define CLW_DPHYCONTTX 0x0020
|
|
#define D0W_DPHYCONTRX 0x0024
|
|
#define D1W_DPHYCONTRX 0x0028
|
|
#define D2W_DPHYCONTRX 0x002c
|
|
#define D3W_DPHYCONTRX 0x0030
|
|
#define COM_DPHYCONTRX 0x0038
|
|
#define CLW_CNTRL 0x0040
|
|
#define D0W_CNTRL 0x0044
|
|
#define D1W_CNTRL 0x0048
|
|
#define D2W_CNTRL 0x004c
|
|
#define D3W_CNTRL 0x0050
|
|
#define TESTMODE_CNTRL 0x0054
|
|
|
|
/* PPI layer registers */
|
|
#define PPI_STARTPPI 0x0104 /* START control bit */
|
|
#define PPI_BUSYPPI 0x0108 /* PPI busy status */
|
|
#define PPI_LPTXTIMECNT 0x0114 /* LPTX timing signal */
|
|
#define LPX_PERIOD 3
|
|
#define PPI_LANEENABLE 0x0134
|
|
#define PPI_TX_RX_TA 0x013c
|
|
#define TTA_GET 0x40000
|
|
#define TTA_SURE 6
|
|
#define PPI_D0S_ATMR 0x0144
|
|
#define PPI_D1S_ATMR 0x0148
|
|
#define PPI_D0S_CLRSIPOCOUNT 0x0164 /* Assertion timer for Lane 0 */
|
|
#define PPI_D1S_CLRSIPOCOUNT 0x0168 /* Assertion timer for Lane 1 */
|
|
#define PPI_D2S_CLRSIPOCOUNT 0x016c /* Assertion timer for Lane 2 */
|
|
#define PPI_D3S_CLRSIPOCOUNT 0x0170 /* Assertion timer for Lane 3 */
|
|
#define PPI_START_FUNCTION BIT(0)
|
|
|
|
/* DSI layer registers */
|
|
#define DSI_STARTDSI 0x0204 /* START control bit of DSI-TX */
|
|
#define DSI_BUSYDSI 0x0208 /* DSI busy status */
|
|
#define DSI_LANEENABLE 0x0210 /* Enables each lane */
|
|
#define DSI_RX_START BIT(0)
|
|
|
|
/* Lane enable PPI and DSI register bits */
|
|
#define LANEENABLE_CLEN BIT(0)
|
|
#define LANEENABLE_L0EN BIT(1)
|
|
#define LANEENABLE_L1EN BIT(2)
|
|
#define LANEENABLE_L2EN BIT(1)
|
|
#define LANEENABLE_L3EN BIT(2)
|
|
|
|
#define DSI_LANESTATUS0 0x0214 /* DSI lane status 0 */
|
|
#define DSI_LANESTATUS1 0x0218 /* DSI lane status 1 */
|
|
#define DSI_INTSTATUS 0x0220 /* Interrupt Status */
|
|
#define DSI_INTMASK 0x0224 /* Interrupt Mask */
|
|
#define DSI_INTCLR 0x0228 /* Interrupt Clear */
|
|
#define DSI_LPTXTO 0x0230 /* LPTX Time Out Counter */
|
|
|
|
/* DSI General Registers */
|
|
#define DSIERRCNT 0x0300 /* DSI Error Count Register */
|
|
|
|
/* DSI Application Layer Registers */
|
|
#define APLCTRL 0x0400 /* Application layer Control Register */
|
|
#define RDPKTLN 0x0404 /* DSI Read packet Length Register */
|
|
|
|
/* Display Parallel Input Interface */
|
|
#define DPIPXLFMT 0x0440
|
|
#define VS_POL_ACTIVE_LOW (1 << 10)
|
|
#define HS_POL_ACTIVE_LOW (1 << 9)
|
|
#define DE_POL_ACTIVE_HIGH (0 << 8)
|
|
#define SUB_CFG_TYPE_CONFIG1 (0 << 2) /* LSB aligned */
|
|
#define SUB_CFG_TYPE_CONFIG2 (1 << 2) /* Loosely Packed */
|
|
#define SUB_CFG_TYPE_CONFIG3 (2 << 2) /* LSB aligned 8-bit */
|
|
#define DPI_BPP_RGB888 (0 << 0)
|
|
#define DPI_BPP_RGB666 (1 << 0)
|
|
#define DPI_BPP_RGB565 (2 << 0)
|
|
|
|
/* Display Parallel Output Interface */
|
|
#define POCTRL 0x0448
|
|
#define POCTRL_S2P BIT(7)
|
|
#define POCTRL_PCLK_POL BIT(3)
|
|
#define POCTRL_VS_POL BIT(2)
|
|
#define POCTRL_HS_POL BIT(1)
|
|
#define POCTRL_DE_POL BIT(0)
|
|
|
|
/* Video Path */
|
|
#define VPCTRL0 0x0450
|
|
#define VSDELAY GENMASK(31, 20)
|
|
#define OPXLFMT_RGB666 (0 << 8)
|
|
#define OPXLFMT_RGB888 (1 << 8)
|
|
#define FRMSYNC_DISABLED (0 << 4) /* Video Timing Gen Disabled */
|
|
#define FRMSYNC_ENABLED (1 << 4) /* Video Timing Gen Enabled */
|
|
#define MSF_DISABLED (0 << 0) /* Magic Square FRC disabled */
|
|
#define MSF_ENABLED (1 << 0) /* Magic Square FRC enabled */
|
|
#define HTIM01 0x0454
|
|
#define HPW GENMASK(8, 0)
|
|
#define HBPR GENMASK(24, 16)
|
|
#define HTIM02 0x0458
|
|
#define HDISPR GENMASK(10, 0)
|
|
#define HFPR GENMASK(24, 16)
|
|
#define VTIM01 0x045c
|
|
#define VSPR GENMASK(7, 0)
|
|
#define VBPR GENMASK(23, 16)
|
|
#define VTIM02 0x0460
|
|
#define VFPR GENMASK(23, 16)
|
|
#define VDISPR GENMASK(10, 0)
|
|
#define VFUEN0 0x0464
|
|
#define VFUEN BIT(0) /* Video Frame Timing Upload */
|
|
|
|
/* System */
|
|
#define TC_IDREG 0x0500 /* Chip ID and Revision ID */
|
|
#define SYSBOOT 0x0504 /* System BootStrap Status Register */
|
|
#define SYSSTAT 0x0508 /* System Status Register */
|
|
#define SYSRSTENB 0x050c /* System Reset/Enable Register */
|
|
#define ENBI2C (1 << 0)
|
|
#define ENBLCD0 (1 << 2)
|
|
#define ENBBM (1 << 3)
|
|
#define ENBDSIRX (1 << 4)
|
|
#define ENBREG (1 << 5)
|
|
#define ENBHDCP (1 << 8)
|
|
#define SYSCTRL 0x0510 /* System Control Register */
|
|
#define DP0_AUDSRC_NO_INPUT (0 << 3)
|
|
#define DP0_AUDSRC_I2S_RX (1 << 3)
|
|
#define DP0_VIDSRC_NO_INPUT (0 << 0)
|
|
#define DP0_VIDSRC_DSI_RX (1 << 0)
|
|
#define DP0_VIDSRC_DPI_RX (2 << 0)
|
|
#define DP0_VIDSRC_COLOR_BAR (3 << 0)
|
|
#define GPIOM 0x0540 /* GPIO Mode Control Register */
|
|
#define GPIOC 0x0544 /* GPIO Direction Control Register */
|
|
#define GPIOO 0x0548 /* GPIO Output Register */
|
|
#define GPIOI 0x054c /* GPIO Input Register */
|
|
#define INTCTL_G 0x0560 /* General Interrupts Control Register */
|
|
#define INTSTS_G 0x0564 /* General Interrupts Status Register */
|
|
|
|
#define INT_SYSERR BIT(16)
|
|
#define INT_GPIO_H(x) (1 << (x == 0 ? 2 : 10))
|
|
#define INT_GPIO_LC(x) (1 << (x == 0 ? 3 : 11))
|
|
|
|
#define TEST_INT_C 0x0570 /* Test Interrupts Control Register */
|
|
#define TEST_INT_S 0x0574 /* Test Interrupts Status Register */
|
|
|
|
#define INT_GP0_LCNT 0x0584 /* Interrupt GPIO0 Low Count Value Register */
|
|
#define INT_GP1_LCNT 0x0588 /* Interrupt GPIO1 Low Count Value Register */
|
|
|
|
/* Control */
|
|
#define DP0CTL 0x0600
|
|
#define VID_MN_GEN BIT(6) /* Auto-generate M/N values */
|
|
#define EF_EN BIT(5) /* Enable Enhanced Framing */
|
|
#define VID_EN BIT(1) /* Video transmission enable */
|
|
#define DP_EN BIT(0) /* Enable DPTX function */
|
|
|
|
/* Clocks */
|
|
#define DP0_VIDMNGEN0 0x0610 /* DP0 Video Force M Value Register */
|
|
#define DP0_VIDMNGEN1 0x0614 /* DP0 Video Force N Value Register */
|
|
#define DP0_VMNGENSTATUS 0x0618 /* DP0 Video Current M Value Register */
|
|
#define DP0_AUDMNGEN0 0x0628 /* DP0 Audio Force M Value Register */
|
|
#define DP0_AUDMNGEN1 0x062c /* DP0 Audio Force N Value Register */
|
|
#define DP0_AMNGENSTATUS 0x0630 /* DP0 Audio Current M Value Register */
|
|
|
|
/* Main Channel */
|
|
#define DP0_SECSAMPLE 0x0640
|
|
#define DP0_VIDSYNCDELAY 0x0644
|
|
#define VID_SYNC_DLY GENMASK(15, 0)
|
|
#define THRESH_DLY GENMASK(31, 16)
|
|
|
|
#define DP0_TOTALVAL 0x0648
|
|
#define H_TOTAL GENMASK(15, 0)
|
|
#define V_TOTAL GENMASK(31, 16)
|
|
#define DP0_STARTVAL 0x064c
|
|
#define H_START GENMASK(15, 0)
|
|
#define V_START GENMASK(31, 16)
|
|
#define DP0_ACTIVEVAL 0x0650
|
|
#define H_ACT GENMASK(15, 0)
|
|
#define V_ACT GENMASK(31, 16)
|
|
|
|
#define DP0_SYNCVAL 0x0654
|
|
#define VS_WIDTH GENMASK(30, 16)
|
|
#define HS_WIDTH GENMASK(14, 0)
|
|
#define SYNCVAL_HS_POL_ACTIVE_LOW (1 << 15)
|
|
#define SYNCVAL_VS_POL_ACTIVE_LOW (1 << 31)
|
|
#define DP0_MISC 0x0658
|
|
#define TU_SIZE_RECOMMENDED (63) /* LSCLK cycles per TU */
|
|
#define MAX_TU_SYMBOL GENMASK(28, 23)
|
|
#define TU_SIZE GENMASK(21, 16)
|
|
#define BPC_6 (0 << 5)
|
|
#define BPC_8 (1 << 5)
|
|
|
|
/* AUX channel */
|
|
#define DP0_AUXCFG0 0x0660
|
|
#define DP0_AUXCFG0_BSIZE GENMASK(11, 8)
|
|
#define DP0_AUXCFG0_ADDR_ONLY BIT(4)
|
|
#define DP0_AUXCFG1 0x0664
|
|
#define AUX_RX_FILTER_EN BIT(16)
|
|
|
|
#define DP0_AUXADDR 0x0668
|
|
#define DP0_AUXWDATA(i) (0x066c + (i) * 4)
|
|
#define DP0_AUXRDATA(i) (0x067c + (i) * 4)
|
|
#define DP0_AUXSTATUS 0x068c
|
|
#define AUX_BYTES GENMASK(15, 8)
|
|
#define AUX_STATUS GENMASK(7, 4)
|
|
#define AUX_TIMEOUT BIT(1)
|
|
#define AUX_BUSY BIT(0)
|
|
#define DP0_AUXI2CADR 0x0698
|
|
|
|
/* Link Training */
|
|
#define DP0_SRCCTRL 0x06a0
|
|
#define DP0_SRCCTRL_PRE1 GENMASK(29, 28)
|
|
#define DP0_SRCCTRL_SWG1 GENMASK(25, 24)
|
|
#define DP0_SRCCTRL_PRE0 GENMASK(21, 20)
|
|
#define DP0_SRCCTRL_SWG0 GENMASK(17, 16)
|
|
#define DP0_SRCCTRL_SCRMBLDIS BIT(13)
|
|
#define DP0_SRCCTRL_EN810B BIT(12)
|
|
#define DP0_SRCCTRL_NOTP (0 << 8)
|
|
#define DP0_SRCCTRL_TP1 (1 << 8)
|
|
#define DP0_SRCCTRL_TP2 (2 << 8)
|
|
#define DP0_SRCCTRL_LANESKEW BIT(7)
|
|
#define DP0_SRCCTRL_SSCG BIT(3)
|
|
#define DP0_SRCCTRL_LANES_1 (0 << 2)
|
|
#define DP0_SRCCTRL_LANES_2 (1 << 2)
|
|
#define DP0_SRCCTRL_BW27 (1 << 1)
|
|
#define DP0_SRCCTRL_BW162 (0 << 1)
|
|
#define DP0_SRCCTRL_AUTOCORRECT BIT(0)
|
|
#define DP0_LTSTAT 0x06d0
|
|
#define LT_LOOPDONE BIT(13)
|
|
#define LT_STATUS_MASK (0x1f << 8)
|
|
#define LT_CHANNEL1_EQ_BITS (DP_CHANNEL_EQ_BITS << 4)
|
|
#define LT_INTERLANE_ALIGN_DONE BIT(3)
|
|
#define LT_CHANNEL0_EQ_BITS (DP_CHANNEL_EQ_BITS)
|
|
#define DP0_SNKLTCHGREQ 0x06d4
|
|
#define DP0_LTLOOPCTRL 0x06d8
|
|
#define DP0_SNKLTCTRL 0x06e4
|
|
#define DP0_TPATDAT0 0x06e8 /* DP0 Test Pattern bits 29 to 0 */
|
|
#define DP0_TPATDAT1 0x06ec /* DP0 Test Pattern bits 59 to 30 */
|
|
#define DP0_TPATDAT2 0x06f0 /* DP0 Test Pattern bits 89 to 60 */
|
|
#define DP0_TPATDAT3 0x06f4 /* DP0 Test Pattern bits 119 to 90 */
|
|
|
|
#define AUDCFG0 0x0700 /* DP0 Audio Config0 Register */
|
|
#define AUDCFG1 0x0704 /* DP0 Audio Config1 Register */
|
|
#define AUDIFDATA0 0x0708 /* DP0 Audio Info Frame Bytes 3 to 0 */
|
|
#define AUDIFDATA1 0x070c /* DP0 Audio Info Frame Bytes 7 to 4 */
|
|
#define AUDIFDATA2 0x0710 /* DP0 Audio Info Frame Bytes 11 to 8 */
|
|
#define AUDIFDATA3 0x0714 /* DP0 Audio Info Frame Bytes 15 to 12 */
|
|
#define AUDIFDATA4 0x0718 /* DP0 Audio Info Frame Bytes 19 to 16 */
|
|
#define AUDIFDATA5 0x071c /* DP0 Audio Info Frame Bytes 23 to 20 */
|
|
#define AUDIFDATA6 0x0720 /* DP0 Audio Info Frame Bytes 27 to 24 */
|
|
|
|
#define DP1_SRCCTRL 0x07a0 /* DP1 Control Register */
|
|
#define DP1_SRCCTRL_PRE GENMASK(21, 20)
|
|
#define DP1_SRCCTRL_SWG GENMASK(17, 16)
|
|
|
|
/* PHY */
|
|
#define DP_PHY_CTRL 0x0800
|
|
#define DP_PHY_RST BIT(28) /* DP PHY Global Soft Reset */
|
|
#define BGREN BIT(25) /* AUX PHY BGR Enable */
|
|
#define PWR_SW_EN BIT(24) /* PHY Power Switch Enable */
|
|
#define PHY_M1_RST BIT(12) /* Reset PHY1 Main Channel */
|
|
#define PHY_RDY BIT(16) /* PHY Main Channels Ready */
|
|
#define PHY_M0_RST BIT(8) /* Reset PHY0 Main Channel */
|
|
#define PHY_2LANE BIT(2) /* PHY Enable 2 lanes */
|
|
#define PHY_A0_EN BIT(1) /* PHY Aux Channel0 Enable */
|
|
#define PHY_M0_EN BIT(0) /* PHY Main Channel0 Enable */
|
|
#define DP_PHY_CFG_WR 0x0810 /* DP PHY Configuration Test Write Register */
|
|
#define DP_PHY_CFG_RD 0x0814 /* DP PHY Configuration Test Read Register */
|
|
#define DP0_AUX_PHY_CTRL 0x0820 /* DP0 AUX PHY Control Register */
|
|
#define DP0_MAIN_PHY_DBG 0x0840 /* DP0 Main PHY Test Debug Register */
|
|
|
|
/* I2S */
|
|
#define I2SCFG 0x0880 /* I2S Audio Config 0 Register */
|
|
#define I2SCH0STAT0 0x0888 /* I2S Audio Channel 0 Status Bytes 3 to 0 */
|
|
#define I2SCH0STAT1 0x088c /* I2S Audio Channel 0 Status Bytes 7 to 4 */
|
|
#define I2SCH0STAT2 0x0890 /* I2S Audio Channel 0 Status Bytes 11 to 8 */
|
|
#define I2SCH0STAT3 0x0894 /* I2S Audio Channel 0 Status Bytes 15 to 12 */
|
|
#define I2SCH0STAT4 0x0898 /* I2S Audio Channel 0 Status Bytes 19 to 16 */
|
|
#define I2SCH0STAT5 0x089c /* I2S Audio Channel 0 Status Bytes 23 to 20 */
|
|
#define I2SCH1STAT0 0x08a0 /* I2S Audio Channel 1 Status Bytes 3 to 0 */
|
|
#define I2SCH1STAT1 0x08a4 /* I2S Audio Channel 1 Status Bytes 7 to 4 */
|
|
#define I2SCH1STAT2 0x08a8 /* I2S Audio Channel 1 Status Bytes 11 to 8 */
|
|
#define I2SCH1STAT3 0x08ac /* I2S Audio Channel 1 Status Bytes 15 to 12 */
|
|
#define I2SCH1STAT4 0x08b0 /* I2S Audio Channel 1 Status Bytes 19 to 16 */
|
|
#define I2SCH1STAT5 0x08b4 /* I2S Audio Channel 1 Status Bytes 23 to 20 */
|
|
|
|
/* PLL */
|
|
#define DP0_PLLCTRL 0x0900
|
|
#define DP1_PLLCTRL 0x0904 /* not defined in DS */
|
|
#define PXL_PLLCTRL 0x0908
|
|
#define PLLUPDATE BIT(2)
|
|
#define PLLBYP BIT(1)
|
|
#define PLLEN BIT(0)
|
|
#define PXL_PLLPARAM 0x0914
|
|
#define IN_SEL_REFCLK (0 << 14)
|
|
#define SYS_PLLPARAM 0x0918
|
|
#define REF_FREQ_38M4 (0 << 8) /* 38.4 MHz */
|
|
#define REF_FREQ_19M2 (1 << 8) /* 19.2 MHz */
|
|
#define REF_FREQ_26M (2 << 8) /* 26 MHz */
|
|
#define REF_FREQ_13M (3 << 8) /* 13 MHz */
|
|
#define SYSCLK_SEL_LSCLK (0 << 4)
|
|
#define LSCLK_DIV_1 (0 << 0)
|
|
#define LSCLK_DIV_2 (1 << 0)
|
|
|
|
/* Test & Debug */
|
|
#define TSTCTL 0x0a00
|
|
#define COLOR_R GENMASK(31, 24)
|
|
#define COLOR_G GENMASK(23, 16)
|
|
#define COLOR_B GENMASK(15, 8)
|
|
#define ENI2CFILTER BIT(4)
|
|
#define COLOR_BAR_MODE GENMASK(1, 0)
|
|
#define COLOR_BAR_MODE_BARS 2
|
|
#define PLL_DBG 0x0a04
|
|
|
|
enum tc_mode {
|
|
mode_dpi_to_edp = BIT(1) | BIT(2),
|
|
mode_dpi_to_dp = BIT(1),
|
|
mode_dsi_to_edp = BIT(0) | BIT(2),
|
|
mode_dsi_to_dp = BIT(0),
|
|
mode_dsi_to_dpi = BIT(0) | BIT(1),
|
|
};
|
|
|
|
static bool tc_test_pattern;
|
|
module_param_named(test, tc_test_pattern, bool, 0644);
|
|
|
|
struct tc_edp_link {
|
|
u8 dpcd[DP_RECEIVER_CAP_SIZE];
|
|
unsigned int rate;
|
|
u8 num_lanes;
|
|
u8 assr;
|
|
bool scrambler_dis;
|
|
bool spread;
|
|
};
|
|
|
|
struct tc_data {
|
|
struct device *dev;
|
|
struct regmap *regmap;
|
|
struct drm_dp_aux aux;
|
|
|
|
struct drm_bridge bridge;
|
|
struct drm_bridge *panel_bridge;
|
|
struct drm_connector connector;
|
|
|
|
struct mipi_dsi_device *dsi;
|
|
|
|
/* link settings */
|
|
struct tc_edp_link link;
|
|
|
|
/* current mode */
|
|
struct drm_display_mode mode;
|
|
|
|
u32 rev;
|
|
u8 assr;
|
|
u8 pre_emphasis[2];
|
|
|
|
struct gpio_desc *sd_gpio;
|
|
struct gpio_desc *reset_gpio;
|
|
struct clk *refclk;
|
|
|
|
/* do we have IRQ */
|
|
bool have_irq;
|
|
|
|
/* Input connector type, DSI and not DPI. */
|
|
bool input_connector_dsi;
|
|
|
|
/* HPD pin number (0 or 1) or -ENODEV */
|
|
int hpd_pin;
|
|
};
|
|
|
|
static inline struct tc_data *aux_to_tc(struct drm_dp_aux *a)
|
|
{
|
|
return container_of(a, struct tc_data, aux);
|
|
}
|
|
|
|
static inline struct tc_data *bridge_to_tc(struct drm_bridge *b)
|
|
{
|
|
return container_of(b, struct tc_data, bridge);
|
|
}
|
|
|
|
static inline struct tc_data *connector_to_tc(struct drm_connector *c)
|
|
{
|
|
return container_of(c, struct tc_data, connector);
|
|
}
|
|
|
|
static inline int tc_poll_timeout(struct tc_data *tc, unsigned int addr,
|
|
unsigned int cond_mask,
|
|
unsigned int cond_value,
|
|
unsigned long sleep_us, u64 timeout_us)
|
|
{
|
|
unsigned int val;
|
|
|
|
return regmap_read_poll_timeout(tc->regmap, addr, val,
|
|
(val & cond_mask) == cond_value,
|
|
sleep_us, timeout_us);
|
|
}
|
|
|
|
static int tc_aux_wait_busy(struct tc_data *tc)
|
|
{
|
|
return tc_poll_timeout(tc, DP0_AUXSTATUS, AUX_BUSY, 0, 100, 100000);
|
|
}
|
|
|
|
static int tc_aux_write_data(struct tc_data *tc, const void *data,
|
|
size_t size)
|
|
{
|
|
u32 auxwdata[DP_AUX_MAX_PAYLOAD_BYTES / sizeof(u32)] = { 0 };
|
|
int ret, count = ALIGN(size, sizeof(u32));
|
|
|
|
memcpy(auxwdata, data, size);
|
|
|
|
ret = regmap_raw_write(tc->regmap, DP0_AUXWDATA(0), auxwdata, count);
|
|
if (ret)
|
|
return ret;
|
|
|
|
return size;
|
|
}
|
|
|
|
static int tc_aux_read_data(struct tc_data *tc, void *data, size_t size)
|
|
{
|
|
u32 auxrdata[DP_AUX_MAX_PAYLOAD_BYTES / sizeof(u32)];
|
|
int ret, count = ALIGN(size, sizeof(u32));
|
|
|
|
ret = regmap_raw_read(tc->regmap, DP0_AUXRDATA(0), auxrdata, count);
|
|
if (ret)
|
|
return ret;
|
|
|
|
memcpy(data, auxrdata, size);
|
|
|
|
return size;
|
|
}
|
|
|
|
static u32 tc_auxcfg0(struct drm_dp_aux_msg *msg, size_t size)
|
|
{
|
|
u32 auxcfg0 = msg->request;
|
|
|
|
if (size)
|
|
auxcfg0 |= FIELD_PREP(DP0_AUXCFG0_BSIZE, size - 1);
|
|
else
|
|
auxcfg0 |= DP0_AUXCFG0_ADDR_ONLY;
|
|
|
|
return auxcfg0;
|
|
}
|
|
|
|
static ssize_t tc_aux_transfer(struct drm_dp_aux *aux,
|
|
struct drm_dp_aux_msg *msg)
|
|
{
|
|
struct tc_data *tc = aux_to_tc(aux);
|
|
size_t size = min_t(size_t, DP_AUX_MAX_PAYLOAD_BYTES - 1, msg->size);
|
|
u8 request = msg->request & ~DP_AUX_I2C_MOT;
|
|
u32 auxstatus;
|
|
int ret;
|
|
|
|
ret = tc_aux_wait_busy(tc);
|
|
if (ret)
|
|
return ret;
|
|
|
|
switch (request) {
|
|
case DP_AUX_NATIVE_READ:
|
|
case DP_AUX_I2C_READ:
|
|
break;
|
|
case DP_AUX_NATIVE_WRITE:
|
|
case DP_AUX_I2C_WRITE:
|
|
if (size) {
|
|
ret = tc_aux_write_data(tc, msg->buffer, size);
|
|
if (ret < 0)
|
|
return ret;
|
|
}
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
|
|
/* Store address */
|
|
ret = regmap_write(tc->regmap, DP0_AUXADDR, msg->address);
|
|
if (ret)
|
|
return ret;
|
|
/* Start transfer */
|
|
ret = regmap_write(tc->regmap, DP0_AUXCFG0, tc_auxcfg0(msg, size));
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = tc_aux_wait_busy(tc);
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = regmap_read(tc->regmap, DP0_AUXSTATUS, &auxstatus);
|
|
if (ret)
|
|
return ret;
|
|
|
|
if (auxstatus & AUX_TIMEOUT)
|
|
return -ETIMEDOUT;
|
|
/*
|
|
* For some reason address-only DP_AUX_I2C_WRITE (MOT), still
|
|
* reports 1 byte transferred in its status. To deal we that
|
|
* we ignore aux_bytes field if we know that this was an
|
|
* address-only transfer
|
|
*/
|
|
if (size)
|
|
size = FIELD_GET(AUX_BYTES, auxstatus);
|
|
msg->reply = FIELD_GET(AUX_STATUS, auxstatus);
|
|
|
|
switch (request) {
|
|
case DP_AUX_NATIVE_READ:
|
|
case DP_AUX_I2C_READ:
|
|
if (size)
|
|
return tc_aux_read_data(tc, msg->buffer, size);
|
|
break;
|
|
}
|
|
|
|
return size;
|
|
}
|
|
|
|
static const char * const training_pattern1_errors[] = {
|
|
"No errors",
|
|
"Aux write error",
|
|
"Aux read error",
|
|
"Max voltage reached error",
|
|
"Loop counter expired error",
|
|
"res", "res", "res"
|
|
};
|
|
|
|
static const char * const training_pattern2_errors[] = {
|
|
"No errors",
|
|
"Aux write error",
|
|
"Aux read error",
|
|
"Clock recovery failed error",
|
|
"Loop counter expired error",
|
|
"res", "res", "res"
|
|
};
|
|
|
|
static u32 tc_srcctrl(struct tc_data *tc)
|
|
{
|
|
/*
|
|
* No training pattern, skew lane 1 data by two LSCLK cycles with
|
|
* respect to lane 0 data, AutoCorrect Mode = 0
|
|
*/
|
|
u32 reg = DP0_SRCCTRL_NOTP | DP0_SRCCTRL_LANESKEW | DP0_SRCCTRL_EN810B;
|
|
|
|
if (tc->link.scrambler_dis)
|
|
reg |= DP0_SRCCTRL_SCRMBLDIS; /* Scrambler Disabled */
|
|
if (tc->link.spread)
|
|
reg |= DP0_SRCCTRL_SSCG; /* Spread Spectrum Enable */
|
|
if (tc->link.num_lanes == 2)
|
|
reg |= DP0_SRCCTRL_LANES_2; /* Two Main Channel Lanes */
|
|
if (tc->link.rate != 162000)
|
|
reg |= DP0_SRCCTRL_BW27; /* 2.7 Gbps link */
|
|
return reg;
|
|
}
|
|
|
|
static int tc_pllupdate(struct tc_data *tc, unsigned int pllctrl)
|
|
{
|
|
int ret;
|
|
|
|
ret = regmap_write(tc->regmap, pllctrl, PLLUPDATE | PLLEN);
|
|
if (ret)
|
|
return ret;
|
|
|
|
/* Wait for PLL to lock: up to 7.5 ms, depending on refclk */
|
|
usleep_range(15000, 20000);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int tc_pxl_pll_calc(struct tc_data *tc, u32 refclk, u32 pixelclock,
|
|
int *out_best_pixelclock, u32 *out_pxl_pllparam)
|
|
{
|
|
int i_pre, best_pre = 1;
|
|
int i_post, best_post = 1;
|
|
int div, best_div = 1;
|
|
int mul, best_mul = 1;
|
|
int delta, best_delta;
|
|
int ext_div[] = {1, 2, 3, 5, 7};
|
|
int clk_min, clk_max;
|
|
int best_pixelclock = 0;
|
|
int vco_hi = 0;
|
|
u32 pxl_pllparam;
|
|
|
|
/*
|
|
* refclk * mul / (ext_pre_div * pre_div) should be in range:
|
|
* - DPI ..... 0 to 100 MHz
|
|
* - (e)DP ... 150 to 650 MHz
|
|
*/
|
|
if (tc->bridge.type == DRM_MODE_CONNECTOR_DPI) {
|
|
clk_min = 0;
|
|
clk_max = 100000000;
|
|
} else {
|
|
clk_min = 150000000;
|
|
clk_max = 650000000;
|
|
}
|
|
|
|
dev_dbg(tc->dev, "PLL: requested %d pixelclock, ref %d\n", pixelclock,
|
|
refclk);
|
|
best_delta = pixelclock;
|
|
/* Loop over all possible ext_divs, skipping invalid configurations */
|
|
for (i_pre = 0; i_pre < ARRAY_SIZE(ext_div); i_pre++) {
|
|
/*
|
|
* refclk / ext_pre_div should be in the 1 to 200 MHz range.
|
|
* We don't allow any refclk > 200 MHz, only check lower bounds.
|
|
*/
|
|
if (refclk / ext_div[i_pre] < 1000000)
|
|
continue;
|
|
for (i_post = 0; i_post < ARRAY_SIZE(ext_div); i_post++) {
|
|
for (div = 1; div <= 16; div++) {
|
|
u32 clk, iclk;
|
|
u64 tmp;
|
|
|
|
/* PCLK PLL input unit clock ... 6..40 MHz */
|
|
iclk = refclk / (div * ext_div[i_pre]);
|
|
if (iclk < 6000000 || iclk > 40000000)
|
|
continue;
|
|
|
|
tmp = pixelclock * ext_div[i_pre] *
|
|
ext_div[i_post] * div;
|
|
do_div(tmp, refclk);
|
|
mul = tmp;
|
|
|
|
/* Check limits */
|
|
if ((mul < 1) || (mul > 128))
|
|
continue;
|
|
|
|
clk = (refclk / ext_div[i_pre] / div) * mul;
|
|
if ((clk > clk_max) || (clk < clk_min))
|
|
continue;
|
|
|
|
clk = clk / ext_div[i_post];
|
|
delta = clk - pixelclock;
|
|
|
|
if (abs(delta) < abs(best_delta)) {
|
|
best_pre = i_pre;
|
|
best_post = i_post;
|
|
best_div = div;
|
|
best_mul = mul;
|
|
best_delta = delta;
|
|
best_pixelclock = clk;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (best_pixelclock == 0) {
|
|
dev_err(tc->dev, "Failed to calc clock for %d pixelclock\n",
|
|
pixelclock);
|
|
return -EINVAL;
|
|
}
|
|
|
|
dev_dbg(tc->dev, "PLL: got %d, delta %d\n", best_pixelclock, best_delta);
|
|
dev_dbg(tc->dev, "PLL: %d / %d / %d * %d / %d\n", refclk,
|
|
ext_div[best_pre], best_div, best_mul, ext_div[best_post]);
|
|
|
|
/* if VCO >= 300 MHz */
|
|
if (refclk / ext_div[best_pre] / best_div * best_mul >= 300000000)
|
|
vco_hi = 1;
|
|
/* see DS */
|
|
if (best_div == 16)
|
|
best_div = 0;
|
|
if (best_mul == 128)
|
|
best_mul = 0;
|
|
|
|
pxl_pllparam = vco_hi << 24; /* For PLL VCO >= 300 MHz = 1 */
|
|
pxl_pllparam |= ext_div[best_pre] << 20; /* External Pre-divider */
|
|
pxl_pllparam |= ext_div[best_post] << 16; /* External Post-divider */
|
|
pxl_pllparam |= IN_SEL_REFCLK; /* Use RefClk as PLL input */
|
|
pxl_pllparam |= best_div << 8; /* Divider for PLL RefClk */
|
|
pxl_pllparam |= best_mul; /* Multiplier for PLL */
|
|
|
|
if (out_best_pixelclock)
|
|
*out_best_pixelclock = best_pixelclock;
|
|
|
|
if (out_pxl_pllparam)
|
|
*out_pxl_pllparam = pxl_pllparam;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int tc_pxl_pll_en(struct tc_data *tc, u32 refclk, u32 pixelclock)
|
|
{
|
|
u32 pxl_pllparam = 0;
|
|
int ret;
|
|
|
|
ret = tc_pxl_pll_calc(tc, refclk, pixelclock, NULL, &pxl_pllparam);
|
|
if (ret)
|
|
return ret;
|
|
|
|
/* Power up PLL and switch to bypass */
|
|
ret = regmap_write(tc->regmap, PXL_PLLCTRL, PLLBYP | PLLEN);
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = regmap_write(tc->regmap, PXL_PLLPARAM, pxl_pllparam);
|
|
if (ret)
|
|
return ret;
|
|
|
|
/* Force PLL parameter update and disable bypass */
|
|
return tc_pllupdate(tc, PXL_PLLCTRL);
|
|
}
|
|
|
|
static int tc_pxl_pll_dis(struct tc_data *tc)
|
|
{
|
|
/* Enable PLL bypass, power down PLL */
|
|
return regmap_write(tc->regmap, PXL_PLLCTRL, PLLBYP);
|
|
}
|
|
|
|
static int tc_stream_clock_calc(struct tc_data *tc)
|
|
{
|
|
/*
|
|
* If the Stream clock and Link Symbol clock are
|
|
* asynchronous with each other, the value of M changes over
|
|
* time. This way of generating link clock and stream
|
|
* clock is called Asynchronous Clock mode. The value M
|
|
* must change while the value N stays constant. The
|
|
* value of N in this Asynchronous Clock mode must be set
|
|
* to 2^15 or 32,768.
|
|
*
|
|
* LSCLK = 1/10 of high speed link clock
|
|
*
|
|
* f_STRMCLK = M/N * f_LSCLK
|
|
* M/N = f_STRMCLK / f_LSCLK
|
|
*
|
|
*/
|
|
return regmap_write(tc->regmap, DP0_VIDMNGEN1, 32768);
|
|
}
|
|
|
|
static int tc_set_syspllparam(struct tc_data *tc)
|
|
{
|
|
unsigned long rate;
|
|
u32 pllparam = SYSCLK_SEL_LSCLK | LSCLK_DIV_1;
|
|
|
|
rate = clk_get_rate(tc->refclk);
|
|
switch (rate) {
|
|
case 38400000:
|
|
pllparam |= REF_FREQ_38M4;
|
|
break;
|
|
case 26000000:
|
|
pllparam |= REF_FREQ_26M;
|
|
break;
|
|
case 19200000:
|
|
pllparam |= REF_FREQ_19M2;
|
|
break;
|
|
case 13000000:
|
|
pllparam |= REF_FREQ_13M;
|
|
break;
|
|
default:
|
|
dev_err(tc->dev, "Invalid refclk rate: %lu Hz\n", rate);
|
|
return -EINVAL;
|
|
}
|
|
|
|
return regmap_write(tc->regmap, SYS_PLLPARAM, pllparam);
|
|
}
|
|
|
|
static int tc_aux_link_setup(struct tc_data *tc)
|
|
{
|
|
int ret;
|
|
u32 dp0_auxcfg1;
|
|
|
|
/* Setup DP-PHY / PLL */
|
|
ret = tc_set_syspllparam(tc);
|
|
if (ret)
|
|
goto err;
|
|
|
|
ret = regmap_write(tc->regmap, DP_PHY_CTRL,
|
|
BGREN | PWR_SW_EN | PHY_A0_EN);
|
|
if (ret)
|
|
goto err;
|
|
/*
|
|
* Initially PLLs are in bypass. Force PLL parameter update,
|
|
* disable PLL bypass, enable PLL
|
|
*/
|
|
ret = tc_pllupdate(tc, DP0_PLLCTRL);
|
|
if (ret)
|
|
goto err;
|
|
|
|
ret = tc_pllupdate(tc, DP1_PLLCTRL);
|
|
if (ret)
|
|
goto err;
|
|
|
|
ret = tc_poll_timeout(tc, DP_PHY_CTRL, PHY_RDY, PHY_RDY, 100, 100000);
|
|
if (ret == -ETIMEDOUT) {
|
|
dev_err(tc->dev, "Timeout waiting for PHY to become ready");
|
|
return ret;
|
|
} else if (ret) {
|
|
goto err;
|
|
}
|
|
|
|
/* Setup AUX link */
|
|
dp0_auxcfg1 = AUX_RX_FILTER_EN;
|
|
dp0_auxcfg1 |= 0x06 << 8; /* Aux Bit Period Calculator Threshold */
|
|
dp0_auxcfg1 |= 0x3f << 0; /* Aux Response Timeout Timer */
|
|
|
|
ret = regmap_write(tc->regmap, DP0_AUXCFG1, dp0_auxcfg1);
|
|
if (ret)
|
|
goto err;
|
|
|
|
/* Register DP AUX channel */
|
|
tc->aux.name = "TC358767 AUX i2c adapter";
|
|
tc->aux.dev = tc->dev;
|
|
tc->aux.transfer = tc_aux_transfer;
|
|
drm_dp_aux_init(&tc->aux);
|
|
|
|
return 0;
|
|
err:
|
|
dev_err(tc->dev, "tc_aux_link_setup failed: %d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
static int tc_get_display_props(struct tc_data *tc)
|
|
{
|
|
u8 revision, num_lanes;
|
|
unsigned int rate;
|
|
int ret;
|
|
u8 reg;
|
|
|
|
/* Read DP Rx Link Capability */
|
|
ret = drm_dp_dpcd_read(&tc->aux, DP_DPCD_REV, tc->link.dpcd,
|
|
DP_RECEIVER_CAP_SIZE);
|
|
if (ret < 0)
|
|
goto err_dpcd_read;
|
|
|
|
revision = tc->link.dpcd[DP_DPCD_REV];
|
|
rate = drm_dp_max_link_rate(tc->link.dpcd);
|
|
num_lanes = drm_dp_max_lane_count(tc->link.dpcd);
|
|
|
|
if (rate != 162000 && rate != 270000) {
|
|
dev_dbg(tc->dev, "Falling to 2.7 Gbps rate\n");
|
|
rate = 270000;
|
|
}
|
|
|
|
tc->link.rate = rate;
|
|
|
|
if (num_lanes > 2) {
|
|
dev_dbg(tc->dev, "Falling to 2 lanes\n");
|
|
num_lanes = 2;
|
|
}
|
|
|
|
tc->link.num_lanes = num_lanes;
|
|
|
|
ret = drm_dp_dpcd_readb(&tc->aux, DP_MAX_DOWNSPREAD, ®);
|
|
if (ret < 0)
|
|
goto err_dpcd_read;
|
|
tc->link.spread = reg & DP_MAX_DOWNSPREAD_0_5;
|
|
|
|
ret = drm_dp_dpcd_readb(&tc->aux, DP_MAIN_LINK_CHANNEL_CODING, ®);
|
|
if (ret < 0)
|
|
goto err_dpcd_read;
|
|
|
|
tc->link.scrambler_dis = false;
|
|
/* read assr */
|
|
ret = drm_dp_dpcd_readb(&tc->aux, DP_EDP_CONFIGURATION_SET, ®);
|
|
if (ret < 0)
|
|
goto err_dpcd_read;
|
|
tc->link.assr = reg & DP_ALTERNATE_SCRAMBLER_RESET_ENABLE;
|
|
|
|
dev_dbg(tc->dev, "DPCD rev: %d.%d, rate: %s, lanes: %d, framing: %s\n",
|
|
revision >> 4, revision & 0x0f,
|
|
(tc->link.rate == 162000) ? "1.62Gbps" : "2.7Gbps",
|
|
tc->link.num_lanes,
|
|
drm_dp_enhanced_frame_cap(tc->link.dpcd) ?
|
|
"enhanced" : "default");
|
|
dev_dbg(tc->dev, "Downspread: %s, scrambler: %s\n",
|
|
tc->link.spread ? "0.5%" : "0.0%",
|
|
tc->link.scrambler_dis ? "disabled" : "enabled");
|
|
dev_dbg(tc->dev, "Display ASSR: %d, TC358767 ASSR: %d\n",
|
|
tc->link.assr, tc->assr);
|
|
|
|
return 0;
|
|
|
|
err_dpcd_read:
|
|
dev_err(tc->dev, "failed to read DPCD: %d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
static int tc_set_common_video_mode(struct tc_data *tc,
|
|
const struct drm_display_mode *mode)
|
|
{
|
|
int left_margin = mode->htotal - mode->hsync_end;
|
|
int right_margin = mode->hsync_start - mode->hdisplay;
|
|
int hsync_len = mode->hsync_end - mode->hsync_start;
|
|
int upper_margin = mode->vtotal - mode->vsync_end;
|
|
int lower_margin = mode->vsync_start - mode->vdisplay;
|
|
int vsync_len = mode->vsync_end - mode->vsync_start;
|
|
int ret;
|
|
|
|
dev_dbg(tc->dev, "set mode %dx%d\n",
|
|
mode->hdisplay, mode->vdisplay);
|
|
dev_dbg(tc->dev, "H margin %d,%d sync %d\n",
|
|
left_margin, right_margin, hsync_len);
|
|
dev_dbg(tc->dev, "V margin %d,%d sync %d\n",
|
|
upper_margin, lower_margin, vsync_len);
|
|
dev_dbg(tc->dev, "total: %dx%d\n", mode->htotal, mode->vtotal);
|
|
|
|
/*
|
|
* LCD Ctl Frame Size
|
|
* datasheet is not clear of vsdelay in case of DPI
|
|
* assume we do not need any delay when DPI is a source of
|
|
* sync signals
|
|
*/
|
|
ret = regmap_write(tc->regmap, VPCTRL0,
|
|
FIELD_PREP(VSDELAY, right_margin + 10) |
|
|
OPXLFMT_RGB888 | FRMSYNC_ENABLED | MSF_DISABLED);
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = regmap_write(tc->regmap, HTIM01,
|
|
FIELD_PREP(HBPR, ALIGN(left_margin, 2)) |
|
|
FIELD_PREP(HPW, ALIGN(hsync_len, 2)));
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = regmap_write(tc->regmap, HTIM02,
|
|
FIELD_PREP(HDISPR, ALIGN(mode->hdisplay, 2)) |
|
|
FIELD_PREP(HFPR, ALIGN(right_margin, 2)));
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = regmap_write(tc->regmap, VTIM01,
|
|
FIELD_PREP(VBPR, upper_margin) |
|
|
FIELD_PREP(VSPR, vsync_len));
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = regmap_write(tc->regmap, VTIM02,
|
|
FIELD_PREP(VFPR, lower_margin) |
|
|
FIELD_PREP(VDISPR, mode->vdisplay));
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = regmap_write(tc->regmap, VFUEN0, VFUEN); /* update settings */
|
|
if (ret)
|
|
return ret;
|
|
|
|
/* Test pattern settings */
|
|
ret = regmap_write(tc->regmap, TSTCTL,
|
|
FIELD_PREP(COLOR_R, 120) |
|
|
FIELD_PREP(COLOR_G, 20) |
|
|
FIELD_PREP(COLOR_B, 99) |
|
|
ENI2CFILTER |
|
|
FIELD_PREP(COLOR_BAR_MODE, COLOR_BAR_MODE_BARS));
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int tc_set_dpi_video_mode(struct tc_data *tc,
|
|
const struct drm_display_mode *mode)
|
|
{
|
|
u32 value = POCTRL_S2P;
|
|
|
|
if (tc->mode.flags & DRM_MODE_FLAG_NHSYNC)
|
|
value |= POCTRL_HS_POL;
|
|
|
|
if (tc->mode.flags & DRM_MODE_FLAG_NVSYNC)
|
|
value |= POCTRL_VS_POL;
|
|
|
|
return regmap_write(tc->regmap, POCTRL, value);
|
|
}
|
|
|
|
static int tc_set_edp_video_mode(struct tc_data *tc,
|
|
const struct drm_display_mode *mode)
|
|
{
|
|
int ret;
|
|
int vid_sync_dly;
|
|
int max_tu_symbol;
|
|
|
|
int left_margin = mode->htotal - mode->hsync_end;
|
|
int hsync_len = mode->hsync_end - mode->hsync_start;
|
|
int upper_margin = mode->vtotal - mode->vsync_end;
|
|
int vsync_len = mode->vsync_end - mode->vsync_start;
|
|
u32 dp0_syncval;
|
|
u32 bits_per_pixel = 24;
|
|
u32 in_bw, out_bw;
|
|
u32 dpipxlfmt;
|
|
|
|
/*
|
|
* Recommended maximum number of symbols transferred in a transfer unit:
|
|
* DIV_ROUND_UP((input active video bandwidth in bytes) * tu_size,
|
|
* (output active video bandwidth in bytes))
|
|
* Must be less than tu_size.
|
|
*/
|
|
|
|
in_bw = mode->clock * bits_per_pixel / 8;
|
|
out_bw = tc->link.num_lanes * tc->link.rate;
|
|
max_tu_symbol = DIV_ROUND_UP(in_bw * TU_SIZE_RECOMMENDED, out_bw);
|
|
|
|
/* DP Main Stream Attributes */
|
|
vid_sync_dly = hsync_len + left_margin + mode->hdisplay;
|
|
ret = regmap_write(tc->regmap, DP0_VIDSYNCDELAY,
|
|
FIELD_PREP(THRESH_DLY, max_tu_symbol) |
|
|
FIELD_PREP(VID_SYNC_DLY, vid_sync_dly));
|
|
|
|
ret = regmap_write(tc->regmap, DP0_TOTALVAL,
|
|
FIELD_PREP(H_TOTAL, mode->htotal) |
|
|
FIELD_PREP(V_TOTAL, mode->vtotal));
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = regmap_write(tc->regmap, DP0_STARTVAL,
|
|
FIELD_PREP(H_START, left_margin + hsync_len) |
|
|
FIELD_PREP(V_START, upper_margin + vsync_len));
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = regmap_write(tc->regmap, DP0_ACTIVEVAL,
|
|
FIELD_PREP(V_ACT, mode->vdisplay) |
|
|
FIELD_PREP(H_ACT, mode->hdisplay));
|
|
if (ret)
|
|
return ret;
|
|
|
|
dp0_syncval = FIELD_PREP(VS_WIDTH, vsync_len) |
|
|
FIELD_PREP(HS_WIDTH, hsync_len);
|
|
|
|
if (mode->flags & DRM_MODE_FLAG_NVSYNC)
|
|
dp0_syncval |= SYNCVAL_VS_POL_ACTIVE_LOW;
|
|
|
|
if (mode->flags & DRM_MODE_FLAG_NHSYNC)
|
|
dp0_syncval |= SYNCVAL_HS_POL_ACTIVE_LOW;
|
|
|
|
ret = regmap_write(tc->regmap, DP0_SYNCVAL, dp0_syncval);
|
|
if (ret)
|
|
return ret;
|
|
|
|
dpipxlfmt = DE_POL_ACTIVE_HIGH | SUB_CFG_TYPE_CONFIG1 | DPI_BPP_RGB888;
|
|
|
|
if (mode->flags & DRM_MODE_FLAG_NVSYNC)
|
|
dpipxlfmt |= VS_POL_ACTIVE_LOW;
|
|
|
|
if (mode->flags & DRM_MODE_FLAG_NHSYNC)
|
|
dpipxlfmt |= HS_POL_ACTIVE_LOW;
|
|
|
|
ret = regmap_write(tc->regmap, DPIPXLFMT, dpipxlfmt);
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = regmap_write(tc->regmap, DP0_MISC,
|
|
FIELD_PREP(MAX_TU_SYMBOL, max_tu_symbol) |
|
|
FIELD_PREP(TU_SIZE, TU_SIZE_RECOMMENDED) |
|
|
BPC_8);
|
|
return ret;
|
|
}
|
|
|
|
static int tc_wait_link_training(struct tc_data *tc)
|
|
{
|
|
u32 value;
|
|
int ret;
|
|
|
|
ret = tc_poll_timeout(tc, DP0_LTSTAT, LT_LOOPDONE,
|
|
LT_LOOPDONE, 500, 100000);
|
|
if (ret) {
|
|
dev_err(tc->dev, "Link training timeout waiting for LT_LOOPDONE!\n");
|
|
return ret;
|
|
}
|
|
|
|
ret = regmap_read(tc->regmap, DP0_LTSTAT, &value);
|
|
if (ret)
|
|
return ret;
|
|
|
|
return (value >> 8) & 0x7;
|
|
}
|
|
|
|
static int tc_main_link_enable(struct tc_data *tc)
|
|
{
|
|
struct drm_dp_aux *aux = &tc->aux;
|
|
struct device *dev = tc->dev;
|
|
u32 dp_phy_ctrl;
|
|
u32 value;
|
|
int ret;
|
|
u8 tmp[DP_LINK_STATUS_SIZE];
|
|
|
|
dev_dbg(tc->dev, "link enable\n");
|
|
|
|
ret = regmap_read(tc->regmap, DP0CTL, &value);
|
|
if (ret)
|
|
return ret;
|
|
|
|
if (WARN_ON(value & DP_EN)) {
|
|
ret = regmap_write(tc->regmap, DP0CTL, 0);
|
|
if (ret)
|
|
return ret;
|
|
}
|
|
|
|
ret = regmap_write(tc->regmap, DP0_SRCCTRL,
|
|
tc_srcctrl(tc) |
|
|
FIELD_PREP(DP0_SRCCTRL_PRE0, tc->pre_emphasis[0]) |
|
|
FIELD_PREP(DP0_SRCCTRL_PRE1, tc->pre_emphasis[1]));
|
|
if (ret)
|
|
return ret;
|
|
/* SSCG and BW27 on DP1 must be set to the same as on DP0 */
|
|
ret = regmap_write(tc->regmap, DP1_SRCCTRL,
|
|
(tc->link.spread ? DP0_SRCCTRL_SSCG : 0) |
|
|
((tc->link.rate != 162000) ? DP0_SRCCTRL_BW27 : 0) |
|
|
FIELD_PREP(DP1_SRCCTRL_PRE, tc->pre_emphasis[1]));
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = tc_set_syspllparam(tc);
|
|
if (ret)
|
|
return ret;
|
|
|
|
/* Setup Main Link */
|
|
dp_phy_ctrl = BGREN | PWR_SW_EN | PHY_A0_EN | PHY_M0_EN;
|
|
if (tc->link.num_lanes == 2)
|
|
dp_phy_ctrl |= PHY_2LANE;
|
|
|
|
ret = regmap_write(tc->regmap, DP_PHY_CTRL, dp_phy_ctrl);
|
|
if (ret)
|
|
return ret;
|
|
|
|
/* PLL setup */
|
|
ret = tc_pllupdate(tc, DP0_PLLCTRL);
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = tc_pllupdate(tc, DP1_PLLCTRL);
|
|
if (ret)
|
|
return ret;
|
|
|
|
/* Reset/Enable Main Links */
|
|
dp_phy_ctrl |= DP_PHY_RST | PHY_M1_RST | PHY_M0_RST;
|
|
ret = regmap_write(tc->regmap, DP_PHY_CTRL, dp_phy_ctrl);
|
|
usleep_range(100, 200);
|
|
dp_phy_ctrl &= ~(DP_PHY_RST | PHY_M1_RST | PHY_M0_RST);
|
|
ret = regmap_write(tc->regmap, DP_PHY_CTRL, dp_phy_ctrl);
|
|
|
|
ret = tc_poll_timeout(tc, DP_PHY_CTRL, PHY_RDY, PHY_RDY, 500, 100000);
|
|
if (ret) {
|
|
dev_err(dev, "timeout waiting for phy become ready");
|
|
return ret;
|
|
}
|
|
|
|
/* Set misc: 8 bits per color */
|
|
ret = regmap_update_bits(tc->regmap, DP0_MISC, BPC_8, BPC_8);
|
|
if (ret)
|
|
return ret;
|
|
|
|
/*
|
|
* ASSR mode
|
|
* on TC358767 side ASSR configured through strap pin
|
|
* seems there is no way to change this setting from SW
|
|
*
|
|
* check is tc configured for same mode
|
|
*/
|
|
if (tc->assr != tc->link.assr) {
|
|
dev_dbg(dev, "Trying to set display to ASSR: %d\n",
|
|
tc->assr);
|
|
/* try to set ASSR on display side */
|
|
tmp[0] = tc->assr;
|
|
ret = drm_dp_dpcd_writeb(aux, DP_EDP_CONFIGURATION_SET, tmp[0]);
|
|
if (ret < 0)
|
|
goto err_dpcd_read;
|
|
/* read back */
|
|
ret = drm_dp_dpcd_readb(aux, DP_EDP_CONFIGURATION_SET, tmp);
|
|
if (ret < 0)
|
|
goto err_dpcd_read;
|
|
|
|
if (tmp[0] != tc->assr) {
|
|
dev_dbg(dev, "Failed to switch display ASSR to %d, falling back to unscrambled mode\n",
|
|
tc->assr);
|
|
/* trying with disabled scrambler */
|
|
tc->link.scrambler_dis = true;
|
|
}
|
|
}
|
|
|
|
/* Setup Link & DPRx Config for Training */
|
|
tmp[0] = drm_dp_link_rate_to_bw_code(tc->link.rate);
|
|
tmp[1] = tc->link.num_lanes;
|
|
|
|
if (drm_dp_enhanced_frame_cap(tc->link.dpcd))
|
|
tmp[1] |= DP_LANE_COUNT_ENHANCED_FRAME_EN;
|
|
|
|
ret = drm_dp_dpcd_write(aux, DP_LINK_BW_SET, tmp, 2);
|
|
if (ret < 0)
|
|
goto err_dpcd_write;
|
|
|
|
/* DOWNSPREAD_CTRL */
|
|
tmp[0] = tc->link.spread ? DP_SPREAD_AMP_0_5 : 0x00;
|
|
/* MAIN_LINK_CHANNEL_CODING_SET */
|
|
tmp[1] = DP_SET_ANSI_8B10B;
|
|
ret = drm_dp_dpcd_write(aux, DP_DOWNSPREAD_CTRL, tmp, 2);
|
|
if (ret < 0)
|
|
goto err_dpcd_write;
|
|
|
|
/* Reset voltage-swing & pre-emphasis */
|
|
tmp[0] = DP_TRAIN_VOLTAGE_SWING_LEVEL_0 |
|
|
FIELD_PREP(DP_TRAIN_PRE_EMPHASIS_MASK, tc->pre_emphasis[0]);
|
|
tmp[1] = DP_TRAIN_VOLTAGE_SWING_LEVEL_0 |
|
|
FIELD_PREP(DP_TRAIN_PRE_EMPHASIS_MASK, tc->pre_emphasis[1]);
|
|
ret = drm_dp_dpcd_write(aux, DP_TRAINING_LANE0_SET, tmp, 2);
|
|
if (ret < 0)
|
|
goto err_dpcd_write;
|
|
|
|
/* Clock-Recovery */
|
|
|
|
/* Set DPCD 0x102 for Training Pattern 1 */
|
|
ret = regmap_write(tc->regmap, DP0_SNKLTCTRL,
|
|
DP_LINK_SCRAMBLING_DISABLE |
|
|
DP_TRAINING_PATTERN_1);
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = regmap_write(tc->regmap, DP0_LTLOOPCTRL,
|
|
(15 << 28) | /* Defer Iteration Count */
|
|
(15 << 24) | /* Loop Iteration Count */
|
|
(0xd << 0)); /* Loop Timer Delay */
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = regmap_write(tc->regmap, DP0_SRCCTRL,
|
|
tc_srcctrl(tc) | DP0_SRCCTRL_SCRMBLDIS |
|
|
DP0_SRCCTRL_AUTOCORRECT |
|
|
DP0_SRCCTRL_TP1 |
|
|
FIELD_PREP(DP0_SRCCTRL_PRE0, tc->pre_emphasis[0]) |
|
|
FIELD_PREP(DP0_SRCCTRL_PRE1, tc->pre_emphasis[1]));
|
|
if (ret)
|
|
return ret;
|
|
|
|
/* Enable DP0 to start Link Training */
|
|
ret = regmap_write(tc->regmap, DP0CTL,
|
|
(drm_dp_enhanced_frame_cap(tc->link.dpcd) ?
|
|
EF_EN : 0) | DP_EN);
|
|
if (ret)
|
|
return ret;
|
|
|
|
/* wait */
|
|
|
|
ret = tc_wait_link_training(tc);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
if (ret) {
|
|
dev_err(tc->dev, "Link training phase 1 failed: %s\n",
|
|
training_pattern1_errors[ret]);
|
|
return -ENODEV;
|
|
}
|
|
|
|
/* Channel Equalization */
|
|
|
|
/* Set DPCD 0x102 for Training Pattern 2 */
|
|
ret = regmap_write(tc->regmap, DP0_SNKLTCTRL,
|
|
DP_LINK_SCRAMBLING_DISABLE |
|
|
DP_TRAINING_PATTERN_2);
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = regmap_write(tc->regmap, DP0_SRCCTRL,
|
|
tc_srcctrl(tc) | DP0_SRCCTRL_SCRMBLDIS |
|
|
DP0_SRCCTRL_AUTOCORRECT |
|
|
DP0_SRCCTRL_TP2 |
|
|
FIELD_PREP(DP0_SRCCTRL_PRE0, tc->pre_emphasis[0]) |
|
|
FIELD_PREP(DP0_SRCCTRL_PRE1, tc->pre_emphasis[1]));
|
|
if (ret)
|
|
return ret;
|
|
|
|
/* wait */
|
|
ret = tc_wait_link_training(tc);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
if (ret) {
|
|
dev_err(tc->dev, "Link training phase 2 failed: %s\n",
|
|
training_pattern2_errors[ret]);
|
|
return -ENODEV;
|
|
}
|
|
|
|
/*
|
|
* Toshiba's documentation suggests to first clear DPCD 0x102, then
|
|
* clear the training pattern bit in DP0_SRCCTRL. Testing shows
|
|
* that the link sometimes drops if those steps are done in that order,
|
|
* but if the steps are done in reverse order, the link stays up.
|
|
*
|
|
* So we do the steps differently than documented here.
|
|
*/
|
|
|
|
/* Clear Training Pattern, set AutoCorrect Mode = 1 */
|
|
ret = regmap_write(tc->regmap, DP0_SRCCTRL, tc_srcctrl(tc) |
|
|
DP0_SRCCTRL_AUTOCORRECT |
|
|
FIELD_PREP(DP0_SRCCTRL_PRE0, tc->pre_emphasis[0]) |
|
|
FIELD_PREP(DP0_SRCCTRL_PRE1, tc->pre_emphasis[1]));
|
|
if (ret)
|
|
return ret;
|
|
|
|
/* Clear DPCD 0x102 */
|
|
/* Note: Can Not use DP0_SNKLTCTRL (0x06E4) short cut */
|
|
tmp[0] = tc->link.scrambler_dis ? DP_LINK_SCRAMBLING_DISABLE : 0x00;
|
|
ret = drm_dp_dpcd_writeb(aux, DP_TRAINING_PATTERN_SET, tmp[0]);
|
|
if (ret < 0)
|
|
goto err_dpcd_write;
|
|
|
|
/* Check link status */
|
|
ret = drm_dp_dpcd_read_link_status(aux, tmp);
|
|
if (ret < 0)
|
|
goto err_dpcd_read;
|
|
|
|
ret = 0;
|
|
|
|
value = tmp[0] & DP_CHANNEL_EQ_BITS;
|
|
|
|
if (value != DP_CHANNEL_EQ_BITS) {
|
|
dev_err(tc->dev, "Lane 0 failed: %x\n", value);
|
|
ret = -ENODEV;
|
|
}
|
|
|
|
if (tc->link.num_lanes == 2) {
|
|
value = (tmp[0] >> 4) & DP_CHANNEL_EQ_BITS;
|
|
|
|
if (value != DP_CHANNEL_EQ_BITS) {
|
|
dev_err(tc->dev, "Lane 1 failed: %x\n", value);
|
|
ret = -ENODEV;
|
|
}
|
|
|
|
if (!(tmp[2] & DP_INTERLANE_ALIGN_DONE)) {
|
|
dev_err(tc->dev, "Interlane align failed\n");
|
|
ret = -ENODEV;
|
|
}
|
|
}
|
|
|
|
if (ret) {
|
|
dev_err(dev, "0x0202 LANE0_1_STATUS: 0x%02x\n", tmp[0]);
|
|
dev_err(dev, "0x0203 LANE2_3_STATUS 0x%02x\n", tmp[1]);
|
|
dev_err(dev, "0x0204 LANE_ALIGN_STATUS_UPDATED: 0x%02x\n", tmp[2]);
|
|
dev_err(dev, "0x0205 SINK_STATUS: 0x%02x\n", tmp[3]);
|
|
dev_err(dev, "0x0206 ADJUST_REQUEST_LANE0_1: 0x%02x\n", tmp[4]);
|
|
dev_err(dev, "0x0207 ADJUST_REQUEST_LANE2_3: 0x%02x\n", tmp[5]);
|
|
return ret;
|
|
}
|
|
|
|
return 0;
|
|
err_dpcd_read:
|
|
dev_err(tc->dev, "Failed to read DPCD: %d\n", ret);
|
|
return ret;
|
|
err_dpcd_write:
|
|
dev_err(tc->dev, "Failed to write DPCD: %d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
static int tc_main_link_disable(struct tc_data *tc)
|
|
{
|
|
int ret;
|
|
|
|
dev_dbg(tc->dev, "link disable\n");
|
|
|
|
ret = regmap_write(tc->regmap, DP0_SRCCTRL, 0);
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = regmap_write(tc->regmap, DP0CTL, 0);
|
|
if (ret)
|
|
return ret;
|
|
|
|
return regmap_update_bits(tc->regmap, DP_PHY_CTRL,
|
|
PHY_M0_RST | PHY_M1_RST | PHY_M0_EN,
|
|
PHY_M0_RST | PHY_M1_RST);
|
|
}
|
|
|
|
static int tc_dsi_rx_enable(struct tc_data *tc)
|
|
{
|
|
u32 value;
|
|
int ret;
|
|
|
|
regmap_write(tc->regmap, PPI_D0S_CLRSIPOCOUNT, 5);
|
|
regmap_write(tc->regmap, PPI_D1S_CLRSIPOCOUNT, 5);
|
|
regmap_write(tc->regmap, PPI_D2S_CLRSIPOCOUNT, 5);
|
|
regmap_write(tc->regmap, PPI_D3S_CLRSIPOCOUNT, 5);
|
|
regmap_write(tc->regmap, PPI_D0S_ATMR, 0);
|
|
regmap_write(tc->regmap, PPI_D1S_ATMR, 0);
|
|
regmap_write(tc->regmap, PPI_TX_RX_TA, TTA_GET | TTA_SURE);
|
|
regmap_write(tc->regmap, PPI_LPTXTIMECNT, LPX_PERIOD);
|
|
|
|
value = ((LANEENABLE_L0EN << tc->dsi->lanes) - LANEENABLE_L0EN) |
|
|
LANEENABLE_CLEN;
|
|
regmap_write(tc->regmap, PPI_LANEENABLE, value);
|
|
regmap_write(tc->regmap, DSI_LANEENABLE, value);
|
|
|
|
/* Set input interface */
|
|
value = DP0_AUDSRC_NO_INPUT;
|
|
if (tc_test_pattern)
|
|
value |= DP0_VIDSRC_COLOR_BAR;
|
|
else
|
|
value |= DP0_VIDSRC_DSI_RX;
|
|
ret = regmap_write(tc->regmap, SYSCTRL, value);
|
|
if (ret)
|
|
return ret;
|
|
|
|
usleep_range(120, 150);
|
|
|
|
regmap_write(tc->regmap, PPI_STARTPPI, PPI_START_FUNCTION);
|
|
regmap_write(tc->regmap, DSI_STARTDSI, DSI_RX_START);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int tc_dpi_rx_enable(struct tc_data *tc)
|
|
{
|
|
u32 value;
|
|
|
|
/* Set input interface */
|
|
value = DP0_AUDSRC_NO_INPUT;
|
|
if (tc_test_pattern)
|
|
value |= DP0_VIDSRC_COLOR_BAR;
|
|
else
|
|
value |= DP0_VIDSRC_DPI_RX;
|
|
return regmap_write(tc->regmap, SYSCTRL, value);
|
|
}
|
|
|
|
static int tc_dpi_stream_enable(struct tc_data *tc)
|
|
{
|
|
int ret;
|
|
|
|
dev_dbg(tc->dev, "enable video stream\n");
|
|
|
|
/* Setup PLL */
|
|
ret = tc_set_syspllparam(tc);
|
|
if (ret)
|
|
return ret;
|
|
|
|
/*
|
|
* Initially PLLs are in bypass. Force PLL parameter update,
|
|
* disable PLL bypass, enable PLL
|
|
*/
|
|
ret = tc_pllupdate(tc, DP0_PLLCTRL);
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = tc_pllupdate(tc, DP1_PLLCTRL);
|
|
if (ret)
|
|
return ret;
|
|
|
|
/* Pixel PLL must always be enabled for DPI mode */
|
|
ret = tc_pxl_pll_en(tc, clk_get_rate(tc->refclk),
|
|
1000 * tc->mode.clock);
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = tc_set_common_video_mode(tc, &tc->mode);
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = tc_set_dpi_video_mode(tc, &tc->mode);
|
|
if (ret)
|
|
return ret;
|
|
|
|
return tc_dsi_rx_enable(tc);
|
|
}
|
|
|
|
static int tc_dpi_stream_disable(struct tc_data *tc)
|
|
{
|
|
dev_dbg(tc->dev, "disable video stream\n");
|
|
|
|
tc_pxl_pll_dis(tc);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int tc_edp_stream_enable(struct tc_data *tc)
|
|
{
|
|
int ret;
|
|
u32 value;
|
|
|
|
dev_dbg(tc->dev, "enable video stream\n");
|
|
|
|
/*
|
|
* Pixel PLL must be enabled for DSI input mode and test pattern.
|
|
*
|
|
* Per TC9595XBG datasheet Revision 0.1 2018-12-27 Figure 4.18
|
|
* "Clock Mode Selection and Clock Sources", either Pixel PLL
|
|
* or DPI_PCLK supplies StrmClk. DPI_PCLK is only available in
|
|
* case valid Pixel Clock are supplied to the chip DPI input.
|
|
* In case built-in test pattern is desired OR DSI input mode
|
|
* is used, DPI_PCLK is not available and thus Pixel PLL must
|
|
* be used instead.
|
|
*/
|
|
if (tc->input_connector_dsi || tc_test_pattern) {
|
|
ret = tc_pxl_pll_en(tc, clk_get_rate(tc->refclk),
|
|
1000 * tc->mode.clock);
|
|
if (ret)
|
|
return ret;
|
|
}
|
|
|
|
ret = tc_set_common_video_mode(tc, &tc->mode);
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = tc_set_edp_video_mode(tc, &tc->mode);
|
|
if (ret)
|
|
return ret;
|
|
|
|
/* Set M/N */
|
|
ret = tc_stream_clock_calc(tc);
|
|
if (ret)
|
|
return ret;
|
|
|
|
value = VID_MN_GEN | DP_EN;
|
|
if (drm_dp_enhanced_frame_cap(tc->link.dpcd))
|
|
value |= EF_EN;
|
|
ret = regmap_write(tc->regmap, DP0CTL, value);
|
|
if (ret)
|
|
return ret;
|
|
/*
|
|
* VID_EN assertion should be delayed by at least N * LSCLK
|
|
* cycles from the time VID_MN_GEN is enabled in order to
|
|
* generate stable values for VID_M. LSCLK is 270 MHz or
|
|
* 162 MHz, VID_N is set to 32768 in tc_stream_clock_calc(),
|
|
* so a delay of at least 203 us should suffice.
|
|
*/
|
|
usleep_range(500, 1000);
|
|
value |= VID_EN;
|
|
ret = regmap_write(tc->regmap, DP0CTL, value);
|
|
if (ret)
|
|
return ret;
|
|
|
|
/* Set input interface */
|
|
if (tc->input_connector_dsi)
|
|
return tc_dsi_rx_enable(tc);
|
|
else
|
|
return tc_dpi_rx_enable(tc);
|
|
}
|
|
|
|
static int tc_edp_stream_disable(struct tc_data *tc)
|
|
{
|
|
int ret;
|
|
|
|
dev_dbg(tc->dev, "disable video stream\n");
|
|
|
|
ret = regmap_update_bits(tc->regmap, DP0CTL, VID_EN, 0);
|
|
if (ret)
|
|
return ret;
|
|
|
|
tc_pxl_pll_dis(tc);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void tc_dpi_bridge_atomic_enable(struct drm_bridge *bridge,
|
|
struct drm_atomic_state *state)
|
|
|
|
{
|
|
struct tc_data *tc = bridge_to_tc(bridge);
|
|
int ret;
|
|
|
|
ret = tc_dpi_stream_enable(tc);
|
|
if (ret < 0) {
|
|
dev_err(tc->dev, "main link stream start error: %d\n", ret);
|
|
tc_main_link_disable(tc);
|
|
return;
|
|
}
|
|
}
|
|
|
|
static void tc_dpi_bridge_atomic_disable(struct drm_bridge *bridge,
|
|
struct drm_atomic_state *state)
|
|
{
|
|
struct tc_data *tc = bridge_to_tc(bridge);
|
|
int ret;
|
|
|
|
ret = tc_dpi_stream_disable(tc);
|
|
if (ret < 0)
|
|
dev_err(tc->dev, "main link stream stop error: %d\n", ret);
|
|
}
|
|
|
|
static void tc_edp_bridge_atomic_enable(struct drm_bridge *bridge,
|
|
struct drm_atomic_state *state)
|
|
{
|
|
struct tc_data *tc = bridge_to_tc(bridge);
|
|
int ret;
|
|
|
|
ret = tc_get_display_props(tc);
|
|
if (ret < 0) {
|
|
dev_err(tc->dev, "failed to read display props: %d\n", ret);
|
|
return;
|
|
}
|
|
|
|
ret = tc_main_link_enable(tc);
|
|
if (ret < 0) {
|
|
dev_err(tc->dev, "main link enable error: %d\n", ret);
|
|
return;
|
|
}
|
|
|
|
ret = tc_edp_stream_enable(tc);
|
|
if (ret < 0) {
|
|
dev_err(tc->dev, "main link stream start error: %d\n", ret);
|
|
tc_main_link_disable(tc);
|
|
return;
|
|
}
|
|
}
|
|
|
|
static void tc_edp_bridge_atomic_disable(struct drm_bridge *bridge,
|
|
struct drm_atomic_state *state)
|
|
{
|
|
struct tc_data *tc = bridge_to_tc(bridge);
|
|
int ret;
|
|
|
|
ret = tc_edp_stream_disable(tc);
|
|
if (ret < 0)
|
|
dev_err(tc->dev, "main link stream stop error: %d\n", ret);
|
|
|
|
ret = tc_main_link_disable(tc);
|
|
if (ret < 0)
|
|
dev_err(tc->dev, "main link disable error: %d\n", ret);
|
|
}
|
|
|
|
static int tc_dpi_atomic_check(struct drm_bridge *bridge,
|
|
struct drm_bridge_state *bridge_state,
|
|
struct drm_crtc_state *crtc_state,
|
|
struct drm_connector_state *conn_state)
|
|
{
|
|
struct tc_data *tc = bridge_to_tc(bridge);
|
|
int adjusted_clock = 0;
|
|
int ret;
|
|
|
|
ret = tc_pxl_pll_calc(tc, clk_get_rate(tc->refclk),
|
|
crtc_state->mode.clock * 1000,
|
|
&adjusted_clock, NULL);
|
|
if (ret)
|
|
return ret;
|
|
|
|
crtc_state->adjusted_mode.clock = adjusted_clock / 1000;
|
|
|
|
/* DSI->DPI interface clock limitation: upto 100 MHz */
|
|
if (crtc_state->adjusted_mode.clock > 100000)
|
|
return -EINVAL;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int tc_edp_atomic_check(struct drm_bridge *bridge,
|
|
struct drm_bridge_state *bridge_state,
|
|
struct drm_crtc_state *crtc_state,
|
|
struct drm_connector_state *conn_state)
|
|
{
|
|
struct tc_data *tc = bridge_to_tc(bridge);
|
|
int adjusted_clock = 0;
|
|
int ret;
|
|
|
|
ret = tc_pxl_pll_calc(tc, clk_get_rate(tc->refclk),
|
|
crtc_state->mode.clock * 1000,
|
|
&adjusted_clock, NULL);
|
|
if (ret)
|
|
return ret;
|
|
|
|
crtc_state->adjusted_mode.clock = adjusted_clock / 1000;
|
|
|
|
/* DPI->(e)DP interface clock limitation: upto 154 MHz */
|
|
if (crtc_state->adjusted_mode.clock > 154000)
|
|
return -EINVAL;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static enum drm_mode_status
|
|
tc_dpi_mode_valid(struct drm_bridge *bridge,
|
|
const struct drm_display_info *info,
|
|
const struct drm_display_mode *mode)
|
|
{
|
|
/* DPI interface clock limitation: upto 100 MHz */
|
|
if (mode->clock > 100000)
|
|
return MODE_CLOCK_HIGH;
|
|
|
|
return MODE_OK;
|
|
}
|
|
|
|
static enum drm_mode_status
|
|
tc_edp_mode_valid(struct drm_bridge *bridge,
|
|
const struct drm_display_info *info,
|
|
const struct drm_display_mode *mode)
|
|
{
|
|
struct tc_data *tc = bridge_to_tc(bridge);
|
|
u32 req, avail;
|
|
u32 bits_per_pixel = 24;
|
|
|
|
/* DPI->(e)DP interface clock limitation: up to 154 MHz */
|
|
if (mode->clock > 154000)
|
|
return MODE_CLOCK_HIGH;
|
|
|
|
req = mode->clock * bits_per_pixel / 8;
|
|
avail = tc->link.num_lanes * tc->link.rate;
|
|
|
|
if (req > avail)
|
|
return MODE_BAD;
|
|
|
|
return MODE_OK;
|
|
}
|
|
|
|
static void tc_bridge_mode_set(struct drm_bridge *bridge,
|
|
const struct drm_display_mode *mode,
|
|
const struct drm_display_mode *adj)
|
|
{
|
|
struct tc_data *tc = bridge_to_tc(bridge);
|
|
|
|
drm_mode_copy(&tc->mode, adj);
|
|
}
|
|
|
|
static const struct drm_edid *tc_edid_read(struct drm_bridge *bridge,
|
|
struct drm_connector *connector)
|
|
{
|
|
struct tc_data *tc = bridge_to_tc(bridge);
|
|
int ret;
|
|
|
|
ret = tc_get_display_props(tc);
|
|
if (ret < 0) {
|
|
dev_err(tc->dev, "failed to read display props: %d\n", ret);
|
|
return 0;
|
|
}
|
|
|
|
return drm_edid_read_ddc(connector, &tc->aux.ddc);
|
|
}
|
|
|
|
static int tc_connector_get_modes(struct drm_connector *connector)
|
|
{
|
|
struct tc_data *tc = connector_to_tc(connector);
|
|
int num_modes;
|
|
const struct drm_edid *drm_edid;
|
|
int ret;
|
|
|
|
ret = tc_get_display_props(tc);
|
|
if (ret < 0) {
|
|
dev_err(tc->dev, "failed to read display props: %d\n", ret);
|
|
return 0;
|
|
}
|
|
|
|
if (tc->panel_bridge) {
|
|
num_modes = drm_bridge_get_modes(tc->panel_bridge, connector);
|
|
if (num_modes > 0)
|
|
return num_modes;
|
|
}
|
|
|
|
drm_edid = tc_edid_read(&tc->bridge, connector);
|
|
drm_edid_connector_update(connector, drm_edid);
|
|
num_modes = drm_edid_connector_add_modes(connector);
|
|
drm_edid_free(drm_edid);
|
|
|
|
return num_modes;
|
|
}
|
|
|
|
static const struct drm_connector_helper_funcs tc_connector_helper_funcs = {
|
|
.get_modes = tc_connector_get_modes,
|
|
};
|
|
|
|
static enum drm_connector_status
|
|
tc_bridge_detect(struct drm_bridge *bridge, struct drm_connector *connector)
|
|
{
|
|
struct tc_data *tc = bridge_to_tc(bridge);
|
|
bool conn;
|
|
u32 val;
|
|
int ret;
|
|
|
|
ret = regmap_read(tc->regmap, GPIOI, &val);
|
|
if (ret)
|
|
return connector_status_unknown;
|
|
|
|
conn = val & BIT(tc->hpd_pin);
|
|
|
|
if (conn)
|
|
return connector_status_connected;
|
|
else
|
|
return connector_status_disconnected;
|
|
}
|
|
|
|
static enum drm_connector_status
|
|
tc_connector_detect(struct drm_connector *connector, bool force)
|
|
{
|
|
struct tc_data *tc = connector_to_tc(connector);
|
|
|
|
if (tc->hpd_pin >= 0)
|
|
return tc_bridge_detect(&tc->bridge, connector);
|
|
|
|
if (tc->panel_bridge)
|
|
return connector_status_connected;
|
|
else
|
|
return connector_status_unknown;
|
|
}
|
|
|
|
static const struct drm_connector_funcs tc_connector_funcs = {
|
|
.detect = tc_connector_detect,
|
|
.fill_modes = drm_helper_probe_single_connector_modes,
|
|
.destroy = drm_connector_cleanup,
|
|
.reset = drm_atomic_helper_connector_reset,
|
|
.atomic_duplicate_state = drm_atomic_helper_connector_duplicate_state,
|
|
.atomic_destroy_state = drm_atomic_helper_connector_destroy_state,
|
|
};
|
|
|
|
static int tc_dpi_bridge_attach(struct drm_bridge *bridge,
|
|
struct drm_encoder *encoder,
|
|
enum drm_bridge_attach_flags flags)
|
|
{
|
|
struct tc_data *tc = bridge_to_tc(bridge);
|
|
|
|
if (!tc->panel_bridge)
|
|
return 0;
|
|
|
|
return drm_bridge_attach(tc->bridge.encoder, tc->panel_bridge,
|
|
&tc->bridge, flags);
|
|
}
|
|
|
|
static int tc_edp_bridge_attach(struct drm_bridge *bridge,
|
|
struct drm_encoder *encoder,
|
|
enum drm_bridge_attach_flags flags)
|
|
{
|
|
u32 bus_format = MEDIA_BUS_FMT_RGB888_1X24;
|
|
struct tc_data *tc = bridge_to_tc(bridge);
|
|
struct drm_device *drm = bridge->dev;
|
|
int ret;
|
|
|
|
if (tc->panel_bridge) {
|
|
/* If a connector is required then this driver shall create it */
|
|
ret = drm_bridge_attach(tc->bridge.encoder, tc->panel_bridge,
|
|
&tc->bridge, flags | DRM_BRIDGE_ATTACH_NO_CONNECTOR);
|
|
if (ret)
|
|
return ret;
|
|
}
|
|
|
|
if (flags & DRM_BRIDGE_ATTACH_NO_CONNECTOR)
|
|
return 0;
|
|
|
|
tc->aux.drm_dev = drm;
|
|
ret = drm_dp_aux_register(&tc->aux);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
/* Create DP/eDP connector */
|
|
drm_connector_helper_add(&tc->connector, &tc_connector_helper_funcs);
|
|
ret = drm_connector_init(drm, &tc->connector, &tc_connector_funcs, tc->bridge.type);
|
|
if (ret)
|
|
goto aux_unregister;
|
|
|
|
/* Don't poll if don't have HPD connected */
|
|
if (tc->hpd_pin >= 0) {
|
|
if (tc->have_irq)
|
|
tc->connector.polled = DRM_CONNECTOR_POLL_HPD;
|
|
else
|
|
tc->connector.polled = DRM_CONNECTOR_POLL_CONNECT |
|
|
DRM_CONNECTOR_POLL_DISCONNECT;
|
|
}
|
|
|
|
drm_display_info_set_bus_formats(&tc->connector.display_info,
|
|
&bus_format, 1);
|
|
tc->connector.display_info.bus_flags =
|
|
DRM_BUS_FLAG_DE_HIGH |
|
|
DRM_BUS_FLAG_PIXDATA_DRIVE_NEGEDGE |
|
|
DRM_BUS_FLAG_SYNC_DRIVE_NEGEDGE;
|
|
drm_connector_attach_encoder(&tc->connector, tc->bridge.encoder);
|
|
|
|
return 0;
|
|
aux_unregister:
|
|
drm_dp_aux_unregister(&tc->aux);
|
|
return ret;
|
|
}
|
|
|
|
static void tc_edp_bridge_detach(struct drm_bridge *bridge)
|
|
{
|
|
drm_dp_aux_unregister(&bridge_to_tc(bridge)->aux);
|
|
}
|
|
|
|
#define MAX_INPUT_SEL_FORMATS 1
|
|
#define MAX_OUTPUT_SEL_FORMATS 1
|
|
|
|
static u32 *
|
|
tc_dpi_atomic_get_input_bus_fmts(struct drm_bridge *bridge,
|
|
struct drm_bridge_state *bridge_state,
|
|
struct drm_crtc_state *crtc_state,
|
|
struct drm_connector_state *conn_state,
|
|
u32 output_fmt,
|
|
unsigned int *num_input_fmts)
|
|
{
|
|
u32 *input_fmts;
|
|
|
|
*num_input_fmts = 0;
|
|
|
|
input_fmts = kcalloc(MAX_INPUT_SEL_FORMATS, sizeof(*input_fmts),
|
|
GFP_KERNEL);
|
|
if (!input_fmts)
|
|
return NULL;
|
|
|
|
/* This is the DSI-end bus format */
|
|
input_fmts[0] = MEDIA_BUS_FMT_RGB888_1X24;
|
|
*num_input_fmts = 1;
|
|
|
|
return input_fmts;
|
|
}
|
|
|
|
static u32 *
|
|
tc_edp_atomic_get_output_bus_fmts(struct drm_bridge *bridge,
|
|
struct drm_bridge_state *bridge_state,
|
|
struct drm_crtc_state *crtc_state,
|
|
struct drm_connector_state *conn_state,
|
|
unsigned int *num_output_fmts)
|
|
{
|
|
u32 *output_fmts;
|
|
|
|
*num_output_fmts = 0;
|
|
|
|
output_fmts = kcalloc(MAX_OUTPUT_SEL_FORMATS, sizeof(*output_fmts),
|
|
GFP_KERNEL);
|
|
if (!output_fmts)
|
|
return NULL;
|
|
|
|
output_fmts[0] = MEDIA_BUS_FMT_RGB888_1X24;
|
|
*num_output_fmts = 1;
|
|
|
|
return output_fmts;
|
|
}
|
|
|
|
static const struct drm_bridge_funcs tc_dpi_bridge_funcs = {
|
|
.attach = tc_dpi_bridge_attach,
|
|
.mode_valid = tc_dpi_mode_valid,
|
|
.mode_set = tc_bridge_mode_set,
|
|
.atomic_check = tc_dpi_atomic_check,
|
|
.atomic_enable = tc_dpi_bridge_atomic_enable,
|
|
.atomic_disable = tc_dpi_bridge_atomic_disable,
|
|
.atomic_duplicate_state = drm_atomic_helper_bridge_duplicate_state,
|
|
.atomic_destroy_state = drm_atomic_helper_bridge_destroy_state,
|
|
.atomic_reset = drm_atomic_helper_bridge_reset,
|
|
.atomic_get_input_bus_fmts = tc_dpi_atomic_get_input_bus_fmts,
|
|
};
|
|
|
|
static const struct drm_bridge_funcs tc_edp_bridge_funcs = {
|
|
.attach = tc_edp_bridge_attach,
|
|
.detach = tc_edp_bridge_detach,
|
|
.mode_valid = tc_edp_mode_valid,
|
|
.mode_set = tc_bridge_mode_set,
|
|
.atomic_check = tc_edp_atomic_check,
|
|
.atomic_enable = tc_edp_bridge_atomic_enable,
|
|
.atomic_disable = tc_edp_bridge_atomic_disable,
|
|
.detect = tc_bridge_detect,
|
|
.edid_read = tc_edid_read,
|
|
.atomic_duplicate_state = drm_atomic_helper_bridge_duplicate_state,
|
|
.atomic_destroy_state = drm_atomic_helper_bridge_destroy_state,
|
|
.atomic_reset = drm_atomic_helper_bridge_reset,
|
|
.atomic_get_input_bus_fmts = drm_atomic_helper_bridge_propagate_bus_fmt,
|
|
.atomic_get_output_bus_fmts = tc_edp_atomic_get_output_bus_fmts,
|
|
};
|
|
|
|
static bool tc_readable_reg(struct device *dev, unsigned int reg)
|
|
{
|
|
switch (reg) {
|
|
/* DSI D-PHY Layer */
|
|
case 0x004:
|
|
case 0x020:
|
|
case 0x024:
|
|
case 0x028:
|
|
case 0x02c:
|
|
case 0x030:
|
|
case 0x038:
|
|
case 0x040:
|
|
case 0x044:
|
|
case 0x048:
|
|
case 0x04c:
|
|
case 0x050:
|
|
case 0x054:
|
|
/* DSI PPI Layer */
|
|
case PPI_STARTPPI:
|
|
case 0x108:
|
|
case 0x110:
|
|
case PPI_LPTXTIMECNT:
|
|
case PPI_LANEENABLE:
|
|
case PPI_TX_RX_TA:
|
|
case 0x140:
|
|
case PPI_D0S_ATMR:
|
|
case PPI_D1S_ATMR:
|
|
case 0x14c:
|
|
case 0x150:
|
|
case PPI_D0S_CLRSIPOCOUNT:
|
|
case PPI_D1S_CLRSIPOCOUNT:
|
|
case PPI_D2S_CLRSIPOCOUNT:
|
|
case PPI_D3S_CLRSIPOCOUNT:
|
|
case 0x180:
|
|
case 0x184:
|
|
case 0x188:
|
|
case 0x18c:
|
|
case 0x190:
|
|
case 0x1a0:
|
|
case 0x1a4:
|
|
case 0x1a8:
|
|
case 0x1ac:
|
|
case 0x1b0:
|
|
case 0x1c0:
|
|
case 0x1c4:
|
|
case 0x1c8:
|
|
case 0x1cc:
|
|
case 0x1d0:
|
|
case 0x1e0:
|
|
case 0x1e4:
|
|
case 0x1f0:
|
|
case 0x1f4:
|
|
/* DSI Protocol Layer */
|
|
case DSI_STARTDSI:
|
|
case DSI_BUSYDSI:
|
|
case DSI_LANEENABLE:
|
|
case DSI_LANESTATUS0:
|
|
case DSI_LANESTATUS1:
|
|
case DSI_INTSTATUS:
|
|
case 0x224:
|
|
case 0x228:
|
|
case 0x230:
|
|
/* DSI General */
|
|
case DSIERRCNT:
|
|
/* DSI Application Layer */
|
|
case 0x400:
|
|
case 0x404:
|
|
/* DPI */
|
|
case DPIPXLFMT:
|
|
/* Parallel Output */
|
|
case POCTRL:
|
|
/* Video Path0 Configuration */
|
|
case VPCTRL0:
|
|
case HTIM01:
|
|
case HTIM02:
|
|
case VTIM01:
|
|
case VTIM02:
|
|
case VFUEN0:
|
|
/* System */
|
|
case TC_IDREG:
|
|
case 0x504:
|
|
case SYSSTAT:
|
|
case SYSRSTENB:
|
|
case SYSCTRL:
|
|
/* I2C */
|
|
case 0x520:
|
|
/* GPIO */
|
|
case GPIOM:
|
|
case GPIOC:
|
|
case GPIOO:
|
|
case GPIOI:
|
|
/* Interrupt */
|
|
case INTCTL_G:
|
|
case INTSTS_G:
|
|
case 0x570:
|
|
case 0x574:
|
|
case INT_GP0_LCNT:
|
|
case INT_GP1_LCNT:
|
|
/* DisplayPort Control */
|
|
case DP0CTL:
|
|
/* DisplayPort Clock */
|
|
case DP0_VIDMNGEN0:
|
|
case DP0_VIDMNGEN1:
|
|
case DP0_VMNGENSTATUS:
|
|
case 0x628:
|
|
case 0x62c:
|
|
case 0x630:
|
|
/* DisplayPort Main Channel */
|
|
case DP0_SECSAMPLE:
|
|
case DP0_VIDSYNCDELAY:
|
|
case DP0_TOTALVAL:
|
|
case DP0_STARTVAL:
|
|
case DP0_ACTIVEVAL:
|
|
case DP0_SYNCVAL:
|
|
case DP0_MISC:
|
|
/* DisplayPort Aux Channel */
|
|
case DP0_AUXCFG0:
|
|
case DP0_AUXCFG1:
|
|
case DP0_AUXADDR:
|
|
case 0x66c:
|
|
case 0x670:
|
|
case 0x674:
|
|
case 0x678:
|
|
case 0x67c:
|
|
case 0x680:
|
|
case 0x684:
|
|
case 0x688:
|
|
case DP0_AUXSTATUS:
|
|
case DP0_AUXI2CADR:
|
|
/* DisplayPort Link Training */
|
|
case DP0_SRCCTRL:
|
|
case DP0_LTSTAT:
|
|
case DP0_SNKLTCHGREQ:
|
|
case DP0_LTLOOPCTRL:
|
|
case DP0_SNKLTCTRL:
|
|
case 0x6e8:
|
|
case 0x6ec:
|
|
case 0x6f0:
|
|
case 0x6f4:
|
|
/* DisplayPort Audio */
|
|
case 0x700:
|
|
case 0x704:
|
|
case 0x708:
|
|
case 0x70c:
|
|
case 0x710:
|
|
case 0x714:
|
|
case 0x718:
|
|
case 0x71c:
|
|
case 0x720:
|
|
/* DisplayPort Source Control */
|
|
case DP1_SRCCTRL:
|
|
/* DisplayPort PHY */
|
|
case DP_PHY_CTRL:
|
|
case 0x810:
|
|
case 0x814:
|
|
case 0x820:
|
|
case 0x840:
|
|
/* I2S */
|
|
case 0x880:
|
|
case 0x888:
|
|
case 0x88c:
|
|
case 0x890:
|
|
case 0x894:
|
|
case 0x898:
|
|
case 0x89c:
|
|
case 0x8a0:
|
|
case 0x8a4:
|
|
case 0x8a8:
|
|
case 0x8ac:
|
|
case 0x8b0:
|
|
case 0x8b4:
|
|
/* PLL */
|
|
case DP0_PLLCTRL:
|
|
case DP1_PLLCTRL:
|
|
case PXL_PLLCTRL:
|
|
case PXL_PLLPARAM:
|
|
case SYS_PLLPARAM:
|
|
/* HDCP */
|
|
case 0x980:
|
|
case 0x984:
|
|
case 0x988:
|
|
case 0x98c:
|
|
case 0x990:
|
|
case 0x994:
|
|
case 0x998:
|
|
case 0x99c:
|
|
case 0x9a0:
|
|
case 0x9a4:
|
|
case 0x9a8:
|
|
case 0x9ac:
|
|
/* Debug */
|
|
case TSTCTL:
|
|
case PLL_DBG:
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static const struct regmap_range tc_volatile_ranges[] = {
|
|
regmap_reg_range(PPI_BUSYPPI, PPI_BUSYPPI),
|
|
regmap_reg_range(DSI_BUSYDSI, DSI_BUSYDSI),
|
|
regmap_reg_range(DSI_LANESTATUS0, DSI_INTSTATUS),
|
|
regmap_reg_range(DSIERRCNT, DSIERRCNT),
|
|
regmap_reg_range(VFUEN0, VFUEN0),
|
|
regmap_reg_range(SYSSTAT, SYSSTAT),
|
|
regmap_reg_range(GPIOI, GPIOI),
|
|
regmap_reg_range(INTSTS_G, INTSTS_G),
|
|
regmap_reg_range(DP0_VMNGENSTATUS, DP0_VMNGENSTATUS),
|
|
regmap_reg_range(DP0_AMNGENSTATUS, DP0_AMNGENSTATUS),
|
|
regmap_reg_range(DP0_AUXWDATA(0), DP0_AUXSTATUS),
|
|
regmap_reg_range(DP0_LTSTAT, DP0_SNKLTCHGREQ),
|
|
regmap_reg_range(DP_PHY_CTRL, DP_PHY_CTRL),
|
|
regmap_reg_range(DP0_PLLCTRL, PXL_PLLCTRL),
|
|
};
|
|
|
|
static const struct regmap_access_table tc_volatile_table = {
|
|
.yes_ranges = tc_volatile_ranges,
|
|
.n_yes_ranges = ARRAY_SIZE(tc_volatile_ranges),
|
|
};
|
|
|
|
static const struct regmap_range tc_precious_ranges[] = {
|
|
regmap_reg_range(SYSSTAT, SYSSTAT),
|
|
};
|
|
|
|
static const struct regmap_access_table tc_precious_table = {
|
|
.yes_ranges = tc_precious_ranges,
|
|
.n_yes_ranges = ARRAY_SIZE(tc_precious_ranges),
|
|
};
|
|
|
|
static bool tc_writeable_reg(struct device *dev, unsigned int reg)
|
|
{
|
|
/* RO reg */
|
|
switch (reg) {
|
|
case PPI_BUSYPPI:
|
|
case DSI_BUSYDSI:
|
|
case DSI_LANESTATUS0:
|
|
case DSI_LANESTATUS1:
|
|
case DSI_INTSTATUS:
|
|
case TC_IDREG:
|
|
case SYSBOOT:
|
|
case SYSSTAT:
|
|
case GPIOI:
|
|
case DP0_LTSTAT:
|
|
case DP0_SNKLTCHGREQ:
|
|
return false;
|
|
}
|
|
/* WO reg */
|
|
switch (reg) {
|
|
case DSI_STARTDSI:
|
|
case DSI_INTCLR:
|
|
return true;
|
|
}
|
|
return tc_readable_reg(dev, reg);
|
|
}
|
|
|
|
static const struct regmap_config tc_regmap_config = {
|
|
.name = "tc358767",
|
|
.reg_bits = 16,
|
|
.val_bits = 32,
|
|
.reg_stride = 4,
|
|
.max_register = PLL_DBG,
|
|
.cache_type = REGCACHE_MAPLE,
|
|
.readable_reg = tc_readable_reg,
|
|
.writeable_reg = tc_writeable_reg,
|
|
.volatile_table = &tc_volatile_table,
|
|
.precious_table = &tc_precious_table,
|
|
.reg_format_endian = REGMAP_ENDIAN_BIG,
|
|
.val_format_endian = REGMAP_ENDIAN_LITTLE,
|
|
};
|
|
|
|
static irqreturn_t tc_irq_handler(int irq, void *arg)
|
|
{
|
|
struct tc_data *tc = arg;
|
|
u32 val;
|
|
int r;
|
|
|
|
r = regmap_read(tc->regmap, INTSTS_G, &val);
|
|
if (r)
|
|
return IRQ_NONE;
|
|
|
|
if (!val)
|
|
return IRQ_NONE;
|
|
|
|
if (val & INT_SYSERR) {
|
|
u32 stat = 0;
|
|
|
|
regmap_read(tc->regmap, SYSSTAT, &stat);
|
|
|
|
dev_err(tc->dev, "syserr %x\n", stat);
|
|
}
|
|
|
|
if (tc->hpd_pin >= 0 && tc->bridge.dev && tc->aux.drm_dev) {
|
|
/*
|
|
* H is triggered when the GPIO goes high.
|
|
*
|
|
* LC is triggered when the GPIO goes low and stays low for
|
|
* the duration of LCNT
|
|
*/
|
|
bool h = val & INT_GPIO_H(tc->hpd_pin);
|
|
bool lc = val & INT_GPIO_LC(tc->hpd_pin);
|
|
|
|
if (h || lc) {
|
|
dev_dbg(tc->dev, "GPIO%d: %s %s\n", tc->hpd_pin,
|
|
h ? "H" : "", lc ? "LC" : "");
|
|
drm_kms_helper_hotplug_event(tc->bridge.dev);
|
|
}
|
|
}
|
|
|
|
regmap_write(tc->regmap, INTSTS_G, val);
|
|
|
|
return IRQ_HANDLED;
|
|
}
|
|
|
|
static int tc_mipi_dsi_host_attach(struct tc_data *tc)
|
|
{
|
|
struct device *dev = tc->dev;
|
|
struct device_node *host_node;
|
|
struct device_node *endpoint;
|
|
struct mipi_dsi_device *dsi;
|
|
struct mipi_dsi_host *host;
|
|
const struct mipi_dsi_device_info info = {
|
|
.type = "tc358767",
|
|
.channel = 0,
|
|
.node = NULL,
|
|
};
|
|
int dsi_lanes, ret;
|
|
|
|
endpoint = of_graph_get_endpoint_by_regs(dev->of_node, 0, -1);
|
|
dsi_lanes = drm_of_get_data_lanes_count(endpoint, 1, 4);
|
|
host_node = of_graph_get_remote_port_parent(endpoint);
|
|
host = of_find_mipi_dsi_host_by_node(host_node);
|
|
of_node_put(host_node);
|
|
of_node_put(endpoint);
|
|
|
|
if (!host)
|
|
return -EPROBE_DEFER;
|
|
|
|
if (dsi_lanes < 0)
|
|
return dsi_lanes;
|
|
|
|
dsi = devm_mipi_dsi_device_register_full(dev, host, &info);
|
|
if (IS_ERR(dsi))
|
|
return dev_err_probe(dev, PTR_ERR(dsi),
|
|
"failed to create dsi device\n");
|
|
|
|
tc->dsi = dsi;
|
|
dsi->lanes = dsi_lanes;
|
|
dsi->format = MIPI_DSI_FMT_RGB888;
|
|
dsi->mode_flags = MIPI_DSI_MODE_VIDEO | MIPI_DSI_MODE_VIDEO_BURST |
|
|
MIPI_DSI_MODE_LPM | MIPI_DSI_CLOCK_NON_CONTINUOUS;
|
|
|
|
ret = devm_mipi_dsi_attach(dev, dsi);
|
|
if (ret < 0) {
|
|
dev_err(dev, "failed to attach dsi to host: %d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int tc_probe_dpi_bridge_endpoint(struct tc_data *tc)
|
|
{
|
|
struct device *dev = tc->dev;
|
|
struct drm_bridge *bridge;
|
|
struct drm_panel *panel;
|
|
int ret;
|
|
|
|
/* port@1 is the DPI input/output port */
|
|
ret = drm_of_find_panel_or_bridge(dev->of_node, 1, 0, &panel, &bridge);
|
|
if (ret && ret != -ENODEV)
|
|
return dev_err_probe(dev, ret,
|
|
"Could not find DPI panel or bridge\n");
|
|
|
|
if (panel) {
|
|
bridge = devm_drm_panel_bridge_add(dev, panel);
|
|
if (IS_ERR(bridge))
|
|
return PTR_ERR(bridge);
|
|
}
|
|
|
|
if (bridge) {
|
|
tc->panel_bridge = bridge;
|
|
tc->bridge.type = DRM_MODE_CONNECTOR_DPI;
|
|
|
|
return 0;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int tc_probe_edp_bridge_endpoint(struct tc_data *tc)
|
|
{
|
|
struct device *dev = tc->dev;
|
|
struct drm_panel *panel;
|
|
int ret;
|
|
|
|
/* port@2 is the output port */
|
|
ret = drm_of_find_panel_or_bridge(dev->of_node, 2, 0, &panel, NULL);
|
|
if (ret && ret != -ENODEV)
|
|
return dev_err_probe(dev, ret,
|
|
"Could not find DSI panel or bridge\n");
|
|
|
|
if (panel) {
|
|
struct drm_bridge *panel_bridge;
|
|
|
|
panel_bridge = devm_drm_panel_bridge_add(dev, panel);
|
|
if (IS_ERR(panel_bridge))
|
|
return PTR_ERR(panel_bridge);
|
|
|
|
tc->panel_bridge = panel_bridge;
|
|
tc->bridge.type = DRM_MODE_CONNECTOR_eDP;
|
|
} else {
|
|
tc->bridge.type = DRM_MODE_CONNECTOR_DisplayPort;
|
|
}
|
|
|
|
if (tc->hpd_pin >= 0)
|
|
tc->bridge.ops |= DRM_BRIDGE_OP_DETECT;
|
|
tc->bridge.ops |= DRM_BRIDGE_OP_EDID;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static enum tc_mode tc_probe_get_mode(struct device *dev)
|
|
{
|
|
struct of_endpoint endpoint;
|
|
struct device_node *node = NULL;
|
|
enum tc_mode mode = 0;
|
|
|
|
/*
|
|
* Determine bridge configuration.
|
|
*
|
|
* Port allocation:
|
|
* port@0 - DSI input
|
|
* port@1 - DPI input/output
|
|
* port@2 - eDP output
|
|
*
|
|
* Possible connections:
|
|
* DPI -> port@1 -> port@2 -> eDP :: [port@0 is not connected]
|
|
* DSI -> port@0 -> port@2 -> eDP :: [port@1 is not connected]
|
|
* DSI -> port@0 -> port@1 -> DPI :: [port@2 is not connected]
|
|
*/
|
|
|
|
for_each_endpoint_of_node(dev->of_node, node) {
|
|
of_graph_parse_endpoint(node, &endpoint);
|
|
if (endpoint.port > 2) {
|
|
of_node_put(node);
|
|
return -EINVAL;
|
|
}
|
|
mode |= BIT(endpoint.port);
|
|
}
|
|
|
|
if (mode != mode_dpi_to_edp &&
|
|
mode != mode_dpi_to_dp &&
|
|
mode != mode_dsi_to_dpi &&
|
|
mode != mode_dsi_to_edp &&
|
|
mode != mode_dsi_to_dp) {
|
|
dev_warn(dev, "Invalid mode (0x%x) is not supported!\n", mode);
|
|
return -EINVAL;
|
|
}
|
|
|
|
return mode;
|
|
}
|
|
|
|
static int tc_probe_bridge_endpoint(struct tc_data *tc, enum tc_mode mode)
|
|
{
|
|
struct device *dev = tc->dev;
|
|
struct of_endpoint endpoint;
|
|
struct device_node *node = NULL;
|
|
|
|
for_each_endpoint_of_node(dev->of_node, node) {
|
|
of_graph_parse_endpoint(node, &endpoint);
|
|
if (endpoint.port == 2) {
|
|
of_property_read_u8_array(node, "toshiba,pre-emphasis",
|
|
tc->pre_emphasis,
|
|
ARRAY_SIZE(tc->pre_emphasis));
|
|
|
|
if (tc->pre_emphasis[0] < 0 || tc->pre_emphasis[0] > 2 ||
|
|
tc->pre_emphasis[1] < 0 || tc->pre_emphasis[1] > 2) {
|
|
dev_err(dev, "Incorrect Pre-Emphasis setting, use either 0=0dB 1=3.5dB 2=6dB\n");
|
|
of_node_put(node);
|
|
return -EINVAL;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (mode == mode_dpi_to_edp || mode == mode_dpi_to_dp) {
|
|
tc->input_connector_dsi = false;
|
|
return tc_probe_edp_bridge_endpoint(tc);
|
|
} else if (mode == mode_dsi_to_dpi) {
|
|
tc->input_connector_dsi = true;
|
|
return tc_probe_dpi_bridge_endpoint(tc);
|
|
} else if (mode == mode_dsi_to_edp || mode == mode_dsi_to_dp) {
|
|
tc->input_connector_dsi = true;
|
|
return tc_probe_edp_bridge_endpoint(tc);
|
|
}
|
|
|
|
/* Should never happen, mode was validated by tc_probe_get_mode() */
|
|
return -EINVAL;
|
|
}
|
|
|
|
static int tc_probe(struct i2c_client *client)
|
|
{
|
|
struct device *dev = &client->dev;
|
|
const struct drm_bridge_funcs *funcs;
|
|
struct tc_data *tc;
|
|
int mode;
|
|
int ret;
|
|
|
|
mode = tc_probe_get_mode(dev);
|
|
funcs = (mode == mode_dsi_to_dpi) ? &tc_dpi_bridge_funcs : &tc_edp_bridge_funcs;
|
|
|
|
tc = devm_drm_bridge_alloc(dev, struct tc_data, bridge, funcs);
|
|
if (IS_ERR(tc))
|
|
return PTR_ERR(tc);
|
|
|
|
tc->dev = dev;
|
|
|
|
ret = tc_probe_bridge_endpoint(tc, mode);
|
|
if (ret)
|
|
return ret;
|
|
|
|
tc->refclk = devm_clk_get_enabled(dev, "ref");
|
|
if (IS_ERR(tc->refclk))
|
|
return dev_err_probe(dev, PTR_ERR(tc->refclk),
|
|
"Failed to get and enable the ref clk\n");
|
|
|
|
/* tRSTW = 100 cycles , at 13 MHz that is ~7.69 us */
|
|
usleep_range(10, 15);
|
|
|
|
/* Shut down GPIO is optional */
|
|
tc->sd_gpio = devm_gpiod_get_optional(dev, "shutdown", GPIOD_OUT_HIGH);
|
|
if (IS_ERR(tc->sd_gpio))
|
|
return PTR_ERR(tc->sd_gpio);
|
|
|
|
if (tc->sd_gpio) {
|
|
gpiod_set_value_cansleep(tc->sd_gpio, 0);
|
|
usleep_range(5000, 10000);
|
|
}
|
|
|
|
/* Reset GPIO is optional */
|
|
tc->reset_gpio = devm_gpiod_get_optional(dev, "reset", GPIOD_OUT_LOW);
|
|
if (IS_ERR(tc->reset_gpio))
|
|
return PTR_ERR(tc->reset_gpio);
|
|
|
|
if (tc->reset_gpio) {
|
|
gpiod_set_value_cansleep(tc->reset_gpio, 1);
|
|
usleep_range(5000, 10000);
|
|
}
|
|
|
|
tc->regmap = devm_regmap_init_i2c(client, &tc_regmap_config);
|
|
if (IS_ERR(tc->regmap)) {
|
|
ret = PTR_ERR(tc->regmap);
|
|
dev_err(dev, "Failed to initialize regmap: %d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
ret = of_property_read_u32(dev->of_node, "toshiba,hpd-pin",
|
|
&tc->hpd_pin);
|
|
if (ret) {
|
|
tc->hpd_pin = -ENODEV;
|
|
} else {
|
|
if (tc->hpd_pin < 0 || tc->hpd_pin > 1) {
|
|
dev_err(dev, "failed to parse HPD number\n");
|
|
return -EINVAL;
|
|
}
|
|
}
|
|
|
|
if (client->irq > 0) {
|
|
/* enable SysErr */
|
|
regmap_write(tc->regmap, INTCTL_G, INT_SYSERR);
|
|
|
|
ret = devm_request_threaded_irq(dev, client->irq,
|
|
NULL, tc_irq_handler,
|
|
IRQF_ONESHOT,
|
|
"tc358767-irq", tc);
|
|
if (ret) {
|
|
dev_err(dev, "failed to register dp interrupt\n");
|
|
return ret;
|
|
}
|
|
|
|
tc->have_irq = true;
|
|
}
|
|
|
|
ret = regmap_read(tc->regmap, TC_IDREG, &tc->rev);
|
|
if (ret) {
|
|
dev_err(tc->dev, "can not read device ID: %d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
if ((tc->rev != 0x6601) && (tc->rev != 0x6603)) {
|
|
dev_err(tc->dev, "invalid device ID: 0x%08x\n", tc->rev);
|
|
return -EINVAL;
|
|
}
|
|
|
|
tc->assr = (tc->rev == 0x6601); /* Enable ASSR for eDP panels */
|
|
|
|
if (!tc->reset_gpio) {
|
|
/*
|
|
* If the reset pin isn't present, do a software reset. It isn't
|
|
* as thorough as the hardware reset, as we can't reset the I2C
|
|
* communication block for obvious reasons, but it's getting the
|
|
* chip into a defined state.
|
|
*/
|
|
regmap_update_bits(tc->regmap, SYSRSTENB,
|
|
ENBLCD0 | ENBBM | ENBDSIRX | ENBREG | ENBHDCP,
|
|
0);
|
|
regmap_update_bits(tc->regmap, SYSRSTENB,
|
|
ENBLCD0 | ENBBM | ENBDSIRX | ENBREG | ENBHDCP,
|
|
ENBLCD0 | ENBBM | ENBDSIRX | ENBREG | ENBHDCP);
|
|
usleep_range(5000, 10000);
|
|
}
|
|
|
|
if (tc->hpd_pin >= 0) {
|
|
u32 lcnt_reg = tc->hpd_pin == 0 ? INT_GP0_LCNT : INT_GP1_LCNT;
|
|
u32 h_lc = INT_GPIO_H(tc->hpd_pin) | INT_GPIO_LC(tc->hpd_pin);
|
|
|
|
/* Set LCNT to 2ms */
|
|
regmap_write(tc->regmap, lcnt_reg,
|
|
clk_get_rate(tc->refclk) * 2 / 1000);
|
|
/* We need the "alternate" mode for HPD */
|
|
regmap_write(tc->regmap, GPIOM, BIT(tc->hpd_pin));
|
|
|
|
if (tc->have_irq) {
|
|
/* enable H & LC */
|
|
regmap_update_bits(tc->regmap, INTCTL_G, h_lc, h_lc);
|
|
}
|
|
}
|
|
|
|
if (tc->bridge.type != DRM_MODE_CONNECTOR_DPI) { /* (e)DP output */
|
|
ret = tc_aux_link_setup(tc);
|
|
if (ret)
|
|
return ret;
|
|
}
|
|
|
|
tc->bridge.of_node = dev->of_node;
|
|
drm_bridge_add(&tc->bridge);
|
|
|
|
i2c_set_clientdata(client, tc);
|
|
|
|
if (tc->input_connector_dsi) { /* DSI input */
|
|
ret = tc_mipi_dsi_host_attach(tc);
|
|
if (ret) {
|
|
drm_bridge_remove(&tc->bridge);
|
|
return dev_err_probe(dev, ret, "Failed to attach DSI host\n");
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void tc_remove(struct i2c_client *client)
|
|
{
|
|
struct tc_data *tc = i2c_get_clientdata(client);
|
|
|
|
drm_bridge_remove(&tc->bridge);
|
|
}
|
|
|
|
static const struct i2c_device_id tc358767_i2c_ids[] = {
|
|
{ "tc358767" },
|
|
{ }
|
|
};
|
|
MODULE_DEVICE_TABLE(i2c, tc358767_i2c_ids);
|
|
|
|
static const struct of_device_id tc358767_of_ids[] = {
|
|
{ .compatible = "toshiba,tc358767", },
|
|
{ }
|
|
};
|
|
MODULE_DEVICE_TABLE(of, tc358767_of_ids);
|
|
|
|
static struct i2c_driver tc358767_driver = {
|
|
.driver = {
|
|
.name = "tc358767",
|
|
.of_match_table = tc358767_of_ids,
|
|
},
|
|
.id_table = tc358767_i2c_ids,
|
|
.probe = tc_probe,
|
|
.remove = tc_remove,
|
|
};
|
|
module_i2c_driver(tc358767_driver);
|
|
|
|
MODULE_AUTHOR("Andrey Gusakov <andrey.gusakov@cogentembedded.com>");
|
|
MODULE_DESCRIPTION("tc358767 eDP encoder driver");
|
|
MODULE_LICENSE("GPL");
|