1dee8268fSThierry Reding /* 2dee8268fSThierry Reding * Copyright (C) 2012 Avionic Design GmbH 3dee8268fSThierry Reding * Copyright (C) 2012 NVIDIA CORPORATION. All rights reserved. 4dee8268fSThierry Reding * 5dee8268fSThierry Reding * This program is free software; you can redistribute it and/or modify 6dee8268fSThierry Reding * it under the terms of the GNU General Public License version 2 as 7dee8268fSThierry Reding * published by the Free Software Foundation. 8dee8268fSThierry Reding */ 9dee8268fSThierry Reding 10dee8268fSThierry Reding #include <linux/clk.h> 11dee8268fSThierry Reding #include <linux/debugfs.h> 12df06b759SThierry Reding #include <linux/iommu.h> 13b9ff7aeaSThierry Reding #include <linux/of_device.h> 1433a8eb8dSThierry Reding #include <linux/pm_runtime.h> 15ca48080aSStephen Warren #include <linux/reset.h> 16dee8268fSThierry Reding 179c012700SThierry Reding #include <soc/tegra/pmc.h> 189c012700SThierry Reding 19dee8268fSThierry Reding #include "dc.h" 20dee8268fSThierry Reding #include "drm.h" 21dee8268fSThierry Reding #include "gem.h" 2247307954SThierry Reding #include "hub.h" 235acd3514SThierry Reding #include "plane.h" 24dee8268fSThierry Reding 259d44189fSThierry Reding #include <drm/drm_atomic.h> 264aa3df71SThierry Reding #include <drm/drm_atomic_helper.h> 273cb9ae4fSDaniel Vetter #include <drm/drm_plane_helper.h> 283cb9ae4fSDaniel Vetter 29791ddb1eSThierry Reding static void tegra_dc_stats_reset(struct tegra_dc_stats *stats) 30791ddb1eSThierry Reding { 31791ddb1eSThierry Reding stats->frames = 0; 32791ddb1eSThierry Reding stats->vblank = 0; 33791ddb1eSThierry Reding stats->underflow = 0; 34791ddb1eSThierry Reding stats->overflow = 0; 35791ddb1eSThierry Reding } 36791ddb1eSThierry Reding 37d700ba7aSThierry Reding /* 3886df256fSThierry Reding * Reads the active copy of a register. This takes the dc->lock spinlock to 3986df256fSThierry Reding * prevent races with the VBLANK processing which also needs access to the 4086df256fSThierry Reding * active copy of some registers. 4186df256fSThierry Reding */ 4286df256fSThierry Reding static u32 tegra_dc_readl_active(struct tegra_dc *dc, unsigned long offset) 4386df256fSThierry Reding { 4486df256fSThierry Reding unsigned long flags; 4586df256fSThierry Reding u32 value; 4686df256fSThierry Reding 4786df256fSThierry Reding spin_lock_irqsave(&dc->lock, flags); 4886df256fSThierry Reding 4986df256fSThierry Reding tegra_dc_writel(dc, READ_MUX, DC_CMD_STATE_ACCESS); 5086df256fSThierry Reding value = tegra_dc_readl(dc, offset); 5186df256fSThierry Reding tegra_dc_writel(dc, 0, DC_CMD_STATE_ACCESS); 5286df256fSThierry Reding 5386df256fSThierry Reding spin_unlock_irqrestore(&dc->lock, flags); 5486df256fSThierry Reding return value; 5586df256fSThierry Reding } 5686df256fSThierry Reding 57c57997bcSThierry Reding bool tegra_dc_has_output(struct tegra_dc *dc, struct device *dev) 58c57997bcSThierry Reding { 59c57997bcSThierry Reding struct device_node *np = dc->dev->of_node; 60c57997bcSThierry Reding struct of_phandle_iterator it; 61c57997bcSThierry Reding int err; 62c57997bcSThierry Reding 63c57997bcSThierry Reding of_for_each_phandle(&it, err, np, "nvidia,outputs", NULL, 0) 64c57997bcSThierry Reding if (it.node == dev->of_node) 65c57997bcSThierry Reding return true; 66c57997bcSThierry Reding 67c57997bcSThierry Reding return false; 68c57997bcSThierry Reding } 69c57997bcSThierry Reding 7086df256fSThierry Reding /* 71d700ba7aSThierry Reding * Double-buffered registers have two copies: ASSEMBLY and ACTIVE. When the 72d700ba7aSThierry Reding * *_ACT_REQ bits are set the ASSEMBLY copy is latched into the ACTIVE copy. 73d700ba7aSThierry Reding * Latching happens mmediately if the display controller is in STOP mode or 74d700ba7aSThierry Reding * on the next frame boundary otherwise. 75d700ba7aSThierry Reding * 76d700ba7aSThierry Reding * Triple-buffered registers have three copies: ASSEMBLY, ARM and ACTIVE. The 77d700ba7aSThierry Reding * ASSEMBLY copy is latched into the ARM copy immediately after *_UPDATE bits 78d700ba7aSThierry Reding * are written. When the *_ACT_REQ bits are written, the ARM copy is latched 79d700ba7aSThierry Reding * into the ACTIVE copy, either immediately if the display controller is in 80d700ba7aSThierry Reding * STOP mode, or at the next frame boundary otherwise. 81d700ba7aSThierry Reding */ 8262b9e063SThierry Reding void tegra_dc_commit(struct tegra_dc *dc) 83205d48edSThierry Reding { 84205d48edSThierry Reding tegra_dc_writel(dc, GENERAL_ACT_REQ << 8, DC_CMD_STATE_CONTROL); 85205d48edSThierry Reding tegra_dc_writel(dc, GENERAL_ACT_REQ, DC_CMD_STATE_CONTROL); 86205d48edSThierry Reding } 87205d48edSThierry Reding 8810288eeaSThierry Reding static inline u32 compute_dda_inc(unsigned int in, unsigned int out, bool v, 8910288eeaSThierry Reding unsigned int bpp) 9010288eeaSThierry Reding { 9110288eeaSThierry Reding fixed20_12 outf = dfixed_init(out); 9210288eeaSThierry Reding fixed20_12 inf = dfixed_init(in); 9310288eeaSThierry Reding u32 dda_inc; 9410288eeaSThierry Reding int max; 9510288eeaSThierry Reding 9610288eeaSThierry Reding if (v) 9710288eeaSThierry Reding max = 15; 9810288eeaSThierry Reding else { 9910288eeaSThierry Reding switch (bpp) { 10010288eeaSThierry Reding case 2: 10110288eeaSThierry Reding max = 8; 10210288eeaSThierry Reding break; 10310288eeaSThierry Reding 10410288eeaSThierry Reding default: 10510288eeaSThierry Reding WARN_ON_ONCE(1); 10610288eeaSThierry Reding /* fallthrough */ 10710288eeaSThierry Reding case 4: 10810288eeaSThierry Reding max = 4; 10910288eeaSThierry Reding break; 11010288eeaSThierry Reding } 11110288eeaSThierry Reding } 11210288eeaSThierry Reding 11310288eeaSThierry Reding outf.full = max_t(u32, outf.full - dfixed_const(1), dfixed_const(1)); 11410288eeaSThierry Reding inf.full -= dfixed_const(1); 11510288eeaSThierry Reding 11610288eeaSThierry Reding dda_inc = dfixed_div(inf, outf); 11710288eeaSThierry Reding dda_inc = min_t(u32, dda_inc, dfixed_const(max)); 11810288eeaSThierry Reding 11910288eeaSThierry Reding return dda_inc; 12010288eeaSThierry Reding } 12110288eeaSThierry Reding 12210288eeaSThierry Reding static inline u32 compute_initial_dda(unsigned int in) 12310288eeaSThierry Reding { 12410288eeaSThierry Reding fixed20_12 inf = dfixed_init(in); 12510288eeaSThierry Reding return dfixed_frac(inf); 12610288eeaSThierry Reding } 12710288eeaSThierry Reding 1284aa3df71SThierry Reding static void tegra_dc_setup_window(struct tegra_dc *dc, unsigned int index, 12910288eeaSThierry Reding const struct tegra_dc_window *window) 13010288eeaSThierry Reding { 13110288eeaSThierry Reding unsigned h_offset, v_offset, h_size, v_size, h_dda, v_dda, bpp; 13293396d0fSSean Paul unsigned long value, flags; 13310288eeaSThierry Reding bool yuv, planar; 13410288eeaSThierry Reding 13510288eeaSThierry Reding /* 13610288eeaSThierry Reding * For YUV planar modes, the number of bytes per pixel takes into 13710288eeaSThierry Reding * account only the luma component and therefore is 1. 13810288eeaSThierry Reding */ 1395acd3514SThierry Reding yuv = tegra_plane_format_is_yuv(window->format, &planar); 14010288eeaSThierry Reding if (!yuv) 14110288eeaSThierry Reding bpp = window->bits_per_pixel / 8; 14210288eeaSThierry Reding else 14310288eeaSThierry Reding bpp = planar ? 1 : 2; 14410288eeaSThierry Reding 14593396d0fSSean Paul spin_lock_irqsave(&dc->lock, flags); 14693396d0fSSean Paul 14710288eeaSThierry Reding value = WINDOW_A_SELECT << index; 14810288eeaSThierry Reding tegra_dc_writel(dc, value, DC_CMD_DISPLAY_WINDOW_HEADER); 14910288eeaSThierry Reding 15010288eeaSThierry Reding tegra_dc_writel(dc, window->format, DC_WIN_COLOR_DEPTH); 15110288eeaSThierry Reding tegra_dc_writel(dc, window->swap, DC_WIN_BYTE_SWAP); 15210288eeaSThierry Reding 15310288eeaSThierry Reding value = V_POSITION(window->dst.y) | H_POSITION(window->dst.x); 15410288eeaSThierry Reding tegra_dc_writel(dc, value, DC_WIN_POSITION); 15510288eeaSThierry Reding 15610288eeaSThierry Reding value = V_SIZE(window->dst.h) | H_SIZE(window->dst.w); 15710288eeaSThierry Reding tegra_dc_writel(dc, value, DC_WIN_SIZE); 15810288eeaSThierry Reding 15910288eeaSThierry Reding h_offset = window->src.x * bpp; 16010288eeaSThierry Reding v_offset = window->src.y; 16110288eeaSThierry Reding h_size = window->src.w * bpp; 16210288eeaSThierry Reding v_size = window->src.h; 16310288eeaSThierry Reding 16410288eeaSThierry Reding value = V_PRESCALED_SIZE(v_size) | H_PRESCALED_SIZE(h_size); 16510288eeaSThierry Reding tegra_dc_writel(dc, value, DC_WIN_PRESCALED_SIZE); 16610288eeaSThierry Reding 16710288eeaSThierry Reding /* 16810288eeaSThierry Reding * For DDA computations the number of bytes per pixel for YUV planar 16910288eeaSThierry Reding * modes needs to take into account all Y, U and V components. 17010288eeaSThierry Reding */ 17110288eeaSThierry Reding if (yuv && planar) 17210288eeaSThierry Reding bpp = 2; 17310288eeaSThierry Reding 17410288eeaSThierry Reding h_dda = compute_dda_inc(window->src.w, window->dst.w, false, bpp); 17510288eeaSThierry Reding v_dda = compute_dda_inc(window->src.h, window->dst.h, true, bpp); 17610288eeaSThierry Reding 17710288eeaSThierry Reding value = V_DDA_INC(v_dda) | H_DDA_INC(h_dda); 17810288eeaSThierry Reding tegra_dc_writel(dc, value, DC_WIN_DDA_INC); 17910288eeaSThierry Reding 18010288eeaSThierry Reding h_dda = compute_initial_dda(window->src.x); 18110288eeaSThierry Reding v_dda = compute_initial_dda(window->src.y); 18210288eeaSThierry Reding 18310288eeaSThierry Reding tegra_dc_writel(dc, h_dda, DC_WIN_H_INITIAL_DDA); 18410288eeaSThierry Reding tegra_dc_writel(dc, v_dda, DC_WIN_V_INITIAL_DDA); 18510288eeaSThierry Reding 18610288eeaSThierry Reding tegra_dc_writel(dc, 0, DC_WIN_UV_BUF_STRIDE); 18710288eeaSThierry Reding tegra_dc_writel(dc, 0, DC_WIN_BUF_STRIDE); 18810288eeaSThierry Reding 18910288eeaSThierry Reding tegra_dc_writel(dc, window->base[0], DC_WINBUF_START_ADDR); 19010288eeaSThierry Reding 19110288eeaSThierry Reding if (yuv && planar) { 19210288eeaSThierry Reding tegra_dc_writel(dc, window->base[1], DC_WINBUF_START_ADDR_U); 19310288eeaSThierry Reding tegra_dc_writel(dc, window->base[2], DC_WINBUF_START_ADDR_V); 19410288eeaSThierry Reding value = window->stride[1] << 16 | window->stride[0]; 19510288eeaSThierry Reding tegra_dc_writel(dc, value, DC_WIN_LINE_STRIDE); 19610288eeaSThierry Reding } else { 19710288eeaSThierry Reding tegra_dc_writel(dc, window->stride[0], DC_WIN_LINE_STRIDE); 19810288eeaSThierry Reding } 19910288eeaSThierry Reding 20010288eeaSThierry Reding if (window->bottom_up) 20110288eeaSThierry Reding v_offset += window->src.h - 1; 20210288eeaSThierry Reding 20310288eeaSThierry Reding tegra_dc_writel(dc, h_offset, DC_WINBUF_ADDR_H_OFFSET); 20410288eeaSThierry Reding tegra_dc_writel(dc, v_offset, DC_WINBUF_ADDR_V_OFFSET); 20510288eeaSThierry Reding 206c134f019SThierry Reding if (dc->soc->supports_block_linear) { 207c134f019SThierry Reding unsigned long height = window->tiling.value; 208c134f019SThierry Reding 209c134f019SThierry Reding switch (window->tiling.mode) { 210c134f019SThierry Reding case TEGRA_BO_TILING_MODE_PITCH: 211c134f019SThierry Reding value = DC_WINBUF_SURFACE_KIND_PITCH; 212c134f019SThierry Reding break; 213c134f019SThierry Reding 214c134f019SThierry Reding case TEGRA_BO_TILING_MODE_TILED: 215c134f019SThierry Reding value = DC_WINBUF_SURFACE_KIND_TILED; 216c134f019SThierry Reding break; 217c134f019SThierry Reding 218c134f019SThierry Reding case TEGRA_BO_TILING_MODE_BLOCK: 219c134f019SThierry Reding value = DC_WINBUF_SURFACE_KIND_BLOCK_HEIGHT(height) | 220c134f019SThierry Reding DC_WINBUF_SURFACE_KIND_BLOCK; 221c134f019SThierry Reding break; 222c134f019SThierry Reding } 223c134f019SThierry Reding 224c134f019SThierry Reding tegra_dc_writel(dc, value, DC_WINBUF_SURFACE_KIND); 22510288eeaSThierry Reding } else { 226c134f019SThierry Reding switch (window->tiling.mode) { 227c134f019SThierry Reding case TEGRA_BO_TILING_MODE_PITCH: 22810288eeaSThierry Reding value = DC_WIN_BUFFER_ADDR_MODE_LINEAR_UV | 22910288eeaSThierry Reding DC_WIN_BUFFER_ADDR_MODE_LINEAR; 230c134f019SThierry Reding break; 231c134f019SThierry Reding 232c134f019SThierry Reding case TEGRA_BO_TILING_MODE_TILED: 233c134f019SThierry Reding value = DC_WIN_BUFFER_ADDR_MODE_TILE_UV | 234c134f019SThierry Reding DC_WIN_BUFFER_ADDR_MODE_TILE; 235c134f019SThierry Reding break; 236c134f019SThierry Reding 237c134f019SThierry Reding case TEGRA_BO_TILING_MODE_BLOCK: 2384aa3df71SThierry Reding /* 2394aa3df71SThierry Reding * No need to handle this here because ->atomic_check 2404aa3df71SThierry Reding * will already have filtered it out. 2414aa3df71SThierry Reding */ 2424aa3df71SThierry Reding break; 24310288eeaSThierry Reding } 24410288eeaSThierry Reding 24510288eeaSThierry Reding tegra_dc_writel(dc, value, DC_WIN_BUFFER_ADDR_MODE); 246c134f019SThierry Reding } 24710288eeaSThierry Reding 24810288eeaSThierry Reding value = WIN_ENABLE; 24910288eeaSThierry Reding 25010288eeaSThierry Reding if (yuv) { 25110288eeaSThierry Reding /* setup default colorspace conversion coefficients */ 25210288eeaSThierry Reding tegra_dc_writel(dc, 0x00f0, DC_WIN_CSC_YOF); 25310288eeaSThierry Reding tegra_dc_writel(dc, 0x012a, DC_WIN_CSC_KYRGB); 25410288eeaSThierry Reding tegra_dc_writel(dc, 0x0000, DC_WIN_CSC_KUR); 25510288eeaSThierry Reding tegra_dc_writel(dc, 0x0198, DC_WIN_CSC_KVR); 25610288eeaSThierry Reding tegra_dc_writel(dc, 0x039b, DC_WIN_CSC_KUG); 25710288eeaSThierry Reding tegra_dc_writel(dc, 0x032f, DC_WIN_CSC_KVG); 25810288eeaSThierry Reding tegra_dc_writel(dc, 0x0204, DC_WIN_CSC_KUB); 25910288eeaSThierry Reding tegra_dc_writel(dc, 0x0000, DC_WIN_CSC_KVB); 26010288eeaSThierry Reding 26110288eeaSThierry Reding value |= CSC_ENABLE; 26210288eeaSThierry Reding } else if (window->bits_per_pixel < 24) { 26310288eeaSThierry Reding value |= COLOR_EXPAND; 26410288eeaSThierry Reding } 26510288eeaSThierry Reding 26610288eeaSThierry Reding if (window->bottom_up) 26710288eeaSThierry Reding value |= V_DIRECTION; 26810288eeaSThierry Reding 26910288eeaSThierry Reding tegra_dc_writel(dc, value, DC_WIN_WIN_OPTIONS); 27010288eeaSThierry Reding 27110288eeaSThierry Reding /* 27210288eeaSThierry Reding * Disable blending and assume Window A is the bottom-most window, 27310288eeaSThierry Reding * Window C is the top-most window and Window B is in the middle. 27410288eeaSThierry Reding */ 27510288eeaSThierry Reding tegra_dc_writel(dc, 0xffff00, DC_WIN_BLEND_NOKEY); 27610288eeaSThierry Reding tegra_dc_writel(dc, 0xffff00, DC_WIN_BLEND_1WIN); 27710288eeaSThierry Reding 27810288eeaSThierry Reding switch (index) { 27910288eeaSThierry Reding case 0: 28010288eeaSThierry Reding tegra_dc_writel(dc, 0x000000, DC_WIN_BLEND_2WIN_X); 28110288eeaSThierry Reding tegra_dc_writel(dc, 0x000000, DC_WIN_BLEND_2WIN_Y); 28210288eeaSThierry Reding tegra_dc_writel(dc, 0x000000, DC_WIN_BLEND_3WIN_XY); 28310288eeaSThierry Reding break; 28410288eeaSThierry Reding 28510288eeaSThierry Reding case 1: 28610288eeaSThierry Reding tegra_dc_writel(dc, 0xffff00, DC_WIN_BLEND_2WIN_X); 28710288eeaSThierry Reding tegra_dc_writel(dc, 0x000000, DC_WIN_BLEND_2WIN_Y); 28810288eeaSThierry Reding tegra_dc_writel(dc, 0x000000, DC_WIN_BLEND_3WIN_XY); 28910288eeaSThierry Reding break; 29010288eeaSThierry Reding 29110288eeaSThierry Reding case 2: 29210288eeaSThierry Reding tegra_dc_writel(dc, 0xffff00, DC_WIN_BLEND_2WIN_X); 29310288eeaSThierry Reding tegra_dc_writel(dc, 0xffff00, DC_WIN_BLEND_2WIN_Y); 29410288eeaSThierry Reding tegra_dc_writel(dc, 0xffff00, DC_WIN_BLEND_3WIN_XY); 29510288eeaSThierry Reding break; 29610288eeaSThierry Reding } 29710288eeaSThierry Reding 29893396d0fSSean Paul spin_unlock_irqrestore(&dc->lock, flags); 299c7679306SThierry Reding } 300c7679306SThierry Reding 301*511c7023SThierry Reding static const u32 tegra20_primary_formats[] = { 302*511c7023SThierry Reding DRM_FORMAT_ARGB4444, 303*511c7023SThierry Reding DRM_FORMAT_ARGB1555, 304c7679306SThierry Reding DRM_FORMAT_RGB565, 305*511c7023SThierry Reding DRM_FORMAT_RGBA5551, 306*511c7023SThierry Reding DRM_FORMAT_ABGR8888, 307*511c7023SThierry Reding DRM_FORMAT_ARGB8888, 308*511c7023SThierry Reding }; 309*511c7023SThierry Reding 310*511c7023SThierry Reding static const u32 tegra114_primary_formats[] = { 311*511c7023SThierry Reding DRM_FORMAT_ARGB4444, 312*511c7023SThierry Reding DRM_FORMAT_ARGB1555, 313*511c7023SThierry Reding DRM_FORMAT_RGB565, 314*511c7023SThierry Reding DRM_FORMAT_RGBA5551, 315*511c7023SThierry Reding DRM_FORMAT_ABGR8888, 316*511c7023SThierry Reding DRM_FORMAT_ARGB8888, 317*511c7023SThierry Reding /* new on Tegra114 */ 318*511c7023SThierry Reding DRM_FORMAT_ABGR4444, 319*511c7023SThierry Reding DRM_FORMAT_ABGR1555, 320*511c7023SThierry Reding DRM_FORMAT_BGRA5551, 321*511c7023SThierry Reding DRM_FORMAT_XRGB1555, 322*511c7023SThierry Reding DRM_FORMAT_RGBX5551, 323*511c7023SThierry Reding DRM_FORMAT_XBGR1555, 324*511c7023SThierry Reding DRM_FORMAT_BGRX5551, 325*511c7023SThierry Reding DRM_FORMAT_BGR565, 326*511c7023SThierry Reding DRM_FORMAT_BGRA8888, 327*511c7023SThierry Reding DRM_FORMAT_RGBA8888, 328*511c7023SThierry Reding DRM_FORMAT_XRGB8888, 329*511c7023SThierry Reding DRM_FORMAT_XBGR8888, 330*511c7023SThierry Reding }; 331*511c7023SThierry Reding 332*511c7023SThierry Reding static const u32 tegra124_primary_formats[] = { 333*511c7023SThierry Reding DRM_FORMAT_ARGB4444, 334*511c7023SThierry Reding DRM_FORMAT_ARGB1555, 335*511c7023SThierry Reding DRM_FORMAT_RGB565, 336*511c7023SThierry Reding DRM_FORMAT_RGBA5551, 337*511c7023SThierry Reding DRM_FORMAT_ABGR8888, 338*511c7023SThierry Reding DRM_FORMAT_ARGB8888, 339*511c7023SThierry Reding /* new on Tegra114 */ 340*511c7023SThierry Reding DRM_FORMAT_ABGR4444, 341*511c7023SThierry Reding DRM_FORMAT_ABGR1555, 342*511c7023SThierry Reding DRM_FORMAT_BGRA5551, 343*511c7023SThierry Reding DRM_FORMAT_XRGB1555, 344*511c7023SThierry Reding DRM_FORMAT_RGBX5551, 345*511c7023SThierry Reding DRM_FORMAT_XBGR1555, 346*511c7023SThierry Reding DRM_FORMAT_BGRX5551, 347*511c7023SThierry Reding DRM_FORMAT_BGR565, 348*511c7023SThierry Reding DRM_FORMAT_BGRA8888, 349*511c7023SThierry Reding DRM_FORMAT_RGBA8888, 350*511c7023SThierry Reding DRM_FORMAT_XRGB8888, 351*511c7023SThierry Reding DRM_FORMAT_XBGR8888, 352*511c7023SThierry Reding /* new on Tegra124 */ 353*511c7023SThierry Reding DRM_FORMAT_RGBX8888, 354*511c7023SThierry Reding DRM_FORMAT_BGRX8888, 355c7679306SThierry Reding }; 356c7679306SThierry Reding 3574aa3df71SThierry Reding static int tegra_plane_atomic_check(struct drm_plane *plane, 3584aa3df71SThierry Reding struct drm_plane_state *state) 3594aa3df71SThierry Reding { 3608f604f8cSThierry Reding struct tegra_plane_state *plane_state = to_tegra_plane_state(state); 3618f604f8cSThierry Reding struct tegra_bo_tiling *tiling = &plane_state->tiling; 36247802b09SThierry Reding struct tegra_plane *tegra = to_tegra_plane(plane); 3634aa3df71SThierry Reding struct tegra_dc *dc = to_tegra_dc(state->crtc); 364c7679306SThierry Reding int err; 365c7679306SThierry Reding 3664aa3df71SThierry Reding /* no need for further checks if the plane is being disabled */ 3674aa3df71SThierry Reding if (!state->crtc) 3684aa3df71SThierry Reding return 0; 3694aa3df71SThierry Reding 3705acd3514SThierry Reding err = tegra_plane_format(state->fb->format->format, 3715acd3514SThierry Reding &plane_state->format, 3728f604f8cSThierry Reding &plane_state->swap); 3734aa3df71SThierry Reding if (err < 0) 3744aa3df71SThierry Reding return err; 3754aa3df71SThierry Reding 3768f604f8cSThierry Reding err = tegra_fb_get_tiling(state->fb, tiling); 3778f604f8cSThierry Reding if (err < 0) 3788f604f8cSThierry Reding return err; 3798f604f8cSThierry Reding 3808f604f8cSThierry Reding if (tiling->mode == TEGRA_BO_TILING_MODE_BLOCK && 3814aa3df71SThierry Reding !dc->soc->supports_block_linear) { 3824aa3df71SThierry Reding DRM_ERROR("hardware doesn't support block linear mode\n"); 3834aa3df71SThierry Reding return -EINVAL; 3844aa3df71SThierry Reding } 3854aa3df71SThierry Reding 3864aa3df71SThierry Reding /* 3874aa3df71SThierry Reding * Tegra doesn't support different strides for U and V planes so we 3884aa3df71SThierry Reding * error out if the user tries to display a framebuffer with such a 3894aa3df71SThierry Reding * configuration. 3904aa3df71SThierry Reding */ 391bcb0b461SVille Syrjälä if (state->fb->format->num_planes > 2) { 3924aa3df71SThierry Reding if (state->fb->pitches[2] != state->fb->pitches[1]) { 3934aa3df71SThierry Reding DRM_ERROR("unsupported UV-plane configuration\n"); 3944aa3df71SThierry Reding return -EINVAL; 3954aa3df71SThierry Reding } 3964aa3df71SThierry Reding } 3974aa3df71SThierry Reding 39847802b09SThierry Reding err = tegra_plane_state_add(tegra, state); 39947802b09SThierry Reding if (err < 0) 40047802b09SThierry Reding return err; 40147802b09SThierry Reding 4024aa3df71SThierry Reding return 0; 4034aa3df71SThierry Reding } 4044aa3df71SThierry Reding 405a4bfa096SThierry Reding static void tegra_plane_atomic_disable(struct drm_plane *plane, 406a4bfa096SThierry Reding struct drm_plane_state *old_state) 40780d3eef1SDmitry Osipenko { 408a4bfa096SThierry Reding struct tegra_dc *dc = to_tegra_dc(old_state->crtc); 409a4bfa096SThierry Reding struct tegra_plane *p = to_tegra_plane(plane); 41080d3eef1SDmitry Osipenko unsigned long flags; 41180d3eef1SDmitry Osipenko u32 value; 41280d3eef1SDmitry Osipenko 413a4bfa096SThierry Reding /* rien ne va plus */ 414a4bfa096SThierry Reding if (!old_state || !old_state->crtc) 415a4bfa096SThierry Reding return; 416a4bfa096SThierry Reding 41780d3eef1SDmitry Osipenko spin_lock_irqsave(&dc->lock, flags); 41880d3eef1SDmitry Osipenko 419a4bfa096SThierry Reding value = WINDOW_A_SELECT << p->index; 42080d3eef1SDmitry Osipenko tegra_dc_writel(dc, value, DC_CMD_DISPLAY_WINDOW_HEADER); 42180d3eef1SDmitry Osipenko 42280d3eef1SDmitry Osipenko value = tegra_dc_readl(dc, DC_WIN_WIN_OPTIONS); 42380d3eef1SDmitry Osipenko value &= ~WIN_ENABLE; 42480d3eef1SDmitry Osipenko tegra_dc_writel(dc, value, DC_WIN_WIN_OPTIONS); 42580d3eef1SDmitry Osipenko 42680d3eef1SDmitry Osipenko spin_unlock_irqrestore(&dc->lock, flags); 42780d3eef1SDmitry Osipenko } 42880d3eef1SDmitry Osipenko 4294aa3df71SThierry Reding static void tegra_plane_atomic_update(struct drm_plane *plane, 4304aa3df71SThierry Reding struct drm_plane_state *old_state) 4314aa3df71SThierry Reding { 4328f604f8cSThierry Reding struct tegra_plane_state *state = to_tegra_plane_state(plane->state); 4334aa3df71SThierry Reding struct tegra_dc *dc = to_tegra_dc(plane->state->crtc); 4344aa3df71SThierry Reding struct drm_framebuffer *fb = plane->state->fb; 4354aa3df71SThierry Reding struct tegra_plane *p = to_tegra_plane(plane); 4364aa3df71SThierry Reding struct tegra_dc_window window; 4374aa3df71SThierry Reding unsigned int i; 4384aa3df71SThierry Reding 4394aa3df71SThierry Reding /* rien ne va plus */ 4404aa3df71SThierry Reding if (!plane->state->crtc || !plane->state->fb) 4414aa3df71SThierry Reding return; 4424aa3df71SThierry Reding 44380d3eef1SDmitry Osipenko if (!plane->state->visible) 444a4bfa096SThierry Reding return tegra_plane_atomic_disable(plane, old_state); 44580d3eef1SDmitry Osipenko 446c7679306SThierry Reding memset(&window, 0, sizeof(window)); 4477d205857SDmitry Osipenko window.src.x = plane->state->src.x1 >> 16; 4487d205857SDmitry Osipenko window.src.y = plane->state->src.y1 >> 16; 4497d205857SDmitry Osipenko window.src.w = drm_rect_width(&plane->state->src) >> 16; 4507d205857SDmitry Osipenko window.src.h = drm_rect_height(&plane->state->src) >> 16; 4517d205857SDmitry Osipenko window.dst.x = plane->state->dst.x1; 4527d205857SDmitry Osipenko window.dst.y = plane->state->dst.y1; 4537d205857SDmitry Osipenko window.dst.w = drm_rect_width(&plane->state->dst); 4547d205857SDmitry Osipenko window.dst.h = drm_rect_height(&plane->state->dst); 455272725c7SVille Syrjälä window.bits_per_pixel = fb->format->cpp[0] * 8; 456c7679306SThierry Reding window.bottom_up = tegra_fb_is_bottom_up(fb); 457c7679306SThierry Reding 4588f604f8cSThierry Reding /* copy from state */ 4598f604f8cSThierry Reding window.tiling = state->tiling; 4608f604f8cSThierry Reding window.format = state->format; 4618f604f8cSThierry Reding window.swap = state->swap; 462c7679306SThierry Reding 463bcb0b461SVille Syrjälä for (i = 0; i < fb->format->num_planes; i++) { 4644aa3df71SThierry Reding struct tegra_bo *bo = tegra_fb_get_plane(fb, i); 465c7679306SThierry Reding 4664aa3df71SThierry Reding window.base[i] = bo->paddr + fb->offsets[i]; 46708ee0178SDmitry Osipenko 46808ee0178SDmitry Osipenko /* 46908ee0178SDmitry Osipenko * Tegra uses a shared stride for UV planes. Framebuffers are 47008ee0178SDmitry Osipenko * already checked for this in the tegra_plane_atomic_check() 47108ee0178SDmitry Osipenko * function, so it's safe to ignore the V-plane pitch here. 47208ee0178SDmitry Osipenko */ 47308ee0178SDmitry Osipenko if (i < 2) 4744aa3df71SThierry Reding window.stride[i] = fb->pitches[i]; 475c7679306SThierry Reding } 476c7679306SThierry Reding 4774aa3df71SThierry Reding tegra_dc_setup_window(dc, p->index, &window); 4784aa3df71SThierry Reding } 4794aa3df71SThierry Reding 480a4bfa096SThierry Reding static const struct drm_plane_helper_funcs tegra_plane_helper_funcs = { 4814aa3df71SThierry Reding .atomic_check = tegra_plane_atomic_check, 4824aa3df71SThierry Reding .atomic_disable = tegra_plane_atomic_disable, 483a4bfa096SThierry Reding .atomic_update = tegra_plane_atomic_update, 484c7679306SThierry Reding }; 485c7679306SThierry Reding 48647307954SThierry Reding static struct drm_plane *tegra_primary_plane_create(struct drm_device *drm, 487c7679306SThierry Reding struct tegra_dc *dc) 488c7679306SThierry Reding { 489518e6227SThierry Reding /* 490518e6227SThierry Reding * Ideally this would use drm_crtc_mask(), but that would require the 491518e6227SThierry Reding * CRTC to already be in the mode_config's list of CRTCs. However, it 492518e6227SThierry Reding * will only be added to that list in the drm_crtc_init_with_planes() 493518e6227SThierry Reding * (in tegra_dc_init()), which in turn requires registration of these 494518e6227SThierry Reding * planes. So we have ourselves a nice little chicken and egg problem 495518e6227SThierry Reding * here. 496518e6227SThierry Reding * 497518e6227SThierry Reding * We work around this by manually creating the mask from the number 498518e6227SThierry Reding * of CRTCs that have been registered, and should therefore always be 499518e6227SThierry Reding * the same as drm_crtc_index() after registration. 500518e6227SThierry Reding */ 501518e6227SThierry Reding unsigned long possible_crtcs = 1 << drm->mode_config.num_crtc; 50247307954SThierry Reding enum drm_plane_type type = DRM_PLANE_TYPE_PRIMARY; 503c7679306SThierry Reding struct tegra_plane *plane; 504c7679306SThierry Reding unsigned int num_formats; 505c7679306SThierry Reding const u32 *formats; 506c7679306SThierry Reding int err; 507c7679306SThierry Reding 508c7679306SThierry Reding plane = kzalloc(sizeof(*plane), GFP_KERNEL); 509c7679306SThierry Reding if (!plane) 510c7679306SThierry Reding return ERR_PTR(-ENOMEM); 511c7679306SThierry Reding 512*511c7023SThierry Reding num_formats = dc->soc->num_primary_formats; 513*511c7023SThierry Reding formats = dc->soc->primary_formats; 514c7679306SThierry Reding 515c4755fb9SThierry Reding /* 516c4755fb9SThierry Reding * XXX compute offset so that we can directly access windows. 517c4755fb9SThierry Reding * 518c4755fb9SThierry Reding * Always use window A as primary window. 519c4755fb9SThierry Reding */ 520c4755fb9SThierry Reding plane->offset = 0; 521c4755fb9SThierry Reding plane->index = 0; 522c4755fb9SThierry Reding plane->depth = 255; 523c4755fb9SThierry Reding 524518e6227SThierry Reding err = drm_universal_plane_init(drm, &plane->base, possible_crtcs, 525c1cb4b61SThierry Reding &tegra_plane_funcs, formats, 52647307954SThierry Reding num_formats, NULL, type, NULL); 527c7679306SThierry Reding if (err < 0) { 528c7679306SThierry Reding kfree(plane); 529c7679306SThierry Reding return ERR_PTR(err); 530c7679306SThierry Reding } 531c7679306SThierry Reding 532a4bfa096SThierry Reding drm_plane_helper_add(&plane->base, &tegra_plane_helper_funcs); 5334aa3df71SThierry Reding 534c7679306SThierry Reding return &plane->base; 535c7679306SThierry Reding } 536c7679306SThierry Reding 537c7679306SThierry Reding static const u32 tegra_cursor_plane_formats[] = { 538c7679306SThierry Reding DRM_FORMAT_RGBA8888, 539c7679306SThierry Reding }; 540c7679306SThierry Reding 5414aa3df71SThierry Reding static int tegra_cursor_atomic_check(struct drm_plane *plane, 5424aa3df71SThierry Reding struct drm_plane_state *state) 543c7679306SThierry Reding { 54447802b09SThierry Reding struct tegra_plane *tegra = to_tegra_plane(plane); 54547802b09SThierry Reding int err; 54647802b09SThierry Reding 5474aa3df71SThierry Reding /* no need for further checks if the plane is being disabled */ 5484aa3df71SThierry Reding if (!state->crtc) 5494aa3df71SThierry Reding return 0; 550c7679306SThierry Reding 551c7679306SThierry Reding /* scaling not supported for cursor */ 5524aa3df71SThierry Reding if ((state->src_w >> 16 != state->crtc_w) || 5534aa3df71SThierry Reding (state->src_h >> 16 != state->crtc_h)) 554c7679306SThierry Reding return -EINVAL; 555c7679306SThierry Reding 556c7679306SThierry Reding /* only square cursors supported */ 5574aa3df71SThierry Reding if (state->src_w != state->src_h) 558c7679306SThierry Reding return -EINVAL; 559c7679306SThierry Reding 5604aa3df71SThierry Reding if (state->crtc_w != 32 && state->crtc_w != 64 && 5614aa3df71SThierry Reding state->crtc_w != 128 && state->crtc_w != 256) 5624aa3df71SThierry Reding return -EINVAL; 5634aa3df71SThierry Reding 56447802b09SThierry Reding err = tegra_plane_state_add(tegra, state); 56547802b09SThierry Reding if (err < 0) 56647802b09SThierry Reding return err; 56747802b09SThierry Reding 5684aa3df71SThierry Reding return 0; 5694aa3df71SThierry Reding } 5704aa3df71SThierry Reding 5714aa3df71SThierry Reding static void tegra_cursor_atomic_update(struct drm_plane *plane, 5724aa3df71SThierry Reding struct drm_plane_state *old_state) 5734aa3df71SThierry Reding { 5744aa3df71SThierry Reding struct tegra_bo *bo = tegra_fb_get_plane(plane->state->fb, 0); 5754aa3df71SThierry Reding struct tegra_dc *dc = to_tegra_dc(plane->state->crtc); 5764aa3df71SThierry Reding struct drm_plane_state *state = plane->state; 5774aa3df71SThierry Reding u32 value = CURSOR_CLIP_DISPLAY; 5784aa3df71SThierry Reding 5794aa3df71SThierry Reding /* rien ne va plus */ 5804aa3df71SThierry Reding if (!plane->state->crtc || !plane->state->fb) 5814aa3df71SThierry Reding return; 5824aa3df71SThierry Reding 5834aa3df71SThierry Reding switch (state->crtc_w) { 584c7679306SThierry Reding case 32: 585c7679306SThierry Reding value |= CURSOR_SIZE_32x32; 586c7679306SThierry Reding break; 587c7679306SThierry Reding 588c7679306SThierry Reding case 64: 589c7679306SThierry Reding value |= CURSOR_SIZE_64x64; 590c7679306SThierry Reding break; 591c7679306SThierry Reding 592c7679306SThierry Reding case 128: 593c7679306SThierry Reding value |= CURSOR_SIZE_128x128; 594c7679306SThierry Reding break; 595c7679306SThierry Reding 596c7679306SThierry Reding case 256: 597c7679306SThierry Reding value |= CURSOR_SIZE_256x256; 598c7679306SThierry Reding break; 599c7679306SThierry Reding 600c7679306SThierry Reding default: 6014aa3df71SThierry Reding WARN(1, "cursor size %ux%u not supported\n", state->crtc_w, 6024aa3df71SThierry Reding state->crtc_h); 6034aa3df71SThierry Reding return; 604c7679306SThierry Reding } 605c7679306SThierry Reding 606c7679306SThierry Reding value |= (bo->paddr >> 10) & 0x3fffff; 607c7679306SThierry Reding tegra_dc_writel(dc, value, DC_DISP_CURSOR_START_ADDR); 608c7679306SThierry Reding 609c7679306SThierry Reding #ifdef CONFIG_ARCH_DMA_ADDR_T_64BIT 610c7679306SThierry Reding value = (bo->paddr >> 32) & 0x3; 611c7679306SThierry Reding tegra_dc_writel(dc, value, DC_DISP_CURSOR_START_ADDR_HI); 612c7679306SThierry Reding #endif 613c7679306SThierry Reding 614c7679306SThierry Reding /* enable cursor and set blend mode */ 615c7679306SThierry Reding value = tegra_dc_readl(dc, DC_DISP_DISP_WIN_OPTIONS); 616c7679306SThierry Reding value |= CURSOR_ENABLE; 617c7679306SThierry Reding tegra_dc_writel(dc, value, DC_DISP_DISP_WIN_OPTIONS); 618c7679306SThierry Reding 619c7679306SThierry Reding value = tegra_dc_readl(dc, DC_DISP_BLEND_CURSOR_CONTROL); 620c7679306SThierry Reding value &= ~CURSOR_DST_BLEND_MASK; 621c7679306SThierry Reding value &= ~CURSOR_SRC_BLEND_MASK; 622c7679306SThierry Reding value |= CURSOR_MODE_NORMAL; 623c7679306SThierry Reding value |= CURSOR_DST_BLEND_NEG_K1_TIMES_SRC; 624c7679306SThierry Reding value |= CURSOR_SRC_BLEND_K1_TIMES_SRC; 625c7679306SThierry Reding value |= CURSOR_ALPHA; 626c7679306SThierry Reding tegra_dc_writel(dc, value, DC_DISP_BLEND_CURSOR_CONTROL); 627c7679306SThierry Reding 628c7679306SThierry Reding /* position the cursor */ 6294aa3df71SThierry Reding value = (state->crtc_y & 0x3fff) << 16 | (state->crtc_x & 0x3fff); 630c7679306SThierry Reding tegra_dc_writel(dc, value, DC_DISP_CURSOR_POSITION); 631c7679306SThierry Reding } 632c7679306SThierry Reding 6334aa3df71SThierry Reding static void tegra_cursor_atomic_disable(struct drm_plane *plane, 6344aa3df71SThierry Reding struct drm_plane_state *old_state) 635c7679306SThierry Reding { 6364aa3df71SThierry Reding struct tegra_dc *dc; 637c7679306SThierry Reding u32 value; 638c7679306SThierry Reding 6394aa3df71SThierry Reding /* rien ne va plus */ 6404aa3df71SThierry Reding if (!old_state || !old_state->crtc) 6414aa3df71SThierry Reding return; 6424aa3df71SThierry Reding 6434aa3df71SThierry Reding dc = to_tegra_dc(old_state->crtc); 644c7679306SThierry Reding 645c7679306SThierry Reding value = tegra_dc_readl(dc, DC_DISP_DISP_WIN_OPTIONS); 646c7679306SThierry Reding value &= ~CURSOR_ENABLE; 647c7679306SThierry Reding tegra_dc_writel(dc, value, DC_DISP_DISP_WIN_OPTIONS); 648c7679306SThierry Reding } 649c7679306SThierry Reding 6504aa3df71SThierry Reding static const struct drm_plane_helper_funcs tegra_cursor_plane_helper_funcs = { 6514aa3df71SThierry Reding .atomic_check = tegra_cursor_atomic_check, 6524aa3df71SThierry Reding .atomic_update = tegra_cursor_atomic_update, 6534aa3df71SThierry Reding .atomic_disable = tegra_cursor_atomic_disable, 654c7679306SThierry Reding }; 655c7679306SThierry Reding 656c7679306SThierry Reding static struct drm_plane *tegra_dc_cursor_plane_create(struct drm_device *drm, 657c7679306SThierry Reding struct tegra_dc *dc) 658c7679306SThierry Reding { 659c7679306SThierry Reding struct tegra_plane *plane; 660c7679306SThierry Reding unsigned int num_formats; 661c7679306SThierry Reding const u32 *formats; 662c7679306SThierry Reding int err; 663c7679306SThierry Reding 664c7679306SThierry Reding plane = kzalloc(sizeof(*plane), GFP_KERNEL); 665c7679306SThierry Reding if (!plane) 666c7679306SThierry Reding return ERR_PTR(-ENOMEM); 667c7679306SThierry Reding 66847802b09SThierry Reding /* 669a1df3b24SThierry Reding * This index is kind of fake. The cursor isn't a regular plane, but 670a1df3b24SThierry Reding * its update and activation request bits in DC_CMD_STATE_CONTROL do 671a1df3b24SThierry Reding * use the same programming. Setting this fake index here allows the 672a1df3b24SThierry Reding * code in tegra_add_plane_state() to do the right thing without the 673a1df3b24SThierry Reding * need to special-casing the cursor plane. 67447802b09SThierry Reding */ 67547802b09SThierry Reding plane->index = 6; 67647802b09SThierry Reding 677c7679306SThierry Reding num_formats = ARRAY_SIZE(tegra_cursor_plane_formats); 678c7679306SThierry Reding formats = tegra_cursor_plane_formats; 679c7679306SThierry Reding 680c7679306SThierry Reding err = drm_universal_plane_init(drm, &plane->base, 1 << dc->pipe, 681c1cb4b61SThierry Reding &tegra_plane_funcs, formats, 682e6fc3b68SBen Widawsky num_formats, NULL, 683e6fc3b68SBen Widawsky DRM_PLANE_TYPE_CURSOR, NULL); 684c7679306SThierry Reding if (err < 0) { 685c7679306SThierry Reding kfree(plane); 686c7679306SThierry Reding return ERR_PTR(err); 687c7679306SThierry Reding } 688c7679306SThierry Reding 6894aa3df71SThierry Reding drm_plane_helper_add(&plane->base, &tegra_cursor_plane_helper_funcs); 6904aa3df71SThierry Reding 691c7679306SThierry Reding return &plane->base; 692c7679306SThierry Reding } 693c7679306SThierry Reding 694*511c7023SThierry Reding static const u32 tegra20_overlay_formats[] = { 695*511c7023SThierry Reding DRM_FORMAT_ARGB4444, 696*511c7023SThierry Reding DRM_FORMAT_ARGB1555, 697dee8268fSThierry Reding DRM_FORMAT_RGB565, 698*511c7023SThierry Reding DRM_FORMAT_RGBA5551, 699*511c7023SThierry Reding DRM_FORMAT_ABGR8888, 700*511c7023SThierry Reding DRM_FORMAT_ARGB8888, 701*511c7023SThierry Reding /* planar formats */ 702*511c7023SThierry Reding DRM_FORMAT_UYVY, 703*511c7023SThierry Reding DRM_FORMAT_YUYV, 704*511c7023SThierry Reding DRM_FORMAT_YUV420, 705*511c7023SThierry Reding DRM_FORMAT_YUV422, 706*511c7023SThierry Reding }; 707*511c7023SThierry Reding 708*511c7023SThierry Reding static const u32 tegra114_overlay_formats[] = { 709*511c7023SThierry Reding DRM_FORMAT_ARGB4444, 710*511c7023SThierry Reding DRM_FORMAT_ARGB1555, 711*511c7023SThierry Reding DRM_FORMAT_RGB565, 712*511c7023SThierry Reding DRM_FORMAT_RGBA5551, 713*511c7023SThierry Reding DRM_FORMAT_ABGR8888, 714*511c7023SThierry Reding DRM_FORMAT_ARGB8888, 715*511c7023SThierry Reding /* new on Tegra114 */ 716*511c7023SThierry Reding DRM_FORMAT_ABGR4444, 717*511c7023SThierry Reding DRM_FORMAT_ABGR1555, 718*511c7023SThierry Reding DRM_FORMAT_BGRA5551, 719*511c7023SThierry Reding DRM_FORMAT_XRGB1555, 720*511c7023SThierry Reding DRM_FORMAT_RGBX5551, 721*511c7023SThierry Reding DRM_FORMAT_XBGR1555, 722*511c7023SThierry Reding DRM_FORMAT_BGRX5551, 723*511c7023SThierry Reding DRM_FORMAT_BGR565, 724*511c7023SThierry Reding DRM_FORMAT_BGRA8888, 725*511c7023SThierry Reding DRM_FORMAT_RGBA8888, 726*511c7023SThierry Reding DRM_FORMAT_XRGB8888, 727*511c7023SThierry Reding DRM_FORMAT_XBGR8888, 728*511c7023SThierry Reding /* planar formats */ 729*511c7023SThierry Reding DRM_FORMAT_UYVY, 730*511c7023SThierry Reding DRM_FORMAT_YUYV, 731*511c7023SThierry Reding DRM_FORMAT_YUV420, 732*511c7023SThierry Reding DRM_FORMAT_YUV422, 733*511c7023SThierry Reding }; 734*511c7023SThierry Reding 735*511c7023SThierry Reding static const u32 tegra124_overlay_formats[] = { 736*511c7023SThierry Reding DRM_FORMAT_ARGB4444, 737*511c7023SThierry Reding DRM_FORMAT_ARGB1555, 738*511c7023SThierry Reding DRM_FORMAT_RGB565, 739*511c7023SThierry Reding DRM_FORMAT_RGBA5551, 740*511c7023SThierry Reding DRM_FORMAT_ABGR8888, 741*511c7023SThierry Reding DRM_FORMAT_ARGB8888, 742*511c7023SThierry Reding /* new on Tegra114 */ 743*511c7023SThierry Reding DRM_FORMAT_ABGR4444, 744*511c7023SThierry Reding DRM_FORMAT_ABGR1555, 745*511c7023SThierry Reding DRM_FORMAT_BGRA5551, 746*511c7023SThierry Reding DRM_FORMAT_XRGB1555, 747*511c7023SThierry Reding DRM_FORMAT_RGBX5551, 748*511c7023SThierry Reding DRM_FORMAT_XBGR1555, 749*511c7023SThierry Reding DRM_FORMAT_BGRX5551, 750*511c7023SThierry Reding DRM_FORMAT_BGR565, 751*511c7023SThierry Reding DRM_FORMAT_BGRA8888, 752*511c7023SThierry Reding DRM_FORMAT_RGBA8888, 753*511c7023SThierry Reding DRM_FORMAT_XRGB8888, 754*511c7023SThierry Reding DRM_FORMAT_XBGR8888, 755*511c7023SThierry Reding /* new on Tegra124 */ 756*511c7023SThierry Reding DRM_FORMAT_RGBX8888, 757*511c7023SThierry Reding DRM_FORMAT_BGRX8888, 758*511c7023SThierry Reding /* planar formats */ 759dee8268fSThierry Reding DRM_FORMAT_UYVY, 760f925390eSThierry Reding DRM_FORMAT_YUYV, 761dee8268fSThierry Reding DRM_FORMAT_YUV420, 762dee8268fSThierry Reding DRM_FORMAT_YUV422, 763dee8268fSThierry Reding }; 764dee8268fSThierry Reding 765c7679306SThierry Reding static struct drm_plane *tegra_dc_overlay_plane_create(struct drm_device *drm, 766c7679306SThierry Reding struct tegra_dc *dc, 767c7679306SThierry Reding unsigned int index) 768dee8268fSThierry Reding { 769dee8268fSThierry Reding struct tegra_plane *plane; 770c7679306SThierry Reding unsigned int num_formats; 771c7679306SThierry Reding const u32 *formats; 772c7679306SThierry Reding int err; 773dee8268fSThierry Reding 774f002abc1SThierry Reding plane = kzalloc(sizeof(*plane), GFP_KERNEL); 775dee8268fSThierry Reding if (!plane) 776c7679306SThierry Reding return ERR_PTR(-ENOMEM); 777dee8268fSThierry Reding 778c4755fb9SThierry Reding /* XXX compute offset so that we can directly access windows */ 779c4755fb9SThierry Reding plane->offset = 0; 780c7679306SThierry Reding plane->index = index; 781c4755fb9SThierry Reding plane->depth = 0; 782dee8268fSThierry Reding 783*511c7023SThierry Reding num_formats = dc->soc->num_overlay_formats; 784*511c7023SThierry Reding formats = dc->soc->overlay_formats; 785c7679306SThierry Reding 786c7679306SThierry Reding err = drm_universal_plane_init(drm, &plane->base, 1 << dc->pipe, 787301e0ddbSThierry Reding &tegra_plane_funcs, formats, 788e6fc3b68SBen Widawsky num_formats, NULL, 789e6fc3b68SBen Widawsky DRM_PLANE_TYPE_OVERLAY, NULL); 790f002abc1SThierry Reding if (err < 0) { 791f002abc1SThierry Reding kfree(plane); 792c7679306SThierry Reding return ERR_PTR(err); 793dee8268fSThierry Reding } 794c7679306SThierry Reding 795a4bfa096SThierry Reding drm_plane_helper_add(&plane->base, &tegra_plane_helper_funcs); 7964aa3df71SThierry Reding 797c7679306SThierry Reding return &plane->base; 798c7679306SThierry Reding } 799c7679306SThierry Reding 80047307954SThierry Reding static struct drm_plane *tegra_dc_add_shared_planes(struct drm_device *drm, 80147307954SThierry Reding struct tegra_dc *dc) 802c7679306SThierry Reding { 80347307954SThierry Reding struct drm_plane *plane, *primary = NULL; 80447307954SThierry Reding unsigned int i, j; 80547307954SThierry Reding 80647307954SThierry Reding for (i = 0; i < dc->soc->num_wgrps; i++) { 80747307954SThierry Reding const struct tegra_windowgroup_soc *wgrp = &dc->soc->wgrps[i]; 80847307954SThierry Reding 80947307954SThierry Reding if (wgrp->dc == dc->pipe) { 81047307954SThierry Reding for (j = 0; j < wgrp->num_windows; j++) { 81147307954SThierry Reding unsigned int index = wgrp->windows[j]; 81247307954SThierry Reding 81347307954SThierry Reding plane = tegra_shared_plane_create(drm, dc, 81447307954SThierry Reding wgrp->index, 81547307954SThierry Reding index); 81647307954SThierry Reding if (IS_ERR(plane)) 81747307954SThierry Reding return plane; 81847307954SThierry Reding 81947307954SThierry Reding /* 82047307954SThierry Reding * Choose the first shared plane owned by this 82147307954SThierry Reding * head as the primary plane. 82247307954SThierry Reding */ 82347307954SThierry Reding if (!primary) { 82447307954SThierry Reding plane->type = DRM_PLANE_TYPE_PRIMARY; 82547307954SThierry Reding primary = plane; 82647307954SThierry Reding } 82747307954SThierry Reding } 82847307954SThierry Reding } 82947307954SThierry Reding } 83047307954SThierry Reding 83147307954SThierry Reding return primary; 83247307954SThierry Reding } 83347307954SThierry Reding 83447307954SThierry Reding static struct drm_plane *tegra_dc_add_planes(struct drm_device *drm, 83547307954SThierry Reding struct tegra_dc *dc) 83647307954SThierry Reding { 83747307954SThierry Reding struct drm_plane *plane, *primary; 838c7679306SThierry Reding unsigned int i; 839c7679306SThierry Reding 84047307954SThierry Reding primary = tegra_primary_plane_create(drm, dc); 84147307954SThierry Reding if (IS_ERR(primary)) 84247307954SThierry Reding return primary; 84347307954SThierry Reding 844c7679306SThierry Reding for (i = 0; i < 2; i++) { 845c7679306SThierry Reding plane = tegra_dc_overlay_plane_create(drm, dc, 1 + i); 84647307954SThierry Reding if (IS_ERR(plane)) { 84747307954SThierry Reding /* XXX tegra_plane_destroy() */ 84847307954SThierry Reding drm_plane_cleanup(primary); 84947307954SThierry Reding kfree(primary); 85047307954SThierry Reding return plane; 85147307954SThierry Reding } 852f002abc1SThierry Reding } 853dee8268fSThierry Reding 85447307954SThierry Reding return primary; 855dee8268fSThierry Reding } 856dee8268fSThierry Reding 857f002abc1SThierry Reding static void tegra_dc_destroy(struct drm_crtc *crtc) 858f002abc1SThierry Reding { 859f002abc1SThierry Reding drm_crtc_cleanup(crtc); 860f002abc1SThierry Reding } 861f002abc1SThierry Reding 862ca915b10SThierry Reding static void tegra_crtc_reset(struct drm_crtc *crtc) 863ca915b10SThierry Reding { 864ca915b10SThierry Reding struct tegra_dc_state *state; 865ca915b10SThierry Reding 8663b59b7acSThierry Reding if (crtc->state) 867ec2dc6a0SDaniel Vetter __drm_atomic_helper_crtc_destroy_state(crtc->state); 8683b59b7acSThierry Reding 869ca915b10SThierry Reding kfree(crtc->state); 870ca915b10SThierry Reding crtc->state = NULL; 871ca915b10SThierry Reding 872ca915b10SThierry Reding state = kzalloc(sizeof(*state), GFP_KERNEL); 873332bbe70SThierry Reding if (state) { 874ca915b10SThierry Reding crtc->state = &state->base; 875332bbe70SThierry Reding crtc->state->crtc = crtc; 876332bbe70SThierry Reding } 87731930d4dSThierry Reding 87831930d4dSThierry Reding drm_crtc_vblank_reset(crtc); 879ca915b10SThierry Reding } 880ca915b10SThierry Reding 881ca915b10SThierry Reding static struct drm_crtc_state * 882ca915b10SThierry Reding tegra_crtc_atomic_duplicate_state(struct drm_crtc *crtc) 883ca915b10SThierry Reding { 884ca915b10SThierry Reding struct tegra_dc_state *state = to_dc_state(crtc->state); 885ca915b10SThierry Reding struct tegra_dc_state *copy; 886ca915b10SThierry Reding 8873b59b7acSThierry Reding copy = kmalloc(sizeof(*copy), GFP_KERNEL); 888ca915b10SThierry Reding if (!copy) 889ca915b10SThierry Reding return NULL; 890ca915b10SThierry Reding 8913b59b7acSThierry Reding __drm_atomic_helper_crtc_duplicate_state(crtc, ©->base); 8923b59b7acSThierry Reding copy->clk = state->clk; 8933b59b7acSThierry Reding copy->pclk = state->pclk; 8943b59b7acSThierry Reding copy->div = state->div; 8953b59b7acSThierry Reding copy->planes = state->planes; 896ca915b10SThierry Reding 897ca915b10SThierry Reding return ©->base; 898ca915b10SThierry Reding } 899ca915b10SThierry Reding 900ca915b10SThierry Reding static void tegra_crtc_atomic_destroy_state(struct drm_crtc *crtc, 901ca915b10SThierry Reding struct drm_crtc_state *state) 902ca915b10SThierry Reding { 903ec2dc6a0SDaniel Vetter __drm_atomic_helper_crtc_destroy_state(state); 904ca915b10SThierry Reding kfree(state); 905ca915b10SThierry Reding } 906ca915b10SThierry Reding 907b95800eeSThierry Reding #define DEBUGFS_REG32(_name) { .name = #_name, .offset = _name } 908b95800eeSThierry Reding 909b95800eeSThierry Reding static const struct debugfs_reg32 tegra_dc_regs[] = { 910b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_GENERAL_INCR_SYNCPT), 911b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_GENERAL_INCR_SYNCPT_CNTRL), 912b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_GENERAL_INCR_SYNCPT_ERROR), 913b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_WIN_A_INCR_SYNCPT), 914b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_WIN_A_INCR_SYNCPT_CNTRL), 915b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_WIN_A_INCR_SYNCPT_ERROR), 916b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_WIN_B_INCR_SYNCPT), 917b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_WIN_B_INCR_SYNCPT_CNTRL), 918b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_WIN_B_INCR_SYNCPT_ERROR), 919b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_WIN_C_INCR_SYNCPT), 920b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_WIN_C_INCR_SYNCPT_CNTRL), 921b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_WIN_C_INCR_SYNCPT_ERROR), 922b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_CONT_SYNCPT_VSYNC), 923b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_DISPLAY_COMMAND_OPTION0), 924b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_DISPLAY_COMMAND), 925b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_SIGNAL_RAISE), 926b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_DISPLAY_POWER_CONTROL), 927b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_INT_STATUS), 928b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_INT_MASK), 929b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_INT_ENABLE), 930b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_INT_TYPE), 931b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_INT_POLARITY), 932b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_SIGNAL_RAISE1), 933b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_SIGNAL_RAISE2), 934b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_SIGNAL_RAISE3), 935b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_STATE_ACCESS), 936b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_STATE_CONTROL), 937b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_DISPLAY_WINDOW_HEADER), 938b95800eeSThierry Reding DEBUGFS_REG32(DC_CMD_REG_ACT_CONTROL), 939b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_CRC_CONTROL), 940b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_CRC_CHECKSUM), 941b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_OUTPUT_ENABLE(0)), 942b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_OUTPUT_ENABLE(1)), 943b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_OUTPUT_ENABLE(2)), 944b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_OUTPUT_ENABLE(3)), 945b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_OUTPUT_POLARITY(0)), 946b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_OUTPUT_POLARITY(1)), 947b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_OUTPUT_POLARITY(2)), 948b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_OUTPUT_POLARITY(3)), 949b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_OUTPUT_DATA(0)), 950b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_OUTPUT_DATA(1)), 951b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_OUTPUT_DATA(2)), 952b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_OUTPUT_DATA(3)), 953b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_INPUT_ENABLE(0)), 954b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_INPUT_ENABLE(1)), 955b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_INPUT_ENABLE(2)), 956b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_INPUT_ENABLE(3)), 957b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_INPUT_DATA(0)), 958b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_INPUT_DATA(1)), 959b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_OUTPUT_SELECT(0)), 960b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_OUTPUT_SELECT(1)), 961b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_OUTPUT_SELECT(2)), 962b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_OUTPUT_SELECT(3)), 963b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_OUTPUT_SELECT(4)), 964b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_OUTPUT_SELECT(5)), 965b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_OUTPUT_SELECT(6)), 966b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_MISC_CONTROL), 967b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_PM0_CONTROL), 968b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_PM0_DUTY_CYCLE), 969b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_PM1_CONTROL), 970b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_PIN_PM1_DUTY_CYCLE), 971b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_SPI_CONTROL), 972b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_SPI_START_BYTE), 973b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_HSPI_WRITE_DATA_AB), 974b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_HSPI_WRITE_DATA_CD), 975b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_HSPI_CS_DC), 976b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_SCRATCH_REGISTER_A), 977b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_SCRATCH_REGISTER_B), 978b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_GPIO_CTRL), 979b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_GPIO_DEBOUNCE_COUNTER), 980b95800eeSThierry Reding DEBUGFS_REG32(DC_COM_CRC_CHECKSUM_LATCHED), 981b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_DISP_SIGNAL_OPTIONS0), 982b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_DISP_SIGNAL_OPTIONS1), 983b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_DISP_WIN_OPTIONS), 984b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_DISP_MEM_HIGH_PRIORITY), 985b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_DISP_MEM_HIGH_PRIORITY_TIMER), 986b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_DISP_TIMING_OPTIONS), 987b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_REF_TO_SYNC), 988b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SYNC_WIDTH), 989b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_BACK_PORCH), 990b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_ACTIVE), 991b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_FRONT_PORCH), 992b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_H_PULSE0_CONTROL), 993b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_H_PULSE0_POSITION_A), 994b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_H_PULSE0_POSITION_B), 995b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_H_PULSE0_POSITION_C), 996b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_H_PULSE0_POSITION_D), 997b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_H_PULSE1_CONTROL), 998b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_H_PULSE1_POSITION_A), 999b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_H_PULSE1_POSITION_B), 1000b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_H_PULSE1_POSITION_C), 1001b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_H_PULSE1_POSITION_D), 1002b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_H_PULSE2_CONTROL), 1003b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_H_PULSE2_POSITION_A), 1004b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_H_PULSE2_POSITION_B), 1005b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_H_PULSE2_POSITION_C), 1006b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_H_PULSE2_POSITION_D), 1007b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_V_PULSE0_CONTROL), 1008b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_V_PULSE0_POSITION_A), 1009b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_V_PULSE0_POSITION_B), 1010b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_V_PULSE0_POSITION_C), 1011b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_V_PULSE1_CONTROL), 1012b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_V_PULSE1_POSITION_A), 1013b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_V_PULSE1_POSITION_B), 1014b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_V_PULSE1_POSITION_C), 1015b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_V_PULSE2_CONTROL), 1016b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_V_PULSE2_POSITION_A), 1017b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_V_PULSE3_CONTROL), 1018b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_V_PULSE3_POSITION_A), 1019b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_M0_CONTROL), 1020b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_M1_CONTROL), 1021b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_DI_CONTROL), 1022b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_PP_CONTROL), 1023b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_PP_SELECT_A), 1024b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_PP_SELECT_B), 1025b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_PP_SELECT_C), 1026b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_PP_SELECT_D), 1027b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_DISP_CLOCK_CONTROL), 1028b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_DISP_INTERFACE_CONTROL), 1029b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_DISP_COLOR_CONTROL), 1030b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SHIFT_CLOCK_OPTIONS), 1031b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_DATA_ENABLE_OPTIONS), 1032b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SERIAL_INTERFACE_OPTIONS), 1033b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_LCD_SPI_OPTIONS), 1034b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_BORDER_COLOR), 1035b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_COLOR_KEY0_LOWER), 1036b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_COLOR_KEY0_UPPER), 1037b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_COLOR_KEY1_LOWER), 1038b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_COLOR_KEY1_UPPER), 1039b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_CURSOR_FOREGROUND), 1040b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_CURSOR_BACKGROUND), 1041b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_CURSOR_START_ADDR), 1042b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_CURSOR_START_ADDR_NS), 1043b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_CURSOR_POSITION), 1044b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_CURSOR_POSITION_NS), 1045b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_INIT_SEQ_CONTROL), 1046b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SPI_INIT_SEQ_DATA_A), 1047b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SPI_INIT_SEQ_DATA_B), 1048b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SPI_INIT_SEQ_DATA_C), 1049b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SPI_INIT_SEQ_DATA_D), 1050b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_DC_MCCIF_FIFOCTRL), 1051b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_MCCIF_DISPLAY0A_HYST), 1052b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_MCCIF_DISPLAY0B_HYST), 1053b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_MCCIF_DISPLAY1A_HYST), 1054b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_MCCIF_DISPLAY1B_HYST), 1055b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_DAC_CRT_CTRL), 1056b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_DISP_MISC_CONTROL), 1057b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_CONTROL), 1058b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_CSC_COEFF), 1059b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_LUT(0)), 1060b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_LUT(1)), 1061b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_LUT(2)), 1062b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_LUT(3)), 1063b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_LUT(4)), 1064b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_LUT(5)), 1065b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_LUT(6)), 1066b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_LUT(7)), 1067b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_LUT(8)), 1068b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_FLICKER_CONTROL), 1069b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_DC_PIXEL_COUNT), 1070b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_HISTOGRAM(0)), 1071b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_HISTOGRAM(1)), 1072b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_HISTOGRAM(2)), 1073b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_HISTOGRAM(3)), 1074b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_HISTOGRAM(4)), 1075b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_HISTOGRAM(5)), 1076b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_HISTOGRAM(6)), 1077b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_HISTOGRAM(7)), 1078b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_BL_TF(0)), 1079b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_BL_TF(1)), 1080b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_BL_TF(2)), 1081b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_BL_TF(3)), 1082b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_BL_CONTROL), 1083b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_HW_K_VALUES), 1084b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_SD_MAN_K_VALUES), 1085b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_CURSOR_START_ADDR_HI), 1086b95800eeSThierry Reding DEBUGFS_REG32(DC_DISP_BLEND_CURSOR_CONTROL), 1087b95800eeSThierry Reding DEBUGFS_REG32(DC_WIN_WIN_OPTIONS), 1088b95800eeSThierry Reding DEBUGFS_REG32(DC_WIN_BYTE_SWAP), 1089b95800eeSThierry Reding DEBUGFS_REG32(DC_WIN_BUFFER_CONTROL), 1090b95800eeSThierry Reding DEBUGFS_REG32(DC_WIN_COLOR_DEPTH), 1091b95800eeSThierry Reding DEBUGFS_REG32(DC_WIN_POSITION), 1092b95800eeSThierry Reding DEBUGFS_REG32(DC_WIN_SIZE), 1093b95800eeSThierry Reding DEBUGFS_REG32(DC_WIN_PRESCALED_SIZE), 1094b95800eeSThierry Reding DEBUGFS_REG32(DC_WIN_H_INITIAL_DDA), 1095b95800eeSThierry Reding DEBUGFS_REG32(DC_WIN_V_INITIAL_DDA), 1096b95800eeSThierry Reding DEBUGFS_REG32(DC_WIN_DDA_INC), 1097b95800eeSThierry Reding DEBUGFS_REG32(DC_WIN_LINE_STRIDE), 1098b95800eeSThierry Reding DEBUGFS_REG32(DC_WIN_BUF_STRIDE), 1099b95800eeSThierry Reding DEBUGFS_REG32(DC_WIN_UV_BUF_STRIDE), 1100b95800eeSThierry Reding DEBUGFS_REG32(DC_WIN_BUFFER_ADDR_MODE), 1101b95800eeSThierry Reding DEBUGFS_REG32(DC_WIN_DV_CONTROL), 1102b95800eeSThierry Reding DEBUGFS_REG32(DC_WIN_BLEND_NOKEY), 1103b95800eeSThierry Reding DEBUGFS_REG32(DC_WIN_BLEND_1WIN), 1104b95800eeSThierry Reding DEBUGFS_REG32(DC_WIN_BLEND_2WIN_X), 1105b95800eeSThierry Reding DEBUGFS_REG32(DC_WIN_BLEND_2WIN_Y), 1106b95800eeSThierry Reding DEBUGFS_REG32(DC_WIN_BLEND_3WIN_XY), 1107b95800eeSThierry Reding DEBUGFS_REG32(DC_WIN_HP_FETCH_CONTROL), 1108b95800eeSThierry Reding DEBUGFS_REG32(DC_WINBUF_START_ADDR), 1109b95800eeSThierry Reding DEBUGFS_REG32(DC_WINBUF_START_ADDR_NS), 1110b95800eeSThierry Reding DEBUGFS_REG32(DC_WINBUF_START_ADDR_U), 1111b95800eeSThierry Reding DEBUGFS_REG32(DC_WINBUF_START_ADDR_U_NS), 1112b95800eeSThierry Reding DEBUGFS_REG32(DC_WINBUF_START_ADDR_V), 1113b95800eeSThierry Reding DEBUGFS_REG32(DC_WINBUF_START_ADDR_V_NS), 1114b95800eeSThierry Reding DEBUGFS_REG32(DC_WINBUF_ADDR_H_OFFSET), 1115b95800eeSThierry Reding DEBUGFS_REG32(DC_WINBUF_ADDR_H_OFFSET_NS), 1116b95800eeSThierry Reding DEBUGFS_REG32(DC_WINBUF_ADDR_V_OFFSET), 1117b95800eeSThierry Reding DEBUGFS_REG32(DC_WINBUF_ADDR_V_OFFSET_NS), 1118b95800eeSThierry Reding DEBUGFS_REG32(DC_WINBUF_UFLOW_STATUS), 1119b95800eeSThierry Reding DEBUGFS_REG32(DC_WINBUF_AD_UFLOW_STATUS), 1120b95800eeSThierry Reding DEBUGFS_REG32(DC_WINBUF_BD_UFLOW_STATUS), 1121b95800eeSThierry Reding DEBUGFS_REG32(DC_WINBUF_CD_UFLOW_STATUS), 1122b95800eeSThierry Reding }; 1123b95800eeSThierry Reding 1124b95800eeSThierry Reding static int tegra_dc_show_regs(struct seq_file *s, void *data) 1125b95800eeSThierry Reding { 1126b95800eeSThierry Reding struct drm_info_node *node = s->private; 1127b95800eeSThierry Reding struct tegra_dc *dc = node->info_ent->data; 1128b95800eeSThierry Reding unsigned int i; 1129b95800eeSThierry Reding int err = 0; 1130b95800eeSThierry Reding 1131b95800eeSThierry Reding drm_modeset_lock(&dc->base.mutex, NULL); 1132b95800eeSThierry Reding 1133b95800eeSThierry Reding if (!dc->base.state->active) { 1134b95800eeSThierry Reding err = -EBUSY; 1135b95800eeSThierry Reding goto unlock; 1136b95800eeSThierry Reding } 1137b95800eeSThierry Reding 1138b95800eeSThierry Reding for (i = 0; i < ARRAY_SIZE(tegra_dc_regs); i++) { 1139b95800eeSThierry Reding unsigned int offset = tegra_dc_regs[i].offset; 1140b95800eeSThierry Reding 1141b95800eeSThierry Reding seq_printf(s, "%-40s %#05x %08x\n", tegra_dc_regs[i].name, 1142b95800eeSThierry Reding offset, tegra_dc_readl(dc, offset)); 1143b95800eeSThierry Reding } 1144b95800eeSThierry Reding 1145b95800eeSThierry Reding unlock: 1146b95800eeSThierry Reding drm_modeset_unlock(&dc->base.mutex); 1147b95800eeSThierry Reding return err; 1148b95800eeSThierry Reding } 1149b95800eeSThierry Reding 1150b95800eeSThierry Reding static int tegra_dc_show_crc(struct seq_file *s, void *data) 1151b95800eeSThierry Reding { 1152b95800eeSThierry Reding struct drm_info_node *node = s->private; 1153b95800eeSThierry Reding struct tegra_dc *dc = node->info_ent->data; 1154b95800eeSThierry Reding int err = 0; 1155b95800eeSThierry Reding u32 value; 1156b95800eeSThierry Reding 1157b95800eeSThierry Reding drm_modeset_lock(&dc->base.mutex, NULL); 1158b95800eeSThierry Reding 1159b95800eeSThierry Reding if (!dc->base.state->active) { 1160b95800eeSThierry Reding err = -EBUSY; 1161b95800eeSThierry Reding goto unlock; 1162b95800eeSThierry Reding } 1163b95800eeSThierry Reding 1164b95800eeSThierry Reding value = DC_COM_CRC_CONTROL_ACTIVE_DATA | DC_COM_CRC_CONTROL_ENABLE; 1165b95800eeSThierry Reding tegra_dc_writel(dc, value, DC_COM_CRC_CONTROL); 1166b95800eeSThierry Reding tegra_dc_commit(dc); 1167b95800eeSThierry Reding 1168b95800eeSThierry Reding drm_crtc_wait_one_vblank(&dc->base); 1169b95800eeSThierry Reding drm_crtc_wait_one_vblank(&dc->base); 1170b95800eeSThierry Reding 1171b95800eeSThierry Reding value = tegra_dc_readl(dc, DC_COM_CRC_CHECKSUM); 1172b95800eeSThierry Reding seq_printf(s, "%08x\n", value); 1173b95800eeSThierry Reding 1174b95800eeSThierry Reding tegra_dc_writel(dc, 0, DC_COM_CRC_CONTROL); 1175b95800eeSThierry Reding 1176b95800eeSThierry Reding unlock: 1177b95800eeSThierry Reding drm_modeset_unlock(&dc->base.mutex); 1178b95800eeSThierry Reding return err; 1179b95800eeSThierry Reding } 1180b95800eeSThierry Reding 1181b95800eeSThierry Reding static int tegra_dc_show_stats(struct seq_file *s, void *data) 1182b95800eeSThierry Reding { 1183b95800eeSThierry Reding struct drm_info_node *node = s->private; 1184b95800eeSThierry Reding struct tegra_dc *dc = node->info_ent->data; 1185b95800eeSThierry Reding 1186b95800eeSThierry Reding seq_printf(s, "frames: %lu\n", dc->stats.frames); 1187b95800eeSThierry Reding seq_printf(s, "vblank: %lu\n", dc->stats.vblank); 1188b95800eeSThierry Reding seq_printf(s, "underflow: %lu\n", dc->stats.underflow); 1189b95800eeSThierry Reding seq_printf(s, "overflow: %lu\n", dc->stats.overflow); 1190b95800eeSThierry Reding 1191b95800eeSThierry Reding return 0; 1192b95800eeSThierry Reding } 1193b95800eeSThierry Reding 1194b95800eeSThierry Reding static struct drm_info_list debugfs_files[] = { 1195b95800eeSThierry Reding { "regs", tegra_dc_show_regs, 0, NULL }, 1196b95800eeSThierry Reding { "crc", tegra_dc_show_crc, 0, NULL }, 1197b95800eeSThierry Reding { "stats", tegra_dc_show_stats, 0, NULL }, 1198b95800eeSThierry Reding }; 1199b95800eeSThierry Reding 1200b95800eeSThierry Reding static int tegra_dc_late_register(struct drm_crtc *crtc) 1201b95800eeSThierry Reding { 1202b95800eeSThierry Reding unsigned int i, count = ARRAY_SIZE(debugfs_files); 1203b95800eeSThierry Reding struct drm_minor *minor = crtc->dev->primary; 1204b95800eeSThierry Reding struct dentry *root = crtc->debugfs_entry; 1205b95800eeSThierry Reding struct tegra_dc *dc = to_tegra_dc(crtc); 1206b95800eeSThierry Reding int err; 1207b95800eeSThierry Reding 1208b95800eeSThierry Reding dc->debugfs_files = kmemdup(debugfs_files, sizeof(debugfs_files), 1209b95800eeSThierry Reding GFP_KERNEL); 1210b95800eeSThierry Reding if (!dc->debugfs_files) 1211b95800eeSThierry Reding return -ENOMEM; 1212b95800eeSThierry Reding 1213b95800eeSThierry Reding for (i = 0; i < count; i++) 1214b95800eeSThierry Reding dc->debugfs_files[i].data = dc; 1215b95800eeSThierry Reding 1216b95800eeSThierry Reding err = drm_debugfs_create_files(dc->debugfs_files, count, root, minor); 1217b95800eeSThierry Reding if (err < 0) 1218b95800eeSThierry Reding goto free; 1219b95800eeSThierry Reding 1220b95800eeSThierry Reding return 0; 1221b95800eeSThierry Reding 1222b95800eeSThierry Reding free: 1223b95800eeSThierry Reding kfree(dc->debugfs_files); 1224b95800eeSThierry Reding dc->debugfs_files = NULL; 1225b95800eeSThierry Reding 1226b95800eeSThierry Reding return err; 1227b95800eeSThierry Reding } 1228b95800eeSThierry Reding 1229b95800eeSThierry Reding static void tegra_dc_early_unregister(struct drm_crtc *crtc) 1230b95800eeSThierry Reding { 1231b95800eeSThierry Reding unsigned int count = ARRAY_SIZE(debugfs_files); 1232b95800eeSThierry Reding struct drm_minor *minor = crtc->dev->primary; 1233b95800eeSThierry Reding struct tegra_dc *dc = to_tegra_dc(crtc); 1234b95800eeSThierry Reding 1235b95800eeSThierry Reding drm_debugfs_remove_files(dc->debugfs_files, count, minor); 1236b95800eeSThierry Reding kfree(dc->debugfs_files); 1237b95800eeSThierry Reding dc->debugfs_files = NULL; 1238b95800eeSThierry Reding } 1239b95800eeSThierry Reding 1240c49c81e2SThierry Reding static u32 tegra_dc_get_vblank_counter(struct drm_crtc *crtc) 1241c49c81e2SThierry Reding { 1242c49c81e2SThierry Reding struct tegra_dc *dc = to_tegra_dc(crtc); 1243c49c81e2SThierry Reding 124447307954SThierry Reding /* XXX vblank syncpoints don't work with nvdisplay yet */ 124547307954SThierry Reding if (dc->syncpt && !dc->soc->has_nvdisplay) 1246c49c81e2SThierry Reding return host1x_syncpt_read(dc->syncpt); 1247c49c81e2SThierry Reding 1248c49c81e2SThierry Reding /* fallback to software emulated VBLANK counter */ 1249c49c81e2SThierry Reding return drm_crtc_vblank_count(&dc->base); 1250c49c81e2SThierry Reding } 1251c49c81e2SThierry Reding 1252c49c81e2SThierry Reding static int tegra_dc_enable_vblank(struct drm_crtc *crtc) 1253c49c81e2SThierry Reding { 1254c49c81e2SThierry Reding struct tegra_dc *dc = to_tegra_dc(crtc); 1255c49c81e2SThierry Reding unsigned long value, flags; 1256c49c81e2SThierry Reding 1257c49c81e2SThierry Reding spin_lock_irqsave(&dc->lock, flags); 1258c49c81e2SThierry Reding 1259c49c81e2SThierry Reding value = tegra_dc_readl(dc, DC_CMD_INT_MASK); 1260c49c81e2SThierry Reding value |= VBLANK_INT; 1261c49c81e2SThierry Reding tegra_dc_writel(dc, value, DC_CMD_INT_MASK); 1262c49c81e2SThierry Reding 1263c49c81e2SThierry Reding spin_unlock_irqrestore(&dc->lock, flags); 1264c49c81e2SThierry Reding 1265c49c81e2SThierry Reding return 0; 1266c49c81e2SThierry Reding } 1267c49c81e2SThierry Reding 1268c49c81e2SThierry Reding static void tegra_dc_disable_vblank(struct drm_crtc *crtc) 1269c49c81e2SThierry Reding { 1270c49c81e2SThierry Reding struct tegra_dc *dc = to_tegra_dc(crtc); 1271c49c81e2SThierry Reding unsigned long value, flags; 1272c49c81e2SThierry Reding 1273c49c81e2SThierry Reding spin_lock_irqsave(&dc->lock, flags); 1274c49c81e2SThierry Reding 1275c49c81e2SThierry Reding value = tegra_dc_readl(dc, DC_CMD_INT_MASK); 1276c49c81e2SThierry Reding value &= ~VBLANK_INT; 1277c49c81e2SThierry Reding tegra_dc_writel(dc, value, DC_CMD_INT_MASK); 1278c49c81e2SThierry Reding 1279c49c81e2SThierry Reding spin_unlock_irqrestore(&dc->lock, flags); 1280c49c81e2SThierry Reding } 1281c49c81e2SThierry Reding 1282dee8268fSThierry Reding static const struct drm_crtc_funcs tegra_crtc_funcs = { 12831503ca47SThierry Reding .page_flip = drm_atomic_helper_page_flip, 128474f48791SThierry Reding .set_config = drm_atomic_helper_set_config, 1285f002abc1SThierry Reding .destroy = tegra_dc_destroy, 1286ca915b10SThierry Reding .reset = tegra_crtc_reset, 1287ca915b10SThierry Reding .atomic_duplicate_state = tegra_crtc_atomic_duplicate_state, 1288ca915b10SThierry Reding .atomic_destroy_state = tegra_crtc_atomic_destroy_state, 1289b95800eeSThierry Reding .late_register = tegra_dc_late_register, 1290b95800eeSThierry Reding .early_unregister = tegra_dc_early_unregister, 129110437d9bSShawn Guo .get_vblank_counter = tegra_dc_get_vblank_counter, 129210437d9bSShawn Guo .enable_vblank = tegra_dc_enable_vblank, 129310437d9bSShawn Guo .disable_vblank = tegra_dc_disable_vblank, 1294dee8268fSThierry Reding }; 1295dee8268fSThierry Reding 1296dee8268fSThierry Reding static int tegra_dc_set_timings(struct tegra_dc *dc, 1297dee8268fSThierry Reding struct drm_display_mode *mode) 1298dee8268fSThierry Reding { 12990444c0ffSThierry Reding unsigned int h_ref_to_sync = 1; 13000444c0ffSThierry Reding unsigned int v_ref_to_sync = 1; 1301dee8268fSThierry Reding unsigned long value; 1302dee8268fSThierry Reding 130347307954SThierry Reding if (!dc->soc->has_nvdisplay) { 1304dee8268fSThierry Reding tegra_dc_writel(dc, 0x0, DC_DISP_DISP_TIMING_OPTIONS); 1305dee8268fSThierry Reding 1306dee8268fSThierry Reding value = (v_ref_to_sync << 16) | h_ref_to_sync; 1307dee8268fSThierry Reding tegra_dc_writel(dc, value, DC_DISP_REF_TO_SYNC); 130847307954SThierry Reding } 1309dee8268fSThierry Reding 1310dee8268fSThierry Reding value = ((mode->vsync_end - mode->vsync_start) << 16) | 1311dee8268fSThierry Reding ((mode->hsync_end - mode->hsync_start) << 0); 1312dee8268fSThierry Reding tegra_dc_writel(dc, value, DC_DISP_SYNC_WIDTH); 1313dee8268fSThierry Reding 1314dee8268fSThierry Reding value = ((mode->vtotal - mode->vsync_end) << 16) | 1315dee8268fSThierry Reding ((mode->htotal - mode->hsync_end) << 0); 1316dee8268fSThierry Reding tegra_dc_writel(dc, value, DC_DISP_BACK_PORCH); 1317dee8268fSThierry Reding 1318dee8268fSThierry Reding value = ((mode->vsync_start - mode->vdisplay) << 16) | 1319dee8268fSThierry Reding ((mode->hsync_start - mode->hdisplay) << 0); 1320dee8268fSThierry Reding tegra_dc_writel(dc, value, DC_DISP_FRONT_PORCH); 1321dee8268fSThierry Reding 1322dee8268fSThierry Reding value = (mode->vdisplay << 16) | mode->hdisplay; 1323dee8268fSThierry Reding tegra_dc_writel(dc, value, DC_DISP_ACTIVE); 1324dee8268fSThierry Reding 1325dee8268fSThierry Reding return 0; 1326dee8268fSThierry Reding } 1327dee8268fSThierry Reding 13289d910b60SThierry Reding /** 13299d910b60SThierry Reding * tegra_dc_state_setup_clock - check clock settings and store them in atomic 13309d910b60SThierry Reding * state 13319d910b60SThierry Reding * @dc: display controller 13329d910b60SThierry Reding * @crtc_state: CRTC atomic state 13339d910b60SThierry Reding * @clk: parent clock for display controller 13349d910b60SThierry Reding * @pclk: pixel clock 13359d910b60SThierry Reding * @div: shift clock divider 13369d910b60SThierry Reding * 13379d910b60SThierry Reding * Returns: 13389d910b60SThierry Reding * 0 on success or a negative error-code on failure. 13399d910b60SThierry Reding */ 1340ca915b10SThierry Reding int tegra_dc_state_setup_clock(struct tegra_dc *dc, 1341ca915b10SThierry Reding struct drm_crtc_state *crtc_state, 1342ca915b10SThierry Reding struct clk *clk, unsigned long pclk, 1343ca915b10SThierry Reding unsigned int div) 1344ca915b10SThierry Reding { 1345ca915b10SThierry Reding struct tegra_dc_state *state = to_dc_state(crtc_state); 1346ca915b10SThierry Reding 1347d2982748SThierry Reding if (!clk_has_parent(dc->clk, clk)) 1348d2982748SThierry Reding return -EINVAL; 1349d2982748SThierry Reding 1350ca915b10SThierry Reding state->clk = clk; 1351ca915b10SThierry Reding state->pclk = pclk; 1352ca915b10SThierry Reding state->div = div; 1353ca915b10SThierry Reding 1354ca915b10SThierry Reding return 0; 1355ca915b10SThierry Reding } 1356ca915b10SThierry Reding 135776d59ed0SThierry Reding static void tegra_dc_commit_state(struct tegra_dc *dc, 135876d59ed0SThierry Reding struct tegra_dc_state *state) 135976d59ed0SThierry Reding { 136076d59ed0SThierry Reding u32 value; 136176d59ed0SThierry Reding int err; 136276d59ed0SThierry Reding 136376d59ed0SThierry Reding err = clk_set_parent(dc->clk, state->clk); 136476d59ed0SThierry Reding if (err < 0) 136576d59ed0SThierry Reding dev_err(dc->dev, "failed to set parent clock: %d\n", err); 136676d59ed0SThierry Reding 136776d59ed0SThierry Reding /* 136876d59ed0SThierry Reding * Outputs may not want to change the parent clock rate. This is only 136976d59ed0SThierry Reding * relevant to Tegra20 where only a single display PLL is available. 137076d59ed0SThierry Reding * Since that PLL would typically be used for HDMI, an internal LVDS 137176d59ed0SThierry Reding * panel would need to be driven by some other clock such as PLL_P 137276d59ed0SThierry Reding * which is shared with other peripherals. Changing the clock rate 137376d59ed0SThierry Reding * should therefore be avoided. 137476d59ed0SThierry Reding */ 137576d59ed0SThierry Reding if (state->pclk > 0) { 137676d59ed0SThierry Reding err = clk_set_rate(state->clk, state->pclk); 137776d59ed0SThierry Reding if (err < 0) 137876d59ed0SThierry Reding dev_err(dc->dev, 137976d59ed0SThierry Reding "failed to set clock rate to %lu Hz\n", 138076d59ed0SThierry Reding state->pclk); 138176d59ed0SThierry Reding } 138276d59ed0SThierry Reding 138376d59ed0SThierry Reding DRM_DEBUG_KMS("rate: %lu, div: %u\n", clk_get_rate(dc->clk), 138476d59ed0SThierry Reding state->div); 138576d59ed0SThierry Reding DRM_DEBUG_KMS("pclk: %lu\n", state->pclk); 138676d59ed0SThierry Reding 138747307954SThierry Reding if (!dc->soc->has_nvdisplay) { 138876d59ed0SThierry Reding value = SHIFT_CLK_DIVIDER(state->div) | PIXEL_CLK_DIVIDER_PCD1; 138976d59ed0SThierry Reding tegra_dc_writel(dc, value, DC_DISP_DISP_CLOCK_CONTROL); 139047307954SThierry Reding } 139139e08affSThierry Reding 139239e08affSThierry Reding err = clk_set_rate(dc->clk, state->pclk); 139339e08affSThierry Reding if (err < 0) 139439e08affSThierry Reding dev_err(dc->dev, "failed to set clock %pC to %lu Hz: %d\n", 139539e08affSThierry Reding dc->clk, state->pclk, err); 139676d59ed0SThierry Reding } 139776d59ed0SThierry Reding 1398003fc848SThierry Reding static void tegra_dc_stop(struct tegra_dc *dc) 1399003fc848SThierry Reding { 1400003fc848SThierry Reding u32 value; 1401003fc848SThierry Reding 1402003fc848SThierry Reding /* stop the display controller */ 1403003fc848SThierry Reding value = tegra_dc_readl(dc, DC_CMD_DISPLAY_COMMAND); 1404003fc848SThierry Reding value &= ~DISP_CTRL_MODE_MASK; 1405003fc848SThierry Reding tegra_dc_writel(dc, value, DC_CMD_DISPLAY_COMMAND); 1406003fc848SThierry Reding 1407003fc848SThierry Reding tegra_dc_commit(dc); 1408003fc848SThierry Reding } 1409003fc848SThierry Reding 1410003fc848SThierry Reding static bool tegra_dc_idle(struct tegra_dc *dc) 1411003fc848SThierry Reding { 1412003fc848SThierry Reding u32 value; 1413003fc848SThierry Reding 1414003fc848SThierry Reding value = tegra_dc_readl_active(dc, DC_CMD_DISPLAY_COMMAND); 1415003fc848SThierry Reding 1416003fc848SThierry Reding return (value & DISP_CTRL_MODE_MASK) == 0; 1417003fc848SThierry Reding } 1418003fc848SThierry Reding 1419003fc848SThierry Reding static int tegra_dc_wait_idle(struct tegra_dc *dc, unsigned long timeout) 1420003fc848SThierry Reding { 1421003fc848SThierry Reding timeout = jiffies + msecs_to_jiffies(timeout); 1422003fc848SThierry Reding 1423003fc848SThierry Reding while (time_before(jiffies, timeout)) { 1424003fc848SThierry Reding if (tegra_dc_idle(dc)) 1425003fc848SThierry Reding return 0; 1426003fc848SThierry Reding 1427003fc848SThierry Reding usleep_range(1000, 2000); 1428003fc848SThierry Reding } 1429003fc848SThierry Reding 1430003fc848SThierry Reding dev_dbg(dc->dev, "timeout waiting for DC to become idle\n"); 1431003fc848SThierry Reding return -ETIMEDOUT; 1432003fc848SThierry Reding } 1433003fc848SThierry Reding 143464581714SLaurent Pinchart static void tegra_crtc_atomic_disable(struct drm_crtc *crtc, 143564581714SLaurent Pinchart struct drm_crtc_state *old_state) 1436003fc848SThierry Reding { 1437003fc848SThierry Reding struct tegra_dc *dc = to_tegra_dc(crtc); 1438003fc848SThierry Reding u32 value; 1439003fc848SThierry Reding 1440003fc848SThierry Reding if (!tegra_dc_idle(dc)) { 1441003fc848SThierry Reding tegra_dc_stop(dc); 1442003fc848SThierry Reding 1443003fc848SThierry Reding /* 1444003fc848SThierry Reding * Ignore the return value, there isn't anything useful to do 1445003fc848SThierry Reding * in case this fails. 1446003fc848SThierry Reding */ 1447003fc848SThierry Reding tegra_dc_wait_idle(dc, 100); 1448003fc848SThierry Reding } 1449003fc848SThierry Reding 1450003fc848SThierry Reding /* 1451003fc848SThierry Reding * This should really be part of the RGB encoder driver, but clearing 1452003fc848SThierry Reding * these bits has the side-effect of stopping the display controller. 1453003fc848SThierry Reding * When that happens no VBLANK interrupts will be raised. At the same 1454003fc848SThierry Reding * time the encoder is disabled before the display controller, so the 1455003fc848SThierry Reding * above code is always going to timeout waiting for the controller 1456003fc848SThierry Reding * to go idle. 1457003fc848SThierry Reding * 1458003fc848SThierry Reding * Given the close coupling between the RGB encoder and the display 1459003fc848SThierry Reding * controller doing it here is still kind of okay. None of the other 1460003fc848SThierry Reding * encoder drivers require these bits to be cleared. 1461003fc848SThierry Reding * 1462003fc848SThierry Reding * XXX: Perhaps given that the display controller is switched off at 1463003fc848SThierry Reding * this point anyway maybe clearing these bits isn't even useful for 1464003fc848SThierry Reding * the RGB encoder? 1465003fc848SThierry Reding */ 1466003fc848SThierry Reding if (dc->rgb) { 1467003fc848SThierry Reding value = tegra_dc_readl(dc, DC_CMD_DISPLAY_POWER_CONTROL); 1468003fc848SThierry Reding value &= ~(PW0_ENABLE | PW1_ENABLE | PW2_ENABLE | PW3_ENABLE | 1469003fc848SThierry Reding PW4_ENABLE | PM0_ENABLE | PM1_ENABLE); 1470003fc848SThierry Reding tegra_dc_writel(dc, value, DC_CMD_DISPLAY_POWER_CONTROL); 1471003fc848SThierry Reding } 1472003fc848SThierry Reding 1473003fc848SThierry Reding tegra_dc_stats_reset(&dc->stats); 1474003fc848SThierry Reding drm_crtc_vblank_off(crtc); 147533a8eb8dSThierry Reding 14769d99ab6eSThierry Reding spin_lock_irq(&crtc->dev->event_lock); 14779d99ab6eSThierry Reding 14789d99ab6eSThierry Reding if (crtc->state->event) { 14799d99ab6eSThierry Reding drm_crtc_send_vblank_event(crtc, crtc->state->event); 14809d99ab6eSThierry Reding crtc->state->event = NULL; 14819d99ab6eSThierry Reding } 14829d99ab6eSThierry Reding 14839d99ab6eSThierry Reding spin_unlock_irq(&crtc->dev->event_lock); 14849d99ab6eSThierry Reding 148533a8eb8dSThierry Reding pm_runtime_put_sync(dc->dev); 1486003fc848SThierry Reding } 1487003fc848SThierry Reding 14880b20a0f8SLaurent Pinchart static void tegra_crtc_atomic_enable(struct drm_crtc *crtc, 14890b20a0f8SLaurent Pinchart struct drm_crtc_state *old_state) 1490dee8268fSThierry Reding { 14914aa3df71SThierry Reding struct drm_display_mode *mode = &crtc->state->adjusted_mode; 149276d59ed0SThierry Reding struct tegra_dc_state *state = to_dc_state(crtc->state); 1493dee8268fSThierry Reding struct tegra_dc *dc = to_tegra_dc(crtc); 1494dbb3f2f7SThierry Reding u32 value; 1495dee8268fSThierry Reding 149633a8eb8dSThierry Reding pm_runtime_get_sync(dc->dev); 149733a8eb8dSThierry Reding 149833a8eb8dSThierry Reding /* initialize display controller */ 149933a8eb8dSThierry Reding if (dc->syncpt) { 150047307954SThierry Reding u32 syncpt = host1x_syncpt_id(dc->syncpt), enable; 150147307954SThierry Reding 150247307954SThierry Reding if (dc->soc->has_nvdisplay) 150347307954SThierry Reding enable = 1 << 31; 150447307954SThierry Reding else 150547307954SThierry Reding enable = 1 << 8; 150633a8eb8dSThierry Reding 150733a8eb8dSThierry Reding value = SYNCPT_CNTRL_NO_STALL; 150833a8eb8dSThierry Reding tegra_dc_writel(dc, value, DC_CMD_GENERAL_INCR_SYNCPT_CNTRL); 150933a8eb8dSThierry Reding 151047307954SThierry Reding value = enable | syncpt; 151133a8eb8dSThierry Reding tegra_dc_writel(dc, value, DC_CMD_CONT_SYNCPT_VSYNC); 151233a8eb8dSThierry Reding } 151333a8eb8dSThierry Reding 151447307954SThierry Reding if (dc->soc->has_nvdisplay) { 151547307954SThierry Reding value = DSC_TO_UF_INT | DSC_BBUF_UF_INT | DSC_RBUF_UF_INT | 151647307954SThierry Reding DSC_OBUF_UF_INT; 151747307954SThierry Reding tegra_dc_writel(dc, value, DC_CMD_INT_TYPE); 151847307954SThierry Reding 151947307954SThierry Reding value = DSC_TO_UF_INT | DSC_BBUF_UF_INT | DSC_RBUF_UF_INT | 152047307954SThierry Reding DSC_OBUF_UF_INT | SD3_BUCKET_WALK_DONE_INT | 152147307954SThierry Reding HEAD_UF_INT | MSF_INT | REG_TMOUT_INT | 152247307954SThierry Reding REGION_CRC_INT | V_PULSE2_INT | V_PULSE3_INT | 152347307954SThierry Reding VBLANK_INT | FRAME_END_INT; 152447307954SThierry Reding tegra_dc_writel(dc, value, DC_CMD_INT_POLARITY); 152547307954SThierry Reding 152647307954SThierry Reding value = SD3_BUCKET_WALK_DONE_INT | HEAD_UF_INT | VBLANK_INT | 152747307954SThierry Reding FRAME_END_INT; 152847307954SThierry Reding tegra_dc_writel(dc, value, DC_CMD_INT_ENABLE); 152947307954SThierry Reding 153047307954SThierry Reding value = HEAD_UF_INT | REG_TMOUT_INT | FRAME_END_INT; 153147307954SThierry Reding tegra_dc_writel(dc, value, DC_CMD_INT_MASK); 153247307954SThierry Reding 153347307954SThierry Reding tegra_dc_writel(dc, READ_MUX, DC_CMD_STATE_ACCESS); 153447307954SThierry Reding } else { 153533a8eb8dSThierry Reding value = WIN_A_UF_INT | WIN_B_UF_INT | WIN_C_UF_INT | 153633a8eb8dSThierry Reding WIN_A_OF_INT | WIN_B_OF_INT | WIN_C_OF_INT; 153733a8eb8dSThierry Reding tegra_dc_writel(dc, value, DC_CMD_INT_TYPE); 153833a8eb8dSThierry Reding 153933a8eb8dSThierry Reding value = WIN_A_UF_INT | WIN_B_UF_INT | WIN_C_UF_INT | 154033a8eb8dSThierry Reding WIN_A_OF_INT | WIN_B_OF_INT | WIN_C_OF_INT; 154133a8eb8dSThierry Reding tegra_dc_writel(dc, value, DC_CMD_INT_POLARITY); 154233a8eb8dSThierry Reding 154333a8eb8dSThierry Reding /* initialize timer */ 154433a8eb8dSThierry Reding value = CURSOR_THRESHOLD(0) | WINDOW_A_THRESHOLD(0x20) | 154533a8eb8dSThierry Reding WINDOW_B_THRESHOLD(0x20) | WINDOW_C_THRESHOLD(0x20); 154633a8eb8dSThierry Reding tegra_dc_writel(dc, value, DC_DISP_DISP_MEM_HIGH_PRIORITY); 154733a8eb8dSThierry Reding 154833a8eb8dSThierry Reding value = CURSOR_THRESHOLD(0) | WINDOW_A_THRESHOLD(1) | 154933a8eb8dSThierry Reding WINDOW_B_THRESHOLD(1) | WINDOW_C_THRESHOLD(1); 155033a8eb8dSThierry Reding tegra_dc_writel(dc, value, DC_DISP_DISP_MEM_HIGH_PRIORITY_TIMER); 155133a8eb8dSThierry Reding 155233a8eb8dSThierry Reding value = VBLANK_INT | WIN_A_UF_INT | WIN_B_UF_INT | WIN_C_UF_INT | 155333a8eb8dSThierry Reding WIN_A_OF_INT | WIN_B_OF_INT | WIN_C_OF_INT; 155433a8eb8dSThierry Reding tegra_dc_writel(dc, value, DC_CMD_INT_ENABLE); 155533a8eb8dSThierry Reding 155633a8eb8dSThierry Reding value = WIN_A_UF_INT | WIN_B_UF_INT | WIN_C_UF_INT | 155733a8eb8dSThierry Reding WIN_A_OF_INT | WIN_B_OF_INT | WIN_C_OF_INT; 155833a8eb8dSThierry Reding tegra_dc_writel(dc, value, DC_CMD_INT_MASK); 155947307954SThierry Reding } 156033a8eb8dSThierry Reding 15617116e9a8SThierry Reding if (dc->soc->supports_background_color) 15627116e9a8SThierry Reding tegra_dc_writel(dc, 0, DC_DISP_BLEND_BACKGROUND_COLOR); 15637116e9a8SThierry Reding else 156433a8eb8dSThierry Reding tegra_dc_writel(dc, 0, DC_DISP_BORDER_COLOR); 156533a8eb8dSThierry Reding 156633a8eb8dSThierry Reding /* apply PLL and pixel clock changes */ 156776d59ed0SThierry Reding tegra_dc_commit_state(dc, state); 156876d59ed0SThierry Reding 1569dee8268fSThierry Reding /* program display mode */ 1570dee8268fSThierry Reding tegra_dc_set_timings(dc, mode); 1571dee8268fSThierry Reding 15728620fc62SThierry Reding /* interlacing isn't supported yet, so disable it */ 15738620fc62SThierry Reding if (dc->soc->supports_interlacing) { 15748620fc62SThierry Reding value = tegra_dc_readl(dc, DC_DISP_INTERLACE_CONTROL); 15758620fc62SThierry Reding value &= ~INTERLACE_ENABLE; 15768620fc62SThierry Reding tegra_dc_writel(dc, value, DC_DISP_INTERLACE_CONTROL); 15778620fc62SThierry Reding } 1578666cb873SThierry Reding 1579666cb873SThierry Reding value = tegra_dc_readl(dc, DC_CMD_DISPLAY_COMMAND); 1580666cb873SThierry Reding value &= ~DISP_CTRL_MODE_MASK; 1581666cb873SThierry Reding value |= DISP_CTRL_MODE_C_DISPLAY; 1582666cb873SThierry Reding tegra_dc_writel(dc, value, DC_CMD_DISPLAY_COMMAND); 1583666cb873SThierry Reding 158447307954SThierry Reding if (!dc->soc->has_nvdisplay) { 1585666cb873SThierry Reding value = tegra_dc_readl(dc, DC_CMD_DISPLAY_POWER_CONTROL); 1586666cb873SThierry Reding value |= PW0_ENABLE | PW1_ENABLE | PW2_ENABLE | PW3_ENABLE | 1587666cb873SThierry Reding PW4_ENABLE | PM0_ENABLE | PM1_ENABLE; 1588666cb873SThierry Reding tegra_dc_writel(dc, value, DC_CMD_DISPLAY_POWER_CONTROL); 158947307954SThierry Reding } 159047307954SThierry Reding 159147307954SThierry Reding /* enable underflow reporting and display red for missing pixels */ 159247307954SThierry Reding if (dc->soc->has_nvdisplay) { 159347307954SThierry Reding value = UNDERFLOW_MODE_RED | UNDERFLOW_REPORT_ENABLE; 159447307954SThierry Reding tegra_dc_writel(dc, value, DC_COM_RG_UNDERFLOW); 159547307954SThierry Reding } 1596666cb873SThierry Reding 1597666cb873SThierry Reding tegra_dc_commit(dc); 1598dee8268fSThierry Reding 15998ff64c17SThierry Reding drm_crtc_vblank_on(crtc); 1600dee8268fSThierry Reding } 1601dee8268fSThierry Reding 16024aa3df71SThierry Reding static int tegra_crtc_atomic_check(struct drm_crtc *crtc, 16034aa3df71SThierry Reding struct drm_crtc_state *state) 16044aa3df71SThierry Reding { 1605c4755fb9SThierry Reding struct tegra_atomic_state *s = to_tegra_atomic_state(state->state); 1606c4755fb9SThierry Reding struct tegra_dc_state *tegra = to_dc_state(state); 1607c4755fb9SThierry Reding 1608c4755fb9SThierry Reding /* 1609c4755fb9SThierry Reding * The display hub display clock needs to be fed by the display clock 1610c4755fb9SThierry Reding * with the highest frequency to ensure proper functioning of all the 1611c4755fb9SThierry Reding * displays. 1612c4755fb9SThierry Reding * 1613c4755fb9SThierry Reding * Note that this isn't used before Tegra186, but it doesn't hurt and 1614c4755fb9SThierry Reding * conditionalizing it would make the code less clean. 1615c4755fb9SThierry Reding */ 1616c4755fb9SThierry Reding if (state->active) { 1617c4755fb9SThierry Reding if (!s->clk_disp || tegra->pclk > s->rate) { 1618c4755fb9SThierry Reding s->dc = to_tegra_dc(crtc); 1619c4755fb9SThierry Reding s->clk_disp = s->dc->clk; 1620c4755fb9SThierry Reding s->rate = tegra->pclk; 1621c4755fb9SThierry Reding } 1622c4755fb9SThierry Reding } 1623c4755fb9SThierry Reding 16244aa3df71SThierry Reding return 0; 16254aa3df71SThierry Reding } 16264aa3df71SThierry Reding 1627613d2b27SMaarten Lankhorst static void tegra_crtc_atomic_begin(struct drm_crtc *crtc, 1628613d2b27SMaarten Lankhorst struct drm_crtc_state *old_crtc_state) 16294aa3df71SThierry Reding { 16309d99ab6eSThierry Reding unsigned long flags; 16311503ca47SThierry Reding 16321503ca47SThierry Reding if (crtc->state->event) { 16339d99ab6eSThierry Reding spin_lock_irqsave(&crtc->dev->event_lock, flags); 16341503ca47SThierry Reding 16359d99ab6eSThierry Reding if (drm_crtc_vblank_get(crtc) != 0) 16369d99ab6eSThierry Reding drm_crtc_send_vblank_event(crtc, crtc->state->event); 16379d99ab6eSThierry Reding else 16389d99ab6eSThierry Reding drm_crtc_arm_vblank_event(crtc, crtc->state->event); 16391503ca47SThierry Reding 16409d99ab6eSThierry Reding spin_unlock_irqrestore(&crtc->dev->event_lock, flags); 16419d99ab6eSThierry Reding 16421503ca47SThierry Reding crtc->state->event = NULL; 16431503ca47SThierry Reding } 16444aa3df71SThierry Reding } 16454aa3df71SThierry Reding 1646613d2b27SMaarten Lankhorst static void tegra_crtc_atomic_flush(struct drm_crtc *crtc, 1647613d2b27SMaarten Lankhorst struct drm_crtc_state *old_crtc_state) 16484aa3df71SThierry Reding { 164947802b09SThierry Reding struct tegra_dc_state *state = to_dc_state(crtc->state); 165047802b09SThierry Reding struct tegra_dc *dc = to_tegra_dc(crtc); 165147307954SThierry Reding u32 value; 165247802b09SThierry Reding 165347307954SThierry Reding value = state->planes << 8 | GENERAL_UPDATE; 165447307954SThierry Reding tegra_dc_writel(dc, value, DC_CMD_STATE_CONTROL); 165547307954SThierry Reding value = tegra_dc_readl(dc, DC_CMD_STATE_CONTROL); 165647307954SThierry Reding 165747307954SThierry Reding value = state->planes | GENERAL_ACT_REQ; 165847307954SThierry Reding tegra_dc_writel(dc, value, DC_CMD_STATE_CONTROL); 165947307954SThierry Reding value = tegra_dc_readl(dc, DC_CMD_STATE_CONTROL); 16604aa3df71SThierry Reding } 16614aa3df71SThierry Reding 1662dee8268fSThierry Reding static const struct drm_crtc_helper_funcs tegra_crtc_helper_funcs = { 16634aa3df71SThierry Reding .atomic_check = tegra_crtc_atomic_check, 16644aa3df71SThierry Reding .atomic_begin = tegra_crtc_atomic_begin, 16654aa3df71SThierry Reding .atomic_flush = tegra_crtc_atomic_flush, 16660b20a0f8SLaurent Pinchart .atomic_enable = tegra_crtc_atomic_enable, 166764581714SLaurent Pinchart .atomic_disable = tegra_crtc_atomic_disable, 1668dee8268fSThierry Reding }; 1669dee8268fSThierry Reding 1670dee8268fSThierry Reding static irqreturn_t tegra_dc_irq(int irq, void *data) 1671dee8268fSThierry Reding { 1672dee8268fSThierry Reding struct tegra_dc *dc = data; 1673dee8268fSThierry Reding unsigned long status; 1674dee8268fSThierry Reding 1675dee8268fSThierry Reding status = tegra_dc_readl(dc, DC_CMD_INT_STATUS); 1676dee8268fSThierry Reding tegra_dc_writel(dc, status, DC_CMD_INT_STATUS); 1677dee8268fSThierry Reding 1678dee8268fSThierry Reding if (status & FRAME_END_INT) { 1679dee8268fSThierry Reding /* 1680dee8268fSThierry Reding dev_dbg(dc->dev, "%s(): frame end\n", __func__); 1681dee8268fSThierry Reding */ 1682791ddb1eSThierry Reding dc->stats.frames++; 1683dee8268fSThierry Reding } 1684dee8268fSThierry Reding 1685dee8268fSThierry Reding if (status & VBLANK_INT) { 1686dee8268fSThierry Reding /* 1687dee8268fSThierry Reding dev_dbg(dc->dev, "%s(): vertical blank\n", __func__); 1688dee8268fSThierry Reding */ 1689ed7dae58SThierry Reding drm_crtc_handle_vblank(&dc->base); 1690791ddb1eSThierry Reding dc->stats.vblank++; 1691dee8268fSThierry Reding } 1692dee8268fSThierry Reding 1693dee8268fSThierry Reding if (status & (WIN_A_UF_INT | WIN_B_UF_INT | WIN_C_UF_INT)) { 1694dee8268fSThierry Reding /* 1695dee8268fSThierry Reding dev_dbg(dc->dev, "%s(): underflow\n", __func__); 1696dee8268fSThierry Reding */ 1697791ddb1eSThierry Reding dc->stats.underflow++; 1698791ddb1eSThierry Reding } 1699791ddb1eSThierry Reding 1700791ddb1eSThierry Reding if (status & (WIN_A_OF_INT | WIN_B_OF_INT | WIN_C_OF_INT)) { 1701791ddb1eSThierry Reding /* 1702791ddb1eSThierry Reding dev_dbg(dc->dev, "%s(): overflow\n", __func__); 1703791ddb1eSThierry Reding */ 1704791ddb1eSThierry Reding dc->stats.overflow++; 1705dee8268fSThierry Reding } 1706dee8268fSThierry Reding 170747307954SThierry Reding if (status & HEAD_UF_INT) { 170847307954SThierry Reding dev_dbg_ratelimited(dc->dev, "%s(): head underflow\n", __func__); 170947307954SThierry Reding dc->stats.underflow++; 171047307954SThierry Reding } 171147307954SThierry Reding 1712dee8268fSThierry Reding return IRQ_HANDLED; 1713dee8268fSThierry Reding } 1714dee8268fSThierry Reding 1715dee8268fSThierry Reding static int tegra_dc_init(struct host1x_client *client) 1716dee8268fSThierry Reding { 17179910f5c4SThierry Reding struct drm_device *drm = dev_get_drvdata(client->parent); 17182bcdcbfaSThierry Reding unsigned long flags = HOST1X_SYNCPT_CLIENT_MANAGED; 1719dee8268fSThierry Reding struct tegra_dc *dc = host1x_client_to_dc(client); 1720d1f3e1e0SThierry Reding struct tegra_drm *tegra = drm->dev_private; 1721c7679306SThierry Reding struct drm_plane *primary = NULL; 1722c7679306SThierry Reding struct drm_plane *cursor = NULL; 1723dee8268fSThierry Reding int err; 1724dee8268fSThierry Reding 1725617dd7ccSThierry Reding dc->syncpt = host1x_syncpt_request(client, flags); 17262bcdcbfaSThierry Reding if (!dc->syncpt) 17272bcdcbfaSThierry Reding dev_warn(dc->dev, "failed to allocate syncpoint\n"); 17282bcdcbfaSThierry Reding 1729df06b759SThierry Reding if (tegra->domain) { 1730df06b759SThierry Reding err = iommu_attach_device(tegra->domain, dc->dev); 1731df06b759SThierry Reding if (err < 0) { 1732df06b759SThierry Reding dev_err(dc->dev, "failed to attach to domain: %d\n", 1733df06b759SThierry Reding err); 1734df06b759SThierry Reding return err; 1735df06b759SThierry Reding } 1736df06b759SThierry Reding 1737df06b759SThierry Reding dc->domain = tegra->domain; 1738df06b759SThierry Reding } 1739df06b759SThierry Reding 174047307954SThierry Reding if (dc->soc->wgrps) 174147307954SThierry Reding primary = tegra_dc_add_shared_planes(drm, dc); 174247307954SThierry Reding else 174347307954SThierry Reding primary = tegra_dc_add_planes(drm, dc); 174447307954SThierry Reding 1745c7679306SThierry Reding if (IS_ERR(primary)) { 1746c7679306SThierry Reding err = PTR_ERR(primary); 1747c7679306SThierry Reding goto cleanup; 1748c7679306SThierry Reding } 1749c7679306SThierry Reding 1750c7679306SThierry Reding if (dc->soc->supports_cursor) { 1751c7679306SThierry Reding cursor = tegra_dc_cursor_plane_create(drm, dc); 1752c7679306SThierry Reding if (IS_ERR(cursor)) { 1753c7679306SThierry Reding err = PTR_ERR(cursor); 1754c7679306SThierry Reding goto cleanup; 1755c7679306SThierry Reding } 1756c7679306SThierry Reding } 1757c7679306SThierry Reding 1758c7679306SThierry Reding err = drm_crtc_init_with_planes(drm, &dc->base, primary, cursor, 1759f9882876SVille Syrjälä &tegra_crtc_funcs, NULL); 1760c7679306SThierry Reding if (err < 0) 1761c7679306SThierry Reding goto cleanup; 1762c7679306SThierry Reding 1763dee8268fSThierry Reding drm_crtc_helper_add(&dc->base, &tegra_crtc_helper_funcs); 1764dee8268fSThierry Reding 1765d1f3e1e0SThierry Reding /* 1766d1f3e1e0SThierry Reding * Keep track of the minimum pitch alignment across all display 1767d1f3e1e0SThierry Reding * controllers. 1768d1f3e1e0SThierry Reding */ 1769d1f3e1e0SThierry Reding if (dc->soc->pitch_align > tegra->pitch_align) 1770d1f3e1e0SThierry Reding tegra->pitch_align = dc->soc->pitch_align; 1771d1f3e1e0SThierry Reding 17729910f5c4SThierry Reding err = tegra_dc_rgb_init(drm, dc); 1773dee8268fSThierry Reding if (err < 0 && err != -ENODEV) { 1774dee8268fSThierry Reding dev_err(dc->dev, "failed to initialize RGB output: %d\n", err); 1775c7679306SThierry Reding goto cleanup; 1776dee8268fSThierry Reding } 1777dee8268fSThierry Reding 1778dee8268fSThierry Reding err = devm_request_irq(dc->dev, dc->irq, tegra_dc_irq, 0, 1779dee8268fSThierry Reding dev_name(dc->dev), dc); 1780dee8268fSThierry Reding if (err < 0) { 1781dee8268fSThierry Reding dev_err(dc->dev, "failed to request IRQ#%u: %d\n", dc->irq, 1782dee8268fSThierry Reding err); 1783c7679306SThierry Reding goto cleanup; 1784dee8268fSThierry Reding } 1785dee8268fSThierry Reding 1786dee8268fSThierry Reding return 0; 1787c7679306SThierry Reding 1788c7679306SThierry Reding cleanup: 178947307954SThierry Reding if (!IS_ERR_OR_NULL(cursor)) 1790c7679306SThierry Reding drm_plane_cleanup(cursor); 1791c7679306SThierry Reding 179247307954SThierry Reding if (!IS_ERR(primary)) 1793c7679306SThierry Reding drm_plane_cleanup(primary); 1794c7679306SThierry Reding 1795c7679306SThierry Reding if (tegra->domain) { 1796c7679306SThierry Reding iommu_detach_device(tegra->domain, dc->dev); 1797c7679306SThierry Reding dc->domain = NULL; 1798c7679306SThierry Reding } 1799c7679306SThierry Reding 1800c7679306SThierry Reding return err; 1801dee8268fSThierry Reding } 1802dee8268fSThierry Reding 1803dee8268fSThierry Reding static int tegra_dc_exit(struct host1x_client *client) 1804dee8268fSThierry Reding { 1805dee8268fSThierry Reding struct tegra_dc *dc = host1x_client_to_dc(client); 1806dee8268fSThierry Reding int err; 1807dee8268fSThierry Reding 1808dee8268fSThierry Reding devm_free_irq(dc->dev, dc->irq, dc); 1809dee8268fSThierry Reding 1810dee8268fSThierry Reding err = tegra_dc_rgb_exit(dc); 1811dee8268fSThierry Reding if (err) { 1812dee8268fSThierry Reding dev_err(dc->dev, "failed to shutdown RGB output: %d\n", err); 1813dee8268fSThierry Reding return err; 1814dee8268fSThierry Reding } 1815dee8268fSThierry Reding 1816df06b759SThierry Reding if (dc->domain) { 1817df06b759SThierry Reding iommu_detach_device(dc->domain, dc->dev); 1818df06b759SThierry Reding dc->domain = NULL; 1819df06b759SThierry Reding } 1820df06b759SThierry Reding 18212bcdcbfaSThierry Reding host1x_syncpt_free(dc->syncpt); 18222bcdcbfaSThierry Reding 1823dee8268fSThierry Reding return 0; 1824dee8268fSThierry Reding } 1825dee8268fSThierry Reding 1826dee8268fSThierry Reding static const struct host1x_client_ops dc_client_ops = { 1827dee8268fSThierry Reding .init = tegra_dc_init, 1828dee8268fSThierry Reding .exit = tegra_dc_exit, 1829dee8268fSThierry Reding }; 1830dee8268fSThierry Reding 18318620fc62SThierry Reding static const struct tegra_dc_soc_info tegra20_dc_soc_info = { 18327116e9a8SThierry Reding .supports_background_color = false, 18338620fc62SThierry Reding .supports_interlacing = false, 1834e687651bSThierry Reding .supports_cursor = false, 1835c134f019SThierry Reding .supports_block_linear = false, 1836d1f3e1e0SThierry Reding .pitch_align = 8, 18379c012700SThierry Reding .has_powergate = false, 18386ac1571bSDmitry Osipenko .broken_reset = true, 183947307954SThierry Reding .has_nvdisplay = false, 1840*511c7023SThierry Reding .num_primary_formats = ARRAY_SIZE(tegra20_primary_formats), 1841*511c7023SThierry Reding .primary_formats = tegra20_primary_formats, 1842*511c7023SThierry Reding .num_overlay_formats = ARRAY_SIZE(tegra20_overlay_formats), 1843*511c7023SThierry Reding .overlay_formats = tegra20_overlay_formats, 18448620fc62SThierry Reding }; 18458620fc62SThierry Reding 18468620fc62SThierry Reding static const struct tegra_dc_soc_info tegra30_dc_soc_info = { 18477116e9a8SThierry Reding .supports_background_color = false, 18488620fc62SThierry Reding .supports_interlacing = false, 1849e687651bSThierry Reding .supports_cursor = false, 1850c134f019SThierry Reding .supports_block_linear = false, 1851d1f3e1e0SThierry Reding .pitch_align = 8, 18529c012700SThierry Reding .has_powergate = false, 18536ac1571bSDmitry Osipenko .broken_reset = false, 185447307954SThierry Reding .has_nvdisplay = false, 1855*511c7023SThierry Reding .num_primary_formats = ARRAY_SIZE(tegra20_primary_formats), 1856*511c7023SThierry Reding .primary_formats = tegra20_primary_formats, 1857*511c7023SThierry Reding .num_overlay_formats = ARRAY_SIZE(tegra20_overlay_formats), 1858*511c7023SThierry Reding .overlay_formats = tegra20_overlay_formats, 1859d1f3e1e0SThierry Reding }; 1860d1f3e1e0SThierry Reding 1861d1f3e1e0SThierry Reding static const struct tegra_dc_soc_info tegra114_dc_soc_info = { 18627116e9a8SThierry Reding .supports_background_color = false, 1863d1f3e1e0SThierry Reding .supports_interlacing = false, 1864d1f3e1e0SThierry Reding .supports_cursor = false, 1865d1f3e1e0SThierry Reding .supports_block_linear = false, 1866d1f3e1e0SThierry Reding .pitch_align = 64, 18679c012700SThierry Reding .has_powergate = true, 18686ac1571bSDmitry Osipenko .broken_reset = false, 186947307954SThierry Reding .has_nvdisplay = false, 1870*511c7023SThierry Reding .num_primary_formats = ARRAY_SIZE(tegra114_primary_formats), 1871*511c7023SThierry Reding .primary_formats = tegra114_primary_formats, 1872*511c7023SThierry Reding .num_overlay_formats = ARRAY_SIZE(tegra114_overlay_formats), 1873*511c7023SThierry Reding .overlay_formats = tegra114_overlay_formats, 18748620fc62SThierry Reding }; 18758620fc62SThierry Reding 18768620fc62SThierry Reding static const struct tegra_dc_soc_info tegra124_dc_soc_info = { 18777116e9a8SThierry Reding .supports_background_color = true, 18788620fc62SThierry Reding .supports_interlacing = true, 1879e687651bSThierry Reding .supports_cursor = true, 1880c134f019SThierry Reding .supports_block_linear = true, 1881d1f3e1e0SThierry Reding .pitch_align = 64, 18829c012700SThierry Reding .has_powergate = true, 18836ac1571bSDmitry Osipenko .broken_reset = false, 188447307954SThierry Reding .has_nvdisplay = false, 1885*511c7023SThierry Reding .num_primary_formats = ARRAY_SIZE(tegra124_primary_formats), 1886*511c7023SThierry Reding .primary_formats = tegra114_primary_formats, 1887*511c7023SThierry Reding .num_overlay_formats = ARRAY_SIZE(tegra124_overlay_formats), 1888*511c7023SThierry Reding .overlay_formats = tegra114_overlay_formats, 18898620fc62SThierry Reding }; 18908620fc62SThierry Reding 18915b4f516fSThierry Reding static const struct tegra_dc_soc_info tegra210_dc_soc_info = { 18927116e9a8SThierry Reding .supports_background_color = true, 18935b4f516fSThierry Reding .supports_interlacing = true, 18945b4f516fSThierry Reding .supports_cursor = true, 18955b4f516fSThierry Reding .supports_block_linear = true, 18965b4f516fSThierry Reding .pitch_align = 64, 18975b4f516fSThierry Reding .has_powergate = true, 18986ac1571bSDmitry Osipenko .broken_reset = false, 189947307954SThierry Reding .has_nvdisplay = false, 1900*511c7023SThierry Reding .num_primary_formats = ARRAY_SIZE(tegra114_primary_formats), 1901*511c7023SThierry Reding .primary_formats = tegra114_primary_formats, 1902*511c7023SThierry Reding .num_overlay_formats = ARRAY_SIZE(tegra114_overlay_formats), 1903*511c7023SThierry Reding .overlay_formats = tegra114_overlay_formats, 190447307954SThierry Reding }; 190547307954SThierry Reding 190647307954SThierry Reding static const struct tegra_windowgroup_soc tegra186_dc_wgrps[] = { 190747307954SThierry Reding { 190847307954SThierry Reding .index = 0, 190947307954SThierry Reding .dc = 0, 191047307954SThierry Reding .windows = (const unsigned int[]) { 0 }, 191147307954SThierry Reding .num_windows = 1, 191247307954SThierry Reding }, { 191347307954SThierry Reding .index = 1, 191447307954SThierry Reding .dc = 1, 191547307954SThierry Reding .windows = (const unsigned int[]) { 1 }, 191647307954SThierry Reding .num_windows = 1, 191747307954SThierry Reding }, { 191847307954SThierry Reding .index = 2, 191947307954SThierry Reding .dc = 1, 192047307954SThierry Reding .windows = (const unsigned int[]) { 2 }, 192147307954SThierry Reding .num_windows = 1, 192247307954SThierry Reding }, { 192347307954SThierry Reding .index = 3, 192447307954SThierry Reding .dc = 2, 192547307954SThierry Reding .windows = (const unsigned int[]) { 3 }, 192647307954SThierry Reding .num_windows = 1, 192747307954SThierry Reding }, { 192847307954SThierry Reding .index = 4, 192947307954SThierry Reding .dc = 2, 193047307954SThierry Reding .windows = (const unsigned int[]) { 4 }, 193147307954SThierry Reding .num_windows = 1, 193247307954SThierry Reding }, { 193347307954SThierry Reding .index = 5, 193447307954SThierry Reding .dc = 2, 193547307954SThierry Reding .windows = (const unsigned int[]) { 5 }, 193647307954SThierry Reding .num_windows = 1, 193747307954SThierry Reding }, 193847307954SThierry Reding }; 193947307954SThierry Reding 194047307954SThierry Reding static const struct tegra_dc_soc_info tegra186_dc_soc_info = { 194147307954SThierry Reding .supports_background_color = true, 194247307954SThierry Reding .supports_interlacing = true, 194347307954SThierry Reding .supports_cursor = true, 194447307954SThierry Reding .supports_block_linear = true, 194547307954SThierry Reding .pitch_align = 64, 194647307954SThierry Reding .has_powergate = false, 194747307954SThierry Reding .broken_reset = false, 194847307954SThierry Reding .has_nvdisplay = true, 194947307954SThierry Reding .wgrps = tegra186_dc_wgrps, 195047307954SThierry Reding .num_wgrps = ARRAY_SIZE(tegra186_dc_wgrps), 19515b4f516fSThierry Reding }; 19525b4f516fSThierry Reding 19538620fc62SThierry Reding static const struct of_device_id tegra_dc_of_match[] = { 19548620fc62SThierry Reding { 195547307954SThierry Reding .compatible = "nvidia,tegra186-dc", 195647307954SThierry Reding .data = &tegra186_dc_soc_info, 195747307954SThierry Reding }, { 19585b4f516fSThierry Reding .compatible = "nvidia,tegra210-dc", 19595b4f516fSThierry Reding .data = &tegra210_dc_soc_info, 19605b4f516fSThierry Reding }, { 19618620fc62SThierry Reding .compatible = "nvidia,tegra124-dc", 19628620fc62SThierry Reding .data = &tegra124_dc_soc_info, 19638620fc62SThierry Reding }, { 19649c012700SThierry Reding .compatible = "nvidia,tegra114-dc", 19659c012700SThierry Reding .data = &tegra114_dc_soc_info, 19669c012700SThierry Reding }, { 19678620fc62SThierry Reding .compatible = "nvidia,tegra30-dc", 19688620fc62SThierry Reding .data = &tegra30_dc_soc_info, 19698620fc62SThierry Reding }, { 19708620fc62SThierry Reding .compatible = "nvidia,tegra20-dc", 19718620fc62SThierry Reding .data = &tegra20_dc_soc_info, 19728620fc62SThierry Reding }, { 19738620fc62SThierry Reding /* sentinel */ 19748620fc62SThierry Reding } 19758620fc62SThierry Reding }; 1976ef70728cSStephen Warren MODULE_DEVICE_TABLE(of, tegra_dc_of_match); 19778620fc62SThierry Reding 197813411dddSThierry Reding static int tegra_dc_parse_dt(struct tegra_dc *dc) 197913411dddSThierry Reding { 198013411dddSThierry Reding struct device_node *np; 198113411dddSThierry Reding u32 value = 0; 198213411dddSThierry Reding int err; 198313411dddSThierry Reding 198413411dddSThierry Reding err = of_property_read_u32(dc->dev->of_node, "nvidia,head", &value); 198513411dddSThierry Reding if (err < 0) { 198613411dddSThierry Reding dev_err(dc->dev, "missing \"nvidia,head\" property\n"); 198713411dddSThierry Reding 198813411dddSThierry Reding /* 198913411dddSThierry Reding * If the nvidia,head property isn't present, try to find the 199013411dddSThierry Reding * correct head number by looking up the position of this 199113411dddSThierry Reding * display controller's node within the device tree. Assuming 199213411dddSThierry Reding * that the nodes are ordered properly in the DTS file and 199313411dddSThierry Reding * that the translation into a flattened device tree blob 199413411dddSThierry Reding * preserves that ordering this will actually yield the right 199513411dddSThierry Reding * head number. 199613411dddSThierry Reding * 199713411dddSThierry Reding * If those assumptions don't hold, this will still work for 199813411dddSThierry Reding * cases where only a single display controller is used. 199913411dddSThierry Reding */ 200013411dddSThierry Reding for_each_matching_node(np, tegra_dc_of_match) { 2001cf6b1744SJulia Lawall if (np == dc->dev->of_node) { 2002cf6b1744SJulia Lawall of_node_put(np); 200313411dddSThierry Reding break; 2004cf6b1744SJulia Lawall } 200513411dddSThierry Reding 200613411dddSThierry Reding value++; 200713411dddSThierry Reding } 200813411dddSThierry Reding } 200913411dddSThierry Reding 201013411dddSThierry Reding dc->pipe = value; 201113411dddSThierry Reding 201213411dddSThierry Reding return 0; 201313411dddSThierry Reding } 201413411dddSThierry Reding 2015dee8268fSThierry Reding static int tegra_dc_probe(struct platform_device *pdev) 2016dee8268fSThierry Reding { 2017dee8268fSThierry Reding struct resource *regs; 2018dee8268fSThierry Reding struct tegra_dc *dc; 2019dee8268fSThierry Reding int err; 2020dee8268fSThierry Reding 2021dee8268fSThierry Reding dc = devm_kzalloc(&pdev->dev, sizeof(*dc), GFP_KERNEL); 2022dee8268fSThierry Reding if (!dc) 2023dee8268fSThierry Reding return -ENOMEM; 2024dee8268fSThierry Reding 2025b9ff7aeaSThierry Reding dc->soc = of_device_get_match_data(&pdev->dev); 20268620fc62SThierry Reding 2027dee8268fSThierry Reding spin_lock_init(&dc->lock); 2028dee8268fSThierry Reding INIT_LIST_HEAD(&dc->list); 2029dee8268fSThierry Reding dc->dev = &pdev->dev; 2030dee8268fSThierry Reding 203113411dddSThierry Reding err = tegra_dc_parse_dt(dc); 203213411dddSThierry Reding if (err < 0) 203313411dddSThierry Reding return err; 203413411dddSThierry Reding 2035dee8268fSThierry Reding dc->clk = devm_clk_get(&pdev->dev, NULL); 2036dee8268fSThierry Reding if (IS_ERR(dc->clk)) { 2037dee8268fSThierry Reding dev_err(&pdev->dev, "failed to get clock\n"); 2038dee8268fSThierry Reding return PTR_ERR(dc->clk); 2039dee8268fSThierry Reding } 2040dee8268fSThierry Reding 2041ca48080aSStephen Warren dc->rst = devm_reset_control_get(&pdev->dev, "dc"); 2042ca48080aSStephen Warren if (IS_ERR(dc->rst)) { 2043ca48080aSStephen Warren dev_err(&pdev->dev, "failed to get reset\n"); 2044ca48080aSStephen Warren return PTR_ERR(dc->rst); 2045ca48080aSStephen Warren } 2046ca48080aSStephen Warren 2047a2f2f740SThierry Reding /* assert reset and disable clock */ 2048a2f2f740SThierry Reding if (!dc->soc->broken_reset) { 2049a2f2f740SThierry Reding err = clk_prepare_enable(dc->clk); 2050a2f2f740SThierry Reding if (err < 0) 2051a2f2f740SThierry Reding return err; 2052a2f2f740SThierry Reding 2053a2f2f740SThierry Reding usleep_range(2000, 4000); 2054a2f2f740SThierry Reding 2055a2f2f740SThierry Reding err = reset_control_assert(dc->rst); 2056a2f2f740SThierry Reding if (err < 0) 2057a2f2f740SThierry Reding return err; 2058a2f2f740SThierry Reding 2059a2f2f740SThierry Reding usleep_range(2000, 4000); 2060a2f2f740SThierry Reding 2061a2f2f740SThierry Reding clk_disable_unprepare(dc->clk); 2062a2f2f740SThierry Reding } 206333a8eb8dSThierry Reding 20649c012700SThierry Reding if (dc->soc->has_powergate) { 20659c012700SThierry Reding if (dc->pipe == 0) 20669c012700SThierry Reding dc->powergate = TEGRA_POWERGATE_DIS; 20679c012700SThierry Reding else 20689c012700SThierry Reding dc->powergate = TEGRA_POWERGATE_DISB; 20699c012700SThierry Reding 207033a8eb8dSThierry Reding tegra_powergate_power_off(dc->powergate); 20719c012700SThierry Reding } 2072dee8268fSThierry Reding 2073dee8268fSThierry Reding regs = platform_get_resource(pdev, IORESOURCE_MEM, 0); 2074dee8268fSThierry Reding dc->regs = devm_ioremap_resource(&pdev->dev, regs); 2075dee8268fSThierry Reding if (IS_ERR(dc->regs)) 2076dee8268fSThierry Reding return PTR_ERR(dc->regs); 2077dee8268fSThierry Reding 2078dee8268fSThierry Reding dc->irq = platform_get_irq(pdev, 0); 2079dee8268fSThierry Reding if (dc->irq < 0) { 2080dee8268fSThierry Reding dev_err(&pdev->dev, "failed to get IRQ\n"); 2081dee8268fSThierry Reding return -ENXIO; 2082dee8268fSThierry Reding } 2083dee8268fSThierry Reding 2084dee8268fSThierry Reding err = tegra_dc_rgb_probe(dc); 2085dee8268fSThierry Reding if (err < 0 && err != -ENODEV) { 2086dee8268fSThierry Reding dev_err(&pdev->dev, "failed to probe RGB output: %d\n", err); 2087dee8268fSThierry Reding return err; 2088dee8268fSThierry Reding } 2089dee8268fSThierry Reding 209033a8eb8dSThierry Reding platform_set_drvdata(pdev, dc); 209133a8eb8dSThierry Reding pm_runtime_enable(&pdev->dev); 209233a8eb8dSThierry Reding 209333a8eb8dSThierry Reding INIT_LIST_HEAD(&dc->client.list); 209433a8eb8dSThierry Reding dc->client.ops = &dc_client_ops; 209533a8eb8dSThierry Reding dc->client.dev = &pdev->dev; 209633a8eb8dSThierry Reding 2097dee8268fSThierry Reding err = host1x_client_register(&dc->client); 2098dee8268fSThierry Reding if (err < 0) { 2099dee8268fSThierry Reding dev_err(&pdev->dev, "failed to register host1x client: %d\n", 2100dee8268fSThierry Reding err); 2101dee8268fSThierry Reding return err; 2102dee8268fSThierry Reding } 2103dee8268fSThierry Reding 2104dee8268fSThierry Reding return 0; 2105dee8268fSThierry Reding } 2106dee8268fSThierry Reding 2107dee8268fSThierry Reding static int tegra_dc_remove(struct platform_device *pdev) 2108dee8268fSThierry Reding { 2109dee8268fSThierry Reding struct tegra_dc *dc = platform_get_drvdata(pdev); 2110dee8268fSThierry Reding int err; 2111dee8268fSThierry Reding 2112dee8268fSThierry Reding err = host1x_client_unregister(&dc->client); 2113dee8268fSThierry Reding if (err < 0) { 2114dee8268fSThierry Reding dev_err(&pdev->dev, "failed to unregister host1x client: %d\n", 2115dee8268fSThierry Reding err); 2116dee8268fSThierry Reding return err; 2117dee8268fSThierry Reding } 2118dee8268fSThierry Reding 211959d29c0eSThierry Reding err = tegra_dc_rgb_remove(dc); 212059d29c0eSThierry Reding if (err < 0) { 212159d29c0eSThierry Reding dev_err(&pdev->dev, "failed to remove RGB output: %d\n", err); 212259d29c0eSThierry Reding return err; 212359d29c0eSThierry Reding } 212459d29c0eSThierry Reding 212533a8eb8dSThierry Reding pm_runtime_disable(&pdev->dev); 212633a8eb8dSThierry Reding 212733a8eb8dSThierry Reding return 0; 212833a8eb8dSThierry Reding } 212933a8eb8dSThierry Reding 213033a8eb8dSThierry Reding #ifdef CONFIG_PM 213133a8eb8dSThierry Reding static int tegra_dc_suspend(struct device *dev) 213233a8eb8dSThierry Reding { 213333a8eb8dSThierry Reding struct tegra_dc *dc = dev_get_drvdata(dev); 213433a8eb8dSThierry Reding int err; 213533a8eb8dSThierry Reding 21366ac1571bSDmitry Osipenko if (!dc->soc->broken_reset) { 213733a8eb8dSThierry Reding err = reset_control_assert(dc->rst); 213833a8eb8dSThierry Reding if (err < 0) { 213933a8eb8dSThierry Reding dev_err(dev, "failed to assert reset: %d\n", err); 214033a8eb8dSThierry Reding return err; 214133a8eb8dSThierry Reding } 21426ac1571bSDmitry Osipenko } 21439c012700SThierry Reding 21449c012700SThierry Reding if (dc->soc->has_powergate) 21459c012700SThierry Reding tegra_powergate_power_off(dc->powergate); 21469c012700SThierry Reding 2147dee8268fSThierry Reding clk_disable_unprepare(dc->clk); 2148dee8268fSThierry Reding 2149dee8268fSThierry Reding return 0; 2150dee8268fSThierry Reding } 2151dee8268fSThierry Reding 215233a8eb8dSThierry Reding static int tegra_dc_resume(struct device *dev) 215333a8eb8dSThierry Reding { 215433a8eb8dSThierry Reding struct tegra_dc *dc = dev_get_drvdata(dev); 215533a8eb8dSThierry Reding int err; 215633a8eb8dSThierry Reding 215733a8eb8dSThierry Reding if (dc->soc->has_powergate) { 215833a8eb8dSThierry Reding err = tegra_powergate_sequence_power_up(dc->powergate, dc->clk, 215933a8eb8dSThierry Reding dc->rst); 216033a8eb8dSThierry Reding if (err < 0) { 216133a8eb8dSThierry Reding dev_err(dev, "failed to power partition: %d\n", err); 216233a8eb8dSThierry Reding return err; 216333a8eb8dSThierry Reding } 216433a8eb8dSThierry Reding } else { 216533a8eb8dSThierry Reding err = clk_prepare_enable(dc->clk); 216633a8eb8dSThierry Reding if (err < 0) { 216733a8eb8dSThierry Reding dev_err(dev, "failed to enable clock: %d\n", err); 216833a8eb8dSThierry Reding return err; 216933a8eb8dSThierry Reding } 217033a8eb8dSThierry Reding 21716ac1571bSDmitry Osipenko if (!dc->soc->broken_reset) { 217233a8eb8dSThierry Reding err = reset_control_deassert(dc->rst); 217333a8eb8dSThierry Reding if (err < 0) { 21746ac1571bSDmitry Osipenko dev_err(dev, 21756ac1571bSDmitry Osipenko "failed to deassert reset: %d\n", err); 217633a8eb8dSThierry Reding return err; 217733a8eb8dSThierry Reding } 217833a8eb8dSThierry Reding } 21796ac1571bSDmitry Osipenko } 218033a8eb8dSThierry Reding 218133a8eb8dSThierry Reding return 0; 218233a8eb8dSThierry Reding } 218333a8eb8dSThierry Reding #endif 218433a8eb8dSThierry Reding 218533a8eb8dSThierry Reding static const struct dev_pm_ops tegra_dc_pm_ops = { 218633a8eb8dSThierry Reding SET_RUNTIME_PM_OPS(tegra_dc_suspend, tegra_dc_resume, NULL) 218733a8eb8dSThierry Reding }; 218833a8eb8dSThierry Reding 2189dee8268fSThierry Reding struct platform_driver tegra_dc_driver = { 2190dee8268fSThierry Reding .driver = { 2191dee8268fSThierry Reding .name = "tegra-dc", 2192dee8268fSThierry Reding .of_match_table = tegra_dc_of_match, 219333a8eb8dSThierry Reding .pm = &tegra_dc_pm_ops, 2194dee8268fSThierry Reding }, 2195dee8268fSThierry Reding .probe = tegra_dc_probe, 2196dee8268fSThierry Reding .remove = tegra_dc_remove, 2197dee8268fSThierry Reding }; 2198