]> Git Repo - J-linux.git/blob - drivers/gpu/drm/msm/dp/dp_catalog.c
Merge tag 'vfs-6.13-rc7.fixes' of git://git.kernel.org/pub/scm/linux/kernel/git/vfs/vfs
[J-linux.git] / drivers / gpu / drm / msm / dp / dp_catalog.c
1 // SPDX-License-Identifier: GPL-2.0-only
2 /*
3  * Copyright (c) 2017-2020, The Linux Foundation. All rights reserved.
4  */
5
6 #define pr_fmt(fmt)     "[drm-dp] %s: " fmt, __func__
7
8 #include <linux/delay.h>
9 #include <linux/iopoll.h>
10 #include <linux/platform_device.h>
11 #include <linux/rational.h>
12 #include <drm/display/drm_dp_helper.h>
13 #include <drm/drm_print.h>
14
15 #include "dp_catalog.h"
16 #include "dp_reg.h"
17
18 #define POLLING_SLEEP_US                        1000
19 #define POLLING_TIMEOUT_US                      10000
20
21 #define SCRAMBLER_RESET_COUNT_VALUE             0xFC
22
23 #define DP_INTERRUPT_STATUS_ACK_SHIFT   1
24 #define DP_INTERRUPT_STATUS_MASK_SHIFT  2
25
26 #define DP_INTF_CONFIG_DATABUS_WIDEN     BIT(4)
27
28 #define DP_INTERRUPT_STATUS1 \
29         (DP_INTR_AUX_XFER_DONE| \
30         DP_INTR_WRONG_ADDR | DP_INTR_TIMEOUT | \
31         DP_INTR_NACK_DEFER | DP_INTR_WRONG_DATA_CNT | \
32         DP_INTR_I2C_NACK | DP_INTR_I2C_DEFER | \
33         DP_INTR_PLL_UNLOCKED | DP_INTR_AUX_ERROR)
34
35 #define DP_INTERRUPT_STATUS1_ACK \
36         (DP_INTERRUPT_STATUS1 << DP_INTERRUPT_STATUS_ACK_SHIFT)
37 #define DP_INTERRUPT_STATUS1_MASK \
38         (DP_INTERRUPT_STATUS1 << DP_INTERRUPT_STATUS_MASK_SHIFT)
39
40 #define DP_INTERRUPT_STATUS2 \
41         (DP_INTR_READY_FOR_VIDEO | DP_INTR_IDLE_PATTERN_SENT | \
42         DP_INTR_FRAME_END | DP_INTR_CRC_UPDATED)
43
44 #define DP_INTERRUPT_STATUS2_ACK \
45         (DP_INTERRUPT_STATUS2 << DP_INTERRUPT_STATUS_ACK_SHIFT)
46 #define DP_INTERRUPT_STATUS2_MASK \
47         (DP_INTERRUPT_STATUS2 << DP_INTERRUPT_STATUS_MASK_SHIFT)
48
49 #define DP_INTERRUPT_STATUS4 \
50         (PSR_UPDATE_INT | PSR_CAPTURE_INT | PSR_EXIT_INT | \
51         PSR_UPDATE_ERROR_INT | PSR_WAKE_ERROR_INT)
52
53 #define DP_INTERRUPT_MASK4 \
54         (PSR_UPDATE_MASK | PSR_CAPTURE_MASK | PSR_EXIT_MASK | \
55         PSR_UPDATE_ERROR_MASK | PSR_WAKE_ERROR_MASK)
56
57 #define DP_DEFAULT_AHB_OFFSET   0x0000
58 #define DP_DEFAULT_AHB_SIZE     0x0200
59 #define DP_DEFAULT_AUX_OFFSET   0x0200
60 #define DP_DEFAULT_AUX_SIZE     0x0200
61 #define DP_DEFAULT_LINK_OFFSET  0x0400
62 #define DP_DEFAULT_LINK_SIZE    0x0C00
63 #define DP_DEFAULT_P0_OFFSET    0x1000
64 #define DP_DEFAULT_P0_SIZE      0x0400
65
66 struct dss_io_region {
67         size_t len;
68         void __iomem *base;
69 };
70
71 struct dss_io_data {
72         struct dss_io_region ahb;
73         struct dss_io_region aux;
74         struct dss_io_region link;
75         struct dss_io_region p0;
76 };
77
78 struct msm_dp_catalog_private {
79         struct device *dev;
80         struct drm_device *drm_dev;
81         struct dss_io_data io;
82         u32 (*audio_map)[DP_AUDIO_SDP_HEADER_MAX];
83         struct msm_dp_catalog msm_dp_catalog;
84 };
85
86 void msm_dp_catalog_snapshot(struct msm_dp_catalog *msm_dp_catalog, struct msm_disp_state *disp_state)
87 {
88         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
89                         struct msm_dp_catalog_private, msm_dp_catalog);
90         struct dss_io_data *dss = &catalog->io;
91
92         msm_disp_snapshot_add_block(disp_state, dss->ahb.len, dss->ahb.base, "dp_ahb");
93         msm_disp_snapshot_add_block(disp_state, dss->aux.len, dss->aux.base, "dp_aux");
94         msm_disp_snapshot_add_block(disp_state, dss->link.len, dss->link.base, "dp_link");
95         msm_disp_snapshot_add_block(disp_state, dss->p0.len, dss->p0.base, "dp_p0");
96 }
97
98 static inline u32 msm_dp_read_aux(struct msm_dp_catalog_private *catalog, u32 offset)
99 {
100         return readl_relaxed(catalog->io.aux.base + offset);
101 }
102
103 static inline void msm_dp_write_aux(struct msm_dp_catalog_private *catalog,
104                                u32 offset, u32 data)
105 {
106         /*
107          * To make sure aux reg writes happens before any other operation,
108          * this function uses writel() instread of writel_relaxed()
109          */
110         writel(data, catalog->io.aux.base + offset);
111 }
112
113 static inline u32 msm_dp_read_ahb(const struct msm_dp_catalog_private *catalog, u32 offset)
114 {
115         return readl_relaxed(catalog->io.ahb.base + offset);
116 }
117
118 static inline void msm_dp_write_ahb(struct msm_dp_catalog_private *catalog,
119                                u32 offset, u32 data)
120 {
121         /*
122          * To make sure phy reg writes happens before any other operation,
123          * this function uses writel() instread of writel_relaxed()
124          */
125         writel(data, catalog->io.ahb.base + offset);
126 }
127
128 static inline void msm_dp_write_p0(struct msm_dp_catalog_private *catalog,
129                                u32 offset, u32 data)
130 {
131         /*
132          * To make sure interface reg writes happens before any other operation,
133          * this function uses writel() instread of writel_relaxed()
134          */
135         writel(data, catalog->io.p0.base + offset);
136 }
137
138 static inline u32 msm_dp_read_p0(struct msm_dp_catalog_private *catalog,
139                                u32 offset)
140 {
141         /*
142          * To make sure interface reg writes happens before any other operation,
143          * this function uses writel() instread of writel_relaxed()
144          */
145         return readl_relaxed(catalog->io.p0.base + offset);
146 }
147
148 static inline u32 msm_dp_read_link(struct msm_dp_catalog_private *catalog, u32 offset)
149 {
150         return readl_relaxed(catalog->io.link.base + offset);
151 }
152
153 static inline void msm_dp_write_link(struct msm_dp_catalog_private *catalog,
154                                u32 offset, u32 data)
155 {
156         /*
157          * To make sure link reg writes happens before any other operation,
158          * this function uses writel() instread of writel_relaxed()
159          */
160         writel(data, catalog->io.link.base + offset);
161 }
162
163 /* aux related catalog functions */
164 u32 msm_dp_catalog_aux_read_data(struct msm_dp_catalog *msm_dp_catalog)
165 {
166         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
167                                 struct msm_dp_catalog_private, msm_dp_catalog);
168
169         return msm_dp_read_aux(catalog, REG_DP_AUX_DATA);
170 }
171
172 int msm_dp_catalog_aux_write_data(struct msm_dp_catalog *msm_dp_catalog, u32 data)
173 {
174         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
175                                 struct msm_dp_catalog_private, msm_dp_catalog);
176
177         msm_dp_write_aux(catalog, REG_DP_AUX_DATA, data);
178         return 0;
179 }
180
181 int msm_dp_catalog_aux_write_trans(struct msm_dp_catalog *msm_dp_catalog, u32 data)
182 {
183         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
184                                 struct msm_dp_catalog_private, msm_dp_catalog);
185
186         msm_dp_write_aux(catalog, REG_DP_AUX_TRANS_CTRL, data);
187         return 0;
188 }
189
190 int msm_dp_catalog_aux_clear_trans(struct msm_dp_catalog *msm_dp_catalog, bool read)
191 {
192         u32 data;
193         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
194                                 struct msm_dp_catalog_private, msm_dp_catalog);
195
196         if (read) {
197                 data = msm_dp_read_aux(catalog, REG_DP_AUX_TRANS_CTRL);
198                 data &= ~DP_AUX_TRANS_CTRL_GO;
199                 msm_dp_write_aux(catalog, REG_DP_AUX_TRANS_CTRL, data);
200         } else {
201                 msm_dp_write_aux(catalog, REG_DP_AUX_TRANS_CTRL, 0);
202         }
203         return 0;
204 }
205
206 int msm_dp_catalog_aux_clear_hw_interrupts(struct msm_dp_catalog *msm_dp_catalog)
207 {
208         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
209                                 struct msm_dp_catalog_private, msm_dp_catalog);
210
211         msm_dp_read_aux(catalog, REG_DP_PHY_AUX_INTERRUPT_STATUS);
212         msm_dp_write_aux(catalog, REG_DP_PHY_AUX_INTERRUPT_CLEAR, 0x1f);
213         msm_dp_write_aux(catalog, REG_DP_PHY_AUX_INTERRUPT_CLEAR, 0x9f);
214         msm_dp_write_aux(catalog, REG_DP_PHY_AUX_INTERRUPT_CLEAR, 0);
215         return 0;
216 }
217
218 /**
219  * msm_dp_catalog_aux_reset() - reset AUX controller
220  *
221  * @msm_dp_catalog: DP catalog structure
222  *
223  * return: void
224  *
225  * This function reset AUX controller
226  *
227  * NOTE: reset AUX controller will also clear any pending HPD related interrupts
228  * 
229  */
230 void msm_dp_catalog_aux_reset(struct msm_dp_catalog *msm_dp_catalog)
231 {
232         u32 aux_ctrl;
233         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
234                                 struct msm_dp_catalog_private, msm_dp_catalog);
235
236         aux_ctrl = msm_dp_read_aux(catalog, REG_DP_AUX_CTRL);
237
238         aux_ctrl |= DP_AUX_CTRL_RESET;
239         msm_dp_write_aux(catalog, REG_DP_AUX_CTRL, aux_ctrl);
240         usleep_range(1000, 1100); /* h/w recommended delay */
241
242         aux_ctrl &= ~DP_AUX_CTRL_RESET;
243         msm_dp_write_aux(catalog, REG_DP_AUX_CTRL, aux_ctrl);
244 }
245
246 void msm_dp_catalog_aux_enable(struct msm_dp_catalog *msm_dp_catalog, bool enable)
247 {
248         u32 aux_ctrl;
249         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
250                                 struct msm_dp_catalog_private, msm_dp_catalog);
251
252         aux_ctrl = msm_dp_read_aux(catalog, REG_DP_AUX_CTRL);
253
254         if (enable) {
255                 msm_dp_write_aux(catalog, REG_DP_TIMEOUT_COUNT, 0xffff);
256                 msm_dp_write_aux(catalog, REG_DP_AUX_LIMITS, 0xffff);
257                 aux_ctrl |= DP_AUX_CTRL_ENABLE;
258         } else {
259                 aux_ctrl &= ~DP_AUX_CTRL_ENABLE;
260         }
261
262         msm_dp_write_aux(catalog, REG_DP_AUX_CTRL, aux_ctrl);
263 }
264
265 int msm_dp_catalog_aux_wait_for_hpd_connect_state(struct msm_dp_catalog *msm_dp_catalog,
266                                               unsigned long wait_us)
267 {
268         u32 state;
269         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
270                                 struct msm_dp_catalog_private, msm_dp_catalog);
271
272         /* poll for hpd connected status every 2ms and timeout after wait_us */
273         return readl_poll_timeout(catalog->io.aux.base +
274                                 REG_DP_DP_HPD_INT_STATUS,
275                                 state, state & DP_DP_HPD_STATE_STATUS_CONNECTED,
276                                 min(wait_us, 2000), wait_us);
277 }
278
279 static void dump_regs(void __iomem *base, int len)
280 {
281         int i;
282         u32 x0, x4, x8, xc;
283         u32 addr_off = 0;
284
285         len = DIV_ROUND_UP(len, 16);
286         for (i = 0; i < len; i++) {
287                 x0 = readl_relaxed(base + addr_off);
288                 x4 = readl_relaxed(base + addr_off + 0x04);
289                 x8 = readl_relaxed(base + addr_off + 0x08);
290                 xc = readl_relaxed(base + addr_off + 0x0c);
291
292                 pr_info("%08x: %08x %08x %08x %08x", addr_off, x0, x4, x8, xc);
293                 addr_off += 16;
294         }
295 }
296
297 void msm_dp_catalog_dump_regs(struct msm_dp_catalog *msm_dp_catalog)
298 {
299         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
300                 struct msm_dp_catalog_private, msm_dp_catalog);
301         struct dss_io_data *io = &catalog->io;
302
303         pr_info("AHB regs\n");
304         dump_regs(io->ahb.base, io->ahb.len);
305
306         pr_info("AUXCLK regs\n");
307         dump_regs(io->aux.base, io->aux.len);
308
309         pr_info("LCLK regs\n");
310         dump_regs(io->link.base, io->link.len);
311
312         pr_info("P0CLK regs\n");
313         dump_regs(io->p0.base, io->p0.len);
314 }
315
316 u32 msm_dp_catalog_aux_get_irq(struct msm_dp_catalog *msm_dp_catalog)
317 {
318         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
319                                 struct msm_dp_catalog_private, msm_dp_catalog);
320         u32 intr, intr_ack;
321
322         intr = msm_dp_read_ahb(catalog, REG_DP_INTR_STATUS);
323         intr &= ~DP_INTERRUPT_STATUS1_MASK;
324         intr_ack = (intr & DP_INTERRUPT_STATUS1)
325                         << DP_INTERRUPT_STATUS_ACK_SHIFT;
326         msm_dp_write_ahb(catalog, REG_DP_INTR_STATUS, intr_ack |
327                         DP_INTERRUPT_STATUS1_MASK);
328
329         return intr;
330
331 }
332
333 /* controller related catalog functions */
334 void msm_dp_catalog_ctrl_update_transfer_unit(struct msm_dp_catalog *msm_dp_catalog,
335                                 u32 msm_dp_tu, u32 valid_boundary,
336                                 u32 valid_boundary2)
337 {
338         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
339                                 struct msm_dp_catalog_private, msm_dp_catalog);
340
341         msm_dp_write_link(catalog, REG_DP_VALID_BOUNDARY, valid_boundary);
342         msm_dp_write_link(catalog, REG_DP_TU, msm_dp_tu);
343         msm_dp_write_link(catalog, REG_DP_VALID_BOUNDARY_2, valid_boundary2);
344 }
345
346 void msm_dp_catalog_ctrl_state_ctrl(struct msm_dp_catalog *msm_dp_catalog, u32 state)
347 {
348         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
349                                 struct msm_dp_catalog_private, msm_dp_catalog);
350
351         msm_dp_write_link(catalog, REG_DP_STATE_CTRL, state);
352 }
353
354 void msm_dp_catalog_ctrl_config_ctrl(struct msm_dp_catalog *msm_dp_catalog, u32 cfg)
355 {
356         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
357                                 struct msm_dp_catalog_private, msm_dp_catalog);
358
359         drm_dbg_dp(catalog->drm_dev, "DP_CONFIGURATION_CTRL=0x%x\n", cfg);
360
361         msm_dp_write_link(catalog, REG_DP_CONFIGURATION_CTRL, cfg);
362 }
363
364 void msm_dp_catalog_ctrl_lane_mapping(struct msm_dp_catalog *msm_dp_catalog)
365 {
366         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
367                                 struct msm_dp_catalog_private, msm_dp_catalog);
368         u32 ln_0 = 0, ln_1 = 1, ln_2 = 2, ln_3 = 3; /* One-to-One mapping */
369         u32 ln_mapping;
370
371         ln_mapping = ln_0 << LANE0_MAPPING_SHIFT;
372         ln_mapping |= ln_1 << LANE1_MAPPING_SHIFT;
373         ln_mapping |= ln_2 << LANE2_MAPPING_SHIFT;
374         ln_mapping |= ln_3 << LANE3_MAPPING_SHIFT;
375
376         msm_dp_write_link(catalog, REG_DP_LOGICAL2PHYSICAL_LANE_MAPPING,
377                         ln_mapping);
378 }
379
380 void msm_dp_catalog_ctrl_psr_mainlink_enable(struct msm_dp_catalog *msm_dp_catalog,
381                                                 bool enable)
382 {
383         u32 val;
384         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
385                                 struct msm_dp_catalog_private, msm_dp_catalog);
386
387         val = msm_dp_read_link(catalog, REG_DP_MAINLINK_CTRL);
388
389         if (enable)
390                 val |= DP_MAINLINK_CTRL_ENABLE;
391         else
392                 val &= ~DP_MAINLINK_CTRL_ENABLE;
393
394         msm_dp_write_link(catalog, REG_DP_MAINLINK_CTRL, val);
395 }
396
397 void msm_dp_catalog_ctrl_mainlink_ctrl(struct msm_dp_catalog *msm_dp_catalog,
398                                                 bool enable)
399 {
400         u32 mainlink_ctrl;
401         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
402                                 struct msm_dp_catalog_private, msm_dp_catalog);
403
404         drm_dbg_dp(catalog->drm_dev, "enable=%d\n", enable);
405         if (enable) {
406                 /*
407                  * To make sure link reg writes happens before other operation,
408                  * msm_dp_write_link() function uses writel()
409                  */
410                 mainlink_ctrl = msm_dp_read_link(catalog, REG_DP_MAINLINK_CTRL);
411
412                 mainlink_ctrl &= ~(DP_MAINLINK_CTRL_RESET |
413                                                 DP_MAINLINK_CTRL_ENABLE);
414                 msm_dp_write_link(catalog, REG_DP_MAINLINK_CTRL, mainlink_ctrl);
415
416                 mainlink_ctrl |= DP_MAINLINK_CTRL_RESET;
417                 msm_dp_write_link(catalog, REG_DP_MAINLINK_CTRL, mainlink_ctrl);
418
419                 mainlink_ctrl &= ~DP_MAINLINK_CTRL_RESET;
420                 msm_dp_write_link(catalog, REG_DP_MAINLINK_CTRL, mainlink_ctrl);
421
422                 mainlink_ctrl |= (DP_MAINLINK_CTRL_ENABLE |
423                                         DP_MAINLINK_FB_BOUNDARY_SEL);
424                 msm_dp_write_link(catalog, REG_DP_MAINLINK_CTRL, mainlink_ctrl);
425         } else {
426                 mainlink_ctrl = msm_dp_read_link(catalog, REG_DP_MAINLINK_CTRL);
427                 mainlink_ctrl &= ~DP_MAINLINK_CTRL_ENABLE;
428                 msm_dp_write_link(catalog, REG_DP_MAINLINK_CTRL, mainlink_ctrl);
429         }
430 }
431
432 void msm_dp_catalog_ctrl_config_misc(struct msm_dp_catalog *msm_dp_catalog,
433                                         u32 colorimetry_cfg,
434                                         u32 test_bits_depth)
435 {
436         u32 misc_val;
437         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
438                                 struct msm_dp_catalog_private, msm_dp_catalog);
439
440         misc_val = msm_dp_read_link(catalog, REG_DP_MISC1_MISC0);
441
442         /* clear bpp bits */
443         misc_val &= ~(0x07 << DP_MISC0_TEST_BITS_DEPTH_SHIFT);
444         misc_val |= colorimetry_cfg << DP_MISC0_COLORIMETRY_CFG_SHIFT;
445         misc_val |= test_bits_depth << DP_MISC0_TEST_BITS_DEPTH_SHIFT;
446         /* Configure clock to synchronous mode */
447         misc_val |= DP_MISC0_SYNCHRONOUS_CLK;
448
449         drm_dbg_dp(catalog->drm_dev, "misc settings = 0x%x\n", misc_val);
450         msm_dp_write_link(catalog, REG_DP_MISC1_MISC0, misc_val);
451 }
452
453 void msm_dp_catalog_setup_peripheral_flush(struct msm_dp_catalog *msm_dp_catalog)
454 {
455         u32 mainlink_ctrl, hw_revision;
456         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
457                                 struct msm_dp_catalog_private, msm_dp_catalog);
458
459         mainlink_ctrl = msm_dp_read_link(catalog, REG_DP_MAINLINK_CTRL);
460
461         hw_revision = msm_dp_catalog_hw_revision(msm_dp_catalog);
462         if (hw_revision >= DP_HW_VERSION_1_2)
463                 mainlink_ctrl |= DP_MAINLINK_FLUSH_MODE_SDE_PERIPH_UPDATE;
464         else
465                 mainlink_ctrl |= DP_MAINLINK_FLUSH_MODE_UPDATE_SDP;
466
467         msm_dp_write_link(catalog, REG_DP_MAINLINK_CTRL, mainlink_ctrl);
468 }
469
470 void msm_dp_catalog_ctrl_config_msa(struct msm_dp_catalog *msm_dp_catalog,
471                                         u32 rate, u32 stream_rate_khz,
472                                         bool is_ycbcr_420)
473 {
474         u32 pixel_m, pixel_n;
475         u32 mvid, nvid, pixel_div = 0, dispcc_input_rate;
476         u32 const nvid_fixed = DP_LINK_CONSTANT_N_VALUE;
477         u32 const link_rate_hbr2 = 540000;
478         u32 const link_rate_hbr3 = 810000;
479         unsigned long den, num;
480
481         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
482                                 struct msm_dp_catalog_private, msm_dp_catalog);
483
484         if (rate == link_rate_hbr3)
485                 pixel_div = 6;
486         else if (rate == 162000 || rate == 270000)
487                 pixel_div = 2;
488         else if (rate == link_rate_hbr2)
489                 pixel_div = 4;
490         else
491                 DRM_ERROR("Invalid pixel mux divider\n");
492
493         dispcc_input_rate = (rate * 10) / pixel_div;
494
495         rational_best_approximation(dispcc_input_rate, stream_rate_khz,
496                         (unsigned long)(1 << 16) - 1,
497                         (unsigned long)(1 << 16) - 1, &den, &num);
498
499         den = ~(den - num);
500         den = den & 0xFFFF;
501         pixel_m = num;
502         pixel_n = den;
503
504         mvid = (pixel_m & 0xFFFF) * 5;
505         nvid = (0xFFFF & (~pixel_n)) + (pixel_m & 0xFFFF);
506
507         if (nvid < nvid_fixed) {
508                 u32 temp;
509
510                 temp = (nvid_fixed / nvid) * nvid;
511                 mvid = (nvid_fixed / nvid) * mvid;
512                 nvid = temp;
513         }
514
515         if (is_ycbcr_420)
516                 mvid /= 2;
517
518         if (link_rate_hbr2 == rate)
519                 nvid *= 2;
520
521         if (link_rate_hbr3 == rate)
522                 nvid *= 3;
523
524         drm_dbg_dp(catalog->drm_dev, "mvid=0x%x, nvid=0x%x\n", mvid, nvid);
525         msm_dp_write_link(catalog, REG_DP_SOFTWARE_MVID, mvid);
526         msm_dp_write_link(catalog, REG_DP_SOFTWARE_NVID, nvid);
527         msm_dp_write_p0(catalog, MMSS_DP_DSC_DTO, 0x0);
528 }
529
530 int msm_dp_catalog_ctrl_set_pattern_state_bit(struct msm_dp_catalog *msm_dp_catalog,
531                                         u32 state_bit)
532 {
533         int bit, ret;
534         u32 data;
535         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
536                                 struct msm_dp_catalog_private, msm_dp_catalog);
537
538         bit = BIT(state_bit - 1);
539         drm_dbg_dp(catalog->drm_dev, "hw: bit=%d train=%d\n", bit, state_bit);
540         msm_dp_catalog_ctrl_state_ctrl(msm_dp_catalog, bit);
541
542         bit = BIT(state_bit - 1) << DP_MAINLINK_READY_LINK_TRAINING_SHIFT;
543
544         /* Poll for mainlink ready status */
545         ret = readx_poll_timeout(readl, catalog->io.link.base +
546                                         REG_DP_MAINLINK_READY,
547                                         data, data & bit,
548                                         POLLING_SLEEP_US, POLLING_TIMEOUT_US);
549         if (ret < 0) {
550                 DRM_ERROR("set state_bit for link_train=%d failed\n", state_bit);
551                 return ret;
552         }
553         return 0;
554 }
555
556 /**
557  * msm_dp_catalog_hw_revision() - retrieve DP hw revision
558  *
559  * @msm_dp_catalog: DP catalog structure
560  *
561  * Return: DP controller hw revision
562  *
563  */
564 u32 msm_dp_catalog_hw_revision(const struct msm_dp_catalog *msm_dp_catalog)
565 {
566         const struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
567                                 struct msm_dp_catalog_private, msm_dp_catalog);
568
569         return msm_dp_read_ahb(catalog, REG_DP_HW_VERSION);
570 }
571
572 /**
573  * msm_dp_catalog_ctrl_reset() - reset DP controller
574  *
575  * @msm_dp_catalog: DP catalog structure
576  *
577  * return: void
578  *
579  * This function reset the DP controller
580  *
581  * NOTE: reset DP controller will also clear any pending HPD related interrupts
582  * 
583  */
584 void msm_dp_catalog_ctrl_reset(struct msm_dp_catalog *msm_dp_catalog)
585 {
586         u32 sw_reset;
587         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
588                                 struct msm_dp_catalog_private, msm_dp_catalog);
589
590         sw_reset = msm_dp_read_ahb(catalog, REG_DP_SW_RESET);
591
592         sw_reset |= DP_SW_RESET;
593         msm_dp_write_ahb(catalog, REG_DP_SW_RESET, sw_reset);
594         usleep_range(1000, 1100); /* h/w recommended delay */
595
596         sw_reset &= ~DP_SW_RESET;
597         msm_dp_write_ahb(catalog, REG_DP_SW_RESET, sw_reset);
598 }
599
600 bool msm_dp_catalog_ctrl_mainlink_ready(struct msm_dp_catalog *msm_dp_catalog)
601 {
602         u32 data;
603         int ret;
604         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
605                                 struct msm_dp_catalog_private, msm_dp_catalog);
606
607         /* Poll for mainlink ready status */
608         ret = readl_poll_timeout(catalog->io.link.base +
609                                 REG_DP_MAINLINK_READY,
610                                 data, data & DP_MAINLINK_READY_FOR_VIDEO,
611                                 POLLING_SLEEP_US, POLLING_TIMEOUT_US);
612         if (ret < 0) {
613                 DRM_ERROR("mainlink not ready\n");
614                 return false;
615         }
616
617         return true;
618 }
619
620 void msm_dp_catalog_ctrl_enable_irq(struct msm_dp_catalog *msm_dp_catalog,
621                                                 bool enable)
622 {
623         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
624                                 struct msm_dp_catalog_private, msm_dp_catalog);
625
626         if (enable) {
627                 msm_dp_write_ahb(catalog, REG_DP_INTR_STATUS,
628                                 DP_INTERRUPT_STATUS1_MASK);
629                 msm_dp_write_ahb(catalog, REG_DP_INTR_STATUS2,
630                                 DP_INTERRUPT_STATUS2_MASK);
631         } else {
632                 msm_dp_write_ahb(catalog, REG_DP_INTR_STATUS, 0x00);
633                 msm_dp_write_ahb(catalog, REG_DP_INTR_STATUS2, 0x00);
634         }
635 }
636
637 void msm_dp_catalog_hpd_config_intr(struct msm_dp_catalog *msm_dp_catalog,
638                         u32 intr_mask, bool en)
639 {
640         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
641                                 struct msm_dp_catalog_private, msm_dp_catalog);
642
643         u32 config = msm_dp_read_aux(catalog, REG_DP_DP_HPD_INT_MASK);
644
645         config = (en ? config | intr_mask : config & ~intr_mask);
646
647         drm_dbg_dp(catalog->drm_dev, "intr_mask=%#x config=%#x\n",
648                                         intr_mask, config);
649         msm_dp_write_aux(catalog, REG_DP_DP_HPD_INT_MASK,
650                                 config & DP_DP_HPD_INT_MASK);
651 }
652
653 void msm_dp_catalog_ctrl_hpd_enable(struct msm_dp_catalog *msm_dp_catalog)
654 {
655         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
656                                 struct msm_dp_catalog_private, msm_dp_catalog);
657
658         u32 reftimer = msm_dp_read_aux(catalog, REG_DP_DP_HPD_REFTIMER);
659
660         /* Configure REFTIMER and enable it */
661         reftimer |= DP_DP_HPD_REFTIMER_ENABLE;
662         msm_dp_write_aux(catalog, REG_DP_DP_HPD_REFTIMER, reftimer);
663
664         /* Enable HPD */
665         msm_dp_write_aux(catalog, REG_DP_DP_HPD_CTRL, DP_DP_HPD_CTRL_HPD_EN);
666 }
667
668 void msm_dp_catalog_ctrl_hpd_disable(struct msm_dp_catalog *msm_dp_catalog)
669 {
670         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
671                                 struct msm_dp_catalog_private, msm_dp_catalog);
672
673         u32 reftimer = msm_dp_read_aux(catalog, REG_DP_DP_HPD_REFTIMER);
674
675         reftimer &= ~DP_DP_HPD_REFTIMER_ENABLE;
676         msm_dp_write_aux(catalog, REG_DP_DP_HPD_REFTIMER, reftimer);
677
678         msm_dp_write_aux(catalog, REG_DP_DP_HPD_CTRL, 0);
679 }
680
681 static void msm_dp_catalog_enable_sdp(struct msm_dp_catalog_private *catalog)
682 {
683         /* trigger sdp */
684         msm_dp_write_link(catalog, MMSS_DP_SDP_CFG3, UPDATE_SDP);
685         msm_dp_write_link(catalog, MMSS_DP_SDP_CFG3, 0x0);
686 }
687
688 void msm_dp_catalog_ctrl_config_psr(struct msm_dp_catalog *msm_dp_catalog)
689 {
690         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
691                                 struct msm_dp_catalog_private, msm_dp_catalog);
692         u32 config;
693
694         /* enable PSR1 function */
695         config = msm_dp_read_link(catalog, REG_PSR_CONFIG);
696         config |= PSR1_SUPPORTED;
697         msm_dp_write_link(catalog, REG_PSR_CONFIG, config);
698
699         msm_dp_write_ahb(catalog, REG_DP_INTR_MASK4, DP_INTERRUPT_MASK4);
700         msm_dp_catalog_enable_sdp(catalog);
701 }
702
703 void msm_dp_catalog_ctrl_set_psr(struct msm_dp_catalog *msm_dp_catalog, bool enter)
704 {
705         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
706                         struct msm_dp_catalog_private, msm_dp_catalog);
707         u32 cmd;
708
709         cmd = msm_dp_read_link(catalog, REG_PSR_CMD);
710
711         cmd &= ~(PSR_ENTER | PSR_EXIT);
712
713         if (enter)
714                 cmd |= PSR_ENTER;
715         else
716                 cmd |= PSR_EXIT;
717
718         msm_dp_catalog_enable_sdp(catalog);
719         msm_dp_write_link(catalog, REG_PSR_CMD, cmd);
720 }
721
722 u32 msm_dp_catalog_link_is_connected(struct msm_dp_catalog *msm_dp_catalog)
723 {
724         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
725                                 struct msm_dp_catalog_private, msm_dp_catalog);
726         u32 status;
727
728         status = msm_dp_read_aux(catalog, REG_DP_DP_HPD_INT_STATUS);
729         drm_dbg_dp(catalog->drm_dev, "aux status: %#x\n", status);
730         status >>= DP_DP_HPD_STATE_STATUS_BITS_SHIFT;
731         status &= DP_DP_HPD_STATE_STATUS_BITS_MASK;
732
733         return status;
734 }
735
736 u32 msm_dp_catalog_hpd_get_intr_status(struct msm_dp_catalog *msm_dp_catalog)
737 {
738         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
739                                 struct msm_dp_catalog_private, msm_dp_catalog);
740         int isr, mask;
741
742         isr = msm_dp_read_aux(catalog, REG_DP_DP_HPD_INT_STATUS);
743         msm_dp_write_aux(catalog, REG_DP_DP_HPD_INT_ACK,
744                                  (isr & DP_DP_HPD_INT_MASK));
745         mask = msm_dp_read_aux(catalog, REG_DP_DP_HPD_INT_MASK);
746
747         /*
748          * We only want to return interrupts that are unmasked to the caller.
749          * However, the interrupt status field also contains other
750          * informational bits about the HPD state status, so we only mask
751          * out the part of the register that tells us about which interrupts
752          * are pending.
753          */
754         return isr & (mask | ~DP_DP_HPD_INT_MASK);
755 }
756
757 u32 msm_dp_catalog_ctrl_read_psr_interrupt_status(struct msm_dp_catalog *msm_dp_catalog)
758 {
759         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
760                                 struct msm_dp_catalog_private, msm_dp_catalog);
761         u32 intr, intr_ack;
762
763         intr = msm_dp_read_ahb(catalog, REG_DP_INTR_STATUS4);
764         intr_ack = (intr & DP_INTERRUPT_STATUS4)
765                         << DP_INTERRUPT_STATUS_ACK_SHIFT;
766         msm_dp_write_ahb(catalog, REG_DP_INTR_STATUS4, intr_ack);
767
768         return intr;
769 }
770
771 int msm_dp_catalog_ctrl_get_interrupt(struct msm_dp_catalog *msm_dp_catalog)
772 {
773         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
774                                 struct msm_dp_catalog_private, msm_dp_catalog);
775         u32 intr, intr_ack;
776
777         intr = msm_dp_read_ahb(catalog, REG_DP_INTR_STATUS2);
778         intr &= ~DP_INTERRUPT_STATUS2_MASK;
779         intr_ack = (intr & DP_INTERRUPT_STATUS2)
780                         << DP_INTERRUPT_STATUS_ACK_SHIFT;
781         msm_dp_write_ahb(catalog, REG_DP_INTR_STATUS2,
782                         intr_ack | DP_INTERRUPT_STATUS2_MASK);
783
784         return intr;
785 }
786
787 void msm_dp_catalog_ctrl_phy_reset(struct msm_dp_catalog *msm_dp_catalog)
788 {
789         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
790                                 struct msm_dp_catalog_private, msm_dp_catalog);
791
792         msm_dp_write_ahb(catalog, REG_DP_PHY_CTRL,
793                         DP_PHY_CTRL_SW_RESET | DP_PHY_CTRL_SW_RESET_PLL);
794         usleep_range(1000, 1100); /* h/w recommended delay */
795         msm_dp_write_ahb(catalog, REG_DP_PHY_CTRL, 0x0);
796 }
797
798 void msm_dp_catalog_ctrl_send_phy_pattern(struct msm_dp_catalog *msm_dp_catalog,
799                         u32 pattern)
800 {
801         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
802                                 struct msm_dp_catalog_private, msm_dp_catalog);
803         u32 value = 0x0;
804
805         /* Make sure to clear the current pattern before starting a new one */
806         msm_dp_write_link(catalog, REG_DP_STATE_CTRL, 0x0);
807
808         drm_dbg_dp(catalog->drm_dev, "pattern: %#x\n", pattern);
809         switch (pattern) {
810         case DP_PHY_TEST_PATTERN_D10_2:
811                 msm_dp_write_link(catalog, REG_DP_STATE_CTRL,
812                                 DP_STATE_CTRL_LINK_TRAINING_PATTERN1);
813                 break;
814         case DP_PHY_TEST_PATTERN_ERROR_COUNT:
815                 value &= ~(1 << 16);
816                 msm_dp_write_link(catalog, REG_DP_HBR2_COMPLIANCE_SCRAMBLER_RESET,
817                                         value);
818                 value |= SCRAMBLER_RESET_COUNT_VALUE;
819                 msm_dp_write_link(catalog, REG_DP_HBR2_COMPLIANCE_SCRAMBLER_RESET,
820                                         value);
821                 msm_dp_write_link(catalog, REG_DP_MAINLINK_LEVELS,
822                                         DP_MAINLINK_SAFE_TO_EXIT_LEVEL_2);
823                 msm_dp_write_link(catalog, REG_DP_STATE_CTRL,
824                                         DP_STATE_CTRL_LINK_SYMBOL_ERR_MEASURE);
825                 break;
826         case DP_PHY_TEST_PATTERN_PRBS7:
827                 msm_dp_write_link(catalog, REG_DP_STATE_CTRL,
828                                 DP_STATE_CTRL_LINK_PRBS7);
829                 break;
830         case DP_PHY_TEST_PATTERN_80BIT_CUSTOM:
831                 msm_dp_write_link(catalog, REG_DP_STATE_CTRL,
832                                 DP_STATE_CTRL_LINK_TEST_CUSTOM_PATTERN);
833                 /* 00111110000011111000001111100000 */
834                 msm_dp_write_link(catalog, REG_DP_TEST_80BIT_CUSTOM_PATTERN_REG0,
835                                 0x3E0F83E0);
836                 /* 00001111100000111110000011111000 */
837                 msm_dp_write_link(catalog, REG_DP_TEST_80BIT_CUSTOM_PATTERN_REG1,
838                                 0x0F83E0F8);
839                 /* 1111100000111110 */
840                 msm_dp_write_link(catalog, REG_DP_TEST_80BIT_CUSTOM_PATTERN_REG2,
841                                 0x0000F83E);
842                 break;
843         case DP_PHY_TEST_PATTERN_CP2520:
844                 value = msm_dp_read_link(catalog, REG_DP_MAINLINK_CTRL);
845                 value &= ~DP_MAINLINK_CTRL_SW_BYPASS_SCRAMBLER;
846                 msm_dp_write_link(catalog, REG_DP_MAINLINK_CTRL, value);
847
848                 value = DP_HBR2_ERM_PATTERN;
849                 msm_dp_write_link(catalog, REG_DP_HBR2_COMPLIANCE_SCRAMBLER_RESET,
850                                 value);
851                 value |= SCRAMBLER_RESET_COUNT_VALUE;
852                 msm_dp_write_link(catalog, REG_DP_HBR2_COMPLIANCE_SCRAMBLER_RESET,
853                                         value);
854                 msm_dp_write_link(catalog, REG_DP_MAINLINK_LEVELS,
855                                         DP_MAINLINK_SAFE_TO_EXIT_LEVEL_2);
856                 msm_dp_write_link(catalog, REG_DP_STATE_CTRL,
857                                         DP_STATE_CTRL_LINK_SYMBOL_ERR_MEASURE);
858                 value = msm_dp_read_link(catalog, REG_DP_MAINLINK_CTRL);
859                 value |= DP_MAINLINK_CTRL_ENABLE;
860                 msm_dp_write_link(catalog, REG_DP_MAINLINK_CTRL, value);
861                 break;
862         case DP_PHY_TEST_PATTERN_SEL_MASK:
863                 msm_dp_write_link(catalog, REG_DP_MAINLINK_CTRL,
864                                 DP_MAINLINK_CTRL_ENABLE);
865                 msm_dp_write_link(catalog, REG_DP_STATE_CTRL,
866                                 DP_STATE_CTRL_LINK_TRAINING_PATTERN4);
867                 break;
868         default:
869                 drm_dbg_dp(catalog->drm_dev,
870                                 "No valid test pattern requested: %#x\n", pattern);
871                 break;
872         }
873 }
874
875 u32 msm_dp_catalog_ctrl_read_phy_pattern(struct msm_dp_catalog *msm_dp_catalog)
876 {
877         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
878                                 struct msm_dp_catalog_private, msm_dp_catalog);
879
880         return msm_dp_read_link(catalog, REG_DP_MAINLINK_READY);
881 }
882
883 /* panel related catalog functions */
884 int msm_dp_catalog_panel_timing_cfg(struct msm_dp_catalog *msm_dp_catalog, u32 total,
885                                 u32 sync_start, u32 width_blanking, u32 msm_dp_active)
886 {
887         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
888                                 struct msm_dp_catalog_private, msm_dp_catalog);
889         u32 reg;
890
891         msm_dp_write_link(catalog, REG_DP_TOTAL_HOR_VER, total);
892         msm_dp_write_link(catalog, REG_DP_START_HOR_VER_FROM_SYNC, sync_start);
893         msm_dp_write_link(catalog, REG_DP_HSYNC_VSYNC_WIDTH_POLARITY, width_blanking);
894         msm_dp_write_link(catalog, REG_DP_ACTIVE_HOR_VER, msm_dp_active);
895
896         reg = msm_dp_read_p0(catalog, MMSS_DP_INTF_CONFIG);
897
898         if (msm_dp_catalog->wide_bus_en)
899                 reg |= DP_INTF_CONFIG_DATABUS_WIDEN;
900         else
901                 reg &= ~DP_INTF_CONFIG_DATABUS_WIDEN;
902
903
904         DRM_DEBUG_DP("wide_bus_en=%d reg=%#x\n", msm_dp_catalog->wide_bus_en, reg);
905
906         msm_dp_write_p0(catalog, MMSS_DP_INTF_CONFIG, reg);
907         return 0;
908 }
909
910 static void msm_dp_catalog_panel_send_vsc_sdp(struct msm_dp_catalog *msm_dp_catalog, struct dp_sdp *vsc_sdp)
911 {
912         struct msm_dp_catalog_private *catalog;
913         u32 header[2];
914         u32 val;
915         int i;
916
917         catalog = container_of(msm_dp_catalog, struct msm_dp_catalog_private, msm_dp_catalog);
918
919         msm_dp_utils_pack_sdp_header(&vsc_sdp->sdp_header, header);
920
921         msm_dp_write_link(catalog, MMSS_DP_GENERIC0_0, header[0]);
922         msm_dp_write_link(catalog, MMSS_DP_GENERIC0_1, header[1]);
923
924         for (i = 0; i < sizeof(vsc_sdp->db); i += 4) {
925                 val = ((vsc_sdp->db[i]) | (vsc_sdp->db[i + 1] << 8) | (vsc_sdp->db[i + 2] << 16) |
926                        (vsc_sdp->db[i + 3] << 24));
927                 msm_dp_write_link(catalog, MMSS_DP_GENERIC0_2 + i, val);
928         }
929 }
930
931 static void msm_dp_catalog_panel_update_sdp(struct msm_dp_catalog *msm_dp_catalog)
932 {
933         struct msm_dp_catalog_private *catalog;
934         u32 hw_revision;
935
936         catalog = container_of(msm_dp_catalog, struct msm_dp_catalog_private, msm_dp_catalog);
937
938         hw_revision = msm_dp_catalog_hw_revision(msm_dp_catalog);
939         if (hw_revision < DP_HW_VERSION_1_2 && hw_revision >= DP_HW_VERSION_1_0) {
940                 msm_dp_write_link(catalog, MMSS_DP_SDP_CFG3, 0x01);
941                 msm_dp_write_link(catalog, MMSS_DP_SDP_CFG3, 0x00);
942         }
943 }
944
945 void msm_dp_catalog_panel_enable_vsc_sdp(struct msm_dp_catalog *msm_dp_catalog, struct dp_sdp *vsc_sdp)
946 {
947         struct msm_dp_catalog_private *catalog;
948         u32 cfg, cfg2, misc;
949
950         catalog = container_of(msm_dp_catalog, struct msm_dp_catalog_private, msm_dp_catalog);
951
952         cfg = msm_dp_read_link(catalog, MMSS_DP_SDP_CFG);
953         cfg2 = msm_dp_read_link(catalog, MMSS_DP_SDP_CFG2);
954         misc = msm_dp_read_link(catalog, REG_DP_MISC1_MISC0);
955
956         cfg |= GEN0_SDP_EN;
957         msm_dp_write_link(catalog, MMSS_DP_SDP_CFG, cfg);
958
959         cfg2 |= GENERIC0_SDPSIZE_VALID;
960         msm_dp_write_link(catalog, MMSS_DP_SDP_CFG2, cfg2);
961
962         msm_dp_catalog_panel_send_vsc_sdp(msm_dp_catalog, vsc_sdp);
963
964         /* indicates presence of VSC (BIT(6) of MISC1) */
965         misc |= DP_MISC1_VSC_SDP;
966
967         drm_dbg_dp(catalog->drm_dev, "vsc sdp enable=1\n");
968
969         pr_debug("misc settings = 0x%x\n", misc);
970         msm_dp_write_link(catalog, REG_DP_MISC1_MISC0, misc);
971
972         msm_dp_catalog_panel_update_sdp(msm_dp_catalog);
973 }
974
975 void msm_dp_catalog_panel_disable_vsc_sdp(struct msm_dp_catalog *msm_dp_catalog)
976 {
977         struct msm_dp_catalog_private *catalog;
978         u32 cfg, cfg2, misc;
979
980         catalog = container_of(msm_dp_catalog, struct msm_dp_catalog_private, msm_dp_catalog);
981
982         cfg = msm_dp_read_link(catalog, MMSS_DP_SDP_CFG);
983         cfg2 = msm_dp_read_link(catalog, MMSS_DP_SDP_CFG2);
984         misc = msm_dp_read_link(catalog, REG_DP_MISC1_MISC0);
985
986         cfg &= ~GEN0_SDP_EN;
987         msm_dp_write_link(catalog, MMSS_DP_SDP_CFG, cfg);
988
989         cfg2 &= ~GENERIC0_SDPSIZE_VALID;
990         msm_dp_write_link(catalog, MMSS_DP_SDP_CFG2, cfg2);
991
992         /* switch back to MSA */
993         misc &= ~DP_MISC1_VSC_SDP;
994
995         drm_dbg_dp(catalog->drm_dev, "vsc sdp enable=0\n");
996
997         pr_debug("misc settings = 0x%x\n", misc);
998         msm_dp_write_link(catalog, REG_DP_MISC1_MISC0, misc);
999
1000         msm_dp_catalog_panel_update_sdp(msm_dp_catalog);
1001 }
1002
1003 void msm_dp_catalog_panel_tpg_enable(struct msm_dp_catalog *msm_dp_catalog,
1004                                 struct drm_display_mode *drm_mode)
1005 {
1006         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
1007                                 struct msm_dp_catalog_private, msm_dp_catalog);
1008         u32 hsync_period, vsync_period;
1009         u32 display_v_start, display_v_end;
1010         u32 hsync_start_x, hsync_end_x;
1011         u32 v_sync_width;
1012         u32 hsync_ctl;
1013         u32 display_hctl;
1014
1015         /* TPG config parameters*/
1016         hsync_period = drm_mode->htotal;
1017         vsync_period = drm_mode->vtotal;
1018
1019         display_v_start = ((drm_mode->vtotal - drm_mode->vsync_start) *
1020                                         hsync_period);
1021         display_v_end = ((vsync_period - (drm_mode->vsync_start -
1022                                         drm_mode->vdisplay))
1023                                         * hsync_period) - 1;
1024
1025         display_v_start += drm_mode->htotal - drm_mode->hsync_start;
1026         display_v_end -= (drm_mode->hsync_start - drm_mode->hdisplay);
1027
1028         hsync_start_x = drm_mode->htotal - drm_mode->hsync_start;
1029         hsync_end_x = hsync_period - (drm_mode->hsync_start -
1030                                         drm_mode->hdisplay) - 1;
1031
1032         v_sync_width = drm_mode->vsync_end - drm_mode->vsync_start;
1033
1034         hsync_ctl = (hsync_period << 16) |
1035                         (drm_mode->hsync_end - drm_mode->hsync_start);
1036         display_hctl = (hsync_end_x << 16) | hsync_start_x;
1037
1038
1039         msm_dp_write_p0(catalog, MMSS_DP_INTF_CONFIG, 0x0);
1040         msm_dp_write_p0(catalog, MMSS_DP_INTF_HSYNC_CTL, hsync_ctl);
1041         msm_dp_write_p0(catalog, MMSS_DP_INTF_VSYNC_PERIOD_F0, vsync_period *
1042                         hsync_period);
1043         msm_dp_write_p0(catalog, MMSS_DP_INTF_VSYNC_PULSE_WIDTH_F0, v_sync_width *
1044                         hsync_period);
1045         msm_dp_write_p0(catalog, MMSS_DP_INTF_VSYNC_PERIOD_F1, 0);
1046         msm_dp_write_p0(catalog, MMSS_DP_INTF_VSYNC_PULSE_WIDTH_F1, 0);
1047         msm_dp_write_p0(catalog, MMSS_DP_INTF_DISPLAY_HCTL, display_hctl);
1048         msm_dp_write_p0(catalog, MMSS_DP_INTF_ACTIVE_HCTL, 0);
1049         msm_dp_write_p0(catalog, MMSS_INTF_DISPLAY_V_START_F0, display_v_start);
1050         msm_dp_write_p0(catalog, MMSS_DP_INTF_DISPLAY_V_END_F0, display_v_end);
1051         msm_dp_write_p0(catalog, MMSS_INTF_DISPLAY_V_START_F1, 0);
1052         msm_dp_write_p0(catalog, MMSS_DP_INTF_DISPLAY_V_END_F1, 0);
1053         msm_dp_write_p0(catalog, MMSS_DP_INTF_ACTIVE_V_START_F0, 0);
1054         msm_dp_write_p0(catalog, MMSS_DP_INTF_ACTIVE_V_END_F0, 0);
1055         msm_dp_write_p0(catalog, MMSS_DP_INTF_ACTIVE_V_START_F1, 0);
1056         msm_dp_write_p0(catalog, MMSS_DP_INTF_ACTIVE_V_END_F1, 0);
1057         msm_dp_write_p0(catalog, MMSS_DP_INTF_POLARITY_CTL, 0);
1058
1059         msm_dp_write_p0(catalog, MMSS_DP_TPG_MAIN_CONTROL,
1060                                 DP_TPG_CHECKERED_RECT_PATTERN);
1061         msm_dp_write_p0(catalog, MMSS_DP_TPG_VIDEO_CONFIG,
1062                                 DP_TPG_VIDEO_CONFIG_BPP_8BIT |
1063                                 DP_TPG_VIDEO_CONFIG_RGB);
1064         msm_dp_write_p0(catalog, MMSS_DP_BIST_ENABLE,
1065                                 DP_BIST_ENABLE_DPBIST_EN);
1066         msm_dp_write_p0(catalog, MMSS_DP_TIMING_ENGINE_EN,
1067                                 DP_TIMING_ENGINE_EN_EN);
1068         drm_dbg_dp(catalog->drm_dev, "%s: enabled tpg\n", __func__);
1069 }
1070
1071 void msm_dp_catalog_panel_tpg_disable(struct msm_dp_catalog *msm_dp_catalog)
1072 {
1073         struct msm_dp_catalog_private *catalog = container_of(msm_dp_catalog,
1074                                 struct msm_dp_catalog_private, msm_dp_catalog);
1075
1076         msm_dp_write_p0(catalog, MMSS_DP_TPG_MAIN_CONTROL, 0x0);
1077         msm_dp_write_p0(catalog, MMSS_DP_BIST_ENABLE, 0x0);
1078         msm_dp_write_p0(catalog, MMSS_DP_TIMING_ENGINE_EN, 0x0);
1079 }
1080
1081 static void __iomem *msm_dp_ioremap(struct platform_device *pdev, int idx, size_t *len)
1082 {
1083         struct resource *res;
1084         void __iomem *base;
1085
1086         base = devm_platform_get_and_ioremap_resource(pdev, idx, &res);
1087         if (!IS_ERR(base))
1088                 *len = resource_size(res);
1089
1090         return base;
1091 }
1092
1093 static int msm_dp_catalog_get_io(struct msm_dp_catalog_private *catalog)
1094 {
1095         struct platform_device *pdev = to_platform_device(catalog->dev);
1096         struct dss_io_data *dss = &catalog->io;
1097
1098         dss->ahb.base = msm_dp_ioremap(pdev, 0, &dss->ahb.len);
1099         if (IS_ERR(dss->ahb.base))
1100                 return PTR_ERR(dss->ahb.base);
1101
1102         dss->aux.base = msm_dp_ioremap(pdev, 1, &dss->aux.len);
1103         if (IS_ERR(dss->aux.base)) {
1104                 /*
1105                  * The initial binding had a single reg, but in order to
1106                  * support variation in the sub-region sizes this was split.
1107                  * msm_dp_ioremap() will fail with -EINVAL here if only a single
1108                  * reg is specified, so fill in the sub-region offsets and
1109                  * lengths based on this single region.
1110                  */
1111                 if (PTR_ERR(dss->aux.base) == -EINVAL) {
1112                         if (dss->ahb.len < DP_DEFAULT_P0_OFFSET + DP_DEFAULT_P0_SIZE) {
1113                                 DRM_ERROR("legacy memory region not large enough\n");
1114                                 return -EINVAL;
1115                         }
1116
1117                         dss->ahb.len = DP_DEFAULT_AHB_SIZE;
1118                         dss->aux.base = dss->ahb.base + DP_DEFAULT_AUX_OFFSET;
1119                         dss->aux.len = DP_DEFAULT_AUX_SIZE;
1120                         dss->link.base = dss->ahb.base + DP_DEFAULT_LINK_OFFSET;
1121                         dss->link.len = DP_DEFAULT_LINK_SIZE;
1122                         dss->p0.base = dss->ahb.base + DP_DEFAULT_P0_OFFSET;
1123                         dss->p0.len = DP_DEFAULT_P0_SIZE;
1124                 } else {
1125                         DRM_ERROR("unable to remap aux region: %pe\n", dss->aux.base);
1126                         return PTR_ERR(dss->aux.base);
1127                 }
1128         } else {
1129                 dss->link.base = msm_dp_ioremap(pdev, 2, &dss->link.len);
1130                 if (IS_ERR(dss->link.base)) {
1131                         DRM_ERROR("unable to remap link region: %pe\n", dss->link.base);
1132                         return PTR_ERR(dss->link.base);
1133                 }
1134
1135                 dss->p0.base = msm_dp_ioremap(pdev, 3, &dss->p0.len);
1136                 if (IS_ERR(dss->p0.base)) {
1137                         DRM_ERROR("unable to remap p0 region: %pe\n", dss->p0.base);
1138                         return PTR_ERR(dss->p0.base);
1139                 }
1140         }
1141
1142         return 0;
1143 }
1144
1145 struct msm_dp_catalog *msm_dp_catalog_get(struct device *dev)
1146 {
1147         struct msm_dp_catalog_private *catalog;
1148         int ret;
1149
1150         catalog  = devm_kzalloc(dev, sizeof(*catalog), GFP_KERNEL);
1151         if (!catalog)
1152                 return ERR_PTR(-ENOMEM);
1153
1154         catalog->dev = dev;
1155
1156         ret = msm_dp_catalog_get_io(catalog);
1157         if (ret)
1158                 return ERR_PTR(ret);
1159
1160         return &catalog->msm_dp_catalog;
1161 }
1162
1163 u32 msm_dp_catalog_audio_get_header(struct msm_dp_catalog *msm_dp_catalog,
1164                                 enum msm_dp_catalog_audio_sdp_type sdp,
1165                                 enum msm_dp_catalog_audio_header_type header)
1166 {
1167         struct msm_dp_catalog_private *catalog;
1168         u32 (*sdp_map)[DP_AUDIO_SDP_HEADER_MAX];
1169
1170         catalog = container_of(msm_dp_catalog,
1171                 struct msm_dp_catalog_private, msm_dp_catalog);
1172
1173         sdp_map = catalog->audio_map;
1174
1175         return msm_dp_read_link(catalog, sdp_map[sdp][header]);
1176 }
1177
1178 void msm_dp_catalog_audio_set_header(struct msm_dp_catalog *msm_dp_catalog,
1179                                  enum msm_dp_catalog_audio_sdp_type sdp,
1180                                  enum msm_dp_catalog_audio_header_type header,
1181                                  u32 data)
1182 {
1183         struct msm_dp_catalog_private *catalog;
1184         u32 (*sdp_map)[DP_AUDIO_SDP_HEADER_MAX];
1185
1186         if (!msm_dp_catalog)
1187                 return;
1188
1189         catalog = container_of(msm_dp_catalog,
1190                 struct msm_dp_catalog_private, msm_dp_catalog);
1191
1192         sdp_map = catalog->audio_map;
1193
1194         msm_dp_write_link(catalog, sdp_map[sdp][header], data);
1195 }
1196
1197 void msm_dp_catalog_audio_config_acr(struct msm_dp_catalog *msm_dp_catalog, u32 select)
1198 {
1199         struct msm_dp_catalog_private *catalog;
1200         u32 acr_ctrl;
1201
1202         if (!msm_dp_catalog)
1203                 return;
1204
1205         catalog = container_of(msm_dp_catalog,
1206                 struct msm_dp_catalog_private, msm_dp_catalog);
1207
1208         acr_ctrl = select << 4 | BIT(31) | BIT(8) | BIT(14);
1209
1210         drm_dbg_dp(catalog->drm_dev, "select: %#x, acr_ctrl: %#x\n",
1211                                         select, acr_ctrl);
1212
1213         msm_dp_write_link(catalog, MMSS_DP_AUDIO_ACR_CTRL, acr_ctrl);
1214 }
1215
1216 void msm_dp_catalog_audio_enable(struct msm_dp_catalog *msm_dp_catalog, bool enable)
1217 {
1218         struct msm_dp_catalog_private *catalog;
1219         u32 audio_ctrl;
1220
1221         if (!msm_dp_catalog)
1222                 return;
1223
1224         catalog = container_of(msm_dp_catalog,
1225                 struct msm_dp_catalog_private, msm_dp_catalog);
1226
1227         audio_ctrl = msm_dp_read_link(catalog, MMSS_DP_AUDIO_CFG);
1228
1229         if (enable)
1230                 audio_ctrl |= BIT(0);
1231         else
1232                 audio_ctrl &= ~BIT(0);
1233
1234         drm_dbg_dp(catalog->drm_dev, "dp_audio_cfg = 0x%x\n", audio_ctrl);
1235
1236         msm_dp_write_link(catalog, MMSS_DP_AUDIO_CFG, audio_ctrl);
1237         /* make sure audio engine is disabled */
1238         wmb();
1239 }
1240
1241 void msm_dp_catalog_audio_config_sdp(struct msm_dp_catalog *msm_dp_catalog)
1242 {
1243         struct msm_dp_catalog_private *catalog;
1244         u32 sdp_cfg = 0;
1245         u32 sdp_cfg2 = 0;
1246
1247         if (!msm_dp_catalog)
1248                 return;
1249
1250         catalog = container_of(msm_dp_catalog,
1251                 struct msm_dp_catalog_private, msm_dp_catalog);
1252
1253         sdp_cfg = msm_dp_read_link(catalog, MMSS_DP_SDP_CFG);
1254         /* AUDIO_TIMESTAMP_SDP_EN */
1255         sdp_cfg |= BIT(1);
1256         /* AUDIO_STREAM_SDP_EN */
1257         sdp_cfg |= BIT(2);
1258         /* AUDIO_COPY_MANAGEMENT_SDP_EN */
1259         sdp_cfg |= BIT(5);
1260         /* AUDIO_ISRC_SDP_EN  */
1261         sdp_cfg |= BIT(6);
1262         /* AUDIO_INFOFRAME_SDP_EN  */
1263         sdp_cfg |= BIT(20);
1264
1265         drm_dbg_dp(catalog->drm_dev, "sdp_cfg = 0x%x\n", sdp_cfg);
1266
1267         msm_dp_write_link(catalog, MMSS_DP_SDP_CFG, sdp_cfg);
1268
1269         sdp_cfg2 = msm_dp_read_link(catalog, MMSS_DP_SDP_CFG2);
1270         /* IFRM_REGSRC -> Do not use reg values */
1271         sdp_cfg2 &= ~BIT(0);
1272         /* AUDIO_STREAM_HB3_REGSRC-> Do not use reg values */
1273         sdp_cfg2 &= ~BIT(1);
1274
1275         drm_dbg_dp(catalog->drm_dev, "sdp_cfg2 = 0x%x\n", sdp_cfg2);
1276
1277         msm_dp_write_link(catalog, MMSS_DP_SDP_CFG2, sdp_cfg2);
1278 }
1279
1280 void msm_dp_catalog_audio_init(struct msm_dp_catalog *msm_dp_catalog)
1281 {
1282         struct msm_dp_catalog_private *catalog;
1283
1284         static u32 sdp_map[][DP_AUDIO_SDP_HEADER_MAX] = {
1285                 {
1286                         MMSS_DP_AUDIO_STREAM_0,
1287                         MMSS_DP_AUDIO_STREAM_1,
1288                         MMSS_DP_AUDIO_STREAM_1,
1289                 },
1290                 {
1291                         MMSS_DP_AUDIO_TIMESTAMP_0,
1292                         MMSS_DP_AUDIO_TIMESTAMP_1,
1293                         MMSS_DP_AUDIO_TIMESTAMP_1,
1294                 },
1295                 {
1296                         MMSS_DP_AUDIO_INFOFRAME_0,
1297                         MMSS_DP_AUDIO_INFOFRAME_1,
1298                         MMSS_DP_AUDIO_INFOFRAME_1,
1299                 },
1300                 {
1301                         MMSS_DP_AUDIO_COPYMANAGEMENT_0,
1302                         MMSS_DP_AUDIO_COPYMANAGEMENT_1,
1303                         MMSS_DP_AUDIO_COPYMANAGEMENT_1,
1304                 },
1305                 {
1306                         MMSS_DP_AUDIO_ISRC_0,
1307                         MMSS_DP_AUDIO_ISRC_1,
1308                         MMSS_DP_AUDIO_ISRC_1,
1309                 },
1310         };
1311
1312         if (!msm_dp_catalog)
1313                 return;
1314
1315         catalog = container_of(msm_dp_catalog,
1316                 struct msm_dp_catalog_private, msm_dp_catalog);
1317
1318         catalog->audio_map = sdp_map;
1319 }
1320
1321 void msm_dp_catalog_audio_sfe_level(struct msm_dp_catalog *msm_dp_catalog, u32 safe_to_exit_level)
1322 {
1323         struct msm_dp_catalog_private *catalog;
1324         u32 mainlink_levels;
1325
1326         if (!msm_dp_catalog)
1327                 return;
1328
1329         catalog = container_of(msm_dp_catalog,
1330                 struct msm_dp_catalog_private, msm_dp_catalog);
1331
1332         mainlink_levels = msm_dp_read_link(catalog, REG_DP_MAINLINK_LEVELS);
1333         mainlink_levels &= 0xFE0;
1334         mainlink_levels |= safe_to_exit_level;
1335
1336         drm_dbg_dp(catalog->drm_dev,
1337                         "mainlink_level = 0x%x, safe_to_exit_level = 0x%x\n",
1338                          mainlink_levels, safe_to_exit_level);
1339
1340         msm_dp_write_link(catalog, REG_DP_MAINLINK_LEVELS, mainlink_levels);
1341 }
This page took 0.133147 seconds and 4 git commands to generate.