Skip to content

Commit 9c56810

Browse files
committed
drm/sun4i: Add HDMI support
The earlier Allwinner SoCs (A10, A10s, A20, A31) have an embedded HDMI controller. That HDMI controller is able to do audio and CEC, but those have been left out for now. Reviewed-by: Chen-Yu Tsai <[email protected]> Signed-off-by: Maxime Ripard <[email protected]>
1 parent 22662f1 commit 9c56810

File tree

6 files changed

+1023
-0
lines changed

6 files changed

+1023
-0
lines changed

drivers/gpu/drm/sun4i/Kconfig

Lines changed: 8 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,14 @@ config DRM_SUN4I
1313
Display Engine. If M is selected the module will be called
1414
sun4i-drm.
1515

16+
config DRM_SUN4I_HDMI
17+
tristate "Allwinner A10 HDMI Controller Support"
18+
depends on DRM_SUN4I
19+
default DRM_SUN4I
20+
help
21+
Choose this option if you have an Allwinner SoC with an HDMI
22+
controller.
23+
1624
config DRM_SUN4I_BACKEND
1725
tristate "Support for Allwinner A10 Display Engine Backend"
1826
depends on DRM_SUN4I

drivers/gpu/drm/sun4i/Makefile

Lines changed: 5 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,10 @@
11
sun4i-drm-y += sun4i_drv.o
22
sun4i-drm-y += sun4i_framebuffer.o
33

4+
sun4i-drm-hdmi-y += sun4i_hdmi_enc.o
5+
sun4i-drm-hdmi-y += sun4i_hdmi_ddc_clk.o
6+
sun4i-drm-hdmi-y += sun4i_hdmi_tmds_clk.o
7+
48
sun4i-tcon-y += sun4i_tcon.o
59
sun4i-tcon-y += sun4i_rgb.o
610
sun4i-tcon-y += sun4i_dotclock.o
@@ -15,4 +19,5 @@ obj-$(CONFIG_DRM_SUN4I) += sun6i_drc.o
1519
obj-$(CONFIG_DRM_SUN4I) += sun4i_tv.o
1620

1721
obj-$(CONFIG_DRM_SUN4I_BACKEND) += sun4i-backend.o
22+
obj-$(CONFIG_DRM_SUN4I_HDMI) += sun4i-drm-hdmi.o
1823
obj-$(CONFIG_DRM_SUN8I_MIXER) += sun8i-mixer.o

drivers/gpu/drm/sun4i/sun4i_hdmi.h

Lines changed: 157 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,157 @@
1+
/*
2+
* Copyright (C) 2016 Maxime Ripard
3+
*
4+
* Maxime Ripard <[email protected]>
5+
*
6+
* This program is free software; you can redistribute it and/or
7+
* modify it under the terms of the GNU General Public License as
8+
* published by the Free Software Foundation; either version 2 of
9+
* the License, or (at your option) any later version.
10+
*/
11+
12+
#ifndef _SUN4I_HDMI_H_
13+
#define _SUN4I_HDMI_H_
14+
15+
#include <drm/drm_connector.h>
16+
#include <drm/drm_encoder.h>
17+
18+
#define SUN4I_HDMI_CTRL_REG 0x004
19+
#define SUN4I_HDMI_CTRL_ENABLE BIT(31)
20+
21+
#define SUN4I_HDMI_IRQ_REG 0x008
22+
#define SUN4I_HDMI_IRQ_STA_MASK 0x73
23+
#define SUN4I_HDMI_IRQ_STA_FIFO_OF BIT(1)
24+
#define SUN4I_HDMI_IRQ_STA_FIFO_UF BIT(0)
25+
26+
#define SUN4I_HDMI_HPD_REG 0x00c
27+
#define SUN4I_HDMI_HPD_HIGH BIT(0)
28+
29+
#define SUN4I_HDMI_VID_CTRL_REG 0x010
30+
#define SUN4I_HDMI_VID_CTRL_ENABLE BIT(31)
31+
#define SUN4I_HDMI_VID_CTRL_HDMI_MODE BIT(30)
32+
33+
#define SUN4I_HDMI_VID_TIMING_ACT_REG 0x014
34+
#define SUN4I_HDMI_VID_TIMING_BP_REG 0x018
35+
#define SUN4I_HDMI_VID_TIMING_FP_REG 0x01c
36+
#define SUN4I_HDMI_VID_TIMING_SPW_REG 0x020
37+
38+
#define SUN4I_HDMI_VID_TIMING_X(x) ((((x) - 1) & GENMASK(11, 0)))
39+
#define SUN4I_HDMI_VID_TIMING_Y(y) ((((y) - 1) & GENMASK(11, 0)) << 16)
40+
41+
#define SUN4I_HDMI_VID_TIMING_POL_REG 0x024
42+
#define SUN4I_HDMI_VID_TIMING_POL_TX_CLK (0x3e0 << 16)
43+
#define SUN4I_HDMI_VID_TIMING_POL_VSYNC BIT(1)
44+
#define SUN4I_HDMI_VID_TIMING_POL_HSYNC BIT(0)
45+
46+
#define SUN4I_HDMI_AVI_INFOFRAME_REG(n) (0x080 + (n))
47+
48+
#define SUN4I_HDMI_PAD_CTRL0_REG 0x200
49+
#define SUN4I_HDMI_PAD_CTRL0_BIASEN BIT(31)
50+
#define SUN4I_HDMI_PAD_CTRL0_LDOCEN BIT(30)
51+
#define SUN4I_HDMI_PAD_CTRL0_LDODEN BIT(29)
52+
#define SUN4I_HDMI_PAD_CTRL0_PWENC BIT(28)
53+
#define SUN4I_HDMI_PAD_CTRL0_PWEND BIT(27)
54+
#define SUN4I_HDMI_PAD_CTRL0_PWENG BIT(26)
55+
#define SUN4I_HDMI_PAD_CTRL0_CKEN BIT(25)
56+
#define SUN4I_HDMI_PAD_CTRL0_TXEN BIT(23)
57+
58+
#define SUN4I_HDMI_PAD_CTRL1_REG 0x204
59+
#define SUN4I_HDMI_PAD_CTRL1_AMP_OPT BIT(23)
60+
#define SUN4I_HDMI_PAD_CTRL1_AMPCK_OPT BIT(22)
61+
#define SUN4I_HDMI_PAD_CTRL1_EMP_OPT BIT(20)
62+
#define SUN4I_HDMI_PAD_CTRL1_EMPCK_OPT BIT(19)
63+
#define SUN4I_HDMI_PAD_CTRL1_REG_DEN BIT(15)
64+
#define SUN4I_HDMI_PAD_CTRL1_REG_DENCK BIT(14)
65+
#define SUN4I_HDMI_PAD_CTRL1_REG_EMP(n) (((n) & 7) << 10)
66+
#define SUN4I_HDMI_PAD_CTRL1_HALVE_CLK BIT(6)
67+
#define SUN4I_HDMI_PAD_CTRL1_REG_AMP(n) (((n) & 7) << 3)
68+
69+
#define SUN4I_HDMI_PLL_CTRL_REG 0x208
70+
#define SUN4I_HDMI_PLL_CTRL_PLL_EN BIT(31)
71+
#define SUN4I_HDMI_PLL_CTRL_BWS BIT(30)
72+
#define SUN4I_HDMI_PLL_CTRL_HV_IS_33 BIT(29)
73+
#define SUN4I_HDMI_PLL_CTRL_LDO1_EN BIT(28)
74+
#define SUN4I_HDMI_PLL_CTRL_LDO2_EN BIT(27)
75+
#define SUN4I_HDMI_PLL_CTRL_SDIV2 BIT(25)
76+
#define SUN4I_HDMI_PLL_CTRL_VCO_GAIN(n) (((n) & 7) << 20)
77+
#define SUN4I_HDMI_PLL_CTRL_S(n) (((n) & 7) << 17)
78+
#define SUN4I_HDMI_PLL_CTRL_CP_S(n) (((n) & 0x1f) << 12)
79+
#define SUN4I_HDMI_PLL_CTRL_CS(n) (((n) & 0xf) << 8)
80+
#define SUN4I_HDMI_PLL_CTRL_DIV(n) (((n) & 0xf) << 4)
81+
#define SUN4I_HDMI_PLL_CTRL_DIV_MASK GENMASK(7, 4)
82+
#define SUN4I_HDMI_PLL_CTRL_VCO_S(n) ((n) & 0xf)
83+
84+
#define SUN4I_HDMI_PLL_DBG0_REG 0x20c
85+
#define SUN4I_HDMI_PLL_DBG0_TMDS_PARENT(n) (((n) & 1) << 21)
86+
#define SUN4I_HDMI_PLL_DBG0_TMDS_PARENT_MASK BIT(21)
87+
#define SUN4I_HDMI_PLL_DBG0_TMDS_PARENT_SHIFT 21
88+
89+
#define SUN4I_HDMI_PKT_CTRL_REG(n) (0x2f0 + (4 * (n)))
90+
#define SUN4I_HDMI_PKT_CTRL_TYPE(n, t) ((t) << (((n) % 4) * 4))
91+
92+
#define SUN4I_HDMI_UNKNOWN_REG 0x300
93+
#define SUN4I_HDMI_UNKNOWN_INPUT_SYNC BIT(27)
94+
95+
#define SUN4I_HDMI_DDC_CTRL_REG 0x500
96+
#define SUN4I_HDMI_DDC_CTRL_ENABLE BIT(31)
97+
#define SUN4I_HDMI_DDC_CTRL_START_CMD BIT(30)
98+
#define SUN4I_HDMI_DDC_CTRL_FIFO_DIR_MASK BIT(8)
99+
#define SUN4I_HDMI_DDC_CTRL_FIFO_DIR_READ (0 << 8)
100+
#define SUN4I_HDMI_DDC_CTRL_RESET BIT(0)
101+
102+
#define SUN4I_HDMI_DDC_ADDR_REG 0x504
103+
#define SUN4I_HDMI_DDC_ADDR_SEGMENT(seg) (((seg) & 0xff) << 24)
104+
#define SUN4I_HDMI_DDC_ADDR_EDDC(addr) (((addr) & 0xff) << 16)
105+
#define SUN4I_HDMI_DDC_ADDR_OFFSET(off) (((off) & 0xff) << 8)
106+
#define SUN4I_HDMI_DDC_ADDR_SLAVE(addr) ((addr) & 0xff)
107+
108+
#define SUN4I_HDMI_DDC_FIFO_CTRL_REG 0x510
109+
#define SUN4I_HDMI_DDC_FIFO_CTRL_CLEAR BIT(31)
110+
111+
#define SUN4I_HDMI_DDC_FIFO_DATA_REG 0x518
112+
#define SUN4I_HDMI_DDC_BYTE_COUNT_REG 0x51c
113+
114+
#define SUN4I_HDMI_DDC_CMD_REG 0x520
115+
#define SUN4I_HDMI_DDC_CMD_EXPLICIT_EDDC_READ 6
116+
117+
#define SUN4I_HDMI_DDC_CLK_REG 0x528
118+
#define SUN4I_HDMI_DDC_CLK_M(m) (((m) & 0x7) << 3)
119+
#define SUN4I_HDMI_DDC_CLK_N(n) ((n) & 0x7)
120+
121+
#define SUN4I_HDMI_DDC_LINE_CTRL_REG 0x540
122+
#define SUN4I_HDMI_DDC_LINE_CTRL_SDA_ENABLE BIT(9)
123+
#define SUN4I_HDMI_DDC_LINE_CTRL_SCL_ENABLE BIT(8)
124+
125+
#define SUN4I_HDMI_DDC_FIFO_SIZE 16
126+
127+
enum sun4i_hdmi_pkt_type {
128+
SUN4I_HDMI_PKT_AVI = 2,
129+
SUN4I_HDMI_PKT_END = 15,
130+
};
131+
132+
struct sun4i_hdmi {
133+
struct drm_connector connector;
134+
struct drm_encoder encoder;
135+
struct device *dev;
136+
137+
void __iomem *base;
138+
139+
/* Parent clocks */
140+
struct clk *bus_clk;
141+
struct clk *mod_clk;
142+
struct clk *pll0_clk;
143+
struct clk *pll1_clk;
144+
145+
/* And the clocks we create */
146+
struct clk *ddc_clk;
147+
struct clk *tmds_clk;
148+
149+
struct sun4i_drv *drv;
150+
151+
bool hdmi_monitor;
152+
};
153+
154+
int sun4i_ddc_create(struct sun4i_hdmi *hdmi, struct clk *clk);
155+
int sun4i_tmds_create(struct sun4i_hdmi *hdmi);
156+
157+
#endif /* _SUN4I_HDMI_H_ */
Lines changed: 127 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,127 @@
1+
/*
2+
* Copyright (C) 2016 Free Electrons
3+
* Copyright (C) 2016 NextThing Co
4+
*
5+
* Maxime Ripard <[email protected]>
6+
*
7+
* This program is free software; you can redistribute it and/or
8+
* modify it under the terms of the GNU General Public License as
9+
* published by the Free Software Foundation; either version 2 of
10+
* the License, or (at your option) any later version.
11+
*/
12+
13+
#include <linux/clk-provider.h>
14+
15+
#include "sun4i_tcon.h"
16+
#include "sun4i_hdmi.h"
17+
18+
struct sun4i_ddc {
19+
struct clk_hw hw;
20+
struct sun4i_hdmi *hdmi;
21+
};
22+
23+
static inline struct sun4i_ddc *hw_to_ddc(struct clk_hw *hw)
24+
{
25+
return container_of(hw, struct sun4i_ddc, hw);
26+
}
27+
28+
static unsigned long sun4i_ddc_calc_divider(unsigned long rate,
29+
unsigned long parent_rate,
30+
u8 *m, u8 *n)
31+
{
32+
unsigned long best_rate = 0;
33+
u8 best_m = 0, best_n = 0, _m, _n;
34+
35+
for (_m = 0; _m < 8; _m++) {
36+
for (_n = 0; _n < 8; _n++) {
37+
unsigned long tmp_rate;
38+
39+
tmp_rate = (((parent_rate / 2) / 10) >> _n) / (_m + 1);
40+
41+
if (tmp_rate > rate)
42+
continue;
43+
44+
if (abs(rate - tmp_rate) < abs(rate - best_rate)) {
45+
best_rate = tmp_rate;
46+
best_m = _m;
47+
best_n = _n;
48+
}
49+
}
50+
}
51+
52+
if (m && n) {
53+
*m = best_m;
54+
*n = best_n;
55+
}
56+
57+
return best_rate;
58+
}
59+
60+
static long sun4i_ddc_round_rate(struct clk_hw *hw, unsigned long rate,
61+
unsigned long *prate)
62+
{
63+
return sun4i_ddc_calc_divider(rate, *prate, NULL, NULL);
64+
}
65+
66+
static unsigned long sun4i_ddc_recalc_rate(struct clk_hw *hw,
67+
unsigned long parent_rate)
68+
{
69+
struct sun4i_ddc *ddc = hw_to_ddc(hw);
70+
u32 reg;
71+
u8 m, n;
72+
73+
reg = readl(ddc->hdmi->base + SUN4I_HDMI_DDC_CLK_REG);
74+
m = (reg >> 3) & 0x7;
75+
n = reg & 0x7;
76+
77+
return (((parent_rate / 2) / 10) >> n) / (m + 1);
78+
}
79+
80+
static int sun4i_ddc_set_rate(struct clk_hw *hw, unsigned long rate,
81+
unsigned long parent_rate)
82+
{
83+
struct sun4i_ddc *ddc = hw_to_ddc(hw);
84+
u8 div_m, div_n;
85+
86+
sun4i_ddc_calc_divider(rate, parent_rate, &div_m, &div_n);
87+
88+
writel(SUN4I_HDMI_DDC_CLK_M(div_m) | SUN4I_HDMI_DDC_CLK_N(div_n),
89+
ddc->hdmi->base + SUN4I_HDMI_DDC_CLK_REG);
90+
91+
return 0;
92+
}
93+
94+
static const struct clk_ops sun4i_ddc_ops = {
95+
.recalc_rate = sun4i_ddc_recalc_rate,
96+
.round_rate = sun4i_ddc_round_rate,
97+
.set_rate = sun4i_ddc_set_rate,
98+
};
99+
100+
int sun4i_ddc_create(struct sun4i_hdmi *hdmi, struct clk *parent)
101+
{
102+
struct clk_init_data init;
103+
struct sun4i_ddc *ddc;
104+
const char *parent_name;
105+
106+
parent_name = __clk_get_name(parent);
107+
if (!parent_name)
108+
return -ENODEV;
109+
110+
ddc = devm_kzalloc(hdmi->dev, sizeof(*ddc), GFP_KERNEL);
111+
if (!ddc)
112+
return -ENOMEM;
113+
114+
init.name = "hdmi-ddc";
115+
init.ops = &sun4i_ddc_ops;
116+
init.parent_names = &parent_name;
117+
init.num_parents = 1;
118+
119+
ddc->hdmi = hdmi;
120+
ddc->hw.init = &init;
121+
122+
hdmi->ddc_clk = devm_clk_register(hdmi->dev, &ddc->hw);
123+
if (IS_ERR(hdmi->ddc_clk))
124+
return PTR_ERR(hdmi->ddc_clk);
125+
126+
return 0;
127+
}

0 commit comments

Comments
 (0)