xref: /OK3568_Linux_fs/kernel/drivers/gpu/drm/aspeed/aspeed_gfx_crtc.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0+
2*4882a593Smuzhiyun // Copyright 2018 IBM Corporation
3*4882a593Smuzhiyun 
4*4882a593Smuzhiyun #include <linux/clk.h>
5*4882a593Smuzhiyun #include <linux/reset.h>
6*4882a593Smuzhiyun #include <linux/regmap.h>
7*4882a593Smuzhiyun 
8*4882a593Smuzhiyun #include <drm/drm_crtc_helper.h>
9*4882a593Smuzhiyun #include <drm/drm_device.h>
10*4882a593Smuzhiyun #include <drm/drm_fb_cma_helper.h>
11*4882a593Smuzhiyun #include <drm/drm_fourcc.h>
12*4882a593Smuzhiyun #include <drm/drm_gem_cma_helper.h>
13*4882a593Smuzhiyun #include <drm/drm_gem_framebuffer_helper.h>
14*4882a593Smuzhiyun #include <drm/drm_panel.h>
15*4882a593Smuzhiyun #include <drm/drm_simple_kms_helper.h>
16*4882a593Smuzhiyun #include <drm/drm_vblank.h>
17*4882a593Smuzhiyun 
18*4882a593Smuzhiyun #include "aspeed_gfx.h"
19*4882a593Smuzhiyun 
20*4882a593Smuzhiyun static struct aspeed_gfx *
drm_pipe_to_aspeed_gfx(struct drm_simple_display_pipe * pipe)21*4882a593Smuzhiyun drm_pipe_to_aspeed_gfx(struct drm_simple_display_pipe *pipe)
22*4882a593Smuzhiyun {
23*4882a593Smuzhiyun 	return container_of(pipe, struct aspeed_gfx, pipe);
24*4882a593Smuzhiyun }
25*4882a593Smuzhiyun 
aspeed_gfx_set_pixel_fmt(struct aspeed_gfx * priv,u32 * bpp)26*4882a593Smuzhiyun static int aspeed_gfx_set_pixel_fmt(struct aspeed_gfx *priv, u32 *bpp)
27*4882a593Smuzhiyun {
28*4882a593Smuzhiyun 	struct drm_crtc *crtc = &priv->pipe.crtc;
29*4882a593Smuzhiyun 	struct drm_device *drm = crtc->dev;
30*4882a593Smuzhiyun 	const u32 format = crtc->primary->state->fb->format->format;
31*4882a593Smuzhiyun 	u32 ctrl1;
32*4882a593Smuzhiyun 
33*4882a593Smuzhiyun 	ctrl1 = readl(priv->base + CRT_CTRL1);
34*4882a593Smuzhiyun 	ctrl1 &= ~CRT_CTRL_COLOR_MASK;
35*4882a593Smuzhiyun 
36*4882a593Smuzhiyun 	switch (format) {
37*4882a593Smuzhiyun 	case DRM_FORMAT_RGB565:
38*4882a593Smuzhiyun 		dev_dbg(drm->dev, "Setting up RGB565 mode\n");
39*4882a593Smuzhiyun 		ctrl1 |= CRT_CTRL_COLOR_RGB565;
40*4882a593Smuzhiyun 		*bpp = 16;
41*4882a593Smuzhiyun 		break;
42*4882a593Smuzhiyun 	case DRM_FORMAT_XRGB8888:
43*4882a593Smuzhiyun 		dev_dbg(drm->dev, "Setting up XRGB8888 mode\n");
44*4882a593Smuzhiyun 		ctrl1 |= CRT_CTRL_COLOR_XRGB8888;
45*4882a593Smuzhiyun 		*bpp = 32;
46*4882a593Smuzhiyun 		break;
47*4882a593Smuzhiyun 	default:
48*4882a593Smuzhiyun 		dev_err(drm->dev, "Unhandled pixel format %08x\n", format);
49*4882a593Smuzhiyun 		return -EINVAL;
50*4882a593Smuzhiyun 	}
51*4882a593Smuzhiyun 
52*4882a593Smuzhiyun 	writel(ctrl1, priv->base + CRT_CTRL1);
53*4882a593Smuzhiyun 
54*4882a593Smuzhiyun 	return 0;
55*4882a593Smuzhiyun }
56*4882a593Smuzhiyun 
aspeed_gfx_enable_controller(struct aspeed_gfx * priv)57*4882a593Smuzhiyun static void aspeed_gfx_enable_controller(struct aspeed_gfx *priv)
58*4882a593Smuzhiyun {
59*4882a593Smuzhiyun 	u32 ctrl1 = readl(priv->base + CRT_CTRL1);
60*4882a593Smuzhiyun 	u32 ctrl2 = readl(priv->base + CRT_CTRL2);
61*4882a593Smuzhiyun 
62*4882a593Smuzhiyun 	/* SCU2C: set DAC source for display output to Graphics CRT (GFX) */
63*4882a593Smuzhiyun 	regmap_update_bits(priv->scu, 0x2c, BIT(16), BIT(16));
64*4882a593Smuzhiyun 
65*4882a593Smuzhiyun 	writel(ctrl1 | CRT_CTRL_EN, priv->base + CRT_CTRL1);
66*4882a593Smuzhiyun 	writel(ctrl2 | CRT_CTRL_DAC_EN, priv->base + CRT_CTRL2);
67*4882a593Smuzhiyun }
68*4882a593Smuzhiyun 
aspeed_gfx_disable_controller(struct aspeed_gfx * priv)69*4882a593Smuzhiyun static void aspeed_gfx_disable_controller(struct aspeed_gfx *priv)
70*4882a593Smuzhiyun {
71*4882a593Smuzhiyun 	u32 ctrl1 = readl(priv->base + CRT_CTRL1);
72*4882a593Smuzhiyun 	u32 ctrl2 = readl(priv->base + CRT_CTRL2);
73*4882a593Smuzhiyun 
74*4882a593Smuzhiyun 	writel(ctrl1 & ~CRT_CTRL_EN, priv->base + CRT_CTRL1);
75*4882a593Smuzhiyun 	writel(ctrl2 & ~CRT_CTRL_DAC_EN, priv->base + CRT_CTRL2);
76*4882a593Smuzhiyun 
77*4882a593Smuzhiyun 	regmap_update_bits(priv->scu, 0x2c, BIT(16), 0);
78*4882a593Smuzhiyun }
79*4882a593Smuzhiyun 
aspeed_gfx_crtc_mode_set_nofb(struct aspeed_gfx * priv)80*4882a593Smuzhiyun static void aspeed_gfx_crtc_mode_set_nofb(struct aspeed_gfx *priv)
81*4882a593Smuzhiyun {
82*4882a593Smuzhiyun 	struct drm_display_mode *m = &priv->pipe.crtc.state->adjusted_mode;
83*4882a593Smuzhiyun 	u32 ctrl1, d_offset, t_count, bpp;
84*4882a593Smuzhiyun 	int err;
85*4882a593Smuzhiyun 
86*4882a593Smuzhiyun 	err = aspeed_gfx_set_pixel_fmt(priv, &bpp);
87*4882a593Smuzhiyun 	if (err)
88*4882a593Smuzhiyun 		return;
89*4882a593Smuzhiyun 
90*4882a593Smuzhiyun #if 0
91*4882a593Smuzhiyun 	/* TODO: we have only been able to test with the 40MHz USB clock. The
92*4882a593Smuzhiyun 	 * clock is fixed, so we cannot adjust it here. */
93*4882a593Smuzhiyun 	clk_set_rate(priv->pixel_clk, m->crtc_clock * 1000);
94*4882a593Smuzhiyun #endif
95*4882a593Smuzhiyun 
96*4882a593Smuzhiyun 	ctrl1 = readl(priv->base + CRT_CTRL1);
97*4882a593Smuzhiyun 	ctrl1 &= ~(CRT_CTRL_INTERLACED |
98*4882a593Smuzhiyun 			CRT_CTRL_HSYNC_NEGATIVE |
99*4882a593Smuzhiyun 			CRT_CTRL_VSYNC_NEGATIVE);
100*4882a593Smuzhiyun 
101*4882a593Smuzhiyun 	if (m->flags & DRM_MODE_FLAG_INTERLACE)
102*4882a593Smuzhiyun 		ctrl1 |= CRT_CTRL_INTERLACED;
103*4882a593Smuzhiyun 
104*4882a593Smuzhiyun 	if (!(m->flags & DRM_MODE_FLAG_PHSYNC))
105*4882a593Smuzhiyun 		ctrl1 |= CRT_CTRL_HSYNC_NEGATIVE;
106*4882a593Smuzhiyun 
107*4882a593Smuzhiyun 	if (!(m->flags & DRM_MODE_FLAG_PVSYNC))
108*4882a593Smuzhiyun 		ctrl1 |= CRT_CTRL_VSYNC_NEGATIVE;
109*4882a593Smuzhiyun 
110*4882a593Smuzhiyun 	writel(ctrl1, priv->base + CRT_CTRL1);
111*4882a593Smuzhiyun 
112*4882a593Smuzhiyun 	/* Horizontal timing */
113*4882a593Smuzhiyun 	writel(CRT_H_TOTAL(m->htotal - 1) | CRT_H_DE(m->hdisplay - 1),
114*4882a593Smuzhiyun 			priv->base + CRT_HORIZ0);
115*4882a593Smuzhiyun 	writel(CRT_H_RS_START(m->hsync_start - 1) | CRT_H_RS_END(m->hsync_end),
116*4882a593Smuzhiyun 			priv->base + CRT_HORIZ1);
117*4882a593Smuzhiyun 
118*4882a593Smuzhiyun 
119*4882a593Smuzhiyun 	/* Vertical timing */
120*4882a593Smuzhiyun 	writel(CRT_V_TOTAL(m->vtotal - 1) | CRT_V_DE(m->vdisplay - 1),
121*4882a593Smuzhiyun 			priv->base + CRT_VERT0);
122*4882a593Smuzhiyun 	writel(CRT_V_RS_START(m->vsync_start) | CRT_V_RS_END(m->vsync_end),
123*4882a593Smuzhiyun 			priv->base + CRT_VERT1);
124*4882a593Smuzhiyun 
125*4882a593Smuzhiyun 	/*
126*4882a593Smuzhiyun 	 * Display Offset: address difference between consecutive scan lines
127*4882a593Smuzhiyun 	 * Terminal Count: memory size of one scan line
128*4882a593Smuzhiyun 	 */
129*4882a593Smuzhiyun 	d_offset = m->hdisplay * bpp / 8;
130*4882a593Smuzhiyun 	t_count = (m->hdisplay * bpp + 127) / 128;
131*4882a593Smuzhiyun 	writel(CRT_DISP_OFFSET(d_offset) | CRT_TERM_COUNT(t_count),
132*4882a593Smuzhiyun 			priv->base + CRT_OFFSET);
133*4882a593Smuzhiyun 
134*4882a593Smuzhiyun 	/*
135*4882a593Smuzhiyun 	 * Threshold: FIFO thresholds of refill and stop (16 byte chunks
136*4882a593Smuzhiyun 	 * per line, rounded up)
137*4882a593Smuzhiyun 	 */
138*4882a593Smuzhiyun 	writel(G5_CRT_THROD_VAL, priv->base + CRT_THROD);
139*4882a593Smuzhiyun }
140*4882a593Smuzhiyun 
aspeed_gfx_pipe_enable(struct drm_simple_display_pipe * pipe,struct drm_crtc_state * crtc_state,struct drm_plane_state * plane_state)141*4882a593Smuzhiyun static void aspeed_gfx_pipe_enable(struct drm_simple_display_pipe *pipe,
142*4882a593Smuzhiyun 			      struct drm_crtc_state *crtc_state,
143*4882a593Smuzhiyun 			      struct drm_plane_state *plane_state)
144*4882a593Smuzhiyun {
145*4882a593Smuzhiyun 	struct aspeed_gfx *priv = drm_pipe_to_aspeed_gfx(pipe);
146*4882a593Smuzhiyun 	struct drm_crtc *crtc = &pipe->crtc;
147*4882a593Smuzhiyun 
148*4882a593Smuzhiyun 	aspeed_gfx_crtc_mode_set_nofb(priv);
149*4882a593Smuzhiyun 	aspeed_gfx_enable_controller(priv);
150*4882a593Smuzhiyun 	drm_crtc_vblank_on(crtc);
151*4882a593Smuzhiyun }
152*4882a593Smuzhiyun 
aspeed_gfx_pipe_disable(struct drm_simple_display_pipe * pipe)153*4882a593Smuzhiyun static void aspeed_gfx_pipe_disable(struct drm_simple_display_pipe *pipe)
154*4882a593Smuzhiyun {
155*4882a593Smuzhiyun 	struct aspeed_gfx *priv = drm_pipe_to_aspeed_gfx(pipe);
156*4882a593Smuzhiyun 	struct drm_crtc *crtc = &pipe->crtc;
157*4882a593Smuzhiyun 
158*4882a593Smuzhiyun 	drm_crtc_vblank_off(crtc);
159*4882a593Smuzhiyun 	aspeed_gfx_disable_controller(priv);
160*4882a593Smuzhiyun }
161*4882a593Smuzhiyun 
aspeed_gfx_pipe_update(struct drm_simple_display_pipe * pipe,struct drm_plane_state * plane_state)162*4882a593Smuzhiyun static void aspeed_gfx_pipe_update(struct drm_simple_display_pipe *pipe,
163*4882a593Smuzhiyun 				   struct drm_plane_state *plane_state)
164*4882a593Smuzhiyun {
165*4882a593Smuzhiyun 	struct aspeed_gfx *priv = drm_pipe_to_aspeed_gfx(pipe);
166*4882a593Smuzhiyun 	struct drm_crtc *crtc = &pipe->crtc;
167*4882a593Smuzhiyun 	struct drm_framebuffer *fb = pipe->plane.state->fb;
168*4882a593Smuzhiyun 	struct drm_pending_vblank_event *event;
169*4882a593Smuzhiyun 	struct drm_gem_cma_object *gem;
170*4882a593Smuzhiyun 
171*4882a593Smuzhiyun 	spin_lock_irq(&crtc->dev->event_lock);
172*4882a593Smuzhiyun 	event = crtc->state->event;
173*4882a593Smuzhiyun 	if (event) {
174*4882a593Smuzhiyun 		crtc->state->event = NULL;
175*4882a593Smuzhiyun 
176*4882a593Smuzhiyun 		if (drm_crtc_vblank_get(crtc) == 0)
177*4882a593Smuzhiyun 			drm_crtc_arm_vblank_event(crtc, event);
178*4882a593Smuzhiyun 		else
179*4882a593Smuzhiyun 			drm_crtc_send_vblank_event(crtc, event);
180*4882a593Smuzhiyun 	}
181*4882a593Smuzhiyun 	spin_unlock_irq(&crtc->dev->event_lock);
182*4882a593Smuzhiyun 
183*4882a593Smuzhiyun 	if (!fb)
184*4882a593Smuzhiyun 		return;
185*4882a593Smuzhiyun 
186*4882a593Smuzhiyun 	gem = drm_fb_cma_get_gem_obj(fb, 0);
187*4882a593Smuzhiyun 	if (!gem)
188*4882a593Smuzhiyun 		return;
189*4882a593Smuzhiyun 	writel(gem->paddr, priv->base + CRT_ADDR);
190*4882a593Smuzhiyun }
191*4882a593Smuzhiyun 
aspeed_gfx_enable_vblank(struct drm_simple_display_pipe * pipe)192*4882a593Smuzhiyun static int aspeed_gfx_enable_vblank(struct drm_simple_display_pipe *pipe)
193*4882a593Smuzhiyun {
194*4882a593Smuzhiyun 	struct aspeed_gfx *priv = drm_pipe_to_aspeed_gfx(pipe);
195*4882a593Smuzhiyun 	u32 reg = readl(priv->base + CRT_CTRL1);
196*4882a593Smuzhiyun 
197*4882a593Smuzhiyun 	/* Clear pending VBLANK IRQ */
198*4882a593Smuzhiyun 	writel(reg | CRT_CTRL_VERTICAL_INTR_STS, priv->base + CRT_CTRL1);
199*4882a593Smuzhiyun 
200*4882a593Smuzhiyun 	reg |= CRT_CTRL_VERTICAL_INTR_EN;
201*4882a593Smuzhiyun 	writel(reg, priv->base + CRT_CTRL1);
202*4882a593Smuzhiyun 
203*4882a593Smuzhiyun 	return 0;
204*4882a593Smuzhiyun }
205*4882a593Smuzhiyun 
aspeed_gfx_disable_vblank(struct drm_simple_display_pipe * pipe)206*4882a593Smuzhiyun static void aspeed_gfx_disable_vblank(struct drm_simple_display_pipe *pipe)
207*4882a593Smuzhiyun {
208*4882a593Smuzhiyun 	struct aspeed_gfx *priv = drm_pipe_to_aspeed_gfx(pipe);
209*4882a593Smuzhiyun 	u32 reg = readl(priv->base + CRT_CTRL1);
210*4882a593Smuzhiyun 
211*4882a593Smuzhiyun 	reg &= ~CRT_CTRL_VERTICAL_INTR_EN;
212*4882a593Smuzhiyun 	writel(reg, priv->base + CRT_CTRL1);
213*4882a593Smuzhiyun 
214*4882a593Smuzhiyun 	/* Clear pending VBLANK IRQ */
215*4882a593Smuzhiyun 	writel(reg | CRT_CTRL_VERTICAL_INTR_STS, priv->base + CRT_CTRL1);
216*4882a593Smuzhiyun }
217*4882a593Smuzhiyun 
218*4882a593Smuzhiyun static const struct drm_simple_display_pipe_funcs aspeed_gfx_funcs = {
219*4882a593Smuzhiyun 	.enable		= aspeed_gfx_pipe_enable,
220*4882a593Smuzhiyun 	.disable	= aspeed_gfx_pipe_disable,
221*4882a593Smuzhiyun 	.update		= aspeed_gfx_pipe_update,
222*4882a593Smuzhiyun 	.prepare_fb	= drm_gem_fb_simple_display_pipe_prepare_fb,
223*4882a593Smuzhiyun 	.enable_vblank	= aspeed_gfx_enable_vblank,
224*4882a593Smuzhiyun 	.disable_vblank	= aspeed_gfx_disable_vblank,
225*4882a593Smuzhiyun };
226*4882a593Smuzhiyun 
227*4882a593Smuzhiyun static const uint32_t aspeed_gfx_formats[] = {
228*4882a593Smuzhiyun 	DRM_FORMAT_XRGB8888,
229*4882a593Smuzhiyun 	DRM_FORMAT_RGB565,
230*4882a593Smuzhiyun };
231*4882a593Smuzhiyun 
aspeed_gfx_create_pipe(struct drm_device * drm)232*4882a593Smuzhiyun int aspeed_gfx_create_pipe(struct drm_device *drm)
233*4882a593Smuzhiyun {
234*4882a593Smuzhiyun 	struct aspeed_gfx *priv = to_aspeed_gfx(drm);
235*4882a593Smuzhiyun 
236*4882a593Smuzhiyun 	return drm_simple_display_pipe_init(drm, &priv->pipe, &aspeed_gfx_funcs,
237*4882a593Smuzhiyun 					    aspeed_gfx_formats,
238*4882a593Smuzhiyun 					    ARRAY_SIZE(aspeed_gfx_formats),
239*4882a593Smuzhiyun 					    NULL,
240*4882a593Smuzhiyun 					    &priv->connector);
241*4882a593Smuzhiyun }
242