266 lines
7.7 KiB
C
266 lines
7.7 KiB
C
/*
|
|
* Copyright 1990,91 by Thomas Roell, Dinkelscherben, Germany
|
|
* Copyright 1993 by David Wexelblat <dwex@goblin.org>
|
|
* Copyright 1999 by David Holland <davidh@iquest.net>
|
|
*
|
|
* 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 the copyright holders not be used in advertising or
|
|
* publicity pertaining to distribution of the software without specific,
|
|
* written prior permission. The copyright holders make no representations
|
|
* about the suitability of this software for any purpose. It is provided "as
|
|
* is" without express or implied warranty.
|
|
*
|
|
* THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
|
|
* INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT
|
|
* SHALL THE COPYRIGHT HOLDERS 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.
|
|
*
|
|
*/
|
|
/* Copyright 2008 Sun Microsystems, Inc. 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.
|
|
*/
|
|
|
|
#ifdef HAVE_XORG_CONFIG_H
|
|
#include <xorg-config.h>
|
|
#endif
|
|
|
|
#include <sys/types.h> /* get __x86 definition if not set by compiler */
|
|
|
|
#if defined(__i386__) || defined(__i386) || defined(__x86)
|
|
# define _NEED_SYSI86
|
|
#endif
|
|
#include "xf86.h"
|
|
#include "xf86Priv.h"
|
|
#include "xf86_OSlib.h"
|
|
#include "xf86OSpriv.h"
|
|
#include <sys/mman.h>
|
|
|
|
/***************************************************************************/
|
|
/* Video Memory Mapping section */
|
|
/***************************************************************************/
|
|
|
|
static char *apertureDevName = NULL;
|
|
static int apertureDevFD_ro = -1;
|
|
static int apertureDevFD_rw = -1;
|
|
|
|
static Bool
|
|
solOpenAperture(void)
|
|
{
|
|
if (apertureDevName == NULL)
|
|
{
|
|
apertureDevName = "/dev/xsvc";
|
|
if ((apertureDevFD_rw = open(apertureDevName, O_RDWR)) < 0)
|
|
{
|
|
xf86MsgVerb(X_WARNING, 0,
|
|
"solOpenAperture: failed to open %s (%s)\n",
|
|
apertureDevName, strerror(errno));
|
|
apertureDevName = "/dev/fbs/aperture";
|
|
apertureDevFD_rw = open(apertureDevName, O_RDWR);
|
|
}
|
|
apertureDevFD_ro = open(apertureDevName, O_RDONLY);
|
|
|
|
if ((apertureDevFD_rw < 0) || (apertureDevFD_ro < 0))
|
|
{
|
|
xf86MsgVerb(X_WARNING, 0,
|
|
"solOpenAperture: failed to open %s (%s)\n",
|
|
apertureDevName, strerror(errno));
|
|
xf86MsgVerb(X_WARNING, 0,
|
|
"solOpenAperture: either /dev/fbs/aperture"
|
|
" or /dev/xsvc required\n");
|
|
|
|
apertureDevName = NULL;
|
|
|
|
if (apertureDevFD_rw >= 0)
|
|
{
|
|
close(apertureDevFD_rw);
|
|
}
|
|
apertureDevFD_rw = -1;
|
|
|
|
if (apertureDevFD_ro >= 0)
|
|
{
|
|
close(apertureDevFD_ro);
|
|
}
|
|
apertureDevFD_ro = -1;
|
|
|
|
return FALSE;
|
|
}
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
static pointer
|
|
solMapVidMem(int ScreenNum, unsigned long Base, unsigned long Size, int Flags)
|
|
{
|
|
pointer base;
|
|
int fd;
|
|
int prot;
|
|
|
|
if (Flags & VIDMEM_READONLY)
|
|
{
|
|
fd = apertureDevFD_ro;
|
|
prot = PROT_READ;
|
|
}
|
|
else
|
|
{
|
|
fd = apertureDevFD_rw;
|
|
prot = PROT_READ | PROT_WRITE;
|
|
}
|
|
|
|
if (fd < 0)
|
|
{
|
|
xf86DrvMsg(ScreenNum, X_ERROR,
|
|
"solMapVidMem: failed to open %s (%s)\n",
|
|
apertureDevName, strerror(errno));
|
|
return NULL;
|
|
}
|
|
|
|
base = mmap(NULL, Size, prot, MAP_SHARED, fd, (off_t)Base);
|
|
|
|
if (base == MAP_FAILED) {
|
|
xf86DrvMsg(ScreenNum, X_ERROR,
|
|
"solMapVidMem: failed to mmap %s (0x%08lx,0x%lx) (%s)\n",
|
|
apertureDevName, Base, Size, strerror(errno));
|
|
return NULL;
|
|
}
|
|
|
|
return base;
|
|
}
|
|
|
|
/* ARGSUSED */
|
|
static void
|
|
solUnMapVidMem(int ScreenNum, pointer Base, unsigned long Size)
|
|
{
|
|
if (munmap(Base, Size) != 0) {
|
|
xf86DrvMsgVerb(ScreenNum, X_WARNING, 0,
|
|
"solUnMapVidMem: failed to unmap %s"
|
|
" (0x%08lx,0x%lx) (%s)\n",
|
|
apertureDevName, Base, Size,
|
|
strerror(errno));
|
|
}
|
|
}
|
|
|
|
_X_HIDDEN void
|
|
xf86OSInitVidMem(VidMemInfoPtr pVidMem)
|
|
{
|
|
pVidMem->linearSupported = solOpenAperture();
|
|
if (pVidMem->linearSupported) {
|
|
pVidMem->mapMem = solMapVidMem;
|
|
pVidMem->unmapMem = solUnMapVidMem;
|
|
} else {
|
|
xf86MsgVerb(X_WARNING, 0,
|
|
"xf86OSInitVidMem: linear memory access disabled\n");
|
|
}
|
|
pVidMem->initialised = TRUE;
|
|
}
|
|
|
|
/*
|
|
* Read BIOS via mmap()ing physical memory.
|
|
*/
|
|
_X_EXPORT int
|
|
xf86ReadBIOS(unsigned long Base, unsigned long Offset, unsigned char *Buf,
|
|
int Len)
|
|
{
|
|
unsigned char *ptr;
|
|
int psize;
|
|
int mlen;
|
|
|
|
psize = getpagesize();
|
|
Offset += Base & (psize - 1);
|
|
Base &= ~(psize - 1);
|
|
mlen = (Offset + Len + psize - 1) & ~(psize - 1);
|
|
|
|
if (solOpenAperture() == FALSE)
|
|
{
|
|
xf86Msg(X_WARNING,
|
|
"xf86ReadBIOS: Failed to open aperture to read BIOS\n");
|
|
return -1;
|
|
}
|
|
|
|
ptr = (unsigned char *)mmap(NULL, mlen, PROT_READ,
|
|
MAP_SHARED, apertureDevFD_ro, (off_t)Base);
|
|
if (ptr == MAP_FAILED)
|
|
{
|
|
xf86Msg(X_WARNING, "xf86ReadBIOS: %s mmap failed [0x%08lx, 0x%04x]\n",
|
|
apertureDevName, Base, mlen);
|
|
return -1;
|
|
}
|
|
|
|
(void)memcpy(Buf, (void *)(ptr + Offset), Len);
|
|
if (munmap((caddr_t)ptr, mlen) != 0) {
|
|
xf86MsgVerb(X_WARNING, 0,
|
|
"solUnMapVidMem: failed to unmap %s"
|
|
" (0x%08lx,0x%lx) (%s)\n",
|
|
apertureDevName, ptr, mlen, strerror(errno));
|
|
}
|
|
|
|
return Len;
|
|
}
|
|
|
|
|
|
/***************************************************************************/
|
|
/* I/O Permissions section */
|
|
/***************************************************************************/
|
|
|
|
#if defined(__i386__) || defined(__i386) || defined(__x86)
|
|
static Bool ExtendedEnabled = FALSE;
|
|
#endif
|
|
|
|
_X_EXPORT Bool
|
|
xf86EnableIO(void)
|
|
{
|
|
#if defined(__i386__) || defined(__i386) || defined(__x86)
|
|
if (ExtendedEnabled)
|
|
return TRUE;
|
|
|
|
if (sysi86(SI86V86, V86SC_IOPL, PS_IOPL) < 0) {
|
|
xf86Msg(X_WARNING, "xf86EnableIOPorts: Failed to set IOPL for I/O\n");
|
|
return FALSE;
|
|
}
|
|
ExtendedEnabled = TRUE;
|
|
#endif /* i386 */
|
|
return TRUE;
|
|
}
|
|
|
|
_X_EXPORT void
|
|
xf86DisableIO(void)
|
|
{
|
|
#if defined(__i386__) || defined(__i386) || defined(__x86)
|
|
if(!ExtendedEnabled)
|
|
return;
|
|
|
|
sysi86(SI86V86, V86SC_IOPL, 0);
|
|
|
|
ExtendedEnabled = FALSE;
|
|
#endif /* i386 */
|
|
}
|