mirror of
				https://github.com/torvalds/linux.git
				synced 2025-10-31 16:48:26 +02:00 
			
		
		
		
	 3d8853dddf
			
		
	
	
		3d8853dddf
		
	
	
	
	
		
			
			Several source files include drm_crtc_helper.h without needing it or only to get its transitive include statements; leading to unnecessary compile-time dependencies. Drop drm_crtc_helper.h where possible. v2: * update commit message (Sam) Signed-off-by: Thomas Zimmermann <tzimmermann@suse.de> Reviewed-by: Sam Ravnborg <sam@ravnborg.org> Acked-by: Jani Nikula <jani.nikula@intel.com> Acked-by: Alex Deucher <alexander.deucher@amd.com> Link: https://patchwork.freedesktop.org/patch/msgid/20230116131235.18917-6-tzimmermann@suse.de
		
			
				
	
	
		
			241 lines
		
	
	
	
		
			6.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			241 lines
		
	
	
	
		
			6.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0+
 | |
| // Copyright 2018 IBM Corporation
 | |
| 
 | |
| #include <linux/clk.h>
 | |
| #include <linux/reset.h>
 | |
| #include <linux/regmap.h>
 | |
| 
 | |
| #include <drm/drm_device.h>
 | |
| #include <drm/drm_fb_dma_helper.h>
 | |
| #include <drm/drm_fourcc.h>
 | |
| #include <drm/drm_framebuffer.h>
 | |
| #include <drm/drm_gem_atomic_helper.h>
 | |
| #include <drm/drm_gem_dma_helper.h>
 | |
| #include <drm/drm_panel.h>
 | |
| #include <drm/drm_simple_kms_helper.h>
 | |
| #include <drm/drm_vblank.h>
 | |
| 
 | |
| #include "aspeed_gfx.h"
 | |
| 
 | |
| static struct aspeed_gfx *
 | |
| drm_pipe_to_aspeed_gfx(struct drm_simple_display_pipe *pipe)
 | |
| {
 | |
| 	return container_of(pipe, struct aspeed_gfx, pipe);
 | |
| }
 | |
| 
 | |
| static int aspeed_gfx_set_pixel_fmt(struct aspeed_gfx *priv, u32 *bpp)
 | |
