xref: /linux/drivers/gpu/drm/drm_panic.c (revision 569d7db70e5dcf13fbf072f10e9096577ac1e565)
1 // SPDX-License-Identifier: GPL-2.0 or MIT
2 /*
3  * Copyright (c) 2023 Red Hat.
4  * Author: Jocelyn Falempe <jfalempe@redhat.com>
5  * inspired by the drm_log driver from David Herrmann <dh.herrmann@gmail.com>
6  * Tux Ascii art taken from cowsay written by Tony Monroe
7  */
8 
9 #include <linux/font.h>
10 #include <linux/iosys-map.h>
11 #include <linux/kdebug.h>
12 #include <linux/kmsg_dump.h>
13 #include <linux/list.h>
14 #include <linux/module.h>
15 #include <linux/printk.h>
16 #include <linux/types.h>
17 
18 #include <drm/drm_drv.h>
19 #include <drm/drm_format_helper.h>
20 #include <drm/drm_fourcc.h>
21 #include <drm/drm_framebuffer.h>
22 #include <drm/drm_modeset_helper_vtables.h>
23 #include <drm/drm_panic.h>
24 #include <drm/drm_plane.h>
25 #include <drm/drm_print.h>
26 
27 MODULE_AUTHOR("Jocelyn Falempe");
28 MODULE_DESCRIPTION("DRM panic handler");
29 MODULE_LICENSE("GPL");
30 
31 static char drm_panic_screen[16] = CONFIG_DRM_PANIC_SCREEN;
32 module_param_string(panic_screen, drm_panic_screen, sizeof(drm_panic_screen), 0644);
33 MODULE_PARM_DESC(panic_screen,
34 		 "Choose what will be displayed by drm_panic, 'user' or 'kmsg' [default="
35 		 CONFIG_DRM_PANIC_SCREEN "]");
36 
37 /**
38  * DOC: overview
39  *
40  * To enable DRM panic for a driver, the primary plane must implement a
41  * &drm_plane_helper_funcs.get_scanout_buffer helper function. It is then
42  * automatically registered to the drm panic handler.
43  * When a panic occurs, the &drm_plane_helper_funcs.get_scanout_buffer will be
44  * called, and the driver can provide a framebuffer so the panic handler can
45  * draw the panic screen on it. Currently only linear buffer and a few color
46  * formats are supported.
47  * Optionally the driver can also provide a &drm_plane_helper_funcs.panic_flush
48  * callback, that will be called after that, to send additional commands to the
49  * hardware to make the scanout buffer visible.
50  */
51 
52 /*
53  * This module displays a user friendly message on screen when a kernel panic
54  * occurs. This is conflicting with fbcon, so you can only enable it when fbcon
55  * is disabled.
56  * It's intended for end-user, so have minimal technical/debug information.
57  *
58  * Implementation details:
59  *
60  * It is a panic handler, so it can't take lock, allocate memory, run tasks/irq,
61  * or attempt to sleep. It's a best effort, and it may not be able to display
62  * the message in all situations (like if the panic occurs in the middle of a
63  * modesetting).
64  * It will display only one static frame, so performance optimizations are low
65  * priority as the machine is already in an unusable state.
66  */
67 
68 struct drm_panic_line {
69 	u32 len;
70 	const char *txt;
71 };
72 
73 #define PANIC_LINE(s) {.len = sizeof(s) - 1, .txt = s}
74 
75 static struct drm_panic_line panic_msg[] = {
76 	PANIC_LINE("KERNEL PANIC !"),
77 	PANIC_LINE(""),
78 	PANIC_LINE("Please reboot your computer."),
79 };
80 
81 static const struct drm_panic_line logo[] = {
82 	PANIC_LINE("     .--.        _"),
83 	PANIC_LINE("    |o_o |      | |"),
84 	PANIC_LINE("    |:_/ |      | |"),
85 	PANIC_LINE("   //   \\ \\     |_|"),
86 	PANIC_LINE("  (|     | )     _"),
87 	PANIC_LINE(" /'\\_   _/`\\    (_)"),
88 	PANIC_LINE(" \\___)=(___/"),
89 };
90 
91 /*
92  * Color conversion
93  */
94 
95 static u16 convert_xrgb8888_to_rgb565(u32 pix)
96 {
97 	return ((pix & 0x00F80000) >> 8) |
98 	       ((pix & 0x0000FC00) >> 5) |
99 	       ((pix & 0x000000F8) >> 3);
100 }
101 
102 static u16 convert_xrgb8888_to_rgba5551(u32 pix)
103 {
104 	return ((pix & 0x00f80000) >> 8) |
105 	       ((pix & 0x0000f800) >> 5) |
106 	       ((pix & 0x000000f8) >> 2) |
107 	       BIT(0); /* set alpha bit */
108 }
109 
110 static u16 convert_xrgb8888_to_xrgb1555(u32 pix)
111 {
112 	return ((pix & 0x00f80000) >> 9) |
113 	       ((pix & 0x0000f800) >> 6) |
114 	       ((pix & 0x000000f8) >> 3);
115 }
116 
117 static u16 convert_xrgb8888_to_argb1555(u32 pix)
118 {
119 	return BIT(15) | /* set alpha bit */
120 	       ((pix & 0x00f80000) >> 9) |
121 	       ((pix & 0x0000f800) >> 6) |
122 	       ((pix & 0x000000f8) >> 3);
123 }
124 
125 static u32 convert_xrgb8888_to_argb8888(u32 pix)
126 {
127 	return pix | GENMASK(31, 24); /* fill alpha bits */
128 }
129 
130 static u32 convert_xrgb8888_to_xbgr8888(u32 pix)
131 {
132 	return ((pix & 0x00ff0000) >> 16) <<  0 |
133 	       ((pix & 0x0000ff00) >>  8) <<  8 |
134 	       ((pix & 0x000000ff) >>  0) << 16 |
135 	       ((pix & 0xff000000) >> 24) << 24;
136 }
137 
138 static u32 convert_xrgb8888_to_abgr8888(u32 pix)
139 {
140 	return ((pix & 0x00ff0000) >> 16) <<  0 |
141 	       ((pix & 0x0000ff00) >>  8) <<  8 |
142 	       ((pix & 0x000000ff) >>  0) << 16 |
143 	       GENMASK(31, 24); /* fill alpha bits */
144 }
145 
146 static u32 convert_xrgb8888_to_xrgb2101010(u32 pix)
147 {
148 	pix = ((pix & 0x000000FF) << 2) |
149 	      ((pix & 0x0000FF00) << 4) |
150 	      ((pix & 0x00FF0000) << 6);
151 	return pix | ((pix >> 8) & 0x00300C03);
152 }
153 
154 static u32 convert_xrgb8888_to_argb2101010(u32 pix)
155 {
156 	pix = ((pix & 0x000000FF) << 2) |
157 	      ((pix & 0x0000FF00) << 4) |
158 	      ((pix & 0x00FF0000) << 6);
159 	return GENMASK(31, 30) /* set alpha bits */ | pix | ((pix >> 8) & 0x00300C03);
160 }
161 
162 /*
163  * convert_from_xrgb8888 - convert one pixel from xrgb8888 to the desired format
164  * @color: input color, in xrgb8888 format
165  * @format: output format
166  *
167  * Returns:
168  * Color in the format specified, casted to u32.
169  * Or 0 if the format is not supported.
170  */
171 static u32 convert_from_xrgb8888(u32 color, u32 format)
172 {
173 	switch (format) {
174 	case DRM_FORMAT_RGB565:
175 		return convert_xrgb8888_to_rgb565(color);
176 	case DRM_FORMAT_RGBA5551:
177 		return convert_xrgb8888_to_rgba5551(color);
178 	case DRM_FORMAT_XRGB1555:
179 		return convert_xrgb8888_to_xrgb1555(color);
180 	case DRM_FORMAT_ARGB1555:
181 		return convert_xrgb8888_to_argb1555(color);
182 	case DRM_FORMAT_RGB888:
183 	case DRM_FORMAT_XRGB8888:
184 		return color;
185 	case DRM_FORMAT_ARGB8888:
186 		return convert_xrgb8888_to_argb8888(color);
187 	case DRM_FORMAT_XBGR8888:
188 		return convert_xrgb8888_to_xbgr8888(color);
189 	case DRM_FORMAT_ABGR8888:
190 		return convert_xrgb8888_to_abgr8888(color);
191 	case DRM_FORMAT_XRGB2101010:
192 		return convert_xrgb8888_to_xrgb2101010(color);
193 	case DRM_FORMAT_ARGB2101010:
194 		return convert_xrgb8888_to_argb2101010(color);
195 	default:
196 		WARN_ONCE(1, "Can't convert to %p4cc\n", &format);
197 		return 0;
198 	}
199 }
200 
201 /*
202  * Blit & Fill
203  */
204 /* check if the pixel at coord x,y is 1 (foreground) or 0 (background) */
205 static bool drm_panic_is_pixel_fg(const u8 *sbuf8, unsigned int spitch, int x, int y)
206 {
207 	return (sbuf8[(y * spitch) + x / 8] & (0x80 >> (x % 8))) != 0;
208 }
209 
210 static void drm_panic_blit16(struct iosys_map *dmap, unsigned int dpitch,
211 			     const u8 *sbuf8, unsigned int spitch,
212 			     unsigned int height, unsigned int width,
213 			     u16 fg16)
214 {
215 	unsigned int y, x;
216 
217 	for (y = 0; y < height; y++)
218 		for (x = 0; x < width; x++)
219 			if (drm_panic_is_pixel_fg(sbuf8, spitch, x, y))
220 				iosys_map_wr(dmap, y * dpitch + x * sizeof(u16), u16, fg16);
221 }
222 
223 static void drm_panic_blit24(struct iosys_map *dmap, unsigned int dpitch,
224 			     const u8 *sbuf8, unsigned int spitch,
225 			     unsigned int height, unsigned int width,
226 			     u32 fg32)
227 {
228 	unsigned int y, x;
229 
230 	for (y = 0; y < height; y++) {
231 		for (x = 0; x < width; x++) {
232 			u32 off = y * dpitch + x * 3;
233 
234 			if (drm_panic_is_pixel_fg(sbuf8, spitch, x, y)) {
235 				/* write blue-green-red to output in little endianness */
236 				iosys_map_wr(dmap, off, u8, (fg32 & 0x000000FF) >> 0);
237 				iosys_map_wr(dmap, off + 1, u8, (fg32 & 0x0000FF00) >> 8);
238 				iosys_map_wr(dmap, off + 2, u8, (fg32 & 0x00FF0000) >> 16);
239 			}
240 		}
241 	}
242 }
243 
244 static void drm_panic_blit32(struct iosys_map *dmap, unsigned int dpitch,
245 			     const u8 *sbuf8, unsigned int spitch,
246 			     unsigned int height, unsigned int width,
247 			     u32 fg32)
248 {
249 	unsigned int y, x;
250 
251 	for (y = 0; y < height; y++)
252 		for (x = 0; x < width; x++)
253 			if (drm_panic_is_pixel_fg(sbuf8, spitch, x, y))
254 				iosys_map_wr(dmap, y * dpitch + x * sizeof(u32), u32, fg32);
255 }
256 
257 static void drm_panic_blit_pixel(struct drm_scanout_buffer *sb, struct drm_rect *clip,
258 				 const u8 *sbuf8, unsigned int spitch, u32 fg_color)
259 {
260 	unsigned int y, x;
261 
262 	for (y = 0; y < drm_rect_height(clip); y++)
263 		for (x = 0; x < drm_rect_width(clip); x++)
264 			if (drm_panic_is_pixel_fg(sbuf8, spitch, x, y))
265 				sb->set_pixel(sb, clip->x1 + x, clip->y1 + y, fg_color);
266 }
267 
268 /*
269  * drm_panic_blit - convert a monochrome image to a linear framebuffer
270  * @sb: destination scanout buffer
271  * @clip: destination rectangle
272  * @sbuf8: source buffer, in monochrome format, 8 pixels per byte.
273  * @spitch: source pitch in bytes
274  * @fg_color: foreground color, in destination format
275  *
276  * This can be used to draw a font character, which is a monochrome image, to a
277  * framebuffer in other supported format.
278  */
279 static void drm_panic_blit(struct drm_scanout_buffer *sb, struct drm_rect *clip,
280 			   const u8 *sbuf8, unsigned int spitch, u32 fg_color)
281 {
282 	struct iosys_map map;
283 
284 	if (sb->set_pixel)
285 		return drm_panic_blit_pixel(sb, clip, sbuf8, spitch, fg_color);
286 
287 	map = sb->map[0];
288 	iosys_map_incr(&map, clip->y1 * sb->pitch[0] + clip->x1 * sb->format->cpp[0]);
289 
290 	switch (sb->format->cpp[0]) {
291 	case 2:
292 		drm_panic_blit16(&map, sb->pitch[0], sbuf8, spitch,
293 				 drm_rect_height(clip), drm_rect_width(clip), fg_color);
294 	break;
295 	case 3:
296 		drm_panic_blit24(&map, sb->pitch[0], sbuf8, spitch,
297 				 drm_rect_height(clip), drm_rect_width(clip), fg_color);
298 	break;
299 	case 4:
300 		drm_panic_blit32(&map, sb->pitch[0], sbuf8, spitch,
301 				 drm_rect_height(clip), drm_rect_width(clip), fg_color);
302 	break;
303 	default:
304 		WARN_ONCE(1, "Can't blit with pixel width %d\n", sb->format->cpp[0]);
305 	}
306 }
307 
308 static void drm_panic_fill16(struct iosys_map *dmap, unsigned int dpitch,
309 			     unsigned int height, unsigned int width,
310 			     u16 color)
311 {
312 	unsigned int y, x;
313 
314 	for (y = 0; y < height; y++)
315 		for (x = 0; x < width; x++)
316 			iosys_map_wr(dmap, y * dpitch + x * sizeof(u16), u16, color);
317 }
318 
319 static void drm_panic_fill24(struct iosys_map *dmap, unsigned int dpitch,
320 			     unsigned int height, unsigned int width,
321 			     u32 color)
322 {
323 	unsigned int y, x;
324 
325 	for (y = 0; y < height; y++) {
326 		for (x = 0; x < width; x++) {
327 			unsigned int off = y * dpitch + x * 3;
328 
329 			/* write blue-green-red to output in little endianness */
330 			iosys_map_wr(dmap, off, u8, (color & 0x000000FF) >> 0);
331 			iosys_map_wr(dmap, off + 1, u8, (color & 0x0000FF00) >> 8);
332 			iosys_map_wr(dmap, off + 2, u8, (color & 0x00FF0000) >> 16);
333 		}
334 	}
335 }
336 
337 static void drm_panic_fill32(struct iosys_map *dmap, unsigned int dpitch,
338 			     unsigned int height, unsigned int width,
339 			     u32 color)
340 {
341 	unsigned int y, x;
342 
343 	for (y = 0; y < height; y++)
344 		for (x = 0; x < width; x++)
345 			iosys_map_wr(dmap, y * dpitch + x * sizeof(u32), u32, color);
346 }
347 
348 static void drm_panic_fill_pixel(struct drm_scanout_buffer *sb,
349 				 struct drm_rect *clip,
350 				 u32 color)
351 {
352 	unsigned int y, x;
353 
354 	for (y = 0; y < drm_rect_height(clip); y++)
355 		for (x = 0; x < drm_rect_width(clip); x++)
356 			sb->set_pixel(sb, clip->x1 + x, clip->y1 + y, color);
357 }
358 
359 /*
360  * drm_panic_fill - Fill a rectangle with a color
361  * @sb: destination scanout buffer
362  * @clip: destination rectangle
363  * @color: foreground color, in destination format
364  *
365  * Fill a rectangle with a color, in a linear framebuffer.
366  */
367 static void drm_panic_fill(struct drm_scanout_buffer *sb, struct drm_rect *clip,
368 			   u32 color)
369 {
370 	struct iosys_map map;
371 
372 	if (sb->set_pixel)
373 		return drm_panic_fill_pixel(sb, clip, color);
374 
375 	map = sb->map[0];
376 	iosys_map_incr(&map, clip->y1 * sb->pitch[0] + clip->x1 * sb->format->cpp[0]);
377 
378 	switch (sb->format->cpp[0]) {
379 	case 2:
380 		drm_panic_fill16(&map, sb->pitch[0], drm_rect_height(clip),
381 				 drm_rect_width(clip), color);
382 	break;
383 	case 3:
384 		drm_panic_fill24(&map, sb->pitch[0], drm_rect_height(clip),
385 				 drm_rect_width(clip), color);
386 	break;
387 	case 4:
388 		drm_panic_fill32(&map, sb->pitch[0], drm_rect_height(clip),
389 				 drm_rect_width(clip), color);
390 	break;
391 	default:
392 		WARN_ONCE(1, "Can't fill with pixel width %d\n", sb->format->cpp[0]);
393 	}
394 }
395 
396 static const u8 *get_char_bitmap(const struct font_desc *font, char c, size_t font_pitch)
397 {
398 	return font->data + (c * font->height) * font_pitch;
399 }
400 
401 static unsigned int get_max_line_len(const struct drm_panic_line *lines, int len)
402 {
403 	int i;
404 	unsigned int max = 0;
405 
406 	for (i = 0; i < len; i++)
407 		max = max(lines[i].len, max);
408 	return max;
409 }
410 
411 /*
412  * Draw a text in a rectangle on a framebuffer. The text is truncated if it overflows the rectangle
413  */
414 static void draw_txt_rectangle(struct drm_scanout_buffer *sb,
415 			       const struct font_desc *font,
416 			       const struct drm_panic_line *msg,
417 			       unsigned int msg_lines,
418 			       bool centered,
419 			       struct drm_rect *clip,
420 			       u32 color)
421 {
422 	int i, j;
423 	const u8 *src;
424 	size_t font_pitch = DIV_ROUND_UP(font->width, 8);
425 	struct drm_rect rec;
426 
427 	msg_lines = min(msg_lines,  drm_rect_height(clip) / font->height);
428 	for (i = 0; i < msg_lines; i++) {
429 		size_t line_len = min(msg[i].len, drm_rect_width(clip) / font->width);
430 
431 		rec.y1 = clip->y1 +  i * font->height;
432 		rec.y2 = rec.y1 + font->height;
433 		rec.x1 = clip->x1;
434 
435 		if (centered)
436 			rec.x1 += (drm_rect_width(clip) - (line_len * font->width)) / 2;
437 
438 		for (j = 0; j < line_len; j++) {
439 			src = get_char_bitmap(font, msg[i].txt[j], font_pitch);
440 			rec.x2 = rec.x1 + font->width;
441 			drm_panic_blit(sb, &rec, src, font_pitch, color);
442 			rec.x1 += font->width;
443 		}
444 	}
445 }
446 
447 static void draw_panic_static_user(struct drm_scanout_buffer *sb)
448 {
449 	size_t msg_lines = ARRAY_SIZE(panic_msg);
450 	size_t logo_lines = ARRAY_SIZE(logo);
451 	u32 fg_color = convert_from_xrgb8888(CONFIG_DRM_PANIC_FOREGROUND_COLOR, sb->format->format);
452 	u32 bg_color = convert_from_xrgb8888(CONFIG_DRM_PANIC_BACKGROUND_COLOR, sb->format->format);
453 	const struct font_desc *font = get_default_font(sb->width, sb->height, NULL, NULL);
454 	struct drm_rect r_screen, r_logo, r_msg;
455 
456 	if (!font)
457 		return;
458 
459 	r_screen = DRM_RECT_INIT(0, 0, sb->width, sb->height);
460 
461 	r_logo = DRM_RECT_INIT(0, 0,
462 			       get_max_line_len(logo, logo_lines) * font->width,
463 			       logo_lines * font->height);
464 	r_msg = DRM_RECT_INIT(0, 0,
465 			      min(get_max_line_len(panic_msg, msg_lines) * font->width, sb->width),
466 			      min(msg_lines * font->height, sb->height));
467 
468 	/* Center the panic message */
469 	drm_rect_translate(&r_msg, (sb->width - r_msg.x2) / 2, (sb->height - r_msg.y2) / 2);
470 
471 	/* Fill with the background color, and draw text on top */
472 	drm_panic_fill(sb, &r_screen, bg_color);
473 
474 	if ((r_msg.x1 >= drm_rect_width(&r_logo) || r_msg.y1 >= drm_rect_height(&r_logo)) &&
475 	    drm_rect_width(&r_logo) < sb->width && drm_rect_height(&r_logo) < sb->height) {
476 		draw_txt_rectangle(sb, font, logo, logo_lines, false, &r_logo, fg_color);
477 	}
478 	draw_txt_rectangle(sb, font, panic_msg, msg_lines, true, &r_msg, fg_color);
479 }
480 
481 /*
482  * Draw one line of kmsg, and handle wrapping if it won't fit in the screen width.
483  * Return the y-offset of the next line.
484  */
485 static int draw_line_with_wrap(struct drm_scanout_buffer *sb, const struct font_desc *font,
486 			       struct drm_panic_line *line, int yoffset, u32 fg_color)
487 {
488 	int chars_per_row = sb->width / font->width;
489 	struct drm_rect r_txt = DRM_RECT_INIT(0, yoffset, sb->width, sb->height);
490 	struct drm_panic_line line_wrap;
491 
492 	if (line->len > chars_per_row) {
493 		line_wrap.len = line->len % chars_per_row;
494 		line_wrap.txt = line->txt + line->len - line_wrap.len;
495 		draw_txt_rectangle(sb, font, &line_wrap, 1, false, &r_txt, fg_color);
496 		r_txt.y1 -= font->height;
497 		if (r_txt.y1 < 0)
498 			return r_txt.y1;
499 		while (line_wrap.txt > line->txt) {
500 			line_wrap.txt -= chars_per_row;
501 			line_wrap.len = chars_per_row;
502 			draw_txt_rectangle(sb, font, &line_wrap, 1, false, &r_txt, fg_color);
503 			r_txt.y1 -= font->height;
504 			if (r_txt.y1 < 0)
505 				return r_txt.y1;
506 		}
507 	} else {
508 		draw_txt_rectangle(sb, font, line, 1, false, &r_txt, fg_color);
509 		r_txt.y1 -= font->height;
510 	}
511 	return r_txt.y1;
512 }
513 
514 /*
515  * Draw the kmsg buffer to the screen, starting from the youngest message at the bottom,
516  * and going up until reaching the top of the screen.
517  */
518 static void draw_panic_static_kmsg(struct drm_scanout_buffer *sb)
519 {
520 	u32 fg_color = convert_from_xrgb8888(CONFIG_DRM_PANIC_FOREGROUND_COLOR, sb->format->format);
521 	u32 bg_color = convert_from_xrgb8888(CONFIG_DRM_PANIC_BACKGROUND_COLOR, sb->format->format);
522 	const struct font_desc *font = get_default_font(sb->width, sb->height, NULL, NULL);
523 	struct drm_rect r_screen = DRM_RECT_INIT(0, 0, sb->width, sb->height);
524 	struct kmsg_dump_iter iter;
525 	char kmsg_buf[512];
526 	size_t kmsg_len;
527 	struct drm_panic_line line;
528 	int yoffset;
529 
530 	if (!font)
531 		return;
532 
533 	yoffset = sb->height - font->height - (sb->height % font->height) / 2;
534 
535 	/* Fill with the background color, and draw text on top */
536 	drm_panic_fill(sb, &r_screen, bg_color);
537 
538 	kmsg_dump_rewind(&iter);
539 	while (kmsg_dump_get_buffer(&iter, false, kmsg_buf, sizeof(kmsg_buf), &kmsg_len)) {
540 		char *start;
541 		char *end;
542 
543 		/* ignore terminating NUL and newline */
544 		start = kmsg_buf + kmsg_len - 2;
545 		end = kmsg_buf + kmsg_len - 1;
546 		while (start > kmsg_buf && yoffset >= 0) {
547 			while (start > kmsg_buf && *start != '\n')
548 				start--;
549 			/* don't count the newline character */
550 			line.txt = start + (start == kmsg_buf ? 0 : 1);
551 			line.len = end - line.txt;
552 
553 			yoffset = draw_line_with_wrap(sb, font, &line, yoffset, fg_color);
554 			end = start;
555 			start--;
556 		}
557 	}
558 }
559 
560 /*
561  * drm_panic_is_format_supported()
562  * @format: a fourcc color code
563  * Returns: true if supported, false otherwise.
564  *
565  * Check if drm_panic will be able to use this color format.
566  */
567 static bool drm_panic_is_format_supported(const struct drm_format_info *format)
568 {
569 	if (format->num_planes != 1)
570 		return false;
571 	return convert_from_xrgb8888(0xffffff, format->format) != 0;
572 }
573 
574 static void draw_panic_dispatch(struct drm_scanout_buffer *sb)
575 {
576 	if (!strcmp(drm_panic_screen, "kmsg")) {
577 		draw_panic_static_kmsg(sb);
578 	} else {
579 		draw_panic_static_user(sb);
580 	}
581 }
582 
583 static void draw_panic_plane(struct drm_plane *plane)
584 {
585 	struct drm_scanout_buffer sb;
586 	int ret;
587 	unsigned long flags;
588 
589 	if (!drm_panic_trylock(plane->dev, flags))
590 		return;
591 
592 	ret = plane->helper_private->get_scanout_buffer(plane, &sb);
593 
594 	if (!ret && drm_panic_is_format_supported(sb.format)) {
595 		draw_panic_dispatch(&sb);
596 		if (plane->helper_private->panic_flush)
597 			plane->helper_private->panic_flush(plane);
598 	}
599 	drm_panic_unlock(plane->dev, flags);
600 }
601 
602 static struct drm_plane *to_drm_plane(struct kmsg_dumper *kd)
603 {
604 	return container_of(kd, struct drm_plane, kmsg_panic);
605 }
606 
607 static void drm_panic(struct kmsg_dumper *dumper, enum kmsg_dump_reason reason)
608 {
609 	struct drm_plane *plane = to_drm_plane(dumper);
610 
611 	if (reason == KMSG_DUMP_PANIC)
612 		draw_panic_plane(plane);
613 }
614 
615 
616 /*
617  * DEBUG FS, This is currently unsafe.
618  * Create one file per plane, so it's possible to debug one plane at a time.
619  * TODO: It would be better to emulate an NMI context.
620  */
621 #ifdef CONFIG_DRM_PANIC_DEBUG
622 #include <linux/debugfs.h>
623 
624 static ssize_t debugfs_trigger_write(struct file *file, const char __user *user_buf,
625 				     size_t count, loff_t *ppos)
626 {
627 	bool run;
628 
629 	if (kstrtobool_from_user(user_buf, count, &run) == 0 && run) {
630 		struct drm_plane *plane = file->private_data;
631 
632 		draw_panic_plane(plane);
633 	}
634 	return count;
635 }
636 
637 static const struct file_operations dbg_drm_panic_ops = {
638 	.owner = THIS_MODULE,
639 	.write = debugfs_trigger_write,
640 	.open = simple_open,
641 };
642 
643 static void debugfs_register_plane(struct drm_plane *plane, int index)
644 {
645 	char fname[32];
646 
647 	snprintf(fname, 32, "drm_panic_plane_%d", index);
648 	debugfs_create_file(fname, 0200, plane->dev->debugfs_root,
649 			    plane, &dbg_drm_panic_ops);
650 }
651 #else
652 static void debugfs_register_plane(struct drm_plane *plane, int index) {}
653 #endif /* CONFIG_DRM_PANIC_DEBUG */
654 
655 /**
656  * drm_panic_register() - Initialize DRM panic for a device
657  * @dev: the drm device on which the panic screen will be displayed.
658  */
659 void drm_panic_register(struct drm_device *dev)
660 {
661 	struct drm_plane *plane;
662 	int registered_plane = 0;
663 
664 	if (!dev->mode_config.num_total_plane)
665 		return;
666 
667 	drm_for_each_plane(plane, dev) {
668 		if (!plane->helper_private || !plane->helper_private->get_scanout_buffer)
669 			continue;
670 		plane->kmsg_panic.dump = drm_panic;
671 		plane->kmsg_panic.max_reason = KMSG_DUMP_PANIC;
672 		if (kmsg_dump_register(&plane->kmsg_panic))
673 			drm_warn(dev, "Failed to register panic handler\n");
674 		else {
675 			debugfs_register_plane(plane, registered_plane);
676 			registered_plane++;
677 		}
678 	}
679 	if (registered_plane)
680 		drm_info(dev, "Registered %d planes with drm panic\n", registered_plane);
681 }
682 EXPORT_SYMBOL(drm_panic_register);
683 
684 /**
685  * drm_panic_unregister()
686  * @dev: the drm device previously registered.
687  */
688 void drm_panic_unregister(struct drm_device *dev)
689 {
690 	struct drm_plane *plane;
691 
692 	if (!dev->mode_config.num_total_plane)
693 		return;
694 
695 	drm_for_each_plane(plane, dev) {
696 		if (!plane->helper_private || !plane->helper_private->get_scanout_buffer)
697 			continue;
698 		kmsg_dump_unregister(&plane->kmsg_panic);
699 	}
700 }
701 EXPORT_SYMBOL(drm_panic_unregister);
702