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