1 // SPDX-License-Identifier: GPL-2.0-only
2 /*
3 * (C) COPYRIGHT 2016 ARM Limited. All rights reserved.
4 * Author: Liviu Dudau <Liviu.Dudau@arm.com>
5 *
6 * ARM Mali DP500/DP550/DP650 hardware manipulation routines. This is where
7 * the difference between various versions of the hardware is being dealt with
8 * in an attempt to provide to the rest of the driver code a unified view
9 */
10
11 #include <linux/clk.h>
12 #include <linux/delay.h>
13 #include <linux/types.h>
14 #include <linux/io.h>
15
16 #include <video/videomode.h>
17 #include <video/display_timing.h>
18
19 #include <drm/drm_fourcc.h>
20 #include <drm/drm_vblank.h>
21 #include <drm/drm_print.h>
22
23 #include "malidp_drv.h"
24 #include "malidp_hw.h"
25 #include "malidp_mw.h"
26
27 enum {
28 MW_NOT_ENABLED = 0, /* SE writeback not enabled */
29 MW_ONESHOT, /* SE in one-shot mode for writeback */
30 MW_START, /* SE started writeback */
31 MW_RESTART, /* SE will start another writeback after this one */
32 MW_STOP, /* SE needs to stop after this writeback */
33 };
34
35 static const struct malidp_format_id malidp500_de_formats[] = {
36 /* fourcc, layers supporting the format, internal id */
37 { DRM_FORMAT_ARGB2101010, DE_VIDEO1 | DE_GRAPHICS1 | DE_GRAPHICS2 | SE_MEMWRITE, 0 },
38 { DRM_FORMAT_ABGR2101010, DE_VIDEO1 | DE_GRAPHICS1 | DE_GRAPHICS2 | SE_MEMWRITE, 1 },
39 { DRM_FORMAT_ARGB8888, DE_VIDEO1 | DE_GRAPHICS1 | DE_GRAPHICS2, 2 },
40 { DRM_FORMAT_ABGR8888, DE_VIDEO1 | DE_GRAPHICS1 | DE_GRAPHICS2, 3 },
41 { DRM_FORMAT_XRGB8888, DE_VIDEO1 | DE_GRAPHICS1 | DE_GRAPHICS2 | SE_MEMWRITE, 4 },
42 { DRM_FORMAT_XBGR8888, DE_VIDEO1 | DE_GRAPHICS1 | DE_GRAPHICS2 | SE_MEMWRITE, 5 },
43 { DRM_FORMAT_RGB888, DE_VIDEO1 | DE_GRAPHICS1 | DE_GRAPHICS2, 6 },
44 { DRM_FORMAT_BGR888, DE_VIDEO1 | DE_GRAPHICS1 | DE_GRAPHICS2, 7 },
45 { DRM_FORMAT_RGBA5551, DE_VIDEO1 | DE_GRAPHICS1 | DE_GRAPHICS2, 8 },
46 { DRM_FORMAT_ABGR1555, DE_VIDEO1 | DE_GRAPHICS1 | DE_GRAPHICS2, 9 },
47 { DRM_FORMAT_RGB565, DE_VIDEO1 | DE_GRAPHICS1 | DE_GRAPHICS2, 10 },
48 { DRM_FORMAT_BGR565, DE_VIDEO1 | DE_GRAPHICS1 | DE_GRAPHICS2, 11 },
49 { DRM_FORMAT_UYVY, DE_VIDEO1, 12 },
50 { DRM_FORMAT_YUYV, DE_VIDEO1, 13 },
51 { DRM_FORMAT_NV12, DE_VIDEO1 | SE_MEMWRITE, 14 },
52 { DRM_FORMAT_YUV420, DE_VIDEO1, 15 },
53 { DRM_FORMAT_XYUV8888, DE_VIDEO1, 16 },
54 /* These are supported with AFBC only */
55 { DRM_FORMAT_YUV420_8BIT, DE_VIDEO1, 14 },
56 { DRM_FORMAT_VUY888, DE_VIDEO1, 16 },
57 { DRM_FORMAT_VUY101010, DE_VIDEO1, 17 },
58 { DRM_FORMAT_YUV420_10BIT, DE_VIDEO1, 18 }
59 };
60
61 #define MALIDP_ID(__group, __format) \
62 ((((__group) & 0x7) << 3) | ((__format) & 0x7))
63
64 #define AFBC_YUV_422_FORMAT_ID MALIDP_ID(5, 1)
65
66 #define MALIDP_COMMON_FORMATS \
67 /* fourcc, layers supporting the format, internal id */ \
68 { DRM_FORMAT_ARGB2101010, DE_VIDEO1 | DE_GRAPHICS1 | DE_VIDEO2 | SE_MEMWRITE, MALIDP_ID(0, 0) }, \
69 { DRM_FORMAT_ABGR2101010, DE_VIDEO1 | DE_GRAPHICS1 | DE_VIDEO2 | SE_MEMWRITE, MALIDP_ID(0, 1) }, \
70 { DRM_FORMAT_RGBA1010102, DE_VIDEO1 | DE_GRAPHICS1 | DE_VIDEO2 | SE_MEMWRITE, MALIDP_ID(0, 2) }, \
71 { DRM_FORMAT_BGRA1010102, DE_VIDEO1 | DE_GRAPHICS1 | DE_VIDEO2 | SE_MEMWRITE, MALIDP_ID(0, 3) }, \
72 { DRM_FORMAT_ARGB8888, DE_VIDEO1 | DE_GRAPHICS1 | DE_VIDEO2 | DE_SMART, MALIDP_ID(1, 0) }, \
73 { DRM_FORMAT_ABGR8888, DE_VIDEO1 | DE_GRAPHICS1 | DE_VIDEO2 | DE_SMART, MALIDP_ID(1, 1) }, \
74 { DRM_FORMAT_RGBA8888, DE_VIDEO1 | DE_GRAPHICS1 | DE_VIDEO2 | DE_SMART, MALIDP_ID(1, 2) }, \
75 { DRM_FORMAT_BGRA8888, DE_VIDEO1 | DE_GRAPHICS1 | DE_VIDEO2 | DE_SMART, MALIDP_ID(1, 3) }, \
76 { DRM_FORMAT_XRGB8888, DE_VIDEO1 | DE_GRAPHICS1 | DE_VIDEO2 | DE_SMART | SE_MEMWRITE, MALIDP_ID(2, 0) }, \
77 { DRM_FORMAT_XBGR8888, DE_VIDEO1 | DE_GRAPHICS1 | DE_VIDEO2 | DE_SMART | SE_MEMWRITE, MALIDP_ID(2, 1) }, \
78 { DRM_FORMAT_RGBX8888, DE_VIDEO1 | DE_GRAPHICS1 | DE_VIDEO2 | DE_SMART | SE_MEMWRITE, MALIDP_ID(2, 2) }, \
79 { DRM_FORMAT_BGRX8888, DE_VIDEO1 | DE_GRAPHICS1 | DE_VIDEO2 | DE_SMART | SE_MEMWRITE, MALIDP_ID(2, 3) }, \
80 { DRM_FORMAT_RGB888, DE_VIDEO1 | DE_GRAPHICS1 | DE_VIDEO2 | SE_MEMWRITE, MALIDP_ID(3, 0) }, \
81 { DRM_FORMAT_BGR888, DE_VIDEO1 | DE_GRAPHICS1 | DE_VIDEO2 | SE_MEMWRITE, MALIDP_ID(3, 1) }, \
82 { DRM_FORMAT_RGBA5551, DE_VIDEO1 | DE_GRAPHICS1 | DE_VIDEO2, MALIDP_ID(4, 0) }, \
83 { DRM_FORMAT_ABGR1555, DE_VIDEO1 | DE_GRAPHICS1 | DE_VIDEO2, MALIDP_ID(4, 1) }, \
84 { DRM_FORMAT_RGB565, DE_VIDEO1 | DE_GRAPHICS1 | DE_VIDEO2, MALIDP_ID(4, 2) }, \
85 { DRM_FORMAT_BGR565, DE_VIDEO1 | DE_GRAPHICS1 | DE_VIDEO2, MALIDP_ID(4, 3) }, \
86 /* This is only supported with linear modifier */ \
87 { DRM_FORMAT_XYUV8888, DE_VIDEO1 | DE_VIDEO2, MALIDP_ID(5, 0) },\
88 /* This is only supported with AFBC modifier */ \
89 { DRM_FORMAT_VUY888, DE_VIDEO1 | DE_VIDEO2, MALIDP_ID(5, 0) }, \
90 { DRM_FORMAT_YUYV, DE_VIDEO1 | DE_VIDEO2, MALIDP_ID(5, 2) }, \
91 /* This is only supported with linear modifier */ \
92 { DRM_FORMAT_UYVY, DE_VIDEO1 | DE_VIDEO2, MALIDP_ID(5, 3) }, \
93 { DRM_FORMAT_NV12, DE_VIDEO1 | DE_VIDEO2 | SE_MEMWRITE, MALIDP_ID(5, 6) }, \
94 /* This is only supported with AFBC modifier */ \
95 { DRM_FORMAT_YUV420_8BIT, DE_VIDEO1 | DE_VIDEO2, MALIDP_ID(5, 6) }, \
96 { DRM_FORMAT_YUV420, DE_VIDEO1 | DE_VIDEO2, MALIDP_ID(5, 7) }, \
97 /* This is only supported with linear modifier */ \
98 { DRM_FORMAT_XVYU2101010, DE_VIDEO1 | DE_VIDEO2, MALIDP_ID(6, 0)}, \
99 /* This is only supported with AFBC modifier */ \
100 { DRM_FORMAT_VUY101010, DE_VIDEO1 | DE_VIDEO2, MALIDP_ID(6, 0)}, \
101 { DRM_FORMAT_X0L2, DE_VIDEO1 | DE_VIDEO2, MALIDP_ID(6, 6)}, \
102 /* This is only supported with AFBC modifier */ \
103 { DRM_FORMAT_YUV420_10BIT, DE_VIDEO1 | DE_VIDEO2, MALIDP_ID(6, 7)}, \
104 { DRM_FORMAT_P010, DE_VIDEO1 | DE_VIDEO2, MALIDP_ID(6, 7)}
105
106 static const struct malidp_format_id malidp550_de_formats[] = {
107 MALIDP_COMMON_FORMATS,
108 };
109
110 static const struct malidp_format_id malidp650_de_formats[] = {
111 MALIDP_COMMON_FORMATS,
112 { DRM_FORMAT_X0L0, DE_VIDEO1 | DE_VIDEO2, MALIDP_ID(5, 4)},
113 };
114
115 static const struct malidp_layer malidp500_layers[] = {
116 /* id, base address, fb pointer address base, stride offset,
117 * yuv2rgb matrix offset, mmu control register offset, rotation_features
118 */
119 { DE_VIDEO1, MALIDP500_DE_LV_BASE, MALIDP500_DE_LV_PTR_BASE,
120 MALIDP_DE_LV_STRIDE0, MALIDP500_LV_YUV2RGB, 0, ROTATE_ANY,
121 MALIDP500_DE_LV_AD_CTRL },
122 { DE_GRAPHICS1, MALIDP500_DE_LG1_BASE, MALIDP500_DE_LG1_PTR_BASE,
123 MALIDP_DE_LG_STRIDE, 0, 0, ROTATE_ANY,
124 MALIDP500_DE_LG1_AD_CTRL },
125 { DE_GRAPHICS2, MALIDP500_DE_LG2_BASE, MALIDP500_DE_LG2_PTR_BASE,
126 MALIDP_DE_LG_STRIDE, 0, 0, ROTATE_ANY,
127 MALIDP500_DE_LG2_AD_CTRL },
128 };
129
130 static const struct malidp_layer malidp550_layers[] = {
131 /* id, base address, fb pointer address base, stride offset,
132 * yuv2rgb matrix offset, mmu control register offset, rotation_features
133 */
134 { DE_VIDEO1, MALIDP550_DE_LV1_BASE, MALIDP550_DE_LV1_PTR_BASE,
135 MALIDP_DE_LV_STRIDE0, MALIDP550_LV_YUV2RGB, 0, ROTATE_ANY,
136 MALIDP550_DE_LV1_AD_CTRL },
137 { DE_GRAPHICS1, MALIDP550_DE_LG_BASE, MALIDP550_DE_LG_PTR_BASE,
138 MALIDP_DE_LG_STRIDE, 0, 0, ROTATE_ANY,
139 MALIDP550_DE_LG_AD_CTRL },
140 { DE_VIDEO2, MALIDP550_DE_LV2_BASE, MALIDP550_DE_LV2_PTR_BASE,
141 MALIDP_DE_LV_STRIDE0, MALIDP550_LV_YUV2RGB, 0, ROTATE_ANY,
142 MALIDP550_DE_LV2_AD_CTRL },
143 { DE_SMART, MALIDP550_DE_LS_BASE, MALIDP550_DE_LS_PTR_BASE,
144 MALIDP550_DE_LS_R1_STRIDE, 0, 0, ROTATE_NONE, 0 },
145 };
146
147 static const struct malidp_layer malidp650_layers[] = {
148 /* id, base address, fb pointer address base, stride offset,
149 * yuv2rgb matrix offset, mmu control register offset,
150 * rotation_features
151 */
152 { DE_VIDEO1, MALIDP550_DE_LV1_BASE, MALIDP550_DE_LV1_PTR_BASE,
153 MALIDP_DE_LV_STRIDE0, MALIDP550_LV_YUV2RGB,
154 MALIDP650_DE_LV_MMU_CTRL, ROTATE_ANY,
155 MALIDP550_DE_LV1_AD_CTRL },
156 { DE_GRAPHICS1, MALIDP550_DE_LG_BASE, MALIDP550_DE_LG_PTR_BASE,
157 MALIDP_DE_LG_STRIDE, 0, MALIDP650_DE_LG_MMU_CTRL,
158 ROTATE_COMPRESSED, MALIDP550_DE_LG_AD_CTRL },
159 { DE_VIDEO2, MALIDP550_DE_LV2_BASE, MALIDP550_DE_LV2_PTR_BASE,
160 MALIDP_DE_LV_STRIDE0, MALIDP550_LV_YUV2RGB,
161 MALIDP650_DE_LV_MMU_CTRL, ROTATE_ANY,
162 MALIDP550_DE_LV2_AD_CTRL },
163 { DE_SMART, MALIDP550_DE_LS_BASE, MALIDP550_DE_LS_PTR_BASE,
164 MALIDP550_DE_LS_R1_STRIDE, 0, MALIDP650_DE_LS_MMU_CTRL,
165 ROTATE_NONE, 0 },
166 };
167
168 const u64 malidp_format_modifiers[] = {
169 /* All RGB formats (except XRGB, RGBX, XBGR, BGRX) */
170 DRM_FORMAT_MOD_ARM_AFBC(AFBC_SIZE_16X16 | AFBC_YTR | AFBC_SPARSE),
171 DRM_FORMAT_MOD_ARM_AFBC(AFBC_SIZE_16X16 | AFBC_YTR),
172
173 /* All RGB formats > 16bpp (except XRGB, RGBX, XBGR, BGRX) */
174 DRM_FORMAT_MOD_ARM_AFBC(AFBC_SIZE_16X16 | AFBC_YTR | AFBC_SPARSE | AFBC_SPLIT),
175
176 /* All 8 or 10 bit YUV 444 formats. */
177 /* In DP550, 10 bit YUV 420 format also supported */
178 DRM_FORMAT_MOD_ARM_AFBC(AFBC_SIZE_16X16 | AFBC_SPARSE | AFBC_SPLIT),
179
180 /* YUV 420, 422 P1 8 bit and YUV 444 8 bit/10 bit formats */
181 DRM_FORMAT_MOD_ARM_AFBC(AFBC_SIZE_16X16 | AFBC_SPARSE),
182 DRM_FORMAT_MOD_ARM_AFBC(AFBC_SIZE_16X16),
183
184 /* YUV 420, 422 P1 8, 10 bit formats */
185 DRM_FORMAT_MOD_ARM_AFBC(AFBC_SIZE_16X16 | AFBC_CBR | AFBC_SPARSE),
186 DRM_FORMAT_MOD_ARM_AFBC(AFBC_SIZE_16X16 | AFBC_CBR),
187
188 /* All formats */
189 DRM_FORMAT_MOD_LINEAR,
190
191 DRM_FORMAT_MOD_INVALID
192 };
193
194 #define SE_N_SCALING_COEFFS 96
195 static const u16 dp500_se_scaling_coeffs[][SE_N_SCALING_COEFFS] = {
196 [MALIDP_UPSCALING_COEFFS - 1] = {
197 0x0000, 0x0001, 0x0007, 0x0011, 0x001e, 0x002e, 0x003f, 0x0052,
198 0x0064, 0x0073, 0x007d, 0x0080, 0x007a, 0x006c, 0x0053, 0x002f,
199 0x0000, 0x3fc6, 0x3f83, 0x3f39, 0x3eea, 0x3e9b, 0x3e4f, 0x3e0a,
200 0x3dd4, 0x3db0, 0x3da2, 0x3db1, 0x3dde, 0x3e2f, 0x3ea5, 0x3f40,
201 0x0000, 0x00e5, 0x01ee, 0x0315, 0x0456, 0x05aa, 0x0709, 0x086c,
202 0x09c9, 0x0b15, 0x0c4a, 0x0d5d, 0x0e4a, 0x0f06, 0x0f91, 0x0fe5,
203 0x1000, 0x0fe5, 0x0f91, 0x0f06, 0x0e4a, 0x0d5d, 0x0c4a, 0x0b15,
204 0x09c9, 0x086c, 0x0709, 0x05aa, 0x0456, 0x0315, 0x01ee, 0x00e5,
205 0x0000, 0x3f40, 0x3ea5, 0x3e2f, 0x3dde, 0x3db1, 0x3da2, 0x3db0,
206 0x3dd4, 0x3e0a, 0x3e4f, 0x3e9b, 0x3eea, 0x3f39, 0x3f83, 0x3fc6,
207 0x0000, 0x002f, 0x0053, 0x006c, 0x007a, 0x0080, 0x007d, 0x0073,
208 0x0064, 0x0052, 0x003f, 0x002e, 0x001e, 0x0011, 0x0007, 0x0001
209 },
210 [MALIDP_DOWNSCALING_1_5_COEFFS - 1] = {
211 0x0059, 0x004f, 0x0041, 0x002e, 0x0016, 0x3ffb, 0x3fd9, 0x3fb4,
212 0x3f8c, 0x3f62, 0x3f36, 0x3f09, 0x3edd, 0x3eb3, 0x3e8d, 0x3e6c,
213 0x3e52, 0x3e3f, 0x3e35, 0x3e37, 0x3e46, 0x3e61, 0x3e8c, 0x3ec5,
214 0x3f0f, 0x3f68, 0x3fd1, 0x004a, 0x00d3, 0x0169, 0x020b, 0x02b8,
215 0x036e, 0x042d, 0x04f2, 0x05b9, 0x0681, 0x0745, 0x0803, 0x08ba,
216 0x0965, 0x0a03, 0x0a91, 0x0b0d, 0x0b75, 0x0bc6, 0x0c00, 0x0c20,
217 0x0c28, 0x0c20, 0x0c00, 0x0bc6, 0x0b75, 0x0b0d, 0x0a91, 0x0a03,
218 0x0965, 0x08ba, 0x0803, 0x0745, 0x0681, 0x05b9, 0x04f2, 0x042d,
219 0x036e, 0x02b8, 0x020b, 0x0169, 0x00d3, 0x004a, 0x3fd1, 0x3f68,
220 0x3f0f, 0x3ec5, 0x3e8c, 0x3e61, 0x3e46, 0x3e37, 0x3e35, 0x3e3f,
221 0x3e52, 0x3e6c, 0x3e8d, 0x3eb3, 0x3edd, 0x3f09, 0x3f36, 0x3f62,
222 0x3f8c, 0x3fb4, 0x3fd9, 0x3ffb, 0x0016, 0x002e, 0x0041, 0x004f
223 },
224 [MALIDP_DOWNSCALING_2_COEFFS - 1] = {
225 0x3f19, 0x3f03, 0x3ef0, 0x3edf, 0x3ed0, 0x3ec5, 0x3ebd, 0x3eb9,
226 0x3eb9, 0x3ebf, 0x3eca, 0x3ed9, 0x3eef, 0x3f0a, 0x3f2c, 0x3f52,
227 0x3f7f, 0x3fb0, 0x3fe8, 0x0026, 0x006a, 0x00b4, 0x0103, 0x0158,
228 0x01b1, 0x020d, 0x026c, 0x02cd, 0x032f, 0x0392, 0x03f4, 0x0455,
229 0x04b4, 0x051e, 0x0585, 0x05eb, 0x064c, 0x06a8, 0x06fe, 0x074e,
230 0x0796, 0x07d5, 0x080c, 0x0839, 0x085c, 0x0875, 0x0882, 0x0887,
231 0x0881, 0x0887, 0x0882, 0x0875, 0x085c, 0x0839, 0x080c, 0x07d5,
232 0x0796, 0x074e, 0x06fe, 0x06a8, 0x064c, 0x05eb, 0x0585, 0x051e,
233 0x04b4, 0x0455, 0x03f4, 0x0392, 0x032f, 0x02cd, 0x026c, 0x020d,
234 0x01b1, 0x0158, 0x0103, 0x00b4, 0x006a, 0x0026, 0x3fe8, 0x3fb0,
235 0x3f7f, 0x3f52, 0x3f2c, 0x3f0a, 0x3eef, 0x3ed9, 0x3eca, 0x3ebf,
236 0x3eb9, 0x3eb9, 0x3ebd, 0x3ec5, 0x3ed0, 0x3edf, 0x3ef0, 0x3f03
237 },
238 [MALIDP_DOWNSCALING_2_75_COEFFS - 1] = {
239 0x3f51, 0x3f60, 0x3f71, 0x3f84, 0x3f98, 0x3faf, 0x3fc8, 0x3fe3,
240 0x0000, 0x001f, 0x0040, 0x0064, 0x008a, 0x00b1, 0x00da, 0x0106,
241 0x0133, 0x0160, 0x018e, 0x01bd, 0x01ec, 0x021d, 0x024e, 0x0280,
242 0x02b2, 0x02e4, 0x0317, 0x0349, 0x037c, 0x03ad, 0x03df, 0x0410,
243 0x0440, 0x0468, 0x048f, 0x04b3, 0x04d6, 0x04f8, 0x0516, 0x0533,
244 0x054e, 0x0566, 0x057c, 0x0590, 0x05a0, 0x05ae, 0x05ba, 0x05c3,
245 0x05c9, 0x05c3, 0x05ba, 0x05ae, 0x05a0, 0x0590, 0x057c, 0x0566,
246 0x054e, 0x0533, 0x0516, 0x04f8, 0x04d6, 0x04b3, 0x048f, 0x0468,
247 0x0440, 0x0410, 0x03df, 0x03ad, 0x037c, 0x0349, 0x0317, 0x02e4,
248 0x02b2, 0x0280, 0x024e, 0x021d, 0x01ec, 0x01bd, 0x018e, 0x0160,
249 0x0133, 0x0106, 0x00da, 0x00b1, 0x008a, 0x0064, 0x0040, 0x001f,
250 0x0000, 0x3fe3, 0x3fc8, 0x3faf, 0x3f98, 0x3f84, 0x3f71, 0x3f60
251 },
252 [MALIDP_DOWNSCALING_4_COEFFS - 1] = {
253 0x0094, 0x00a9, 0x00be, 0x00d4, 0x00ea, 0x0101, 0x0118, 0x012f,
254 0x0148, 0x0160, 0x017a, 0x0193, 0x01ae, 0x01c8, 0x01e4, 0x01ff,
255 0x021c, 0x0233, 0x024a, 0x0261, 0x0278, 0x028f, 0x02a6, 0x02bd,
256 0x02d4, 0x02eb, 0x0302, 0x0319, 0x032f, 0x0346, 0x035d, 0x0374,
257 0x038a, 0x0397, 0x03a3, 0x03af, 0x03bb, 0x03c6, 0x03d1, 0x03db,
258 0x03e4, 0x03ed, 0x03f6, 0x03fe, 0x0406, 0x040d, 0x0414, 0x041a,
259 0x0420, 0x041a, 0x0414, 0x040d, 0x0406, 0x03fe, 0x03f6, 0x03ed,
260 0x03e4, 0x03db, 0x03d1, 0x03c6, 0x03bb, 0x03af, 0x03a3, 0x0397,
261 0x038a, 0x0374, 0x035d, 0x0346, 0x032f, 0x0319, 0x0302, 0x02eb,
262 0x02d4, 0x02bd, 0x02a6, 0x028f, 0x0278, 0x0261, 0x024a, 0x0233,
263 0x021c, 0x01ff, 0x01e4, 0x01c8, 0x01ae, 0x0193, 0x017a, 0x0160,
264 0x0148, 0x012f, 0x0118, 0x0101, 0x00ea, 0x00d4, 0x00be, 0x00a9
265 },
266 };
267
268 #define MALIDP_DE_DEFAULT_PREFETCH_START 5
269
malidp500_query_hw(struct malidp_hw_device * hwdev)270 static int malidp500_query_hw(struct malidp_hw_device *hwdev)
271 {
272 u32 conf = malidp_hw_read(hwdev, MALIDP500_CONFIG_ID);
273 /* bit 4 of the CONFIG_ID register holds the line size multiplier */
274 u8 ln_size_mult = conf & 0x10 ? 2 : 1;
275
276 hwdev->min_line_size = 2;
277 hwdev->max_line_size = SZ_2K * ln_size_mult;
278 hwdev->rotation_memory[0] = SZ_1K * 64 * ln_size_mult;
279 hwdev->rotation_memory[1] = 0; /* no second rotation memory bank */
280
281 return 0;
282 }
283
malidp500_enter_config_mode(struct malidp_hw_device * hwdev)284 static void malidp500_enter_config_mode(struct malidp_hw_device *hwdev)
285 {
286 u32 status, count = 100;
287
288 malidp_hw_setbits(hwdev, MALIDP500_DC_CONFIG_REQ, MALIDP500_DC_CONTROL);
289 while (count) {
290 status = malidp_hw_read(hwdev, hwdev->hw->map.dc_base + MALIDP_REG_STATUS);
291 if ((status & MALIDP500_DC_CONFIG_REQ) == MALIDP500_DC_CONFIG_REQ)
292 break;
293 /*
294 * entering config mode can take as long as the rendering
295 * of a full frame, hence the long sleep here
296 */
297 usleep_range(1000, 10000);
298 count--;
299 }
300 WARN(count == 0, "timeout while entering config mode");
301 }
302
malidp500_leave_config_mode(struct malidp_hw_device * hwdev)303 static void malidp500_leave_config_mode(struct malidp_hw_device *hwdev)
304 {
305 u32 status, count = 100;
306
307 malidp_hw_clearbits(hwdev, MALIDP_CFG_VALID, MALIDP500_CONFIG_VALID);
308 malidp_hw_clearbits(hwdev, MALIDP500_DC_CONFIG_REQ, MALIDP500_DC_CONTROL);
309 while (count) {
310 status = malidp_hw_read(hwdev, hwdev->hw->map.dc_base + MALIDP_REG_STATUS);
311 if ((status & MALIDP500_DC_CONFIG_REQ) == 0)
312 break;
313 usleep_range(100, 1000);
314 count--;
315 }
316 WARN(count == 0, "timeout while leaving config mode");
317 }
318
malidp500_in_config_mode(struct malidp_hw_device * hwdev)319 static bool malidp500_in_config_mode(struct malidp_hw_device *hwdev)
320 {
321 u32 status;
322
323 status = malidp_hw_read(hwdev, hwdev->hw->map.dc_base + MALIDP_REG_STATUS);
324 if ((status & MALIDP500_DC_CONFIG_REQ) == MALIDP500_DC_CONFIG_REQ)
325 return true;
326
327 return false;
328 }
329
malidp500_set_config_valid(struct malidp_hw_device * hwdev,u8 value)330 static void malidp500_set_config_valid(struct malidp_hw_device *hwdev, u8 value)
331 {
332 if (value)
333 malidp_hw_setbits(hwdev, MALIDP_CFG_VALID, MALIDP500_CONFIG_VALID);
334 else
335 malidp_hw_clearbits(hwdev, MALIDP_CFG_VALID, MALIDP500_CONFIG_VALID);
336 }
337
malidp500_modeset(struct malidp_hw_device * hwdev,struct videomode * mode)338 static void malidp500_modeset(struct malidp_hw_device *hwdev, struct videomode *mode)
339 {
340 u32 val = 0;
341
342 malidp_hw_write(hwdev, hwdev->output_color_depth,
343 hwdev->hw->map.out_depth_base);
344 malidp_hw_clearbits(hwdev, MALIDP500_DC_CLEAR_MASK, MALIDP500_DC_CONTROL);
345 if (mode->flags & DISPLAY_FLAGS_HSYNC_HIGH)
346 val |= MALIDP500_HSYNCPOL;
347 if (mode->flags & DISPLAY_FLAGS_VSYNC_HIGH)
348 val |= MALIDP500_VSYNCPOL;
349 val |= MALIDP_DE_DEFAULT_PREFETCH_START;
350 malidp_hw_setbits(hwdev, val, MALIDP500_DC_CONTROL);
351
352 /*
353 * Mali-DP500 encodes the background color like this:
354 * - red @ MALIDP500_BGND_COLOR[12:0]
355 * - green @ MALIDP500_BGND_COLOR[27:16]
356 * - blue @ (MALIDP500_BGND_COLOR + 4)[12:0]
357 */
358 val = ((MALIDP_BGND_COLOR_G & 0xfff) << 16) |
359 (MALIDP_BGND_COLOR_R & 0xfff);
360 malidp_hw_write(hwdev, val, MALIDP500_BGND_COLOR);
361 malidp_hw_write(hwdev, MALIDP_BGND_COLOR_B, MALIDP500_BGND_COLOR + 4);
362
363 val = MALIDP_DE_H_FRONTPORCH(mode->hfront_porch) |
364 MALIDP_DE_H_BACKPORCH(mode->hback_porch);
365 malidp_hw_write(hwdev, val, MALIDP500_TIMINGS_BASE + MALIDP_DE_H_TIMINGS);
366
367 val = MALIDP500_DE_V_FRONTPORCH(mode->vfront_porch) |
368 MALIDP_DE_V_BACKPORCH(mode->vback_porch);
369 malidp_hw_write(hwdev, val, MALIDP500_TIMINGS_BASE + MALIDP_DE_V_TIMINGS);
370
371 val = MALIDP_DE_H_SYNCWIDTH(mode->hsync_len) |
372 MALIDP_DE_V_SYNCWIDTH(mode->vsync_len);
373 malidp_hw_write(hwdev, val, MALIDP500_TIMINGS_BASE + MALIDP_DE_SYNC_WIDTH);
374
375 val = MALIDP_DE_H_ACTIVE(mode->hactive) | MALIDP_DE_V_ACTIVE(mode->vactive);
376 malidp_hw_write(hwdev, val, MALIDP500_TIMINGS_BASE + MALIDP_DE_HV_ACTIVE);
377
378 if (mode->flags & DISPLAY_FLAGS_INTERLACED)
379 malidp_hw_setbits(hwdev, MALIDP_DISP_FUNC_ILACED, MALIDP_DE_DISPLAY_FUNC);
380 else
381 malidp_hw_clearbits(hwdev, MALIDP_DISP_FUNC_ILACED, MALIDP_DE_DISPLAY_FUNC);
382 }
383
malidp_format_get_bpp(u32 fmt)384 int malidp_format_get_bpp(u32 fmt)
385 {
386 const struct drm_format_info *info = drm_format_info(fmt);
387 int bpp = info->cpp[0] * 8;
388
389 if (bpp == 0) {
390 switch (fmt) {
391 case DRM_FORMAT_VUY101010:
392 bpp = 30;
393 break;
394 case DRM_FORMAT_YUV420_10BIT:
395 bpp = 15;
396 break;
397 case DRM_FORMAT_YUV420_8BIT:
398 bpp = 12;
399 break;
400 default:
401 bpp = 0;
402 }
403 }
404
405 return bpp;
406 }
407
malidp500_rotmem_required(struct malidp_hw_device * hwdev,u16 w,u16 h,u32 fmt,bool has_modifier)408 static int malidp500_rotmem_required(struct malidp_hw_device *hwdev, u16 w,
409 u16 h, u32 fmt, bool has_modifier)
410 {
411 /*
412 * Each layer needs enough rotation memory to fit 8 lines
413 * worth of pixel data. Required size is then:
414 * size = rotated_width * (bpp / 8) * 8;
415 */
416 int bpp = malidp_format_get_bpp(fmt);
417
418 return w * bpp;
419 }
420
malidp500_se_write_pp_coefftab(struct malidp_hw_device * hwdev,u32 direction,u16 addr,u8 coeffs_id)421 static void malidp500_se_write_pp_coefftab(struct malidp_hw_device *hwdev,
422 u32 direction,
423 u16 addr,
424 u8 coeffs_id)
425 {
426 int i;
427 u16 scaling_control = MALIDP500_SE_CONTROL + MALIDP_SE_SCALING_CONTROL;
428
429 malidp_hw_write(hwdev,
430 direction | (addr & MALIDP_SE_COEFFTAB_ADDR_MASK),
431 scaling_control + MALIDP_SE_COEFFTAB_ADDR);
432 for (i = 0; i < ARRAY_SIZE(dp500_se_scaling_coeffs); ++i)
433 malidp_hw_write(hwdev, MALIDP_SE_SET_COEFFTAB_DATA(
434 dp500_se_scaling_coeffs[coeffs_id][i]),
435 scaling_control + MALIDP_SE_COEFFTAB_DATA);
436 }
437
malidp500_se_set_scaling_coeffs(struct malidp_hw_device * hwdev,struct malidp_se_config * se_config,struct malidp_se_config * old_config)438 static int malidp500_se_set_scaling_coeffs(struct malidp_hw_device *hwdev,
439 struct malidp_se_config *se_config,
440 struct malidp_se_config *old_config)
441 {
442 /* Get array indices into dp500_se_scaling_coeffs. */
443 u8 h = (u8)se_config->hcoeff - 1;
444 u8 v = (u8)se_config->vcoeff - 1;
445
446 if (WARN_ON(h >= ARRAY_SIZE(dp500_se_scaling_coeffs) ||
447 v >= ARRAY_SIZE(dp500_se_scaling_coeffs)))
448 return -EINVAL;
449
450 if ((h == v) && (se_config->hcoeff != old_config->hcoeff ||
451 se_config->vcoeff != old_config->vcoeff)) {
452 malidp500_se_write_pp_coefftab(hwdev,
453 (MALIDP_SE_V_COEFFTAB |
454 MALIDP_SE_H_COEFFTAB),
455 0, v);
456 } else {
457 if (se_config->vcoeff != old_config->vcoeff)
458 malidp500_se_write_pp_coefftab(hwdev,
459 MALIDP_SE_V_COEFFTAB,
460 0, v);
461 if (se_config->hcoeff != old_config->hcoeff)
462 malidp500_se_write_pp_coefftab(hwdev,
463 MALIDP_SE_H_COEFFTAB,
464 0, h);
465 }
466
467 return 0;
468 }
469
malidp500_se_calc_mclk(struct malidp_hw_device * hwdev,struct malidp_se_config * se_config,struct videomode * vm)470 static long malidp500_se_calc_mclk(struct malidp_hw_device *hwdev,
471 struct malidp_se_config *se_config,
472 struct videomode *vm)
473 {
474 unsigned long mclk;
475 unsigned long pxlclk = vm->pixelclock; /* Hz */
476 unsigned long htotal = vm->hactive + vm->hfront_porch +
477 vm->hback_porch + vm->hsync_len;
478 unsigned long input_size = se_config->input_w * se_config->input_h;
479 unsigned long a = 10;
480 long ret;
481
482 /*
483 * mclk = max(a, 1.5) * pxlclk
484 *
485 * To avoid float calculaiton, using 15 instead of 1.5 and div by
486 * 10 to get mclk.
487 */
488 if (se_config->scale_enable) {
489 a = 15 * input_size / (htotal * se_config->output_h);
490 if (a < 15)
491 a = 15;
492 }
493 mclk = a * pxlclk / 10;
494 ret = clk_get_rate(hwdev->mclk);
495 if (ret < mclk) {
496 DRM_DEBUG_DRIVER("mclk requirement of %lu kHz can't be met.\n",
497 mclk / 1000);
498 return -EINVAL;
499 }
500 return ret;
501 }
502
malidp500_enable_memwrite(struct malidp_hw_device * hwdev,dma_addr_t * addrs,s32 * pitches,int num_planes,u16 w,u16 h,u32 fmt_id,const s16 * rgb2yuv_coeffs)503 static int malidp500_enable_memwrite(struct malidp_hw_device *hwdev,
504 dma_addr_t *addrs, s32 *pitches,
505 int num_planes, u16 w, u16 h, u32 fmt_id,
506 const s16 *rgb2yuv_coeffs)
507 {
508 u32 base = MALIDP500_SE_MEMWRITE_BASE;
509 u32 de_base = malidp_get_block_base(hwdev, MALIDP_DE_BLOCK);
510
511 /* enable the scaling engine block */
512 malidp_hw_setbits(hwdev, MALIDP_SCALE_ENGINE_EN, de_base + MALIDP_DE_DISPLAY_FUNC);
513
514 /* restart the writeback if already enabled */
515 if (hwdev->mw_state != MW_NOT_ENABLED)
516 hwdev->mw_state = MW_RESTART;
517 else
518 hwdev->mw_state = MW_START;
519
520 malidp_hw_write(hwdev, fmt_id, base + MALIDP_MW_FORMAT);
521 switch (num_planes) {
522 case 2:
523 malidp_hw_write(hwdev, lower_32_bits(addrs[1]), base + MALIDP_MW_P2_PTR_LOW);
524 malidp_hw_write(hwdev, upper_32_bits(addrs[1]), base + MALIDP_MW_P2_PTR_HIGH);
525 malidp_hw_write(hwdev, pitches[1], base + MALIDP_MW_P2_STRIDE);
526 /* fall through */
527 case 1:
528 malidp_hw_write(hwdev, lower_32_bits(addrs[0]), base + MALIDP_MW_P1_PTR_LOW);
529 malidp_hw_write(hwdev, upper_32_bits(addrs[0]), base + MALIDP_MW_P1_PTR_HIGH);
530 malidp_hw_write(hwdev, pitches[0], base + MALIDP_MW_P1_STRIDE);
531 break;
532 default:
533 WARN(1, "Invalid number of planes");
534 }
535
536 malidp_hw_write(hwdev, MALIDP_DE_H_ACTIVE(w) | MALIDP_DE_V_ACTIVE(h),
537 MALIDP500_SE_MEMWRITE_OUT_SIZE);
538
539 if (rgb2yuv_coeffs) {
540 int i;
541
542 for (i = 0; i < MALIDP_COLORADJ_NUM_COEFFS; i++) {
543 malidp_hw_write(hwdev, rgb2yuv_coeffs[i],
544 MALIDP500_SE_RGB_YUV_COEFFS + i * 4);
545 }
546 }
547
548 malidp_hw_setbits(hwdev, MALIDP_SE_MEMWRITE_EN, MALIDP500_SE_CONTROL);
549
550 return 0;
551 }
552
malidp500_disable_memwrite(struct malidp_hw_device * hwdev)553 static void malidp500_disable_memwrite(struct malidp_hw_device *hwdev)
554 {
555 u32 base = malidp_get_block_base(hwdev, MALIDP_DE_BLOCK);
556
557 if (hwdev->mw_state == MW_START || hwdev->mw_state == MW_RESTART)
558 hwdev->mw_state = MW_STOP;
559 malidp_hw_clearbits(hwdev, MALIDP_SE_MEMWRITE_EN, MALIDP500_SE_CONTROL);
560 malidp_hw_clearbits(hwdev, MALIDP_SCALE_ENGINE_EN, base + MALIDP_DE_DISPLAY_FUNC);
561 }
562
malidp550_query_hw(struct malidp_hw_device * hwdev)563 static int malidp550_query_hw(struct malidp_hw_device *hwdev)
564 {
565 u32 conf = malidp_hw_read(hwdev, MALIDP550_CONFIG_ID);
566 u8 ln_size = (conf >> 4) & 0x3, rsize;
567
568 hwdev->min_line_size = 2;
569
570 switch (ln_size) {
571 case 0:
572 hwdev->max_line_size = SZ_2K;
573 /* two banks of 64KB for rotation memory */
574 rsize = 64;
575 break;
576 case 1:
577 hwdev->max_line_size = SZ_4K;
578 /* two banks of 128KB for rotation memory */
579 rsize = 128;
580 break;
581 case 2:
582 hwdev->max_line_size = 1280;
583 /* two banks of 40KB for rotation memory */
584 rsize = 40;
585 break;
586 case 3:
587 /* reserved value */
588 hwdev->max_line_size = 0;
589 return -EINVAL;
590 }
591
592 hwdev->rotation_memory[0] = hwdev->rotation_memory[1] = rsize * SZ_1K;
593 return 0;
594 }
595
malidp550_enter_config_mode(struct malidp_hw_device * hwdev)596 static void malidp550_enter_config_mode(struct malidp_hw_device *hwdev)
597 {
598 u32 status, count = 100;
599
600 malidp_hw_setbits(hwdev, MALIDP550_DC_CONFIG_REQ, MALIDP550_DC_CONTROL);
601 while (count) {
602 status = malidp_hw_read(hwdev, hwdev->hw->map.dc_base + MALIDP_REG_STATUS);
603 if ((status & MALIDP550_DC_CONFIG_REQ) == MALIDP550_DC_CONFIG_REQ)
604 break;
605 /*
606 * entering config mode can take as long as the rendering
607 * of a full frame, hence the long sleep here
608 */
609 usleep_range(1000, 10000);
610 count--;
611 }
612 WARN(count == 0, "timeout while entering config mode");
613 }
614
malidp550_leave_config_mode(struct malidp_hw_device * hwdev)615 static void malidp550_leave_config_mode(struct malidp_hw_device *hwdev)
616 {
617 u32 status, count = 100;
618
619 malidp_hw_clearbits(hwdev, MALIDP_CFG_VALID, MALIDP550_CONFIG_VALID);
620 malidp_hw_clearbits(hwdev, MALIDP550_DC_CONFIG_REQ, MALIDP550_DC_CONTROL);
621 while (count) {
622 status = malidp_hw_read(hwdev, hwdev->hw->map.dc_base + MALIDP_REG_STATUS);
623 if ((status & MALIDP550_DC_CONFIG_REQ) == 0)
624 break;
625 usleep_range(100, 1000);
626 count--;
627 }
628 WARN(count == 0, "timeout while leaving config mode");
629 }
630
malidp550_in_config_mode(struct malidp_hw_device * hwdev)631 static bool malidp550_in_config_mode(struct malidp_hw_device *hwdev)
632 {
633 u32 status;
634
635 status = malidp_hw_read(hwdev, hwdev->hw->map.dc_base + MALIDP_REG_STATUS);
636 if ((status & MALIDP550_DC_CONFIG_REQ) == MALIDP550_DC_CONFIG_REQ)
637 return true;
638
639 return false;
640 }
641
malidp550_set_config_valid(struct malidp_hw_device * hwdev,u8 value)642 static void malidp550_set_config_valid(struct malidp_hw_device *hwdev, u8 value)
643 {
644 if (value)
645 malidp_hw_setbits(hwdev, MALIDP_CFG_VALID, MALIDP550_CONFIG_VALID);
646 else
647 malidp_hw_clearbits(hwdev, MALIDP_CFG_VALID, MALIDP550_CONFIG_VALID);
648 }
649
malidp550_modeset(struct malidp_hw_device * hwdev,struct videomode * mode)650 static void malidp550_modeset(struct malidp_hw_device *hwdev, struct videomode *mode)
651 {
652 u32 val = MALIDP_DE_DEFAULT_PREFETCH_START;
653
654 malidp_hw_write(hwdev, hwdev->output_color_depth,
655 hwdev->hw->map.out_depth_base);
656 malidp_hw_write(hwdev, val, MALIDP550_DE_CONTROL);
657 /*
658 * Mali-DP550 and Mali-DP650 encode the background color like this:
659 * - red @ MALIDP550_DE_BGND_COLOR[23:16]
660 * - green @ MALIDP550_DE_BGND_COLOR[15:8]
661 * - blue @ MALIDP550_DE_BGND_COLOR[7:0]
662 *
663 * We need to truncate the least significant 4 bits from the default
664 * MALIDP_BGND_COLOR_x values
665 */
666 val = (((MALIDP_BGND_COLOR_R >> 4) & 0xff) << 16) |
667 (((MALIDP_BGND_COLOR_G >> 4) & 0xff) << 8) |
668 ((MALIDP_BGND_COLOR_B >> 4) & 0xff);
669 malidp_hw_write(hwdev, val, MALIDP550_DE_BGND_COLOR);
670
671 val = MALIDP_DE_H_FRONTPORCH(mode->hfront_porch) |
672 MALIDP_DE_H_BACKPORCH(mode->hback_porch);
673 malidp_hw_write(hwdev, val, MALIDP550_TIMINGS_BASE + MALIDP_DE_H_TIMINGS);
674
675 val = MALIDP550_DE_V_FRONTPORCH(mode->vfront_porch) |
676 MALIDP_DE_V_BACKPORCH(mode->vback_porch);
677 malidp_hw_write(hwdev, val, MALIDP550_TIMINGS_BASE + MALIDP_DE_V_TIMINGS);
678
679 val = MALIDP_DE_H_SYNCWIDTH(mode->hsync_len) |
680 MALIDP_DE_V_SYNCWIDTH(mode->vsync_len);
681 if (mode->flags & DISPLAY_FLAGS_HSYNC_HIGH)
682 val |= MALIDP550_HSYNCPOL;
683 if (mode->flags & DISPLAY_FLAGS_VSYNC_HIGH)
684 val |= MALIDP550_VSYNCPOL;
685 malidp_hw_write(hwdev, val, MALIDP550_TIMINGS_BASE + MALIDP_DE_SYNC_WIDTH);
686
687 val = MALIDP_DE_H_ACTIVE(mode->hactive) | MALIDP_DE_V_ACTIVE(mode->vactive);
688 malidp_hw_write(hwdev, val, MALIDP550_TIMINGS_BASE + MALIDP_DE_HV_ACTIVE);
689
690 if (mode->flags & DISPLAY_FLAGS_INTERLACED)
691 malidp_hw_setbits(hwdev, MALIDP_DISP_FUNC_ILACED, MALIDP_DE_DISPLAY_FUNC);
692 else
693 malidp_hw_clearbits(hwdev, MALIDP_DISP_FUNC_ILACED, MALIDP_DE_DISPLAY_FUNC);
694 }
695
malidpx50_get_bytes_per_column(u32 fmt)696 static int malidpx50_get_bytes_per_column(u32 fmt)
697 {
698 u32 bytes_per_column;
699
700 switch (fmt) {
701 /* 8 lines at 4 bytes per pixel */
702 case DRM_FORMAT_ARGB2101010:
703 case DRM_FORMAT_ABGR2101010:
704 case DRM_FORMAT_RGBA1010102:
705 case DRM_FORMAT_BGRA1010102:
706 case DRM_FORMAT_ARGB8888:
707 case DRM_FORMAT_ABGR8888:
708 case DRM_FORMAT_RGBA8888:
709 case DRM_FORMAT_BGRA8888:
710 case DRM_FORMAT_XRGB8888:
711 case DRM_FORMAT_XBGR8888:
712 case DRM_FORMAT_RGBX8888:
713 case DRM_FORMAT_BGRX8888:
714 case DRM_FORMAT_RGB888:
715 case DRM_FORMAT_BGR888:
716 /* 16 lines at 2 bytes per pixel */
717 case DRM_FORMAT_RGBA5551:
718 case DRM_FORMAT_ABGR1555:
719 case DRM_FORMAT_RGB565:
720 case DRM_FORMAT_BGR565:
721 case DRM_FORMAT_UYVY:
722 case DRM_FORMAT_YUYV:
723 case DRM_FORMAT_X0L0:
724 bytes_per_column = 32;
725 break;
726 /* 16 lines at 1.5 bytes per pixel */
727 case DRM_FORMAT_NV12:
728 case DRM_FORMAT_YUV420:
729 /* 8 lines at 3 bytes per pixel */
730 case DRM_FORMAT_VUY888:
731 /* 16 lines at 12 bits per pixel */
732 case DRM_FORMAT_YUV420_8BIT:
733 /* 8 lines at 3 bytes per pixel */
734 case DRM_FORMAT_P010:
735 bytes_per_column = 24;
736 break;
737 /* 8 lines at 30 bits per pixel */
738 case DRM_FORMAT_VUY101010:
739 /* 16 lines at 15 bits per pixel */
740 case DRM_FORMAT_YUV420_10BIT:
741 bytes_per_column = 30;
742 break;
743 default:
744 return -EINVAL;
745 }
746
747 return bytes_per_column;
748 }
749
malidp550_rotmem_required(struct malidp_hw_device * hwdev,u16 w,u16 h,u32 fmt,bool has_modifier)750 static int malidp550_rotmem_required(struct malidp_hw_device *hwdev, u16 w,
751 u16 h, u32 fmt, bool has_modifier)
752 {
753 int bytes_per_column = 0;
754
755 switch (fmt) {
756 /* 8 lines at 15 bits per pixel */
757 case DRM_FORMAT_YUV420_10BIT:
758 bytes_per_column = 15;
759 break;
760 /* Uncompressed YUV 420 10 bit single plane cannot be rotated */
761 case DRM_FORMAT_X0L2:
762 if (has_modifier)
763 bytes_per_column = 8;
764 else
765 return -EINVAL;
766 break;
767 default:
768 bytes_per_column = malidpx50_get_bytes_per_column(fmt);
769 }
770
771 if (bytes_per_column == -EINVAL)
772 return bytes_per_column;
773
774 return w * bytes_per_column;
775 }
776
malidp650_rotmem_required(struct malidp_hw_device * hwdev,u16 w,u16 h,u32 fmt,bool has_modifier)777 static int malidp650_rotmem_required(struct malidp_hw_device *hwdev, u16 w,
778 u16 h, u32 fmt, bool has_modifier)
779 {
780 int bytes_per_column = 0;
781
782 switch (fmt) {
783 /* 16 lines at 2 bytes per pixel */
784 case DRM_FORMAT_X0L2:
785 bytes_per_column = 32;
786 break;
787 default:
788 bytes_per_column = malidpx50_get_bytes_per_column(fmt);
789 }
790
791 if (bytes_per_column == -EINVAL)
792 return bytes_per_column;
793
794 return w * bytes_per_column;
795 }
796
malidp550_se_set_scaling_coeffs(struct malidp_hw_device * hwdev,struct malidp_se_config * se_config,struct malidp_se_config * old_config)797 static int malidp550_se_set_scaling_coeffs(struct malidp_hw_device *hwdev,
798 struct malidp_se_config *se_config,
799 struct malidp_se_config *old_config)
800 {
801 u32 mask = MALIDP550_SE_CTL_VCSEL(MALIDP550_SE_CTL_SEL_MASK) |
802 MALIDP550_SE_CTL_HCSEL(MALIDP550_SE_CTL_SEL_MASK);
803 u32 new_value = MALIDP550_SE_CTL_VCSEL(se_config->vcoeff) |
804 MALIDP550_SE_CTL_HCSEL(se_config->hcoeff);
805
806 malidp_hw_clearbits(hwdev, mask, MALIDP550_SE_CONTROL);
807 malidp_hw_setbits(hwdev, new_value, MALIDP550_SE_CONTROL);
808 return 0;
809 }
810
malidp550_se_calc_mclk(struct malidp_hw_device * hwdev,struct malidp_se_config * se_config,struct videomode * vm)811 static long malidp550_se_calc_mclk(struct malidp_hw_device *hwdev,
812 struct malidp_se_config *se_config,
813 struct videomode *vm)
814 {
815 unsigned long mclk;
816 unsigned long pxlclk = vm->pixelclock;
817 unsigned long htotal = vm->hactive + vm->hfront_porch +
818 vm->hback_porch + vm->hsync_len;
819 unsigned long numerator = 1, denominator = 1;
820 long ret;
821
822 if (se_config->scale_enable) {
823 numerator = max(se_config->input_w, se_config->output_w) *
824 se_config->input_h;
825 numerator += se_config->output_w *
826 (se_config->output_h -
827 min(se_config->input_h, se_config->output_h));
828 denominator = (htotal - 2) * se_config->output_h;
829 }
830
831 /* mclk can't be slower than pxlclk. */
832 if (numerator < denominator)
833 numerator = denominator = 1;
834 mclk = (pxlclk * numerator) / denominator;
835 ret = clk_get_rate(hwdev->mclk);
836 if (ret < mclk) {
837 DRM_DEBUG_DRIVER("mclk requirement of %lu kHz can't be met.\n",
838 mclk / 1000);
839 return -EINVAL;
840 }
841 return ret;
842 }
843
malidp550_enable_memwrite(struct malidp_hw_device * hwdev,dma_addr_t * addrs,s32 * pitches,int num_planes,u16 w,u16 h,u32 fmt_id,const s16 * rgb2yuv_coeffs)844 static int malidp550_enable_memwrite(struct malidp_hw_device *hwdev,
845 dma_addr_t *addrs, s32 *pitches,
846 int num_planes, u16 w, u16 h, u32 fmt_id,
847 const s16 *rgb2yuv_coeffs)
848 {
849 u32 base = MALIDP550_SE_MEMWRITE_BASE;
850 u32 de_base = malidp_get_block_base(hwdev, MALIDP_DE_BLOCK);
851
852 /* enable the scaling engine block */
853 malidp_hw_setbits(hwdev, MALIDP_SCALE_ENGINE_EN, de_base + MALIDP_DE_DISPLAY_FUNC);
854
855 hwdev->mw_state = MW_ONESHOT;
856
857 malidp_hw_write(hwdev, fmt_id, base + MALIDP_MW_FORMAT);
858 switch (num_planes) {
859 case 2:
860 malidp_hw_write(hwdev, lower_32_bits(addrs[1]), base + MALIDP_MW_P2_PTR_LOW);
861 malidp_hw_write(hwdev, upper_32_bits(addrs[1]), base + MALIDP_MW_P2_PTR_HIGH);
862 malidp_hw_write(hwdev, pitches[1], base + MALIDP_MW_P2_STRIDE);
863 /* fall through */
864 case 1:
865 malidp_hw_write(hwdev, lower_32_bits(addrs[0]), base + MALIDP_MW_P1_PTR_LOW);
866 malidp_hw_write(hwdev, upper_32_bits(addrs[0]), base + MALIDP_MW_P1_PTR_HIGH);
867 malidp_hw_write(hwdev, pitches[0], base + MALIDP_MW_P1_STRIDE);
868 break;
869 default:
870 WARN(1, "Invalid number of planes");
871 }
872
873 malidp_hw_write(hwdev, MALIDP_DE_H_ACTIVE(w) | MALIDP_DE_V_ACTIVE(h),
874 MALIDP550_SE_MEMWRITE_OUT_SIZE);
875 malidp_hw_setbits(hwdev, MALIDP550_SE_MEMWRITE_ONESHOT | MALIDP_SE_MEMWRITE_EN,
876 MALIDP550_SE_CONTROL);
877
878 if (rgb2yuv_coeffs) {
879 int i;
880
881 for (i = 0; i < MALIDP_COLORADJ_NUM_COEFFS; i++) {
882 malidp_hw_write(hwdev, rgb2yuv_coeffs[i],
883 MALIDP550_SE_RGB_YUV_COEFFS + i * 4);
884 }
885 }
886
887 return 0;
888 }
889
malidp550_disable_memwrite(struct malidp_hw_device * hwdev)890 static void malidp550_disable_memwrite(struct malidp_hw_device *hwdev)
891 {
892 u32 base = malidp_get_block_base(hwdev, MALIDP_DE_BLOCK);
893
894 malidp_hw_clearbits(hwdev, MALIDP550_SE_MEMWRITE_ONESHOT | MALIDP_SE_MEMWRITE_EN,
895 MALIDP550_SE_CONTROL);
896 malidp_hw_clearbits(hwdev, MALIDP_SCALE_ENGINE_EN, base + MALIDP_DE_DISPLAY_FUNC);
897 }
898
malidp650_query_hw(struct malidp_hw_device * hwdev)899 static int malidp650_query_hw(struct malidp_hw_device *hwdev)
900 {
901 u32 conf = malidp_hw_read(hwdev, MALIDP550_CONFIG_ID);
902 u8 ln_size = (conf >> 4) & 0x3, rsize;
903
904 hwdev->min_line_size = 4;
905
906 switch (ln_size) {
907 case 0:
908 case 2:
909 /* reserved values */
910 hwdev->max_line_size = 0;
911 return -EINVAL;
912 case 1:
913 hwdev->max_line_size = SZ_4K;
914 /* two banks of 128KB for rotation memory */
915 rsize = 128;
916 break;
917 case 3:
918 hwdev->max_line_size = 2560;
919 /* two banks of 80KB for rotation memory */
920 rsize = 80;
921 }
922
923 hwdev->rotation_memory[0] = hwdev->rotation_memory[1] = rsize * SZ_1K;
924 return 0;
925 }
926
927 const struct malidp_hw malidp_device[MALIDP_MAX_DEVICES] = {
928 [MALIDP_500] = {
929 .map = {
930 .coeffs_base = MALIDP500_COEFFS_BASE,
931 .se_base = MALIDP500_SE_BASE,
932 .dc_base = MALIDP500_DC_BASE,
933 .out_depth_base = MALIDP500_OUTPUT_DEPTH,
934 .features = 0, /* no CLEARIRQ register */
935 .n_layers = ARRAY_SIZE(malidp500_layers),
936 .layers = malidp500_layers,
937 .de_irq_map = {
938 .irq_mask = MALIDP_DE_IRQ_UNDERRUN |
939 MALIDP500_DE_IRQ_AXI_ERR |
940 MALIDP500_DE_IRQ_VSYNC |
941 MALIDP500_DE_IRQ_GLOBAL,
942 .vsync_irq = MALIDP500_DE_IRQ_VSYNC,
943 .err_mask = MALIDP_DE_IRQ_UNDERRUN |
944 MALIDP500_DE_IRQ_AXI_ERR |
945 MALIDP500_DE_IRQ_SATURATION,
946 },
947 .se_irq_map = {
948 .irq_mask = MALIDP500_SE_IRQ_CONF_MODE |
949 MALIDP500_SE_IRQ_CONF_VALID |
950 MALIDP500_SE_IRQ_GLOBAL,
951 .vsync_irq = MALIDP500_SE_IRQ_CONF_VALID,
952 .err_mask = MALIDP500_SE_IRQ_INIT_BUSY |
953 MALIDP500_SE_IRQ_AXI_ERROR |
954 MALIDP500_SE_IRQ_OVERRUN,
955 },
956 .dc_irq_map = {
957 .irq_mask = MALIDP500_DE_IRQ_CONF_VALID,
958 .vsync_irq = MALIDP500_DE_IRQ_CONF_VALID,
959 },
960 .pixel_formats = malidp500_de_formats,
961 .n_pixel_formats = ARRAY_SIZE(malidp500_de_formats),
962 .bus_align_bytes = 8,
963 },
964 .query_hw = malidp500_query_hw,
965 .enter_config_mode = malidp500_enter_config_mode,
966 .leave_config_mode = malidp500_leave_config_mode,
967 .in_config_mode = malidp500_in_config_mode,
968 .set_config_valid = malidp500_set_config_valid,
969 .modeset = malidp500_modeset,
970 .rotmem_required = malidp500_rotmem_required,
971 .se_set_scaling_coeffs = malidp500_se_set_scaling_coeffs,
972 .se_calc_mclk = malidp500_se_calc_mclk,
973 .enable_memwrite = malidp500_enable_memwrite,
974 .disable_memwrite = malidp500_disable_memwrite,
975 .features = MALIDP_DEVICE_LV_HAS_3_STRIDES,
976 },
977 [MALIDP_550] = {
978 .map = {
979 .coeffs_base = MALIDP550_COEFFS_BASE,
980 .se_base = MALIDP550_SE_BASE,
981 .dc_base = MALIDP550_DC_BASE,
982 .out_depth_base = MALIDP550_DE_OUTPUT_DEPTH,
983 .features = MALIDP_REGMAP_HAS_CLEARIRQ |
984 MALIDP_DEVICE_AFBC_SUPPORT_SPLIT |
985 MALIDP_DEVICE_AFBC_YUV_420_10_SUPPORT_SPLIT |
986 MALIDP_DEVICE_AFBC_YUYV_USE_422_P2,
987 .n_layers = ARRAY_SIZE(malidp550_layers),
988 .layers = malidp550_layers,
989 .de_irq_map = {
990 .irq_mask = MALIDP_DE_IRQ_UNDERRUN |
991 MALIDP550_DE_IRQ_VSYNC,
992 .vsync_irq = MALIDP550_DE_IRQ_VSYNC,
993 .err_mask = MALIDP_DE_IRQ_UNDERRUN |
994 MALIDP550_DE_IRQ_SATURATION |
995 MALIDP550_DE_IRQ_AXI_ERR,
996 },
997 .se_irq_map = {
998 .irq_mask = MALIDP550_SE_IRQ_EOW,
999 .vsync_irq = MALIDP550_SE_IRQ_EOW,
1000 .err_mask = MALIDP550_SE_IRQ_AXI_ERR |
1001 MALIDP550_SE_IRQ_OVR |
1002 MALIDP550_SE_IRQ_IBSY,
1003 },
1004 .dc_irq_map = {
1005 .irq_mask = MALIDP550_DC_IRQ_CONF_VALID |
1006 MALIDP550_DC_IRQ_SE,
1007 .vsync_irq = MALIDP550_DC_IRQ_CONF_VALID,
1008 },
1009 .pixel_formats = malidp550_de_formats,
1010 .n_pixel_formats = ARRAY_SIZE(malidp550_de_formats),
1011 .bus_align_bytes = 8,
1012 },
1013 .query_hw = malidp550_query_hw,
1014 .enter_config_mode = malidp550_enter_config_mode,
1015 .leave_config_mode = malidp550_leave_config_mode,
1016 .in_config_mode = malidp550_in_config_mode,
1017 .set_config_valid = malidp550_set_config_valid,
1018 .modeset = malidp550_modeset,
1019 .rotmem_required = malidp550_rotmem_required,
1020 .se_set_scaling_coeffs = malidp550_se_set_scaling_coeffs,
1021 .se_calc_mclk = malidp550_se_calc_mclk,
1022 .enable_memwrite = malidp550_enable_memwrite,
1023 .disable_memwrite = malidp550_disable_memwrite,
1024 .features = 0,
1025 },
1026 [MALIDP_650] = {
1027 .map = {
1028 .coeffs_base = MALIDP550_COEFFS_BASE,
1029 .se_base = MALIDP550_SE_BASE,
1030 .dc_base = MALIDP550_DC_BASE,
1031 .out_depth_base = MALIDP550_DE_OUTPUT_DEPTH,
1032 .features = MALIDP_REGMAP_HAS_CLEARIRQ |
1033 MALIDP_DEVICE_AFBC_SUPPORT_SPLIT |
1034 MALIDP_DEVICE_AFBC_YUYV_USE_422_P2,
1035 .n_layers = ARRAY_SIZE(malidp650_layers),
1036 .layers = malidp650_layers,
1037 .de_irq_map = {
1038 .irq_mask = MALIDP_DE_IRQ_UNDERRUN |
1039 MALIDP650_DE_IRQ_DRIFT |
1040 MALIDP550_DE_IRQ_VSYNC,
1041 .vsync_irq = MALIDP550_DE_IRQ_VSYNC,
1042 .err_mask = MALIDP_DE_IRQ_UNDERRUN |
1043 MALIDP650_DE_IRQ_DRIFT |
1044 MALIDP550_DE_IRQ_SATURATION |
1045 MALIDP550_DE_IRQ_AXI_ERR |
1046 MALIDP650_DE_IRQ_ACEV1 |
1047 MALIDP650_DE_IRQ_ACEV2 |
1048 MALIDP650_DE_IRQ_ACEG |
1049 MALIDP650_DE_IRQ_AXIEP,
1050 },
1051 .se_irq_map = {
1052 .irq_mask = MALIDP550_SE_IRQ_EOW,
1053 .vsync_irq = MALIDP550_SE_IRQ_EOW,
1054 .err_mask = MALIDP550_SE_IRQ_AXI_ERR |
1055 MALIDP550_SE_IRQ_OVR |
1056 MALIDP550_SE_IRQ_IBSY,
1057 },
1058 .dc_irq_map = {
1059 .irq_mask = MALIDP550_DC_IRQ_CONF_VALID |
1060 MALIDP550_DC_IRQ_SE,
1061 .vsync_irq = MALIDP550_DC_IRQ_CONF_VALID,
1062 },
1063 .pixel_formats = malidp650_de_formats,
1064 .n_pixel_formats = ARRAY_SIZE(malidp650_de_formats),
1065 .bus_align_bytes = 16,
1066 },
1067 .query_hw = malidp650_query_hw,
1068 .enter_config_mode = malidp550_enter_config_mode,
1069 .leave_config_mode = malidp550_leave_config_mode,
1070 .in_config_mode = malidp550_in_config_mode,
1071 .set_config_valid = malidp550_set_config_valid,
1072 .modeset = malidp550_modeset,
1073 .rotmem_required = malidp650_rotmem_required,
1074 .se_set_scaling_coeffs = malidp550_se_set_scaling_coeffs,
1075 .se_calc_mclk = malidp550_se_calc_mclk,
1076 .enable_memwrite = malidp550_enable_memwrite,
1077 .disable_memwrite = malidp550_disable_memwrite,
1078 .features = 0,
1079 },
1080 };
1081
malidp_hw_get_format_id(const struct malidp_hw_regmap * map,u8 layer_id,u32 format,bool has_modifier)1082 u8 malidp_hw_get_format_id(const struct malidp_hw_regmap *map,
1083 u8 layer_id, u32 format, bool has_modifier)
1084 {
1085 unsigned int i;
1086
1087 for (i = 0; i < map->n_pixel_formats; i++) {
1088 if (((map->pixel_formats[i].layer & layer_id) == layer_id) &&
1089 (map->pixel_formats[i].format == format)) {
1090 /*
1091 * In some DP550 and DP650, DRM_FORMAT_YUYV + AFBC modifier
1092 * is supported by a different h/w format id than
1093 * DRM_FORMAT_YUYV (only).
1094 */
1095 if (format == DRM_FORMAT_YUYV &&
1096 (has_modifier) &&
1097 (map->features & MALIDP_DEVICE_AFBC_YUYV_USE_422_P2))
1098 return AFBC_YUV_422_FORMAT_ID;
1099 else
1100 return map->pixel_formats[i].id;
1101 }
1102 }
1103
1104 return MALIDP_INVALID_FORMAT_ID;
1105 }
1106
malidp_hw_format_is_linear_only(u32 format)1107 bool malidp_hw_format_is_linear_only(u32 format)
1108 {
1109 switch (format) {
1110 case DRM_FORMAT_ARGB2101010:
1111 case DRM_FORMAT_RGBA1010102:
1112 case DRM_FORMAT_BGRA1010102:
1113 case DRM_FORMAT_ARGB8888:
1114 case DRM_FORMAT_RGBA8888:
1115 case DRM_FORMAT_BGRA8888:
1116 case DRM_FORMAT_XBGR8888:
1117 case DRM_FORMAT_XRGB8888:
1118 case DRM_FORMAT_RGBX8888:
1119 case DRM_FORMAT_BGRX8888:
1120 case DRM_FORMAT_RGB888:
1121 case DRM_FORMAT_RGB565:
1122 case DRM_FORMAT_ARGB1555:
1123 case DRM_FORMAT_RGBA5551:
1124 case DRM_FORMAT_BGRA5551:
1125 case DRM_FORMAT_UYVY:
1126 case DRM_FORMAT_XYUV8888:
1127 case DRM_FORMAT_XVYU2101010:
1128 case DRM_FORMAT_X0L2:
1129 case DRM_FORMAT_X0L0:
1130 return true;
1131 default:
1132 return false;
1133 }
1134 }
1135
malidp_hw_format_is_afbc_only(u32 format)1136 bool malidp_hw_format_is_afbc_only(u32 format)
1137 {
1138 switch (format) {
1139 case DRM_FORMAT_VUY888:
1140 case DRM_FORMAT_VUY101010:
1141 case DRM_FORMAT_YUV420_8BIT:
1142 case DRM_FORMAT_YUV420_10BIT:
1143 return true;
1144 default:
1145 return false;
1146 }
1147 }
1148
malidp_hw_clear_irq(struct malidp_hw_device * hwdev,u8 block,u32 irq)1149 static void malidp_hw_clear_irq(struct malidp_hw_device *hwdev, u8 block, u32 irq)
1150 {
1151 u32 base = malidp_get_block_base(hwdev, block);
1152
1153 if (hwdev->hw->map.features & MALIDP_REGMAP_HAS_CLEARIRQ)
1154 malidp_hw_write(hwdev, irq, base + MALIDP_REG_CLEARIRQ);
1155 else
1156 malidp_hw_write(hwdev, irq, base + MALIDP_REG_STATUS);
1157 }
1158
malidp_de_irq(int irq,void * arg)1159 static irqreturn_t malidp_de_irq(int irq, void *arg)
1160 {
1161 struct drm_device *drm = arg;
1162 struct malidp_drm *malidp = drm->dev_private;
1163 struct malidp_hw_device *hwdev;
1164 struct malidp_hw *hw;
1165 const struct malidp_irq_map *de;
1166 u32 status, mask, dc_status;
1167 irqreturn_t ret = IRQ_NONE;
1168
1169 hwdev = malidp->dev;
1170 hw = hwdev->hw;
1171 de = &hw->map.de_irq_map;
1172
1173 /*
1174 * if we are suspended it is likely that we were invoked because
1175 * we share an interrupt line with some other driver, don't try
1176 * to read the hardware registers
1177 */
1178 if (hwdev->pm_suspended)
1179 return IRQ_NONE;
1180
1181 /* first handle the config valid IRQ */
1182 dc_status = malidp_hw_read(hwdev, hw->map.dc_base + MALIDP_REG_STATUS);
1183 if (dc_status & hw->map.dc_irq_map.vsync_irq) {
1184 malidp_hw_clear_irq(hwdev, MALIDP_DC_BLOCK, dc_status);
1185 /* do we have a page flip event? */
1186 if (malidp->event != NULL) {
1187 spin_lock(&drm->event_lock);
1188 drm_crtc_send_vblank_event(&malidp->crtc, malidp->event);
1189 malidp->event = NULL;
1190 spin_unlock(&drm->event_lock);
1191 }
1192 atomic_set(&malidp->config_valid, MALIDP_CONFIG_VALID_DONE);
1193 ret = IRQ_WAKE_THREAD;
1194 }
1195
1196 status = malidp_hw_read(hwdev, MALIDP_REG_STATUS);
1197 if (!(status & de->irq_mask))
1198 return ret;
1199
1200 mask = malidp_hw_read(hwdev, MALIDP_REG_MASKIRQ);
1201 /* keep the status of the enabled interrupts, plus the error bits */
1202 status &= (mask | de->err_mask);
1203 if ((status & de->vsync_irq) && malidp->crtc.enabled)
1204 drm_crtc_handle_vblank(&malidp->crtc);
1205
1206 #ifdef CONFIG_DEBUG_FS
1207 if (status & de->err_mask) {
1208 malidp_error(malidp, &malidp->de_errors, status,
1209 drm_crtc_vblank_count(&malidp->crtc));
1210 }
1211 #endif
1212 malidp_hw_clear_irq(hwdev, MALIDP_DE_BLOCK, status);
1213
1214 return (ret == IRQ_NONE) ? IRQ_HANDLED : ret;
1215 }
1216
malidp_de_irq_thread_handler(int irq,void * arg)1217 static irqreturn_t malidp_de_irq_thread_handler(int irq, void *arg)
1218 {
1219 struct drm_device *drm = arg;
1220 struct malidp_drm *malidp = drm->dev_private;
1221
1222 wake_up(&malidp->wq);
1223
1224 return IRQ_HANDLED;
1225 }
1226
malidp_de_irq_hw_init(struct malidp_hw_device * hwdev)1227 void malidp_de_irq_hw_init(struct malidp_hw_device *hwdev)
1228 {
1229 /* ensure interrupts are disabled */
1230 malidp_hw_disable_irq(hwdev, MALIDP_DE_BLOCK, 0xffffffff);
1231 malidp_hw_clear_irq(hwdev, MALIDP_DE_BLOCK, 0xffffffff);
1232 malidp_hw_disable_irq(hwdev, MALIDP_DC_BLOCK, 0xffffffff);
1233 malidp_hw_clear_irq(hwdev, MALIDP_DC_BLOCK, 0xffffffff);
1234
1235 /* first enable the DC block IRQs */
1236 malidp_hw_enable_irq(hwdev, MALIDP_DC_BLOCK,
1237 hwdev->hw->map.dc_irq_map.irq_mask);
1238
1239 /* now enable the DE block IRQs */
1240 malidp_hw_enable_irq(hwdev, MALIDP_DE_BLOCK,
1241 hwdev->hw->map.de_irq_map.irq_mask);
1242 }
1243
malidp_de_irq_init(struct drm_device * drm,int irq)1244 int malidp_de_irq_init(struct drm_device *drm, int irq)
1245 {
1246 struct malidp_drm *malidp = drm->dev_private;
1247 struct malidp_hw_device *hwdev = malidp->dev;
1248 int ret;
1249
1250 /* ensure interrupts are disabled */
1251 malidp_hw_disable_irq(hwdev, MALIDP_DE_BLOCK, 0xffffffff);
1252 malidp_hw_clear_irq(hwdev, MALIDP_DE_BLOCK, 0xffffffff);
1253 malidp_hw_disable_irq(hwdev, MALIDP_DC_BLOCK, 0xffffffff);
1254 malidp_hw_clear_irq(hwdev, MALIDP_DC_BLOCK, 0xffffffff);
1255
1256 ret = devm_request_threaded_irq(drm->dev, irq, malidp_de_irq,
1257 malidp_de_irq_thread_handler,
1258 IRQF_SHARED, "malidp-de", drm);
1259 if (ret < 0) {
1260 DRM_ERROR("failed to install DE IRQ handler\n");
1261 return ret;
1262 }
1263
1264 malidp_de_irq_hw_init(hwdev);
1265
1266 return 0;
1267 }
1268
malidp_de_irq_fini(struct malidp_hw_device * hwdev)1269 void malidp_de_irq_fini(struct malidp_hw_device *hwdev)
1270 {
1271 malidp_hw_disable_irq(hwdev, MALIDP_DE_BLOCK,
1272 hwdev->hw->map.de_irq_map.irq_mask);
1273 malidp_hw_disable_irq(hwdev, MALIDP_DC_BLOCK,
1274 hwdev->hw->map.dc_irq_map.irq_mask);
1275 }
1276
malidp_se_irq(int irq,void * arg)1277 static irqreturn_t malidp_se_irq(int irq, void *arg)
1278 {
1279 struct drm_device *drm = arg;
1280 struct malidp_drm *malidp = drm->dev_private;
1281 struct malidp_hw_device *hwdev = malidp->dev;
1282 struct malidp_hw *hw = hwdev->hw;
1283 const struct malidp_irq_map *se = &hw->map.se_irq_map;
1284 u32 status, mask;
1285
1286 /*
1287 * if we are suspended it is likely that we were invoked because
1288 * we share an interrupt line with some other driver, don't try
1289 * to read the hardware registers
1290 */
1291 if (hwdev->pm_suspended)
1292 return IRQ_NONE;
1293
1294 status = malidp_hw_read(hwdev, hw->map.se_base + MALIDP_REG_STATUS);
1295 if (!(status & (se->irq_mask | se->err_mask)))
1296 return IRQ_NONE;
1297
1298 #ifdef CONFIG_DEBUG_FS
1299 if (status & se->err_mask)
1300 malidp_error(malidp, &malidp->se_errors, status,
1301 drm_crtc_vblank_count(&malidp->crtc));
1302 #endif
1303 mask = malidp_hw_read(hwdev, hw->map.se_base + MALIDP_REG_MASKIRQ);
1304 status &= mask;
1305
1306 if (status & se->vsync_irq) {
1307 switch (hwdev->mw_state) {
1308 case MW_ONESHOT:
1309 drm_writeback_signal_completion(&malidp->mw_connector, 0);
1310 break;
1311 case MW_STOP:
1312 drm_writeback_signal_completion(&malidp->mw_connector, 0);
1313 /* disable writeback after stop */
1314 hwdev->mw_state = MW_NOT_ENABLED;
1315 break;
1316 case MW_RESTART:
1317 drm_writeback_signal_completion(&malidp->mw_connector, 0);
1318 /* fall through - to a new start */
1319 case MW_START:
1320 /* writeback started, need to emulate one-shot mode */
1321 hw->disable_memwrite(hwdev);
1322 /*
1323 * only set config_valid HW bit if there is no other update
1324 * in progress or if we raced ahead of the DE IRQ handler
1325 * and config_valid flag will not be update until later
1326 */
1327 status = malidp_hw_read(hwdev, hw->map.dc_base + MALIDP_REG_STATUS);
1328 if ((atomic_read(&malidp->config_valid) != MALIDP_CONFIG_START) ||
1329 (status & hw->map.dc_irq_map.vsync_irq))
1330 hw->set_config_valid(hwdev, 1);
1331 break;
1332 }
1333 }
1334
1335 malidp_hw_clear_irq(hwdev, MALIDP_SE_BLOCK, status);
1336
1337 return IRQ_HANDLED;
1338 }
1339
malidp_se_irq_hw_init(struct malidp_hw_device * hwdev)1340 void malidp_se_irq_hw_init(struct malidp_hw_device *hwdev)
1341 {
1342 /* ensure interrupts are disabled */
1343 malidp_hw_disable_irq(hwdev, MALIDP_SE_BLOCK, 0xffffffff);
1344 malidp_hw_clear_irq(hwdev, MALIDP_SE_BLOCK, 0xffffffff);
1345
1346 malidp_hw_enable_irq(hwdev, MALIDP_SE_BLOCK,
1347 hwdev->hw->map.se_irq_map.irq_mask);
1348 }
1349
malidp_se_irq_thread_handler(int irq,void * arg)1350 static irqreturn_t malidp_se_irq_thread_handler(int irq, void *arg)
1351 {
1352 return IRQ_HANDLED;
1353 }
1354
malidp_se_irq_init(struct drm_device * drm,int irq)1355 int malidp_se_irq_init(struct drm_device *drm, int irq)
1356 {
1357 struct malidp_drm *malidp = drm->dev_private;
1358 struct malidp_hw_device *hwdev = malidp->dev;
1359 int ret;
1360
1361 /* ensure interrupts are disabled */
1362 malidp_hw_disable_irq(hwdev, MALIDP_SE_BLOCK, 0xffffffff);
1363 malidp_hw_clear_irq(hwdev, MALIDP_SE_BLOCK, 0xffffffff);
1364
1365 ret = devm_request_threaded_irq(drm->dev, irq, malidp_se_irq,
1366 malidp_se_irq_thread_handler,
1367 IRQF_SHARED, "malidp-se", drm);
1368 if (ret < 0) {
1369 DRM_ERROR("failed to install SE IRQ handler\n");
1370 return ret;
1371 }
1372
1373 hwdev->mw_state = MW_NOT_ENABLED;
1374 malidp_se_irq_hw_init(hwdev);
1375
1376 return 0;
1377 }
1378
malidp_se_irq_fini(struct malidp_hw_device * hwdev)1379 void malidp_se_irq_fini(struct malidp_hw_device *hwdev)
1380 {
1381 malidp_hw_disable_irq(hwdev, MALIDP_SE_BLOCK,
1382 hwdev->hw->map.se_irq_map.irq_mask);
1383 }
1384