Maxime Ripard | 9c56810 | 2017-05-27 18:09:35 +0200 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (C) 2016 Maxime Ripard |
| 3 | * |
| 4 | * Maxime Ripard <maxime.ripard@free-electrons.com> |
| 5 | * |
| 6 | * This program is free software; you can redistribute it and/or |
| 7 | * modify it under the terms of the GNU General Public License as |
| 8 | * published by the Free Software Foundation; either version 2 of |
| 9 | * the License, or (at your option) any later version. |
| 10 | */ |
| 11 | |
| 12 | #ifndef _SUN4I_HDMI_H_ |
| 13 | #define _SUN4I_HDMI_H_ |
| 14 | |
| 15 | #include <drm/drm_connector.h> |
| 16 | #include <drm/drm_encoder.h> |
Chen-Yu Tsai | 939d749 | 2017-10-10 11:20:04 +0800 | [diff] [blame] | 17 | #include <linux/regmap.h> |
Maxime Ripard | 9c56810 | 2017-05-27 18:09:35 +0200 | [diff] [blame] | 18 | |
Hans Verkuil | 4ba72fc | 2017-09-21 22:34:54 +0200 | [diff] [blame] | 19 | #include <media/cec-pin.h> |
Hans Verkuil | 998140d | 2017-07-11 08:30:44 +0200 | [diff] [blame] | 20 | |
Maxime Ripard | 9c56810 | 2017-05-27 18:09:35 +0200 | [diff] [blame] | 21 | #define SUN4I_HDMI_CTRL_REG 0x004 |
| 22 | #define SUN4I_HDMI_CTRL_ENABLE BIT(31) |
| 23 | |
| 24 | #define SUN4I_HDMI_IRQ_REG 0x008 |
| 25 | #define SUN4I_HDMI_IRQ_STA_MASK 0x73 |
| 26 | #define SUN4I_HDMI_IRQ_STA_FIFO_OF BIT(1) |
| 27 | #define SUN4I_HDMI_IRQ_STA_FIFO_UF BIT(0) |
| 28 | |
| 29 | #define SUN4I_HDMI_HPD_REG 0x00c |
| 30 | #define SUN4I_HDMI_HPD_HIGH BIT(0) |
| 31 | |
| 32 | #define SUN4I_HDMI_VID_CTRL_REG 0x010 |
| 33 | #define SUN4I_HDMI_VID_CTRL_ENABLE BIT(31) |
| 34 | #define SUN4I_HDMI_VID_CTRL_HDMI_MODE BIT(30) |
| 35 | |
| 36 | #define SUN4I_HDMI_VID_TIMING_ACT_REG 0x014 |
| 37 | #define SUN4I_HDMI_VID_TIMING_BP_REG 0x018 |
| 38 | #define SUN4I_HDMI_VID_TIMING_FP_REG 0x01c |
| 39 | #define SUN4I_HDMI_VID_TIMING_SPW_REG 0x020 |
| 40 | |
| 41 | #define SUN4I_HDMI_VID_TIMING_X(x) ((((x) - 1) & GENMASK(11, 0))) |
| 42 | #define SUN4I_HDMI_VID_TIMING_Y(y) ((((y) - 1) & GENMASK(11, 0)) << 16) |
| 43 | |
| 44 | #define SUN4I_HDMI_VID_TIMING_POL_REG 0x024 |
| 45 | #define SUN4I_HDMI_VID_TIMING_POL_TX_CLK (0x3e0 << 16) |
| 46 | #define SUN4I_HDMI_VID_TIMING_POL_VSYNC BIT(1) |
| 47 | #define SUN4I_HDMI_VID_TIMING_POL_HSYNC BIT(0) |
| 48 | |
| 49 | #define SUN4I_HDMI_AVI_INFOFRAME_REG(n) (0x080 + (n)) |
| 50 | |
| 51 | #define SUN4I_HDMI_PAD_CTRL0_REG 0x200 |
| 52 | #define SUN4I_HDMI_PAD_CTRL0_BIASEN BIT(31) |
| 53 | #define SUN4I_HDMI_PAD_CTRL0_LDOCEN BIT(30) |
| 54 | #define SUN4I_HDMI_PAD_CTRL0_LDODEN BIT(29) |
| 55 | #define SUN4I_HDMI_PAD_CTRL0_PWENC BIT(28) |
| 56 | #define SUN4I_HDMI_PAD_CTRL0_PWEND BIT(27) |
| 57 | #define SUN4I_HDMI_PAD_CTRL0_PWENG BIT(26) |
| 58 | #define SUN4I_HDMI_PAD_CTRL0_CKEN BIT(25) |
| 59 | #define SUN4I_HDMI_PAD_CTRL0_TXEN BIT(23) |
| 60 | |
| 61 | #define SUN4I_HDMI_PAD_CTRL1_REG 0x204 |
| 62 | #define SUN4I_HDMI_PAD_CTRL1_AMP_OPT BIT(23) |
| 63 | #define SUN4I_HDMI_PAD_CTRL1_AMPCK_OPT BIT(22) |
| 64 | #define SUN4I_HDMI_PAD_CTRL1_EMP_OPT BIT(20) |
| 65 | #define SUN4I_HDMI_PAD_CTRL1_EMPCK_OPT BIT(19) |
| 66 | #define SUN4I_HDMI_PAD_CTRL1_REG_DEN BIT(15) |
| 67 | #define SUN4I_HDMI_PAD_CTRL1_REG_DENCK BIT(14) |
| 68 | #define SUN4I_HDMI_PAD_CTRL1_REG_EMP(n) (((n) & 7) << 10) |
| 69 | #define SUN4I_HDMI_PAD_CTRL1_HALVE_CLK BIT(6) |
| 70 | #define SUN4I_HDMI_PAD_CTRL1_REG_AMP(n) (((n) & 7) << 3) |
| 71 | |
| 72 | #define SUN4I_HDMI_PLL_CTRL_REG 0x208 |
| 73 | #define SUN4I_HDMI_PLL_CTRL_PLL_EN BIT(31) |
| 74 | #define SUN4I_HDMI_PLL_CTRL_BWS BIT(30) |
| 75 | #define SUN4I_HDMI_PLL_CTRL_HV_IS_33 BIT(29) |
| 76 | #define SUN4I_HDMI_PLL_CTRL_LDO1_EN BIT(28) |
| 77 | #define SUN4I_HDMI_PLL_CTRL_LDO2_EN BIT(27) |
| 78 | #define SUN4I_HDMI_PLL_CTRL_SDIV2 BIT(25) |
| 79 | #define SUN4I_HDMI_PLL_CTRL_VCO_GAIN(n) (((n) & 7) << 20) |
| 80 | #define SUN4I_HDMI_PLL_CTRL_S(n) (((n) & 7) << 17) |
| 81 | #define SUN4I_HDMI_PLL_CTRL_CP_S(n) (((n) & 0x1f) << 12) |
| 82 | #define SUN4I_HDMI_PLL_CTRL_CS(n) (((n) & 0xf) << 8) |
| 83 | #define SUN4I_HDMI_PLL_CTRL_DIV(n) (((n) & 0xf) << 4) |
| 84 | #define SUN4I_HDMI_PLL_CTRL_DIV_MASK GENMASK(7, 4) |
| 85 | #define SUN4I_HDMI_PLL_CTRL_VCO_S(n) ((n) & 0xf) |
| 86 | |
| 87 | #define SUN4I_HDMI_PLL_DBG0_REG 0x20c |
| 88 | #define SUN4I_HDMI_PLL_DBG0_TMDS_PARENT(n) (((n) & 1) << 21) |
| 89 | #define SUN4I_HDMI_PLL_DBG0_TMDS_PARENT_MASK BIT(21) |
| 90 | #define SUN4I_HDMI_PLL_DBG0_TMDS_PARENT_SHIFT 21 |
| 91 | |
Hans Verkuil | 998140d | 2017-07-11 08:30:44 +0200 | [diff] [blame] | 92 | #define SUN4I_HDMI_CEC 0x214 |
| 93 | #define SUN4I_HDMI_CEC_ENABLE BIT(11) |
| 94 | #define SUN4I_HDMI_CEC_TX BIT(9) |
| 95 | #define SUN4I_HDMI_CEC_RX BIT(8) |
| 96 | |
Maxime Ripard | 9c56810 | 2017-05-27 18:09:35 +0200 | [diff] [blame] | 97 | #define SUN4I_HDMI_PKT_CTRL_REG(n) (0x2f0 + (4 * (n))) |
| 98 | #define SUN4I_HDMI_PKT_CTRL_TYPE(n, t) ((t) << (((n) % 4) * 4)) |
| 99 | |
| 100 | #define SUN4I_HDMI_UNKNOWN_REG 0x300 |
| 101 | #define SUN4I_HDMI_UNKNOWN_INPUT_SYNC BIT(27) |
| 102 | |
| 103 | #define SUN4I_HDMI_DDC_CTRL_REG 0x500 |
| 104 | #define SUN4I_HDMI_DDC_CTRL_ENABLE BIT(31) |
| 105 | #define SUN4I_HDMI_DDC_CTRL_START_CMD BIT(30) |
| 106 | #define SUN4I_HDMI_DDC_CTRL_FIFO_DIR_MASK BIT(8) |
Jonathan Liu | f0a3dd3 | 2017-07-02 17:27:10 +1000 | [diff] [blame] | 107 | #define SUN4I_HDMI_DDC_CTRL_FIFO_DIR_WRITE (1 << 8) |
Maxime Ripard | 9c56810 | 2017-05-27 18:09:35 +0200 | [diff] [blame] | 108 | #define SUN4I_HDMI_DDC_CTRL_FIFO_DIR_READ (0 << 8) |
| 109 | #define SUN4I_HDMI_DDC_CTRL_RESET BIT(0) |
| 110 | |
| 111 | #define SUN4I_HDMI_DDC_ADDR_REG 0x504 |
| 112 | #define SUN4I_HDMI_DDC_ADDR_SEGMENT(seg) (((seg) & 0xff) << 24) |
| 113 | #define SUN4I_HDMI_DDC_ADDR_EDDC(addr) (((addr) & 0xff) << 16) |
| 114 | #define SUN4I_HDMI_DDC_ADDR_OFFSET(off) (((off) & 0xff) << 8) |
| 115 | #define SUN4I_HDMI_DDC_ADDR_SLAVE(addr) ((addr) & 0xff) |
| 116 | |
Jonathan Liu | f0a3dd3 | 2017-07-02 17:27:10 +1000 | [diff] [blame] | 117 | #define SUN4I_HDMI_DDC_INT_STATUS_REG 0x50c |
| 118 | #define SUN4I_HDMI_DDC_INT_STATUS_ILLEGAL_FIFO_OPERATION BIT(7) |
| 119 | #define SUN4I_HDMI_DDC_INT_STATUS_DDC_RX_FIFO_UNDERFLOW BIT(6) |
| 120 | #define SUN4I_HDMI_DDC_INT_STATUS_DDC_TX_FIFO_OVERFLOW BIT(5) |
| 121 | #define SUN4I_HDMI_DDC_INT_STATUS_FIFO_REQUEST BIT(4) |
| 122 | #define SUN4I_HDMI_DDC_INT_STATUS_ARBITRATION_ERROR BIT(3) |
| 123 | #define SUN4I_HDMI_DDC_INT_STATUS_ACK_ERROR BIT(2) |
| 124 | #define SUN4I_HDMI_DDC_INT_STATUS_BUS_ERROR BIT(1) |
| 125 | #define SUN4I_HDMI_DDC_INT_STATUS_TRANSFER_COMPLETE BIT(0) |
| 126 | |
Maxime Ripard | 9c56810 | 2017-05-27 18:09:35 +0200 | [diff] [blame] | 127 | #define SUN4I_HDMI_DDC_FIFO_CTRL_REG 0x510 |
| 128 | #define SUN4I_HDMI_DDC_FIFO_CTRL_CLEAR BIT(31) |
Jonathan Liu | f0a3dd3 | 2017-07-02 17:27:10 +1000 | [diff] [blame] | 129 | #define SUN4I_HDMI_DDC_FIFO_CTRL_RX_THRES(n) (((n) & 0xf) << 4) |
| 130 | #define SUN4I_HDMI_DDC_FIFO_CTRL_RX_THRES_MASK GENMASK(7, 4) |
| 131 | #define SUN4I_HDMI_DDC_FIFO_CTRL_RX_THRES_MAX (BIT(4) - 1) |
| 132 | #define SUN4I_HDMI_DDC_FIFO_CTRL_TX_THRES(n) ((n) & 0xf) |
| 133 | #define SUN4I_HDMI_DDC_FIFO_CTRL_TX_THRES_MASK GENMASK(3, 0) |
| 134 | #define SUN4I_HDMI_DDC_FIFO_CTRL_TX_THRES_MAX (BIT(4) - 1) |
Maxime Ripard | 9c56810 | 2017-05-27 18:09:35 +0200 | [diff] [blame] | 135 | |
| 136 | #define SUN4I_HDMI_DDC_FIFO_DATA_REG 0x518 |
Jonathan Liu | f0a3dd3 | 2017-07-02 17:27:10 +1000 | [diff] [blame] | 137 | |
Maxime Ripard | 9c56810 | 2017-05-27 18:09:35 +0200 | [diff] [blame] | 138 | #define SUN4I_HDMI_DDC_BYTE_COUNT_REG 0x51c |
Jonathan Liu | f0a3dd3 | 2017-07-02 17:27:10 +1000 | [diff] [blame] | 139 | #define SUN4I_HDMI_DDC_BYTE_COUNT_MAX (BIT(10) - 1) |
Maxime Ripard | 9c56810 | 2017-05-27 18:09:35 +0200 | [diff] [blame] | 140 | |
| 141 | #define SUN4I_HDMI_DDC_CMD_REG 0x520 |
| 142 | #define SUN4I_HDMI_DDC_CMD_EXPLICIT_EDDC_READ 6 |
Jonathan Liu | f0a3dd3 | 2017-07-02 17:27:10 +1000 | [diff] [blame] | 143 | #define SUN4I_HDMI_DDC_CMD_IMPLICIT_READ 5 |
| 144 | #define SUN4I_HDMI_DDC_CMD_IMPLICIT_WRITE 3 |
Maxime Ripard | 9c56810 | 2017-05-27 18:09:35 +0200 | [diff] [blame] | 145 | |
| 146 | #define SUN4I_HDMI_DDC_CLK_REG 0x528 |
| 147 | #define SUN4I_HDMI_DDC_CLK_M(m) (((m) & 0x7) << 3) |
| 148 | #define SUN4I_HDMI_DDC_CLK_N(n) ((n) & 0x7) |
| 149 | |
| 150 | #define SUN4I_HDMI_DDC_LINE_CTRL_REG 0x540 |
| 151 | #define SUN4I_HDMI_DDC_LINE_CTRL_SDA_ENABLE BIT(9) |
| 152 | #define SUN4I_HDMI_DDC_LINE_CTRL_SCL_ENABLE BIT(8) |
| 153 | |
| 154 | #define SUN4I_HDMI_DDC_FIFO_SIZE 16 |
| 155 | |
Chen-Yu Tsai | c4a9aec | 2017-10-10 11:20:05 +0800 | [diff] [blame^] | 156 | /* A31 specific */ |
| 157 | #define SUN6I_HDMI_DDC_CTRL_REG 0x500 |
| 158 | #define SUN6I_HDMI_DDC_CTRL_RESET BIT(31) |
| 159 | #define SUN6I_HDMI_DDC_CTRL_START_CMD BIT(27) |
| 160 | #define SUN6I_HDMI_DDC_CTRL_SDA_ENABLE BIT(6) |
| 161 | #define SUN6I_HDMI_DDC_CTRL_SCL_ENABLE BIT(4) |
| 162 | #define SUN6I_HDMI_DDC_CTRL_ENABLE BIT(0) |
| 163 | |
| 164 | #define SUN6I_HDMI_DDC_CMD_REG 0x508 |
| 165 | #define SUN6I_HDMI_DDC_CMD_BYTE_COUNT(count) ((count) << 16) |
| 166 | /* command types in lower 3 bits are the same as sun4i */ |
| 167 | |
| 168 | #define SUN6I_HDMI_DDC_ADDR_REG 0x50c |
| 169 | #define SUN6I_HDMI_DDC_ADDR_SEGMENT(seg) (((seg) & 0xff) << 24) |
| 170 | #define SUN6I_HDMI_DDC_ADDR_EDDC(addr) (((addr) & 0xff) << 16) |
| 171 | #define SUN6I_HDMI_DDC_ADDR_OFFSET(off) (((off) & 0xff) << 8) |
| 172 | #define SUN6I_HDMI_DDC_ADDR_SLAVE(addr) (((addr) & 0xff) << 1) |
| 173 | |
| 174 | #define SUN6I_HDMI_DDC_INT_STATUS_REG 0x514 |
| 175 | #define SUN6I_HDMI_DDC_INT_STATUS_TIMEOUT BIT(8) |
| 176 | /* lower 8 bits are the same as sun4i */ |
| 177 | |
| 178 | #define SUN6I_HDMI_DDC_FIFO_CTRL_REG 0x518 |
| 179 | #define SUN6I_HDMI_DDC_FIFO_CTRL_CLEAR BIT(15) |
| 180 | /* lower 9 bits are the same as sun4i */ |
| 181 | |
| 182 | #define SUN6I_HDMI_DDC_CLK_REG 0x520 |
| 183 | /* DDC CLK bit fields are the same, but the formula is not */ |
| 184 | |
| 185 | #define SUN6I_HDMI_DDC_FIFO_DATA_REG 0x580 |
| 186 | |
Maxime Ripard | 9c56810 | 2017-05-27 18:09:35 +0200 | [diff] [blame] | 187 | enum sun4i_hdmi_pkt_type { |
| 188 | SUN4I_HDMI_PKT_AVI = 2, |
| 189 | SUN4I_HDMI_PKT_END = 15, |
| 190 | }; |
| 191 | |
Chen-Yu Tsai | 939d749 | 2017-10-10 11:20:04 +0800 | [diff] [blame] | 192 | struct sun4i_hdmi_variant { |
| 193 | bool has_ddc_parent_clk; |
| 194 | bool has_reset_control; |
| 195 | |
| 196 | u32 pad_ctrl0_init_val; |
| 197 | u32 pad_ctrl1_init_val; |
| 198 | u32 pll_ctrl_init_val; |
| 199 | |
| 200 | struct reg_field ddc_clk_reg; |
| 201 | u8 ddc_clk_pre_divider; |
| 202 | u8 ddc_clk_m_offset; |
| 203 | |
| 204 | u8 tmds_clk_div_offset; |
| 205 | |
| 206 | /* Register fields for I2C adapter */ |
| 207 | struct reg_field field_ddc_en; |
| 208 | struct reg_field field_ddc_start; |
| 209 | struct reg_field field_ddc_reset; |
| 210 | struct reg_field field_ddc_addr_reg; |
| 211 | struct reg_field field_ddc_slave_addr; |
| 212 | struct reg_field field_ddc_int_mask; |
| 213 | struct reg_field field_ddc_int_status; |
| 214 | struct reg_field field_ddc_fifo_clear; |
| 215 | struct reg_field field_ddc_fifo_rx_thres; |
| 216 | struct reg_field field_ddc_fifo_tx_thres; |
| 217 | struct reg_field field_ddc_byte_count; |
| 218 | struct reg_field field_ddc_cmd; |
| 219 | struct reg_field field_ddc_sda_en; |
| 220 | struct reg_field field_ddc_sck_en; |
| 221 | |
| 222 | /* DDC FIFO register offset */ |
| 223 | u32 ddc_fifo_reg; |
| 224 | |
| 225 | /* |
| 226 | * DDC FIFO threshold boundary conditions |
| 227 | * |
| 228 | * This is used to cope with the threshold boundary condition |
| 229 | * being slightly different on sun5i and sun6i. |
| 230 | * |
| 231 | * On sun5i the threshold is exclusive, i.e. does not include, |
| 232 | * the value of the threshold. ( > for RX; < for TX ) |
| 233 | * On sun6i the threshold is inclusive, i.e. includes, the |
| 234 | * value of the threshold. ( >= for RX; <= for TX ) |
| 235 | */ |
| 236 | bool ddc_fifo_thres_incl; |
| 237 | |
| 238 | bool ddc_fifo_has_dir; |
| 239 | }; |
| 240 | |
Maxime Ripard | 9c56810 | 2017-05-27 18:09:35 +0200 | [diff] [blame] | 241 | struct sun4i_hdmi { |
| 242 | struct drm_connector connector; |
| 243 | struct drm_encoder encoder; |
| 244 | struct device *dev; |
| 245 | |
| 246 | void __iomem *base; |
Chen-Yu Tsai | 4b1c924 | 2017-10-10 11:20:01 +0800 | [diff] [blame] | 247 | struct regmap *regmap; |
Maxime Ripard | 9c56810 | 2017-05-27 18:09:35 +0200 | [diff] [blame] | 248 | |
Chen-Yu Tsai | 939d749 | 2017-10-10 11:20:04 +0800 | [diff] [blame] | 249 | /* Reset control */ |
| 250 | struct reset_control *reset; |
| 251 | |
Maxime Ripard | 9c56810 | 2017-05-27 18:09:35 +0200 | [diff] [blame] | 252 | /* Parent clocks */ |
| 253 | struct clk *bus_clk; |
| 254 | struct clk *mod_clk; |
Chen-Yu Tsai | 939d749 | 2017-10-10 11:20:04 +0800 | [diff] [blame] | 255 | struct clk *ddc_parent_clk; |
Maxime Ripard | 9c56810 | 2017-05-27 18:09:35 +0200 | [diff] [blame] | 256 | struct clk *pll0_clk; |
| 257 | struct clk *pll1_clk; |
| 258 | |
| 259 | /* And the clocks we create */ |
| 260 | struct clk *ddc_clk; |
| 261 | struct clk *tmds_clk; |
| 262 | |
Jonathan Liu | f0a3dd3 | 2017-07-02 17:27:10 +1000 | [diff] [blame] | 263 | struct i2c_adapter *i2c; |
| 264 | |
Chen-Yu Tsai | 939d749 | 2017-10-10 11:20:04 +0800 | [diff] [blame] | 265 | /* Regmap fields for I2C adapter */ |
| 266 | struct regmap_field *field_ddc_en; |
| 267 | struct regmap_field *field_ddc_start; |
| 268 | struct regmap_field *field_ddc_reset; |
| 269 | struct regmap_field *field_ddc_addr_reg; |
| 270 | struct regmap_field *field_ddc_slave_addr; |
| 271 | struct regmap_field *field_ddc_int_mask; |
| 272 | struct regmap_field *field_ddc_int_status; |
| 273 | struct regmap_field *field_ddc_fifo_clear; |
| 274 | struct regmap_field *field_ddc_fifo_rx_thres; |
| 275 | struct regmap_field *field_ddc_fifo_tx_thres; |
| 276 | struct regmap_field *field_ddc_byte_count; |
| 277 | struct regmap_field *field_ddc_cmd; |
| 278 | struct regmap_field *field_ddc_sda_en; |
| 279 | struct regmap_field *field_ddc_sck_en; |
| 280 | |
Maxime Ripard | 9c56810 | 2017-05-27 18:09:35 +0200 | [diff] [blame] | 281 | struct sun4i_drv *drv; |
| 282 | |
| 283 | bool hdmi_monitor; |
Hans Verkuil | 998140d | 2017-07-11 08:30:44 +0200 | [diff] [blame] | 284 | struct cec_adapter *cec_adap; |
Chen-Yu Tsai | 939d749 | 2017-10-10 11:20:04 +0800 | [diff] [blame] | 285 | |
| 286 | const struct sun4i_hdmi_variant *variant; |
Maxime Ripard | 9c56810 | 2017-05-27 18:09:35 +0200 | [diff] [blame] | 287 | }; |
| 288 | |
| 289 | int sun4i_ddc_create(struct sun4i_hdmi *hdmi, struct clk *clk); |
| 290 | int sun4i_tmds_create(struct sun4i_hdmi *hdmi); |
Jonathan Liu | f0a3dd3 | 2017-07-02 17:27:10 +1000 | [diff] [blame] | 291 | int sun4i_hdmi_i2c_create(struct device *dev, struct sun4i_hdmi *hdmi); |
Maxime Ripard | 9c56810 | 2017-05-27 18:09:35 +0200 | [diff] [blame] | 292 | |
| 293 | #endif /* _SUN4I_HDMI_H_ */ |