Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux
at master 360 lines 11 kB view raw
1// SPDX-License-Identifier: MIT 2/* 3 * Permission is hereby granted, free of charge, to any person obtaining a 4 * copy of this software and associated documentation files (the 5 * "Software"), to deal in the Software without restriction, including 6 * without limitation the rights to use, copy, modify, merge, publish, 7 * distribute, sub license, and/or sell copies of the Software, and to 8 * permit persons to whom the Software is furnished to do so, subject to 9 * the following conditions: 10 * 11 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 12 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 13 * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL 14 * THE COPYRIGHT HOLDERS, AUTHORS AND/OR ITS SUPPLIERS BE LIABLE FOR ANY CLAIM, 15 * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR 16 * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE 17 * USE OR OTHER DEALINGS IN THE SOFTWARE. 18 * 19 * The above copyright notice and this permission notice (including the 20 * next paragraph) shall be included in all copies or substantial portions 21 * of the Software. 22 */ 23 24#include <linux/bits.h> 25#include <linux/sizes.h> 26 27#include <drm/drm_atomic.h> 28#include <drm/drm_damage_helper.h> 29#include <drm/drm_format_helper.h> 30#include <drm/drm_gem_atomic_helper.h> 31#include <drm/drm_gem_framebuffer_helper.h> 32#include <drm/drm_print.h> 33 34#include "ast_drv.h" 35 36/* 37 * Hardware cursor 38 */ 39 40/* define for signature structure */ 41#define AST_HWC_SIGNATURE_SIZE SZ_32 42#define AST_HWC_SIGNATURE_CHECKSUM 0x00 43#define AST_HWC_SIGNATURE_SizeX 0x04 44#define AST_HWC_SIGNATURE_SizeY 0x08 45#define AST_HWC_SIGNATURE_X 0x0C 46#define AST_HWC_SIGNATURE_Y 0x10 47#define AST_HWC_SIGNATURE_HOTSPOTX 0x14 48#define AST_HWC_SIGNATURE_HOTSPOTY 0x18 49 50static unsigned long ast_cursor_vram_size(void) 51{ 52 return AST_HWC_SIZE + AST_HWC_SIGNATURE_SIZE; 53} 54 55long ast_cursor_vram_offset(struct ast_device *ast) 56{ 57 unsigned long size = ast_cursor_vram_size(); 58 59 if (size > ast->vram_size) 60 return -EINVAL; 61 62 return ALIGN_DOWN(ast->vram_size - size, SZ_8); 63} 64 65static u32 ast_cursor_calculate_checksum(const void *src, unsigned int width, unsigned int height) 66{ 67 u32 csum = 0; 68 unsigned int one_pixel_copy = width & BIT(0); 69 unsigned int two_pixel_copy = width - one_pixel_copy; 70 unsigned int trailing_bytes = (AST_MAX_HWC_WIDTH - width) * sizeof(u16); 71 unsigned int x, y; 72 73 for (y = 0; y < height; y++) { 74 for (x = 0; x < two_pixel_copy; x += 2) { 75 const u32 *src32 = (const u32 *)src; 76 77 csum += *src32; 78 src += SZ_4; 79 } 80 if (one_pixel_copy) { 81 const u16 *src16 = (const u16 *)src; 82 83 csum += *src16; 84 src += SZ_2; 85 } 86 src += trailing_bytes; 87 } 88 89 return csum; 90} 91 92static void ast_set_cursor_image(struct ast_device *ast, const u8 *src, 93 unsigned int width, unsigned int height) 94{ 95 u8 __iomem *dst = ast_plane_vaddr(&ast->cursor_plane.base); 96 u32 csum = ast_cursor_calculate_checksum(src, width, height); 97 98 /* write pixel data */ 99#if defined(__BIG_ENDIAN) 100 unsigned int i; 101 102 for (i = 0; i < AST_HWC_SIZE; i += 2) 103 writew(swab16(*(const __u16 *)&src[i]), &dst[i]); 104#else 105 memcpy_toio(dst, src, AST_HWC_SIZE); 106#endif 107 108 /* write checksum + signature */ 109 dst += AST_HWC_SIZE; 110 writel(csum, dst); 111 writel(width, dst + AST_HWC_SIGNATURE_SizeX); 112 writel(height, dst + AST_HWC_SIGNATURE_SizeY); 113 writel(0, dst + AST_HWC_SIGNATURE_HOTSPOTX); 114 writel(0, dst + AST_HWC_SIGNATURE_HOTSPOTY); 115} 116 117static void ast_set_cursor_base(struct ast_device *ast, u64 address) 118{ 119 u8 addr0 = (address >> 3) & 0xff; 120 u8 addr1 = (address >> 11) & 0xff; 121 u8 addr2 = (address >> 19) & 0xff; 122 123 ast_set_index_reg(ast, AST_IO_VGACRI, 0xc8, addr0); 124 ast_set_index_reg(ast, AST_IO_VGACRI, 0xc9, addr1); 125 ast_set_index_reg(ast, AST_IO_VGACRI, 0xca, addr2); 126} 127 128static void ast_set_cursor_location(struct ast_device *ast, u16 x, u16 y, 129 u8 x_offset, u8 y_offset) 130{ 131 u8 x0 = (x & 0x00ff); 132 u8 x1 = (x & 0x0f00) >> 8; 133 u8 y0 = (y & 0x00ff); 134 u8 y1 = (y & 0x0700) >> 8; 135 136 ast_set_index_reg(ast, AST_IO_VGACRI, 0xc2, x_offset); 137 ast_set_index_reg(ast, AST_IO_VGACRI, 0xc3, y_offset); 138 ast_set_index_reg(ast, AST_IO_VGACRI, 0xc4, x0); 139 ast_set_index_reg(ast, AST_IO_VGACRI, 0xc5, x1); 140 ast_set_index_reg(ast, AST_IO_VGACRI, 0xc6, y0); 141 ast_set_index_reg(ast, AST_IO_VGACRI, 0xc7, y1); 142} 143 144static void ast_set_cursor_enabled(struct ast_device *ast, bool enabled) 145{ 146 static const u8 mask = (u8)~(AST_IO_VGACRCB_HWC_16BPP | 147 AST_IO_VGACRCB_HWC_ENABLED); 148 149 u8 vgacrcb = AST_IO_VGACRCB_HWC_16BPP; 150 151 if (enabled) 152 vgacrcb |= AST_IO_VGACRCB_HWC_ENABLED; 153 154 ast_set_index_reg_mask(ast, AST_IO_VGACRI, 0xcb, mask, vgacrcb); 155} 156 157/* 158 * Cursor plane 159 */ 160 161static const uint32_t ast_cursor_plane_formats[] = { 162 DRM_FORMAT_ARGB4444, 163 DRM_FORMAT_ARGB8888, 164}; 165 166static int ast_cursor_plane_helper_atomic_check(struct drm_plane *plane, 167 struct drm_atomic_state *state) 168{ 169 struct drm_plane_state *new_plane_state = drm_atomic_get_new_plane_state(state, plane); 170 struct drm_framebuffer *new_fb = new_plane_state->fb; 171 struct drm_crtc_state *new_crtc_state = NULL; 172 int ret; 173 174 if (new_plane_state->crtc) 175 new_crtc_state = drm_atomic_get_new_crtc_state(state, new_plane_state->crtc); 176 177 ret = drm_atomic_helper_check_plane_state(new_plane_state, new_crtc_state, 178 DRM_PLANE_NO_SCALING, 179 DRM_PLANE_NO_SCALING, 180 true, true); 181 if (ret || !new_plane_state->visible) 182 return ret; 183 184 if (new_fb->width > AST_MAX_HWC_WIDTH || new_fb->height > AST_MAX_HWC_HEIGHT) 185 return -EINVAL; 186 187 return 0; 188} 189 190static const u8 *ast_cursor_plane_get_argb4444(struct ast_cursor_plane *ast_cursor_plane, 191 struct drm_shadow_plane_state *shadow_plane_state, 192 const struct drm_rect *clip) 193{ 194 struct drm_plane_state *plane_state = &shadow_plane_state->base; 195 struct drm_framebuffer *fb = plane_state->fb; 196 u8 *argb4444 = NULL; 197 198 if (drm_gem_fb_begin_cpu_access(fb, DMA_FROM_DEVICE) == 0) { 199 switch (fb->format->format) { 200 case DRM_FORMAT_ARGB4444: 201 if (shadow_plane_state->data[0].is_iomem) { 202 struct iosys_map argb4444_dst[DRM_FORMAT_MAX_PLANES] = { 203 IOSYS_MAP_INIT_VADDR(ast_cursor_plane->argb4444), 204 }; 205 unsigned int argb4444_dst_pitch[DRM_FORMAT_MAX_PLANES] = { 206 AST_HWC_PITCH, 207 }; 208 209 drm_fb_memcpy(argb4444_dst, argb4444_dst_pitch, 210 shadow_plane_state->data, fb, clip); 211 argb4444 = argb4444_dst[0].vaddr; 212 } else { 213 argb4444 = shadow_plane_state->data[0].vaddr; 214 } 215 break; 216 case DRM_FORMAT_ARGB8888: 217 { 218 struct iosys_map argb4444_dst[DRM_FORMAT_MAX_PLANES] = { 219 IOSYS_MAP_INIT_VADDR(ast_cursor_plane->argb4444), 220 }; 221 unsigned int argb4444_dst_pitch[DRM_FORMAT_MAX_PLANES] = { 222 AST_HWC_PITCH, 223 }; 224 225 drm_fb_argb8888_to_argb4444(argb4444_dst, argb4444_dst_pitch, 226 shadow_plane_state->data, fb, clip, 227 &shadow_plane_state->fmtcnv_state); 228 argb4444 = argb4444_dst[0].vaddr; 229 } 230 break; 231 } 232 233 drm_gem_fb_end_cpu_access(fb, DMA_FROM_DEVICE); 234 } else { 235 /* 236 * Fall back to white square if GEM object is not ready. Gives 237 * the user an indication where the cursor is located. 238 */ 239 memset(ast_cursor_plane->argb4444, 0xff, sizeof(ast_cursor_plane->argb4444)); 240 argb4444 = ast_cursor_plane->argb4444; 241 } 242 243 return argb4444; 244} 245 246static void ast_cursor_plane_helper_atomic_update(struct drm_plane *plane, 247 struct drm_atomic_state *state) 248{ 249 struct ast_cursor_plane *ast_cursor_plane = to_ast_cursor_plane(plane); 250 struct ast_plane *ast_plane = to_ast_plane(plane); 251 struct drm_plane_state *plane_state = drm_atomic_get_new_plane_state(state, plane); 252 struct drm_shadow_plane_state *shadow_plane_state = to_drm_shadow_plane_state(plane_state); 253 struct drm_framebuffer *fb = plane_state->fb; 254 struct drm_plane_state *old_plane_state = drm_atomic_get_old_plane_state(state, plane); 255 struct ast_device *ast = to_ast_device(plane->dev); 256 struct drm_rect damage; 257 u64 dst_off = ast_plane->offset; 258 u8 __iomem *dst = ast_plane_vaddr(ast_plane); /* TODO: Use mapping abstraction properly */ 259 u8 __iomem *sig = dst + AST_HWC_SIZE; /* TODO: Use mapping abstraction properly */ 260 unsigned int offset_x, offset_y; 261 u16 x, y; 262 u8 x_offset, y_offset; 263 264 /* 265 * Do data transfer to hardware buffer and point the scanout 266 * engine to the offset. 267 */ 268 269 if (drm_atomic_helper_damage_merged(old_plane_state, plane_state, &damage)) { 270 const u8 *argb4444 = ast_cursor_plane_get_argb4444(ast_cursor_plane, 271 shadow_plane_state, 272 &damage); 273 274 if (argb4444) 275 ast_set_cursor_image(ast, argb4444, fb->width, fb->height); 276 277 ast_set_cursor_base(ast, dst_off); 278 } 279 280 /* 281 * Update location in HWC signature and registers. 282 */ 283 284 writel(plane_state->crtc_x, sig + AST_HWC_SIGNATURE_X); 285 writel(plane_state->crtc_y, sig + AST_HWC_SIGNATURE_Y); 286 287 offset_x = AST_MAX_HWC_WIDTH - fb->width; 288 offset_y = AST_MAX_HWC_HEIGHT - fb->height; 289 290 if (plane_state->crtc_x < 0) { 291 x_offset = (-plane_state->crtc_x) + offset_x; 292 x = 0; 293 } else { 294 x_offset = offset_x; 295 x = plane_state->crtc_x; 296 } 297 if (plane_state->crtc_y < 0) { 298 y_offset = (-plane_state->crtc_y) + offset_y; 299 y = 0; 300 } else { 301 y_offset = offset_y; 302 y = plane_state->crtc_y; 303 } 304 305 ast_set_cursor_location(ast, x, y, x_offset, y_offset); 306 307 /* Dummy write to enable HWC and make the HW pick-up the changes. */ 308 ast_set_cursor_enabled(ast, true); 309} 310 311static void ast_cursor_plane_helper_atomic_disable(struct drm_plane *plane, 312 struct drm_atomic_state *state) 313{ 314 struct ast_device *ast = to_ast_device(plane->dev); 315 316 ast_set_cursor_enabled(ast, false); 317} 318 319static const struct drm_plane_helper_funcs ast_cursor_plane_helper_funcs = { 320 DRM_GEM_SHADOW_PLANE_HELPER_FUNCS, 321 .atomic_check = ast_cursor_plane_helper_atomic_check, 322 .atomic_update = ast_cursor_plane_helper_atomic_update, 323 .atomic_disable = ast_cursor_plane_helper_atomic_disable, 324}; 325 326static const struct drm_plane_funcs ast_cursor_plane_funcs = { 327 .update_plane = drm_atomic_helper_update_plane, 328 .disable_plane = drm_atomic_helper_disable_plane, 329 .destroy = drm_plane_cleanup, 330 DRM_GEM_SHADOW_PLANE_FUNCS, 331}; 332 333int ast_cursor_plane_init(struct ast_device *ast) 334{ 335 struct drm_device *dev = &ast->base; 336 struct ast_cursor_plane *ast_cursor_plane = &ast->cursor_plane; 337 struct ast_plane *ast_plane = &ast_cursor_plane->base; 338 struct drm_plane *cursor_plane = &ast_plane->base; 339 unsigned long size; 340 long offset; 341 int ret; 342 343 size = ast_cursor_vram_size(); 344 offset = ast_cursor_vram_offset(ast); 345 if (offset < 0) 346 return offset; 347 348 ret = ast_plane_init(dev, ast_plane, offset, size, 349 0x01, &ast_cursor_plane_funcs, 350 ast_cursor_plane_formats, ARRAY_SIZE(ast_cursor_plane_formats), 351 NULL, DRM_PLANE_TYPE_CURSOR); 352 if (ret) { 353 drm_err(dev, "ast_plane_init() failed: %d\n", ret); 354 return ret; 355 } 356 drm_plane_helper_add(cursor_plane, &ast_cursor_plane_helper_funcs); 357 drm_plane_enable_fb_damage_clips(cursor_plane); 358 359 return 0; 360}