blob: c1c65624fd5d37da4d6b1588faa71210e768564d [file] [log] [blame]
Archit Taneja5cac5ae2013-10-08 13:07:00 +05301/*
2 * HDMI PHY
3 *
4 * Copyright (C) 2013 Texas Instruments Incorporated
5 *
6 * This program is free software; you can redistribute it and/or modify it
7 * under the terms of the GNU General Public License version 2 as published by
8 * the Free Software Foundation.
9 */
10
11#include <linux/kernel.h>
Archit Taneja5cac5ae2013-10-08 13:07:00 +053012#include <linux/err.h>
13#include <linux/io.h>
14#include <linux/platform_device.h>
15#include <video/omapdss.h>
16
17#include "dss.h"
Archit Tanejaef269582013-09-12 17:45:57 +053018#include "hdmi.h"
Archit Taneja5cac5ae2013-10-08 13:07:00 +053019
Archit Taneja5cac5ae2013-10-08 13:07:00 +053020void hdmi_phy_dump(struct hdmi_phy_data *phy, struct seq_file *s)
21{
22#define DUMPPHY(r) seq_printf(s, "%-35s %08x\n", #r,\
23 hdmi_read_reg(phy->base, r))
24
25 DUMPPHY(HDMI_TXPHY_TX_CTRL);
26 DUMPPHY(HDMI_TXPHY_DIGITAL_CTRL);
27 DUMPPHY(HDMI_TXPHY_POWER_CTRL);
28 DUMPPHY(HDMI_TXPHY_PAD_CFG_CTRL);
29}
30
31static irqreturn_t hdmi_irq_handler(int irq, void *data)
32{
33 struct hdmi_wp_data *wp = data;
34 u32 irqstatus;
35
36 irqstatus = hdmi_wp_get_irqstatus(wp);
37 hdmi_wp_set_irqstatus(wp, irqstatus);
38
39 if ((irqstatus & HDMI_IRQ_LINK_CONNECT) &&
40 irqstatus & HDMI_IRQ_LINK_DISCONNECT) {
41 /*
42 * If we get both connect and disconnect interrupts at the same
43 * time, turn off the PHY, clear interrupts, and restart, which
44 * raises connect interrupt if a cable is connected, or nothing
45 * if cable is not connected.
46 */
47 hdmi_wp_set_phy_pwr(wp, HDMI_PHYPWRCMD_OFF);
48
49 hdmi_wp_set_irqstatus(wp, HDMI_IRQ_LINK_CONNECT |
50 HDMI_IRQ_LINK_DISCONNECT);
51
52 hdmi_wp_set_phy_pwr(wp, HDMI_PHYPWRCMD_LDOON);
53 } else if (irqstatus & HDMI_IRQ_LINK_CONNECT) {
54 hdmi_wp_set_phy_pwr(wp, HDMI_PHYPWRCMD_TXON);
55 } else if (irqstatus & HDMI_IRQ_LINK_DISCONNECT) {
56 hdmi_wp_set_phy_pwr(wp, HDMI_PHYPWRCMD_LDOON);
57 }
58
59 return IRQ_HANDLED;
60}
61
Tomi Valkeinen2f5dc672014-04-17 12:54:02 +030062int hdmi_phy_parse_lanes(struct hdmi_phy_data *phy, const u32 *lanes)
63{
64 int i;
65
66 for (i = 0; i < 8; i += 2) {
67 u8 lane, pol;
68 int dx, dy;
69
70 dx = lanes[i];
71 dy = lanes[i + 1];
72
73 if (dx < 0 || dx >= 8)
74 return -EINVAL;
75
76 if (dy < 0 || dy >= 8)
77 return -EINVAL;
78
79 if (dx & 1) {
80 if (dy != dx - 1)
81 return -EINVAL;
82 pol = 1;
83 } else {
84 if (dy != dx + 1)
85 return -EINVAL;
86 pol = 0;
87 }
88
89 lane = dx / 2;
90
91 phy->lane_function[lane] = i / 2;
92 phy->lane_polarity[lane] = pol;
93 }
94
95 return 0;
96}
97
98static void hdmi_phy_configure_lanes(struct hdmi_phy_data *phy)
99{
100 static const u16 pad_cfg_list[] = {
101 0x0123,
102 0x0132,
103 0x0312,
104 0x0321,
105 0x0231,
106 0x0213,
107 0x1023,
108 0x1032,
109 0x3012,
110 0x3021,
111 0x2031,
112 0x2013,
113 0x1203,
114 0x1302,
115 0x3102,
116 0x3201,
117 0x2301,
118 0x2103,
119 0x1230,
120 0x1320,
121 0x3120,
122 0x3210,
123 0x2310,
124 0x2130,
125 };
126
127 u16 lane_cfg = 0;
128 int i;
129 unsigned lane_cfg_val;
130 u16 pol_val = 0;
131
132 for (i = 0; i < 4; ++i)
133 lane_cfg |= phy->lane_function[i] << ((3 - i) * 4);
134
135 pol_val |= phy->lane_polarity[0] << 0;
136 pol_val |= phy->lane_polarity[1] << 3;
137 pol_val |= phy->lane_polarity[2] << 2;
138 pol_val |= phy->lane_polarity[3] << 1;
139
140 for (i = 0; i < ARRAY_SIZE(pad_cfg_list); ++i)
141 if (pad_cfg_list[i] == lane_cfg)
142 break;
143
144 if (WARN_ON(i == ARRAY_SIZE(pad_cfg_list)))
145 i = 0;
146
147 lane_cfg_val = i;
148
149 REG_FLD_MOD(phy->base, HDMI_TXPHY_PAD_CFG_CTRL, lane_cfg_val, 26, 22);
150 REG_FLD_MOD(phy->base, HDMI_TXPHY_PAD_CFG_CTRL, pol_val, 30, 27);
151}
152
Archit Taneja5cac5ae2013-10-08 13:07:00 +0530153int hdmi_phy_enable(struct hdmi_phy_data *phy, struct hdmi_wp_data *wp,
154 struct hdmi_config *cfg)
155{
156 u16 r = 0;
157 u32 irqstatus;
158
159 hdmi_wp_clear_irqenable(wp, 0xffffffff);
160
161 irqstatus = hdmi_wp_get_irqstatus(wp);
162 hdmi_wp_set_irqstatus(wp, irqstatus);
163
164 r = hdmi_wp_set_phy_pwr(wp, HDMI_PHYPWRCMD_LDOON);
165 if (r)
166 return r;
167
168 /*
169 * Read address 0 in order to get the SCP reset done completed
170 * Dummy access performed to make sure reset is done
171 */
172 hdmi_read_reg(phy->base, HDMI_TXPHY_TX_CTRL);
173
174 /*
175 * Write to phy address 0 to configure the clock
176 * use HFBITCLK write HDMI_TXPHY_TX_CONTROL_FREQOUT field
177 */
178 REG_FLD_MOD(phy->base, HDMI_TXPHY_TX_CTRL, 0x1, 31, 30);
179
180 /* Write to phy address 1 to start HDMI line (TXVALID and TMDSCLKEN) */
181 hdmi_write_reg(phy->base, HDMI_TXPHY_DIGITAL_CTRL, 0xF0000000);
182
183 /* Setup max LDO voltage */
184 REG_FLD_MOD(phy->base, HDMI_TXPHY_POWER_CTRL, 0xB, 3, 0);
185
Tomi Valkeinen2f5dc672014-04-17 12:54:02 +0300186 hdmi_phy_configure_lanes(phy);
Archit Taneja5cac5ae2013-10-08 13:07:00 +0530187
188 r = request_threaded_irq(phy->irq, NULL, hdmi_irq_handler,
189 IRQF_ONESHOT, "OMAP HDMI", wp);
190 if (r) {
191 DSSERR("HDMI IRQ request failed\n");
192 hdmi_wp_set_phy_pwr(wp, HDMI_PHYPWRCMD_OFF);
193 return r;
194 }
195
196 hdmi_wp_set_irqenable(wp,
197 HDMI_IRQ_LINK_CONNECT | HDMI_IRQ_LINK_DISCONNECT);
198
199 return 0;
200}
201
202void hdmi_phy_disable(struct hdmi_phy_data *phy, struct hdmi_wp_data *wp)
203{
204 free_irq(phy->irq, wp);
205
206 hdmi_wp_set_phy_pwr(wp, HDMI_PHYPWRCMD_OFF);
207}
208
209#define PHY_OFFSET 0x300
210#define PHY_SIZE 0x100
211
212int hdmi_phy_init(struct platform_device *pdev, struct hdmi_phy_data *phy)
213{
214 struct resource *res;
215 struct resource temp_res;
216
Tomi Valkeinen77601502013-12-17 14:41:14 +0200217 res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "phy");
Archit Taneja5cac5ae2013-10-08 13:07:00 +0530218 if (!res) {
219 DSSDBG("can't get PHY mem resource by name\n");
220 /*
221 * if hwmod/DT doesn't have the memory resource information
222 * split into HDMI sub blocks by name, we try again by getting
223 * the platform's first resource. this code will be removed when
224 * the driver can get the mem resources by name
225 */
226 res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
227 if (!res) {
228 DSSERR("can't get PHY mem resource\n");
229 return -EINVAL;
230 }
231
232 temp_res.start = res->start + PHY_OFFSET;
233 temp_res.end = temp_res.start + PHY_SIZE - 1;
234 res = &temp_res;
235 }
236
237 phy->base = devm_ioremap(&pdev->dev, res->start, resource_size(res));
238 if (!phy->base) {
239 DSSERR("can't ioremap TX PHY\n");
240 return -ENOMEM;
241 }
242
243 phy->irq = platform_get_irq(pdev, 0);
244 if (phy->irq < 0) {
245 DSSERR("platform_get_irq failed\n");
246 return -ENODEV;
247 }
248
249 return 0;
250}