blob: 1be1e83047207b637b158947f24c2942146eeed0 [file] [log] [blame]
Eric Anholtc8b75bc2015-03-02 13:01:12 -08001/*
2 * Copyright (C) 2015 Broadcom
3 * Copyright (c) 2014 The Linux Foundation. All rights reserved.
4 * Copyright (C) 2013 Red Hat
5 * Author: Rob Clark <robdclark@gmail.com>
6 *
7 * This program is free software; you can redistribute it and/or modify it
8 * under the terms of the GNU General Public License version 2 as published by
9 * the Free Software Foundation.
10 *
11 * This program is distributed in the hope that it will be useful, but WITHOUT
12 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
14 * more details.
15 *
16 * You should have received a copy of the GNU General Public License along with
17 * this program. If not, see <http://www.gnu.org/licenses/>.
18 */
19
20/**
21 * DOC: VC4 Falcon HDMI module
22 *
Eric Anholtf6c01532017-02-27 12:11:43 -080023 * The HDMI core has a state machine and a PHY. On BCM2835, most of
24 * the unit operates off of the HSM clock from CPRMAN. It also
25 * internally uses the PLLH_PIX clock for the PHY.
26 *
27 * HDMI infoframes are kept within a small packet ram, where each
28 * packet can be individually enabled for including in a frame.
29 *
30 * HDMI audio is implemented entirely within the HDMI IP block. A
31 * register in the HDMI encoder takes SPDIF frames from the DMA engine
32 * and transfers them over an internal MAI (multi-channel audio
33 * interconnect) bus to the encoder side for insertion into the video
34 * blank regions.
35 *
36 * The driver's HDMI encoder does not yet support power management.
37 * The HDMI encoder's power domain and the HSM/pixel clocks are kept
38 * continuously running, and only the HDMI logic and packet ram are
39 * powered off/on at disable/enable time.
40 *
41 * The driver does not yet support CEC control, though the HDMI
42 * encoder block has CEC support.
Eric Anholtc8b75bc2015-03-02 13:01:12 -080043 */
44
45#include "drm_atomic_helper.h"
46#include "drm_crtc_helper.h"
47#include "drm_edid.h"
48#include "linux/clk.h"
49#include "linux/component.h"
50#include "linux/i2c.h"
51#include "linux/of_gpio.h"
52#include "linux/of_platform.h"
53#include "vc4_drv.h"
54#include "vc4_regs.h"
55
56/* General HDMI hardware state. */
57struct vc4_hdmi {
58 struct platform_device *pdev;
59
60 struct drm_encoder *encoder;
61 struct drm_connector *connector;
62
63 struct i2c_adapter *ddc;
64 void __iomem *hdmicore_regs;
65 void __iomem *hd_regs;
66 int hpd_gpio;
Eric Anholt0b06e0a2016-02-29 17:53:01 -080067 bool hpd_active_low;
Eric Anholtc8b75bc2015-03-02 13:01:12 -080068
69 struct clk *pixel_clock;
70 struct clk *hsm_clock;
71};
72
73#define HDMI_READ(offset) readl(vc4->hdmi->hdmicore_regs + offset)
74#define HDMI_WRITE(offset, val) writel(val, vc4->hdmi->hdmicore_regs + offset)
75#define HD_READ(offset) readl(vc4->hdmi->hd_regs + offset)
76#define HD_WRITE(offset, val) writel(val, vc4->hdmi->hd_regs + offset)
77
78/* VC4 HDMI encoder KMS struct */
79struct vc4_hdmi_encoder {
80 struct vc4_encoder base;
81 bool hdmi_monitor;
Eric Anholt21317b32016-09-29 15:34:43 -070082 bool limited_rgb_range;
83 bool rgb_range_selectable;
Eric Anholtc8b75bc2015-03-02 13:01:12 -080084};
85
86static inline struct vc4_hdmi_encoder *
87to_vc4_hdmi_encoder(struct drm_encoder *encoder)
88{
89 return container_of(encoder, struct vc4_hdmi_encoder, base.base);
90}
91
92/* VC4 HDMI connector KMS struct */
93struct vc4_hdmi_connector {
94 struct drm_connector base;
95
96 /* Since the connector is attached to just the one encoder,
97 * this is the reference to it so we can do the best_encoder()
98 * hook.
99 */
100 struct drm_encoder *encoder;
101};
102
103static inline struct vc4_hdmi_connector *
104to_vc4_hdmi_connector(struct drm_connector *connector)
105{
106 return container_of(connector, struct vc4_hdmi_connector, base);
107}
108
109#define HDMI_REG(reg) { reg, #reg }
110static const struct {
111 u32 reg;
112 const char *name;
113} hdmi_regs[] = {
114 HDMI_REG(VC4_HDMI_CORE_REV),
115 HDMI_REG(VC4_HDMI_SW_RESET_CONTROL),
116 HDMI_REG(VC4_HDMI_HOTPLUG_INT),
117 HDMI_REG(VC4_HDMI_HOTPLUG),
Eric Anholt936f1a52016-02-12 15:16:56 -0800118 HDMI_REG(VC4_HDMI_RAM_PACKET_CONFIG),
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800119 HDMI_REG(VC4_HDMI_HORZA),
120 HDMI_REG(VC4_HDMI_HORZB),
121 HDMI_REG(VC4_HDMI_FIFO_CTL),
122 HDMI_REG(VC4_HDMI_SCHEDULER_CONTROL),
123 HDMI_REG(VC4_HDMI_VERTA0),
124 HDMI_REG(VC4_HDMI_VERTA1),
125 HDMI_REG(VC4_HDMI_VERTB0),
126 HDMI_REG(VC4_HDMI_VERTB1),
127 HDMI_REG(VC4_HDMI_TX_PHY_RESET_CTL),
128};
129
130static const struct {
131 u32 reg;
132 const char *name;
133} hd_regs[] = {
134 HDMI_REG(VC4_HD_M_CTL),
135 HDMI_REG(VC4_HD_MAI_CTL),
136 HDMI_REG(VC4_HD_VID_CTL),
137 HDMI_REG(VC4_HD_CSC_CTL),
138 HDMI_REG(VC4_HD_FRAME_COUNT),
139};
140
141#ifdef CONFIG_DEBUG_FS
142int vc4_hdmi_debugfs_regs(struct seq_file *m, void *unused)
143{
144 struct drm_info_node *node = (struct drm_info_node *)m->private;
145 struct drm_device *dev = node->minor->dev;
146 struct vc4_dev *vc4 = to_vc4_dev(dev);
147 int i;
148
149 for (i = 0; i < ARRAY_SIZE(hdmi_regs); i++) {
150 seq_printf(m, "%s (0x%04x): 0x%08x\n",
151 hdmi_regs[i].name, hdmi_regs[i].reg,
152 HDMI_READ(hdmi_regs[i].reg));
153 }
154
155 for (i = 0; i < ARRAY_SIZE(hd_regs); i++) {
156 seq_printf(m, "%s (0x%04x): 0x%08x\n",
157 hd_regs[i].name, hd_regs[i].reg,
158 HD_READ(hd_regs[i].reg));
159 }
160
161 return 0;
162}
163#endif /* CONFIG_DEBUG_FS */
164
165static void vc4_hdmi_dump_regs(struct drm_device *dev)
166{
167 struct vc4_dev *vc4 = to_vc4_dev(dev);
168 int i;
169
170 for (i = 0; i < ARRAY_SIZE(hdmi_regs); i++) {
171 DRM_INFO("0x%04x (%s): 0x%08x\n",
172 hdmi_regs[i].reg, hdmi_regs[i].name,
173 HDMI_READ(hdmi_regs[i].reg));
174 }
175 for (i = 0; i < ARRAY_SIZE(hd_regs); i++) {
176 DRM_INFO("0x%04x (%s): 0x%08x\n",
177 hd_regs[i].reg, hd_regs[i].name,
178 HD_READ(hd_regs[i].reg));
179 }
180}
181
182static enum drm_connector_status
183vc4_hdmi_connector_detect(struct drm_connector *connector, bool force)
184{
185 struct drm_device *dev = connector->dev;
186 struct vc4_dev *vc4 = to_vc4_dev(dev);
187
188 if (vc4->hdmi->hpd_gpio) {
Eric Anholt0b06e0a2016-02-29 17:53:01 -0800189 if (gpio_get_value_cansleep(vc4->hdmi->hpd_gpio) ^
190 vc4->hdmi->hpd_active_low)
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800191 return connector_status_connected;
192 else
193 return connector_status_disconnected;
194 }
195
Eric Anholt9d44abb2016-09-14 19:21:29 +0100196 if (drm_probe_ddc(vc4->hdmi->ddc))
197 return connector_status_connected;
198
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800199 if (HDMI_READ(VC4_HDMI_HOTPLUG) & VC4_HDMI_HOTPLUG_CONNECTED)
200 return connector_status_connected;
201 else
202 return connector_status_disconnected;
203}
204
205static void vc4_hdmi_connector_destroy(struct drm_connector *connector)
206{
207 drm_connector_unregister(connector);
208 drm_connector_cleanup(connector);
209}
210
211static int vc4_hdmi_connector_get_modes(struct drm_connector *connector)
212{
213 struct vc4_hdmi_connector *vc4_connector =
214 to_vc4_hdmi_connector(connector);
215 struct drm_encoder *encoder = vc4_connector->encoder;
216 struct vc4_hdmi_encoder *vc4_encoder = to_vc4_hdmi_encoder(encoder);
217 struct drm_device *dev = connector->dev;
218 struct vc4_dev *vc4 = to_vc4_dev(dev);
219 int ret = 0;
220 struct edid *edid;
221
222 edid = drm_get_edid(connector, vc4->hdmi->ddc);
223 if (!edid)
224 return -ENODEV;
225
226 vc4_encoder->hdmi_monitor = drm_detect_hdmi_monitor(edid);
Eric Anholt21317b32016-09-29 15:34:43 -0700227
228 if (edid && edid->input & DRM_EDID_INPUT_DIGITAL) {
229 vc4_encoder->rgb_range_selectable =
230 drm_rgb_quant_range_selectable(edid);
231 }
232
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800233 drm_mode_connector_update_edid_property(connector, edid);
234 ret = drm_add_edid_modes(connector, edid);
235
236 return ret;
237}
238
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800239static const struct drm_connector_funcs vc4_hdmi_connector_funcs = {
240 .dpms = drm_atomic_helper_connector_dpms,
241 .detect = vc4_hdmi_connector_detect,
Eric Anholt682e62c2016-09-28 17:30:25 -0700242 .fill_modes = drm_helper_probe_single_connector_modes,
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800243 .destroy = vc4_hdmi_connector_destroy,
244 .reset = drm_atomic_helper_connector_reset,
245 .atomic_duplicate_state = drm_atomic_helper_connector_duplicate_state,
246 .atomic_destroy_state = drm_atomic_helper_connector_destroy_state,
247};
248
249static const struct drm_connector_helper_funcs vc4_hdmi_connector_helper_funcs = {
250 .get_modes = vc4_hdmi_connector_get_modes,
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800251};
252
253static struct drm_connector *vc4_hdmi_connector_init(struct drm_device *dev,
254 struct drm_encoder *encoder)
255{
256 struct drm_connector *connector = NULL;
257 struct vc4_hdmi_connector *hdmi_connector;
258 int ret = 0;
259
260 hdmi_connector = devm_kzalloc(dev->dev, sizeof(*hdmi_connector),
261 GFP_KERNEL);
262 if (!hdmi_connector) {
263 ret = -ENOMEM;
264 goto fail;
265 }
266 connector = &hdmi_connector->base;
267
268 hdmi_connector->encoder = encoder;
269
270 drm_connector_init(dev, connector, &vc4_hdmi_connector_funcs,
271 DRM_MODE_CONNECTOR_HDMIA);
272 drm_connector_helper_add(connector, &vc4_hdmi_connector_helper_funcs);
273
274 connector->polled = (DRM_CONNECTOR_POLL_CONNECT |
275 DRM_CONNECTOR_POLL_DISCONNECT);
276
Mario Kleineracc1be12016-07-19 20:58:58 +0200277 connector->interlace_allowed = 1;
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800278 connector->doublescan_allowed = 0;
279
280 drm_mode_connector_attach_encoder(connector, encoder);
281
282 return connector;
283
284 fail:
285 if (connector)
286 vc4_hdmi_connector_destroy(connector);
287
288 return ERR_PTR(ret);
289}
290
291static void vc4_hdmi_encoder_destroy(struct drm_encoder *encoder)
292{
293 drm_encoder_cleanup(encoder);
294}
295
296static const struct drm_encoder_funcs vc4_hdmi_encoder_funcs = {
297 .destroy = vc4_hdmi_encoder_destroy,
298};
299
Eric Anholt21317b32016-09-29 15:34:43 -0700300static int vc4_hdmi_stop_packet(struct drm_encoder *encoder,
301 enum hdmi_infoframe_type type)
302{
303 struct drm_device *dev = encoder->dev;
304 struct vc4_dev *vc4 = to_vc4_dev(dev);
305 u32 packet_id = type - 0x80;
306
307 HDMI_WRITE(VC4_HDMI_RAM_PACKET_CONFIG,
308 HDMI_READ(VC4_HDMI_RAM_PACKET_CONFIG) & ~BIT(packet_id));
309
310 return wait_for(!(HDMI_READ(VC4_HDMI_RAM_PACKET_STATUS) &
311 BIT(packet_id)), 100);
312}
313
314static void vc4_hdmi_write_infoframe(struct drm_encoder *encoder,
315 union hdmi_infoframe *frame)
316{
317 struct drm_device *dev = encoder->dev;
318 struct vc4_dev *vc4 = to_vc4_dev(dev);
319 u32 packet_id = frame->any.type - 0x80;
320 u32 packet_reg = VC4_HDMI_GCP_0 + VC4_HDMI_PACKET_STRIDE * packet_id;
321 uint8_t buffer[VC4_HDMI_PACKET_STRIDE];
322 ssize_t len, i;
323 int ret;
324
325 WARN_ONCE(!(HDMI_READ(VC4_HDMI_RAM_PACKET_CONFIG) &
326 VC4_HDMI_RAM_PACKET_ENABLE),
327 "Packet RAM has to be on to store the packet.");
328
329 len = hdmi_infoframe_pack(frame, buffer, sizeof(buffer));
330 if (len < 0)
331 return;
332
333 ret = vc4_hdmi_stop_packet(encoder, frame->any.type);
334 if (ret) {
335 DRM_ERROR("Failed to wait for infoframe to go idle: %d\n", ret);
336 return;
337 }
338
339 for (i = 0; i < len; i += 7) {
340 HDMI_WRITE(packet_reg,
341 buffer[i + 0] << 0 |
342 buffer[i + 1] << 8 |
343 buffer[i + 2] << 16);
344 packet_reg += 4;
345
346 HDMI_WRITE(packet_reg,
347 buffer[i + 3] << 0 |
348 buffer[i + 4] << 8 |
349 buffer[i + 5] << 16 |
350 buffer[i + 6] << 24);
351 packet_reg += 4;
352 }
353
354 HDMI_WRITE(VC4_HDMI_RAM_PACKET_CONFIG,
355 HDMI_READ(VC4_HDMI_RAM_PACKET_CONFIG) | BIT(packet_id));
356 ret = wait_for((HDMI_READ(VC4_HDMI_RAM_PACKET_STATUS) &
357 BIT(packet_id)), 100);
358 if (ret)
359 DRM_ERROR("Failed to wait for infoframe to start: %d\n", ret);
360}
361
362static void vc4_hdmi_set_avi_infoframe(struct drm_encoder *encoder)
363{
364 struct vc4_hdmi_encoder *vc4_encoder = to_vc4_hdmi_encoder(encoder);
365 struct drm_crtc *crtc = encoder->crtc;
366 const struct drm_display_mode *mode = &crtc->state->adjusted_mode;
367 union hdmi_infoframe frame;
368 int ret;
369
370 ret = drm_hdmi_avi_infoframe_from_display_mode(&frame.avi, mode);
371 if (ret < 0) {
372 DRM_ERROR("couldn't fill AVI infoframe\n");
373 return;
374 }
375
Ville Syrjälä779c4c22017-01-11 14:57:24 +0200376 drm_hdmi_avi_infoframe_quant_range(&frame.avi, mode,
Ville Syrjäläa2ce26f2017-01-11 14:57:23 +0200377 vc4_encoder->limited_rgb_range ?
378 HDMI_QUANTIZATION_RANGE_LIMITED :
379 HDMI_QUANTIZATION_RANGE_FULL,
380 vc4_encoder->rgb_range_selectable);
Eric Anholt21317b32016-09-29 15:34:43 -0700381
382 vc4_hdmi_write_infoframe(encoder, &frame);
383}
384
385static void vc4_hdmi_set_spd_infoframe(struct drm_encoder *encoder)
386{
387 union hdmi_infoframe frame;
388 int ret;
389
390 ret = hdmi_spd_infoframe_init(&frame.spd, "Broadcom", "Videocore");
391 if (ret < 0) {
392 DRM_ERROR("couldn't fill SPD infoframe\n");
393 return;
394 }
395
396 frame.spd.sdi = HDMI_SPD_SDI_PC;
397
398 vc4_hdmi_write_infoframe(encoder, &frame);
399}
400
401static void vc4_hdmi_set_infoframes(struct drm_encoder *encoder)
402{
403 vc4_hdmi_set_avi_infoframe(encoder);
404 vc4_hdmi_set_spd_infoframe(encoder);
405}
406
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800407static void vc4_hdmi_encoder_mode_set(struct drm_encoder *encoder,
408 struct drm_display_mode *unadjusted_mode,
409 struct drm_display_mode *mode)
410{
Eric Anholt6e1cbba2016-09-16 10:59:45 +0100411 struct vc4_hdmi_encoder *vc4_encoder = to_vc4_hdmi_encoder(encoder);
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800412 struct drm_device *dev = encoder->dev;
413 struct vc4_dev *vc4 = to_vc4_dev(dev);
414 bool debug_dump_regs = false;
415 bool hsync_pos = mode->flags & DRM_MODE_FLAG_PHSYNC;
416 bool vsync_pos = mode->flags & DRM_MODE_FLAG_PVSYNC;
Eric Anholt682e62c2016-09-28 17:30:25 -0700417 bool interlaced = mode->flags & DRM_MODE_FLAG_INTERLACE;
Eric Anholtdfccd932016-09-29 15:34:44 -0700418 u32 pixel_rep = (mode->flags & DRM_MODE_FLAG_DBLCLK) ? 2 : 1;
Eric Anholt682e62c2016-09-28 17:30:25 -0700419 u32 verta = (VC4_SET_FIELD(mode->crtc_vsync_end - mode->crtc_vsync_start,
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800420 VC4_HDMI_VERTA_VSP) |
Eric Anholt682e62c2016-09-28 17:30:25 -0700421 VC4_SET_FIELD(mode->crtc_vsync_start - mode->crtc_vdisplay,
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800422 VC4_HDMI_VERTA_VFP) |
Eric Anholt682e62c2016-09-28 17:30:25 -0700423 VC4_SET_FIELD(mode->crtc_vdisplay, VC4_HDMI_VERTA_VAL));
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800424 u32 vertb = (VC4_SET_FIELD(0, VC4_HDMI_VERTB_VSPO) |
Eric Anholt682e62c2016-09-28 17:30:25 -0700425 VC4_SET_FIELD(mode->crtc_vtotal - mode->crtc_vsync_end,
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800426 VC4_HDMI_VERTB_VBP));
Eric Anholt682e62c2016-09-28 17:30:25 -0700427 u32 vertb_even = (VC4_SET_FIELD(0, VC4_HDMI_VERTB_VSPO) |
428 VC4_SET_FIELD(mode->crtc_vtotal -
429 mode->crtc_vsync_end -
430 interlaced,
431 VC4_HDMI_VERTB_VBP));
Eric Anholt6e1cbba2016-09-16 10:59:45 +0100432 u32 csc_ctl;
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800433
434 if (debug_dump_regs) {
435 DRM_INFO("HDMI regs before:\n");
436 vc4_hdmi_dump_regs(dev);
437 }
438
439 HD_WRITE(VC4_HD_VID_CTL, 0);
440
Eric Anholtdfccd932016-09-29 15:34:44 -0700441 clk_set_rate(vc4->hdmi->pixel_clock, mode->clock * 1000 *
442 ((mode->flags & DRM_MODE_FLAG_DBLCLK) ? 2 : 1));
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800443
444 HDMI_WRITE(VC4_HDMI_SCHEDULER_CONTROL,
445 HDMI_READ(VC4_HDMI_SCHEDULER_CONTROL) |
446 VC4_HDMI_SCHEDULER_CONTROL_MANUAL_FORMAT |
447 VC4_HDMI_SCHEDULER_CONTROL_IGNORE_VSYNC_PREDICTS);
448
449 HDMI_WRITE(VC4_HDMI_HORZA,
450 (vsync_pos ? VC4_HDMI_HORZA_VPOS : 0) |
451 (hsync_pos ? VC4_HDMI_HORZA_HPOS : 0) |
Eric Anholtdfccd932016-09-29 15:34:44 -0700452 VC4_SET_FIELD(mode->hdisplay * pixel_rep,
453 VC4_HDMI_HORZA_HAP));
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800454
455 HDMI_WRITE(VC4_HDMI_HORZB,
Eric Anholtdfccd932016-09-29 15:34:44 -0700456 VC4_SET_FIELD((mode->htotal -
457 mode->hsync_end) * pixel_rep,
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800458 VC4_HDMI_HORZB_HBP) |
Eric Anholtdfccd932016-09-29 15:34:44 -0700459 VC4_SET_FIELD((mode->hsync_end -
460 mode->hsync_start) * pixel_rep,
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800461 VC4_HDMI_HORZB_HSP) |
Eric Anholtdfccd932016-09-29 15:34:44 -0700462 VC4_SET_FIELD((mode->hsync_start -
463 mode->hdisplay) * pixel_rep,
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800464 VC4_HDMI_HORZB_HFP));
465
466 HDMI_WRITE(VC4_HDMI_VERTA0, verta);
467 HDMI_WRITE(VC4_HDMI_VERTA1, verta);
468
Eric Anholt682e62c2016-09-28 17:30:25 -0700469 HDMI_WRITE(VC4_HDMI_VERTB0, vertb_even);
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800470 HDMI_WRITE(VC4_HDMI_VERTB1, vertb);
471
472 HD_WRITE(VC4_HD_VID_CTL,
473 (vsync_pos ? 0 : VC4_HD_VID_CTL_VSYNC_LOW) |
474 (hsync_pos ? 0 : VC4_HD_VID_CTL_HSYNC_LOW));
475
Eric Anholt6e1cbba2016-09-16 10:59:45 +0100476 csc_ctl = VC4_SET_FIELD(VC4_HD_CSC_CTL_ORDER_BGR,
477 VC4_HD_CSC_CTL_ORDER);
478
Ville Syrjäläc8127cf02017-01-11 16:18:35 +0200479 if (vc4_encoder->hdmi_monitor &&
480 drm_default_rgb_quant_range(mode) ==
481 HDMI_QUANTIZATION_RANGE_LIMITED) {
Eric Anholt6e1cbba2016-09-16 10:59:45 +0100482 /* CEA VICs other than #1 requre limited range RGB
Eric Anholt21317b32016-09-29 15:34:43 -0700483 * output unless overridden by an AVI infoframe.
484 * Apply a colorspace conversion to squash 0-255 down
485 * to 16-235. The matrix here is:
Eric Anholt6e1cbba2016-09-16 10:59:45 +0100486 *
487 * [ 0 0 0.8594 16]
488 * [ 0 0.8594 0 16]
489 * [ 0.8594 0 0 16]
490 * [ 0 0 0 1]
491 */
492 csc_ctl |= VC4_HD_CSC_CTL_ENABLE;
493 csc_ctl |= VC4_HD_CSC_CTL_RGB2YCC;
494 csc_ctl |= VC4_SET_FIELD(VC4_HD_CSC_CTL_MODE_CUSTOM,
495 VC4_HD_CSC_CTL_MODE);
496
497 HD_WRITE(VC4_HD_CSC_12_11, (0x000 << 16) | 0x000);
498 HD_WRITE(VC4_HD_CSC_14_13, (0x100 << 16) | 0x6e0);
499 HD_WRITE(VC4_HD_CSC_22_21, (0x6e0 << 16) | 0x000);
500 HD_WRITE(VC4_HD_CSC_24_23, (0x100 << 16) | 0x000);
501 HD_WRITE(VC4_HD_CSC_32_31, (0x000 << 16) | 0x6e0);
502 HD_WRITE(VC4_HD_CSC_34_33, (0x100 << 16) | 0x000);
Eric Anholt21317b32016-09-29 15:34:43 -0700503 vc4_encoder->limited_rgb_range = true;
504 } else {
505 vc4_encoder->limited_rgb_range = false;
Eric Anholt6e1cbba2016-09-16 10:59:45 +0100506 }
507
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800508 /* The RGB order applies even when CSC is disabled. */
Eric Anholt6e1cbba2016-09-16 10:59:45 +0100509 HD_WRITE(VC4_HD_CSC_CTL, csc_ctl);
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800510
511 HDMI_WRITE(VC4_HDMI_FIFO_CTL, VC4_HDMI_FIFO_CTL_MASTER_SLAVE_N);
512
513 if (debug_dump_regs) {
514 DRM_INFO("HDMI regs after:\n");
515 vc4_hdmi_dump_regs(dev);
516 }
517}
518
519static void vc4_hdmi_encoder_disable(struct drm_encoder *encoder)
520{
521 struct drm_device *dev = encoder->dev;
522 struct vc4_dev *vc4 = to_vc4_dev(dev);
523
Eric Anholt21317b32016-09-29 15:34:43 -0700524 HDMI_WRITE(VC4_HDMI_RAM_PACKET_CONFIG, 0);
525
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800526 HDMI_WRITE(VC4_HDMI_TX_PHY_RESET_CTL, 0xf << 16);
527 HD_WRITE(VC4_HD_VID_CTL,
528 HD_READ(VC4_HD_VID_CTL) & ~VC4_HD_VID_CTL_ENABLE);
529}
530
531static void vc4_hdmi_encoder_enable(struct drm_encoder *encoder)
532{
533 struct vc4_hdmi_encoder *vc4_encoder = to_vc4_hdmi_encoder(encoder);
534 struct drm_device *dev = encoder->dev;
535 struct vc4_dev *vc4 = to_vc4_dev(dev);
536 int ret;
537
538 HDMI_WRITE(VC4_HDMI_TX_PHY_RESET_CTL, 0);
539
540 HD_WRITE(VC4_HD_VID_CTL,
541 HD_READ(VC4_HD_VID_CTL) |
542 VC4_HD_VID_CTL_ENABLE |
543 VC4_HD_VID_CTL_UNDERFLOW_ENABLE |
544 VC4_HD_VID_CTL_FRAME_COUNTER_RESET);
545
546 if (vc4_encoder->hdmi_monitor) {
547 HDMI_WRITE(VC4_HDMI_SCHEDULER_CONTROL,
548 HDMI_READ(VC4_HDMI_SCHEDULER_CONTROL) |
549 VC4_HDMI_SCHEDULER_CONTROL_MODE_HDMI);
550
551 ret = wait_for(HDMI_READ(VC4_HDMI_SCHEDULER_CONTROL) &
Eric Anholt2b29bf12016-09-28 17:21:05 -0700552 VC4_HDMI_SCHEDULER_CONTROL_HDMI_ACTIVE, 1000);
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800553 WARN_ONCE(ret, "Timeout waiting for "
554 "VC4_HDMI_SCHEDULER_CONTROL_HDMI_ACTIVE\n");
555 } else {
556 HDMI_WRITE(VC4_HDMI_RAM_PACKET_CONFIG,
557 HDMI_READ(VC4_HDMI_RAM_PACKET_CONFIG) &
558 ~(VC4_HDMI_RAM_PACKET_ENABLE));
559 HDMI_WRITE(VC4_HDMI_SCHEDULER_CONTROL,
560 HDMI_READ(VC4_HDMI_SCHEDULER_CONTROL) &
561 ~VC4_HDMI_SCHEDULER_CONTROL_MODE_HDMI);
562
563 ret = wait_for(!(HDMI_READ(VC4_HDMI_SCHEDULER_CONTROL) &
Eric Anholt2b29bf12016-09-28 17:21:05 -0700564 VC4_HDMI_SCHEDULER_CONTROL_HDMI_ACTIVE), 1000);
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800565 WARN_ONCE(ret, "Timeout waiting for "
566 "!VC4_HDMI_SCHEDULER_CONTROL_HDMI_ACTIVE\n");
567 }
568
569 if (vc4_encoder->hdmi_monitor) {
570 u32 drift;
571
572 WARN_ON(!(HDMI_READ(VC4_HDMI_SCHEDULER_CONTROL) &
573 VC4_HDMI_SCHEDULER_CONTROL_HDMI_ACTIVE));
574 HDMI_WRITE(VC4_HDMI_SCHEDULER_CONTROL,
575 HDMI_READ(VC4_HDMI_SCHEDULER_CONTROL) |
576 VC4_HDMI_SCHEDULER_CONTROL_VERT_ALWAYS_KEEPOUT);
577
Eric Anholt21317b32016-09-29 15:34:43 -0700578 HDMI_WRITE(VC4_HDMI_RAM_PACKET_CONFIG,
579 VC4_HDMI_RAM_PACKET_ENABLE);
580
581 vc4_hdmi_set_infoframes(encoder);
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800582
583 drift = HDMI_READ(VC4_HDMI_FIFO_CTL);
584 drift &= VC4_HDMI_FIFO_VALID_WRITE_MASK;
585
586 HDMI_WRITE(VC4_HDMI_FIFO_CTL,
587 drift & ~VC4_HDMI_FIFO_CTL_RECENTER);
588 HDMI_WRITE(VC4_HDMI_FIFO_CTL,
589 drift | VC4_HDMI_FIFO_CTL_RECENTER);
590 udelay(1000);
591 HDMI_WRITE(VC4_HDMI_FIFO_CTL,
592 drift & ~VC4_HDMI_FIFO_CTL_RECENTER);
593 HDMI_WRITE(VC4_HDMI_FIFO_CTL,
594 drift | VC4_HDMI_FIFO_CTL_RECENTER);
595
596 ret = wait_for(HDMI_READ(VC4_HDMI_FIFO_CTL) &
597 VC4_HDMI_FIFO_CTL_RECENTER_DONE, 1);
598 WARN_ONCE(ret, "Timeout waiting for "
599 "VC4_HDMI_FIFO_CTL_RECENTER_DONE");
600 }
601}
602
603static const struct drm_encoder_helper_funcs vc4_hdmi_encoder_helper_funcs = {
604 .mode_set = vc4_hdmi_encoder_mode_set,
605 .disable = vc4_hdmi_encoder_disable,
606 .enable = vc4_hdmi_encoder_enable,
607};
608
609static int vc4_hdmi_bind(struct device *dev, struct device *master, void *data)
610{
611 struct platform_device *pdev = to_platform_device(dev);
612 struct drm_device *drm = dev_get_drvdata(master);
613 struct vc4_dev *vc4 = drm->dev_private;
614 struct vc4_hdmi *hdmi;
615 struct vc4_hdmi_encoder *vc4_hdmi_encoder;
616 struct device_node *ddc_node;
617 u32 value;
618 int ret;
619
620 hdmi = devm_kzalloc(dev, sizeof(*hdmi), GFP_KERNEL);
621 if (!hdmi)
622 return -ENOMEM;
623
624 vc4_hdmi_encoder = devm_kzalloc(dev, sizeof(*vc4_hdmi_encoder),
625 GFP_KERNEL);
626 if (!vc4_hdmi_encoder)
627 return -ENOMEM;
628 vc4_hdmi_encoder->base.type = VC4_ENCODER_TYPE_HDMI;
629 hdmi->encoder = &vc4_hdmi_encoder->base.base;
630
631 hdmi->pdev = pdev;
632 hdmi->hdmicore_regs = vc4_ioremap_regs(pdev, 0);
633 if (IS_ERR(hdmi->hdmicore_regs))
634 return PTR_ERR(hdmi->hdmicore_regs);
635
636 hdmi->hd_regs = vc4_ioremap_regs(pdev, 1);
637 if (IS_ERR(hdmi->hd_regs))
638 return PTR_ERR(hdmi->hd_regs);
639
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800640 hdmi->pixel_clock = devm_clk_get(dev, "pixel");
641 if (IS_ERR(hdmi->pixel_clock)) {
642 DRM_ERROR("Failed to get pixel clock\n");
643 return PTR_ERR(hdmi->pixel_clock);
644 }
645 hdmi->hsm_clock = devm_clk_get(dev, "hdmi");
646 if (IS_ERR(hdmi->hsm_clock)) {
647 DRM_ERROR("Failed to get HDMI state machine clock\n");
648 return PTR_ERR(hdmi->hsm_clock);
649 }
650
Peter Chen027a6972016-07-05 10:04:54 +0800651 ddc_node = of_parse_phandle(dev->of_node, "ddc", 0);
652 if (!ddc_node) {
653 DRM_ERROR("Failed to find ddc node in device tree\n");
654 return -ENODEV;
655 }
656
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800657 hdmi->ddc = of_find_i2c_adapter_by_node(ddc_node);
Peter Chen027a6972016-07-05 10:04:54 +0800658 of_node_put(ddc_node);
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800659 if (!hdmi->ddc) {
660 DRM_DEBUG("Failed to get ddc i2c adapter by node\n");
661 return -EPROBE_DEFER;
662 }
663
664 /* Enable the clocks at startup. We can't quite recover from
665 * turning off the pixel clock during disable/enables yet, so
666 * it's always running.
667 */
668 ret = clk_prepare_enable(hdmi->pixel_clock);
669 if (ret) {
670 DRM_ERROR("Failed to turn on pixel clock: %d\n", ret);
671 goto err_put_i2c;
672 }
673
Eric Anholt851479a2016-02-12 14:15:14 -0800674 /* This is the rate that is set by the firmware. The number
675 * needs to be a bit higher than the pixel clock rate
676 * (generally 148.5Mhz).
677 */
678 ret = clk_set_rate(hdmi->hsm_clock, 163682864);
679 if (ret) {
680 DRM_ERROR("Failed to set HSM clock rate: %d\n", ret);
681 goto err_unprepare_pix;
682 }
683
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800684 ret = clk_prepare_enable(hdmi->hsm_clock);
685 if (ret) {
686 DRM_ERROR("Failed to turn on HDMI state machine clock: %d\n",
687 ret);
688 goto err_unprepare_pix;
689 }
690
691 /* Only use the GPIO HPD pin if present in the DT, otherwise
692 * we'll use the HDMI core's register.
693 */
694 if (of_find_property(dev->of_node, "hpd-gpios", &value)) {
Eric Anholt0b06e0a2016-02-29 17:53:01 -0800695 enum of_gpio_flags hpd_gpio_flags;
696
697 hdmi->hpd_gpio = of_get_named_gpio_flags(dev->of_node,
698 "hpd-gpios", 0,
699 &hpd_gpio_flags);
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800700 if (hdmi->hpd_gpio < 0) {
701 ret = hdmi->hpd_gpio;
702 goto err_unprepare_hsm;
703 }
Eric Anholt0b06e0a2016-02-29 17:53:01 -0800704
705 hdmi->hpd_active_low = hpd_gpio_flags & OF_GPIO_ACTIVE_LOW;
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800706 }
707
708 vc4->hdmi = hdmi;
709
710 /* HDMI core must be enabled. */
Eric Anholt851479a2016-02-12 14:15:14 -0800711 if (!(HD_READ(VC4_HD_M_CTL) & VC4_HD_M_ENABLE)) {
712 HD_WRITE(VC4_HD_M_CTL, VC4_HD_M_SW_RST);
713 udelay(1);
714 HD_WRITE(VC4_HD_M_CTL, 0);
715
716 HD_WRITE(VC4_HD_M_CTL, VC4_HD_M_ENABLE);
717
718 HDMI_WRITE(VC4_HDMI_SW_RESET_CONTROL,
719 VC4_HDMI_SW_RESET_HDMI |
720 VC4_HDMI_SW_RESET_FORMAT_DETECT);
721
722 HDMI_WRITE(VC4_HDMI_SW_RESET_CONTROL, 0);
723
724 /* PHY should be in reset, like
725 * vc4_hdmi_encoder_disable() does.
726 */
727 HDMI_WRITE(VC4_HDMI_TX_PHY_RESET_CTL, 0xf << 16);
728 }
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800729
730 drm_encoder_init(drm, hdmi->encoder, &vc4_hdmi_encoder_funcs,
Ville Syrjälä13a3d912015-12-09 16:20:18 +0200731 DRM_MODE_ENCODER_TMDS, NULL);
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800732 drm_encoder_helper_add(hdmi->encoder, &vc4_hdmi_encoder_helper_funcs);
733
734 hdmi->connector = vc4_hdmi_connector_init(drm, hdmi->encoder);
735 if (IS_ERR(hdmi->connector)) {
736 ret = PTR_ERR(hdmi->connector);
737 goto err_destroy_encoder;
738 }
739
740 return 0;
741
742err_destroy_encoder:
743 vc4_hdmi_encoder_destroy(hdmi->encoder);
744err_unprepare_hsm:
745 clk_disable_unprepare(hdmi->hsm_clock);
746err_unprepare_pix:
747 clk_disable_unprepare(hdmi->pixel_clock);
748err_put_i2c:
Eric Anholt58839802016-04-04 14:25:59 -0700749 put_device(&hdmi->ddc->dev);
Eric Anholtc8b75bc2015-03-02 13:01:12 -0800750
751 return ret;
752}
753
754static void vc4_hdmi_unbind(struct device *dev, struct device *master,
755 void *data)
756{
757 struct drm_device *drm = dev_get_drvdata(master);
758 struct vc4_dev *vc4 = drm->dev_private;
759 struct vc4_hdmi *hdmi = vc4->hdmi;
760
761 vc4_hdmi_connector_destroy(hdmi->connector);
762 vc4_hdmi_encoder_destroy(hdmi->encoder);
763
764 clk_disable_unprepare(hdmi->pixel_clock);
765 clk_disable_unprepare(hdmi->hsm_clock);
766 put_device(&hdmi->ddc->dev);
767
768 vc4->hdmi = NULL;
769}
770
771static const struct component_ops vc4_hdmi_ops = {
772 .bind = vc4_hdmi_bind,
773 .unbind = vc4_hdmi_unbind,
774};
775
776static int vc4_hdmi_dev_probe(struct platform_device *pdev)
777{
778 return component_add(&pdev->dev, &vc4_hdmi_ops);
779}
780
781static int vc4_hdmi_dev_remove(struct platform_device *pdev)
782{
783 component_del(&pdev->dev, &vc4_hdmi_ops);
784 return 0;
785}
786
787static const struct of_device_id vc4_hdmi_dt_match[] = {
788 { .compatible = "brcm,bcm2835-hdmi" },
789 {}
790};
791
792struct platform_driver vc4_hdmi_driver = {
793 .probe = vc4_hdmi_dev_probe,
794 .remove = vc4_hdmi_dev_remove,
795 .driver = {
796 .name = "vc4_hdmi",
797 .of_match_table = vc4_hdmi_dt_match,
798 },
799};