Commit 3c9bedd1 authored by Antonino Daplas's avatar Antonino Daplas Committed by Linus Torvalds

[PATCH] fbcon: Do not touch hardware if vc_mode != KD_TEXT

The frambuffer console is still writing to the graphics hardware even if the
hardware is not owned anymore by the console.  This can lead to crashes with
some hardware.

A trace of the source indicates that it comes from almost anywhere, anytime.

The fix is to return immediately if vc_mode != KD_TEXT.  The test is placed in
the following functions if not already present.

fbcon_cursor()
fbcon_putcs()
fbcon_clear()
fbcon_bmove()
fbcon_blank()
Signed-off-by: default avatarAntonino Daplas <adaplas@pol.net>
Signed-off-by: default avatarAndrew Morton <akpm@osdl.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@osdl.org>
parent 857c46ac
...@@ -983,6 +983,9 @@ static void fbcon_clear(struct vc_data *vc, int sy, int sx, int height, ...@@ -983,6 +983,9 @@ static void fbcon_clear(struct vc_data *vc, int sy, int sx, int height,
if (info->state != FBINFO_STATE_RUNNING) if (info->state != FBINFO_STATE_RUNNING)
return; return;
if (vt_cons[vc->vc_num]->vc_mode != KD_TEXT)
return;
if (!height || !width) if (!height || !width)
return; return;
...@@ -1007,6 +1010,7 @@ static void fbcon_putcs(struct vc_data *vc, const unsigned short *s, ...@@ -1007,6 +1010,7 @@ static void fbcon_putcs(struct vc_data *vc, const unsigned short *s,
if (!info->fbops->fb_blank && console_blanked) if (!info->fbops->fb_blank && console_blanked)
return; return;
if (info->state != FBINFO_STATE_RUNNING) if (info->state != FBINFO_STATE_RUNNING)
return; return;
...@@ -1042,6 +1046,9 @@ static void fbcon_cursor(struct vc_data *vc, int mode) ...@@ -1042,6 +1046,9 @@ static void fbcon_cursor(struct vc_data *vc, int mode)
int y = real_y(p, vc->vc_y); int y = real_y(p, vc->vc_y);
int c = scr_readw((u16 *) vc->vc_pos); int c = scr_readw((u16 *) vc->vc_pos);
if (vt_cons[vc->vc_num]->vc_mode != KD_TEXT)
return;
ops->cursor_flash = 1; ops->cursor_flash = 1;
if (mode & CM_SOFTBACK) { if (mode & CM_SOFTBACK) {
mode &= ~CM_SOFTBACK; mode &= ~CM_SOFTBACK;
...@@ -1690,6 +1697,9 @@ static void fbcon_bmove(struct vc_data *vc, int sy, int sx, int dy, int dx, ...@@ -1690,6 +1697,9 @@ static void fbcon_bmove(struct vc_data *vc, int sy, int sx, int dy, int dx,
if (!info->fbops->fb_blank && console_blanked) if (!info->fbops->fb_blank && console_blanked)
return; return;
if (vt_cons[vc->vc_num]->vc_mode != KD_TEXT)
return;
if (!width || !height) if (!width || !height)
return; return;
...@@ -1952,6 +1962,7 @@ static int fbcon_blank(struct vc_data *vc, int blank, int mode_switch) ...@@ -1952,6 +1962,7 @@ static int fbcon_blank(struct vc_data *vc, int blank, int mode_switch)
struct fb_info *info = registered_fb[con2fb_map[vc->vc_num]]; struct fb_info *info = registered_fb[con2fb_map[vc->vc_num]];
struct fbcon_ops *ops = (struct fbcon_ops *) info->fbcon_par; struct fbcon_ops *ops = (struct fbcon_ops *) info->fbcon_par;
struct display *p = &fb_display[vc->vc_num]; struct display *p = &fb_display[vc->vc_num];
int retval = 0;
if (mode_switch) { if (mode_switch) {
struct fb_var_screeninfo var = info->var; struct fb_var_screeninfo var = info->var;
...@@ -1968,19 +1979,16 @@ static int fbcon_blank(struct vc_data *vc, int blank, int mode_switch) ...@@ -1968,19 +1979,16 @@ static int fbcon_blank(struct vc_data *vc, int blank, int mode_switch)
if (info->flags & FBINFO_MISC_MODESWITCHLATE) if (info->flags & FBINFO_MISC_MODESWITCHLATE)
info->flags |= FBINFO_MISC_MODESWITCH; info->flags |= FBINFO_MISC_MODESWITCH;
if (blank) { if (!blank && !(info->flags & FBINFO_MISC_MODESWITCHLATE)) {
fbcon_cursor(vc, CM_ERASE);
return 0;
}
if (!(info->flags & FBINFO_MISC_MODESWITCHLATE)) {
var.activate = FB_ACTIVATE_NOW | FB_ACTIVATE_FORCE; var.activate = FB_ACTIVATE_NOW | FB_ACTIVATE_FORCE;
fb_set_var(info, &var); fb_set_var(info, &var);
} }
return 0;
} }
fbcon_cursor(vc, blank ? CM_ERASE : CM_DRAW);
ops->cursor_flash = (!blank); ops->cursor_flash = (!blank);
fbcon_cursor(vc, blank ? CM_ERASE : CM_DRAW);
if (!info->fbops->fb_blank) { if (!info->fbops->fb_blank) {
if (blank) { if (blank) {
...@@ -2001,9 +2009,10 @@ static int fbcon_blank(struct vc_data *vc, int blank, int mode_switch) ...@@ -2001,9 +2009,10 @@ static int fbcon_blank(struct vc_data *vc, int blank, int mode_switch)
vc->vc_video_erase_char = oldc; vc->vc_video_erase_char = oldc;
} else } else
update_screen(vc->vc_num); update_screen(vc->vc_num);
return 0; } else if (vt_cons[vc->vc_num]->vc_mode == KD_TEXT)
} else retval = fb_blank(info, blank);
return fb_blank(info, blank);
return retval;
} }
static void fbcon_free_font(struct display *p) static void fbcon_free_font(struct display *p)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment