Commit c0ce26a8 authored by Andrew Morton's avatar Andrew Morton Committed by Linus Torvalds

[PATCH] Fix VT mode change vs. fbcon

From: Benjamin Herrenschmidt <benh@kernel.crashing.org>

This patch implements what we discussed earlier to fix the switch bewteen
KD_GRAPHICS and KD_TEXT.

It has been tested for a few days now and appear to resolve the problem for
affected users.

James: I know you have some objections, I don't fully agree with them, and
I want that in asap now, that bug has been plaguing fbdev since the very
beginning and it's time to get rid of that and my corresponding todolist
entry.  You are welcome to propose a patch on top of this one if you feel
you can make things cleaner.  The approach of adding a parameter to
con_blank() is Linus idea btw :)

I didn't add a separate function as that would have made the butchering of
drivers/char/vt beyond what I want to deal with in 2.6.
parent 3d5daf73
...@@ -2744,11 +2744,11 @@ static void vesa_powerdown(void) ...@@ -2744,11 +2744,11 @@ static void vesa_powerdown(void)
*/ */
switch (vesa_blank_mode) { switch (vesa_blank_mode) {
case VESA_NO_BLANKING: case VESA_NO_BLANKING:
c->vc_sw->con_blank(c, VESA_VSYNC_SUSPEND+1); c->vc_sw->con_blank(c, VESA_VSYNC_SUSPEND+1, 0);
break; break;
case VESA_VSYNC_SUSPEND: case VESA_VSYNC_SUSPEND:
case VESA_HSYNC_SUSPEND: case VESA_HSYNC_SUSPEND:
c->vc_sw->con_blank(c, VESA_POWERDOWN+1); c->vc_sw->con_blank(c, VESA_POWERDOWN+1, 0);
break; break;
} }
} }
...@@ -2776,7 +2776,7 @@ void do_blank_screen(int entering_gfx) ...@@ -2776,7 +2776,7 @@ void do_blank_screen(int entering_gfx)
if (entering_gfx) { if (entering_gfx) {
hide_cursor(currcons); hide_cursor(currcons);
save_screen(currcons); save_screen(currcons);
sw->con_blank(vc_cons[currcons].d, -1); sw->con_blank(vc_cons[currcons].d, -1, 1);
console_blanked = fg_console + 1; console_blanked = fg_console + 1;
set_origin(currcons); set_origin(currcons);
return; return;
...@@ -2794,7 +2794,7 @@ void do_blank_screen(int entering_gfx) ...@@ -2794,7 +2794,7 @@ void do_blank_screen(int entering_gfx)
save_screen(currcons); save_screen(currcons);
/* In case we need to reset origin, blanking hook returns 1 */ /* In case we need to reset origin, blanking hook returns 1 */
i = sw->con_blank(vc_cons[currcons].d, 1); i = sw->con_blank(vc_cons[currcons].d, 1, 0);
console_blanked = fg_console + 1; console_blanked = fg_console + 1;
if (i) if (i)
set_origin(currcons); set_origin(currcons);
...@@ -2808,14 +2808,14 @@ void do_blank_screen(int entering_gfx) ...@@ -2808,14 +2808,14 @@ void do_blank_screen(int entering_gfx)
} }
if (vesa_blank_mode) if (vesa_blank_mode)
sw->con_blank(vc_cons[currcons].d, vesa_blank_mode + 1); sw->con_blank(vc_cons[currcons].d, vesa_blank_mode + 1, 0);
} }
/* /*
* Called by timer as well as from vt_console_driver * Called by timer as well as from vt_console_driver
*/ */
void unblank_screen(void) void do_unblank_screen(int leaving_gfx)
{ {
int currcons; int currcons;
...@@ -2839,7 +2839,7 @@ void unblank_screen(void) ...@@ -2839,7 +2839,7 @@ void unblank_screen(void)
} }
console_blanked = 0; console_blanked = 0;
if (sw->con_blank(vc_cons[currcons].d, 0)) if (sw->con_blank(vc_cons[currcons].d, 0, leaving_gfx))
/* Low-level driver cannot restore -> do it ourselves */ /* Low-level driver cannot restore -> do it ourselves */
update_screen(fg_console); update_screen(fg_console);
if (console_blank_hook) if (console_blank_hook)
...@@ -2848,6 +2848,17 @@ void unblank_screen(void) ...@@ -2848,6 +2848,17 @@ void unblank_screen(void)
set_cursor(fg_console); set_cursor(fg_console);
} }
/*
* This is called by the outside world to cause a forced unblank, mostly for
* oopses. Currently, I just call do_unblank_screen(0), but we could eventually
* call it with 1 as an argument and so force a mode restore... that may kill
* X or at least garbage the screen but would also make the Oops visible...
*/
void unblank_screen(void)
{
do_unblank_screen(0);
}
/* /*
* We defer the timer blanking to work queue so it can take the console semaphore * We defer the timer blanking to work queue so it can take the console semaphore
* (console operations can still happen at irq time, but only from printk which * (console operations can still happen at irq time, but only from printk which
......
...@@ -497,7 +497,7 @@ int vt_ioctl(struct tty_struct *tty, struct file * file, ...@@ -497,7 +497,7 @@ int vt_ioctl(struct tty_struct *tty, struct file * file,
*/ */
acquire_console_sem(); acquire_console_sem();
if (arg == KD_TEXT) if (arg == KD_TEXT)
unblank_screen(); do_unblank_screen(1);
else else
do_blank_screen(1); do_blank_screen(1);
release_console_sem(); release_console_sem();
...@@ -1103,7 +1103,7 @@ void complete_change_console(unsigned int new_console) ...@@ -1103,7 +1103,7 @@ void complete_change_console(unsigned int new_console)
if (old_vc_mode != vt_cons[new_console]->vc_mode) if (old_vc_mode != vt_cons[new_console]->vc_mode)
{ {
if (vt_cons[new_console]->vc_mode == KD_TEXT) if (vt_cons[new_console]->vc_mode == KD_TEXT)
unblank_screen(); do_unblank_screen(1);
else else
do_blank_screen(1); do_blank_screen(1);
} }
...@@ -1138,7 +1138,7 @@ void complete_change_console(unsigned int new_console) ...@@ -1138,7 +1138,7 @@ void complete_change_console(unsigned int new_console)
if (old_vc_mode != vt_cons[new_console]->vc_mode) if (old_vc_mode != vt_cons[new_console]->vc_mode)
{ {
if (vt_cons[new_console]->vc_mode == KD_TEXT) if (vt_cons[new_console]->vc_mode == KD_TEXT)
unblank_screen(); do_unblank_screen(1);
else else
do_blank_screen(1); do_blank_screen(1);
} }
......
...@@ -159,7 +159,7 @@ static int fbcon_scroll(struct vc_data *vc, int t, int b, int dir, ...@@ -159,7 +159,7 @@ static int fbcon_scroll(struct vc_data *vc, int t, int b, int dir,
static void fbcon_bmove(struct vc_data *vc, int sy, int sx, int dy, int dx, static void fbcon_bmove(struct vc_data *vc, int sy, int sx, int dy, int dx,
int height, int width); int height, int width);
static int fbcon_switch(struct vc_data *vc); static int fbcon_switch(struct vc_data *vc);
static int fbcon_blank(struct vc_data *vc, int blank); static int fbcon_blank(struct vc_data *vc, int blank, int mode_switch);
static int fbcon_font_op(struct vc_data *vc, struct console_font_op *op); static int fbcon_font_op(struct vc_data *vc, struct console_font_op *op);
static int fbcon_set_palette(struct vc_data *vc, unsigned char *table); static int fbcon_set_palette(struct vc_data *vc, unsigned char *table);
static int fbcon_scrolldelta(struct vc_data *vc, int lines); static int fbcon_scrolldelta(struct vc_data *vc, int lines);
...@@ -1697,14 +1697,23 @@ static int fbcon_switch(struct vc_data *vc) ...@@ -1697,14 +1697,23 @@ static int fbcon_switch(struct vc_data *vc)
return 1; return 1;
} }
static int fbcon_blank(struct vc_data *vc, int blank) static int fbcon_blank(struct vc_data *vc, int blank, int mode_switch)
{ {
unsigned short charmask = vc->vc_hi_font_mask ? 0x1ff : 0xff; unsigned short charmask = vc->vc_hi_font_mask ? 0x1ff : 0xff;
struct fb_info *info = registered_fb[(int) con2fb_map[vc->vc_num]]; struct fb_info *info = registered_fb[(int) con2fb_map[vc->vc_num]];
struct display *p = &fb_display[vc->vc_num]; struct display *p = &fb_display[vc->vc_num];
if (blank < 0) /* Entering graphics mode */ if (mode_switch) {
struct fb_info *info = registered_fb[(int) con2fb_map[vc->vc_num]];
struct fb_var_screeninfo var = info->var;
if (blank) {
fbcon_cursor(vc, CM_ERASE);
return 0; return 0;
}
var.activate = FB_ACTIVATE_NOW | FB_ACTIVATE_FORCE;
fb_set_var(info, &var);
}
fbcon_cursor(vc, blank ? CM_ERASE : CM_DRAW); fbcon_cursor(vc, blank ? CM_ERASE : CM_DRAW);
......
...@@ -463,7 +463,7 @@ promcon_font_op(struct vc_data *conp, struct console_font_op *op) ...@@ -463,7 +463,7 @@ promcon_font_op(struct vc_data *conp, struct console_font_op *op)
} }
static int static int
promcon_blank(struct vc_data *conp, int blank) promcon_blank(struct vc_data *conp, int blank, int mode_switch)
{ {
if (blank) { if (blank) {
promcon_puts("\033[H\033[J\033[7m \033[m\b", 15); promcon_puts("\033[H\033[J\033[7m \033[m\b", 15);
......
...@@ -250,26 +250,18 @@ static int sticon_set_origin(struct vc_data *conp) ...@@ -250,26 +250,18 @@ static int sticon_set_origin(struct vc_data *conp)
return 0; return 0;
} }
static int sticon_blank(struct vc_data *c, int blank) static int sticon_blank(struct vc_data *c, int blank, int mode_switch)
{ {
switch (blank) { if (blank == 0) {
case 0: /* unblank */ if (mode_switch)
vga_is_gfx = 0; vga_is_gfx = 0;
/* Tell console.c that it has to restore the screen itself */
return 1; return 1;
case 1: /* normal blanking */ }
default: /* VESA blanking */
if (vga_is_gfx)
return 0;
sticon_set_origin(c); sticon_set_origin(c);
sti_clear(sticon_sti, 0,0, c->vc_rows, c->vc_cols, BLANK); sti_clear(sticon_sti, 0,0, c->vc_rows, c->vc_cols, BLANK);
return 1; if (mode_switch)
case -1: /* Entering graphic mode */
sti_clear(sticon_sti, 0,0, c->vc_rows, c->vc_cols, BLANK);
vga_is_gfx = 1; vga_is_gfx = 1;
return 1; return 1;
}
return 1; /* console needs to restore screen itself */
} }
static int sticon_scrolldelta(struct vc_data *conp, int lines) static int sticon_scrolldelta(struct vc_data *conp, int lines)
......
...@@ -76,7 +76,7 @@ static void vgacon_init(struct vc_data *c, int init); ...@@ -76,7 +76,7 @@ static void vgacon_init(struct vc_data *c, int init);
static void vgacon_deinit(struct vc_data *c); static void vgacon_deinit(struct vc_data *c);
static void vgacon_cursor(struct vc_data *c, int mode); static void vgacon_cursor(struct vc_data *c, int mode);
static int vgacon_switch(struct vc_data *c); static int vgacon_switch(struct vc_data *c);
static int vgacon_blank(struct vc_data *c, int blank); static int vgacon_blank(struct vc_data *c, int blank, int mode_switch);
static int vgacon_font_op(struct vc_data *c, struct console_font_op *op); static int vgacon_font_op(struct vc_data *c, struct console_font_op *op);
static int vgacon_set_palette(struct vc_data *vc, unsigned char *table); static int vgacon_set_palette(struct vc_data *vc, unsigned char *table);
static int vgacon_scrolldelta(struct vc_data *c, int lines); static int vgacon_scrolldelta(struct vc_data *c, int lines);
...@@ -661,7 +661,7 @@ static void vga_pal_blank(struct vgastate *state) ...@@ -661,7 +661,7 @@ static void vga_pal_blank(struct vgastate *state)
} }
} }
static int vgacon_blank(struct vc_data *c, int blank) static int vgacon_blank(struct vc_data *c, int blank, int mode_switch)
{ {
switch (blank) { switch (blank) {
case 0: /* Unblank */ case 0: /* Unblank */
...@@ -678,7 +678,8 @@ static int vgacon_blank(struct vc_data *c, int blank) ...@@ -678,7 +678,8 @@ static int vgacon_blank(struct vc_data *c, int blank)
/* Tell console.c that it has to restore the screen itself */ /* Tell console.c that it has to restore the screen itself */
return 1; return 1;
case 1: /* Normal blanking */ case 1: /* Normal blanking */
if (vga_video_type == VIDEO_TYPE_VGAC) { case -1: /* Obsolete */
if (!mode_switch && vga_video_type == VIDEO_TYPE_VGAC) {
vga_pal_blank(&state); vga_pal_blank(&state);
vga_palette_blanked = 1; vga_palette_blanked = 1;
return 0; return 0;
...@@ -686,10 +687,7 @@ static int vgacon_blank(struct vc_data *c, int blank) ...@@ -686,10 +687,7 @@ static int vgacon_blank(struct vc_data *c, int blank)
vgacon_set_origin(c); vgacon_set_origin(c);
scr_memsetw((void *) vga_vram_base, BLANK, scr_memsetw((void *) vga_vram_base, BLANK,
c->vc_screenbuf_size); c->vc_screenbuf_size);
return 1; if (mode_switch)
case -1: /* Entering graphic mode */
scr_memsetw((void *) vga_vram_base, BLANK,
c->vc_screenbuf_size);
vga_is_gfx = 1; vga_is_gfx = 1;
return 1; return 1;
default: /* VESA blanking */ default: /* VESA blanking */
......
...@@ -955,7 +955,8 @@ fb_set_var(struct fb_info *info, struct fb_var_screeninfo *var) ...@@ -955,7 +955,8 @@ fb_set_var(struct fb_info *info, struct fb_var_screeninfo *var)
{ {
int err; int err;
if (memcmp(&info->var, var, sizeof(struct fb_var_screeninfo))) { if ((var->activate & FB_ACTIVATE_FORCE) ||
memcmp(&info->var, var, sizeof(struct fb_var_screeninfo))) {
if (!info->fbops->fb_check_var) { if (!info->fbops->fb_check_var) {
*var = info->var; *var = info->var;
return 0; return 0;
......
...@@ -37,7 +37,7 @@ struct consw { ...@@ -37,7 +37,7 @@ struct consw {
int (*con_scroll)(struct vc_data *, int, int, int, int); int (*con_scroll)(struct vc_data *, int, int, int, int);
void (*con_bmove)(struct vc_data *, int, int, int, int, int, int); void (*con_bmove)(struct vc_data *, int, int, int, int, int, int);
int (*con_switch)(struct vc_data *); int (*con_switch)(struct vc_data *);
int (*con_blank)(struct vc_data *, int); int (*con_blank)(struct vc_data *, int, int);
int (*con_font_op)(struct vc_data *, struct console_font_op *); int (*con_font_op)(struct vc_data *, struct console_font_op *);
int (*con_resize)(struct vc_data *, unsigned int, unsigned int); int (*con_resize)(struct vc_data *, unsigned int, unsigned int);
int (*con_set_palette)(struct vc_data *, unsigned char *); int (*con_set_palette)(struct vc_data *, unsigned char *);
......
...@@ -152,6 +152,7 @@ struct fb_bitfield { ...@@ -152,6 +152,7 @@ struct fb_bitfield {
#define FB_ACTIVATE_VBL 16 /* activate values on next vbl */ #define FB_ACTIVATE_VBL 16 /* activate values on next vbl */
#define FB_CHANGE_CMAP_VBL 32 /* change colormap on vbl */ #define FB_CHANGE_CMAP_VBL 32 /* change colormap on vbl */
#define FB_ACTIVATE_ALL 64 /* change all VCs on this fb */ #define FB_ACTIVATE_ALL 64 /* change all VCs on this fb */
#define FB_ACTIVATE_FORCE 128 /* force apply even when no change*/
#define FB_ACCELF_TEXT 1 /* text mode acceleration */ #define FB_ACCELF_TEXT 1 /* text mode acceleration */
......
...@@ -44,7 +44,8 @@ int vc_resize(int currcons, unsigned int cols, unsigned int lines); ...@@ -44,7 +44,8 @@ int vc_resize(int currcons, unsigned int cols, unsigned int lines);
void vc_disallocate(unsigned int console); void vc_disallocate(unsigned int console);
void reset_palette(int currcons); void reset_palette(int currcons);
void set_palette(int currcons); void set_palette(int currcons);
void do_blank_screen(int gfx_mode); void do_blank_screen(int entering_gfx);
void do_unblank_screen(int leaving_gfx);
void unblank_screen(void); void unblank_screen(void);
void poke_blanked_console(void); void poke_blanked_console(void);
int con_font_op(int currcons, struct console_font_op *op); int con_font_op(int currcons, struct console_font_op *op);
......
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