[PATCH] vesafb: Fix mtrr bugs
>> vesafb: mode is 800x600x16, linelength=1600, pages=16 >> vesafb: scrolling: redraw >> vesafb: Truecolor: size=0:5:6:5, shift=0:11:5:0 >> mtrr: type mismatch for fc000000,1000000 old: write-back new: write- >> combining Range is already set to write-back, vesafb attempts to add a write-combining mtrr (default for vesafb). >> mtrr: size and base must be multiples of 4 kiB This is a bug, vesafb attempts to add a size < PAGE_SIZE triggering the messages below. To eliminate the warning messages, you can add the option mtrr:2 to add a write-back mtrr for vesafb. Or just use nomtrr option. 1. Fix algorithm for finding the best power of 2 size with mtrr_add(). 2. Add option to choose the mtrr type by extending the mtrr boot option: mtrr:n where n 0 = no mtrr (equivalent to using the nomtrr option) 1 = uncachable 2 = write back 3 = write combining (default) 4 = write through Signed-off-by: Antonino Daplas <adaplas@pol.net> Signed-off-by: Andrew Morton <akpm@osdl.org> Signed-off-by: Linus Torvalds <torvalds@osdl.org>
This commit is contained in:
parent
655a0a7799
commit
8062594209
@ -45,7 +45,7 @@ static struct fb_fix_screeninfo vesafb_fix __initdata = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
static int inverse = 0;
|
static int inverse = 0;
|
||||||
static int mtrr = 1;
|
static int mtrr = 3; /* default to write-combining */
|
||||||
static int vram_remap __initdata = 0; /* Set amount of memory to be used */
|
static int vram_remap __initdata = 0; /* Set amount of memory to be used */
|
||||||
static int vram_total __initdata = 0; /* Set total amount of memory */
|
static int vram_total __initdata = 0; /* Set total amount of memory */
|
||||||
static int pmi_setpal = 0; /* pmi for palette changes ??? */
|
static int pmi_setpal = 0; /* pmi for palette changes ??? */
|
||||||
@ -204,8 +204,8 @@ static int __init vesafb_setup(char *options)
|
|||||||
pmi_setpal=0;
|
pmi_setpal=0;
|
||||||
else if (! strcmp(this_opt, "pmipal"))
|
else if (! strcmp(this_opt, "pmipal"))
|
||||||
pmi_setpal=1;
|
pmi_setpal=1;
|
||||||
else if (! strcmp(this_opt, "mtrr"))
|
else if (! strncmp(this_opt, "mtrr:", 5))
|
||||||
mtrr=1;
|
mtrr = simple_strtoul(this_opt+5, NULL, 0);
|
||||||
else if (! strcmp(this_opt, "nomtrr"))
|
else if (! strcmp(this_opt, "nomtrr"))
|
||||||
mtrr=0;
|
mtrr=0;
|
||||||
else if (! strncmp(this_opt, "vtotal:", 7))
|
else if (! strncmp(this_opt, "vtotal:", 7))
|
||||||
@ -387,14 +387,39 @@ static int __init vesafb_probe(struct device *device)
|
|||||||
|
|
||||||
if (mtrr) {
|
if (mtrr) {
|
||||||
unsigned int temp_size = size_total;
|
unsigned int temp_size = size_total;
|
||||||
/* Find the largest power-of-two */
|
unsigned int type = 0;
|
||||||
while (temp_size & (temp_size - 1))
|
|
||||||
temp_size &= (temp_size - 1);
|
|
||||||
|
|
||||||
/* Try and find a power of two to add */
|
switch (mtrr) {
|
||||||
while (temp_size > PAGE_SIZE &&
|
case 1:
|
||||||
mtrr_add(vesafb_fix.smem_start, temp_size, MTRR_TYPE_WRCOMB, 1)==-EINVAL) {
|
type = MTRR_TYPE_UNCACHABLE;
|
||||||
temp_size >>= 1;
|
break;
|
||||||
|
case 2:
|
||||||
|
type = MTRR_TYPE_WRBACK;
|
||||||
|
break;
|
||||||
|
case 3:
|
||||||
|
type = MTRR_TYPE_WRCOMB;
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
type = MTRR_TYPE_WRTHROUGH;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
type = 0;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (type) {
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
/* Find the largest power-of-two */
|
||||||
|
while (temp_size & (temp_size - 1))
|
||||||
|
temp_size &= (temp_size - 1);
|
||||||
|
|
||||||
|
/* Try and find a power of two to add */
|
||||||
|
do {
|
||||||
|
rc = mtrr_add(vesafb_fix.smem_start, temp_size,
|
||||||
|
type, 1);
|
||||||
|
temp_size >>= 1;
|
||||||
|
} while (temp_size >= PAGE_SIZE && rc == -EINVAL);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user