blob: a3fe2f06b80c3fbda25c7cf10745324f344ef5cd [file] [log] [blame]
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001/*
2 * Copyright (C) 2011 Samsung Electronics Co.Ltd
3 * Authors:
4 * Seung-Woo Kim <sw0312.kim@samsung.com>
5 * Inki Dae <inki.dae@samsung.com>
6 * Joonyoung Shim <jy0922.shim@samsung.com>
7 *
8 * Based on drivers/media/video/s5p-tv/hdmi_drv.c
9 *
10 * This program is free software; you can redistribute it and/or modify it
11 * under the terms of the GNU General Public License as published by the
12 * Free Software Foundation; either version 2 of the License, or (at your
13 * option) any later version.
14 *
15 */
16
David Howells760285e2012-10-02 18:01:07 +010017#include <drm/drmP.h>
18#include <drm/drm_edid.h>
19#include <drm/drm_crtc_helper.h>
Gustavo Padovan4ea95262015-06-01 12:04:44 -030020#include <drm/drm_atomic_helper.h>
Seung-Woo Kimd8408322011-12-21 17:39:39 +090021
22#include "regs-hdmi.h"
23
24#include <linux/kernel.h>
Seung-Woo Kimd8408322011-12-21 17:39:39 +090025#include <linux/wait.h>
26#include <linux/i2c.h>
Seung-Woo Kimd8408322011-12-21 17:39:39 +090027#include <linux/platform_device.h>
28#include <linux/interrupt.h>
29#include <linux/irq.h>
30#include <linux/delay.h>
31#include <linux/pm_runtime.h>
32#include <linux/clk.h>
33#include <linux/regulator/consumer.h>
Rahul Sharma22c4f422012-10-04 20:48:55 +053034#include <linux/io.h>
Rahul Sharmad5e9ca42014-05-09 15:34:18 +090035#include <linux/of_address.h>
Andrzej Hajdacd240cd2015-07-09 16:28:09 +020036#include <linux/of_device.h>
Rahul Sharma22c4f422012-10-04 20:48:55 +053037#include <linux/of_gpio.h>
Sachin Kamatd34d59b2014-02-04 08:40:18 +053038#include <linux/hdmi.h>
Inki Daef37cd5e2014-05-09 14:25:20 +090039#include <linux/component.h>
Rahul Sharma049d34e2014-05-20 10:36:05 +053040#include <linux/mfd/syscon.h>
41#include <linux/regmap.h>
Seung-Woo Kimd8408322011-12-21 17:39:39 +090042
43#include <drm/exynos_drm.h>
44
45#include "exynos_drm_drv.h"
Inki Daef37cd5e2014-05-09 14:25:20 +090046#include "exynos_drm_crtc.h"
Sean Paulf041b252014-01-30 16:19:15 -050047#include "exynos_mixer.h"
Seung-Woo Kimd8408322011-12-21 17:39:39 +090048
Tomasz Stanislawskifca57122012-10-04 20:48:46 +053049#include <linux/gpio.h>
Tomasz Stanislawskifca57122012-10-04 20:48:46 +053050
Sean Pauld9716ee2014-01-30 16:19:29 -050051#define ctx_from_connector(c) container_of(c, struct hdmi_context, connector)
Seung-Woo Kimd8408322011-12-21 17:39:39 +090052
Sean Paul724fd142014-05-09 15:05:10 +090053#define HOTPLUG_DEBOUNCE_MS 1100
54
Rahul Sharmaa144c2e2012-11-26 10:52:57 +053055/* AVI header and aspect ratio */
56#define HDMI_AVI_VERSION 0x02
57#define HDMI_AVI_LENGTH 0x0D
Rahul Sharmaa144c2e2012-11-26 10:52:57 +053058
59/* AUI header info */
60#define HDMI_AUI_VERSION 0x01
61#define HDMI_AUI_LENGTH 0x0A
Shirish S46154152014-03-13 10:58:28 +053062#define AVI_SAME_AS_PIC_ASPECT_RATIO 0x8
63#define AVI_4_3_CENTER_RATIO 0x9
64#define AVI_16_9_CENTER_RATIO 0xa
Rahul Sharmaa144c2e2012-11-26 10:52:57 +053065
Rahul Sharma5a325072012-10-04 20:48:54 +053066enum hdmi_type {
67 HDMI_TYPE13,
68 HDMI_TYPE14,
69};
70
Inki Daebfe4e842014-03-06 14:18:17 +090071struct hdmi_driver_data {
72 unsigned int type;
Rahul Sharmad5e9ca42014-05-09 15:34:18 +090073 const struct hdmiphy_config *phy_confs;
74 unsigned int phy_conf_count;
Inki Daebfe4e842014-03-06 14:18:17 +090075 unsigned int is_apb_phy:1;
76};
77
Joonyoung Shim590f4182012-03-16 18:47:14 +090078struct hdmi_resources {
79 struct clk *hdmi;
80 struct clk *sclk_hdmi;
81 struct clk *sclk_pixel;
82 struct clk *sclk_hdmiphy;
Rahul Sharma59956d32013-06-11 12:24:03 +053083 struct clk *mout_hdmi;
Joonyoung Shim590f4182012-03-16 18:47:14 +090084 struct regulator_bulk_data *regul_bulk;
Marek Szyprowski05fdf982014-07-01 10:10:06 +020085 struct regulator *reg_hdmi_en;
Joonyoung Shim590f4182012-03-16 18:47:14 +090086 int regul_count;
87};
88
Sean Paul2f7e2ed2013-01-15 08:11:08 -050089struct hdmi_tg_regs {
90 u8 cmd[1];
91 u8 h_fsz[2];
92 u8 hact_st[2];
93 u8 hact_sz[2];
94 u8 v_fsz[2];
95 u8 vsync[2];
96 u8 vsync2[2];
97 u8 vact_st[2];
98 u8 vact_sz[2];
99 u8 field_chg[2];
100 u8 vact_st2[2];
101 u8 vact_st3[2];
102 u8 vact_st4[2];
103 u8 vsync_top_hdmi[2];
104 u8 vsync_bot_hdmi[2];
105 u8 field_top_hdmi[2];
106 u8 field_bot_hdmi[2];
107 u8 tg_3d[1];
108};
109
Rahul Sharma6b986ed2013-03-06 17:33:29 +0900110struct hdmi_v13_core_regs {
111 u8 h_blank[2];
112 u8 v_blank[3];
113 u8 h_v_line[3];
114 u8 vsync_pol[1];
115 u8 int_pro_mode[1];
116 u8 v_blank_f[3];
117 u8 h_sync_gen[3];
118 u8 v_sync_gen1[3];
119 u8 v_sync_gen2[3];
120 u8 v_sync_gen3[3];
121};
122
123struct hdmi_v14_core_regs {
Sean Paul2f7e2ed2013-01-15 08:11:08 -0500124 u8 h_blank[2];
125 u8 v2_blank[2];
126 u8 v1_blank[2];
127 u8 v_line[2];
128 u8 h_line[2];
129 u8 hsync_pol[1];
130 u8 vsync_pol[1];
131 u8 int_pro_mode[1];
132 u8 v_blank_f0[2];
133 u8 v_blank_f1[2];
134 u8 h_sync_start[2];
135 u8 h_sync_end[2];
136 u8 v_sync_line_bef_2[2];
137 u8 v_sync_line_bef_1[2];
138 u8 v_sync_line_aft_2[2];
139 u8 v_sync_line_aft_1[2];
140 u8 v_sync_line_aft_pxl_2[2];
141 u8 v_sync_line_aft_pxl_1[2];
142 u8 v_blank_f2[2]; /* for 3D mode */
143 u8 v_blank_f3[2]; /* for 3D mode */
144 u8 v_blank_f4[2]; /* for 3D mode */
145 u8 v_blank_f5[2]; /* for 3D mode */
146 u8 v_sync_line_aft_3[2];
147 u8 v_sync_line_aft_4[2];
148 u8 v_sync_line_aft_5[2];
149 u8 v_sync_line_aft_6[2];
150 u8 v_sync_line_aft_pxl_3[2];
151 u8 v_sync_line_aft_pxl_4[2];
152 u8 v_sync_line_aft_pxl_5[2];
153 u8 v_sync_line_aft_pxl_6[2];
154 u8 vact_space_1[2];
155 u8 vact_space_2[2];
156 u8 vact_space_3[2];
157 u8 vact_space_4[2];
158 u8 vact_space_5[2];
159 u8 vact_space_6[2];
160};
161
Rahul Sharma6b986ed2013-03-06 17:33:29 +0900162struct hdmi_v13_conf {
163 struct hdmi_v13_core_regs core;
Sean Paul2f7e2ed2013-01-15 08:11:08 -0500164 struct hdmi_tg_regs tg;
Rahul Sharma6b986ed2013-03-06 17:33:29 +0900165};
166
167struct hdmi_v14_conf {
168 struct hdmi_v14_core_regs core;
169 struct hdmi_tg_regs tg;
170};
171
Andrzej Hajdac93aaeb2015-07-09 16:28:10 +0200172union hdmi_conf_regs {
173 struct hdmi_v13_conf v13_conf;
174 struct hdmi_v14_conf v14_conf;
Sean Paul2f7e2ed2013-01-15 08:11:08 -0500175};
176
Joonyoung Shim590f4182012-03-16 18:47:14 +0900177struct hdmi_context {
Andrzej Hajda930865f2014-11-17 09:54:20 +0100178 struct exynos_drm_display display;
Joonyoung Shim590f4182012-03-16 18:47:14 +0900179 struct device *dev;
180 struct drm_device *drm_dev;
Sean Pauld9716ee2014-01-30 16:19:29 -0500181 struct drm_connector connector;
182 struct drm_encoder *encoder;
Joonyoung Shimcf8fc4f2012-04-23 19:35:50 +0900183 bool powered;
Seung-Woo Kim872d20d62012-04-24 17:39:15 +0900184 bool dvi_mode;
Joonyoung Shim590f4182012-03-16 18:47:14 +0900185
Joonyoung Shim590f4182012-03-16 18:47:14 +0900186 void __iomem *regs;
Sean Paul77006a72013-01-16 10:17:20 -0500187 int irq;
Sean Paul724fd142014-05-09 15:05:10 +0900188 struct delayed_work hotplug_work;
Joonyoung Shim590f4182012-03-16 18:47:14 +0900189
Inki Dae8fa04aa2014-03-13 16:38:31 +0900190 struct i2c_adapter *ddc_adpt;
Joonyoung Shim590f4182012-03-16 18:47:14 +0900191 struct i2c_client *hdmiphy_port;
192
Rahul Sharma6b986ed2013-03-06 17:33:29 +0900193 /* current hdmiphy conf regs */
Rahul Sharmabfa48422014-04-03 20:41:04 +0530194 struct drm_display_mode current_mode;
Andrzej Hajdac93aaeb2015-07-09 16:28:10 +0200195 u8 cea_video_id;
196 union hdmi_conf_regs mode_conf;
Joonyoung Shim590f4182012-03-16 18:47:14 +0900197
198 struct hdmi_resources res;
Andrzej Hajdacd240cd2015-07-09 16:28:09 +0200199 const struct hdmi_driver_data *drv_data;
Joonyoung Shim7ecd34e2012-04-23 19:35:47 +0900200
Tomasz Stanislawskifca57122012-10-04 20:48:46 +0530201 int hpd_gpio;
Rahul Sharmad5e9ca42014-05-09 15:34:18 +0900202 void __iomem *regs_hdmiphy;
Rahul Sharma5a325072012-10-04 20:48:54 +0530203
Rahul Sharma049d34e2014-05-20 10:36:05 +0530204 struct regmap *pmureg;
Joonyoung Shim590f4182012-03-16 18:47:14 +0900205};
206
Andrzej Hajda0d8424f82014-11-17 09:54:21 +0100207static inline struct hdmi_context *display_to_hdmi(struct exynos_drm_display *d)
208{
209 return container_of(d, struct hdmi_context, display);
210}
211
Sean Paul2f7e2ed2013-01-15 08:11:08 -0500212struct hdmiphy_config {
213 int pixel_clock;
214 u8 conf[32];
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +0900215};
216
Rahul Sharma6b986ed2013-03-06 17:33:29 +0900217/* list of phy config settings */
218static const struct hdmiphy_config hdmiphy_v13_configs[] = {
219 {
220 .pixel_clock = 27000000,
221 .conf = {
222 0x01, 0x05, 0x00, 0xD8, 0x10, 0x1C, 0x30, 0x40,
223 0x6B, 0x10, 0x02, 0x51, 0xDF, 0xF2, 0x54, 0x87,
224 0x84, 0x00, 0x30, 0x38, 0x00, 0x08, 0x10, 0xE0,
225 0x22, 0x40, 0xE3, 0x26, 0x00, 0x00, 0x00, 0x00,
226 },
227 },
228 {
229 .pixel_clock = 27027000,
230 .conf = {
231 0x01, 0x05, 0x00, 0xD4, 0x10, 0x9C, 0x09, 0x64,
232 0x6B, 0x10, 0x02, 0x51, 0xDF, 0xF2, 0x54, 0x87,
233 0x84, 0x00, 0x30, 0x38, 0x00, 0x08, 0x10, 0xE0,
234 0x22, 0x40, 0xE3, 0x26, 0x00, 0x00, 0x00, 0x00,
235 },
236 },
237 {
238 .pixel_clock = 74176000,
239 .conf = {
240 0x01, 0x05, 0x00, 0xD8, 0x10, 0x9C, 0xef, 0x5B,
241 0x6D, 0x10, 0x01, 0x51, 0xef, 0xF3, 0x54, 0xb9,
242 0x84, 0x00, 0x30, 0x38, 0x00, 0x08, 0x10, 0xE0,
243 0x22, 0x40, 0xa5, 0x26, 0x01, 0x00, 0x00, 0x00,
244 },
245 },
246 {
247 .pixel_clock = 74250000,
248 .conf = {
249 0x01, 0x05, 0x00, 0xd8, 0x10, 0x9c, 0xf8, 0x40,
250 0x6a, 0x10, 0x01, 0x51, 0xff, 0xf1, 0x54, 0xba,
251 0x84, 0x00, 0x10, 0x38, 0x00, 0x08, 0x10, 0xe0,
252 0x22, 0x40, 0xa4, 0x26, 0x01, 0x00, 0x00, 0x00,
253 },
254 },
255 {
256 .pixel_clock = 148500000,
257 .conf = {
258 0x01, 0x05, 0x00, 0xD8, 0x10, 0x9C, 0xf8, 0x40,
259 0x6A, 0x18, 0x00, 0x51, 0xff, 0xF1, 0x54, 0xba,
260 0x84, 0x00, 0x10, 0x38, 0x00, 0x08, 0x10, 0xE0,
261 0x22, 0x40, 0xa4, 0x26, 0x02, 0x00, 0x00, 0x00,
262 },
263 },
264};
265
Sean Paul2f7e2ed2013-01-15 08:11:08 -0500266static const struct hdmiphy_config hdmiphy_v14_configs[] = {
267 {
268 .pixel_clock = 25200000,
269 .conf = {
270 0x01, 0x51, 0x2A, 0x75, 0x40, 0x01, 0x00, 0x08,
271 0x82, 0x80, 0xfc, 0xd8, 0x45, 0xa0, 0xac, 0x80,
272 0x08, 0x80, 0x11, 0x04, 0x02, 0x22, 0x44, 0x86,
273 0x54, 0xf4, 0x24, 0x00, 0x00, 0x00, 0x01, 0x80,
274 },
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +0900275 },
Sean Paul2f7e2ed2013-01-15 08:11:08 -0500276 {
277 .pixel_clock = 27000000,
278 .conf = {
279 0x01, 0xd1, 0x22, 0x51, 0x40, 0x08, 0xfc, 0x20,
280 0x98, 0xa0, 0xcb, 0xd8, 0x45, 0xa0, 0xac, 0x80,
281 0x06, 0x80, 0x11, 0x04, 0x02, 0x22, 0x44, 0x86,
282 0x54, 0xe4, 0x24, 0x00, 0x00, 0x00, 0x01, 0x80,
283 },
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +0900284 },
Sean Paul2f7e2ed2013-01-15 08:11:08 -0500285 {
286 .pixel_clock = 27027000,
287 .conf = {
288 0x01, 0xd1, 0x2d, 0x72, 0x40, 0x64, 0x12, 0x08,
289 0x43, 0xa0, 0x0e, 0xd9, 0x45, 0xa0, 0xac, 0x80,
290 0x08, 0x80, 0x11, 0x04, 0x02, 0x22, 0x44, 0x86,
291 0x54, 0xe3, 0x24, 0x00, 0x00, 0x00, 0x01, 0x00,
292 },
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +0900293 },
Sean Paul2f7e2ed2013-01-15 08:11:08 -0500294 {
295 .pixel_clock = 36000000,
296 .conf = {
297 0x01, 0x51, 0x2d, 0x55, 0x40, 0x01, 0x00, 0x08,
298 0x82, 0x80, 0x0e, 0xd9, 0x45, 0xa0, 0xac, 0x80,
299 0x08, 0x80, 0x11, 0x04, 0x02, 0x22, 0x44, 0x86,
300 0x54, 0xab, 0x24, 0x00, 0x00, 0x00, 0x01, 0x80,
301 },
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +0900302 },
Sean Paul2f7e2ed2013-01-15 08:11:08 -0500303 {
304 .pixel_clock = 40000000,
305 .conf = {
306 0x01, 0x51, 0x32, 0x55, 0x40, 0x01, 0x00, 0x08,
307 0x82, 0x80, 0x2c, 0xd9, 0x45, 0xa0, 0xac, 0x80,
308 0x08, 0x80, 0x11, 0x04, 0x02, 0x22, 0x44, 0x86,
309 0x54, 0x9a, 0x24, 0x00, 0x00, 0x00, 0x01, 0x80,
310 },
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +0900311 },
Sean Paul2f7e2ed2013-01-15 08:11:08 -0500312 {
313 .pixel_clock = 65000000,
314 .conf = {
315 0x01, 0xd1, 0x36, 0x34, 0x40, 0x1e, 0x0a, 0x08,
316 0x82, 0xa0, 0x45, 0xd9, 0x45, 0xa0, 0xac, 0x80,
317 0x08, 0x80, 0x11, 0x04, 0x02, 0x22, 0x44, 0x86,
318 0x54, 0xbd, 0x24, 0x01, 0x00, 0x00, 0x01, 0x80,
319 },
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +0900320 },
Sean Paul2f7e2ed2013-01-15 08:11:08 -0500321 {
Shirish Se1d883c2014-03-13 14:28:27 +0900322 .pixel_clock = 71000000,
323 .conf = {
Shirish S96d26532014-05-05 10:27:51 +0530324 0x01, 0xd1, 0x3b, 0x35, 0x40, 0x0c, 0x04, 0x08,
325 0x85, 0xa0, 0x63, 0xd9, 0x45, 0xa0, 0xac, 0x80,
326 0x08, 0x80, 0x11, 0x04, 0x02, 0x22, 0x44, 0x86,
Shirish Se1d883c2014-03-13 14:28:27 +0900327 0x54, 0xad, 0x24, 0x01, 0x00, 0x00, 0x01, 0x80,
328 },
329 },
330 {
331 .pixel_clock = 73250000,
332 .conf = {
Shirish S96d26532014-05-05 10:27:51 +0530333 0x01, 0xd1, 0x3d, 0x35, 0x40, 0x18, 0x02, 0x08,
334 0x83, 0xa0, 0x6e, 0xd9, 0x45, 0xa0, 0xac, 0x80,
335 0x08, 0x80, 0x11, 0x04, 0x02, 0x22, 0x44, 0x86,
Shirish Se1d883c2014-03-13 14:28:27 +0900336 0x54, 0xa8, 0x24, 0x01, 0x00, 0x00, 0x01, 0x80,
337 },
338 },
339 {
Sean Paul2f7e2ed2013-01-15 08:11:08 -0500340 .pixel_clock = 74176000,
341 .conf = {
342 0x01, 0xd1, 0x3e, 0x35, 0x40, 0x5b, 0xde, 0x08,
343 0x82, 0xa0, 0x73, 0xd9, 0x45, 0xa0, 0xac, 0x80,
344 0x56, 0x80, 0x11, 0x04, 0x02, 0x22, 0x44, 0x86,
345 0x54, 0xa6, 0x24, 0x01, 0x00, 0x00, 0x01, 0x80,
346 },
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +0900347 },
Sean Paul2f7e2ed2013-01-15 08:11:08 -0500348 {
349 .pixel_clock = 74250000,
350 .conf = {
351 0x01, 0xd1, 0x1f, 0x10, 0x40, 0x40, 0xf8, 0x08,
352 0x81, 0xa0, 0xba, 0xd8, 0x45, 0xa0, 0xac, 0x80,
353 0x3c, 0x80, 0x11, 0x04, 0x02, 0x22, 0x44, 0x86,
354 0x54, 0xa5, 0x24, 0x01, 0x00, 0x00, 0x01, 0x00,
355 },
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +0900356 },
Sean Paul2f7e2ed2013-01-15 08:11:08 -0500357 {
358 .pixel_clock = 83500000,
359 .conf = {
360 0x01, 0xd1, 0x23, 0x11, 0x40, 0x0c, 0xfb, 0x08,
361 0x85, 0xa0, 0xd1, 0xd8, 0x45, 0xa0, 0xac, 0x80,
362 0x08, 0x80, 0x11, 0x04, 0x02, 0x22, 0x44, 0x86,
363 0x54, 0x93, 0x24, 0x01, 0x00, 0x00, 0x01, 0x80,
364 },
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +0900365 },
Sean Paul2f7e2ed2013-01-15 08:11:08 -0500366 {
367 .pixel_clock = 106500000,
368 .conf = {
369 0x01, 0xd1, 0x2c, 0x12, 0x40, 0x0c, 0x09, 0x08,
370 0x84, 0xa0, 0x0a, 0xd9, 0x45, 0xa0, 0xac, 0x80,
371 0x08, 0x80, 0x11, 0x04, 0x02, 0x22, 0x44, 0x86,
372 0x54, 0x73, 0x24, 0x01, 0x00, 0x00, 0x01, 0x80,
373 },
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +0900374 },
Sean Paul2f7e2ed2013-01-15 08:11:08 -0500375 {
376 .pixel_clock = 108000000,
377 .conf = {
378 0x01, 0x51, 0x2d, 0x15, 0x40, 0x01, 0x00, 0x08,
379 0x82, 0x80, 0x0e, 0xd9, 0x45, 0xa0, 0xac, 0x80,
380 0x08, 0x80, 0x11, 0x04, 0x02, 0x22, 0x44, 0x86,
381 0x54, 0xc7, 0x25, 0x03, 0x00, 0x00, 0x01, 0x80,
382 },
Seung-Woo Kime540adf2012-04-24 17:55:06 +0900383 },
Sean Paul2f7e2ed2013-01-15 08:11:08 -0500384 {
Shirish Se1d883c2014-03-13 14:28:27 +0900385 .pixel_clock = 115500000,
386 .conf = {
Shirish S96d26532014-05-05 10:27:51 +0530387 0x01, 0xd1, 0x30, 0x12, 0x40, 0x40, 0x10, 0x08,
388 0x80, 0x80, 0x21, 0xd9, 0x45, 0xa0, 0xac, 0x80,
389 0x08, 0x80, 0x11, 0x04, 0x02, 0x22, 0x44, 0x86,
Shirish Se1d883c2014-03-13 14:28:27 +0900390 0x54, 0xaa, 0x25, 0x03, 0x00, 0x00, 0x01, 0x80,
391 },
392 },
393 {
394 .pixel_clock = 119000000,
395 .conf = {
Shirish S96d26532014-05-05 10:27:51 +0530396 0x01, 0xd1, 0x32, 0x1a, 0x40, 0x30, 0xd8, 0x08,
397 0x04, 0xa0, 0x2a, 0xd9, 0x45, 0xa0, 0xac, 0x80,
398 0x08, 0x80, 0x11, 0x04, 0x02, 0x22, 0x44, 0x86,
Shirish Se1d883c2014-03-13 14:28:27 +0900399 0x54, 0x9d, 0x25, 0x03, 0x00, 0x00, 0x01, 0x80,
400 },
401 },
402 {
Sean Paul2f7e2ed2013-01-15 08:11:08 -0500403 .pixel_clock = 146250000,
404 .conf = {
405 0x01, 0xd1, 0x3d, 0x15, 0x40, 0x18, 0xfd, 0x08,
406 0x83, 0xa0, 0x6e, 0xd9, 0x45, 0xa0, 0xac, 0x80,
407 0x08, 0x80, 0x11, 0x04, 0x02, 0x22, 0x44, 0x86,
408 0x54, 0x50, 0x25, 0x03, 0x00, 0x00, 0x01, 0x80,
409 },
Seung-Woo Kime540adf2012-04-24 17:55:06 +0900410 },
Sean Paul2f7e2ed2013-01-15 08:11:08 -0500411 {
412 .pixel_clock = 148500000,
413 .conf = {
414 0x01, 0xd1, 0x1f, 0x00, 0x40, 0x40, 0xf8, 0x08,
415 0x81, 0xa0, 0xba, 0xd8, 0x45, 0xa0, 0xac, 0x80,
416 0x3c, 0x80, 0x11, 0x04, 0x02, 0x22, 0x44, 0x86,
417 0x54, 0x4b, 0x25, 0x03, 0x00, 0x00, 0x01, 0x00,
418 },
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +0900419 },
Seung-Woo Kimd8408322011-12-21 17:39:39 +0900420};
421
Rahul Sharmaa18a2dd2014-04-20 15:51:17 +0530422static const struct hdmiphy_config hdmiphy_5420_configs[] = {
423 {
424 .pixel_clock = 25200000,
425 .conf = {
426 0x01, 0x52, 0x3F, 0x55, 0x40, 0x01, 0x00, 0xC8,
427 0x82, 0xC8, 0xBD, 0xD8, 0x45, 0xA0, 0xAC, 0x80,
428 0x06, 0x80, 0x01, 0x84, 0x05, 0x02, 0x24, 0x66,
429 0x54, 0xF4, 0x24, 0x00, 0x00, 0x00, 0x01, 0x80,
430 },
431 },
432 {
433 .pixel_clock = 27000000,
434 .conf = {
435 0x01, 0xD1, 0x22, 0x51, 0x40, 0x08, 0xFC, 0xE0,
436 0x98, 0xE8, 0xCB, 0xD8, 0x45, 0xA0, 0xAC, 0x80,
437 0x06, 0x80, 0x09, 0x84, 0x05, 0x02, 0x24, 0x66,
438 0x54, 0xE4, 0x24, 0x00, 0x00, 0x00, 0x01, 0x80,
439 },
440 },
441 {
442 .pixel_clock = 27027000,
443 .conf = {
444 0x01, 0xD1, 0x2D, 0x72, 0x40, 0x64, 0x12, 0xC8,
445 0x43, 0xE8, 0x0E, 0xD9, 0x45, 0xA0, 0xAC, 0x80,
446 0x26, 0x80, 0x09, 0x84, 0x05, 0x02, 0x24, 0x66,
447 0x54, 0xE3, 0x24, 0x00, 0x00, 0x00, 0x01, 0x80,
448 },
449 },
450 {
451 .pixel_clock = 36000000,
452 .conf = {
453 0x01, 0x51, 0x2D, 0x55, 0x40, 0x40, 0x00, 0xC8,
454 0x02, 0xC8, 0x0E, 0xD9, 0x45, 0xA0, 0xAC, 0x80,
455 0x08, 0x80, 0x09, 0x84, 0x05, 0x02, 0x24, 0x66,
456 0x54, 0xAB, 0x24, 0x00, 0x00, 0x00, 0x01, 0x80,
457 },
458 },
459 {
460 .pixel_clock = 40000000,
461 .conf = {
462 0x01, 0xD1, 0x21, 0x31, 0x40, 0x3C, 0x28, 0xC8,
463 0x87, 0xE8, 0xC8, 0xD8, 0x45, 0xA0, 0xAC, 0x80,
464 0x08, 0x80, 0x09, 0x84, 0x05, 0x02, 0x24, 0x66,
465 0x54, 0x9A, 0x24, 0x00, 0x00, 0x00, 0x01, 0x80,
466 },
467 },
468 {
469 .pixel_clock = 65000000,
470 .conf = {
471 0x01, 0xD1, 0x36, 0x34, 0x40, 0x0C, 0x04, 0xC8,
472 0x82, 0xE8, 0x45, 0xD9, 0x45, 0xA0, 0xAC, 0x80,
473 0x08, 0x80, 0x09, 0x84, 0x05, 0x02, 0x24, 0x66,
474 0x54, 0xBD, 0x24, 0x01, 0x00, 0x00, 0x01, 0x80,
475 },
476 },
477 {
478 .pixel_clock = 71000000,
479 .conf = {
480 0x01, 0xD1, 0x3B, 0x35, 0x40, 0x0C, 0x04, 0xC8,
481 0x85, 0xE8, 0x63, 0xD9, 0x45, 0xA0, 0xAC, 0x80,
482 0x08, 0x80, 0x09, 0x84, 0x05, 0x02, 0x24, 0x66,
483 0x54, 0x57, 0x24, 0x00, 0x00, 0x00, 0x01, 0x80,
484 },
485 },
486 {
487 .pixel_clock = 73250000,
488 .conf = {
489 0x01, 0xD1, 0x1F, 0x10, 0x40, 0x78, 0x8D, 0xC8,
490 0x81, 0xE8, 0xB7, 0xD8, 0x45, 0xA0, 0xAC, 0x80,
491 0x56, 0x80, 0x09, 0x84, 0x05, 0x02, 0x24, 0x66,
492 0x54, 0xA8, 0x24, 0x01, 0x00, 0x00, 0x01, 0x80,
493 },
494 },
495 {
496 .pixel_clock = 74176000,
497 .conf = {
498 0x01, 0xD1, 0x1F, 0x10, 0x40, 0x5B, 0xEF, 0xC8,
499 0x81, 0xE8, 0xB9, 0xD8, 0x45, 0xA0, 0xAC, 0x80,
500 0x56, 0x80, 0x09, 0x84, 0x05, 0x02, 0x24, 0x66,
501 0x54, 0xA6, 0x24, 0x01, 0x00, 0x00, 0x01, 0x80,
502 },
503 },
504 {
505 .pixel_clock = 74250000,
506 .conf = {
507 0x01, 0xD1, 0x1F, 0x10, 0x40, 0x40, 0xF8, 0x08,
508 0x81, 0xE8, 0xBA, 0xD8, 0x45, 0xA0, 0xAC, 0x80,
509 0x26, 0x80, 0x09, 0x84, 0x05, 0x22, 0x24, 0x66,
510 0x54, 0xA5, 0x24, 0x01, 0x00, 0x00, 0x01, 0x80,
511 },
512 },
513 {
514 .pixel_clock = 83500000,
515 .conf = {
516 0x01, 0xD1, 0x23, 0x11, 0x40, 0x0C, 0xFB, 0xC8,
517 0x85, 0xE8, 0xD1, 0xD8, 0x45, 0xA0, 0xAC, 0x80,
518 0x08, 0x80, 0x09, 0x84, 0x05, 0x02, 0x24, 0x66,
519 0x54, 0x4A, 0x24, 0x00, 0x00, 0x00, 0x01, 0x80,
520 },
521 },
522 {
523 .pixel_clock = 88750000,
524 .conf = {
525 0x01, 0xD1, 0x25, 0x11, 0x40, 0x18, 0xFF, 0xC8,
526 0x83, 0xE8, 0xDE, 0xD8, 0x45, 0xA0, 0xAC, 0x80,
527 0x08, 0x80, 0x09, 0x84, 0x05, 0x02, 0x24, 0x66,
528 0x54, 0x45, 0x24, 0x00, 0x00, 0x00, 0x01, 0x80,
529 },
530 },
531 {
532 .pixel_clock = 106500000,
533 .conf = {
534 0x01, 0xD1, 0x2C, 0x12, 0x40, 0x0C, 0x09, 0xC8,
535 0x84, 0xE8, 0x0A, 0xD9, 0x45, 0xA0, 0xAC, 0x80,
536 0x08, 0x80, 0x09, 0x84, 0x05, 0x02, 0x24, 0x66,
537 0x54, 0x73, 0x24, 0x01, 0x00, 0x00, 0x01, 0x80,
538 },
539 },
540 {
541 .pixel_clock = 108000000,
542 .conf = {
543 0x01, 0x51, 0x2D, 0x15, 0x40, 0x01, 0x00, 0xC8,
544 0x82, 0xC8, 0x0E, 0xD9, 0x45, 0xA0, 0xAC, 0x80,
545 0x08, 0x80, 0x09, 0x84, 0x05, 0x02, 0x24, 0x66,
546 0x54, 0xC7, 0x25, 0x03, 0x00, 0x00, 0x01, 0x80,
547 },
548 },
549 {
550 .pixel_clock = 115500000,
551 .conf = {
552 0x01, 0xD1, 0x30, 0x14, 0x40, 0x0C, 0x03, 0xC8,
553 0x88, 0xE8, 0x21, 0xD9, 0x45, 0xA0, 0xAC, 0x80,
554 0x08, 0x80, 0x09, 0x84, 0x05, 0x02, 0x24, 0x66,
555 0x54, 0x6A, 0x24, 0x01, 0x00, 0x00, 0x01, 0x80,
556 },
557 },
558 {
559 .pixel_clock = 146250000,
560 .conf = {
561 0x01, 0xD1, 0x3D, 0x15, 0x40, 0x18, 0xFD, 0xC8,
562 0x83, 0xE8, 0x6E, 0xD9, 0x45, 0xA0, 0xAC, 0x80,
563 0x08, 0x80, 0x09, 0x84, 0x05, 0x02, 0x24, 0x66,
564 0x54, 0x54, 0x24, 0x01, 0x00, 0x00, 0x01, 0x80,
565 },
566 },
567 {
568 .pixel_clock = 148500000,
569 .conf = {
570 0x01, 0xD1, 0x1F, 0x00, 0x40, 0x40, 0xF8, 0x08,
571 0x81, 0xE8, 0xBA, 0xD8, 0x45, 0xA0, 0xAC, 0x80,
572 0x26, 0x80, 0x09, 0x84, 0x05, 0x22, 0x24, 0x66,
573 0x54, 0x4B, 0x25, 0x03, 0x00, 0x80, 0x01, 0x80,
574 },
575 },
576};
577
Sachin Kamat16337072014-05-22 10:32:56 +0530578static struct hdmi_driver_data exynos5420_hdmi_driver_data = {
Rahul Sharmaa18a2dd2014-04-20 15:51:17 +0530579 .type = HDMI_TYPE14,
580 .phy_confs = hdmiphy_5420_configs,
581 .phy_conf_count = ARRAY_SIZE(hdmiphy_5420_configs),
582 .is_apb_phy = 1,
583};
Rahul Sharmad5e9ca42014-05-09 15:34:18 +0900584
Sachin Kamat16337072014-05-22 10:32:56 +0530585static struct hdmi_driver_data exynos4212_hdmi_driver_data = {
Rahul Sharmad5e9ca42014-05-09 15:34:18 +0900586 .type = HDMI_TYPE14,
587 .phy_confs = hdmiphy_v14_configs,
588 .phy_conf_count = ARRAY_SIZE(hdmiphy_v14_configs),
589 .is_apb_phy = 0,
590};
591
Marek Szyprowskiff830c92014-07-01 10:10:07 +0200592static struct hdmi_driver_data exynos4210_hdmi_driver_data = {
593 .type = HDMI_TYPE13,
594 .phy_confs = hdmiphy_v13_configs,
595 .phy_conf_count = ARRAY_SIZE(hdmiphy_v13_configs),
596 .is_apb_phy = 0,
597};
598
Sachin Kamat16337072014-05-22 10:32:56 +0530599static struct hdmi_driver_data exynos5_hdmi_driver_data = {
Rahul Sharmad5e9ca42014-05-09 15:34:18 +0900600 .type = HDMI_TYPE14,
601 .phy_confs = hdmiphy_v13_configs,
602 .phy_conf_count = ARRAY_SIZE(hdmiphy_v13_configs),
603 .is_apb_phy = 0,
604};
605
Seung-Woo Kimd8408322011-12-21 17:39:39 +0900606static inline u32 hdmi_reg_read(struct hdmi_context *hdata, u32 reg_id)
607{
608 return readl(hdata->regs + reg_id);
609}
610
611static inline void hdmi_reg_writeb(struct hdmi_context *hdata,
612 u32 reg_id, u8 value)
613{
614 writeb(value, hdata->regs + reg_id);
615}
616
617static inline void hdmi_reg_writemask(struct hdmi_context *hdata,
618 u32 reg_id, u32 value, u32 mask)
619{
620 u32 old = readl(hdata->regs + reg_id);
621 value = (value & mask) | (old & ~mask);
622 writel(value, hdata->regs + reg_id);
623}
624
Rahul Sharmad5e9ca42014-05-09 15:34:18 +0900625static int hdmiphy_reg_writeb(struct hdmi_context *hdata,
626 u32 reg_offset, u8 value)
627{
628 if (hdata->hdmiphy_port) {
629 u8 buffer[2];
630 int ret;
631
632 buffer[0] = reg_offset;
633 buffer[1] = value;
634
635 ret = i2c_master_send(hdata->hdmiphy_port, buffer, 2);
636 if (ret == 2)
637 return 0;
638 return ret;
639 } else {
640 writeb(value, hdata->regs_hdmiphy + (reg_offset<<2));
641 return 0;
642 }
643}
644
645static int hdmiphy_reg_write_buf(struct hdmi_context *hdata,
646 u32 reg_offset, const u8 *buf, u32 len)
647{
648 if ((reg_offset + len) > 32)
649 return -EINVAL;
650
651 if (hdata->hdmiphy_port) {
652 int ret;
653
654 ret = i2c_master_send(hdata->hdmiphy_port, buf, len);
655 if (ret == len)
656 return 0;
657 return ret;
658 } else {
659 int i;
660 for (i = 0; i < len; i++)
661 writeb(buf[i], hdata->regs_hdmiphy +
662 ((reg_offset + i)<<2));
663 return 0;
664 }
665}
666
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +0900667static void hdmi_v13_regs_dump(struct hdmi_context *hdata, char *prefix)
Seung-Woo Kimd8408322011-12-21 17:39:39 +0900668{
669#define DUMPREG(reg_id) \
670 DRM_DEBUG_KMS("%s:" #reg_id " = %08x\n", prefix, \
671 readl(hdata->regs + reg_id))
672 DRM_DEBUG_KMS("%s: ---- CONTROL REGISTERS ----\n", prefix);
673 DUMPREG(HDMI_INTC_FLAG);
674 DUMPREG(HDMI_INTC_CON);
675 DUMPREG(HDMI_HPD_STATUS);
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +0900676 DUMPREG(HDMI_V13_PHY_RSTOUT);
677 DUMPREG(HDMI_V13_PHY_VPLL);
678 DUMPREG(HDMI_V13_PHY_CMU);
679 DUMPREG(HDMI_V13_CORE_RSTOUT);
Seung-Woo Kimd8408322011-12-21 17:39:39 +0900680
681 DRM_DEBUG_KMS("%s: ---- CORE REGISTERS ----\n", prefix);
682 DUMPREG(HDMI_CON_0);
683 DUMPREG(HDMI_CON_1);
684 DUMPREG(HDMI_CON_2);
685 DUMPREG(HDMI_SYS_STATUS);
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +0900686 DUMPREG(HDMI_V13_PHY_STATUS);
Seung-Woo Kimd8408322011-12-21 17:39:39 +0900687 DUMPREG(HDMI_STATUS_EN);
688 DUMPREG(HDMI_HPD);
689 DUMPREG(HDMI_MODE_SEL);
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +0900690 DUMPREG(HDMI_V13_HPD_GEN);
691 DUMPREG(HDMI_V13_DC_CONTROL);
692 DUMPREG(HDMI_V13_VIDEO_PATTERN_GEN);
Seung-Woo Kimd8408322011-12-21 17:39:39 +0900693
694 DRM_DEBUG_KMS("%s: ---- CORE SYNC REGISTERS ----\n", prefix);
695 DUMPREG(HDMI_H_BLANK_0);
696 DUMPREG(HDMI_H_BLANK_1);
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +0900697 DUMPREG(HDMI_V13_V_BLANK_0);
698 DUMPREG(HDMI_V13_V_BLANK_1);
699 DUMPREG(HDMI_V13_V_BLANK_2);
700 DUMPREG(HDMI_V13_H_V_LINE_0);
701 DUMPREG(HDMI_V13_H_V_LINE_1);
702 DUMPREG(HDMI_V13_H_V_LINE_2);
Seung-Woo Kimd8408322011-12-21 17:39:39 +0900703 DUMPREG(HDMI_VSYNC_POL);
704 DUMPREG(HDMI_INT_PRO_MODE);
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +0900705 DUMPREG(HDMI_V13_V_BLANK_F_0);
706 DUMPREG(HDMI_V13_V_BLANK_F_1);
707 DUMPREG(HDMI_V13_V_BLANK_F_2);
708 DUMPREG(HDMI_V13_H_SYNC_GEN_0);
709 DUMPREG(HDMI_V13_H_SYNC_GEN_1);
710 DUMPREG(HDMI_V13_H_SYNC_GEN_2);
711 DUMPREG(HDMI_V13_V_SYNC_GEN_1_0);
712 DUMPREG(HDMI_V13_V_SYNC_GEN_1_1);
713 DUMPREG(HDMI_V13_V_SYNC_GEN_1_2);
714 DUMPREG(HDMI_V13_V_SYNC_GEN_2_0);
715 DUMPREG(HDMI_V13_V_SYNC_GEN_2_1);
716 DUMPREG(HDMI_V13_V_SYNC_GEN_2_2);
717 DUMPREG(HDMI_V13_V_SYNC_GEN_3_0);
718 DUMPREG(HDMI_V13_V_SYNC_GEN_3_1);
719 DUMPREG(HDMI_V13_V_SYNC_GEN_3_2);
Seung-Woo Kimd8408322011-12-21 17:39:39 +0900720
721 DRM_DEBUG_KMS("%s: ---- TG REGISTERS ----\n", prefix);
722 DUMPREG(HDMI_TG_CMD);
723 DUMPREG(HDMI_TG_H_FSZ_L);
724 DUMPREG(HDMI_TG_H_FSZ_H);
725 DUMPREG(HDMI_TG_HACT_ST_L);
726 DUMPREG(HDMI_TG_HACT_ST_H);
727 DUMPREG(HDMI_TG_HACT_SZ_L);
728 DUMPREG(HDMI_TG_HACT_SZ_H);
729 DUMPREG(HDMI_TG_V_FSZ_L);
730 DUMPREG(HDMI_TG_V_FSZ_H);
731 DUMPREG(HDMI_TG_VSYNC_L);
732 DUMPREG(HDMI_TG_VSYNC_H);
733 DUMPREG(HDMI_TG_VSYNC2_L);
734 DUMPREG(HDMI_TG_VSYNC2_H);
735 DUMPREG(HDMI_TG_VACT_ST_L);
736 DUMPREG(HDMI_TG_VACT_ST_H);
737 DUMPREG(HDMI_TG_VACT_SZ_L);
738 DUMPREG(HDMI_TG_VACT_SZ_H);
739 DUMPREG(HDMI_TG_FIELD_CHG_L);
740 DUMPREG(HDMI_TG_FIELD_CHG_H);
741 DUMPREG(HDMI_TG_VACT_ST2_L);
742 DUMPREG(HDMI_TG_VACT_ST2_H);
743 DUMPREG(HDMI_TG_VSYNC_TOP_HDMI_L);
744 DUMPREG(HDMI_TG_VSYNC_TOP_HDMI_H);
745 DUMPREG(HDMI_TG_VSYNC_BOT_HDMI_L);
746 DUMPREG(HDMI_TG_VSYNC_BOT_HDMI_H);
747 DUMPREG(HDMI_TG_FIELD_TOP_HDMI_L);
748 DUMPREG(HDMI_TG_FIELD_TOP_HDMI_H);
749 DUMPREG(HDMI_TG_FIELD_BOT_HDMI_L);
750 DUMPREG(HDMI_TG_FIELD_BOT_HDMI_H);
751#undef DUMPREG
752}
753
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +0900754static void hdmi_v14_regs_dump(struct hdmi_context *hdata, char *prefix)
755{
756 int i;
757
758#define DUMPREG(reg_id) \
759 DRM_DEBUG_KMS("%s:" #reg_id " = %08x\n", prefix, \
760 readl(hdata->regs + reg_id))
761
762 DRM_DEBUG_KMS("%s: ---- CONTROL REGISTERS ----\n", prefix);
763 DUMPREG(HDMI_INTC_CON);
764 DUMPREG(HDMI_INTC_FLAG);
765 DUMPREG(HDMI_HPD_STATUS);
766 DUMPREG(HDMI_INTC_CON_1);
767 DUMPREG(HDMI_INTC_FLAG_1);
768 DUMPREG(HDMI_PHY_STATUS_0);
769 DUMPREG(HDMI_PHY_STATUS_PLL);
770 DUMPREG(HDMI_PHY_CON_0);
771 DUMPREG(HDMI_PHY_RSTOUT);
772 DUMPREG(HDMI_PHY_VPLL);
773 DUMPREG(HDMI_PHY_CMU);
774 DUMPREG(HDMI_CORE_RSTOUT);
775
776 DRM_DEBUG_KMS("%s: ---- CORE REGISTERS ----\n", prefix);
777 DUMPREG(HDMI_CON_0);
778 DUMPREG(HDMI_CON_1);
779 DUMPREG(HDMI_CON_2);
780 DUMPREG(HDMI_SYS_STATUS);
781 DUMPREG(HDMI_PHY_STATUS_0);
782 DUMPREG(HDMI_STATUS_EN);
783 DUMPREG(HDMI_HPD);
784 DUMPREG(HDMI_MODE_SEL);
785 DUMPREG(HDMI_ENC_EN);
786 DUMPREG(HDMI_DC_CONTROL);
787 DUMPREG(HDMI_VIDEO_PATTERN_GEN);
788
789 DRM_DEBUG_KMS("%s: ---- CORE SYNC REGISTERS ----\n", prefix);
790 DUMPREG(HDMI_H_BLANK_0);
791 DUMPREG(HDMI_H_BLANK_1);
792 DUMPREG(HDMI_V2_BLANK_0);
793 DUMPREG(HDMI_V2_BLANK_1);
794 DUMPREG(HDMI_V1_BLANK_0);
795 DUMPREG(HDMI_V1_BLANK_1);
796 DUMPREG(HDMI_V_LINE_0);
797 DUMPREG(HDMI_V_LINE_1);
798 DUMPREG(HDMI_H_LINE_0);
799 DUMPREG(HDMI_H_LINE_1);
800 DUMPREG(HDMI_HSYNC_POL);
801
802 DUMPREG(HDMI_VSYNC_POL);
803 DUMPREG(HDMI_INT_PRO_MODE);
804 DUMPREG(HDMI_V_BLANK_F0_0);
805 DUMPREG(HDMI_V_BLANK_F0_1);
806 DUMPREG(HDMI_V_BLANK_F1_0);
807 DUMPREG(HDMI_V_BLANK_F1_1);
808
809 DUMPREG(HDMI_H_SYNC_START_0);
810 DUMPREG(HDMI_H_SYNC_START_1);
811 DUMPREG(HDMI_H_SYNC_END_0);
812 DUMPREG(HDMI_H_SYNC_END_1);
813
814 DUMPREG(HDMI_V_SYNC_LINE_BEF_2_0);
815 DUMPREG(HDMI_V_SYNC_LINE_BEF_2_1);
816 DUMPREG(HDMI_V_SYNC_LINE_BEF_1_0);
817 DUMPREG(HDMI_V_SYNC_LINE_BEF_1_1);
818
819 DUMPREG(HDMI_V_SYNC_LINE_AFT_2_0);
820 DUMPREG(HDMI_V_SYNC_LINE_AFT_2_1);
821 DUMPREG(HDMI_V_SYNC_LINE_AFT_1_0);
822 DUMPREG(HDMI_V_SYNC_LINE_AFT_1_1);
823
824 DUMPREG(HDMI_V_SYNC_LINE_AFT_PXL_2_0);
825 DUMPREG(HDMI_V_SYNC_LINE_AFT_PXL_2_1);
826 DUMPREG(HDMI_V_SYNC_LINE_AFT_PXL_1_0);
827 DUMPREG(HDMI_V_SYNC_LINE_AFT_PXL_1_1);
828
829 DUMPREG(HDMI_V_BLANK_F2_0);
830 DUMPREG(HDMI_V_BLANK_F2_1);
831 DUMPREG(HDMI_V_BLANK_F3_0);
832 DUMPREG(HDMI_V_BLANK_F3_1);
833 DUMPREG(HDMI_V_BLANK_F4_0);
834 DUMPREG(HDMI_V_BLANK_F4_1);
835 DUMPREG(HDMI_V_BLANK_F5_0);
836 DUMPREG(HDMI_V_BLANK_F5_1);
837
838 DUMPREG(HDMI_V_SYNC_LINE_AFT_3_0);
839 DUMPREG(HDMI_V_SYNC_LINE_AFT_3_1);
840 DUMPREG(HDMI_V_SYNC_LINE_AFT_4_0);
841 DUMPREG(HDMI_V_SYNC_LINE_AFT_4_1);
842 DUMPREG(HDMI_V_SYNC_LINE_AFT_5_0);
843 DUMPREG(HDMI_V_SYNC_LINE_AFT_5_1);
844 DUMPREG(HDMI_V_SYNC_LINE_AFT_6_0);
845 DUMPREG(HDMI_V_SYNC_LINE_AFT_6_1);
846
847 DUMPREG(HDMI_V_SYNC_LINE_AFT_PXL_3_0);
848 DUMPREG(HDMI_V_SYNC_LINE_AFT_PXL_3_1);
849 DUMPREG(HDMI_V_SYNC_LINE_AFT_PXL_4_0);
850 DUMPREG(HDMI_V_SYNC_LINE_AFT_PXL_4_1);
851 DUMPREG(HDMI_V_SYNC_LINE_AFT_PXL_5_0);
852 DUMPREG(HDMI_V_SYNC_LINE_AFT_PXL_5_1);
853 DUMPREG(HDMI_V_SYNC_LINE_AFT_PXL_6_0);
854 DUMPREG(HDMI_V_SYNC_LINE_AFT_PXL_6_1);
855
856 DUMPREG(HDMI_VACT_SPACE_1_0);
857 DUMPREG(HDMI_VACT_SPACE_1_1);
858 DUMPREG(HDMI_VACT_SPACE_2_0);
859 DUMPREG(HDMI_VACT_SPACE_2_1);
860 DUMPREG(HDMI_VACT_SPACE_3_0);
861 DUMPREG(HDMI_VACT_SPACE_3_1);
862 DUMPREG(HDMI_VACT_SPACE_4_0);
863 DUMPREG(HDMI_VACT_SPACE_4_1);
864 DUMPREG(HDMI_VACT_SPACE_5_0);
865 DUMPREG(HDMI_VACT_SPACE_5_1);
866 DUMPREG(HDMI_VACT_SPACE_6_0);
867 DUMPREG(HDMI_VACT_SPACE_6_1);
868
869 DRM_DEBUG_KMS("%s: ---- TG REGISTERS ----\n", prefix);
870 DUMPREG(HDMI_TG_CMD);
871 DUMPREG(HDMI_TG_H_FSZ_L);
872 DUMPREG(HDMI_TG_H_FSZ_H);
873 DUMPREG(HDMI_TG_HACT_ST_L);
874 DUMPREG(HDMI_TG_HACT_ST_H);
875 DUMPREG(HDMI_TG_HACT_SZ_L);
876 DUMPREG(HDMI_TG_HACT_SZ_H);
877 DUMPREG(HDMI_TG_V_FSZ_L);
878 DUMPREG(HDMI_TG_V_FSZ_H);
879 DUMPREG(HDMI_TG_VSYNC_L);
880 DUMPREG(HDMI_TG_VSYNC_H);
881 DUMPREG(HDMI_TG_VSYNC2_L);
882 DUMPREG(HDMI_TG_VSYNC2_H);
883 DUMPREG(HDMI_TG_VACT_ST_L);
884 DUMPREG(HDMI_TG_VACT_ST_H);
885 DUMPREG(HDMI_TG_VACT_SZ_L);
886 DUMPREG(HDMI_TG_VACT_SZ_H);
887 DUMPREG(HDMI_TG_FIELD_CHG_L);
888 DUMPREG(HDMI_TG_FIELD_CHG_H);
889 DUMPREG(HDMI_TG_VACT_ST2_L);
890 DUMPREG(HDMI_TG_VACT_ST2_H);
891 DUMPREG(HDMI_TG_VACT_ST3_L);
892 DUMPREG(HDMI_TG_VACT_ST3_H);
893 DUMPREG(HDMI_TG_VACT_ST4_L);
894 DUMPREG(HDMI_TG_VACT_ST4_H);
895 DUMPREG(HDMI_TG_VSYNC_TOP_HDMI_L);
896 DUMPREG(HDMI_TG_VSYNC_TOP_HDMI_H);
897 DUMPREG(HDMI_TG_VSYNC_BOT_HDMI_L);
898 DUMPREG(HDMI_TG_VSYNC_BOT_HDMI_H);
899 DUMPREG(HDMI_TG_FIELD_TOP_HDMI_L);
900 DUMPREG(HDMI_TG_FIELD_TOP_HDMI_H);
901 DUMPREG(HDMI_TG_FIELD_BOT_HDMI_L);
902 DUMPREG(HDMI_TG_FIELD_BOT_HDMI_H);
903 DUMPREG(HDMI_TG_3D);
904
905 DRM_DEBUG_KMS("%s: ---- PACKET REGISTERS ----\n", prefix);
906 DUMPREG(HDMI_AVI_CON);
907 DUMPREG(HDMI_AVI_HEADER0);
908 DUMPREG(HDMI_AVI_HEADER1);
909 DUMPREG(HDMI_AVI_HEADER2);
910 DUMPREG(HDMI_AVI_CHECK_SUM);
911 DUMPREG(HDMI_VSI_CON);
912 DUMPREG(HDMI_VSI_HEADER0);
913 DUMPREG(HDMI_VSI_HEADER1);
914 DUMPREG(HDMI_VSI_HEADER2);
915 for (i = 0; i < 7; ++i)
916 DUMPREG(HDMI_VSI_DATA(i));
917
918#undef DUMPREG
919}
920
921static void hdmi_regs_dump(struct hdmi_context *hdata, char *prefix)
922{
Andrzej Hajdacd240cd2015-07-09 16:28:09 +0200923 if (hdata->drv_data->type == HDMI_TYPE13)
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +0900924 hdmi_v13_regs_dump(hdata, prefix);
925 else
926 hdmi_v14_regs_dump(hdata, prefix);
927}
928
Rahul Sharmaa144c2e2012-11-26 10:52:57 +0530929static u8 hdmi_chksum(struct hdmi_context *hdata,
930 u32 start, u8 len, u32 hdr_sum)
931{
932 int i;
933
934 /* hdr_sum : header0 + header1 + header2
935 * start : start address of packet byte1
936 * len : packet bytes - 1 */
937 for (i = 0; i < len; ++i)
938 hdr_sum += 0xff & hdmi_reg_read(hdata, start + i * 4);
939
940 /* return 2's complement of 8 bit hdr_sum */
941 return (u8)(~(hdr_sum & 0xff) + 1);
942}
943
944static void hdmi_reg_infoframe(struct hdmi_context *hdata,
Sachin Kamatd34d59b2014-02-04 08:40:18 +0530945 union hdmi_infoframe *infoframe)
Rahul Sharmaa144c2e2012-11-26 10:52:57 +0530946{
947 u32 hdr_sum;
948 u8 chksum;
Rahul Sharmaa144c2e2012-11-26 10:52:57 +0530949 u32 mod;
Andrzej Hajdac93aaeb2015-07-09 16:28:10 +0200950 u8 ar;
Rahul Sharmaa144c2e2012-11-26 10:52:57 +0530951
Rahul Sharmaa144c2e2012-11-26 10:52:57 +0530952 mod = hdmi_reg_read(hdata, HDMI_MODE_SEL);
953 if (hdata->dvi_mode) {
954 hdmi_reg_writeb(hdata, HDMI_VSI_CON,
955 HDMI_VSI_CON_DO_NOT_TRANSMIT);
956 hdmi_reg_writeb(hdata, HDMI_AVI_CON,
957 HDMI_AVI_CON_DO_NOT_TRANSMIT);
958 hdmi_reg_writeb(hdata, HDMI_AUI_CON, HDMI_AUI_CON_NO_TRAN);
959 return;
960 }
961
Sachin Kamatd34d59b2014-02-04 08:40:18 +0530962 switch (infoframe->any.type) {
963 case HDMI_INFOFRAME_TYPE_AVI:
Rahul Sharmaa144c2e2012-11-26 10:52:57 +0530964 hdmi_reg_writeb(hdata, HDMI_AVI_CON, HDMI_AVI_CON_EVERY_VSYNC);
Sachin Kamatd34d59b2014-02-04 08:40:18 +0530965 hdmi_reg_writeb(hdata, HDMI_AVI_HEADER0, infoframe->any.type);
966 hdmi_reg_writeb(hdata, HDMI_AVI_HEADER1,
967 infoframe->any.version);
968 hdmi_reg_writeb(hdata, HDMI_AVI_HEADER2, infoframe->any.length);
969 hdr_sum = infoframe->any.type + infoframe->any.version +
970 infoframe->any.length;
Rahul Sharmaa144c2e2012-11-26 10:52:57 +0530971
972 /* Output format zero hardcoded ,RGB YBCR selection */
973 hdmi_reg_writeb(hdata, HDMI_AVI_BYTE(1), 0 << 5 |
974 AVI_ACTIVE_FORMAT_VALID |
975 AVI_UNDERSCANNED_DISPLAY_VALID);
976
Shirish S46154152014-03-13 10:58:28 +0530977 /*
978 * Set the aspect ratio as per the mode, mentioned in
979 * Table 9 AVI InfoFrame Data Byte 2 of CEA-861-D Standard
980 */
Andrzej Hajdac93aaeb2015-07-09 16:28:10 +0200981 ar = hdata->current_mode.picture_aspect_ratio;
982 switch (ar) {
Shirish S46154152014-03-13 10:58:28 +0530983 case HDMI_PICTURE_ASPECT_4_3:
Andrzej Hajdac93aaeb2015-07-09 16:28:10 +0200984 ar |= AVI_4_3_CENTER_RATIO;
Shirish S46154152014-03-13 10:58:28 +0530985 break;
986 case HDMI_PICTURE_ASPECT_16_9:
Andrzej Hajdac93aaeb2015-07-09 16:28:10 +0200987 ar |= AVI_16_9_CENTER_RATIO;
Shirish S46154152014-03-13 10:58:28 +0530988 break;
989 case HDMI_PICTURE_ASPECT_NONE:
990 default:
Andrzej Hajdac93aaeb2015-07-09 16:28:10 +0200991 ar |= AVI_SAME_AS_PIC_ASPECT_RATIO;
Shirish S46154152014-03-13 10:58:28 +0530992 break;
993 }
Andrzej Hajdac93aaeb2015-07-09 16:28:10 +0200994 hdmi_reg_writeb(hdata, HDMI_AVI_BYTE(2), ar);
Rahul Sharmaa144c2e2012-11-26 10:52:57 +0530995
Andrzej Hajdac93aaeb2015-07-09 16:28:10 +0200996 hdmi_reg_writeb(hdata, HDMI_AVI_BYTE(4), hdata->cea_video_id);
Rahul Sharmaa144c2e2012-11-26 10:52:57 +0530997
998 chksum = hdmi_chksum(hdata, HDMI_AVI_BYTE(1),
Sachin Kamatd34d59b2014-02-04 08:40:18 +0530999 infoframe->any.length, hdr_sum);
Rahul Sharmaa144c2e2012-11-26 10:52:57 +05301000 DRM_DEBUG_KMS("AVI checksum = 0x%x\n", chksum);
1001 hdmi_reg_writeb(hdata, HDMI_AVI_CHECK_SUM, chksum);
1002 break;
Sachin Kamatd34d59b2014-02-04 08:40:18 +05301003 case HDMI_INFOFRAME_TYPE_AUDIO:
Rahul Sharmaa144c2e2012-11-26 10:52:57 +05301004 hdmi_reg_writeb(hdata, HDMI_AUI_CON, 0x02);
Sachin Kamatd34d59b2014-02-04 08:40:18 +05301005 hdmi_reg_writeb(hdata, HDMI_AUI_HEADER0, infoframe->any.type);
1006 hdmi_reg_writeb(hdata, HDMI_AUI_HEADER1,
1007 infoframe->any.version);
1008 hdmi_reg_writeb(hdata, HDMI_AUI_HEADER2, infoframe->any.length);
1009 hdr_sum = infoframe->any.type + infoframe->any.version +
1010 infoframe->any.length;
Rahul Sharmaa144c2e2012-11-26 10:52:57 +05301011 chksum = hdmi_chksum(hdata, HDMI_AUI_BYTE(1),
Sachin Kamatd34d59b2014-02-04 08:40:18 +05301012 infoframe->any.length, hdr_sum);
Rahul Sharmaa144c2e2012-11-26 10:52:57 +05301013 DRM_DEBUG_KMS("AUI checksum = 0x%x\n", chksum);
1014 hdmi_reg_writeb(hdata, HDMI_AUI_CHECK_SUM, chksum);
1015 break;
1016 default:
1017 break;
1018 }
1019}
1020
Sean Pauld9716ee2014-01-30 16:19:29 -05001021static enum drm_connector_status hdmi_detect(struct drm_connector *connector,
1022 bool force)
Sean Paul45517892014-01-30 16:19:05 -05001023{
Sean Pauld9716ee2014-01-30 16:19:29 -05001024 struct hdmi_context *hdata = ctx_from_connector(connector);
Sean Paul45517892014-01-30 16:19:05 -05001025
Andrzej Hajdaef6ce282015-07-09 16:28:07 +02001026 if (gpio_get_value(hdata->hpd_gpio))
1027 return connector_status_connected;
Sean Paul5137c8c2014-04-03 20:41:03 +05301028
Andrzej Hajdaef6ce282015-07-09 16:28:07 +02001029 return connector_status_disconnected;
Sean Paul45517892014-01-30 16:19:05 -05001030}
1031
Sean Pauld9716ee2014-01-30 16:19:29 -05001032static void hdmi_connector_destroy(struct drm_connector *connector)
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001033{
Andrzej Hajdaad279312014-09-09 15:16:13 +02001034 drm_connector_unregister(connector);
1035 drm_connector_cleanup(connector);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001036}
1037
Sean Pauld9716ee2014-01-30 16:19:29 -05001038static struct drm_connector_funcs hdmi_connector_funcs = {
Gustavo Padovan63498e32015-06-01 12:04:53 -03001039 .dpms = drm_atomic_helper_connector_dpms,
Sean Pauld9716ee2014-01-30 16:19:29 -05001040 .fill_modes = drm_helper_probe_single_connector_modes,
1041 .detect = hdmi_detect,
1042 .destroy = hdmi_connector_destroy,
Gustavo Padovan4ea95262015-06-01 12:04:44 -03001043 .reset = drm_atomic_helper_connector_reset,
1044 .atomic_duplicate_state = drm_atomic_helper_connector_duplicate_state,
1045 .atomic_destroy_state = drm_atomic_helper_connector_destroy_state,
Sean Pauld9716ee2014-01-30 16:19:29 -05001046};
1047
1048static int hdmi_get_modes(struct drm_connector *connector)
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001049{
Sean Pauld9716ee2014-01-30 16:19:29 -05001050 struct hdmi_context *hdata = ctx_from_connector(connector);
1051 struct edid *edid;
Andrzej Hajda64ebd892015-07-09 08:25:38 +02001052 int ret;
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001053
Inki Dae8fa04aa2014-03-13 16:38:31 +09001054 if (!hdata->ddc_adpt)
Sean Pauld9716ee2014-01-30 16:19:29 -05001055 return -ENODEV;
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001056
Inki Dae8fa04aa2014-03-13 16:38:31 +09001057 edid = drm_get_edid(connector, hdata->ddc_adpt);
Sean Pauld9716ee2014-01-30 16:19:29 -05001058 if (!edid)
1059 return -ENODEV;
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001060
Sean Pauld9716ee2014-01-30 16:19:29 -05001061 hdata->dvi_mode = !drm_detect_hdmi_monitor(edid);
Rahul Sharma9c08e4b2013-01-04 07:59:11 -05001062 DRM_DEBUG_KMS("%s : width[%d] x height[%d]\n",
1063 (hdata->dvi_mode ? "dvi monitor" : "hdmi monitor"),
Sean Pauld9716ee2014-01-30 16:19:29 -05001064 edid->width_cm, edid->height_cm);
Rahul Sharma9c08e4b2013-01-04 07:59:11 -05001065
Sean Pauld9716ee2014-01-30 16:19:29 -05001066 drm_mode_connector_update_edid_property(connector, edid);
1067
Andrzej Hajda64ebd892015-07-09 08:25:38 +02001068 ret = drm_add_edid_modes(connector, edid);
1069
1070 kfree(edid);
1071
1072 return ret;
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001073}
1074
Rahul Sharma6b986ed2013-03-06 17:33:29 +09001075static int hdmi_find_phy_conf(struct hdmi_context *hdata, u32 pixel_clock)
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001076{
Rahul Sharmad5e9ca42014-05-09 15:34:18 +09001077 int i;
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001078
Andrzej Hajdacd240cd2015-07-09 16:28:09 +02001079 for (i = 0; i < hdata->drv_data->phy_conf_count; i++)
1080 if (hdata->drv_data->phy_confs[i].pixel_clock == pixel_clock)
Sean Paul2f7e2ed2013-01-15 08:11:08 -05001081 return i;
Sean Paul2f7e2ed2013-01-15 08:11:08 -05001082
1083 DRM_DEBUG_KMS("Could not find phy config for %d\n", pixel_clock);
1084 return -EINVAL;
1085}
1086
Sean Pauld9716ee2014-01-30 16:19:29 -05001087static int hdmi_mode_valid(struct drm_connector *connector,
Sean Paulf041b252014-01-30 16:19:15 -05001088 struct drm_display_mode *mode)
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +09001089{
Sean Pauld9716ee2014-01-30 16:19:29 -05001090 struct hdmi_context *hdata = ctx_from_connector(connector);
Rahul Sharma6b986ed2013-03-06 17:33:29 +09001091 int ret;
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +09001092
Rahul Sharma16844fb2013-06-10 14:50:00 +05301093 DRM_DEBUG_KMS("xres=%d, yres=%d, refresh=%d, intl=%d clock=%d\n",
1094 mode->hdisplay, mode->vdisplay, mode->vrefresh,
1095 (mode->flags & DRM_MODE_FLAG_INTERLACE) ? true :
1096 false, mode->clock * 1000);
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +09001097
Sean Paulf041b252014-01-30 16:19:15 -05001098 ret = mixer_check_mode(mode);
1099 if (ret)
Sean Pauld9716ee2014-01-30 16:19:29 -05001100 return MODE_BAD;
Sean Paulf041b252014-01-30 16:19:15 -05001101
Rahul Sharma16844fb2013-06-10 14:50:00 +05301102 ret = hdmi_find_phy_conf(hdata, mode->clock * 1000);
Rahul Sharma6b986ed2013-03-06 17:33:29 +09001103 if (ret < 0)
Sean Pauld9716ee2014-01-30 16:19:29 -05001104 return MODE_BAD;
1105
1106 return MODE_OK;
1107}
1108
1109static struct drm_encoder *hdmi_best_encoder(struct drm_connector *connector)
1110{
1111 struct hdmi_context *hdata = ctx_from_connector(connector);
1112
1113 return hdata->encoder;
1114}
1115
1116static struct drm_connector_helper_funcs hdmi_connector_helper_funcs = {
1117 .get_modes = hdmi_get_modes,
1118 .mode_valid = hdmi_mode_valid,
1119 .best_encoder = hdmi_best_encoder,
1120};
1121
1122static int hdmi_create_connector(struct exynos_drm_display *display,
1123 struct drm_encoder *encoder)
1124{
Andrzej Hajda0d8424f82014-11-17 09:54:21 +01001125 struct hdmi_context *hdata = display_to_hdmi(display);
Sean Pauld9716ee2014-01-30 16:19:29 -05001126 struct drm_connector *connector = &hdata->connector;
1127 int ret;
1128
1129 hdata->encoder = encoder;
1130 connector->interlace_allowed = true;
1131 connector->polled = DRM_CONNECTOR_POLL_HPD;
1132
1133 ret = drm_connector_init(hdata->drm_dev, connector,
1134 &hdmi_connector_funcs, DRM_MODE_CONNECTOR_HDMIA);
1135 if (ret) {
1136 DRM_ERROR("Failed to initialize connector with drm\n");
Rahul Sharma6b986ed2013-03-06 17:33:29 +09001137 return ret;
Sean Pauld9716ee2014-01-30 16:19:29 -05001138 }
1139
1140 drm_connector_helper_add(connector, &hdmi_connector_helper_funcs);
Thomas Wood34ea3d32014-05-29 16:57:41 +01001141 drm_connector_register(connector);
Sean Pauld9716ee2014-01-30 16:19:29 -05001142 drm_mode_connector_attach_encoder(connector, encoder);
1143
1144 return 0;
1145}
1146
Sean Paulf041b252014-01-30 16:19:15 -05001147static void hdmi_mode_fixup(struct exynos_drm_display *display,
1148 struct drm_connector *connector,
1149 const struct drm_display_mode *mode,
1150 struct drm_display_mode *adjusted_mode)
1151{
1152 struct drm_display_mode *m;
1153 int mode_ok;
1154
1155 DRM_DEBUG_KMS("%s\n", __FILE__);
1156
1157 drm_mode_set_crtcinfo(adjusted_mode, 0);
1158
Sean Pauld9716ee2014-01-30 16:19:29 -05001159 mode_ok = hdmi_mode_valid(connector, adjusted_mode);
Sean Paulf041b252014-01-30 16:19:15 -05001160
1161 /* just return if user desired mode exists. */
Sean Pauld9716ee2014-01-30 16:19:29 -05001162 if (mode_ok == MODE_OK)
Sean Paulf041b252014-01-30 16:19:15 -05001163 return;
1164
1165 /*
1166 * otherwise, find the most suitable mode among modes and change it
1167 * to adjusted_mode.
1168 */
1169 list_for_each_entry(m, &connector->modes, head) {
Sean Pauld9716ee2014-01-30 16:19:29 -05001170 mode_ok = hdmi_mode_valid(connector, m);
Sean Paulf041b252014-01-30 16:19:15 -05001171
Sean Pauld9716ee2014-01-30 16:19:29 -05001172 if (mode_ok == MODE_OK) {
Sean Paulf041b252014-01-30 16:19:15 -05001173 DRM_INFO("desired mode doesn't exist so\n");
1174 DRM_INFO("use the most suitable mode among modes.\n");
1175
1176 DRM_DEBUG_KMS("Adjusted Mode: [%d]x[%d] [%d]Hz\n",
1177 m->hdisplay, m->vdisplay, m->vrefresh);
1178
Sean Paul75626852014-01-30 16:19:16 -05001179 drm_mode_copy(adjusted_mode, m);
Sean Paulf041b252014-01-30 16:19:15 -05001180 break;
1181 }
1182 }
1183}
1184
Seung-Woo Kim3e148ba2012-03-16 18:47:16 +09001185static void hdmi_set_acr(u32 freq, u8 *acr)
1186{
1187 u32 n, cts;
1188
1189 switch (freq) {
1190 case 32000:
1191 n = 4096;
1192 cts = 27000;
1193 break;
1194 case 44100:
1195 n = 6272;
1196 cts = 30000;
1197 break;
1198 case 88200:
1199 n = 12544;
1200 cts = 30000;
1201 break;
1202 case 176400:
1203 n = 25088;
1204 cts = 30000;
1205 break;
1206 case 48000:
1207 n = 6144;
1208 cts = 27000;
1209 break;
1210 case 96000:
1211 n = 12288;
1212 cts = 27000;
1213 break;
1214 case 192000:
1215 n = 24576;
1216 cts = 27000;
1217 break;
1218 default:
1219 n = 0;
1220 cts = 0;
1221 break;
1222 }
1223
1224 acr[1] = cts >> 16;
1225 acr[2] = cts >> 8 & 0xff;
1226 acr[3] = cts & 0xff;
1227
1228 acr[4] = n >> 16;
1229 acr[5] = n >> 8 & 0xff;
1230 acr[6] = n & 0xff;
1231}
1232
1233static void hdmi_reg_acr(struct hdmi_context *hdata, u8 *acr)
1234{
1235 hdmi_reg_writeb(hdata, HDMI_ACR_N0, acr[6]);
1236 hdmi_reg_writeb(hdata, HDMI_ACR_N1, acr[5]);
1237 hdmi_reg_writeb(hdata, HDMI_ACR_N2, acr[4]);
1238 hdmi_reg_writeb(hdata, HDMI_ACR_MCTS0, acr[3]);
1239 hdmi_reg_writeb(hdata, HDMI_ACR_MCTS1, acr[2]);
1240 hdmi_reg_writeb(hdata, HDMI_ACR_MCTS2, acr[1]);
1241 hdmi_reg_writeb(hdata, HDMI_ACR_CTS0, acr[3]);
1242 hdmi_reg_writeb(hdata, HDMI_ACR_CTS1, acr[2]);
1243 hdmi_reg_writeb(hdata, HDMI_ACR_CTS2, acr[1]);
1244
Andrzej Hajdacd240cd2015-07-09 16:28:09 +02001245 if (hdata->drv_data->type == HDMI_TYPE13)
Seung-Woo Kim3e148ba2012-03-16 18:47:16 +09001246 hdmi_reg_writeb(hdata, HDMI_V13_ACR_CON, 4);
1247 else
1248 hdmi_reg_writeb(hdata, HDMI_ACR_CON, 4);
1249}
1250
1251static void hdmi_audio_init(struct hdmi_context *hdata)
1252{
Sachin Kamat7a9bf6e2014-07-02 09:33:07 +05301253 u32 sample_rate, bits_per_sample;
Seung-Woo Kim3e148ba2012-03-16 18:47:16 +09001254 u32 data_num, bit_ch, sample_frq;
1255 u32 val;
1256 u8 acr[7];
1257
1258 sample_rate = 44100;
1259 bits_per_sample = 16;
Seung-Woo Kim3e148ba2012-03-16 18:47:16 +09001260
1261 switch (bits_per_sample) {
1262 case 20:
1263 data_num = 2;
1264 bit_ch = 1;
1265 break;
1266 case 24:
1267 data_num = 3;
1268 bit_ch = 1;
1269 break;
1270 default:
1271 data_num = 1;
1272 bit_ch = 0;
1273 break;
1274 }
1275
1276 hdmi_set_acr(sample_rate, acr);
1277 hdmi_reg_acr(hdata, acr);
1278
1279 hdmi_reg_writeb(hdata, HDMI_I2S_MUX_CON, HDMI_I2S_IN_DISABLE
1280 | HDMI_I2S_AUD_I2S | HDMI_I2S_CUV_I2S_ENABLE
1281 | HDMI_I2S_MUX_ENABLE);
1282
1283 hdmi_reg_writeb(hdata, HDMI_I2S_MUX_CH, HDMI_I2S_CH0_EN
1284 | HDMI_I2S_CH1_EN | HDMI_I2S_CH2_EN);
1285
1286 hdmi_reg_writeb(hdata, HDMI_I2S_MUX_CUV, HDMI_I2S_CUV_RL_EN);
1287
1288 sample_frq = (sample_rate == 44100) ? 0 :
1289 (sample_rate == 48000) ? 2 :
1290 (sample_rate == 32000) ? 3 :
1291 (sample_rate == 96000) ? 0xa : 0x0;
1292
1293 hdmi_reg_writeb(hdata, HDMI_I2S_CLK_CON, HDMI_I2S_CLK_DIS);
1294 hdmi_reg_writeb(hdata, HDMI_I2S_CLK_CON, HDMI_I2S_CLK_EN);
1295
1296 val = hdmi_reg_read(hdata, HDMI_I2S_DSD_CON) | 0x01;
1297 hdmi_reg_writeb(hdata, HDMI_I2S_DSD_CON, val);
1298
1299 /* Configuration I2S input ports. Configure I2S_PIN_SEL_0~4 */
1300 hdmi_reg_writeb(hdata, HDMI_I2S_PIN_SEL_0, HDMI_I2S_SEL_SCLK(5)
1301 | HDMI_I2S_SEL_LRCK(6));
1302 hdmi_reg_writeb(hdata, HDMI_I2S_PIN_SEL_1, HDMI_I2S_SEL_SDATA1(1)
1303 | HDMI_I2S_SEL_SDATA2(4));
1304 hdmi_reg_writeb(hdata, HDMI_I2S_PIN_SEL_2, HDMI_I2S_SEL_SDATA3(1)
1305 | HDMI_I2S_SEL_SDATA2(2));
1306 hdmi_reg_writeb(hdata, HDMI_I2S_PIN_SEL_3, HDMI_I2S_SEL_DSD(0));
1307
1308 /* I2S_CON_1 & 2 */
1309 hdmi_reg_writeb(hdata, HDMI_I2S_CON_1, HDMI_I2S_SCLK_FALLING_EDGE
1310 | HDMI_I2S_L_CH_LOW_POL);
1311 hdmi_reg_writeb(hdata, HDMI_I2S_CON_2, HDMI_I2S_MSB_FIRST_MODE
1312 | HDMI_I2S_SET_BIT_CH(bit_ch)
1313 | HDMI_I2S_SET_SDATA_BIT(data_num)
1314 | HDMI_I2S_BASIC_FORMAT);
1315
1316 /* Configure register related to CUV information */
1317 hdmi_reg_writeb(hdata, HDMI_I2S_CH_ST_0, HDMI_I2S_CH_STATUS_MODE_0
1318 | HDMI_I2S_2AUD_CH_WITHOUT_PREEMPH
1319 | HDMI_I2S_COPYRIGHT
1320 | HDMI_I2S_LINEAR_PCM
1321 | HDMI_I2S_CONSUMER_FORMAT);
1322 hdmi_reg_writeb(hdata, HDMI_I2S_CH_ST_1, HDMI_I2S_CD_PLAYER);
1323 hdmi_reg_writeb(hdata, HDMI_I2S_CH_ST_2, HDMI_I2S_SET_SOURCE_NUM(0));
1324 hdmi_reg_writeb(hdata, HDMI_I2S_CH_ST_3, HDMI_I2S_CLK_ACCUR_LEVEL_2
1325 | HDMI_I2S_SET_SMP_FREQ(sample_frq));
1326 hdmi_reg_writeb(hdata, HDMI_I2S_CH_ST_4,
1327 HDMI_I2S_ORG_SMP_FREQ_44_1
1328 | HDMI_I2S_WORD_LEN_MAX24_24BITS
1329 | HDMI_I2S_WORD_LEN_MAX_24BITS);
1330
1331 hdmi_reg_writeb(hdata, HDMI_I2S_CH_ST_CON, HDMI_I2S_CH_STATUS_RELOAD);
1332}
1333
1334static void hdmi_audio_control(struct hdmi_context *hdata, bool onoff)
1335{
Seung-Woo Kim872d20d62012-04-24 17:39:15 +09001336 if (hdata->dvi_mode)
Seung-Woo Kim3e148ba2012-03-16 18:47:16 +09001337 return;
1338
1339 hdmi_reg_writeb(hdata, HDMI_AUI_CON, onoff ? 2 : 0);
1340 hdmi_reg_writemask(hdata, HDMI_CON_0, onoff ?
1341 HDMI_ASP_EN : HDMI_ASP_DIS, HDMI_ASP_MASK);
1342}
1343
Rahul Sharmabfa48422014-04-03 20:41:04 +05301344static void hdmi_start(struct hdmi_context *hdata, bool start)
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001345{
Rahul Sharmabfa48422014-04-03 20:41:04 +05301346 u32 val = start ? HDMI_TG_EN : 0;
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +09001347
Rahul Sharmabfa48422014-04-03 20:41:04 +05301348 if (hdata->current_mode.flags & DRM_MODE_FLAG_INTERLACE)
1349 val |= HDMI_FIELD_EN;
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +09001350
Rahul Sharmabfa48422014-04-03 20:41:04 +05301351 hdmi_reg_writemask(hdata, HDMI_CON_0, val, HDMI_EN);
1352 hdmi_reg_writemask(hdata, HDMI_TG_CMD, val, HDMI_TG_EN | HDMI_FIELD_EN);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001353}
1354
1355static void hdmi_conf_init(struct hdmi_context *hdata)
1356{
Sachin Kamatd34d59b2014-02-04 08:40:18 +05301357 union hdmi_infoframe infoframe;
Rahul Sharmaa144c2e2012-11-26 10:52:57 +05301358
Sean Paul77006a72013-01-16 10:17:20 -05001359 /* disable HPD interrupts from HDMI IP block, use GPIO instead */
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001360 hdmi_reg_writemask(hdata, HDMI_INTC_CON, 0, HDMI_INTC_EN_GLOBAL |
1361 HDMI_INTC_EN_HPD_PLUG | HDMI_INTC_EN_HPD_UNPLUG);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001362
1363 /* choose HDMI mode */
1364 hdmi_reg_writemask(hdata, HDMI_MODE_SEL,
1365 HDMI_MODE_HDMI_EN, HDMI_MODE_MASK);
Shirish S9a8e1cb2014-02-14 13:04:57 +05301366 /* Apply Video preable and Guard band in HDMI mode only */
1367 hdmi_reg_writeb(hdata, HDMI_CON_2, 0);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001368 /* disable bluescreen */
1369 hdmi_reg_writemask(hdata, HDMI_CON_0, 0, HDMI_BLUE_SCR_EN);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001370
Seung-Woo Kim872d20d62012-04-24 17:39:15 +09001371 if (hdata->dvi_mode) {
1372 /* choose DVI mode */
1373 hdmi_reg_writemask(hdata, HDMI_MODE_SEL,
1374 HDMI_MODE_DVI_EN, HDMI_MODE_MASK);
1375 hdmi_reg_writeb(hdata, HDMI_CON_2,
1376 HDMI_VID_PREAMBLE_DIS | HDMI_GUARD_BAND_DIS);
1377 }
1378
Andrzej Hajdacd240cd2015-07-09 16:28:09 +02001379 if (hdata->drv_data->type == HDMI_TYPE13) {
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +09001380 /* choose bluescreen (fecal) color */
1381 hdmi_reg_writeb(hdata, HDMI_V13_BLUE_SCREEN_0, 0x12);
1382 hdmi_reg_writeb(hdata, HDMI_V13_BLUE_SCREEN_1, 0x34);
1383 hdmi_reg_writeb(hdata, HDMI_V13_BLUE_SCREEN_2, 0x56);
1384
1385 /* enable AVI packet every vsync, fixes purple line problem */
1386 hdmi_reg_writeb(hdata, HDMI_V13_AVI_CON, 0x02);
1387 /* force RGB, look to CEA-861-D, table 7 for more detail */
1388 hdmi_reg_writeb(hdata, HDMI_V13_AVI_BYTE(0), 0 << 5);
1389 hdmi_reg_writemask(hdata, HDMI_CON_1, 0x10 << 5, 0x11 << 5);
1390
1391 hdmi_reg_writeb(hdata, HDMI_V13_SPD_CON, 0x02);
1392 hdmi_reg_writeb(hdata, HDMI_V13_AUI_CON, 0x02);
1393 hdmi_reg_writeb(hdata, HDMI_V13_ACR_CON, 0x04);
1394 } else {
Sachin Kamatd34d59b2014-02-04 08:40:18 +05301395 infoframe.any.type = HDMI_INFOFRAME_TYPE_AVI;
1396 infoframe.any.version = HDMI_AVI_VERSION;
1397 infoframe.any.length = HDMI_AVI_LENGTH;
Rahul Sharmaa144c2e2012-11-26 10:52:57 +05301398 hdmi_reg_infoframe(hdata, &infoframe);
1399
Sachin Kamatd34d59b2014-02-04 08:40:18 +05301400 infoframe.any.type = HDMI_INFOFRAME_TYPE_AUDIO;
1401 infoframe.any.version = HDMI_AUI_VERSION;
1402 infoframe.any.length = HDMI_AUI_LENGTH;
Rahul Sharmaa144c2e2012-11-26 10:52:57 +05301403 hdmi_reg_infoframe(hdata, &infoframe);
1404
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +09001405 /* enable AVI packet every vsync, fixes purple line problem */
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +09001406 hdmi_reg_writemask(hdata, HDMI_CON_1, 2, 3 << 5);
1407 }
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001408}
1409
Rahul Sharma16844fb2013-06-10 14:50:00 +05301410static void hdmi_v13_mode_apply(struct hdmi_context *hdata)
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001411{
Andrzej Hajdac93aaeb2015-07-09 16:28:10 +02001412 const struct hdmi_tg_regs *tg = &hdata->mode_conf.v13_conf.tg;
1413 const struct hdmi_v13_core_regs *core = &hdata->mode_conf.v13_conf.core;
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001414 int tries;
1415
1416 /* setting core registers */
1417 hdmi_reg_writeb(hdata, HDMI_H_BLANK_0, core->h_blank[0]);
1418 hdmi_reg_writeb(hdata, HDMI_H_BLANK_1, core->h_blank[1]);
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +09001419 hdmi_reg_writeb(hdata, HDMI_V13_V_BLANK_0, core->v_blank[0]);
1420 hdmi_reg_writeb(hdata, HDMI_V13_V_BLANK_1, core->v_blank[1]);
1421 hdmi_reg_writeb(hdata, HDMI_V13_V_BLANK_2, core->v_blank[2]);
1422 hdmi_reg_writeb(hdata, HDMI_V13_H_V_LINE_0, core->h_v_line[0]);
1423 hdmi_reg_writeb(hdata, HDMI_V13_H_V_LINE_1, core->h_v_line[1]);
1424 hdmi_reg_writeb(hdata, HDMI_V13_H_V_LINE_2, core->h_v_line[2]);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001425 hdmi_reg_writeb(hdata, HDMI_VSYNC_POL, core->vsync_pol[0]);
1426 hdmi_reg_writeb(hdata, HDMI_INT_PRO_MODE, core->int_pro_mode[0]);
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +09001427 hdmi_reg_writeb(hdata, HDMI_V13_V_BLANK_F_0, core->v_blank_f[0]);
1428 hdmi_reg_writeb(hdata, HDMI_V13_V_BLANK_F_1, core->v_blank_f[1]);
1429 hdmi_reg_writeb(hdata, HDMI_V13_V_BLANK_F_2, core->v_blank_f[2]);
1430 hdmi_reg_writeb(hdata, HDMI_V13_H_SYNC_GEN_0, core->h_sync_gen[0]);
1431 hdmi_reg_writeb(hdata, HDMI_V13_H_SYNC_GEN_1, core->h_sync_gen[1]);
1432 hdmi_reg_writeb(hdata, HDMI_V13_H_SYNC_GEN_2, core->h_sync_gen[2]);
1433 hdmi_reg_writeb(hdata, HDMI_V13_V_SYNC_GEN_1_0, core->v_sync_gen1[0]);
1434 hdmi_reg_writeb(hdata, HDMI_V13_V_SYNC_GEN_1_1, core->v_sync_gen1[1]);
1435 hdmi_reg_writeb(hdata, HDMI_V13_V_SYNC_GEN_1_2, core->v_sync_gen1[2]);
1436 hdmi_reg_writeb(hdata, HDMI_V13_V_SYNC_GEN_2_0, core->v_sync_gen2[0]);
1437 hdmi_reg_writeb(hdata, HDMI_V13_V_SYNC_GEN_2_1, core->v_sync_gen2[1]);
1438 hdmi_reg_writeb(hdata, HDMI_V13_V_SYNC_GEN_2_2, core->v_sync_gen2[2]);
1439 hdmi_reg_writeb(hdata, HDMI_V13_V_SYNC_GEN_3_0, core->v_sync_gen3[0]);
1440 hdmi_reg_writeb(hdata, HDMI_V13_V_SYNC_GEN_3_1, core->v_sync_gen3[1]);
1441 hdmi_reg_writeb(hdata, HDMI_V13_V_SYNC_GEN_3_2, core->v_sync_gen3[2]);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001442 /* Timing generator registers */
Rahul Sharma6b986ed2013-03-06 17:33:29 +09001443 hdmi_reg_writeb(hdata, HDMI_TG_H_FSZ_L, tg->h_fsz[0]);
1444 hdmi_reg_writeb(hdata, HDMI_TG_H_FSZ_H, tg->h_fsz[1]);
1445 hdmi_reg_writeb(hdata, HDMI_TG_HACT_ST_L, tg->hact_st[0]);
1446 hdmi_reg_writeb(hdata, HDMI_TG_HACT_ST_H, tg->hact_st[1]);
1447 hdmi_reg_writeb(hdata, HDMI_TG_HACT_SZ_L, tg->hact_sz[0]);
1448 hdmi_reg_writeb(hdata, HDMI_TG_HACT_SZ_H, tg->hact_sz[1]);
1449 hdmi_reg_writeb(hdata, HDMI_TG_V_FSZ_L, tg->v_fsz[0]);
1450 hdmi_reg_writeb(hdata, HDMI_TG_V_FSZ_H, tg->v_fsz[1]);
1451 hdmi_reg_writeb(hdata, HDMI_TG_VSYNC_L, tg->vsync[0]);
1452 hdmi_reg_writeb(hdata, HDMI_TG_VSYNC_H, tg->vsync[1]);
1453 hdmi_reg_writeb(hdata, HDMI_TG_VSYNC2_L, tg->vsync2[0]);
1454 hdmi_reg_writeb(hdata, HDMI_TG_VSYNC2_H, tg->vsync2[1]);
1455 hdmi_reg_writeb(hdata, HDMI_TG_VACT_ST_L, tg->vact_st[0]);
1456 hdmi_reg_writeb(hdata, HDMI_TG_VACT_ST_H, tg->vact_st[1]);
1457 hdmi_reg_writeb(hdata, HDMI_TG_VACT_SZ_L, tg->vact_sz[0]);
1458 hdmi_reg_writeb(hdata, HDMI_TG_VACT_SZ_H, tg->vact_sz[1]);
1459 hdmi_reg_writeb(hdata, HDMI_TG_FIELD_CHG_L, tg->field_chg[0]);
1460 hdmi_reg_writeb(hdata, HDMI_TG_FIELD_CHG_H, tg->field_chg[1]);
1461 hdmi_reg_writeb(hdata, HDMI_TG_VACT_ST2_L, tg->vact_st2[0]);
1462 hdmi_reg_writeb(hdata, HDMI_TG_VACT_ST2_H, tg->vact_st2[1]);
1463 hdmi_reg_writeb(hdata, HDMI_TG_VSYNC_TOP_HDMI_L, tg->vsync_top_hdmi[0]);
1464 hdmi_reg_writeb(hdata, HDMI_TG_VSYNC_TOP_HDMI_H, tg->vsync_top_hdmi[1]);
1465 hdmi_reg_writeb(hdata, HDMI_TG_VSYNC_BOT_HDMI_L, tg->vsync_bot_hdmi[0]);
1466 hdmi_reg_writeb(hdata, HDMI_TG_VSYNC_BOT_HDMI_H, tg->vsync_bot_hdmi[1]);
1467 hdmi_reg_writeb(hdata, HDMI_TG_FIELD_TOP_HDMI_L, tg->field_top_hdmi[0]);
1468 hdmi_reg_writeb(hdata, HDMI_TG_FIELD_TOP_HDMI_H, tg->field_top_hdmi[1]);
1469 hdmi_reg_writeb(hdata, HDMI_TG_FIELD_BOT_HDMI_L, tg->field_bot_hdmi[0]);
1470 hdmi_reg_writeb(hdata, HDMI_TG_FIELD_BOT_HDMI_H, tg->field_bot_hdmi[1]);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001471
1472 /* waiting for HDMIPHY's PLL to get to steady state */
1473 for (tries = 100; tries; --tries) {
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +09001474 u32 val = hdmi_reg_read(hdata, HDMI_V13_PHY_STATUS);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001475 if (val & HDMI_PHY_STATUS_READY)
1476 break;
Sean Paul09760ea2013-01-14 17:03:20 -05001477 usleep_range(1000, 2000);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001478 }
1479 /* steady state not achieved */
1480 if (tries == 0) {
1481 DRM_ERROR("hdmiphy's pll could not reach steady state.\n");
1482 hdmi_regs_dump(hdata, "timing apply");
1483 }
1484
Sean Paul0bfb1f82013-06-11 12:24:02 +05301485 clk_disable_unprepare(hdata->res.sclk_hdmi);
Rahul Sharma59956d32013-06-11 12:24:03 +05301486 clk_set_parent(hdata->res.mout_hdmi, hdata->res.sclk_hdmiphy);
Sean Paul0bfb1f82013-06-11 12:24:02 +05301487 clk_prepare_enable(hdata->res.sclk_hdmi);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001488
1489 /* enable HDMI and timing generator */
Rahul Sharmabfa48422014-04-03 20:41:04 +05301490 hdmi_start(hdata, true);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001491}
1492
Rahul Sharma16844fb2013-06-10 14:50:00 +05301493static void hdmi_v14_mode_apply(struct hdmi_context *hdata)
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +09001494{
Andrzej Hajdac93aaeb2015-07-09 16:28:10 +02001495 const struct hdmi_tg_regs *tg = &hdata->mode_conf.v14_conf.tg;
1496 const struct hdmi_v14_core_regs *core = &hdata->mode_conf.v14_conf.core;
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +09001497 int tries;
1498
1499 /* setting core registers */
1500 hdmi_reg_writeb(hdata, HDMI_H_BLANK_0, core->h_blank[0]);
1501 hdmi_reg_writeb(hdata, HDMI_H_BLANK_1, core->h_blank[1]);
1502 hdmi_reg_writeb(hdata, HDMI_V2_BLANK_0, core->v2_blank[0]);
1503 hdmi_reg_writeb(hdata, HDMI_V2_BLANK_1, core->v2_blank[1]);
1504 hdmi_reg_writeb(hdata, HDMI_V1_BLANK_0, core->v1_blank[0]);
1505 hdmi_reg_writeb(hdata, HDMI_V1_BLANK_1, core->v1_blank[1]);
1506 hdmi_reg_writeb(hdata, HDMI_V_LINE_0, core->v_line[0]);
1507 hdmi_reg_writeb(hdata, HDMI_V_LINE_1, core->v_line[1]);
1508 hdmi_reg_writeb(hdata, HDMI_H_LINE_0, core->h_line[0]);
1509 hdmi_reg_writeb(hdata, HDMI_H_LINE_1, core->h_line[1]);
1510 hdmi_reg_writeb(hdata, HDMI_HSYNC_POL, core->hsync_pol[0]);
1511 hdmi_reg_writeb(hdata, HDMI_VSYNC_POL, core->vsync_pol[0]);
1512 hdmi_reg_writeb(hdata, HDMI_INT_PRO_MODE, core->int_pro_mode[0]);
1513 hdmi_reg_writeb(hdata, HDMI_V_BLANK_F0_0, core->v_blank_f0[0]);
1514 hdmi_reg_writeb(hdata, HDMI_V_BLANK_F0_1, core->v_blank_f0[1]);
1515 hdmi_reg_writeb(hdata, HDMI_V_BLANK_F1_0, core->v_blank_f1[0]);
1516 hdmi_reg_writeb(hdata, HDMI_V_BLANK_F1_1, core->v_blank_f1[1]);
1517 hdmi_reg_writeb(hdata, HDMI_H_SYNC_START_0, core->h_sync_start[0]);
1518 hdmi_reg_writeb(hdata, HDMI_H_SYNC_START_1, core->h_sync_start[1]);
1519 hdmi_reg_writeb(hdata, HDMI_H_SYNC_END_0, core->h_sync_end[0]);
1520 hdmi_reg_writeb(hdata, HDMI_H_SYNC_END_1, core->h_sync_end[1]);
1521 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_BEF_2_0,
1522 core->v_sync_line_bef_2[0]);
1523 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_BEF_2_1,
1524 core->v_sync_line_bef_2[1]);
1525 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_BEF_1_0,
1526 core->v_sync_line_bef_1[0]);
1527 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_BEF_1_1,
1528 core->v_sync_line_bef_1[1]);
1529 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_AFT_2_0,
1530 core->v_sync_line_aft_2[0]);
1531 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_AFT_2_1,
1532 core->v_sync_line_aft_2[1]);
1533 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_AFT_1_0,
1534 core->v_sync_line_aft_1[0]);
1535 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_AFT_1_1,
1536 core->v_sync_line_aft_1[1]);
1537 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_AFT_PXL_2_0,
1538 core->v_sync_line_aft_pxl_2[0]);
1539 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_AFT_PXL_2_1,
1540 core->v_sync_line_aft_pxl_2[1]);
1541 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_AFT_PXL_1_0,
1542 core->v_sync_line_aft_pxl_1[0]);
1543 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_AFT_PXL_1_1,
1544 core->v_sync_line_aft_pxl_1[1]);
1545 hdmi_reg_writeb(hdata, HDMI_V_BLANK_F2_0, core->v_blank_f2[0]);
1546 hdmi_reg_writeb(hdata, HDMI_V_BLANK_F2_1, core->v_blank_f2[1]);
1547 hdmi_reg_writeb(hdata, HDMI_V_BLANK_F3_0, core->v_blank_f3[0]);
1548 hdmi_reg_writeb(hdata, HDMI_V_BLANK_F3_1, core->v_blank_f3[1]);
1549 hdmi_reg_writeb(hdata, HDMI_V_BLANK_F4_0, core->v_blank_f4[0]);
1550 hdmi_reg_writeb(hdata, HDMI_V_BLANK_F4_1, core->v_blank_f4[1]);
1551 hdmi_reg_writeb(hdata, HDMI_V_BLANK_F5_0, core->v_blank_f5[0]);
1552 hdmi_reg_writeb(hdata, HDMI_V_BLANK_F5_1, core->v_blank_f5[1]);
1553 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_AFT_3_0,
1554 core->v_sync_line_aft_3[0]);
1555 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_AFT_3_1,
1556 core->v_sync_line_aft_3[1]);
1557 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_AFT_4_0,
1558 core->v_sync_line_aft_4[0]);
1559 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_AFT_4_1,
1560 core->v_sync_line_aft_4[1]);
1561 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_AFT_5_0,
1562 core->v_sync_line_aft_5[0]);
1563 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_AFT_5_1,
1564 core->v_sync_line_aft_5[1]);
1565 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_AFT_6_0,
1566 core->v_sync_line_aft_6[0]);
1567 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_AFT_6_1,
1568 core->v_sync_line_aft_6[1]);
1569 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_AFT_PXL_3_0,
1570 core->v_sync_line_aft_pxl_3[0]);
1571 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_AFT_PXL_3_1,
1572 core->v_sync_line_aft_pxl_3[1]);
1573 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_AFT_PXL_4_0,
1574 core->v_sync_line_aft_pxl_4[0]);
1575 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_AFT_PXL_4_1,
1576 core->v_sync_line_aft_pxl_4[1]);
1577 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_AFT_PXL_5_0,
1578 core->v_sync_line_aft_pxl_5[0]);
1579 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_AFT_PXL_5_1,
1580 core->v_sync_line_aft_pxl_5[1]);
1581 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_AFT_PXL_6_0,
1582 core->v_sync_line_aft_pxl_6[0]);
1583 hdmi_reg_writeb(hdata, HDMI_V_SYNC_LINE_AFT_PXL_6_1,
1584 core->v_sync_line_aft_pxl_6[1]);
1585 hdmi_reg_writeb(hdata, HDMI_VACT_SPACE_1_0, core->vact_space_1[0]);
1586 hdmi_reg_writeb(hdata, HDMI_VACT_SPACE_1_1, core->vact_space_1[1]);
1587 hdmi_reg_writeb(hdata, HDMI_VACT_SPACE_2_0, core->vact_space_2[0]);
1588 hdmi_reg_writeb(hdata, HDMI_VACT_SPACE_2_1, core->vact_space_2[1]);
1589 hdmi_reg_writeb(hdata, HDMI_VACT_SPACE_3_0, core->vact_space_3[0]);
1590 hdmi_reg_writeb(hdata, HDMI_VACT_SPACE_3_1, core->vact_space_3[1]);
1591 hdmi_reg_writeb(hdata, HDMI_VACT_SPACE_4_0, core->vact_space_4[0]);
1592 hdmi_reg_writeb(hdata, HDMI_VACT_SPACE_4_1, core->vact_space_4[1]);
1593 hdmi_reg_writeb(hdata, HDMI_VACT_SPACE_5_0, core->vact_space_5[0]);
1594 hdmi_reg_writeb(hdata, HDMI_VACT_SPACE_5_1, core->vact_space_5[1]);
1595 hdmi_reg_writeb(hdata, HDMI_VACT_SPACE_6_0, core->vact_space_6[0]);
1596 hdmi_reg_writeb(hdata, HDMI_VACT_SPACE_6_1, core->vact_space_6[1]);
1597
1598 /* Timing generator registers */
Sean Paul2f7e2ed2013-01-15 08:11:08 -05001599 hdmi_reg_writeb(hdata, HDMI_TG_H_FSZ_L, tg->h_fsz[0]);
1600 hdmi_reg_writeb(hdata, HDMI_TG_H_FSZ_H, tg->h_fsz[1]);
1601 hdmi_reg_writeb(hdata, HDMI_TG_HACT_ST_L, tg->hact_st[0]);
1602 hdmi_reg_writeb(hdata, HDMI_TG_HACT_ST_H, tg->hact_st[1]);
1603 hdmi_reg_writeb(hdata, HDMI_TG_HACT_SZ_L, tg->hact_sz[0]);
1604 hdmi_reg_writeb(hdata, HDMI_TG_HACT_SZ_H, tg->hact_sz[1]);
1605 hdmi_reg_writeb(hdata, HDMI_TG_V_FSZ_L, tg->v_fsz[0]);
1606 hdmi_reg_writeb(hdata, HDMI_TG_V_FSZ_H, tg->v_fsz[1]);
1607 hdmi_reg_writeb(hdata, HDMI_TG_VSYNC_L, tg->vsync[0]);
1608 hdmi_reg_writeb(hdata, HDMI_TG_VSYNC_H, tg->vsync[1]);
1609 hdmi_reg_writeb(hdata, HDMI_TG_VSYNC2_L, tg->vsync2[0]);
1610 hdmi_reg_writeb(hdata, HDMI_TG_VSYNC2_H, tg->vsync2[1]);
1611 hdmi_reg_writeb(hdata, HDMI_TG_VACT_ST_L, tg->vact_st[0]);
1612 hdmi_reg_writeb(hdata, HDMI_TG_VACT_ST_H, tg->vact_st[1]);
1613 hdmi_reg_writeb(hdata, HDMI_TG_VACT_SZ_L, tg->vact_sz[0]);
1614 hdmi_reg_writeb(hdata, HDMI_TG_VACT_SZ_H, tg->vact_sz[1]);
1615 hdmi_reg_writeb(hdata, HDMI_TG_FIELD_CHG_L, tg->field_chg[0]);
1616 hdmi_reg_writeb(hdata, HDMI_TG_FIELD_CHG_H, tg->field_chg[1]);
1617 hdmi_reg_writeb(hdata, HDMI_TG_VACT_ST2_L, tg->vact_st2[0]);
1618 hdmi_reg_writeb(hdata, HDMI_TG_VACT_ST2_H, tg->vact_st2[1]);
1619 hdmi_reg_writeb(hdata, HDMI_TG_VACT_ST3_L, tg->vact_st3[0]);
1620 hdmi_reg_writeb(hdata, HDMI_TG_VACT_ST3_H, tg->vact_st3[1]);
1621 hdmi_reg_writeb(hdata, HDMI_TG_VACT_ST4_L, tg->vact_st4[0]);
1622 hdmi_reg_writeb(hdata, HDMI_TG_VACT_ST4_H, tg->vact_st4[1]);
1623 hdmi_reg_writeb(hdata, HDMI_TG_VSYNC_TOP_HDMI_L, tg->vsync_top_hdmi[0]);
1624 hdmi_reg_writeb(hdata, HDMI_TG_VSYNC_TOP_HDMI_H, tg->vsync_top_hdmi[1]);
1625 hdmi_reg_writeb(hdata, HDMI_TG_VSYNC_BOT_HDMI_L, tg->vsync_bot_hdmi[0]);
1626 hdmi_reg_writeb(hdata, HDMI_TG_VSYNC_BOT_HDMI_H, tg->vsync_bot_hdmi[1]);
1627 hdmi_reg_writeb(hdata, HDMI_TG_FIELD_TOP_HDMI_L, tg->field_top_hdmi[0]);
1628 hdmi_reg_writeb(hdata, HDMI_TG_FIELD_TOP_HDMI_H, tg->field_top_hdmi[1]);
1629 hdmi_reg_writeb(hdata, HDMI_TG_FIELD_BOT_HDMI_L, tg->field_bot_hdmi[0]);
1630 hdmi_reg_writeb(hdata, HDMI_TG_FIELD_BOT_HDMI_H, tg->field_bot_hdmi[1]);
1631 hdmi_reg_writeb(hdata, HDMI_TG_3D, tg->tg_3d[0]);
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +09001632
1633 /* waiting for HDMIPHY's PLL to get to steady state */
1634 for (tries = 100; tries; --tries) {
1635 u32 val = hdmi_reg_read(hdata, HDMI_PHY_STATUS_0);
1636 if (val & HDMI_PHY_STATUS_READY)
1637 break;
Sean Paul09760ea2013-01-14 17:03:20 -05001638 usleep_range(1000, 2000);
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +09001639 }
1640 /* steady state not achieved */
1641 if (tries == 0) {
1642 DRM_ERROR("hdmiphy's pll could not reach steady state.\n");
1643 hdmi_regs_dump(hdata, "timing apply");
1644 }
1645
Sean Paul0bfb1f82013-06-11 12:24:02 +05301646 clk_disable_unprepare(hdata->res.sclk_hdmi);
Rahul Sharma59956d32013-06-11 12:24:03 +05301647 clk_set_parent(hdata->res.mout_hdmi, hdata->res.sclk_hdmiphy);
Sean Paul0bfb1f82013-06-11 12:24:02 +05301648 clk_prepare_enable(hdata->res.sclk_hdmi);
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +09001649
1650 /* enable HDMI and timing generator */
Rahul Sharmabfa48422014-04-03 20:41:04 +05301651 hdmi_start(hdata, true);
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +09001652}
1653
Rahul Sharma16844fb2013-06-10 14:50:00 +05301654static void hdmi_mode_apply(struct hdmi_context *hdata)
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +09001655{
Andrzej Hajdacd240cd2015-07-09 16:28:09 +02001656 if (hdata->drv_data->type == HDMI_TYPE13)
Rahul Sharma16844fb2013-06-10 14:50:00 +05301657 hdmi_v13_mode_apply(hdata);
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +09001658 else
Rahul Sharma16844fb2013-06-10 14:50:00 +05301659 hdmi_v14_mode_apply(hdata);
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +09001660}
1661
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001662static void hdmiphy_conf_reset(struct hdmi_context *hdata)
1663{
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +09001664 u32 reg;
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001665
Sean Paul0bfb1f82013-06-11 12:24:02 +05301666 clk_disable_unprepare(hdata->res.sclk_hdmi);
Rahul Sharma59956d32013-06-11 12:24:03 +05301667 clk_set_parent(hdata->res.mout_hdmi, hdata->res.sclk_pixel);
Sean Paul0bfb1f82013-06-11 12:24:02 +05301668 clk_prepare_enable(hdata->res.sclk_hdmi);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001669
1670 /* operation mode */
Joonyoung Shim265134a2015-01-12 14:35:16 +09001671 hdmiphy_reg_writeb(hdata, HDMIPHY_MODE_SET_DONE,
1672 HDMI_PHY_ENABLE_MODE_SET);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001673
Andrzej Hajdacd240cd2015-07-09 16:28:09 +02001674 if (hdata->drv_data->type == HDMI_TYPE13)
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +09001675 reg = HDMI_V13_PHY_RSTOUT;
1676 else
1677 reg = HDMI_PHY_RSTOUT;
1678
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001679 /* reset hdmiphy */
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +09001680 hdmi_reg_writemask(hdata, reg, ~0, HDMI_PHY_SW_RSTOUT);
Sean Paul09760ea2013-01-14 17:03:20 -05001681 usleep_range(10000, 12000);
Joonyoung Shim3ecd70b2012-03-16 18:47:03 +09001682 hdmi_reg_writemask(hdata, reg, 0, HDMI_PHY_SW_RSTOUT);
Sean Paul09760ea2013-01-14 17:03:20 -05001683 usleep_range(10000, 12000);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001684}
1685
Rahul Sharmaa5562252012-11-28 11:30:25 +05301686static void hdmiphy_poweron(struct hdmi_context *hdata)
1687{
Andrzej Hajdacd240cd2015-07-09 16:28:09 +02001688 if (hdata->drv_data->type != HDMI_TYPE14)
Shirish S6a296e22014-04-03 20:41:02 +05301689 return;
1690
1691 DRM_DEBUG_KMS("\n");
1692
1693 /* For PHY Mode Setting */
1694 hdmiphy_reg_writeb(hdata, HDMIPHY_MODE_SET_DONE,
1695 HDMI_PHY_ENABLE_MODE_SET);
1696 /* Phy Power On */
1697 hdmiphy_reg_writeb(hdata, HDMIPHY_POWER,
1698 HDMI_PHY_POWER_ON);
1699 /* For PHY Mode Setting */
1700 hdmiphy_reg_writeb(hdata, HDMIPHY_MODE_SET_DONE,
1701 HDMI_PHY_DISABLE_MODE_SET);
1702 /* PHY SW Reset */
1703 hdmiphy_conf_reset(hdata);
Rahul Sharmaa5562252012-11-28 11:30:25 +05301704}
1705
1706static void hdmiphy_poweroff(struct hdmi_context *hdata)
1707{
Andrzej Hajdacd240cd2015-07-09 16:28:09 +02001708 if (hdata->drv_data->type != HDMI_TYPE14)
Shirish S6a296e22014-04-03 20:41:02 +05301709 return;
1710
1711 DRM_DEBUG_KMS("\n");
1712
1713 /* PHY SW Reset */
1714 hdmiphy_conf_reset(hdata);
1715 /* For PHY Mode Setting */
1716 hdmiphy_reg_writeb(hdata, HDMIPHY_MODE_SET_DONE,
1717 HDMI_PHY_ENABLE_MODE_SET);
1718
1719 /* PHY Power Off */
1720 hdmiphy_reg_writeb(hdata, HDMIPHY_POWER,
1721 HDMI_PHY_POWER_OFF);
1722
1723 /* For PHY Mode Setting */
1724 hdmiphy_reg_writeb(hdata, HDMIPHY_MODE_SET_DONE,
1725 HDMI_PHY_DISABLE_MODE_SET);
Rahul Sharmaa5562252012-11-28 11:30:25 +05301726}
1727
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001728static void hdmiphy_conf_apply(struct hdmi_context *hdata)
1729{
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001730 int ret;
1731 int i;
1732
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001733 /* pixel clock */
Andrzej Hajdac93aaeb2015-07-09 16:28:10 +02001734 i = hdmi_find_phy_conf(hdata, hdata->current_mode.clock * 1000);
Rahul Sharma6b986ed2013-03-06 17:33:29 +09001735 if (i < 0) {
1736 DRM_ERROR("failed to find hdmiphy conf\n");
1737 return;
1738 }
Sean Paul2f7e2ed2013-01-15 08:11:08 -05001739
Andrzej Hajdacd240cd2015-07-09 16:28:09 +02001740 ret = hdmiphy_reg_write_buf(hdata, 0,
1741 hdata->drv_data->phy_confs[i].conf, 32);
Rahul Sharmad5e9ca42014-05-09 15:34:18 +09001742 if (ret) {
1743 DRM_ERROR("failed to configure hdmiphy\n");
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001744 return;
1745 }
1746
Sean Paul09760ea2013-01-14 17:03:20 -05001747 usleep_range(10000, 12000);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001748
Rahul Sharmad5e9ca42014-05-09 15:34:18 +09001749 ret = hdmiphy_reg_writeb(hdata, HDMIPHY_MODE_SET_DONE,
1750 HDMI_PHY_DISABLE_MODE_SET);
1751 if (ret) {
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001752 DRM_ERROR("failed to enable hdmiphy\n");
1753 return;
1754 }
1755
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001756}
1757
1758static void hdmi_conf_apply(struct hdmi_context *hdata)
1759{
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001760 hdmiphy_conf_reset(hdata);
1761 hdmiphy_conf_apply(hdata);
1762
Rahul Sharmabfa48422014-04-03 20:41:04 +05301763 hdmi_start(hdata, false);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001764 hdmi_conf_init(hdata);
Joonyoung Shimcf8fc4f2012-04-23 19:35:50 +09001765
Seung-Woo Kim3e148ba2012-03-16 18:47:16 +09001766 hdmi_audio_init(hdata);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001767
1768 /* setting core registers */
Rahul Sharma16844fb2013-06-10 14:50:00 +05301769 hdmi_mode_apply(hdata);
Seung-Woo Kim3e148ba2012-03-16 18:47:16 +09001770 hdmi_audio_control(hdata, true);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001771
1772 hdmi_regs_dump(hdata, "start");
1773}
1774
Sean Paul2f7e2ed2013-01-15 08:11:08 -05001775static void hdmi_set_reg(u8 *reg_pair, int num_bytes, u32 value)
1776{
1777 int i;
1778 BUG_ON(num_bytes > 4);
1779 for (i = 0; i < num_bytes; i++)
1780 reg_pair[i] = (value >> (8 * i)) & 0xff;
1781}
1782
Rahul Sharma6b986ed2013-03-06 17:33:29 +09001783static void hdmi_v13_mode_set(struct hdmi_context *hdata,
1784 struct drm_display_mode *m)
1785{
Andrzej Hajdac93aaeb2015-07-09 16:28:10 +02001786 struct hdmi_v13_core_regs *core = &hdata->mode_conf.v13_conf.core;
1787 struct hdmi_tg_regs *tg = &hdata->mode_conf.v13_conf.tg;
Rahul Sharma6b986ed2013-03-06 17:33:29 +09001788 unsigned int val;
1789
Rahul Sharma6b986ed2013-03-06 17:33:29 +09001790 hdmi_set_reg(core->h_blank, 2, m->htotal - m->hdisplay);
1791 hdmi_set_reg(core->h_v_line, 3, (m->htotal << 12) | m->vtotal);
1792
1793 val = (m->flags & DRM_MODE_FLAG_NVSYNC) ? 1 : 0;
1794 hdmi_set_reg(core->vsync_pol, 1, val);
1795
1796 val = (m->flags & DRM_MODE_FLAG_INTERLACE) ? 1 : 0;
1797 hdmi_set_reg(core->int_pro_mode, 1, val);
1798
1799 val = (m->hsync_start - m->hdisplay - 2);
1800 val |= ((m->hsync_end - m->hdisplay - 2) << 10);
1801 val |= ((m->flags & DRM_MODE_FLAG_NHSYNC) ? 1 : 0)<<20;
1802 hdmi_set_reg(core->h_sync_gen, 3, val);
1803
1804 /*
1805 * Quirk requirement for exynos HDMI IP design,
1806 * 2 pixels less than the actual calculation for hsync_start
1807 * and end.
1808 */
1809
1810 /* Following values & calculations differ for different type of modes */
1811 if (m->flags & DRM_MODE_FLAG_INTERLACE) {
1812 /* Interlaced Mode */
1813 val = ((m->vsync_end - m->vdisplay) / 2);
1814 val |= ((m->vsync_start - m->vdisplay) / 2) << 12;
1815 hdmi_set_reg(core->v_sync_gen1, 3, val);
1816
1817 val = m->vtotal / 2;
1818 val |= ((m->vtotal - m->vdisplay) / 2) << 11;
1819 hdmi_set_reg(core->v_blank, 3, val);
1820
1821 val = (m->vtotal +
1822 ((m->vsync_end - m->vsync_start) * 4) + 5) / 2;
1823 val |= m->vtotal << 11;
1824 hdmi_set_reg(core->v_blank_f, 3, val);
1825
1826 val = ((m->vtotal / 2) + 7);
1827 val |= ((m->vtotal / 2) + 2) << 12;
1828 hdmi_set_reg(core->v_sync_gen2, 3, val);
1829
1830 val = ((m->htotal / 2) + (m->hsync_start - m->hdisplay));
1831 val |= ((m->htotal / 2) +
1832 (m->hsync_start - m->hdisplay)) << 12;
1833 hdmi_set_reg(core->v_sync_gen3, 3, val);
1834
1835 hdmi_set_reg(tg->vact_st, 2, (m->vtotal - m->vdisplay) / 2);
1836 hdmi_set_reg(tg->vact_sz, 2, m->vdisplay / 2);
1837
1838 hdmi_set_reg(tg->vact_st2, 2, 0x249);/* Reset value + 1*/
1839 } else {
1840 /* Progressive Mode */
1841
1842 val = m->vtotal;
1843 val |= (m->vtotal - m->vdisplay) << 11;
1844 hdmi_set_reg(core->v_blank, 3, val);
1845
1846 hdmi_set_reg(core->v_blank_f, 3, 0);
1847
1848 val = (m->vsync_end - m->vdisplay);
1849 val |= ((m->vsync_start - m->vdisplay) << 12);
1850 hdmi_set_reg(core->v_sync_gen1, 3, val);
1851
1852 hdmi_set_reg(core->v_sync_gen2, 3, 0x1001);/* Reset value */
1853 hdmi_set_reg(core->v_sync_gen3, 3, 0x1001);/* Reset value */
1854 hdmi_set_reg(tg->vact_st, 2, m->vtotal - m->vdisplay);
1855 hdmi_set_reg(tg->vact_sz, 2, m->vdisplay);
1856 hdmi_set_reg(tg->vact_st2, 2, 0x248); /* Reset value */
1857 }
1858
1859 /* Timing generator registers */
1860 hdmi_set_reg(tg->cmd, 1, 0x0);
1861 hdmi_set_reg(tg->h_fsz, 2, m->htotal);
1862 hdmi_set_reg(tg->hact_st, 2, m->htotal - m->hdisplay);
1863 hdmi_set_reg(tg->hact_sz, 2, m->hdisplay);
1864 hdmi_set_reg(tg->v_fsz, 2, m->vtotal);
1865 hdmi_set_reg(tg->vsync, 2, 0x1);
1866 hdmi_set_reg(tg->vsync2, 2, 0x233); /* Reset value */
1867 hdmi_set_reg(tg->field_chg, 2, 0x233); /* Reset value */
1868 hdmi_set_reg(tg->vsync_top_hdmi, 2, 0x1); /* Reset value */
1869 hdmi_set_reg(tg->vsync_bot_hdmi, 2, 0x233); /* Reset value */
1870 hdmi_set_reg(tg->field_top_hdmi, 2, 0x1); /* Reset value */
1871 hdmi_set_reg(tg->field_bot_hdmi, 2, 0x233); /* Reset value */
1872 hdmi_set_reg(tg->tg_3d, 1, 0x0); /* Not used */
1873}
1874
Sean Paul2f7e2ed2013-01-15 08:11:08 -05001875static void hdmi_v14_mode_set(struct hdmi_context *hdata,
1876 struct drm_display_mode *m)
1877{
Andrzej Hajdac93aaeb2015-07-09 16:28:10 +02001878 struct hdmi_tg_regs *tg = &hdata->mode_conf.v14_conf.tg;
1879 struct hdmi_v14_core_regs *core = &hdata->mode_conf.v14_conf.core;
Rahul Sharma6b986ed2013-03-06 17:33:29 +09001880
Sean Paul2f7e2ed2013-01-15 08:11:08 -05001881 hdmi_set_reg(core->h_blank, 2, m->htotal - m->hdisplay);
1882 hdmi_set_reg(core->v_line, 2, m->vtotal);
1883 hdmi_set_reg(core->h_line, 2, m->htotal);
1884 hdmi_set_reg(core->hsync_pol, 1,
1885 (m->flags & DRM_MODE_FLAG_NHSYNC) ? 1 : 0);
1886 hdmi_set_reg(core->vsync_pol, 1,
1887 (m->flags & DRM_MODE_FLAG_NVSYNC) ? 1 : 0);
1888 hdmi_set_reg(core->int_pro_mode, 1,
1889 (m->flags & DRM_MODE_FLAG_INTERLACE) ? 1 : 0);
1890
1891 /*
1892 * Quirk requirement for exynos 5 HDMI IP design,
1893 * 2 pixels less than the actual calculation for hsync_start
1894 * and end.
1895 */
1896
1897 /* Following values & calculations differ for different type of modes */
1898 if (m->flags & DRM_MODE_FLAG_INTERLACE) {
1899 /* Interlaced Mode */
1900 hdmi_set_reg(core->v_sync_line_bef_2, 2,
1901 (m->vsync_end - m->vdisplay) / 2);
1902 hdmi_set_reg(core->v_sync_line_bef_1, 2,
1903 (m->vsync_start - m->vdisplay) / 2);
1904 hdmi_set_reg(core->v2_blank, 2, m->vtotal / 2);
1905 hdmi_set_reg(core->v1_blank, 2, (m->vtotal - m->vdisplay) / 2);
Rahul Sharma14829952013-06-18 18:19:37 +05301906 hdmi_set_reg(core->v_blank_f0, 2, m->vtotal - m->vdisplay / 2);
Sean Paul2f7e2ed2013-01-15 08:11:08 -05001907 hdmi_set_reg(core->v_blank_f1, 2, m->vtotal);
1908 hdmi_set_reg(core->v_sync_line_aft_2, 2, (m->vtotal / 2) + 7);
1909 hdmi_set_reg(core->v_sync_line_aft_1, 2, (m->vtotal / 2) + 2);
1910 hdmi_set_reg(core->v_sync_line_aft_pxl_2, 2,
1911 (m->htotal / 2) + (m->hsync_start - m->hdisplay));
1912 hdmi_set_reg(core->v_sync_line_aft_pxl_1, 2,
1913 (m->htotal / 2) + (m->hsync_start - m->hdisplay));
1914 hdmi_set_reg(tg->vact_st, 2, (m->vtotal - m->vdisplay) / 2);
1915 hdmi_set_reg(tg->vact_sz, 2, m->vdisplay / 2);
Rahul Sharma14829952013-06-18 18:19:37 +05301916 hdmi_set_reg(tg->vact_st2, 2, m->vtotal - m->vdisplay / 2);
1917 hdmi_set_reg(tg->vsync2, 2, (m->vtotal / 2) + 1);
1918 hdmi_set_reg(tg->vsync_bot_hdmi, 2, (m->vtotal / 2) + 1);
1919 hdmi_set_reg(tg->field_bot_hdmi, 2, (m->vtotal / 2) + 1);
Sean Paul2f7e2ed2013-01-15 08:11:08 -05001920 hdmi_set_reg(tg->vact_st3, 2, 0x0);
1921 hdmi_set_reg(tg->vact_st4, 2, 0x0);
1922 } else {
1923 /* Progressive Mode */
1924 hdmi_set_reg(core->v_sync_line_bef_2, 2,
1925 m->vsync_end - m->vdisplay);
1926 hdmi_set_reg(core->v_sync_line_bef_1, 2,
1927 m->vsync_start - m->vdisplay);
1928 hdmi_set_reg(core->v2_blank, 2, m->vtotal);
1929 hdmi_set_reg(core->v1_blank, 2, m->vtotal - m->vdisplay);
1930 hdmi_set_reg(core->v_blank_f0, 2, 0xffff);
1931 hdmi_set_reg(core->v_blank_f1, 2, 0xffff);
1932 hdmi_set_reg(core->v_sync_line_aft_2, 2, 0xffff);
1933 hdmi_set_reg(core->v_sync_line_aft_1, 2, 0xffff);
1934 hdmi_set_reg(core->v_sync_line_aft_pxl_2, 2, 0xffff);
1935 hdmi_set_reg(core->v_sync_line_aft_pxl_1, 2, 0xffff);
1936 hdmi_set_reg(tg->vact_st, 2, m->vtotal - m->vdisplay);
1937 hdmi_set_reg(tg->vact_sz, 2, m->vdisplay);
1938 hdmi_set_reg(tg->vact_st2, 2, 0x248); /* Reset value */
1939 hdmi_set_reg(tg->vact_st3, 2, 0x47b); /* Reset value */
1940 hdmi_set_reg(tg->vact_st4, 2, 0x6ae); /* Reset value */
Rahul Sharma14829952013-06-18 18:19:37 +05301941 hdmi_set_reg(tg->vsync2, 2, 0x233); /* Reset value */
1942 hdmi_set_reg(tg->vsync_bot_hdmi, 2, 0x233); /* Reset value */
1943 hdmi_set_reg(tg->field_bot_hdmi, 2, 0x233); /* Reset value */
Sean Paul2f7e2ed2013-01-15 08:11:08 -05001944 }
1945
1946 /* Following values & calculations are same irrespective of mode type */
1947 hdmi_set_reg(core->h_sync_start, 2, m->hsync_start - m->hdisplay - 2);
1948 hdmi_set_reg(core->h_sync_end, 2, m->hsync_end - m->hdisplay - 2);
1949 hdmi_set_reg(core->vact_space_1, 2, 0xffff);
1950 hdmi_set_reg(core->vact_space_2, 2, 0xffff);
1951 hdmi_set_reg(core->vact_space_3, 2, 0xffff);
1952 hdmi_set_reg(core->vact_space_4, 2, 0xffff);
1953 hdmi_set_reg(core->vact_space_5, 2, 0xffff);
1954 hdmi_set_reg(core->vact_space_6, 2, 0xffff);
1955 hdmi_set_reg(core->v_blank_f2, 2, 0xffff);
1956 hdmi_set_reg(core->v_blank_f3, 2, 0xffff);
1957 hdmi_set_reg(core->v_blank_f4, 2, 0xffff);
1958 hdmi_set_reg(core->v_blank_f5, 2, 0xffff);
1959 hdmi_set_reg(core->v_sync_line_aft_3, 2, 0xffff);
1960 hdmi_set_reg(core->v_sync_line_aft_4, 2, 0xffff);
1961 hdmi_set_reg(core->v_sync_line_aft_5, 2, 0xffff);
1962 hdmi_set_reg(core->v_sync_line_aft_6, 2, 0xffff);
1963 hdmi_set_reg(core->v_sync_line_aft_pxl_3, 2, 0xffff);
1964 hdmi_set_reg(core->v_sync_line_aft_pxl_4, 2, 0xffff);
1965 hdmi_set_reg(core->v_sync_line_aft_pxl_5, 2, 0xffff);
1966 hdmi_set_reg(core->v_sync_line_aft_pxl_6, 2, 0xffff);
1967
1968 /* Timing generator registers */
1969 hdmi_set_reg(tg->cmd, 1, 0x0);
1970 hdmi_set_reg(tg->h_fsz, 2, m->htotal);
1971 hdmi_set_reg(tg->hact_st, 2, m->htotal - m->hdisplay);
1972 hdmi_set_reg(tg->hact_sz, 2, m->hdisplay);
1973 hdmi_set_reg(tg->v_fsz, 2, m->vtotal);
1974 hdmi_set_reg(tg->vsync, 2, 0x1);
Sean Paul2f7e2ed2013-01-15 08:11:08 -05001975 hdmi_set_reg(tg->field_chg, 2, 0x233); /* Reset value */
1976 hdmi_set_reg(tg->vsync_top_hdmi, 2, 0x1); /* Reset value */
Sean Paul2f7e2ed2013-01-15 08:11:08 -05001977 hdmi_set_reg(tg->field_top_hdmi, 2, 0x1); /* Reset value */
Sean Paul2f7e2ed2013-01-15 08:11:08 -05001978 hdmi_set_reg(tg->tg_3d, 1, 0x0);
Sean Paul2f7e2ed2013-01-15 08:11:08 -05001979}
1980
Sean Paulf041b252014-01-30 16:19:15 -05001981static void hdmi_mode_set(struct exynos_drm_display *display,
1982 struct drm_display_mode *mode)
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001983{
Andrzej Hajda0d8424f82014-11-17 09:54:21 +01001984 struct hdmi_context *hdata = display_to_hdmi(display);
Rahul Sharma6b986ed2013-03-06 17:33:29 +09001985 struct drm_display_mode *m = mode;
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001986
YoungJun Chocbc4c332013-06-12 10:44:40 +09001987 DRM_DEBUG_KMS("xres=%d, yres=%d, refresh=%d, intl=%s\n",
1988 m->hdisplay, m->vdisplay,
Rahul Sharma6b986ed2013-03-06 17:33:29 +09001989 m->vrefresh, (m->flags & DRM_MODE_FLAG_INTERLACE) ?
Tobias Jakobi1e6d4592015-04-07 01:14:50 +02001990 "INTERLACED" : "PROGRESSIVE");
Seung-Woo Kimd8408322011-12-21 17:39:39 +09001991
Rahul Sharmabfa48422014-04-03 20:41:04 +05301992 /* preserve mode information for later use. */
1993 drm_mode_copy(&hdata->current_mode, mode);
1994
Andrzej Hajdac93aaeb2015-07-09 16:28:10 +02001995 hdata->cea_video_id = drm_match_cea_mode(mode);
1996
Andrzej Hajdacd240cd2015-07-09 16:28:09 +02001997 if (hdata->drv_data->type == HDMI_TYPE13)
Rahul Sharma6b986ed2013-03-06 17:33:29 +09001998 hdmi_v13_mode_set(hdata, mode);
Sachin Kamat5f46c332013-04-26 11:29:00 +05301999 else
Sean Paul2f7e2ed2013-01-15 08:11:08 -05002000 hdmi_v14_mode_set(hdata, mode);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002001}
2002
Sean Paulf041b252014-01-30 16:19:15 -05002003static void hdmi_commit(struct exynos_drm_display *display)
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002004{
Andrzej Hajda0d8424f82014-11-17 09:54:21 +01002005 struct hdmi_context *hdata = display_to_hdmi(display);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002006
Andrzej Hajda882a0642015-07-09 16:28:08 +02002007 if (!hdata->powered)
Shirish Sdda90122013-01-23 22:03:18 -05002008 return;
Shirish Sdda90122013-01-23 22:03:18 -05002009
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002010 hdmi_conf_apply(hdata);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002011}
2012
Joonyoung Shim92dc7a02015-01-30 16:43:02 +09002013static void hdmi_poweron(struct hdmi_context *hdata)
Joonyoung Shimcf8fc4f2012-04-23 19:35:50 +09002014{
2015 struct hdmi_resources *res = &hdata->res;
2016
Andrzej Hajda882a0642015-07-09 16:28:08 +02002017 if (hdata->powered)
Joonyoung Shimcf8fc4f2012-04-23 19:35:50 +09002018 return;
Joonyoung Shimcf8fc4f2012-04-23 19:35:50 +09002019
2020 hdata->powered = true;
2021
Sean Paulaf65c802014-01-30 16:19:27 -05002022 pm_runtime_get_sync(hdata->dev);
2023
Seung-Woo Kimad079452013-06-05 14:34:38 +09002024 if (regulator_bulk_enable(res->regul_count, res->regul_bulk))
2025 DRM_DEBUG_KMS("failed to enable regulator bulk\n");
2026
Rahul Sharma049d34e2014-05-20 10:36:05 +05302027 /* set pmu hdmiphy control bit to enable hdmiphy */
2028 regmap_update_bits(hdata->pmureg, PMU_HDMI_PHY_CONTROL,
2029 PMU_HDMI_PHY_ENABLE_BIT, 1);
2030
Sean Paul0bfb1f82013-06-11 12:24:02 +05302031 clk_prepare_enable(res->hdmi);
2032 clk_prepare_enable(res->sclk_hdmi);
Rahul Sharmaa5562252012-11-28 11:30:25 +05302033
2034 hdmiphy_poweron(hdata);
Joonyoung Shim92dc7a02015-01-30 16:43:02 +09002035 hdmi_commit(&hdata->display);
Joonyoung Shimcf8fc4f2012-04-23 19:35:50 +09002036}
2037
Joonyoung Shim92dc7a02015-01-30 16:43:02 +09002038static void hdmi_poweroff(struct hdmi_context *hdata)
Joonyoung Shimcf8fc4f2012-04-23 19:35:50 +09002039{
2040 struct hdmi_resources *res = &hdata->res;
2041
Joonyoung Shimcf8fc4f2012-04-23 19:35:50 +09002042 if (!hdata->powered)
Andrzej Hajda882a0642015-07-09 16:28:08 +02002043 return;
Joonyoung Shimcf8fc4f2012-04-23 19:35:50 +09002044
Rahul Sharmabfa48422014-04-03 20:41:04 +05302045 /* HDMI System Disable */
2046 hdmi_reg_writemask(hdata, HDMI_CON_0, 0, HDMI_EN);
2047
Rahul Sharmaa5562252012-11-28 11:30:25 +05302048 hdmiphy_poweroff(hdata);
Joonyoung Shimcf8fc4f2012-04-23 19:35:50 +09002049
Sean Paul724fd142014-05-09 15:05:10 +09002050 cancel_delayed_work(&hdata->hotplug_work);
2051
Sean Paul0bfb1f82013-06-11 12:24:02 +05302052 clk_disable_unprepare(res->sclk_hdmi);
2053 clk_disable_unprepare(res->hdmi);
Rahul Sharma049d34e2014-05-20 10:36:05 +05302054
2055 /* reset pmu hdmiphy control bit to disable hdmiphy */
2056 regmap_update_bits(hdata->pmureg, PMU_HDMI_PHY_CONTROL,
2057 PMU_HDMI_PHY_ENABLE_BIT, 0);
2058
Joonyoung Shimcf8fc4f2012-04-23 19:35:50 +09002059 regulator_bulk_disable(res->regul_count, res->regul_bulk);
2060
Sean Paulaf65c802014-01-30 16:19:27 -05002061 pm_runtime_put_sync(hdata->dev);
Joonyoung Shimcf8fc4f2012-04-23 19:35:50 +09002062
2063 hdata->powered = false;
Joonyoung Shimcf8fc4f2012-04-23 19:35:50 +09002064}
2065
Sean Paulf041b252014-01-30 16:19:15 -05002066static void hdmi_dpms(struct exynos_drm_display *display, int mode)
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002067{
Andrzej Hajda0d8424f82014-11-17 09:54:21 +01002068 struct hdmi_context *hdata = display_to_hdmi(display);
Inki Dae245f98f2014-06-13 17:44:40 +09002069 struct drm_encoder *encoder = hdata->encoder;
2070 struct drm_crtc *crtc = encoder->crtc;
Jani Nikulab0f87782015-03-11 11:50:59 +02002071 const struct drm_crtc_helper_funcs *funcs = NULL;
Inki Dae245f98f2014-06-13 17:44:40 +09002072
YoungJun Chocbc4c332013-06-12 10:44:40 +09002073 DRM_DEBUG_KMS("mode %d\n", mode);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002074
Joonyoung Shimcf8fc4f2012-04-23 19:35:50 +09002075 switch (mode) {
2076 case DRM_MODE_DPMS_ON:
Joonyoung Shim92dc7a02015-01-30 16:43:02 +09002077 hdmi_poweron(hdata);
Joonyoung Shimcf8fc4f2012-04-23 19:35:50 +09002078 break;
2079 case DRM_MODE_DPMS_STANDBY:
2080 case DRM_MODE_DPMS_SUSPEND:
2081 case DRM_MODE_DPMS_OFF:
Inki Dae245f98f2014-06-13 17:44:40 +09002082 /*
2083 * The SFRs of VP and Mixer are updated by Vertical Sync of
2084 * Timing generator which is a part of HDMI so the sequence
2085 * to disable TV Subsystem should be as following,
2086 * VP -> Mixer -> HDMI
2087 *
2088 * Below codes will try to disable Mixer and VP(if used)
2089 * prior to disabling HDMI.
2090 */
2091 if (crtc)
2092 funcs = crtc->helper_private;
Gustavo Padovan63498e32015-06-01 12:04:53 -03002093 if (funcs && funcs->disable)
2094 (*funcs->disable)(crtc);
Inki Dae245f98f2014-06-13 17:44:40 +09002095
Joonyoung Shim92dc7a02015-01-30 16:43:02 +09002096 hdmi_poweroff(hdata);
Joonyoung Shimcf8fc4f2012-04-23 19:35:50 +09002097 break;
2098 default:
2099 DRM_DEBUG_KMS("unknown dpms mode: %d\n", mode);
2100 break;
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002101 }
2102}
2103
Sean Paulf041b252014-01-30 16:19:15 -05002104static struct exynos_drm_display_ops hdmi_display_ops = {
Sean Pauld9716ee2014-01-30 16:19:29 -05002105 .create_connector = hdmi_create_connector,
Sean Paulf041b252014-01-30 16:19:15 -05002106 .mode_fixup = hdmi_mode_fixup,
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002107 .mode_set = hdmi_mode_set,
Sean Paulf041b252014-01-30 16:19:15 -05002108 .dpms = hdmi_dpms,
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002109 .commit = hdmi_commit,
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002110};
2111
Sean Paul724fd142014-05-09 15:05:10 +09002112static void hdmi_hotplug_work_func(struct work_struct *work)
Joonyoung Shimcf8fc4f2012-04-23 19:35:50 +09002113{
Sean Paul724fd142014-05-09 15:05:10 +09002114 struct hdmi_context *hdata;
2115
2116 hdata = container_of(work, struct hdmi_context, hotplug_work.work);
Joonyoung Shimcf8fc4f2012-04-23 19:35:50 +09002117
Sean Paul45517892014-01-30 16:19:05 -05002118 if (hdata->drm_dev)
2119 drm_helper_hpd_irq_event(hdata->drm_dev);
Sean Paul724fd142014-05-09 15:05:10 +09002120}
2121
2122static irqreturn_t hdmi_irq_thread(int irq, void *arg)
2123{
2124 struct hdmi_context *hdata = arg;
2125
2126 mod_delayed_work(system_wq, &hdata->hotplug_work,
2127 msecs_to_jiffies(HOTPLUG_DEBOUNCE_MS));
Joonyoung Shimcf8fc4f2012-04-23 19:35:50 +09002128
Joonyoung Shimcf8fc4f2012-04-23 19:35:50 +09002129 return IRQ_HANDLED;
2130}
2131
Greg Kroah-Hartman56550d92012-12-21 15:09:25 -08002132static int hdmi_resources_init(struct hdmi_context *hdata)
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002133{
2134 struct device *dev = hdata->dev;
2135 struct hdmi_resources *res = &hdata->res;
2136 static char *supply[] = {
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002137 "vdd",
2138 "vdd_osc",
2139 "vdd_pll",
2140 };
2141 int i, ret;
2142
2143 DRM_DEBUG_KMS("HDMI resource init\n");
2144
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002145 /* get clocks, power */
Sachin Kamat9f49d9f2012-11-23 14:13:27 +05302146 res->hdmi = devm_clk_get(dev, "hdmi");
Sachin Kamatee7cbaf2013-03-21 15:33:57 +05302147 if (IS_ERR(res->hdmi)) {
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002148 DRM_ERROR("failed to get clock 'hdmi'\n");
Inki Daedf5225b2014-05-29 18:28:02 +09002149 ret = PTR_ERR(res->hdmi);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002150 goto fail;
2151 }
Sachin Kamat9f49d9f2012-11-23 14:13:27 +05302152 res->sclk_hdmi = devm_clk_get(dev, "sclk_hdmi");
Sachin Kamatee7cbaf2013-03-21 15:33:57 +05302153 if (IS_ERR(res->sclk_hdmi)) {
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002154 DRM_ERROR("failed to get clock 'sclk_hdmi'\n");
Inki Daedf5225b2014-05-29 18:28:02 +09002155 ret = PTR_ERR(res->sclk_hdmi);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002156 goto fail;
2157 }
Sachin Kamat9f49d9f2012-11-23 14:13:27 +05302158 res->sclk_pixel = devm_clk_get(dev, "sclk_pixel");
Sachin Kamatee7cbaf2013-03-21 15:33:57 +05302159 if (IS_ERR(res->sclk_pixel)) {
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002160 DRM_ERROR("failed to get clock 'sclk_pixel'\n");
Inki Daedf5225b2014-05-29 18:28:02 +09002161 ret = PTR_ERR(res->sclk_pixel);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002162 goto fail;
2163 }
Sachin Kamat9f49d9f2012-11-23 14:13:27 +05302164 res->sclk_hdmiphy = devm_clk_get(dev, "sclk_hdmiphy");
Sachin Kamatee7cbaf2013-03-21 15:33:57 +05302165 if (IS_ERR(res->sclk_hdmiphy)) {
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002166 DRM_ERROR("failed to get clock 'sclk_hdmiphy'\n");
Inki Daedf5225b2014-05-29 18:28:02 +09002167 ret = PTR_ERR(res->sclk_hdmiphy);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002168 goto fail;
2169 }
Rahul Sharma59956d32013-06-11 12:24:03 +05302170 res->mout_hdmi = devm_clk_get(dev, "mout_hdmi");
2171 if (IS_ERR(res->mout_hdmi)) {
2172 DRM_ERROR("failed to get clock 'mout_hdmi'\n");
Inki Daedf5225b2014-05-29 18:28:02 +09002173 ret = PTR_ERR(res->mout_hdmi);
Rahul Sharma59956d32013-06-11 12:24:03 +05302174 goto fail;
2175 }
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002176
Rahul Sharma59956d32013-06-11 12:24:03 +05302177 clk_set_parent(res->mout_hdmi, res->sclk_pixel);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002178
Sachin Kamat9f49d9f2012-11-23 14:13:27 +05302179 res->regul_bulk = devm_kzalloc(dev, ARRAY_SIZE(supply) *
Sachin Kamatadc837a2012-08-31 15:50:47 +05302180 sizeof(res->regul_bulk[0]), GFP_KERNEL);
Inki Daedf5225b2014-05-29 18:28:02 +09002181 if (!res->regul_bulk) {
2182 ret = -ENOMEM;
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002183 goto fail;
Inki Daedf5225b2014-05-29 18:28:02 +09002184 }
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002185 for (i = 0; i < ARRAY_SIZE(supply); ++i) {
2186 res->regul_bulk[i].supply = supply[i];
2187 res->regul_bulk[i].consumer = NULL;
2188 }
Sachin Kamat9f49d9f2012-11-23 14:13:27 +05302189 ret = devm_regulator_bulk_get(dev, ARRAY_SIZE(supply), res->regul_bulk);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002190 if (ret) {
2191 DRM_ERROR("failed to get regulators\n");
Inki Daedf5225b2014-05-29 18:28:02 +09002192 return ret;
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002193 }
2194 res->regul_count = ARRAY_SIZE(supply);
2195
Marek Szyprowski05fdf982014-07-01 10:10:06 +02002196 res->reg_hdmi_en = devm_regulator_get(dev, "hdmi-en");
2197 if (IS_ERR(res->reg_hdmi_en) && PTR_ERR(res->reg_hdmi_en) != -ENOENT) {
2198 DRM_ERROR("failed to get hdmi-en regulator\n");
2199 return PTR_ERR(res->reg_hdmi_en);
2200 }
2201 if (!IS_ERR(res->reg_hdmi_en)) {
2202 ret = regulator_enable(res->reg_hdmi_en);
2203 if (ret) {
2204 DRM_ERROR("failed to enable hdmi-en regulator\n");
2205 return ret;
2206 }
2207 } else
2208 res->reg_hdmi_en = NULL;
2209
Inki Daedf5225b2014-05-29 18:28:02 +09002210 return ret;
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002211fail:
2212 DRM_ERROR("HDMI resource init - failed\n");
Inki Daedf5225b2014-05-29 18:28:02 +09002213 return ret;
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002214}
2215
Rahul Sharma22c4f422012-10-04 20:48:55 +05302216static struct of_device_id hdmi_match_types[] = {
2217 {
2218 .compatible = "samsung,exynos5-hdmi",
Inki Daebfe4e842014-03-06 14:18:17 +09002219 .data = &exynos5_hdmi_driver_data,
Tomasz Stanislawskic119ed02012-10-04 20:48:44 +05302220 }, {
Marek Szyprowskiff830c92014-07-01 10:10:07 +02002221 .compatible = "samsung,exynos4210-hdmi",
2222 .data = &exynos4210_hdmi_driver_data,
2223 }, {
Rahul Sharmacc57caf2013-06-19 18:21:07 +05302224 .compatible = "samsung,exynos4212-hdmi",
Inki Daebfe4e842014-03-06 14:18:17 +09002225 .data = &exynos4212_hdmi_driver_data,
Rahul Sharmacc57caf2013-06-19 18:21:07 +05302226 }, {
Rahul Sharmaa18a2dd2014-04-20 15:51:17 +05302227 .compatible = "samsung,exynos5420-hdmi",
2228 .data = &exynos5420_hdmi_driver_data,
2229 }, {
Tomasz Stanislawskic119ed02012-10-04 20:48:44 +05302230 /* end node */
2231 }
2232};
Sjoerd Simons39b58a32014-07-18 22:36:41 +02002233MODULE_DEVICE_TABLE (of, hdmi_match_types);
Tomasz Stanislawskic119ed02012-10-04 20:48:44 +05302234
Inki Daef37cd5e2014-05-09 14:25:20 +09002235static int hdmi_bind(struct device *dev, struct device *master, void *data)
2236{
2237 struct drm_device *drm_dev = data;
Andrzej Hajda930865f2014-11-17 09:54:20 +01002238 struct hdmi_context *hdata = dev_get_drvdata(dev);
Inki Daef37cd5e2014-05-09 14:25:20 +09002239
Inki Daef37cd5e2014-05-09 14:25:20 +09002240 hdata->drm_dev = drm_dev;
2241
Andrzej Hajda930865f2014-11-17 09:54:20 +01002242 return exynos_drm_create_enc_conn(drm_dev, &hdata->display);
Inki Daef37cd5e2014-05-09 14:25:20 +09002243}
2244
2245static void hdmi_unbind(struct device *dev, struct device *master, void *data)
2246{
Inki Daef37cd5e2014-05-09 14:25:20 +09002247}
2248
2249static const struct component_ops hdmi_component_ops = {
2250 .bind = hdmi_bind,
2251 .unbind = hdmi_unbind,
2252};
2253
Inki Daee2a562d2014-05-09 16:46:10 +09002254static struct device_node *hdmi_legacy_ddc_dt_binding(struct device *dev)
2255{
2256 const char *compatible_str = "samsung,exynos4210-hdmiddc";
2257 struct device_node *np;
2258
2259 np = of_find_compatible_node(NULL, NULL, compatible_str);
2260 if (np)
2261 return of_get_next_parent(np);
2262
2263 return NULL;
2264}
2265
2266static struct device_node *hdmi_legacy_phy_dt_binding(struct device *dev)
2267{
2268 const char *compatible_str = "samsung,exynos4212-hdmiphy";
2269
2270 return of_find_compatible_node(NULL, NULL, compatible_str);
2271}
2272
Greg Kroah-Hartman56550d92012-12-21 15:09:25 -08002273static int hdmi_probe(struct platform_device *pdev)
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002274{
Inki Daef37cd5e2014-05-09 14:25:20 +09002275 struct device_node *ddc_node, *phy_node;
Inki Daef37cd5e2014-05-09 14:25:20 +09002276 const struct of_device_id *match;
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002277 struct device *dev = &pdev->dev;
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002278 struct hdmi_context *hdata;
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002279 struct resource *res;
2280 int ret;
2281
Andrzej Hajda930865f2014-11-17 09:54:20 +01002282 hdata = devm_kzalloc(dev, sizeof(struct hdmi_context), GFP_KERNEL);
2283 if (!hdata)
2284 return -ENOMEM;
2285
Andrzej Hajdacd240cd2015-07-09 16:28:09 +02002286 match = of_match_device(hdmi_match_types, dev);
2287 if (!match)
2288 return -ENODEV;
2289
2290 hdata->drv_data = match->data;
Andrzej Hajda930865f2014-11-17 09:54:20 +01002291 hdata->display.type = EXYNOS_DISPLAY_TYPE_HDMI;
2292 hdata->display.ops = &hdmi_display_ops;
2293
Andrzej Hajda930865f2014-11-17 09:54:20 +01002294 platform_set_drvdata(pdev, hdata);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002295
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002296 hdata->dev = dev;
Andrzej Hajdad36b3002015-07-09 16:28:06 +02002297 hdata->hpd_gpio = of_get_named_gpio(dev->of_node, "hpd-gpio", 0);
2298 if (hdata->hpd_gpio < 0) {
2299 DRM_ERROR("cannot get hpd gpio property\n");
2300 return hdata->hpd_gpio;
2301 }
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002302
2303 ret = hdmi_resources_init(hdata);
2304 if (ret) {
Rahul Sharma22c4f422012-10-04 20:48:55 +05302305 DRM_ERROR("hdmi_resources_init failed\n");
Inki Daedf5225b2014-05-29 18:28:02 +09002306 return ret;
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002307 }
2308
2309 res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
Seung-Woo Kimd873ab92013-05-22 21:14:14 +09002310 hdata->regs = devm_ioremap_resource(dev, res);
Inki Daedf5225b2014-05-29 18:28:02 +09002311 if (IS_ERR(hdata->regs)) {
2312 ret = PTR_ERR(hdata->regs);
Andrzej Hajda86650402015-06-11 23:23:37 +09002313 return ret;
Inki Daedf5225b2014-05-29 18:28:02 +09002314 }
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002315
Seung-Woo Kimd873ab92013-05-22 21:14:14 +09002316 ret = devm_gpio_request(dev, hdata->hpd_gpio, "HPD");
Tomasz Stanislawskifca57122012-10-04 20:48:46 +05302317 if (ret) {
2318 DRM_ERROR("failed to request HPD gpio\n");
Andrzej Hajda86650402015-06-11 23:23:37 +09002319 return ret;
Tomasz Stanislawskifca57122012-10-04 20:48:46 +05302320 }
2321
Inki Daee2a562d2014-05-09 16:46:10 +09002322 ddc_node = hdmi_legacy_ddc_dt_binding(dev);
2323 if (ddc_node)
2324 goto out_get_ddc_adpt;
2325
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002326 /* DDC i2c driver */
Daniel Kurtz2b768132014-02-24 18:52:51 +09002327 ddc_node = of_parse_phandle(dev->of_node, "ddc", 0);
2328 if (!ddc_node) {
2329 DRM_ERROR("Failed to find ddc node in device tree\n");
Andrzej Hajda86650402015-06-11 23:23:37 +09002330 return -ENODEV;
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002331 }
Inki Daee2a562d2014-05-09 16:46:10 +09002332
2333out_get_ddc_adpt:
Inki Dae8fa04aa2014-03-13 16:38:31 +09002334 hdata->ddc_adpt = of_find_i2c_adapter_by_node(ddc_node);
2335 if (!hdata->ddc_adpt) {
2336 DRM_ERROR("Failed to get ddc i2c adapter by node\n");
Inki Daedf5225b2014-05-29 18:28:02 +09002337 return -EPROBE_DEFER;
Daniel Kurtz2b768132014-02-24 18:52:51 +09002338 }
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002339
Inki Daee2a562d2014-05-09 16:46:10 +09002340 phy_node = hdmi_legacy_phy_dt_binding(dev);
2341 if (phy_node)
2342 goto out_get_phy_port;
2343
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002344 /* hdmiphy i2c driver */
Daniel Kurtz2b768132014-02-24 18:52:51 +09002345 phy_node = of_parse_phandle(dev->of_node, "phy", 0);
2346 if (!phy_node) {
2347 DRM_ERROR("Failed to find hdmiphy node in device tree\n");
2348 ret = -ENODEV;
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002349 goto err_ddc;
2350 }
Rahul Sharmad5e9ca42014-05-09 15:34:18 +09002351
Inki Daee2a562d2014-05-09 16:46:10 +09002352out_get_phy_port:
Andrzej Hajdacd240cd2015-07-09 16:28:09 +02002353 if (hdata->drv_data->is_apb_phy) {
Rahul Sharmad5e9ca42014-05-09 15:34:18 +09002354 hdata->regs_hdmiphy = of_iomap(phy_node, 0);
2355 if (!hdata->regs_hdmiphy) {
2356 DRM_ERROR("failed to ioremap hdmi phy\n");
2357 ret = -ENOMEM;
2358 goto err_ddc;
2359 }
2360 } else {
2361 hdata->hdmiphy_port = of_find_i2c_device_by_node(phy_node);
2362 if (!hdata->hdmiphy_port) {
2363 DRM_ERROR("Failed to get hdmi phy i2c client\n");
Inki Daedf5225b2014-05-29 18:28:02 +09002364 ret = -EPROBE_DEFER;
Rahul Sharmad5e9ca42014-05-09 15:34:18 +09002365 goto err_ddc;
2366 }
Daniel Kurtz2b768132014-02-24 18:52:51 +09002367 }
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002368
Sean Paul77006a72013-01-16 10:17:20 -05002369 hdata->irq = gpio_to_irq(hdata->hpd_gpio);
2370 if (hdata->irq < 0) {
2371 DRM_ERROR("failed to get GPIO irq\n");
2372 ret = hdata->irq;
Joonyoung Shim66265a22012-04-23 19:35:49 +09002373 goto err_hdmiphy;
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002374 }
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002375
Sean Paul724fd142014-05-09 15:05:10 +09002376 INIT_DELAYED_WORK(&hdata->hotplug_work, hdmi_hotplug_work_func);
2377
Seung-Woo Kimdcb9a7c2013-05-22 21:14:17 +09002378 ret = devm_request_threaded_irq(dev, hdata->irq, NULL,
Sean Paul77006a72013-01-16 10:17:20 -05002379 hdmi_irq_thread, IRQF_TRIGGER_RISING |
Joonyoung Shimcf8fc4f2012-04-23 19:35:50 +09002380 IRQF_TRIGGER_FALLING | IRQF_ONESHOT,
Sean Paulf041b252014-01-30 16:19:15 -05002381 "hdmi", hdata);
Joonyoung Shimcf8fc4f2012-04-23 19:35:50 +09002382 if (ret) {
Sean Paul77006a72013-01-16 10:17:20 -05002383 DRM_ERROR("failed to register hdmi interrupt\n");
Joonyoung Shimcf8fc4f2012-04-23 19:35:50 +09002384 goto err_hdmiphy;
2385 }
2386
Rahul Sharma049d34e2014-05-20 10:36:05 +05302387 hdata->pmureg = syscon_regmap_lookup_by_phandle(dev->of_node,
2388 "samsung,syscon-phandle");
2389 if (IS_ERR(hdata->pmureg)) {
2390 DRM_ERROR("syscon regmap lookup failed.\n");
Inki Daedf5225b2014-05-29 18:28:02 +09002391 ret = -EPROBE_DEFER;
Rahul Sharma049d34e2014-05-20 10:36:05 +05302392 goto err_hdmiphy;
2393 }
2394
Sean Paulaf65c802014-01-30 16:19:27 -05002395 pm_runtime_enable(dev);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002396
Inki Daedf5225b2014-05-29 18:28:02 +09002397 ret = component_add(&pdev->dev, &hdmi_component_ops);
2398 if (ret)
2399 goto err_disable_pm_runtime;
2400
2401 return ret;
2402
2403err_disable_pm_runtime:
2404 pm_runtime_disable(dev);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002405
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002406err_hdmiphy:
Paul Taysomb21a3bf2014-05-09 15:06:28 +09002407 if (hdata->hdmiphy_port)
2408 put_device(&hdata->hdmiphy_port->dev);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002409err_ddc:
Inki Dae8fa04aa2014-03-13 16:38:31 +09002410 put_device(&hdata->ddc_adpt->dev);
Inki Daedf5225b2014-05-29 18:28:02 +09002411
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002412 return ret;
2413}
2414
Greg Kroah-Hartman56550d92012-12-21 15:09:25 -08002415static int hdmi_remove(struct platform_device *pdev)
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002416{
Andrzej Hajda930865f2014-11-17 09:54:20 +01002417 struct hdmi_context *hdata = platform_get_drvdata(pdev);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002418
Sean Paul724fd142014-05-09 15:05:10 +09002419 cancel_delayed_work_sync(&hdata->hotplug_work);
2420
Marek Szyprowski05fdf982014-07-01 10:10:06 +02002421 if (hdata->res.reg_hdmi_en)
2422 regulator_disable(hdata->res.reg_hdmi_en);
2423
Seung-Woo Kim9d1e25c2014-07-28 17:15:22 +09002424 if (hdata->hdmiphy_port)
2425 put_device(&hdata->hdmiphy_port->dev);
Inki Dae8fa04aa2014-03-13 16:38:31 +09002426 put_device(&hdata->ddc_adpt->dev);
Inki Daef37cd5e2014-05-09 14:25:20 +09002427
Sean Paulaf65c802014-01-30 16:19:27 -05002428 pm_runtime_disable(&pdev->dev);
Inki Daedf5225b2014-05-29 18:28:02 +09002429 component_del(&pdev->dev, &hdmi_component_ops);
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002430
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002431 return 0;
2432}
2433
2434struct platform_driver hdmi_driver = {
2435 .probe = hdmi_probe,
Greg Kroah-Hartman56550d92012-12-21 15:09:25 -08002436 .remove = hdmi_remove,
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002437 .driver = {
Rahul Sharma22c4f422012-10-04 20:48:55 +05302438 .name = "exynos-hdmi",
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002439 .owner = THIS_MODULE,
Sachin Kamat88c49812013-08-28 10:47:57 +05302440 .of_match_table = hdmi_match_types,
Seung-Woo Kimd8408322011-12-21 17:39:39 +09002441 },
2442};