/* SPDX-License-Identifier: GPL-2.0-only * * Copyright © 2019-2020 Intel Corporation */ #ifndef __KMB_DSI_H__ #define __KMB_DSI_H__ #include #include /* MIPI TX CFG */ #define MIPI_TX_LANE_DATA_RATE_MBPS 891 #define MIPI_TX_REF_CLK_KHZ 24000 #define MIPI_TX_CFG_CLK_KHZ 24000 #define MIPI_TX_BPP 24 /* DPHY Tx test codes*/ #define TEST_CODE_FSM_CONTROL 0x03 #define TEST_CODE_MULTIPLE_PHY_CTRL 0x0C #define TEST_CODE_PLL_PROPORTIONAL_CHARGE_PUMP_CTRL 0x0E #define TEST_CODE_PLL_INTEGRAL_CHARGE_PUMP_CTRL 0x0F #define TEST_CODE_PLL_VCO_CTRL 0x12 #define TEST_CODE_PLL_GMP_CTRL 0x13 #define TEST_CODE_PLL_PHASE_ERR_CTRL 0x14 #define TEST_CODE_PLL_LOCK_FILTER 0x15 #define TEST_CODE_PLL_UNLOCK_FILTER 0x16 #define TEST_CODE_PLL_INPUT_DIVIDER 0x17 #define TEST_CODE_PLL_FEEDBACK_DIVIDER 0x18 #define PLL_FEEDBACK_DIVIDER_HIGH BIT(7) #define TEST_CODE_PLL_OUTPUT_CLK_SEL 0x19 #define PLL_N_OVR_EN BIT(4) #define PLL_M_OVR_EN BIT(5) #define TEST_CODE_VOD_LEVEL 0x24 #define TEST_CODE_PLL_CHARGE_PUMP_BIAS 0x1C #define TEST_CODE_PLL_LOCK_DETECTOR 0x1D #define TEST_CODE_HS_FREQ_RANGE_CFG 0x44 #define TEST_CODE_PLL_ANALOG_PROG 0x1F #define TEST_CODE_SLEW_RATE_OVERRIDE_CTRL 0xA0 #define TEST_CODE_SLEW_RATE_DDL_LOOP_CTRL 0xA3 #define TEST_CODE_SLEW_RATE_DDL_CYCLES 0xA4 /* DPHY params */ #define PLL_N_MIN 0 #define PLL_N_MAX 15 #define PLL_M_MIN 62 #define PLL_M_MAX 623 #define PLL_FVCO_MAX 1250 #define TIMEOUT 600 #define MIPI_TX_FRAME_GEN 4 #define MIPI_TX_FRAME_GEN_SECTIONS 4 #define MIPI_CTRL_VIRTUAL_CHANNELS 4 #define MIPI_D_LANES_PER_DPHY 2 #define MIPI_CTRL_2LANE_MAX_MC_FIFO_LOC 255 #define MIPI_CTRL_4LANE_MAX_MC_FIFO_LOC 511 /* 2 Data Lanes per D-PHY */ #define MIPI_DPHY_D_LANES 2 #define MIPI_DPHY_DEFAULT_BIT_RATES 63 #define KMB_MIPI_DEFAULT_CLK 24000000 #define KMB_MIPI_DEFAULT_CFG_CLK 24000000 #define to_kmb_dsi(x) container_of(x, struct kmb_dsi, base) struct kmb_dsi { struct drm_encoder base; struct device *dev; struct platform_device *pdev; struct mipi_dsi_host *host; struct mipi_dsi_device *device; struct drm_bridge *adv_bridge; void __iomem *mipi_mmio; struct clk *clk_mipi; struct clk *clk_mipi_ecfg; struct clk *clk_mipi_cfg; int sys_clk_mhz; }; /* DPHY Tx test codes */ enum mipi_ctrl_num { MIPI_CTRL0 = 0, MIPI_CTRL1, MIPI_CTRL2, MIPI_CTRL3, MIPI_CTRL4, MIPI_CTRL5, MIPI_CTRL6, MIPI_CTRL7, MIPI_CTRL8, MIPI_CTRL9, MIPI_CTRL_NA }; enum mipi_dphy_num { MIPI_DPHY0 = 0, MIPI_DPHY1, MIPI_DPHY2, MIPI_DPHY3, MIPI_DPHY4, MIPI_DPHY5, MIPI_DPHY6, MIPI_DPHY7, MIPI_DPHY8, MIPI_DPHY9, MIPI_DPHY_NA }; enum mipi_dir { MIPI_RX, MIPI_TX }; enum mipi_ctrl_type { MIPI_DSI, MIPI_CSI }; enum mipi_data_if { MIPI_IF_DMA, MIPI_IF_PARALLEL }; enum mipi_data_mode { MIPI_DATA_MODE0, MIPI_DATA_MODE1, MIPI_DATA_MODE2, MIPI_DATA_MODE3 }; enum mipi_dsi_video_mode { DSI_VIDEO_MODE_NO_BURST_PULSE, DSI_VIDEO_MODE_NO_BURST_EVENT, DSI_VIDEO_MODE_BURST }; enum mipi_dsi_blanking_mode { TRANSITION_TO_LOW_POWER, SEND_BLANK_PACKET }; enum mipi_dsi_eotp { DSI_EOTP_DISABLED, DSI_EOTP_ENABLES }; enum mipi_dsi_data_type { DSI_SP_DT_RESERVED_00 = 0x00, DSI_SP_DT_VSYNC_START = 0x01, DSI_SP_DT_COLOR_MODE_OFF = 0x02, DSI_SP_DT_GENERIC_SHORT_WR = 0x03, DSI_SP_DT_GENERIC_RD = 0x04, DSI_SP_DT_DCS_SHORT_WR = 0x05, DSI_SP_DT_DCS_RD = 0x06, DSI_SP_DT_EOTP = 0x08, DSI_LP_DT_NULL = 0x09, DSI_LP_DT_RESERVED_0A = 0x0a, DSI_LP_DT_RESERVED_0B = 0x0b, DSI_LP_DT_LPPS_YCBCR422_20B = 0x0c, DSI_LP_DT_PPS_RGB101010_30B = 0x0d, DSI_LP_DT_PPS_RGB565_16B = 0x0e, DSI_LP_DT_RESERVED_0F = 0x0f, DSI_SP_DT_RESERVED_10 = 0x10, DSI_SP_DT_VSYNC_END = 0x11, DSI_SP_DT_COLOR_MODE_ON = 0x12, DSI_SP_DT_GENERIC_SHORT_WR_1PAR = 0x13, DSI_SP_DT_GENERIC_RD_1PAR = 0x14, DSI_SP_DT_DCS_SHORT_WR_1PAR = 0x15, DSI_SP_DT_RESERVED_16 = 0x16, DSI_SP_DT_RESERVED_17 = 0x17, DSI_SP_DT_RESERVED_18 = 0x18, DSI_LP_DT_BLANK = 0x19, DSI_LP_DT_RESERVED_1A = 0x1a, DSI_LP_DT_RESERVED_1B = 0x1b, DSI_LP_DT_PPS_YCBCR422_24B = 0x1c, DSI_LP_DT_PPS_RGB121212_36B = 0x1d, DSI_LP_DT_PPS_RGB666_18B = 0x1e, DSI_LP_DT_RESERVED_1F = 0x1f, DSI_SP_DT_RESERVED_20 = 0x20, DSI_SP_DT_HSYNC_START = 0x21, DSI_SP_DT_SHUT_DOWN_PERIPH_CMD = 0x22, DSI_SP_DT_GENERIC_SHORT_WR_2PAR = 0x23, DSI_SP_DT_GENERIC_RD_2PAR = 0x24, DSI_SP_DT_RESERVED_25 = 0x25, DSI_SP_DT_RESERVED_26 = 0x26, DSI_SP_DT_RESERVED_27 = 0x27, DSI_SP_DT_RESERVED_28 = 0x28, DSI_LP_DT_GENERIC_LONG_WR = 0x29, DSI_LP_DT_RESERVED_2A = 0x2a, DSI_LP_DT_RESERVED_2B = 0x2b, DSI_LP_DT_PPS_YCBCR422_16B = 0x2c, DSI_LP_DT_RESERVED_2D = 0x2d, DSI_LP_DT_LPPS_RGB666_18B = 0x2e, DSI_LP_DT_RESERVED_2F = 0x2f, DSI_SP_DT_RESERVED_30 = 0x30, DSI_SP_DT_HSYNC_END = 0x31, DSI_SP_DT_TURN_ON_PERIPH_CMD = 0x32, DSI_SP_DT_RESERVED_33 = 0x33, DSI_SP_DT_RESERVED_34 = 0x34, DSI_SP_DT_RESERVED_35 = 0x35, DSI_SP_DT_RESERVED_36 = 0x36, DSI_SP_DT_SET_MAX_RETURN_PKT_SIZE = 0x37, DSI_SP_DT_RESERVED_38 = 0x38, DSI_LP_DT_DSC_LONG_WR = 0x39, DSI_LP_DT_RESERVED_3A = 0x3a, DSI_LP_DT_RESERVED_3B = 0x3b, DSI_LP_DT_RESERVED_3C = 0x3c, DSI_LP_DT_PPS_YCBCR420_12B = 0x3d, DSI_LP_DT_PPS_RGB888_24B = 0x3e, DSI_LP_DT_RESERVED_3F = 0x3f }; enum mipi_tx_hs_tp_sel { MIPI_TX_HS_TP_WHOLE_FRAME_COLOR0 = 0, MIPI_TX_HS_TP_WHOLE_FRAME_COLOR1, MIPI_TX_HS_TP_V_STRIPES, MIPI_TX_HS_TP_H_STRIPES, }; enum dphy_mode { MIPI_DPHY_SLAVE = 0, MIPI_DPHY_MASTER }; enum dphy_tx_fsm { DPHY_TX_POWERDWN = 0, DPHY_TX_BGPON, DPHY_TX_TERMCAL, DPHY_TX_TERMCALUP, DPHY_TX_OFFSETCAL, DPHY_TX_LOCK, DPHY_TX_SRCAL, DPHY_TX_IDLE, DPHY_TX_ULP, DPHY_TX_LANESTART, DPHY_TX_CLKALIGN, DPHY_TX_DDLTUNNING, DPHY_TX_ULP_FORCE_PLL, DPHY_TX_LOCK_LOSS }; struct mipi_data_type_params { u8 size_constraint_pixels; u8 size_constraint_bytes; u8 pixels_per_pclk; u8 bits_per_pclk; }; struct mipi_tx_dsi_cfg { u8 hfp_blank_en; /* Horizontal front porch blanking enable */ u8 eotp_en; /* End of transmission packet enable */ /* Last vertical front porch blanking mode */ u8 lpm_last_vfp_line; /* First vertical sync active blanking mode */ u8 lpm_first_vsa_line; u8 sync_pulse_eventn; /* Sync type */ u8 hfp_blanking; /* Horizontal front porch blanking mode */ u8 hbp_blanking; /* Horizontal back porch blanking mode */ u8 hsa_blanking; /* Horizontal sync active blanking mode */ u8 v_blanking; /* Vertical timing blanking mode */ }; struct mipi_tx_frame_section_cfg { u32 dma_v_stride; u16 dma_v_scale_cfg; u16 width_pixels; u16 height_lines; u8 dma_packed; u8 bpp; u8 bpp_unpacked; u8 dma_h_stride; u8 data_type; u8 data_mode; u8 dma_flip_rotate_sel; }; struct mipi_tx_frame_timing_cfg { u32 bpp; u32 lane_rate_mbps; u32 hsync_width; u32 h_backporch; u32 h_frontporch; u32 h_active; u16 vsync_width; u16 v_backporch; u16 v_frontporch; u16 v_active; u8 active_lanes; }; struct mipi_tx_frame_sect_phcfg { u32 wc; enum mipi_data_mode data_mode; enum mipi_dsi_data_type data_type; u8 vchannel; u8 dma_packed; }; struct mipi_tx_frame_cfg { struct mipi_tx_frame_section_cfg *sections[MIPI_TX_FRAME_GEN_SECTIONS]; u32 hsync_width; /* in pixels */ u32 h_backporch; /* in pixels */ u32 h_frontporch; /* in pixels */ u16 vsync_width; /* in lines */ u16 v_backporch; /* in lines */ u16 v_frontporch; /* in lines */ }; struct mipi_tx_ctrl_cfg { struct mipi_tx_frame_cfg *frames[MIPI_TX_FRAME_GEN]; const struct mipi_tx_dsi_cfg *tx_dsi_cfg; u8 line_sync_pkt_en; u8 line_counter_active; u8 frame_counter_active; u8 tx_hsclkkidle_cnt; u8 tx_hsexit_cnt; u8 tx_crc_en; u8 tx_hact_wait_stop; u8 tx_always_use_hact; u8 tx_wait_trig; u8 tx_wait_all_sect; }; /* configuration structure for MIPI control */ struct mipi_ctrl_cfg { u8 active_lanes; /* # active lanes per controller 2/4 */ u32 lane_rate_mbps; /* MBPS */ u32 ref_clk_khz; u32 cfg_clk_khz; struct mipi_tx_ctrl_cfg tx_ctrl_cfg; }; static inline void kmb_write_mipi(struct kmb_dsi *kmb_dsi, unsigned int reg, u32 value) { writel(value, (kmb_dsi->mipi_mmio + reg)); } static inline u32 kmb_read_mipi(struct kmb_dsi *kmb_dsi, unsigned int reg) { return readl(kmb_dsi->mipi_mmio + reg); } static inline void kmb_write_bits_mipi(struct kmb_dsi *kmb_dsi, unsigned int reg, u32 offset, u32 num_bits, u32 value) { u32 reg_val = kmb_read_mipi(kmb_dsi, reg); u32 mask = (1 << num_bits) - 1; value &= mask; mask <<= offset; reg_val &= (~mask); reg_val |= (value << offset); kmb_write_mipi(kmb_dsi, reg, reg_val); } static inline void kmb_set_bit_mipi(struct kmb_dsi *kmb_dsi, unsigned int reg, u32 offset) { u32 reg_val = kmb_read_mipi(kmb_dsi, reg); kmb_write_mipi(kmb_dsi, reg, reg_val | (1 << offset)); } static inline void kmb_clr_bit_mipi(struct kmb_dsi *kmb_dsi, unsigned int reg, u32 offset) { u32 reg_val = kmb_read_mipi(kmb_dsi, reg); kmb_write_mipi(kmb_dsi, reg, reg_val & (~(1 << offset))); } int kmb_dsi_host_bridge_init(struct device *dev); struct kmb_dsi *kmb_dsi_init(struct platform_device *pdev); void kmb_dsi_host_unregister(struct kmb_dsi *kmb_dsi); int kmb_dsi_mode_set(struct kmb_dsi *kmb_dsi, struct drm_display_mode *mode, int sys_clk_mhz); int kmb_dsi_map_mmio(struct kmb_dsi *kmb_dsi); int kmb_dsi_clk_init(struct kmb_dsi *kmb_dsi); int kmb_dsi_encoder_init(struct drm_device *dev, struct kmb_dsi *kmb_dsi); #endif /* __KMB_DSI_H__ */