2113 lines
56 KiB
C
2113 lines
56 KiB
C
/* COPYRIGHT AND PERMISSION NOTICE
|
|
|
|
Copyright (c) 2000, 2001 Nokia Home Communications
|
|
|
|
All rights reserved.
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining
|
|
a copy of this software and associated documentation files (the
|
|
"Software"), to deal in the Software without restriction, including
|
|
without limitation the rights to use, copy, modify, merge, publish,
|
|
distribute, and/or sell copies of the Software, and to permit persons
|
|
to whom the Software is furnished to do so, provided that the above
|
|
copyright notice(s) and this permission notice appear in all copies of
|
|
the Software and that both the above copyright notice(s) and this
|
|
permission notice appear in supporting documentation.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT
|
|
OF THIRD PARTY RIGHTS. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
|
|
HOLDERS INCLUDED IN THIS NOTICE BE LIABLE FOR ANY CLAIM, OR ANY
|
|
SPECIAL INDIRECT OR CONSEQUENTIAL DAMAGES, OR ANY DAMAGES WHATSOEVER
|
|
RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF
|
|
CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
|
|
CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
Except as contained in this notice, the name of a copyright holder
|
|
shall not be used in advertising or otherwise to promote the sale, use
|
|
or other dealings in this Software without prior written authorization
|
|
of the copyright holder.
|
|
|
|
X Window System is a trademark of The Open Group */
|
|
|
|
/*
|
|
* i810.c - KDrive driver for the i810 chipset
|
|
*
|
|
* Authors:
|
|
* Pontus Lidman <pontus.lidman@nokia.com>
|
|
*
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include <kdrive-config.h>
|
|
#endif
|
|
#include "kdrive.h"
|
|
#include "kxv.h"
|
|
#include "klinux.h"
|
|
|
|
#include "i810.h"
|
|
#include "agp.h"
|
|
|
|
#include "i810draw.h"
|
|
|
|
#ifndef I810_DEBUG
|
|
int I810_DEBUG = (0
|
|
/* | DEBUG_ALWAYS_SYNC */
|
|
/* | DEBUG_VERBOSE_ACCEL */
|
|
/* | DEBUG_VERBOSE_SYNC */
|
|
/* | DEBUG_VERBOSE_VGA */
|
|
/* | DEBUG_VERBOSE_RING */
|
|
/* | DEBUG_VERBOSE_OUTREG */
|
|
/* | DEBUG_VERBOSE_MEMORY */
|
|
/* | DEBUG_VERBOSE_CURSOR */
|
|
);
|
|
#endif
|
|
|
|
|
|
static Bool
|
|
i810ModeInit(KdScreenInfo *screen, const KdMonitorTiming *t);
|
|
|
|
static void
|
|
i810PrintMode( vgaRegPtr vgaReg, I810RegPtr mode );
|
|
|
|
Bool
|
|
i810CardInit (KdCardInfo *card)
|
|
{
|
|
int i;
|
|
|
|
I810CardInfo *i810c;
|
|
|
|
/* fprintf(stderr,"i810CardInit\n"); */
|
|
|
|
i810c = (I810CardInfo *) xalloc (sizeof (I810CardInfo));
|
|
|
|
if (!i810c)
|
|
return FALSE;
|
|
|
|
/* 2MB Video RAM */
|
|
i810c->videoRam=2048;
|
|
|
|
/* Find FB address */
|
|
|
|
if (card->attr.address[1] != 0) {
|
|
i810c->LinearAddr = card->attr.address[0] & 0xFF000000;
|
|
|
|
if (!i810c->LinearAddr) {
|
|
fprintf(stderr,"No valid FB address in PCI config space(1)\n");
|
|
xfree(i810c);
|
|
return FALSE;
|
|
} else {
|
|
/* fprintf(stderr,"Linear framebuffer at %lx\n",i810c->LinearAddr); */
|
|
}
|
|
} else {
|
|
fprintf(stderr,"No valid FB address in PCI config space(2)\n");
|
|
xfree(i810c);
|
|
return FALSE;
|
|
}
|
|
|
|
if (card->attr.address[1]) {
|
|
|
|
i810c->MMIOAddr = card->attr.address[1] & 0xFFF80000;
|
|
|
|
i810c->MMIOBase =
|
|
KdMapDevice (i810c->MMIOAddr, I810_REG_SIZE);
|
|
if (!i810c->MMIOBase) {
|
|
fprintf(stderr,"No valid MMIO address in PCI config space(1)\n");
|
|
xfree(i810c);
|
|
return FALSE;
|
|
} else {
|
|
|
|
}
|
|
} else {
|
|
fprintf(stderr,"No valid MMIO address in PCI config space(2)\n");
|
|
xfree(i810c);
|
|
return FALSE;
|
|
}
|
|
|
|
/* fprintf(stderr,"Mapped 0x%x bytes of MMIO regs at phys 0x%lx virt %p\n", */
|
|
/* I810_REG_SIZE,i810c->MMIOAddr,i810c->MMIOBase); */
|
|
|
|
/* Find out memory bus frequency.
|
|
*/
|
|
|
|
{
|
|
unsigned long *p;
|
|
|
|
if (!(p= (unsigned long *) LinuxGetPciCfg(&card->attr)))
|
|
return FALSE;
|
|
|
|
/* fprintf(stderr,"Frequency long %lx\n",p[WHTCFG_PAMR_DRP]); */
|
|
|
|
if ( (p[WHTCFG_PAMR_DRP] & LM_FREQ_MASK) == LM_FREQ_133 )
|
|
i810c->LmFreqSel = 133;
|
|
else
|
|
i810c->LmFreqSel = 100;
|
|
|
|
xfree(p);
|
|
|
|
/* fprintf(stderr,"Selected frequency %d\n",i810c->LmFreqSel); */
|
|
}
|
|
|
|
/* fprintf(stderr,"Will alloc AGP framebuffer: %d kByte\n",i810c->videoRam); */
|
|
|
|
/* Since we always want write combining on first 32 mb of framebuffer
|
|
* we pass a mapsize of 32 mb */
|
|
i810c->FbMapSize = 32*1024*1024;
|
|
|
|
for (i = 2 ; i < i810c->FbMapSize ; i <<= 1);
|
|
i810c->FbMapSize = i;
|
|
|
|
i810c->FbBase =
|
|
KdMapDevice (i810c->LinearAddr, i810c->FbMapSize);
|
|
|
|
if (!i810c->FbBase) return FALSE;
|
|
/* fprintf(stderr,"Mapped 0x%lx bytes of framebuffer at %p\n", */
|
|
/* i810c->FbMapSize,i810c->FbBase); */
|
|
|
|
card->driver=i810c;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
i810ScreenFini (KdScreenInfo *screen)
|
|
{
|
|
I810ScreenInfo *i810s = (I810ScreenInfo *) screen->driver;
|
|
|
|
xfree (i810s);
|
|
screen->driver = 0;
|
|
}
|
|
|
|
static Bool
|
|
i810InitScreen (ScreenPtr pScreen) {
|
|
|
|
#ifdef XV
|
|
i810InitVideo(pScreen);
|
|
#endif
|
|
return TRUE;
|
|
}
|
|
|
|
static Bool
|
|
i810FinishInitScreen(ScreenPtr pScreen)
|
|
{
|
|
/* XXX: RandR init */
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
i810CardFini (KdCardInfo *card)
|
|
{
|
|
I810CardInfo *i810c = (I810CardInfo *) card->driver;
|
|
|
|
KdUnmapDevice (i810c->FbBase, i810c->FbMapSize);
|
|
KdUnmapDevice (i810c->MMIOBase, I810_REG_SIZE);
|
|
xfree (i810c);
|
|
card->driver = 0;
|
|
}
|
|
|
|
struct wm_info {
|
|
double freq;
|
|
unsigned int wm;
|
|
};
|
|
|
|
struct wm_info i810_wm_8_100[] = {
|
|
{ 0, 0x22003000 },
|
|
{ 25.2, 0x22003000 },
|
|
{ 28.0, 0x22003000 },
|
|
{ 31.5, 0x22003000 },
|
|
{ 36.0, 0x22007000 },
|
|
{ 40.0, 0x22007000 },
|
|
{ 45.0, 0x22007000 },
|
|
{ 49.5, 0x22008000 },
|
|
{ 50.0, 0x22008000 },
|
|
{ 56.3, 0x22008000 },
|
|
{ 65.0, 0x22008000 },
|
|
{ 75.0, 0x22008000 },
|
|
{ 78.8, 0x22008000 },
|
|
{ 80.0, 0x22008000 },
|
|
{ 94.0, 0x22008000 },
|
|
{ 96.0, 0x22107000 },
|
|
{ 99.0, 0x22107000 },
|
|
{ 108.0, 0x22107000 },
|
|
{ 121.0, 0x22107000 },
|
|
{ 128.9, 0x22107000 },
|
|
{ 132.0, 0x22109000 },
|
|
{ 135.0, 0x22109000 },
|
|
{ 157.5, 0x2210b000 },
|
|
{ 162.0, 0x2210b000 },
|
|
{ 175.5, 0x2210b000 },
|
|
{ 189.0, 0x2220e000 },
|
|
{ 202.5, 0x2220e000 }
|
|
};
|
|
|
|
struct wm_info i810_wm_16_100[] = {
|
|
{ 0, 0x22004000 },
|
|
{ 25.2, 0x22006000 },
|
|
{ 28.0, 0x22006000 },
|
|
{ 31.5, 0x22007000 },
|
|
{ 36.0, 0x22007000 },
|
|
{ 40.0, 0x22007000 },
|
|
{ 45.0, 0x22007000 },
|
|
{ 49.5, 0x22009000 },
|
|
{ 50.0, 0x22009000 },
|
|
{ 56.3, 0x22108000 },
|
|
{ 65.0, 0x2210e000 },
|
|
{ 75.0, 0x2210e000 },
|
|
{ 78.8, 0x2210e000 },
|
|
{ 80.0, 0x22210000 },
|
|
{ 94.5, 0x22210000 },
|
|
{ 96.0, 0x22210000 },
|
|
{ 99.0, 0x22210000 },
|
|
{ 108.0, 0x22210000 },
|
|
{ 121.0, 0x22210000 },
|
|
{ 128.9, 0x22210000 },
|
|
{ 132.0, 0x22314000 },
|
|
{ 135.0, 0x22314000 },
|
|
{ 157.5, 0x22415000 },
|
|
{ 162.0, 0x22416000 },
|
|
{ 175.5, 0x22416000 },
|
|
{ 189.0, 0x22416000 },
|
|
{ 195.0, 0x22416000 },
|
|
{ 202.5, 0x22416000 }
|
|
};
|
|
|
|
|
|
struct wm_info i810_wm_24_100[] = {
|
|
{ 0, 0x22006000 },
|
|
{ 25.2, 0x22009000 },
|
|
{ 28.0, 0x22009000 },
|
|
{ 31.5, 0x2200a000 },
|
|
{ 36.0, 0x2210c000 },
|
|
{ 40.0, 0x2210c000 },
|
|
{ 45.0, 0x2210c000 },
|
|
{ 49.5, 0x22111000 },
|
|
{ 50.0, 0x22111000 },
|
|
{ 56.3, 0x22111000 },
|
|
{ 65.0, 0x22214000 },
|
|
{ 75.0, 0x22214000 },
|
|
{ 78.8, 0x22215000 },
|
|
{ 80.0, 0x22216000 },
|
|
{ 94.5, 0x22218000 },
|
|
{ 96.0, 0x22418000 },
|
|
{ 99.0, 0x22418000 },
|
|
{ 108.0, 0x22418000 },
|
|
{ 121.0, 0x22418000 },
|
|
{ 128.9, 0x22419000 },
|
|
{ 132.0, 0x22519000 },
|
|
{ 135.0, 0x4441d000 },
|
|
{ 157.5, 0x44419000 },
|
|
{ 162.0, 0x44419000 },
|
|
{ 175.5, 0x44419000 },
|
|
{ 189.0, 0x44419000 },
|
|
{ 195.0, 0x44419000 },
|
|
{ 202.5, 0x44419000 }
|
|
};
|
|
|
|
struct wm_info i810_wm_32_100[] = {
|
|
{ 0, 0x2210b000 },
|
|
{ 60, 0x22415000 }, /* 0x314000 works too */
|
|
{ 80, 0x22419000 } /* 0x518000 works too */
|
|
};
|
|
|
|
|
|
struct wm_info i810_wm_8_133[] = {
|
|
{ 0, 0x22003000 },
|
|
{ 25.2, 0x22003000 },
|
|
{ 28.0, 0x22003000 },
|
|
{ 31.5, 0x22003000 },
|
|
{ 36.0, 0x22007000 },
|
|
{ 40.0, 0x22007000 },
|
|
{ 45.0, 0x22007000 },
|
|
{ 49.5, 0x22008000 },
|
|
{ 50.0, 0x22008000 },
|
|
{ 56.3, 0x22008000 },
|
|
{ 65.0, 0x22008000 },
|
|
{ 75.0, 0x22008000 },
|
|
{ 78.8, 0x22008000 },
|
|
{ 80.0, 0x22008000 },
|
|
{ 94.0, 0x22008000 },
|
|
{ 96.0, 0x22107000 },
|
|
{ 99.0, 0x22107000 },
|
|
{ 108.0, 0x22107000 },
|
|
{ 121.0, 0x22107000 },
|
|
{ 128.9, 0x22107000 },
|
|
{ 132.0, 0x22109000 },
|
|
{ 135.0, 0x22109000 },
|
|
{ 157.5, 0x2210b000 },
|
|
{ 162.0, 0x2210b000 },
|
|
{ 175.5, 0x2210b000 },
|
|
{ 189.0, 0x2220e000 },
|
|
{ 202.5, 0x2220e000 }
|
|
};
|
|
|
|
|
|
struct wm_info i810_wm_16_133[] = {
|
|
{ 0, 0x22004000 },
|
|
{ 25.2, 0x22006000 },
|
|
{ 28.0, 0x22006000 },
|
|
{ 31.5, 0x22007000 },
|
|
{ 36.0, 0x22007000 },
|
|
{ 40.0, 0x22007000 },
|
|
{ 45.0, 0x22007000 },
|
|
{ 49.5, 0x22009000 },
|
|
{ 50.0, 0x22009000 },
|
|
{ 56.3, 0x22108000 },
|
|
{ 65.0, 0x2210e000 },
|
|
{ 75.0, 0x2210e000 },
|
|
{ 78.8, 0x2210e000 },
|
|
{ 80.0, 0x22210000 },
|
|
{ 94.5, 0x22210000 },
|
|
{ 96.0, 0x22210000 },
|
|
{ 99.0, 0x22210000 },
|
|
{ 108.0, 0x22210000 },
|
|
{ 121.0, 0x22210000 },
|
|
{ 128.9, 0x22210000 },
|
|
{ 132.0, 0x22314000 },
|
|
{ 135.0, 0x22314000 },
|
|
{ 157.5, 0x22415000 },
|
|
{ 162.0, 0x22416000 },
|
|
{ 175.5, 0x22416000 },
|
|
{ 189.0, 0x22416000 },
|
|
{ 195.0, 0x22416000 },
|
|
{ 202.5, 0x22416000 }
|
|
};
|
|
|
|
struct wm_info i810_wm_24_133[] = {
|
|
{ 0, 0x22006000 },
|
|
{ 25.2, 0x22009000 },
|
|
{ 28.0, 0x22009000 },
|
|
{ 31.5, 0x2200a000 },
|
|
{ 36.0, 0x2210c000 },
|
|
{ 40.0, 0x2210c000 },
|
|
{ 45.0, 0x2210c000 },
|
|
{ 49.5, 0x22111000 },
|
|
{ 50.0, 0x22111000 },
|
|
{ 56.3, 0x22111000 },
|
|
{ 65.0, 0x22214000 },
|
|
{ 75.0, 0x22214000 },
|
|
{ 78.8, 0x22215000 },
|
|
{ 80.0, 0x22216000 },
|
|
{ 94.5, 0x22218000 },
|
|
{ 96.0, 0x22418000 },
|
|
{ 99.0, 0x22418000 },
|
|
{ 108.0, 0x22418000 },
|
|
{ 121.0, 0x22418000 },
|
|
{ 128.9, 0x22419000 },
|
|
{ 132.0, 0x22519000 },
|
|
{ 135.0, 0x4441d000 },
|
|
{ 157.5, 0x44419000 },
|
|
{ 162.0, 0x44419000 },
|
|
{ 175.5, 0x44419000 },
|
|
{ 189.0, 0x44419000 },
|
|
{ 195.0, 0x44419000 },
|
|
{ 202.5, 0x44419000 }
|
|
};
|
|
|
|
static void
|
|
i810WriteControlMMIO(I810CardInfo *i810c, int addr, CARD8 index, CARD8 val) {
|
|
moutb(addr, index);
|
|
moutb(addr+1, val);
|
|
}
|
|
|
|
static CARD8
|
|
i810ReadControlMMIO(I810CardInfo *i810c, int addr, CARD8 index) {
|
|
moutb(addr, index);
|
|
return minb(addr+1);
|
|
}
|
|
|
|
static Bool
|
|
i810ModeSupported (KdScreenInfo *screen, const KdMonitorTiming *t)
|
|
{
|
|
/* This is just a guess. */
|
|
if (t->horizontal > 1600 || t->horizontal < 640) return FALSE;
|
|
if (t->vertical > 1200 || t->horizontal < 350) return FALSE;
|
|
return TRUE;
|
|
}
|
|
|
|
static Bool
|
|
i810ModeUsable (KdScreenInfo *screen)
|
|
{
|
|
KdCardInfo *card = screen->card;
|
|
I810CardInfo *i810c = (I810CardInfo *) card->driver;
|
|
int byte_width, pixel_width, screen_size;
|
|
|
|
/* fprintf(stderr,"i810ModeUsable\n"); */
|
|
|
|
if (screen->fb[0].depth >= 24)
|
|
{
|
|
screen->fb[0].depth = 24;
|
|
screen->fb[0].bitsPerPixel = 24;
|
|
screen->dumb = TRUE;
|
|
}
|
|
else if (screen->fb[0].depth >= 16)
|
|
{
|
|
screen->fb[0].depth = 16;
|
|
screen->fb[0].bitsPerPixel = 16;
|
|
}
|
|
else if (screen->fb[0].depth >= 15)
|
|
{
|
|
screen->fb[0].depth = 15;
|
|
screen->fb[0].bitsPerPixel = 16;
|
|
}
|
|
else
|
|
{
|
|
screen->fb[0].depth = 8;
|
|
screen->fb[0].bitsPerPixel = 8;
|
|
}
|
|
byte_width = screen->width * (screen->fb[0].bitsPerPixel >> 3);
|
|
pixel_width = screen->width;
|
|
|
|
screen->fb[0].pixelStride = pixel_width;
|
|
screen->fb[0].byteStride = byte_width;
|
|
|
|
screen_size = byte_width * screen->height;
|
|
|
|
return screen_size <= (i810c->videoRam * 1024);
|
|
}
|
|
|
|
static int i810AllocateGARTMemory( KdScreenInfo *screen )
|
|
{
|
|
KdCardInfo *card = screen->card;
|
|
I810CardInfo *i810c = (I810CardInfo *) card->driver;
|
|
unsigned long size = i810c->videoRam * 1024;
|
|
|
|
int key;
|
|
long tom = 0;
|
|
unsigned long physical;
|
|
|
|
if (!KdAgpGARTSupported())
|
|
return FALSE;
|
|
|
|
if (!KdAcquireGART(screen->mynum))
|
|
return FALSE;
|
|
|
|
/* This allows the 2d only Xserver to regen */
|
|
i810c->agpAcquired2d = TRUE;
|
|
|
|
/* Treat the gart like video memory - we assume we own all that is
|
|
* there, so ignore EBUSY errors. Don't try to remove it on
|
|
* failure, either, as other X server may be using it.
|
|
*/
|
|
|
|
if ((key = KdAllocateGARTMemory(screen->mynum, size, 0, NULL)) == -1)
|
|
return FALSE;
|
|
|
|
i810c->VramOffset = 0;
|
|
i810c->VramKey = key;
|
|
|
|
if (!KdBindGARTMemory(screen->mynum, key, 0))
|
|
return FALSE;
|
|
|
|
|
|
i810c->SysMem.Start = 0;
|
|
i810c->SysMem.Size = size;
|
|
i810c->SysMem.End = size;
|
|
i810c->SavedSysMem = i810c->SysMem;
|
|
|
|
tom = i810c->SysMem.End;
|
|
|
|
i810c->DcacheMem.Start = 0;
|
|
i810c->DcacheMem.End = 0;
|
|
i810c->DcacheMem.Size = 0;
|
|
i810c->CursorPhysical = 0;
|
|
|
|
/* Dcache - half the speed of normal ram, so not really useful for
|
|
* a 2d server. Don't bother reporting its presence. This is
|
|
* mapped in addition to the requested amount of system ram.
|
|
*/
|
|
size = 1024 * 4096;
|
|
|
|
/* Keep it 512K aligned for the sake of tiled regions.
|
|
*/
|
|
tom += 0x7ffff;
|
|
tom &= ~0x7ffff;
|
|
|
|
if ((key = KdAllocateGARTMemory(screen->mynum, size, AGP_DCACHE_MEMORY, NULL)) != -1) {
|
|
i810c->DcacheOffset= tom;
|
|
i810c->DcacheKey = key;
|
|
if (!KdBindGARTMemory(screen->mynum, key, tom)) {
|
|
fprintf(stderr,"Allocation of %ld bytes for DCACHE failed\n", size);
|
|
i810c->DcacheKey = -1;
|
|
} else {
|
|
i810c->DcacheMem.Start = tom;
|
|
i810c->DcacheMem.Size = size;
|
|
i810c->DcacheMem.End = i810c->DcacheMem.Start + i810c->DcacheMem.Size;
|
|
tom = i810c->DcacheMem.End;
|
|
}
|
|
} else {
|
|
fprintf(stderr,
|
|
"No physical memory available for %ld bytes of DCACHE\n",
|
|
size);
|
|
i810c->DcacheKey = -1;
|
|
}
|
|
|
|
/* Mouse cursor -- The i810 (crazy) needs a physical address in
|
|
* system memory from which to upload the cursor. We get this from
|
|
* the agpgart module using a special memory type.
|
|
*/
|
|
|
|
/* 4k for the cursor is excessive, I'm going to steal 3k for
|
|
* overlay registers later
|
|
*/
|
|
|
|
size = 4096;
|
|
|
|
if ((key = KdAllocateGARTMemory(screen->mynum, size, AGP_PHYS_MEMORY,
|
|
&physical)) == -1) {
|
|
fprintf(stderr,
|
|
"No physical memory available for HW cursor\n");
|
|
i810c->HwcursKey = -1;
|
|
} else {
|
|
i810c->HwcursOffset= tom;
|
|
i810c->HwcursKey = key;
|
|
if (!KdBindGARTMemory(screen->mynum, key, tom)) {
|
|
fprintf(stderr,
|
|
"Allocation of %ld bytes for HW cursor failed\n",
|
|
size);
|
|
i810c->HwcursKey = -1;
|
|
} else {
|
|
i810c->CursorPhysical = physical;
|
|
i810c->CursorStart = tom;
|
|
tom += size;
|
|
}
|
|
}
|
|
|
|
/* Overlay register buffer -- Just like the cursor, the i810 needs a
|
|
* physical address in system memory from which to upload the overlay
|
|
* registers.
|
|
*/
|
|
if (i810c->CursorStart != 0) {
|
|
i810c->OverlayPhysical = i810c->CursorPhysical + 1024;
|
|
i810c->OverlayStart = i810c->CursorStart + 1024;
|
|
}
|
|
|
|
|
|
i810c->GttBound = 1;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/* Allocate from a memrange, returns success */
|
|
|
|
static int i810AllocLow( I810MemRange *result, I810MemRange *pool, int size )
|
|
{
|
|
if (size > pool->Size) return FALSE;
|
|
|
|
pool->Size -= size;
|
|
result->Size = size;
|
|
result->Start = pool->Start;
|
|
result->End = pool->Start += size;
|
|
return TRUE;
|
|
}
|
|
|
|
static int i810AllocHigh( I810MemRange *result, I810MemRange *pool, int size )
|
|
{
|
|
if (size > pool->Size) return 0;
|
|
|
|
pool->Size -= size;
|
|
result->Size = size;
|
|
result->End = pool->End;
|
|
result->Start = pool->End -= size;
|
|
return 1;
|
|
}
|
|
|
|
static Bool
|
|
i810AllocateFront(KdScreenInfo *screen) {
|
|
|
|
KdCardInfo *card = screen->card;
|
|
I810CardInfo *i810c = (I810CardInfo *) card->driver;
|
|
|
|
int cache_lines = -1;
|
|
|
|
if(i810c->DoneFrontAlloc)
|
|
return TRUE;
|
|
|
|
memset(&(i810c->FbMemBox), 0, sizeof(BoxRec));
|
|
/* Alloc FrontBuffer/Ring/Accel memory */
|
|
i810c->FbMemBox.x1=0;
|
|
i810c->FbMemBox.x2=screen->width;
|
|
i810c->FbMemBox.y1=0;
|
|
i810c->FbMemBox.y2=screen->height;
|
|
|
|
/* This could be made a command line option */
|
|
cache_lines = 0;
|
|
|
|
if(cache_lines >= 0)
|
|
i810c->FbMemBox.y2 += cache_lines;
|
|
else {
|
|
/* make sure there is enough for two DVD sized YUV buffers */
|
|
i810c->FbMemBox.y2 += (screen->fb[0].depth == 24) ? 256 : 384;
|
|
if (screen->width <= 1024)
|
|
i810c->FbMemBox.y2 += (screen->fb[0].depth == 24) ? 256 : 384;
|
|
cache_lines = i810c->FbMemBox.y2 - screen->height;
|
|
}
|
|
|
|
if (I810_DEBUG)
|
|
ErrorF("Adding %i scanlines for pixmap caching\n", cache_lines);
|
|
|
|
/* Reserve room for the framebuffer and pixcache. Put at the top
|
|
* of memory so we can have nice alignment for the tiled regions at
|
|
* the start of memory.
|
|
*/
|
|
i810AllocLow( &(i810c->FrontBuffer),
|
|
&(i810c->SysMem),
|
|
((i810c->FbMemBox.x2 *
|
|
i810c->FbMemBox.y2 *
|
|
i810c->cpp) + 4095) & ~4095);
|
|
|
|
memset( &(i810c->LpRing), 0, sizeof( I810RingBuffer ) );
|
|
if(i810AllocLow( &(i810c->LpRing.mem), &(i810c->SysMem), 16*4096 )) {
|
|
if (I810_DEBUG & DEBUG_VERBOSE_MEMORY)
|
|
ErrorF( "ring buffer at local %lx\n",
|
|
i810c->LpRing.mem.Start);
|
|
|
|
i810c->LpRing.tail_mask = i810c->LpRing.mem.Size - 1;
|
|
i810c->LpRing.virtual_start = i810c->FbBase + i810c->LpRing.mem.Start;
|
|
i810c->LpRing.head = 0;
|
|
i810c->LpRing.tail = 0;
|
|
i810c->LpRing.space = 0;
|
|
}
|
|
|
|
if ( i810AllocLow( &i810c->Scratch, &(i810c->SysMem), 64*1024 ) ||
|
|
i810AllocLow( &i810c->Scratch, &(i810c->SysMem), 16*1024 ) ) {
|
|
if (I810_DEBUG & DEBUG_VERBOSE_MEMORY)
|
|
ErrorF("Allocated Scratch Memory\n");
|
|
}
|
|
|
|
#ifdef XV
|
|
/* 720x720 is just how much memory the mpeg player needs for overlays */
|
|
|
|
if ( i810AllocHigh( &i810c->XvMem, &(i810c->SysMem), 720*720*2 )) {
|
|
if (I810_DEBUG & DEBUG_VERBOSE_MEMORY)
|
|
ErrorF("Allocated overlay Memory\n");
|
|
}
|
|
#endif
|
|
|
|
i810c->DoneFrontAlloc = TRUE;
|
|
return TRUE;
|
|
}
|
|
|
|
static Bool
|
|
i810MapMem(KdScreenInfo *screen)
|
|
{
|
|
|
|
KdCardInfo *card = screen->card;
|
|
I810CardInfo *i810c = (I810CardInfo *) card->driver;
|
|
|
|
i810c->LpRing.virtual_start = i810c->FbBase + i810c->LpRing.mem.Start;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
Bool
|
|
i810ScreenInit (KdScreenInfo *screen)
|
|
{
|
|
KdCardInfo *card = screen->card;
|
|
I810CardInfo *i810c = (I810CardInfo *) card->driver;
|
|
I810ScreenInfo *i810s;
|
|
|
|
int i;
|
|
|
|
const KdMonitorTiming *t;
|
|
|
|
/* fprintf(stderr,"i810ScreenInit\n"); */
|
|
|
|
i810s = (I810ScreenInfo *) xalloc (sizeof (I810ScreenInfo));
|
|
if (!i810s)
|
|
return FALSE;
|
|
|
|
memset (i810s, '\0', sizeof (I810ScreenInfo));
|
|
|
|
i810s->i810c = i810c;
|
|
|
|
/* Default dimensions */
|
|
if (!screen->width || !screen->height)
|
|
{
|
|
screen->width = 720;
|
|
screen->height = 576;
|
|
screen->rate = 52;
|
|
#if 0
|
|
screen->width = 1024;
|
|
screen->height = 768;
|
|
screen->rate = 72;
|
|
#endif
|
|
}
|
|
|
|
if (!screen->fb[0].depth)
|
|
screen->fb[0].depth = 16;
|
|
|
|
t = KdFindMode (screen, i810ModeSupported);
|
|
|
|
screen->rate = t->rate;
|
|
screen->width = t->horizontal;
|
|
screen->height = t->vertical;
|
|
|
|
if (!KdTuneMode (screen, i810ModeUsable, i810ModeSupported))
|
|
{
|
|
xfree (i810c);
|
|
return FALSE;
|
|
}
|
|
|
|
/* fprintf(stderr,"Screen rate %d horiz %d vert %d\n",t->rate,t->horizontal,t->vertical); */
|
|
|
|
switch (screen->fb[0].depth) {
|
|
case 8:
|
|
screen->fb[0].visuals = ((1 << StaticGray) |
|
|
(1 << GrayScale) |
|
|
(1 << StaticColor) |
|
|
(1 << PseudoColor) |
|
|
(1 << TrueColor) |
|
|
(1 << DirectColor));
|
|
screen->fb[0].blueMask = 0x00;
|
|
screen->fb[0].greenMask = 0x00;
|
|
screen->fb[0].redMask = 0x00;
|
|
break;
|
|
case 15:
|
|
screen->fb[0].visuals = (1 << TrueColor);
|
|
screen->fb[0].blueMask = 0x001f;
|
|
screen->fb[0].greenMask = 0x03e0;
|
|
screen->fb[0].redMask = 0x7c00;
|
|
|
|
i810c->colorKey = 0x043f;
|
|
|
|
break;
|
|
case 16:
|
|
screen->fb[0].visuals = (1 << TrueColor);
|
|
screen->fb[0].blueMask = 0x001f;
|
|
screen->fb[0].greenMask = 0x07e0;
|
|
screen->fb[0].redMask = 0xf800;
|
|
|
|
i810c->colorKey = 0x083f;
|
|
|
|
break;
|
|
case 24:
|
|
screen->fb[0].visuals = (1 << TrueColor);
|
|
screen->fb[0].blueMask = 0x0000ff;
|
|
screen->fb[0].greenMask = 0x00ff00;
|
|
screen->fb[0].redMask = 0xff0000;
|
|
|
|
i810c->colorKey = 0x0101ff;
|
|
|
|
break;
|
|
default:
|
|
fprintf(stderr,"Unsupported depth %d\n",screen->fb[0].depth);
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
|
|
/* Set all colours to black */
|
|
for (i=0; i<768; i++) i810c->vga.ModeReg.DAC[i] = 0x00;
|
|
|
|
/* ... and the overscan */
|
|
if (screen->fb[0].depth >= 4)
|
|
i810c->vga.ModeReg.Attribute[OVERSCAN] = 0xFF;
|
|
|
|
/* Could be made a command-line option */
|
|
|
|
#ifdef I810CFG_SHOW_OVERSCAN
|
|
i810c->vga.ModeReg.DAC[765] = 0x3F;
|
|
i810c->vga.ModeReg.DAC[766] = 0x00;
|
|
i810c->vga.ModeReg.DAC[767] = 0x3F;
|
|
i810c->vga.ModeReg.Attribute[OVERSCAN] = 0xFF;
|
|
i810c->vga.ShowOverscan = TRUE;
|
|
#else
|
|
i810c->vga.ShowOverscan = FALSE;
|
|
#endif
|
|
|
|
i810c->vga.paletteEnabled = FALSE;
|
|
i810c->vga.cmapSaved = FALSE;
|
|
i810c->vga.MMIOBase = i810c->MMIOBase;
|
|
|
|
i810c->cpp = screen->fb[0].bitsPerPixel/8;
|
|
|
|
/* move to initscreen? */
|
|
|
|
switch (screen->fb[0].bitsPerPixel) {
|
|
case 8:
|
|
i810c->MaxClock = 203000;
|
|
break;
|
|
case 16:
|
|
i810c->MaxClock = 163000;
|
|
break;
|
|
case 24:
|
|
i810c->MaxClock = 136000;
|
|
break;
|
|
case 32: /* not supported */
|
|
i810c->MaxClock = 86000;
|
|
default:
|
|
fprintf(stderr,"Unsupported bpp %d\n",screen->fb[0].bitsPerPixel);
|
|
return FALSE;
|
|
}
|
|
|
|
if (!i810AllocateGARTMemory( screen )) {
|
|
return FALSE;
|
|
}
|
|
|
|
i810AllocateFront(screen);
|
|
|
|
/* Map LpRing memory */
|
|
if (!i810MapMem(screen)) return FALSE;
|
|
|
|
screen->fb[0].frameBuffer = i810c->FbBase;
|
|
|
|
screen->driver = i810s;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/*
|
|
* I810Save --
|
|
*
|
|
* This function saves the video state. It reads all of the SVGA registers
|
|
* into the vgaI810Rec data structure. There is in general no need to
|
|
* mask out bits here - just read the registers.
|
|
*/
|
|
static void
|
|
DoSave(KdCardInfo *card, vgaRegPtr vgaReg, I810RegPtr i810Reg, Bool saveFonts)
|
|
{
|
|
|
|
I810CardInfo *i810c = card->driver;
|
|
i810VGAPtr vgap = &i810c->vga;
|
|
|
|
int i;
|
|
|
|
/* Save VGA registers */
|
|
|
|
vgaReg->MiscOutReg = mmioReadMiscOut(vgap);
|
|
if (vgaReg->MiscOutReg & 0x01)
|
|
vgap->IOBase = VGA_IOBASE_COLOR;
|
|
else
|
|
vgap->IOBase = VGA_IOBASE_MONO;
|
|
|
|
for (i = 0; i < VGA_NUM_CRTC; i++) {
|
|
vgaReg->CRTC[i] = mmioReadCrtc(vgap, i);
|
|
}
|
|
|
|
mmioEnablePalette(vgap);
|
|
for (i = 0; i < VGA_NUM_ATTR; i++) {
|
|
vgaReg->Attribute[i] = mmioReadAttr(vgap, i);
|
|
}
|
|
mmioDisablePalette(vgap);
|
|
|
|
for (i = 0; i < VGA_NUM_GFX; i++) {
|
|
vgaReg->Graphics[i] = mmioReadGr(vgap, i);
|
|
}
|
|
|
|
for (i = 1; i < VGA_NUM_SEQ; i++) {
|
|
vgaReg->Sequencer[i] = mmioReadSeq(vgap, i);
|
|
}
|
|
|
|
/*
|
|
* The port I/O code necessary to read in the extended registers
|
|
* into the fields of the I810Rec structure goes here.
|
|
*/
|
|
i810Reg->IOControl = mmioReadCrtc(vgap, IO_CTNL);
|
|
i810Reg->AddressMapping = i810ReadControlMMIO(i810c, GRX, ADDRESS_MAPPING);
|
|
i810Reg->BitBLTControl = INREG8(BITBLT_CNTL);
|
|
i810Reg->VideoClk2_M = INREG16(VCLK2_VCO_M);
|
|
i810Reg->VideoClk2_N = INREG16(VCLK2_VCO_N);
|
|
i810Reg->VideoClk2_DivisorSel = INREG8(VCLK2_VCO_DIV_SEL);
|
|
|
|
i810Reg->ExtVertTotal=mmioReadCrtc(vgap, EXT_VERT_TOTAL);
|
|
i810Reg->ExtVertDispEnd=mmioReadCrtc(vgap, EXT_VERT_DISPLAY);
|
|
i810Reg->ExtVertSyncStart=mmioReadCrtc(vgap, EXT_VERT_SYNC_START);
|
|
i810Reg->ExtVertBlankStart=mmioReadCrtc(vgap, EXT_VERT_BLANK_START);
|
|
i810Reg->ExtHorizTotal=mmioReadCrtc(vgap, EXT_HORIZ_TOTAL);
|
|
i810Reg->ExtHorizBlank=mmioReadCrtc(vgap, EXT_HORIZ_BLANK);
|
|
i810Reg->ExtOffset=mmioReadCrtc(vgap, EXT_OFFSET);
|
|
i810Reg->InterlaceControl=mmioReadCrtc(vgap, INTERLACE_CNTL);
|
|
|
|
i810Reg->PixelPipeCfg0 = INREG8(PIXPIPE_CONFIG_0);
|
|
i810Reg->PixelPipeCfg1 = INREG8(PIXPIPE_CONFIG_1);
|
|
i810Reg->PixelPipeCfg2 = INREG8(PIXPIPE_CONFIG_2);
|
|
i810Reg->DisplayControl = INREG8(DISPLAY_CNTL);
|
|
i810Reg->LMI_FIFO_Watermark = INREG(FWATER_BLC);
|
|
|
|
for (i = 0 ; i < 8 ; i++)
|
|
i810Reg->Fence[i] = INREG(FENCE+i*4);
|
|
|
|
i810Reg->LprbTail = INREG(LP_RING + RING_TAIL);
|
|
i810Reg->LprbHead = INREG(LP_RING + RING_HEAD);
|
|
i810Reg->LprbStart = INREG(LP_RING + RING_START);
|
|
i810Reg->LprbLen = INREG(LP_RING + RING_LEN);
|
|
|
|
if ((i810Reg->LprbTail & TAIL_ADDR) != (i810Reg->LprbHead & HEAD_ADDR) &&
|
|
i810Reg->LprbLen & RING_VALID) {
|
|
i810PrintErrorState( i810c );
|
|
FatalError( "Active ring not flushed\n");
|
|
}
|
|
|
|
if (I810_DEBUG) {
|
|
fprintf(stderr,"Got mode in I810Save:\n");
|
|
i810PrintMode( vgaReg, i810Reg );
|
|
}
|
|
}
|
|
|
|
static void
|
|
i810Preserve(KdCardInfo *card)
|
|
{
|
|
I810CardInfo *i810c = card->driver;
|
|
i810VGAPtr vgap = &i810c->vga;
|
|
|
|
/* fprintf(stderr,"i810Preserve\n"); */
|
|
DoSave(card, &vgap->SavedReg, &i810c->SavedReg, TRUE);
|
|
}
|
|
|
|
/* Famous last words
|
|
*/
|
|
void
|
|
i810PrintErrorState(i810CardInfo *i810c)
|
|
{
|
|
|
|
fprintf(stderr, "pgetbl_ctl: 0x%lx pgetbl_err: 0x%lx\n",
|
|
INREG(PGETBL_CTL),
|
|
INREG(PGE_ERR));
|
|
|
|
fprintf(stderr, "ipeir: %lx iphdr: %lx\n",
|
|
INREG(IPEIR),
|
|
INREG(IPEHR));
|
|
|
|
fprintf(stderr, "LP ring tail: %lx head: %lx len: %lx start %lx\n",
|
|
INREG(LP_RING + RING_TAIL),
|
|
INREG(LP_RING + RING_HEAD) & HEAD_ADDR,
|
|
INREG(LP_RING + RING_LEN),
|
|
INREG(LP_RING + RING_START));
|
|
|
|
fprintf(stderr, "eir: %x esr: %x emr: %x\n",
|
|
INREG16(EIR),
|
|
INREG16(ESR),
|
|
INREG16(EMR));
|
|
|
|
fprintf(stderr, "instdone: %x instpm: %x\n",
|
|
INREG16(INST_DONE),
|
|
INREG8(INST_PM));
|
|
|
|
fprintf(stderr, "memmode: %lx instps: %lx\n",
|
|
INREG(MEMMODE),
|
|
INREG(INST_PS));
|
|
|
|
fprintf(stderr, "hwstam: %x ier: %x imr: %x iir: %x\n",
|
|
INREG16(HWSTAM),
|
|
INREG16(IER),
|
|
INREG16(IMR),
|
|
INREG16(IIR));
|
|
}
|
|
|
|
static Bool
|
|
i810BindGARTMemory( KdScreenInfo *screen )
|
|
{
|
|
|
|
KdCardInfo *card = screen->card;
|
|
I810CardInfo *i810c = card->driver;
|
|
|
|
if (!i810c->GttBound) {
|
|
if (!KdAcquireGART(screen->mynum))
|
|
return FALSE;
|
|
if (!KdBindGARTMemory(screen->mynum, i810c->VramKey,
|
|
i810c->VramOffset))
|
|
|
|
return FALSE;
|
|
if (i810c->DcacheKey != -1) {
|
|
if (!KdBindGARTMemory(screen->mynum, i810c->DcacheKey,
|
|
i810c->DcacheOffset))
|
|
return FALSE;
|
|
}
|
|
if (i810c->HwcursKey != -1) {
|
|
if (!KdBindGARTMemory(screen->mynum, i810c->HwcursKey,
|
|
i810c->HwcursOffset))
|
|
return FALSE;
|
|
}
|
|
i810c->GttBound = 1;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
static Bool
|
|
i810UnbindGARTMemory(KdScreenInfo *screen)
|
|
{
|
|
KdCardInfo *card = screen->card;
|
|
I810CardInfo *i810c = card->driver;
|
|
|
|
|
|
if (KdAgpGARTSupported() && i810c->GttBound) {
|
|
if (!KdUnbindGARTMemory(screen->mynum, i810c->VramKey))
|
|
return FALSE;
|
|
if (i810c->DcacheKey != -1) {
|
|
if (!KdUnbindGARTMemory(screen->mynum, i810c->DcacheKey))
|
|
return FALSE;
|
|
}
|
|
if (i810c->HwcursKey != -1) {
|
|
if (!KdUnbindGARTMemory(screen->mynum, i810c->HwcursKey))
|
|
return FALSE;
|
|
}
|
|
if (!KdReleaseGART(screen->mynum))
|
|
return FALSE;
|
|
i810c->GttBound = 0;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
/*
|
|
* I810CalcVCLK --
|
|
*
|
|
* Determine the closest clock frequency to the one requested.
|
|
*/
|
|
|
|
#define MAX_VCO_FREQ 600.0
|
|
#define TARGET_MAX_N 30
|
|
#define REF_FREQ 24.0
|
|
|
|
#define CALC_VCLK(m,n,p) \
|
|
(double)m / ((double)n * (1 << p)) * 4 * REF_FREQ
|
|
|
|
static void
|
|
i810CalcVCLK( KdScreenInfo *screen, double freq )
|
|
{
|
|
|
|
KdCardInfo *card = screen->card;
|
|
I810CardInfo *i810c = card->driver;
|
|
I810RegPtr i810Reg = &i810c->ModeReg;
|
|
|
|
int m, n, p;
|
|
double f_out, f_best;
|
|
double f_err;
|
|
double f_vco;
|
|
int m_best = 0, n_best = 0, p_best = 0;
|
|
double f_target = freq;
|
|
double err_max = 0.005;
|
|
double err_target = 0.001;
|
|
double err_best = 999999.0;
|
|
|
|
p_best = p = log(MAX_VCO_FREQ/f_target)/log((double)2);
|
|
f_vco = f_target * (1 << p);
|
|
|
|
n = 2;
|
|
do {
|
|
n++;
|
|
m = f_vco / (REF_FREQ / (double)n) / (double)4.0 + 0.5;
|
|
if (m < 3) m = 3;
|
|
f_out = CALC_VCLK(m,n,p);
|
|
f_err = 1.0 - (f_target/f_out);
|
|
if (fabs(f_err) < err_max) {
|
|
m_best = m;
|
|
n_best = n;
|
|
f_best = f_out;
|
|
err_best = f_err;
|
|
}
|
|
} while ((fabs(f_err) >= err_target) &&
|
|
((n <= TARGET_MAX_N) || (fabs(err_best) > err_max)));
|
|
|
|
if (fabs(f_err) < err_target) {
|
|
m_best = m;
|
|
n_best = n;
|
|
}
|
|
|
|
i810Reg->VideoClk2_M = (m_best-2) & 0x3FF;
|
|
i810Reg->VideoClk2_N = (n_best-2) & 0x3FF;
|
|
i810Reg->VideoClk2_DivisorSel = (p_best << 4);
|
|
|
|
/* fprintf(stderr, "Setting dot clock to %.1f MHz " */
|
|
/* "[ 0x%x 0x%x 0x%x ] " */
|
|
/* "[ %d %d %d ]\n", */
|
|
/* CALC_VCLK(m_best,n_best,p_best), */
|
|
/* i810Reg->VideoClk2_M, */
|
|
/* i810Reg->VideoClk2_N, */
|
|
/* i810Reg->VideoClk2_DivisorSel, */
|
|
/* m_best, n_best, p_best); */
|
|
}
|
|
|
|
/*
|
|
* I810CalcFIFO --
|
|
*
|
|
* Calculate burst length and FIFO watermark.
|
|
*/
|
|
|
|
#define Elements(x) (sizeof(x)/sizeof(*x))
|
|
|
|
static unsigned int
|
|
i810CalcWatermark( KdScreenInfo *screen, double freq, Bool dcache )
|
|
{
|
|
|
|
KdCardInfo *card = screen->card;
|
|
I810CardInfo *i810c = card->driver;
|
|
|
|
|
|
struct wm_info *tab;
|
|
int nr;
|
|
int i;
|
|
|
|
if (i810c->LmFreqSel == 100) {
|
|
switch(screen->fb[0].bitsPerPixel) {
|
|
case 8:
|
|
tab = i810_wm_8_100;
|
|
nr = Elements(i810_wm_8_100);
|
|
break;
|
|
case 16:
|
|
tab = i810_wm_16_100;
|
|
nr = Elements(i810_wm_16_100);
|
|
break;
|
|
case 24:
|
|
tab = i810_wm_24_100;
|
|
nr = Elements(i810_wm_24_100);
|
|
break;
|
|
default:
|
|
return 0;
|
|
}
|
|
} else {
|
|
switch(screen->fb[0].bitsPerPixel) {
|
|
case 8:
|
|
tab = i810_wm_8_133;
|
|
nr = Elements(i810_wm_8_133);
|
|
break;
|
|
case 16:
|
|
tab = i810_wm_16_133;
|
|
nr = Elements(i810_wm_16_133);
|
|
break;
|
|
case 24:
|
|
tab = i810_wm_24_133;
|
|
nr = Elements(i810_wm_24_133);
|
|
break;
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
for (i = 0 ; i < nr && tab[i].freq < freq ; i++);
|
|
|
|
if (i == nr)
|
|
i--;
|
|
|
|
/* fprintf(stderr,"chose watermark 0x%x: (tab.freq %.1f)\n", */
|
|
/* tab[i].wm, tab[i].freq); */
|
|
|
|
/* None of these values (sourced from intel) have watermarks for
|
|
* the dcache memory. Fake it for now by using the same watermark
|
|
* for both...
|
|
*
|
|
* Update: this is probably because dcache isn't real useful as
|
|
* framebuffer memory, so intel's drivers don't need watermarks
|
|
* for that memory because they never use it to feed the ramdacs.
|
|
* We do use it in the fallback mode, so keep the watermarks for
|
|
* now.
|
|
*/
|
|
if (dcache)
|
|
return (tab[i].wm & ~0xffffff) | ((tab[i].wm>>12) & 0xfff);
|
|
else
|
|
return tab[i].wm;
|
|
}
|
|
|
|
static void i810PrintMode( vgaRegPtr vgaReg, I810RegPtr mode )
|
|
{
|
|
int i;
|
|
|
|
fprintf(stderr," MiscOut: %x\n", vgaReg->MiscOutReg);
|
|
|
|
|
|
fprintf(stderr,"SEQ: ");
|
|
for (i = 0 ; i < VGA_NUM_SEQ ; i++) {
|
|
if ((i&7)==0) fprintf(stderr,"\n");
|
|
fprintf(stderr," %d: %x", i, vgaReg->Sequencer[i]);
|
|
}
|
|
fprintf(stderr,"\n");
|
|
|
|
fprintf(stderr,"CRTC: ");
|
|
for (i = 0 ; i < VGA_NUM_CRTC ; i++) {
|
|
if ((i&3)==0) fprintf(stderr,"\n");
|
|
fprintf(stderr," CR%02x: %2x", i, vgaReg->CRTC[i]);
|
|
}
|
|
fprintf(stderr,"\n");
|
|
|
|
fprintf(stderr,"GFX: ");
|
|
for (i = 0 ; i < VGA_NUM_GFX ; i++) {
|
|
if ((i&3)==0) fprintf(stderr,"\n");
|
|
fprintf(stderr," GR%02x: %02x", i, vgaReg->Graphics[i]);
|
|
}
|
|
fprintf(stderr,"\n");
|
|
|
|
fprintf(stderr,"ATTR: ");
|
|
for (i = 0 ; i < VGA_NUM_ATTR ; i++) {
|
|
if ((i&7)==0) fprintf(stderr,"\n");
|
|
fprintf(stderr," %d: %x", i, vgaReg->Attribute[i]);
|
|
}
|
|
fprintf(stderr,"\n");
|
|
|
|
|
|
fprintf(stderr," DisplayControl: %x\n", mode->DisplayControl);
|
|
fprintf(stderr," PixelPipeCfg0: %x\n", mode->PixelPipeCfg0);
|
|
fprintf(stderr," PixelPipeCfg1: %x\n", mode->PixelPipeCfg1);
|
|
fprintf(stderr," PixelPipeCfg2: %x\n", mode->PixelPipeCfg2);
|
|
fprintf(stderr," VideoClk2_M: %x\n", mode->VideoClk2_M);
|
|
fprintf(stderr," VideoClk2_N: %x\n", mode->VideoClk2_N);
|
|
fprintf(stderr," VideoClk2_DivisorSel: %x\n", mode->VideoClk2_DivisorSel);
|
|
fprintf(stderr," AddressMapping: %x\n", mode->AddressMapping);
|
|
fprintf(stderr," IOControl: %x\n", mode->IOControl);
|
|
fprintf(stderr," BitBLTControl: %x\n", mode->BitBLTControl);
|
|
fprintf(stderr," ExtVertTotal: %x\n", mode->ExtVertTotal);
|
|
fprintf(stderr," ExtVertDispEnd: %x\n", mode->ExtVertDispEnd);
|
|
fprintf(stderr," ExtVertSyncStart: %x\n", mode->ExtVertSyncStart);
|
|
fprintf(stderr," ExtVertBlankStart: %x\n", mode->ExtVertBlankStart);
|
|
fprintf(stderr," ExtHorizTotal: %x\n", mode->ExtHorizTotal);
|
|
fprintf(stderr," ExtHorizBlank: %x\n", mode->ExtHorizBlank);
|
|
fprintf(stderr," ExtOffset: %x\n", mode->ExtOffset);
|
|
fprintf(stderr," InterlaceControl: %x\n", mode->InterlaceControl);
|
|
fprintf(stderr," LMI_FIFO_Watermark: %x\n", mode->LMI_FIFO_Watermark);
|
|
fprintf(stderr," LprbTail: %x\n", mode->LprbTail);
|
|
fprintf(stderr," LprbHead: %x\n", mode->LprbHead);
|
|
fprintf(stderr," LprbStart: %x\n", mode->LprbStart);
|
|
fprintf(stderr," LprbLen: %x\n", mode->LprbLen);
|
|
fprintf(stderr," OverlayActiveStart: %x\n", mode->OverlayActiveStart);
|
|
fprintf(stderr," OverlayActiveEnd: %x\n", mode->OverlayActiveEnd);
|
|
}
|
|
|
|
|
|
/*
|
|
* i810VGASeqReset
|
|
* perform a sequencer reset.
|
|
*
|
|
* The i815 documentation states that these bits are not used by the
|
|
* HW, but still warns about not programming them...
|
|
*/
|
|
|
|
static void
|
|
i810VGASeqReset(i810VGAPtr vgap, Bool start)
|
|
{
|
|
if (start)
|
|
{
|
|
mmioWriteSeq(vgap, 0x00, 0x01); /* Synchronous Reset */
|
|
}
|
|
else
|
|
{
|
|
mmioWriteSeq(vgap, 0x00, 0x03); /* End Reset */
|
|
}
|
|
}
|
|
|
|
static void
|
|
i810VGAProtect(KdCardInfo *card, Bool on)
|
|
{
|
|
|
|
I810CardInfo *i810c = card->driver;
|
|
i810VGAPtr vgap = &i810c->vga;
|
|
|
|
unsigned char tmp;
|
|
|
|
if (on) {
|
|
/*
|
|
* Turn off screen and disable sequencer.
|
|
*/
|
|
tmp = mmioReadSeq(vgap, 0x01);
|
|
|
|
i810VGASeqReset(vgap, TRUE); /* start synchronous reset */
|
|
mmioWriteSeq(vgap, 0x01, tmp | 0x20); /* disable the display */
|
|
|
|
mmioEnablePalette(vgap);
|
|
} else {
|
|
/*
|
|
* Reenable sequencer, then turn on screen.
|
|
*/
|
|
|
|
tmp = mmioReadSeq(vgap, 0x01);
|
|
|
|
mmioWriteSeq(vgap, 0x01, tmp & ~0x20); /* reenable display */
|
|
i810VGASeqReset(vgap, FALSE); /* clear synchronousreset */
|
|
|
|
mmioDisablePalette(vgap);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* i810VGABlankScreen -- blank the screen.
|
|
*/
|
|
|
|
void
|
|
i810VGABlankScreen(KdCardInfo *card, Bool on)
|
|
{
|
|
I810CardInfo *i810c = card->driver;
|
|
i810VGAPtr vgap = &i810c->vga;
|
|
|
|
unsigned char scrn;
|
|
|
|
scrn = mmioReadSeq(vgap, 0x01);
|
|
|
|
if (on) {
|
|
scrn &= ~0x20; /* enable screen */
|
|
} else {
|
|
scrn |= 0x20; /* blank screen */
|
|
}
|
|
|
|
mmioWriteSeq(vgap,0x00,0x01);
|
|
mmioWriteSeq(vgap, 0x01, scrn); /* change mode */
|
|
mmioWriteSeq(vgap,0x00,0x03);
|
|
}
|
|
|
|
/* Restore hardware state */
|
|
|
|
static void
|
|
DoRestore(KdCardInfo *card, vgaRegPtr vgaReg, I810RegPtr i810Reg,
|
|
Bool restoreFonts) {
|
|
|
|
|
|
I810CardInfo *i810c = card->driver;
|
|
|
|
i810VGAPtr vgap = &i810c->vga;
|
|
|
|
unsigned char temp;
|
|
unsigned int itemp;
|
|
int i;
|
|
|
|
if (I810_DEBUG & DEBUG_VERBOSE_VGA) {
|
|
fprintf(stderr,"Setting mode in DoRestore:\n");
|
|
i810PrintMode( vgaReg, i810Reg );
|
|
}
|
|
|
|
/* Blank screen (i810vgaprotect) */
|
|
i810VGAProtect(card, TRUE);
|
|
|
|
/* Should wait for at least two hsync and no more than two vsync
|
|
before writing PIXCONF and turning the display on (?) */
|
|
usleep(50000);
|
|
|
|
/* Turn off DRAM Refresh */
|
|
temp = INREG8( DRAM_ROW_CNTL_HI );
|
|
temp &= ~DRAM_REFRESH_RATE;
|
|
temp |= DRAM_REFRESH_DISABLE;
|
|
OUTREG8( DRAM_ROW_CNTL_HI, temp );
|
|
|
|
usleep(1000); /* Wait 1 ms */
|
|
|
|
/* Write the M, N and P values */
|
|
OUTREG16( VCLK2_VCO_M, i810Reg->VideoClk2_M);
|
|
OUTREG16( VCLK2_VCO_N, i810Reg->VideoClk2_N);
|
|
OUTREG8( VCLK2_VCO_DIV_SEL, i810Reg->VideoClk2_DivisorSel);
|
|
|
|
/*
|
|
* Turn on 8 bit dac mode, if requested. This is needed to make
|
|
* sure that vgaHWRestore writes the values into the DAC properly.
|
|
* The problem occurs if 8 bit dac mode is requested and the HW is
|
|
* in 6 bit dac mode. If this happens, all the values are
|
|
* automatically shifted left twice by the HW and incorrect colors
|
|
* will be displayed on the screen. The only time this can happen
|
|
* is at server startup time and when switching back from a VT.
|
|
*/
|
|
temp = INREG8(PIXPIPE_CONFIG_0);
|
|
temp &= 0x7F; /* Save all but the 8 bit dac mode bit */
|
|
temp |= (i810Reg->PixelPipeCfg0 & DAC_8_BIT);
|
|
OUTREG8( PIXPIPE_CONFIG_0, temp );
|
|
|
|
/*
|
|
* Code to restore any SVGA registers that have been saved/modified
|
|
* goes here. Note that it is allowable, and often correct, to
|
|
* only modify certain bits in a register by a read/modify/write cycle.
|
|
*
|
|
* A special case - when using an external clock-setting program,
|
|
* this function must not change bits associated with the clock
|
|
* selection. This condition can be checked by the condition:
|
|
*
|
|
* if (i810Reg->std.NoClock >= 0)
|
|
* restore clock-select bits.
|
|
*/
|
|
|
|
/* VGA restore */
|
|
if (vgaReg->MiscOutReg & 0x01)
|
|
vgap->IOBase = VGA_IOBASE_COLOR;
|
|
else
|
|
vgap->IOBase = VGA_IOBASE_MONO;
|
|
|
|
mmioWriteMiscOut(vgap, vgaReg->MiscOutReg);
|
|
|
|
for (i = 1; i < VGA_NUM_SEQ; i++)
|
|
mmioWriteSeq(vgap, i, vgaReg->Sequencer[i]);
|
|
|
|
/* Ensure CRTC registers 0-7 are unlocked by clearing bit 7 or CRTC[17] */
|
|
/* = CR11 */
|
|
mmioWriteCrtc(vgap, 17, vgaReg->CRTC[17] & ~0x80);
|
|
|
|
for (i = 0; i < VGA_NUM_CRTC; i++) {
|
|
mmioWriteCrtc(vgap, i, vgaReg->CRTC[i]);
|
|
}
|
|
|
|
for (i = 0; i < VGA_NUM_GFX; i++)
|
|
mmioWriteGr(vgap, i, vgaReg->Graphics[i]);
|
|
|
|
mmioEnablePalette(vgap);
|
|
for (i = 0; i < VGA_NUM_ATTR; i++)
|
|
mmioWriteAttr(vgap, i, vgaReg->Attribute[i]);
|
|
mmioDisablePalette(vgap);
|
|
|
|
|
|
mmioWriteCrtc(vgap, EXT_VERT_TOTAL, i810Reg->ExtVertTotal);
|
|
mmioWriteCrtc(vgap, EXT_VERT_DISPLAY, i810Reg->ExtVertDispEnd);
|
|
mmioWriteCrtc(vgap, EXT_VERT_SYNC_START, i810Reg->ExtVertSyncStart);
|
|
mmioWriteCrtc(vgap, EXT_VERT_BLANK_START, i810Reg->ExtVertBlankStart);
|
|
mmioWriteCrtc(vgap, EXT_HORIZ_TOTAL, i810Reg->ExtHorizTotal);
|
|
mmioWriteCrtc(vgap, EXT_HORIZ_BLANK, i810Reg->ExtHorizBlank);
|
|
|
|
/* write CR40, CR42 first etc to get CR13 written as described in PRM */
|
|
|
|
mmioWriteCrtc(vgap, EXT_START_ADDR_HI, 0);
|
|
mmioWriteCrtc(vgap, EXT_START_ADDR, EXT_START_ADDR_ENABLE);
|
|
|
|
mmioWriteCrtc(vgap, EXT_OFFSET, i810Reg->ExtOffset);
|
|
mmioWriteCrtc(vgap, 0x13, vgaReg->CRTC[0x13]);
|
|
|
|
temp=mmioReadCrtc(vgap, INTERLACE_CNTL);
|
|
temp &= ~INTERLACE_ENABLE;
|
|
temp |= i810Reg->InterlaceControl;
|
|
mmioWriteCrtc(vgap, INTERLACE_CNTL, temp);
|
|
|
|
temp=i810ReadControlMMIO(i810c, GRX, ADDRESS_MAPPING);
|
|
temp &= 0xE0; /* Save reserved bits 7:5 */
|
|
temp |= i810Reg->AddressMapping;
|
|
i810WriteControlMMIO(i810c, GRX, ADDRESS_MAPPING, temp);
|
|
|
|
/* Setting the OVRACT Register for video overlay*/
|
|
OUTREG(0x6001C, (i810Reg->OverlayActiveEnd << 16) | i810Reg->OverlayActiveStart);
|
|
|
|
/* Turn on DRAM Refresh */
|
|
temp = INREG8( DRAM_ROW_CNTL_HI );
|
|
temp &= ~DRAM_REFRESH_RATE;
|
|
temp |= DRAM_REFRESH_60HZ;
|
|
OUTREG8( DRAM_ROW_CNTL_HI, temp );
|
|
|
|
temp = INREG8( BITBLT_CNTL );
|
|
temp &= ~COLEXP_MODE;
|
|
temp |= i810Reg->BitBLTControl;
|
|
OUTREG8( BITBLT_CNTL, temp );
|
|
|
|
temp = INREG8( DISPLAY_CNTL );
|
|
temp &= ~(VGA_WRAP_MODE | GUI_MODE);
|
|
temp |= i810Reg->DisplayControl;
|
|
OUTREG8( DISPLAY_CNTL, temp );
|
|
|
|
|
|
temp = INREG8( PIXPIPE_CONFIG_0 );
|
|
temp &= 0x64; /* Save reserved bits 6:5,2 */
|
|
temp |= i810Reg->PixelPipeCfg0;
|
|
OUTREG8( PIXPIPE_CONFIG_0, temp );
|
|
|
|
temp = INREG8( PIXPIPE_CONFIG_2 );
|
|
temp &= 0xF3; /* Save reserved bits 7:4,1:0 */
|
|
temp |= i810Reg->PixelPipeCfg2;
|
|
OUTREG8( PIXPIPE_CONFIG_2, temp );
|
|
|
|
temp = INREG8( PIXPIPE_CONFIG_1 );
|
|
temp &= ~DISPLAY_COLOR_MODE;
|
|
temp &= 0xEF; /* Restore the CRT control bit */
|
|
temp |= i810Reg->PixelPipeCfg1;
|
|
OUTREG8( PIXPIPE_CONFIG_1, temp );
|
|
|
|
OUTREG16(EIR, 0);
|
|
|
|
itemp = INREG(FWATER_BLC);
|
|
itemp &= ~(LM_BURST_LENGTH | LM_FIFO_WATERMARK |
|
|
MM_BURST_LENGTH | MM_FIFO_WATERMARK );
|
|
itemp |= i810Reg->LMI_FIFO_Watermark;
|
|
OUTREG(FWATER_BLC, itemp);
|
|
|
|
|
|
for (i = 0 ; i < 8 ; i++) {
|
|
OUTREG( FENCE+i*4, i810Reg->Fence[i] );
|
|
if (I810_DEBUG & DEBUG_VERBOSE_VGA)
|
|
fprintf(stderr,"Fence Register : %x\n", i810Reg->Fence[i]);
|
|
}
|
|
|
|
/* First disable the ring buffer (Need to wait for empty first?, if so
|
|
* should probably do it before entering this section)
|
|
*/
|
|
itemp = INREG(LP_RING + RING_LEN);
|
|
itemp &= ~RING_VALID_MASK;
|
|
OUTREG(LP_RING + RING_LEN, itemp );
|
|
|
|
/* Set up the low priority ring buffer.
|
|
*/
|
|
OUTREG(LP_RING + RING_TAIL, 0 );
|
|
OUTREG(LP_RING + RING_HEAD, 0 );
|
|
|
|
i810c->LpRing.head = 0;
|
|
i810c->LpRing.tail = 0;
|
|
|
|
itemp = INREG(LP_RING + RING_START);
|
|
itemp &= ~(START_ADDR);
|
|
itemp |= i810Reg->LprbStart;
|
|
OUTREG(LP_RING + RING_START, itemp );
|
|
|
|
itemp = INREG(LP_RING + RING_LEN);
|
|
itemp &= ~(RING_NR_PAGES | RING_REPORT_MASK | RING_VALID_MASK);
|
|
itemp |= i810Reg->LprbLen;
|
|
OUTREG(LP_RING + RING_LEN, itemp );
|
|
|
|
i810VGAProtect(card, FALSE);
|
|
|
|
temp=mmioReadCrtc(vgap, IO_CTNL);
|
|
temp &= ~(EXTENDED_ATTR_CNTL | EXTENDED_CRTC_CNTL);
|
|
temp |= i810Reg->IOControl;
|
|
mmioWriteCrtc(vgap, IO_CTNL, temp);
|
|
/* Protect CRTC[0-7] */
|
|
mmioWriteCrtc(vgap, 0x11, mmioReadCrtc(vgap, 0x11) | 0x80);
|
|
}
|
|
|
|
|
|
static Bool
|
|
i810SetMode(KdScreenInfo *screen, const KdMonitorTiming *t)
|
|
{
|
|
|
|
KdCardInfo *card = screen->card;
|
|
I810CardInfo *i810c = card->driver;
|
|
i810VGAPtr vgap = &i810c->vga;
|
|
|
|
I810RegPtr i810Reg = &i810c->ModeReg;
|
|
vgaRegPtr pVga = &vgap->ModeReg;
|
|
|
|
double dclk = t->clock/1000.0;
|
|
|
|
switch (screen->fb[0].bitsPerPixel) {
|
|
case 8:
|
|
pVga->CRTC[0x13] = screen->width >> 3;
|
|
i810Reg->ExtOffset = screen->width >> 11;
|
|
i810Reg->PixelPipeCfg1 = DISPLAY_8BPP_MODE;
|
|
i810Reg->BitBLTControl = COLEXP_8BPP;
|
|
break;
|
|
case 16:
|
|
i810Reg->PixelPipeCfg1 = DISPLAY_16BPP_MODE;
|
|
pVga->CRTC[0x13] = screen->width >> 2;
|
|
i810Reg->ExtOffset = screen->width >> 10;
|
|
i810Reg->BitBLTControl = COLEXP_16BPP;
|
|
break;
|
|
case 24:
|
|
pVga->CRTC[0x13] = (screen->width * 3) >> 3;
|
|
i810Reg->ExtOffset = (screen->width * 3) >> 11;
|
|
|
|
i810Reg->PixelPipeCfg1 = DISPLAY_24BPP_MODE;
|
|
i810Reg->BitBLTControl = COLEXP_24BPP;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
i810Reg->PixelPipeCfg0 = DAC_8_BIT;
|
|
|
|
/* Do not delay CRT Blank: needed for video overlay */
|
|
i810Reg->PixelPipeCfg1 |= 0x10;
|
|
|
|
/* Turn on Extended VGA Interpretation */
|
|
i810Reg->IOControl = EXTENDED_CRTC_CNTL;
|
|
|
|
/* Turn on linear and page mapping */
|
|
i810Reg->AddressMapping = (LINEAR_MODE_ENABLE |
|
|
GTT_MEM_MAP_ENABLE);
|
|
|
|
/* Turn on GUI mode */
|
|
i810Reg->DisplayControl = HIRES_MODE;
|
|
|
|
i810Reg->OverlayActiveStart = t->horizontal + t->hblank - 32;
|
|
i810Reg->OverlayActiveEnd = t->horizontal - 32;
|
|
|
|
/* Turn on interlaced mode if necessary (it's not) */
|
|
i810Reg->InterlaceControl = INTERLACE_DISABLE;
|
|
|
|
/*
|
|
* Set the overscan color to 0.
|
|
* NOTE: This only affects >8bpp mode.
|
|
*/
|
|
pVga->Attribute[0x11] = 0;
|
|
|
|
/*
|
|
* Calculate the VCLK that most closely matches the requested dot
|
|
* clock.
|
|
*/
|
|
i810CalcVCLK(screen, dclk);
|
|
|
|
/* Since we program the clocks ourselves, always use VCLK2. */
|
|
pVga->MiscOutReg |= 0x0C;
|
|
|
|
/* Calculate the FIFO Watermark and Burst Length. */
|
|
i810Reg->LMI_FIFO_Watermark = i810CalcWatermark(screen, dclk, FALSE);
|
|
|
|
/* Setup the ring buffer */
|
|
i810Reg->LprbTail = 0;
|
|
i810Reg->LprbHead = 0;
|
|
i810Reg->LprbStart = i810c->LpRing.mem.Start;
|
|
|
|
if (i810Reg->LprbStart)
|
|
i810Reg->LprbLen = ((i810c->LpRing.mem.Size-4096) |
|
|
RING_NO_REPORT | RING_VALID);
|
|
else
|
|
i810Reg->LprbLen = RING_INVALID;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static Bool
|
|
i810ModeInit(KdScreenInfo *screen, const KdMonitorTiming *t)
|
|
{
|
|
|
|
KdCardInfo *card = screen->card;
|
|
I810CardInfo *i810c = card->driver;
|
|
i810VGAPtr vgap = &i810c->vga;
|
|
vgaRegPtr pVga;
|
|
|
|
/* fprintf(stderr,"i810ModeInit\n"); */
|
|
|
|
i810VGAUnlock(vgap);
|
|
|
|
if (!i810VGAInit(screen, t)) return FALSE;
|
|
pVga = &vgap->ModeReg;
|
|
|
|
if (!i810SetMode(screen, t)) return FALSE;
|
|
|
|
DoRestore(screen->card, &vgap->ModeReg, &i810c->ModeReg, FALSE);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
Bool
|
|
i810VGAInit(KdScreenInfo *screen, const KdMonitorTiming *t)
|
|
{
|
|
unsigned int i;
|
|
|
|
int hactive, hblank, hbp, hfp;
|
|
int vactive, vblank, vbp, vfp;
|
|
int h_screen_off = 0, h_adjust = 0, h_total, h_display_end, h_blank_start;
|
|
int h_blank_end, h_sync_start, h_sync_end, v_total, v_retrace_start;
|
|
int v_retrace_end, v_display_end, v_blank_start, v_blank_end;
|
|
|
|
KdCardInfo *card = screen->card;
|
|
I810CardInfo *i810c = card->driver;
|
|
|
|
i810VGAPtr vgap = &i810c->vga;
|
|
I810RegPtr ireg = &i810c->ModeReg;
|
|
|
|
|
|
vgaRegPtr regp;
|
|
int depth = screen->fb[0].depth;
|
|
|
|
regp = &vgap->ModeReg;
|
|
|
|
/*
|
|
* compute correct Hsync & Vsync polarity
|
|
*/
|
|
|
|
regp->MiscOutReg = 0x23;
|
|
if (t->vpol == KdSyncNegative) regp->MiscOutReg |= 0x40;
|
|
if (t->hpol == KdSyncNegative) regp->MiscOutReg |= 0x80;
|
|
|
|
/*
|
|
* Time Sequencer
|
|
*/
|
|
if (depth == 4)
|
|
regp->Sequencer[0] = 0x02;
|
|
else
|
|
regp->Sequencer[0] = 0x00;
|
|
/* No support for 320 or 360 x resolution */
|
|
regp->Sequencer[1] = 0x01;
|
|
|
|
if (depth == 1)
|
|
regp->Sequencer[2] = 1 << BIT_PLANE;
|
|
else
|
|
regp->Sequencer[2] = 0x0F;
|
|
|
|
regp->Sequencer[3] = 0x00; /* Font select */
|
|
|
|
if (depth < 8)
|
|
regp->Sequencer[4] = 0x06; /* Misc */
|
|
else
|
|
regp->Sequencer[4] = 0x0E; /* Misc */
|
|
|
|
hactive = t->horizontal;
|
|
hblank = t->hblank;
|
|
hbp = t->hbp;
|
|
hfp = t->hfp;
|
|
|
|
vactive = t->vertical;
|
|
vblank = t->vblank;
|
|
vbp = t->vbp;
|
|
vfp = t->vfp;
|
|
|
|
switch (screen->fb[0].bitsPerPixel) {
|
|
case 8:
|
|
hactive /= 8;
|
|
hblank /= 8;
|
|
hfp /= 8;
|
|
hbp /= 8;
|
|
h_screen_off = hactive;
|
|
h_adjust = 1;
|
|
break;
|
|
case 16:
|
|
hactive /= 8;
|
|
hblank /= 8;
|
|
hfp /= 8;
|
|
hbp /= 8;
|
|
|
|
h_screen_off = hactive * 2;
|
|
h_adjust = 1;
|
|
break;
|
|
case 24:
|
|
hactive /= 8;
|
|
hblank /= 8;
|
|
hfp /= 8;
|
|
hbp /= 8;
|
|
|
|
h_screen_off = hactive * 3;
|
|
h_adjust = 1;
|
|
break;
|
|
case 32:
|
|
hactive /= 8;
|
|
hblank /= 8;
|
|
hfp /= 8;
|
|
hbp /= 8;
|
|
|
|
h_screen_off = hactive * 4;
|
|
h_adjust = 1;
|
|
break;
|
|
}
|
|
|
|
/*
|
|
* Compute horizontal register values from timings
|
|
*/
|
|
h_total = hactive + hblank - 5;
|
|
h_display_end = hactive - 1;
|
|
h_blank_start = h_display_end;
|
|
h_blank_end = h_blank_start + hblank;
|
|
|
|
h_sync_start = hactive + hfp + h_adjust;
|
|
h_sync_end = h_sync_start + hblank - hbp - hfp;
|
|
|
|
/* Set CRTC regs for horizontal timings */
|
|
regp->CRTC[0x0] = h_total;
|
|
ireg->ExtHorizTotal=(h_total & 0x100) >> 8;
|
|
|
|
regp->CRTC[0x1] = h_display_end;
|
|
|
|
regp->CRTC[0x2] = h_blank_start;
|
|
|
|
regp->CRTC[0x3] = 0x80 | (h_blank_end & 0x1f);
|
|
regp->CRTC[0x5] = (h_blank_end & 0x20) << 2;
|
|
|
|
regp->CRTC[0x4] = h_sync_start;
|
|
|
|
regp->CRTC[0x5] |= h_sync_end & 0x1f;
|
|
|
|
regp->CRTC[0x13] = h_screen_off;
|
|
ireg->ExtOffset = h_screen_off >> 8;
|
|
|
|
/* Compute vertical timings */
|
|
v_total = vactive + vblank - 2;
|
|
v_retrace_start = vactive + vfp - 1;
|
|
v_retrace_end = v_retrace_start + vblank - vbp - vfp;
|
|
v_display_end = vactive - 1;
|
|
v_blank_start = vactive - 1;
|
|
v_blank_end = v_blank_start + vblank /* - 1 */;
|
|
|
|
regp->CRTC[0x6] = v_total;
|
|
ireg->ExtVertTotal = v_total >> 8;
|
|
|
|
regp->CRTC[0x10] = v_retrace_start;
|
|
ireg->ExtVertSyncStart = v_retrace_start >> 8;
|
|
|
|
regp->CRTC[0x11] = v_retrace_end;
|
|
|
|
regp->CRTC[0x12] = v_display_end;
|
|
ireg->ExtVertDispEnd = v_display_end >> 8;
|
|
|
|
regp->CRTC[0x15] = v_blank_start;
|
|
ireg->ExtVertBlankStart = v_blank_start >> 8;
|
|
|
|
regp->CRTC[0x16] = v_blank_end;
|
|
|
|
if (depth < 8)
|
|
regp->CRTC[23] = 0xE3;
|
|
else
|
|
regp->CRTC[23] = 0xC3;
|
|
regp->CRTC[24] = 0xFF;
|
|
|
|
/*
|
|
* Graphics Display Controller
|
|
*/
|
|
regp->Graphics[0] = 0x00;
|
|
regp->Graphics[1] = 0x00;
|
|
regp->Graphics[2] = 0x00;
|
|
regp->Graphics[3] = 0x00;
|
|
if (depth == 1) {
|
|
regp->Graphics[4] = BIT_PLANE;
|
|
regp->Graphics[5] = 0x00;
|
|
} else {
|
|
regp->Graphics[4] = 0x00;
|
|
if (depth == 4)
|
|
regp->Graphics[5] = 0x02;
|
|
else
|
|
regp->Graphics[5] = 0x40;
|
|
}
|
|
regp->Graphics[6] = 0x05;
|
|
regp->Graphics[7] = 0x0F;
|
|
regp->Graphics[8] = 0xFF;
|
|
|
|
if (depth == 1) {
|
|
/* Initialise the Mono map according to which bit-plane gets used */
|
|
|
|
Bool flipPixels = FALSE; /* maybe support this in the future? */
|
|
|
|
for (i=0; i<16; i++)
|
|
if (((i & (1 << BIT_PLANE)) != 0) != flipPixels)
|
|
regp->Attribute[i] = WHITE_VALUE;
|
|
else
|
|
regp->Attribute[i] = BLACK_VALUE;
|
|
|
|
regp->Attribute[16] = 0x01; /* -VGA2- */
|
|
if (!vgap->ShowOverscan)
|
|
regp->Attribute[OVERSCAN] = OVERSCAN_VALUE; /* -VGA2- */
|
|
} else {
|
|
regp->Attribute[0] = 0x00; /* standard colormap translation */
|
|
regp->Attribute[1] = 0x01;
|
|
regp->Attribute[2] = 0x02;
|
|
regp->Attribute[3] = 0x03;
|
|
regp->Attribute[4] = 0x04;
|
|
regp->Attribute[5] = 0x05;
|
|
regp->Attribute[6] = 0x06;
|
|
regp->Attribute[7] = 0x07;
|
|
regp->Attribute[8] = 0x08;
|
|
regp->Attribute[9] = 0x09;
|
|
regp->Attribute[10] = 0x0A;
|
|
regp->Attribute[11] = 0x0B;
|
|
regp->Attribute[12] = 0x0C;
|
|
regp->Attribute[13] = 0x0D;
|
|
regp->Attribute[14] = 0x0E;
|
|
regp->Attribute[15] = 0x0F;
|
|
if (depth == 4)
|
|
regp->Attribute[16] = 0x81;
|
|
else
|
|
regp->Attribute[16] = 0x41;
|
|
/* Attribute[17] (overscan) was initialised earlier */
|
|
}
|
|
regp->Attribute[18] = 0x0F;
|
|
regp->Attribute[19] = 0x00;
|
|
regp->Attribute[20] = 0x00;
|
|
|
|
return(TRUE);
|
|
}
|
|
|
|
void
|
|
i810VGALock(i810VGAPtr vgap)
|
|
{
|
|
/* Protect CRTC[0-7] */
|
|
mmioWriteCrtc(vgap, 0x11, mmioReadCrtc(vgap, 0x11) & ~0x80);
|
|
}
|
|
|
|
void
|
|
i810VGAUnlock(i810VGAPtr vgap)
|
|
{
|
|
/* Unprotect CRTC[0-7] */
|
|
mmioWriteCrtc(vgap, 0x11, mmioReadCrtc(vgap, 0x11) | 0x80);
|
|
}
|
|
|
|
static void
|
|
i810Restore(KdCardInfo *card) {
|
|
|
|
I810CardInfo *i810c = card->driver;
|
|
|
|
i810VGAPtr vgap = &i810c->vga;
|
|
|
|
if (I810_DEBUG)
|
|
fprintf(stderr,"i810Restore\n");
|
|
|
|
DoRestore(card, &vgap->SavedReg, &i810c->SavedReg, TRUE);
|
|
}
|
|
|
|
static Bool
|
|
i810Enable (ScreenPtr pScreen)
|
|
{
|
|
KdScreenPriv(pScreen);
|
|
KdScreenInfo *screen = pScreenPriv->screen;
|
|
KdCardInfo *card = pScreenPriv->card;
|
|
I810CardInfo *i810c = card->driver;
|
|
i810VGAPtr vgap = &i810c->vga;
|
|
const KdMonitorTiming *t;
|
|
|
|
if (I810_DEBUG)
|
|
fprintf(stderr,"i810Enable\n");
|
|
|
|
vgap->IOBase = (mmioReadMiscOut(vgap) & 0x01) ?
|
|
VGA_IOBASE_COLOR : VGA_IOBASE_MONO;
|
|
|
|
{
|
|
I810RegPtr i810Reg = &i810c->ModeReg;
|
|
int i;
|
|
|
|
for (i = 0 ; i < 8 ; i++)
|
|
i810Reg->Fence[i] = 0;
|
|
}
|
|
|
|
t = KdFindMode (screen, i810ModeSupported);
|
|
|
|
if (!i810BindGARTMemory(screen))
|
|
return FALSE;
|
|
|
|
if (!i810ModeInit(screen, t)) return FALSE;
|
|
|
|
{
|
|
/* DPMS power on state */
|
|
|
|
unsigned char SEQ01=0;
|
|
int DPMSSyncSelect=0;
|
|
|
|
SEQ01 = 0x00;
|
|
DPMSSyncSelect = HSYNC_ON | VSYNC_ON;
|
|
|
|
SEQ01 |= i810ReadControlMMIO(i810c, SRX, 0x01) & ~0x20;
|
|
i810WriteControlMMIO(i810c, SRX, 0x01, SEQ01);
|
|
|
|
/* Set the DPMS mode */
|
|
OUTREG8(DPMS_SYNC_SELECT, DPMSSyncSelect);
|
|
}
|
|
#ifdef XV
|
|
KdXVEnable (pScreen);
|
|
#endif
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static void
|
|
i810Disable(ScreenPtr pScreen) {
|
|
|
|
KdScreenPriv(pScreen);
|
|
KdScreenInfo *screen = pScreenPriv->screen;
|
|
KdCardInfo *card = pScreenPriv->card;
|
|
I810CardInfo *i810c = card->driver;
|
|
|
|
i810VGAPtr vgap = &i810c->vga;
|
|
|
|
if (I810_DEBUG)
|
|
fprintf(stderr,"i810Disable\n");
|
|
|
|
#ifdef XV
|
|
KdXVDisable (pScreen);
|
|
#endif
|
|
i810Restore(screen->card);
|
|
|
|
if (!i810UnbindGARTMemory(screen))
|
|
return;
|
|
|
|
i810VGALock(vgap);
|
|
}
|
|
|
|
|
|
static Bool
|
|
i810DPMS(ScreenPtr pScreen, int mode)
|
|
{
|
|
KdScreenPriv(pScreen);
|
|
KdCardInfo *card = pScreenPriv->card;
|
|
I810CardInfo *i810c = card->driver;
|
|
|
|
unsigned char SEQ01=0;
|
|
int DPMSSyncSelect=0;
|
|
|
|
if (I810_DEBUG)
|
|
fprintf(stderr,"i810DPMS: %d\n",mode);
|
|
|
|
switch (mode) {
|
|
case KD_DPMS_NORMAL:
|
|
/* Screen: On; HSync: On, VSync: On */
|
|
SEQ01 = 0x00;
|
|
DPMSSyncSelect = HSYNC_ON | VSYNC_ON;
|
|
break;
|
|
case KD_DPMS_STANDBY:
|
|
/* Screen: Off; HSync: Off, VSync: On */
|
|
SEQ01 = 0x20;
|
|
DPMSSyncSelect = HSYNC_OFF | VSYNC_ON;
|
|
break;
|
|
case KD_DPMS_SUSPEND:
|
|
/* Screen: Off; HSync: On, VSync: Off */
|
|
SEQ01 = 0x20;
|
|
DPMSSyncSelect = HSYNC_ON | VSYNC_OFF;
|
|
break;
|
|
case KD_DPMS_POWERDOWN:
|
|
/* Screen: Off; HSync: Off, VSync: Off */
|
|
SEQ01 = 0x20;
|
|
DPMSSyncSelect = HSYNC_OFF | VSYNC_OFF;
|
|
break;
|
|
}
|
|
|
|
/* Turn the screen on/off */
|
|
SEQ01 |= i810ReadControlMMIO(i810c, SRX, 0x01) & ~0x20;
|
|
i810WriteControlMMIO(i810c, SRX, 0x01, SEQ01);
|
|
|
|
/* Set the DPMS mode */
|
|
OUTREG8(DPMS_SYNC_SELECT, DPMSSyncSelect);
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static void
|
|
i810GetColors (ScreenPtr pScreen, int fb, int ndefs, xColorItem *c)
|
|
{
|
|
|
|
if (I810_DEBUG)
|
|
fprintf(stderr,"i810GetColors (NOT IMPLEMENTED)\n");
|
|
}
|
|
|
|
#define DACDelay(hw) \
|
|
do { \
|
|
unsigned char temp = Vminb((hw)->IOBase + VGA_IN_STAT_1_OFFSET); \
|
|
temp = Vminb((hw)->IOBase + VGA_IN_STAT_1_OFFSET); \
|
|
} while (0)
|
|
|
|
static void
|
|
i810PutColors (ScreenPtr pScreen, int fb, int ndef, xColorItem *pdefs)
|
|
{
|
|
|
|
KdScreenPriv(pScreen);
|
|
KdScreenInfo *screen = pScreenPriv->screen;
|
|
KdCardInfo *card = screen->card;
|
|
I810CardInfo *i810c = (I810CardInfo *) card->driver;
|
|
|
|
i810VGAPtr vgap = &i810c->vga;
|
|
|
|
if (I810_DEBUG)
|
|
fprintf(stderr,"i810PutColors\n");
|
|
|
|
while (ndef--)
|
|
{
|
|
mmioWriteDacWriteAddr(vgap, pdefs->pixel);
|
|
DACDelay(vgap);
|
|
mmioWriteDacData(vgap, pdefs->red);
|
|
DACDelay(vgap);
|
|
mmioWriteDacData(vgap, pdefs->green);
|
|
DACDelay(vgap);
|
|
mmioWriteDacData(vgap, pdefs->blue);
|
|
DACDelay(vgap);
|
|
|
|
pdefs++;
|
|
}
|
|
}
|
|
|
|
|
|
KdCardFuncs i810Funcs = {
|
|
i810CardInit, /* cardinit */
|
|
i810ScreenInit, /* scrinit */
|
|
i810InitScreen, /* initScreen */
|
|
i810FinishInitScreen, /* finishInitScreen */
|
|
NULL, /* createResources */
|
|
i810Preserve, /* preserve */
|
|
i810Enable, /* enable */
|
|
i810DPMS, /* dpms */
|
|
i810Disable, /* disable */
|
|
i810Restore, /* restore */
|
|
i810ScreenFini, /* scrfini */
|
|
i810CardFini, /* cardfini */
|
|
|
|
i810CursorInit, /* initCursor */
|
|
i810CursorEnable, /* enableCursor */
|
|
i810CursorDisable, /* disableCursor */
|
|
i810CursorFini, /* finiCursor */
|
|
NULL, /* recolorCursor */
|
|
|
|
i810InitAccel, /* initAccel */
|
|
i810EnableAccel, /* enableAccel */
|
|
i810DisableAccel, /* disableAccel */
|
|
i810FiniAccel, /* finiAccel */
|
|
|
|
i810GetColors, /* getColors */
|
|
i810PutColors, /* putColors */
|
|
};
|