Commit 0137ecfd authored by Benjamin Herrenschmidt's avatar Benjamin Herrenschmidt Committed by Linus Torvalds

[PATCH] nvidiafb: Fixes for new G5

Recent X "nv" driver was fixed for various issues with modern 6xxx and 7xxx
cards.  This patch ports those fixes to nvidiafb.  This makes it work fine
on the 6600 bundled with the newest G5 macs.  I've verified it still works
on the 5200FX of the iMacG5.
Signed-off-by: default avatarBenjamin Herrenschmidt <benh@kernel.crashing.org>
Acked-by: default avatar"Antonino A. Daplas" <adaplas@pol.net>
Signed-off-by: default avatarAndrew Morton <akpm@osdl.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@osdl.org>
parent 2308acca
...@@ -848,7 +848,7 @@ void NVCalcStateExt(struct nvidia_par *par, ...@@ -848,7 +848,7 @@ void NVCalcStateExt(struct nvidia_par *par,
int width, int width,
int hDisplaySize, int height, int dotClock, int flags) int hDisplaySize, int height, int dotClock, int flags)
{ {
int pixelDepth, VClk; int pixelDepth, VClk = 0;
/* /*
* Save mode parameters. * Save mode parameters.
*/ */
...@@ -938,15 +938,24 @@ void NVLoadStateExt(struct nvidia_par *par, RIVA_HW_STATE * state) ...@@ -938,15 +938,24 @@ void NVLoadStateExt(struct nvidia_par *par, RIVA_HW_STATE * state)
if (par->Architecture == NV_ARCH_04) { if (par->Architecture == NV_ARCH_04) {
NV_WR32(par->PFB, 0x0200, state->config); NV_WR32(par->PFB, 0x0200, state->config);
} else if ((par->Chipset & 0xfff0) == 0x0090) { } else if ((par->Architecture < NV_ARCH_40) ||
for (i = 0; i < 15; i++) { (par->Chipset & 0xfff0) == 0x0040) {
NV_WR32(par->PFB, 0x0600 + (i * 0x10), 0);
NV_WR32(par->PFB, 0x0604 + (i * 0x10), par->FbMapSize - 1);
}
} else {
for (i = 0; i < 8; i++) { for (i = 0; i < 8; i++) {
NV_WR32(par->PFB, 0x0240 + (i * 0x10), 0); NV_WR32(par->PFB, 0x0240 + (i * 0x10), 0);
NV_WR32(par->PFB, 0x0244 + (i * 0x10), par->FbMapSize - 1); NV_WR32(par->PFB, 0x0244 + (i * 0x10),
par->FbMapSize - 1);
}
} else {
int regions = 12;
if (((par->Chipset & 0xfff0) == 0x0090) ||
((par->Chipset & 0xfff0) == 0x01D0) ||
((par->Chipset & 0xfff0) == 0x0290))
regions = 15;
for(i = 0; i < regions; i++) {
NV_WR32(par->PFB, 0x0600 + (i * 0x10), 0);
NV_WR32(par->PFB, 0x0604 + (i * 0x10),
par->FbMapSize - 1);
} }
} }
...@@ -1182,11 +1191,17 @@ void NVLoadStateExt(struct nvidia_par *par, RIVA_HW_STATE * state) ...@@ -1182,11 +1191,17 @@ void NVLoadStateExt(struct nvidia_par *par, RIVA_HW_STATE * state)
NV_WR32(par->PGRAPH, 0x0608, 0xFFFFFFFF); NV_WR32(par->PGRAPH, 0x0608, 0xFFFFFFFF);
} else { } else {
if (par->Architecture >= NV_ARCH_40) { if (par->Architecture >= NV_ARCH_40) {
u32 tmp;
NV_WR32(par->PGRAPH, 0x0084, 0x401287c0); NV_WR32(par->PGRAPH, 0x0084, 0x401287c0);
NV_WR32(par->PGRAPH, 0x008C, 0x60de8051); NV_WR32(par->PGRAPH, 0x008C, 0x60de8051);
NV_WR32(par->PGRAPH, 0x0090, 0x00008000); NV_WR32(par->PGRAPH, 0x0090, 0x00008000);
NV_WR32(par->PGRAPH, 0x0610, 0x00be3c5f); NV_WR32(par->PGRAPH, 0x0610, 0x00be3c5f);
tmp = NV_RD32(par->REGS, 0x1540) & 0xff;
for(i = 0; tmp && !(tmp & 1); tmp >>= 1, i++);
NV_WR32(par->PGRAPH, 0x5000, i);
if ((par->Chipset & 0xfff0) == 0x0040) { if ((par->Chipset & 0xfff0) == 0x0040) {
NV_WR32(par->PGRAPH, 0x09b0, NV_WR32(par->PGRAPH, 0x09b0,
0x83280fff); 0x83280fff);
...@@ -1211,6 +1226,7 @@ void NVLoadStateExt(struct nvidia_par *par, RIVA_HW_STATE * state) ...@@ -1211,6 +1226,7 @@ void NVLoadStateExt(struct nvidia_par *par, RIVA_HW_STATE * state)
0xffff7fff); 0xffff7fff);
break; break;
case 0x00C0: case 0x00C0:
case 0x0120:
NV_WR32(par->PGRAPH, 0x0828, NV_WR32(par->PGRAPH, 0x0828,
0x007596ff); 0x007596ff);
NV_WR32(par->PGRAPH, 0x082C, NV_WR32(par->PGRAPH, 0x082C,
...@@ -1245,6 +1261,7 @@ void NVLoadStateExt(struct nvidia_par *par, RIVA_HW_STATE * state) ...@@ -1245,6 +1261,7 @@ void NVLoadStateExt(struct nvidia_par *par, RIVA_HW_STATE * state)
0x00100000); 0x00100000);
break; break;
case 0x0090: case 0x0090:
case 0x0290:
NV_WR32(par->PRAMDAC, 0x0608, NV_WR32(par->PRAMDAC, 0x0608,
NV_RD32(par->PRAMDAC, 0x0608) | NV_RD32(par->PRAMDAC, 0x0608) |
0x00100000); 0x00100000);
...@@ -1310,14 +1327,44 @@ void NVLoadStateExt(struct nvidia_par *par, RIVA_HW_STATE * state) ...@@ -1310,14 +1327,44 @@ void NVLoadStateExt(struct nvidia_par *par, RIVA_HW_STATE * state)
} }
} }
if ((par->Chipset & 0xfff0) == 0x0090) { if ((par->Architecture < NV_ARCH_40) ||
for (i = 0; i < 60; i++) ((par->Chipset & 0xfff0) == 0x0040)) {
NV_WR32(par->PGRAPH, 0x0D00 + i, for (i = 0; i < 32; i++) {
NV_RD32(par->PFB, 0x0600 + i)); NV_WR32(par->PGRAPH, 0x0900 + i*4,
NV_RD32(par->PFB, 0x0240 +i*4));
NV_WR32(par->PGRAPH, 0x6900 + i*4,
NV_RD32(par->PFB, 0x0240 +i*4));
}
} else { } else {
for (i = 0; i < 32; i++) if (((par->Chipset & 0xfff0) == 0x0090) ||
NV_WR32(par->PGRAPH, 0x0900 + i, ((par->Chipset & 0xfff0) == 0x01D0) ||
NV_RD32(par->PFB, 0x0240 + i)); ((par->Chipset & 0xfff0) == 0x0290)) {
for (i = 0; i < 60; i++) {
NV_WR32(par->PGRAPH,
0x0D00 + i*4,
NV_RD32(par->PFB,
0x0600 + i*4));
NV_WR32(par->PGRAPH,
0x6900 + i*4,
NV_RD32(par->PFB,
0x0600 + i*4));
}
} else {
for (i = 0; i < 48; i++) {
NV_WR32(par->PGRAPH,
0x0900 + i*4,
NV_RD32(par->PFB,
0x0600 + i*4));
if(((par->Chipset & 0xfff0)
!= 0x0160) &&
((par->Chipset & 0xfff0)
!= 0x0220))
NV_WR32(par->PGRAPH,
0x6900 + i*4,
NV_RD32(par->PFB,
0x0600 + i*4));
}
}
} }
if (par->Architecture >= NV_ARCH_40) { if (par->Architecture >= NV_ARCH_40) {
...@@ -1338,7 +1385,9 @@ void NVLoadStateExt(struct nvidia_par *par, RIVA_HW_STATE * state) ...@@ -1338,7 +1385,9 @@ void NVLoadStateExt(struct nvidia_par *par, RIVA_HW_STATE * state)
NV_WR32(par->PGRAPH, 0x0868, NV_WR32(par->PGRAPH, 0x0868,
par->FbMapSize - 1); par->FbMapSize - 1);
} else { } else {
if((par->Chipset & 0xfff0) == 0x0090) { if ((par->Chipset & 0xfff0) == 0x0090 ||
(par->Chipset & 0xfff0) == 0x01D0 ||
(par->Chipset & 0xfff0) == 0x0290) {
NV_WR32(par->PGRAPH, 0x0DF0, NV_WR32(par->PGRAPH, 0x0DF0,
NV_RD32(par->PFB, 0x0200)); NV_RD32(par->PFB, 0x0200));
NV_WR32(par->PGRAPH, 0x0DF4, NV_WR32(par->PGRAPH, 0x0DF4,
......
...@@ -285,7 +285,6 @@ static void nv10GetConfig(struct nvidia_par *par) ...@@ -285,7 +285,6 @@ static void nv10GetConfig(struct nvidia_par *par)
par->CrystalFreqKHz = 27000; par->CrystalFreqKHz = 27000;
} }
par->CursorStart = (par->RamAmountKBytes - 96) * 1024;
par->CURSOR = NULL; /* can't set this here */ par->CURSOR = NULL; /* can't set this here */
par->MinVClockFreqKHz = 12000; par->MinVClockFreqKHz = 12000;
par->MaxVClockFreqKHz = par->twoStagePLL ? 400000 : 350000; par->MaxVClockFreqKHz = par->twoStagePLL ? 400000 : 350000;
...@@ -382,6 +381,8 @@ void NVCommonSetup(struct fb_info *info) ...@@ -382,6 +381,8 @@ void NVCommonSetup(struct fb_info *info)
case 0x0146: case 0x0146:
case 0x0147: case 0x0147:
case 0x0148: case 0x0148:
case 0x0098:
case 0x0099:
mobile = 1; mobile = 1;
break; break;
default: default:
......
...@@ -1485,6 +1485,8 @@ static u32 __devinit nvidia_get_arch(struct pci_dev *pd) ...@@ -1485,6 +1485,8 @@ static u32 __devinit nvidia_get_arch(struct pci_dev *pd)
case 0x0210: case 0x0210:
case 0x0220: case 0x0220:
case 0x0230: case 0x0230:
case 0x0290:
case 0x0390:
arch = NV_ARCH_40; arch = NV_ARCH_40;
break; break;
case 0x0020: /* TNT, TNT2 */ case 0x0020: /* TNT, TNT2 */
...@@ -1581,10 +1583,15 @@ static int __devinit nvidiafb_probe(struct pci_dev *pd, ...@@ -1581,10 +1583,15 @@ static int __devinit nvidiafb_probe(struct pci_dev *pd,
if (par->FbMapSize > 64 * 1024 * 1024) if (par->FbMapSize > 64 * 1024 * 1024)
par->FbMapSize = 64 * 1024 * 1024; par->FbMapSize = 64 * 1024 * 1024;
par->FbUsableSize = par->FbMapSize - (128 * 1024); if(par->Architecture >= NV_ARCH_40)
par->FbUsableSize = par->FbMapSize - (560 * 1024);
else
par->FbUsableSize = par->FbMapSize - (128 * 1024);
par->ScratchBufferSize = (par->Architecture < NV_ARCH_10) ? 8 * 1024 : par->ScratchBufferSize = (par->Architecture < NV_ARCH_10) ? 8 * 1024 :
16 * 1024; 16 * 1024;
par->ScratchBufferStart = par->FbUsableSize - par->ScratchBufferSize; par->ScratchBufferStart = par->FbUsableSize - par->ScratchBufferSize;
par->CursorStart = par->FbUsableSize + (32 * 1024);
info->screen_base = ioremap(nvidiafb_fix.smem_start, par->FbMapSize); info->screen_base = ioremap(nvidiafb_fix.smem_start, par->FbMapSize);
info->screen_size = par->FbUsableSize; info->screen_size = par->FbUsableSize;
nvidiafb_fix.smem_len = par->RamAmountKBytes * 1024; nvidiafb_fix.smem_len = par->RamAmountKBytes * 1024;
......
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