| {
 | |
| 	struct drm_crtc *crtc = &priv->pipe.crtc;
 | |
| 	struct drm_device *drm = crtc->dev;
 | |
| 	const u32 format = crtc->primary->state->fb->format->format;
 | |
| 	u32 ctrl1;
 | |
| 
 | |
| 	ctrl1 = readl(priv->base + CRT_CTRL1);
 | |
| 	ctrl1 &= ~CRT_CTRL_COLOR_MASK;
 | |
| 
 | |
| 	switch (format) {
 | |
| 	case DRM_FORMAT_RGB565:
 | |
| 		dev_dbg(drm->dev, "Setting up RGB565 mode\n");
 | |
| 		ctrl1 |= CRT_CTRL_COLOR_RGB565;
 | |
| 		*bpp = 16;
 | |
| 		break;
 | |
| 	case DRM_FORMAT_XRGB8888:
 | |
| 		dev_dbg(drm->dev, "Setting up XRGB8888 mode\n");
 | |
| 		ctrl1 |= CRT_CTRL_COLOR_XRGB8888;
 | |
| 		*bpp = 32;
 | |
| 		break;
 | |
| 	default:
 | |
| 		dev_err(drm->dev, "Unhandled pixel format %08x\n", format);
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 
 | |
| 	writel(ctrl1, priv->base + CRT_CTRL1);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static void aspeed_gfx_enable_controller(struct aspeed_gfx *priv)
 | |
| {
 | |
| 	u32 ctrl1 = readl(priv->base + CRT_CTRL1);
 | |
| 	u32 ctrl2 = readl(priv->base + CRT_CTRL2);
 | |
| 
 | |
| 	/* Set DAC source for display output to Graphics CRT (GFX) */
 | |
| 	regmap_update_bits(priv->scu, priv->dac_reg, BIT(16), BIT(16));
 | |
| 
 | |
| 	writel(ctrl1 | CRT_CTRL_EN, priv->base + CRT_CTRL1);
 | |
| 	writel(ctrl2 | CRT_CTRL_DAC_EN, priv->base + CRT_CTRL2);
 | |
| }
 | |
| 
 | |
| static void aspeed_gfx_disable_controller(struct aspeed_gfx *priv)
 | |
| {
 | |
| 	u32 ctrl1 = readl(priv->base + CRT_CTRL1);
 | |
| 	u32 ctrl2 = readl(priv->base + CRT_CTRL2);
 | |
| 
 | |
| 	writel(ctrl1 & ~CRT_CTRL_EN, priv->base + CRT_CTRL1);
 | |
| 	writel(ctrl2 & ~CRT_CTRL_DAC_EN, priv->base + CRT_CTRL2);
 | |
| 
 | |
| 	regmap_update_bits(priv->scu, priv->dac_reg, BIT(16), 0);
 | |
| }
 | |
| 
 | |
| static void aspeed_gfx_crtc_mode_set_nofb(struct aspeed_gfx *priv)
 | |
| {
 | |
| 	struct drm_display_mode *m = &priv->pipe.crtc.state->adjusted_mode;
 | |
| 	u32 ctrl1, d_offset, t_count, bpp;
 | |
| 	int err;
 | |
| 
 | |
| 	err = aspeed_gfx_set_pixel_fmt(priv, &bpp);
 | |
| 	if (err)
 | |
| 		return;
 | |
| 
 | |
| #if 0
 | |
| 	/* TODO: we have only been able to test with the 40MHz USB clock. The
 | |
| 	 * clock is fixed, so we cannot adjust it here. */
 | |
| 	clk_set_rate(priv->pixel_clk, m->crtc_clock * 1000);
 | |
| #endif
 | |
| 
 | |
| 	ctrl1 = readl(priv->base + CRT_CTRL1);
 | |
| 	ctrl1 &= ~(CRT_CTRL_INTERLACED |
 | |
| 			CRT_CTRL_HSYNC_NEGATIVE |
 | |
| 			CRT_CTRL_VSYNC_NEGATIVE);
 | |
| 
 | |
| 	if (m->flags & DRM_MODE_FLAG_INTERLACE)
 | |
| 		ctrl1 |= CRT_CTRL_INTERLACED;
 | |
| 
 | |
| 	if (!(m->flags & DRM_MODE_FLAG_PHSYNC))
 | |
| 		ctrl1 |= CRT_CTRL_HSYNC_NEGATIVE;
 | |
| 
 | |
| 	if (!(m->flags & DRM_MODE_FLAG_PVSYNC))
 | |
| 		ctrl1 |= CRT_CTRL_VSYNC_NEGATIVE;
 | |
| 
 | |
| 	writel(ctrl1, priv->base + CRT_CTRL1);
 | |
| 
 | |
| 	/* Horizontal timing */
 | |
| 	writel(CRT_H_TOTAL(m->htotal - 1) | CRT_H_DE(m->hdisplay - 1),
 | |
| 			priv->base + CRT_HORIZ0);
 | |
| 	writel(CRT_H_RS_START(m->hsync_start - 1) | CRT_H_RS_END(m->hsync_end),
 | |
| 			priv->base + CRT_HORIZ1);
 | |
| 
 | |
| 
 | |
| 	/* Vertical timing */
 | |
| 	writel(CRT_V_TOTAL(m->vtotal - 1) | CRT_V_DE(m->vdisplay - 1),
 | |
| 			priv->base + CRT_VERT0);
 | |
| 	writel(CRT_V_RS_START(m->vsync_start) | CRT_V_RS_END(m->vsync_end),
 | |
| 			priv->base + CRT_VERT1);
 | |
| 
 | |
| 	/*
 | |
| 	 * Display Offset: address difference between consecutive scan lines
 | |
| 	 * Terminal Count: memory size of one scan line
 | |
| 	 */
 | |
| 	d_offset = m->hdisplay * bpp / 8;
 | |
| 	t_count = DIV_ROUND_UP(m->hdisplay * bpp, priv->scan_line_max);
 | |
| 
 | |
| 	writel(CRT_DISP_OFFSET(d_offset) | CRT_TERM_COUNT(t_count),
 | |
| 			priv->base + CRT_OFFSET);
 | |
| 
 | |
| 	/*
 | |
| 	 * Threshold: FIFO thresholds of refill and stop (16 byte chunks
 | |
| 	 * per line, rounded up)
 | |
| 	 */
 | |
| 	writel(priv->throd_val, priv->base + CRT_THROD);
 | |
| }
 | |
| 
 | |
| static void aspeed_gfx_pipe_enable(struct drm_simple_display_pipe *pipe,
 | |
| 			      struct drm_crtc_state *crtc_state,
 | |
| 			      struct drm_plane_state *plane_state)
 | |
| {
 | |
| 	struct aspeed_gfx *priv = drm_pipe_to_aspeed_gfx(pipe);
 | |
| 	struct drm_crtc *crtc = &pipe->crtc;
 | |
| 
 | |
| 	aspeed_gfx_crtc_mode_set_nofb(priv);
 | |
| 	aspeed_gfx_enable_controller(priv);
 | |
| 	drm_crtc_vblank_on(crtc);
 | |
| }
 | |
| 
 | |
| static void aspeed_gfx_pipe_disable(struct drm_simple_display_pipe *pipe)
 | |
| {
 | |
| 	struct aspeed_gfx *priv = drm_pipe_to_aspeed_gfx(pipe);
 | |
| 	struct drm_crtc *crtc = &pipe->crtc;
 | |
| 
 | |
| 	drm_crtc_vblank_off(crtc);
 | |
| 	aspeed_gfx_disable_controller(priv);
 | |
| }
 | |
| 
 | |
| static void aspeed_gfx_pipe_update(struct drm_simple_display_pipe *pipe,
 | |
| 				   struct drm_plane_state *plane_state)
 | |
| {
 | |
| 	struct aspeed_gfx *priv = drm_pipe_to_aspeed_gfx(pipe);
 | |
| 	struct drm_crtc *crtc = &pipe->crtc;
 | |
| 	struct drm_framebuffer *fb = pipe->plane.state->fb;
 | |
| 	struct drm_pending_vblank_event *event;
 | |
| 	struct drm_gem_dma_object *gem;
 | |
| 
 | |
| 	spin_lock_irq(&crtc->dev->event_lock);
 | |
| 	event = crtc->state->event;
 | |
| 	if (event) {
 | |
| 		crtc->state->event = NULL;
 | |
| 
 | |
| 		if (drm_crtc_vblank_get(crtc) == 0)
 | |
| 			drm_crtc_arm_vblank_event(crtc, event);
 | |
| 		else
 | |
| 			drm_crtc_send_vblank_event(crtc, event);
 | |
| 	}
 | |
| 	spin_unlock_irq(&crtc->dev->event_lock);
 | |
| 
 | |
| 	if (!fb)
 | |
| 		return;
 | |
| 
 | |
| 	gem = drm_fb_dma_get_gem_obj(fb, 0);
 | |
| 	if (!gem)
 | |
| 		return;
 | |
| 	writel(gem->dma_addr, priv->base + CRT_ADDR);
 | |
| }
 | |
| 
 | |
| static int aspeed_gfx_enable_vblank(struct drm_simple_display_pipe *pipe)
 | |
| {
 | |
| 	struct aspeed_gfx *priv = drm_pipe_to_aspeed_gfx(pipe);
 | |
| 	u32 reg = readl(priv->base + CRT_CTRL1);
 | |
| 
 | |
| 	/* Clear pending VBLANK IRQ */
 | |
| 	writel(reg | CRT_CTRL_VERTICAL_INTR_STS, priv->base + CRT_CTRL1);
 | |
| 
 | |
| 	reg |= CRT_CTRL_VERTICAL_INTR_EN;
 | |
| 	writel(reg, priv->base + CRT_CTRL1);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static void aspeed_gfx_disable_vblank(struct drm_simple_display_pipe *pipe)
 | |
| {
 | |
| 	struct aspeed_gfx *priv = drm_pipe_to_aspeed_gfx(pipe);
 | |
| 	u32 reg = readl(priv->base + CRT_CTRL1);
 | |
| 
 | |
| 	reg &= ~CRT_CTRL_VERTICAL_INTR_EN;
 | |
| 	writel(reg, priv->base + CRT_CTRL1);
 | |
| 
 | |
| 	/* Clear pending VBLANK IRQ */
 | |
| 	writel(reg | CRT_CTRL_VERTICAL_INTR_STS, priv->base + CRT_CTRL1);
 | |
| }
 | |
| 
 | |
| static const struct drm_simple_display_pipe_funcs aspeed_gfx_funcs = {
 | |
| 	.enable		= aspeed_gfx_pipe_enable,
 | |
| 	.disable	= aspeed_gfx_pipe_disable,
 | |
| 	.update		= aspeed_gfx_pipe_update,
 | |
| 	.enable_vblank	= aspeed_gfx_enable_vblank,
 | |
| 	.disable_vblank	= aspeed_gfx_disable_vblank,
 | |
| };
 | |
| 
 | |
| static const uint32_t aspeed_gfx_formats[] = {
 | |
| 	DRM_FORMAT_XRGB8888,
 | |
| 	DRM_FORMAT_RGB565,
 | |
| };
 | |
| 
 | |
| int aspeed_gfx_create_pipe(struct drm_device *drm)
 | |
| {
 | |
| 	struct aspeed_gfx *priv = to_aspeed_gfx(drm);
 | |
| 
 | |
| 	return drm_simple_display_pipe_init(drm, &priv->pipe, &aspeed_gfx_funcs,
 | |
| 					    aspeed_gfx_formats,
 | |
| 					    ARRAY_SIZE(aspeed_gfx_formats),
 | |
| 					    NULL,
 | |
| 					    &priv->connector);
 | |
| }
 |