635 lines
14 KiB
C
635 lines
14 KiB
C
/* $XFree86: xc/programs/Xserver/hw/xfree86/os-support/linux/lnx_video.c,v 3.13.2.1 1997/05/11 05:04:25 dawes Exp $ */
|
|
/*
|
|
* Copyright 1992 by Orest Zborowski <obz@Kodak.com>
|
|
* Copyright 1993 by David Wexelblat <dwex@goblin.org>
|
|
*
|
|
* Permission to use, copy, modify, distribute, and sell this software and its
|
|
* documentation for any purpose is hereby granted without fee, provided that
|
|
* the above copyright notice appear in all copies and that both that
|
|
* copyright notice and this permission notice appear in supporting
|
|
* documentation, and that the names of Orest Zborowski and David Wexelblat
|
|
* not be used in advertising or publicity pertaining to distribution of
|
|
* the software without specific, written prior permission. Orest Zborowski
|
|
* and David Wexelblat make no representations about the suitability of this
|
|
* software for any purpose. It is provided "as is" without express or
|
|
* implied warranty.
|
|
*
|
|
* OREST ZBOROWSKI AND DAVID WEXELBLAT DISCLAIMS ALL WARRANTIES WITH REGARD
|
|
* TO THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
|
|
* FITNESS, IN NO EVENT SHALL OREST ZBOROWSKI OR DAVID WEXELBLAT BE LIABLE
|
|
* FOR 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.
|
|
*
|
|
*/
|
|
/* $Xorg: lnx_video.c,v 1.3 2000/08/17 19:51:23 cpqbld Exp $ */
|
|
|
|
#include "X.h"
|
|
#include "input.h"
|
|
#include "scrnintstr.h"
|
|
|
|
#include "xf86.h"
|
|
#include "xf86Priv.h"
|
|
#include "xf86_OSlib.h"
|
|
|
|
#ifdef __alpha__
|
|
|
|
/*
|
|
* The Jensen lacks dense memory, thus we have to address the bus via
|
|
* the sparse addressing scheme.
|
|
*
|
|
* Martin Ostermann (ost@comnets.rwth-aachen.de) - Apr.-Sep. 1996
|
|
*/
|
|
|
|
#ifdef TEST_JENSEN_CODE /* define to test the Sparse addressing on a non-Jensen */
|
|
#define SPARSE (5)
|
|
#define isJensen (1)
|
|
#else
|
|
#define isJensen (!_bus_base())
|
|
#define SPARSE (7)
|
|
#endif
|
|
|
|
#define BUS_BASE (isJensen ? _bus_base_sparse() : _bus_base())
|
|
#define JENSEN_SHIFT(x) (isJensen ? ((long)x<<SPARSE) : (long)x)
|
|
#else
|
|
#define BUS_BASE 0
|
|
#define JENSEN_SHIFT(x) (x)
|
|
#endif
|
|
|
|
/***************************************************************************/
|
|
/* Video Memory Mapping section */
|
|
/***************************************************************************/
|
|
|
|
/*
|
|
* Unfortunatly mmap without MAP_FIXED only works the first time :-(
|
|
* This is now fixed in pl13 ALPHA, but still seems to have problems.
|
|
*/
|
|
#undef ONLY_MMAP_FIXED_WORKS
|
|
|
|
#ifdef ONLY_MMAP_FIXED_WORKS
|
|
static pointer AllocAddress[MAXSCREENS][NUM_REGIONS];
|
|
#endif
|
|
|
|
#if 0
|
|
static struct xf86memMap {
|
|
int offset;
|
|
int memSize;
|
|
} xf86memMaps[MAXSCREENS];
|
|
#endif
|
|
|
|
pointer xf86MapVidMem(ScreenNum, Region, Base, Size)
|
|
int ScreenNum;
|
|
int Region;
|
|
pointer Base;
|
|
unsigned long Size;
|
|
{
|
|
pointer base;
|
|
int fd;
|
|
|
|
#ifdef ONLY_MMAP_FIXED_WORKS
|
|
#ifdef __alpha__
|
|
FatalError("xf86MapVidMem: Unexpected code for Alpha (pagesize=8k!)\n");
|
|
#endif
|
|
AllocAddress[ScreenNum][Region] = (pointer)xalloc(Size + 0x1000);
|
|
if (AllocAddress[ScreenNum][Region] == NULL)
|
|
{
|
|
FatalError("xf86MapVidMem: can't alloc framebuffer space\n");
|
|
}
|
|
base = (pointer)(((unsigned int)AllocAddress[ScreenNum][Region]
|
|
& ~0xFFF) + 0x1000);
|
|
if ((fd = open("/dev/mem", O_RDWR)) < 0)
|
|
{
|
|
FatalError("xf86MapVidMem: failed to open /dev/mem (%s)\n",
|
|
strerror(errno));
|
|
}
|
|
base = (pointer)mmap((caddr_t)base, Size, PROT_READ|PROT_WRITE,
|
|
MAP_FIXED|MAP_SHARED, fd, (off_t)Base);
|
|
#else
|
|
if ((fd = open("/dev/mem", O_RDWR)) < 0)
|
|
{
|
|
FatalError("xf86MapVidMem: failed to open /dev/mem (%s)\n",
|
|
strerror(errno));
|
|
}
|
|
/* This requirers linux-0.99.pl10 or above */
|
|
base = (pointer)mmap((caddr_t)0, JENSEN_SHIFT(Size),
|
|
PROT_READ|PROT_WRITE,
|
|
MAP_SHARED, fd,
|
|
(off_t)(JENSEN_SHIFT((off_t)Base) + BUS_BASE));
|
|
#endif
|
|
close(fd);
|
|
if ((long)base == -1)
|
|
{
|
|
FatalError("xf86MapVidMem: Could not mmap framebuffer (%s)\n",
|
|
strerror(errno));
|
|
}
|
|
#if 0
|
|
xf86memMaps[ScreenNum].offset = (int) Base;
|
|
xf86memMaps[ScreenNum].memSize = Size;
|
|
#endif
|
|
return base;
|
|
}
|
|
|
|
#if 0
|
|
void xf86GetVidMemData(ScreenNum, Base, Size)
|
|
int ScreenNum;
|
|
int *Base;
|
|
int *Size;
|
|
{
|
|
*Base = xf86memMaps[ScreenNum].offset;
|
|
*Size = xf86memMaps[ScreenNum].memSize;
|
|
}
|
|
#endif
|
|
|
|
void xf86UnMapVidMem(ScreenNum, Region, Base, Size)
|
|
int ScreenNum;
|
|
int Region;
|
|
pointer Base;
|
|
unsigned long Size;
|
|
{
|
|
munmap((caddr_t)JENSEN_SHIFT(Base), JENSEN_SHIFT(Size));
|
|
#ifdef ONLY_MMAP_FIXED_WORKS
|
|
xfree(AllocAddress[ScreenNum][Region]);
|
|
#endif
|
|
}
|
|
|
|
Bool xf86LinearVidMem()
|
|
{
|
|
return(TRUE);
|
|
}
|
|
|
|
/***************************************************************************/
|
|
/* I/O Permissions section */
|
|
/***************************************************************************/
|
|
|
|
/*
|
|
* Linux handles regular (<= 0x3ff) ports with the TSS I/O bitmap, and
|
|
* extended ports with the iopl() system call.
|
|
*
|
|
* For testing, it's useful to enable only the ports we need, but for
|
|
* production purposes, it's faster to enable all ports.
|
|
*/
|
|
#define ALWAYS_USE_EXTENDED
|
|
|
|
#ifdef ALWAYS_USE_EXTENDED
|
|
|
|
static Bool ScreenEnabled[MAXSCREENS];
|
|
static Bool ExtendedEnabled = FALSE;
|
|
static Bool InitDone = FALSE;
|
|
|
|
void xf86ClearIOPortList(ScreenNum)
|
|
int ScreenNum;
|
|
{
|
|
if (!InitDone)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < MAXSCREENS; i++)
|
|
ScreenEnabled[i] = FALSE;
|
|
InitDone = TRUE;
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
void xf86AddIOPorts(ScreenNum, NumPorts, Ports)
|
|
int ScreenNum;
|
|
int NumPorts;
|
|
unsigned *Ports;
|
|
{
|
|
return;
|
|
}
|
|
|
|
void xf86EnableIOPorts(ScreenNum)
|
|
int ScreenNum;
|
|
{
|
|
int i;
|
|
|
|
ScreenEnabled[ScreenNum] = TRUE;
|
|
|
|
if (ExtendedEnabled)
|
|
return;
|
|
|
|
#ifndef __mc68000__
|
|
if (iopl(3))
|
|
FatalError("%s: Failed to set IOPL for I/O\n",
|
|
"xf86EnableIOPorts");
|
|
#endif
|
|
ExtendedEnabled = TRUE;
|
|
|
|
return;
|
|
}
|
|
|
|
void xf86DisableIOPorts(ScreenNum)
|
|
int ScreenNum;
|
|
{
|
|
int i;
|
|
|
|
ScreenEnabled[ScreenNum] = FALSE;
|
|
|
|
if (!ExtendedEnabled)
|
|
return;
|
|
|
|
for (i = 0; i < MAXSCREENS; i++)
|
|
if (ScreenEnabled[i])
|
|
return;
|
|
|
|
#ifndef __mc68000__
|
|
iopl(0);
|
|
#endif
|
|
ExtendedEnabled = FALSE;
|
|
|
|
return;
|
|
}
|
|
|
|
#else /* !ALWAYS_USE_EXTENDED */
|
|
|
|
static unsigned *EnabledPorts[MAXSCREENS];
|
|
static int NumEnabledPorts[MAXSCREENS];
|
|
static Bool ScreenEnabled[MAXSCREENS];
|
|
static Bool ExtendedPorts[MAXSCREENS];
|
|
static Bool ExtendedEnabled = FALSE;
|
|
static Bool InitDone = FALSE;
|
|
|
|
void xf86ClearIOPortList(ScreenNum)
|
|
int ScreenNum;
|
|
{
|
|
if (!InitDone)
|
|
{
|
|
xf86InitPortLists(EnabledPorts, NumEnabledPorts,
|
|
ScreenEnabled, ExtendedPorts, MAXSCREENS);
|
|
InitDone = TRUE;
|
|
return;
|
|
}
|
|
ExtendedPorts[ScreenNum] = FALSE;
|
|
if (EnabledPorts[ScreenNum] != (unsigned *)NULL)
|
|
xfree(EnabledPorts[ScreenNum]);
|
|
EnabledPorts[ScreenNum] = (unsigned *)NULL;
|
|
NumEnabledPorts[ScreenNum] = 0;
|
|
}
|
|
|
|
void xf86AddIOPorts(ScreenNum, NumPorts, Ports)
|
|
int ScreenNum;
|
|
int NumPorts;
|
|
unsigned *Ports;
|
|
{
|
|
int i;
|
|
|
|
if (!InitDone)
|
|
{
|
|
FatalError("xf86AddIOPorts: I/O control lists not initialised\n");
|
|
}
|
|
EnabledPorts[ScreenNum] = (unsigned *)xrealloc(EnabledPorts[ScreenNum],
|
|
(NumEnabledPorts[ScreenNum]+NumPorts)*sizeof(unsigned));
|
|
for (i = 0; i < NumPorts; i++)
|
|
{
|
|
EnabledPorts[ScreenNum][NumEnabledPorts[ScreenNum] + i] =
|
|
Ports[i];
|
|
if (Ports[i] > 0x3FF)
|
|
ExtendedPorts[ScreenNum] = TRUE;
|
|
}
|
|
NumEnabledPorts[ScreenNum] += NumPorts;
|
|
}
|
|
|
|
void xf86EnableIOPorts(ScreenNum)
|
|
int ScreenNum;
|
|
{
|
|
int i;
|
|
|
|
if (ScreenEnabled[ScreenNum])
|
|
return;
|
|
|
|
for (i = 0; i < MAXSCREENS; i++)
|
|
{
|
|
if (ExtendedPorts[i] && (ScreenEnabled[i] || i == ScreenNum))
|
|
{
|
|
#ifndef __mc68000__
|
|
if (iopl(3))
|
|
{
|
|
FatalError("%s: Failed to set IOPL for extended I/O\n",
|
|
"xf86EnableIOPorts");
|
|
}
|
|
#endif
|
|
ExtendedEnabled = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
/* Extended I/O was used, but not any more */
|
|
if (ExtendedEnabled && i == MAXSCREENS)
|
|
{
|
|
#ifndef __mc68000__
|
|
iopl(0);
|
|
#endif
|
|
ExtendedEnabled = FALSE;
|
|
}
|
|
/*
|
|
* Turn on non-extended ports even when using extended I/O
|
|
* so they are there if extended I/O gets turned off when it's no
|
|
* longer needed.
|
|
*/
|
|
for (i = 0; i < NumEnabledPorts[ScreenNum]; i++)
|
|
{
|
|
unsigned port = EnabledPorts[ScreenNum][i];
|
|
|
|
if (port > 0x3FF)
|
|
continue;
|
|
|
|
if (xf86CheckPorts(port, EnabledPorts, NumEnabledPorts,
|
|
ScreenEnabled, MAXSCREENS))
|
|
{
|
|
if (ioperm(port, 1, TRUE) < 0)
|
|
{
|
|
FatalError("%s: Failed to enable I/O port 0x%x (%s)\n",
|
|
"xf86EnableIOPorts", port, strerror(errno));
|
|
}
|
|
}
|
|
}
|
|
ScreenEnabled[ScreenNum] = TRUE;
|
|
return;
|
|
}
|
|
|
|
void xf86DisableIOPorts(ScreenNum)
|
|
int ScreenNum;
|
|
{
|
|
int i;
|
|
|
|
if (!ScreenEnabled[ScreenNum])
|
|
return;
|
|
|
|
ScreenEnabled[ScreenNum] = FALSE;
|
|
for (i = 0; i < MAXSCREENS; i++)
|
|
{
|
|
if (ScreenEnabled[i] && ExtendedPorts[i])
|
|
break;
|
|
}
|
|
if (ExtendedEnabled && i == MAXSCREENS)
|
|
{
|
|
#ifndef __mc68000__
|
|
iopl(0);
|
|
#endif
|
|
ExtendedEnabled = FALSE;
|
|
}
|
|
for (i = 0; i < NumEnabledPorts[ScreenNum]; i++)
|
|
{
|
|
unsigned port = EnabledPorts[ScreenNum][i];
|
|
|
|
if (port > 0x3FF)
|
|
continue;
|
|
|
|
if (xf86CheckPorts(port, EnabledPorts, NumEnabledPorts,
|
|
ScreenEnabled, MAXSCREENS))
|
|
{
|
|
ioperm(port, 1, FALSE);
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
|
|
#endif /* ALWAYS_USE_EXTENDED */
|
|
|
|
void xf86DisableIOPrivs()
|
|
{
|
|
#ifndef __mc68000__
|
|
if (ExtendedEnabled)
|
|
iopl(0);
|
|
#endif
|
|
return;
|
|
}
|
|
|
|
/***************************************************************************/
|
|
/* Interrupt Handling section */
|
|
/***************************************************************************/
|
|
|
|
Bool xf86DisableInterrupts()
|
|
{
|
|
if (!ExtendedEnabled)
|
|
#ifndef __mc68000__
|
|
if (iopl(3))
|
|
return (FALSE);
|
|
#endif
|
|
#if defined(__alpha__) || defined(__mc68000__)
|
|
#else
|
|
#ifdef __GNUC__
|
|
__asm__ __volatile__("cli");
|
|
#else
|
|
asm("cli");
|
|
#endif
|
|
#endif
|
|
#ifndef __mc68000__
|
|
if (!ExtendedEnabled)
|
|
iopl(0);
|
|
#endif
|
|
return (TRUE);
|
|
}
|
|
|
|
void xf86EnableInterrupts()
|
|
{
|
|
if (!ExtendedEnabled)
|
|
#ifndef __mc68000__
|
|
if (iopl(3))
|
|
return;
|
|
#endif
|
|
#if defined(__alpha__) || defined(__mc68000__)
|
|
#else
|
|
#ifdef __GNUC__
|
|
__asm__ __volatile__("sti");
|
|
#else
|
|
asm("sti");
|
|
#endif
|
|
#endif
|
|
#ifndef __mc68000__
|
|
if (!ExtendedEnabled)
|
|
iopl(0);
|
|
#endif
|
|
return;
|
|
}
|
|
|
|
#if defined(__alpha__)
|
|
|
|
static int xf86SparseShift = 5; /* default to all but JENSEN */
|
|
|
|
pointer xf86MapVidMemSparse(ScreenNum, Region, Base, Size)
|
|
int ScreenNum;
|
|
int Region;
|
|
pointer Base;
|
|
unsigned long Size;
|
|
{
|
|
pointer base;
|
|
int fd;
|
|
|
|
if (!_bus_base()) xf86SparseShift = 7; /* Uh, oh, JENSEN... */
|
|
|
|
Size <<= xf86SparseShift;
|
|
Base = (pointer)((unsigned long)Base << xf86SparseShift);
|
|
|
|
if ((fd = open("/dev/mem", O_RDWR)) < 0)
|
|
{
|
|
FatalError("xf86MapVidMem: failed to open /dev/mem (%s)\n",
|
|
strerror(errno));
|
|
}
|
|
/* This requirers linux-0.99.pl10 or above */
|
|
base = (pointer)mmap((caddr_t)0, Size,
|
|
PROT_READ | PROT_WRITE,
|
|
MAP_SHARED, fd,
|
|
(off_t)Base + _bus_base_sparse());
|
|
close(fd);
|
|
if ((long)base == -1)
|
|
{
|
|
FatalError("xf86MapVidMem: Could not mmap framebuffer (%s)\n",
|
|
strerror(errno));
|
|
}
|
|
return base;
|
|
}
|
|
|
|
void xf86UnMapVidMemSparse(ScreenNum, Region, Base, Size)
|
|
int ScreenNum;
|
|
int Region;
|
|
pointer Base;
|
|
unsigned long Size;
|
|
{
|
|
Size <<= xf86SparseShift;
|
|
|
|
munmap((caddr_t)Base, Size);
|
|
}
|
|
|
|
#define vuip volatile unsigned int *
|
|
|
|
extern void sethae(unsigned long hae);
|
|
|
|
int xf86ReadSparse8(Base, Offset)
|
|
pointer Base;
|
|
unsigned long Offset;
|
|
{
|
|
unsigned long result, shift;
|
|
unsigned long msb = 0;
|
|
|
|
shift = (Offset & 0x3) * 8;
|
|
if (xf86SparseShift != 7) { /* if not JENSEN, we may need HAE */
|
|
if (Offset >= (1UL << 24)) {
|
|
msb = Offset & 0xf8000000UL;
|
|
Offset -= msb;
|
|
if (msb) {
|
|
sethae(msb);
|
|
}
|
|
}
|
|
}
|
|
result = *(vuip) ((unsigned long)Base + (Offset << xf86SparseShift));
|
|
if (msb)
|
|
sethae(0);
|
|
result >>= shift;
|
|
return 0xffUL & result;
|
|
}
|
|
|
|
int xf86ReadSparse16(Base, Offset)
|
|
pointer Base;
|
|
unsigned long Offset;
|
|
{
|
|
unsigned long result, shift;
|
|
unsigned long msb = 0;
|
|
|
|
shift = (Offset & 0x2) * 8;
|
|
if (xf86SparseShift != 7) { /* if not JENSEN, we may need HAE */
|
|
if (Offset >= (1UL << 24)) {
|
|
msb = Offset & 0xf8000000UL;
|
|
Offset -= msb;
|
|
if (msb) {
|
|
sethae(msb);
|
|
}
|
|
}
|
|
}
|
|
result = *(vuip)((unsigned long)Base+(Offset<<xf86SparseShift)+(1<<(xf86SparseShift-2)));
|
|
if (msb)
|
|
sethae(0);
|
|
result >>= shift;
|
|
return 0xffffUL & result;
|
|
}
|
|
|
|
int xf86ReadSparse32(Base, Offset)
|
|
pointer Base;
|
|
unsigned long Offset;
|
|
{
|
|
unsigned long result;
|
|
unsigned long msb = 0;
|
|
|
|
if (xf86SparseShift != 7) { /* if not JENSEN, we may need HAE */
|
|
if (Offset >= (1UL << 24)) {
|
|
msb = Offset & 0xf8000000UL;
|
|
Offset -= msb;
|
|
if (msb) {
|
|
sethae(msb);
|
|
}
|
|
}
|
|
}
|
|
result = *(vuip)((unsigned long)Base+(Offset<<xf86SparseShift)+(3<<(xf86SparseShift-2)));
|
|
if (msb)
|
|
sethae(0);
|
|
return result;
|
|
}
|
|
|
|
void xf86WriteSparse8(Value, Base, Offset)
|
|
int Value;
|
|
pointer Base;
|
|
unsigned long Offset;
|
|
{
|
|
unsigned long msb = 0;
|
|
unsigned int b = Value & 0xffU;
|
|
|
|
if (xf86SparseShift != 7) { /* not JENSEN */
|
|
if (Offset >= (1UL << 24)) {
|
|
msb = Offset & 0xf8000000;
|
|
Offset -= msb;
|
|
if (msb) {
|
|
sethae(msb);
|
|
}
|
|
}
|
|
}
|
|
*(vuip) ((unsigned long)Base + (Offset << xf86SparseShift)) = b * 0x01010101;
|
|
if (msb)
|
|
sethae(0);
|
|
}
|
|
|
|
void xf86WriteSparse16(Value, Base, Offset)
|
|
int Value;
|
|
pointer Base;
|
|
unsigned long Offset;
|
|
{
|
|
unsigned long msb = 0;
|
|
unsigned int w = Value & 0xffffU;
|
|
|
|
if (xf86SparseShift != 7) { /* not JENSEN */
|
|
if (Offset >= (1UL << 24)) {
|
|
msb = Offset & 0xf8000000;
|
|
Offset -= msb;
|
|
if (msb) {
|
|
sethae(msb);
|
|
}
|
|
}
|
|
}
|
|
*(vuip)((unsigned long)Base+(Offset<<xf86SparseShift)+(1<<(xf86SparseShift-2))) =
|
|
w * 0x00010001;
|
|
if (msb)
|
|
sethae(0);
|
|
}
|
|
|
|
void xf86WriteSparse32(Value, Base, Offset)
|
|
int Value;
|
|
pointer Base;
|
|
unsigned long Offset;
|
|
{
|
|
unsigned long msb = 0;
|
|
|
|
if (xf86SparseShift != 7) { /* not JENSEN */
|
|
if (Offset >= (1UL << 24)) {
|
|
msb = Offset & 0xf8000000;
|
|
Offset -= msb;
|
|
if (msb) {
|
|
sethae(msb);
|
|
}
|
|
}
|
|
}
|
|
*(vuip)((unsigned long)Base+(Offset<<xf86SparseShift)+(3<<(xf86SparseShift-2))) = Value;
|
|
if (msb)
|
|
sethae(0);
|
|
}
|
|
#endif /* __alpha__ */
|