Commit 15418471 authored by Dave Airlie's avatar Dave Airlie Committed by Ben Hutchings

vgacon/vt: clear buffer attributes when we load a 512 character font (v2)

commit 2a248307 upstream.

When we switch from 256->512 byte font rendering mode, it means the
current contents of the screen is being reinterpreted. The bit that holds
the high bit of the 9-bit font, may have been previously set, and thus
the new font misrenders.

The problem case we see is grub2 writes spaces with the bit set, so it
ends up with data like 0x820, which gets reinterpreted into 0x120 char
which the font translates into G with a circumflex. This flashes up on
screen at boot and is quite ugly.

A current side effect of this patch though is that any rendering on the
screen changes color to a slightly darker color, but at least the screen
no longer corrupts.

v2: as suggested by hpa, always clear the attribute space, whether we
are are going to or from 512 chars.
Signed-off-by: default avatarDave Airlie <airlied@redhat.com>
Signed-off-by: default avatarBen Hutchings <ben@decadent.org.uk>
parent 6684f6e7
...@@ -657,7 +657,7 @@ static inline void save_screen(struct vc_data *vc) ...@@ -657,7 +657,7 @@ static inline void save_screen(struct vc_data *vc)
* Redrawing of screen * Redrawing of screen
*/ */
static void clear_buffer_attributes(struct vc_data *vc) void clear_buffer_attributes(struct vc_data *vc)
{ {
unsigned short *p = (unsigned short *)vc->vc_origin; unsigned short *p = (unsigned short *)vc->vc_origin;
int count = vc->vc_screenbuf_size / 2; int count = vc->vc_screenbuf_size / 2;
......
...@@ -1064,7 +1064,7 @@ static int vgacon_do_font_op(struct vgastate *state,char *arg,int set,int ch512) ...@@ -1064,7 +1064,7 @@ static int vgacon_do_font_op(struct vgastate *state,char *arg,int set,int ch512)
unsigned short video_port_status = vga_video_port_reg + 6; unsigned short video_port_status = vga_video_port_reg + 6;
int font_select = 0x00, beg, i; int font_select = 0x00, beg, i;
char *charmap; char *charmap;
bool clear_attribs = false;
if (vga_video_type != VIDEO_TYPE_EGAM) { if (vga_video_type != VIDEO_TYPE_EGAM) {
charmap = (char *) VGA_MAP_MEM(colourmap, 0); charmap = (char *) VGA_MAP_MEM(colourmap, 0);
beg = 0x0e; beg = 0x0e;
...@@ -1169,12 +1169,6 @@ static int vgacon_do_font_op(struct vgastate *state,char *arg,int set,int ch512) ...@@ -1169,12 +1169,6 @@ static int vgacon_do_font_op(struct vgastate *state,char *arg,int set,int ch512)
/* if 512 char mode is already enabled don't re-enable it. */ /* if 512 char mode is already enabled don't re-enable it. */
if ((set) && (ch512 != vga_512_chars)) { if ((set) && (ch512 != vga_512_chars)) {
/* attribute controller */
for (i = 0; i < MAX_NR_CONSOLES; i++) {
struct vc_data *c = vc_cons[i].d;
if (c && c->vc_sw == &vga_con)
c->vc_hi_font_mask = ch512 ? 0x0800 : 0;
}
vga_512_chars = ch512; vga_512_chars = ch512;
/* 256-char: enable intensity bit /* 256-char: enable intensity bit
512-char: disable intensity bit */ 512-char: disable intensity bit */
...@@ -1185,8 +1179,22 @@ static int vgacon_do_font_op(struct vgastate *state,char *arg,int set,int ch512) ...@@ -1185,8 +1179,22 @@ static int vgacon_do_font_op(struct vgastate *state,char *arg,int set,int ch512)
it means, but it works, and it appears necessary */ it means, but it works, and it appears necessary */
inb_p(video_port_status); inb_p(video_port_status);
vga_wattr(state->vgabase, VGA_AR_ENABLE_DISPLAY, 0); vga_wattr(state->vgabase, VGA_AR_ENABLE_DISPLAY, 0);
clear_attribs = true;
} }
raw_spin_unlock_irq(&vga_lock); raw_spin_unlock_irq(&vga_lock);
if (clear_attribs) {
for (i = 0; i < MAX_NR_CONSOLES; i++) {
struct vc_data *c = vc_cons[i].d;
if (c && c->vc_sw == &vga_con) {
/* force hi font mask to 0, so we always clear
the bit on either transition */
c->vc_hi_font_mask = 0x00;
clear_buffer_attributes(c);
c->vc_hi_font_mask = ch512 ? 0x0800 : 0;
}
}
}
return 0; return 0;
} }
......
...@@ -47,6 +47,7 @@ int con_set_cmap(unsigned char __user *cmap); ...@@ -47,6 +47,7 @@ int con_set_cmap(unsigned char __user *cmap);
int con_get_cmap(unsigned char __user *cmap); int con_get_cmap(unsigned char __user *cmap);
void scrollback(struct vc_data *vc, int lines); void scrollback(struct vc_data *vc, int lines);
void scrollfront(struct vc_data *vc, int lines); void scrollfront(struct vc_data *vc, int lines);
void clear_buffer_attributes(struct vc_data *vc);
void update_region(struct vc_data *vc, unsigned long start, int count); void update_region(struct vc_data *vc, unsigned long start, int count);
void redraw_screen(struct vc_data *vc, int is_switch); void redraw_screen(struct vc_data *vc, int is_switch);
#define update_screen(x) redraw_screen(x, 0) #define update_screen(x) redraw_screen(x, 0)
......
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