2008-12-13 07:51:28 +00:00
|
|
|
/*
|
2009-02-11 06:52:01 +00:00
|
|
|
* Copyright 2007-2008 Luc Verhaegen <lverhaegen@novell.com>
|
|
|
|
* Copyright 2007-2008 Matthias Hopf <mhopf@novell.com>
|
|
|
|
* Copyright 2007-2008 Egbert Eich <eich@novell.com>
|
2008-12-13 07:51:28 +00:00
|
|
|
*
|
|
|
|
* 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, sublicense,
|
|
|
|
* and/or sell copies of the Software, and to permit persons to whom the
|
|
|
|
* Software is furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* 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. IN NO EVENT SHALL
|
|
|
|
* THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
|
|
|
|
* OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
|
|
|
|
* ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
|
|
|
* OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "xf86.h"
|
|
|
|
#if HAVE_XF86_ANSIC_H
|
|
|
|
# include "xf86_ansic.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "rhd.h"
|
|
|
|
#include "rhd_vga.h"
|
|
|
|
#include "rhd_regs.h"
|
|
|
|
#include "rhd_mc.h"
|
|
|
|
|
2009-02-11 06:52:01 +00:00
|
|
|
struct rhdVGA {
|
|
|
|
Bool Stored;
|
|
|
|
|
|
|
|
CARD32 FBOffset;
|
|
|
|
CARD8 *FB;
|
|
|
|
int FBSize; /* 256kB */
|
|
|
|
|
|
|
|
CARD32 Render_Control;
|
|
|
|
CARD32 Mode_Control;
|
|
|
|
CARD32 HDP_Control;
|
|
|
|
CARD32 D1_Control;
|
|
|
|
CARD32 D2_Control;
|
|
|
|
};
|
|
|
|
|
2008-12-13 07:51:28 +00:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
RHDVGAInit(RHDPtr rhdPtr)
|
|
|
|
{
|
2009-02-11 06:52:01 +00:00
|
|
|
struct rhdVGA *VGA;
|
|
|
|
|
|
|
|
RHDFUNC(rhdPtr);
|
2008-12-13 07:51:28 +00:00
|
|
|
|
|
|
|
/* Check whether one of our VGA bits is set */
|
|
|
|
if (!(_RHDRegRead(rhdPtr, VGA_RENDER_CONTROL) & 0x00030000) &&
|
|
|
|
(_RHDRegRead(rhdPtr, VGA_HDP_CONTROL) & 0x00000010) &&
|
|
|
|
!(_RHDRegRead(rhdPtr, D1VGA_CONTROL) & 0x00000001) &&
|
|
|
|
!(_RHDRegRead(rhdPtr, D2VGA_CONTROL) & 0x00000001))
|
|
|
|
return;
|
|
|
|
|
2009-02-11 06:52:01 +00:00
|
|
|
xf86DrvMsg(rhdPtr->scrnIndex, X_INFO, "Detected VGA mode.\n");
|
|
|
|
|
|
|
|
VGA = xnfcalloc(sizeof(struct rhdVGA), 1);
|
|
|
|
VGA->Stored = FALSE;
|
2008-12-13 07:51:28 +00:00
|
|
|
|
2009-02-11 06:52:01 +00:00
|
|
|
rhdPtr->VGA = VGA;
|
2008-12-13 07:51:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
*/
|
2009-02-11 06:52:01 +00:00
|
|
|
static CARD32
|
|
|
|
rhdVGAFBOffsetGet(RHDPtr rhdPtr)
|
2008-12-13 07:51:28 +00:00
|
|
|
{
|
2009-02-11 06:52:01 +00:00
|
|
|
CARD32 FBSize, VGAFBOffset, VGAFBSize = 256 * 1024;
|
|
|
|
CARD64 FBAddress = RHDMCGetFBLocation(rhdPtr, &FBSize);
|
|
|
|
CARD64 VGAFBAddress = RHDRegRead(rhdPtr, VGA_MEMORY_BASE_ADDRESS);
|
|
|
|
|
|
|
|
if (VGAFBAddress < FBAddress)
|
|
|
|
return 0xFFFFFFFF;
|
2008-12-13 07:51:28 +00:00
|
|
|
|
2009-02-11 06:52:01 +00:00
|
|
|
if ((VGAFBAddress + VGAFBSize) > (FBAddress + FBSize))
|
|
|
|
return 0xFFFFFFFF;
|
2008-12-13 07:51:28 +00:00
|
|
|
|
2009-02-11 06:52:01 +00:00
|
|
|
VGAFBOffset = VGAFBAddress - FBAddress; /* < FBSize, so 32bit */
|
|
|
|
|
|
|
|
if ((VGAFBOffset + VGAFBSize) >= rhdPtr->FbMapSize)
|
|
|
|
return 0xFFFFFFFF;
|
|
|
|
|
|
|
|
return VGAFBOffset;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This is (usually) ok, as VGASave is called after the memory has been mapped,
|
|
|
|
* but before the MC is set up. So the use of RHDMCGetFBLocation is correct in
|
|
|
|
* rhdVGAFBOffsetGet.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
rhdVGASaveFB(RHDPtr rhdPtr)
|
|
|
|
{
|
|
|
|
struct rhdVGA *VGA = rhdPtr->VGA;
|
|
|
|
|
|
|
|
ASSERT(rhdPtr->FbBase);
|
|
|
|
|
|
|
|
RHDFUNC(rhdPtr);
|
|
|
|
|
|
|
|
VGA->FBOffset = rhdVGAFBOffsetGet(rhdPtr);
|
|
|
|
|
|
|
|
if (VGA->FBOffset == 0xFFFFFFFF) {
|
|
|
|
xf86DrvMsg(rhdPtr->scrnIndex, X_WARNING, "%s: Unable to access the VGA "
|
|
|
|
"framebuffer (0x%08X)\n", __func__,
|
|
|
|
(unsigned int) RHDRegRead(rhdPtr, VGA_MEMORY_BASE_ADDRESS));
|
|
|
|
if (VGA->FB)
|
|
|
|
xfree(VGA->FB);
|
|
|
|
VGA->FB = NULL;
|
|
|
|
VGA->FBSize = 0;
|
|
|
|
return;
|
|
|
|
}
|
2008-12-13 07:51:28 +00:00
|
|
|
|
|
|
|
VGA->FBSize = 256 * 1024;
|
2009-02-11 06:52:01 +00:00
|
|
|
|
|
|
|
RHDDebug(rhdPtr->scrnIndex, "%s: VGA FB Offset 0x%08X [0x%08X]\n",
|
|
|
|
__func__, VGA->FBOffset, VGA->FBSize);
|
|
|
|
|
|
|
|
if (!VGA->FB)
|
|
|
|
VGA->FB = xcalloc(VGA->FBSize, 1);
|
|
|
|
|
2008-12-13 07:51:28 +00:00
|
|
|
if (VGA->FB)
|
|
|
|
memcpy(VGA->FB, ((CARD8 *) rhdPtr->FbBase) + VGA->FBOffset,
|
|
|
|
VGA->FBSize);
|
|
|
|
else {
|
2009-02-11 06:52:01 +00:00
|
|
|
xf86DrvMsg(rhdPtr->scrnIndex, X_WARNING, "%s: Failed to allocate"
|
2008-12-13 07:51:28 +00:00
|
|
|
" space for storing the VGA framebuffer.\n", __func__);
|
|
|
|
VGA->FBOffset = 0xFFFFFFFF;
|
|
|
|
VGA->FBSize = 0;
|
|
|
|
}
|
2009-02-11 06:52:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
RHDVGASave(RHDPtr rhdPtr)
|
|
|
|
{
|
|
|
|
struct rhdVGA *VGA = rhdPtr->VGA;
|
|
|
|
|
|
|
|
RHDFUNC(rhdPtr);
|
|
|
|
|
|
|
|
if (!VGA)
|
|
|
|
return; /* We don't need to warn , this is intended use */
|
|
|
|
|
|
|
|
VGA->Render_Control = RHDRegRead(rhdPtr, VGA_RENDER_CONTROL);
|
|
|
|
VGA->Mode_Control = RHDRegRead(rhdPtr, VGA_MODE_CONTROL);
|
|
|
|
VGA->HDP_Control = RHDRegRead(rhdPtr, VGA_HDP_CONTROL);
|
|
|
|
VGA->D1_Control = RHDRegRead(rhdPtr, D1VGA_CONTROL);
|
|
|
|
VGA->D2_Control = RHDRegRead(rhdPtr, D2VGA_CONTROL);
|
2008-12-13 07:51:28 +00:00
|
|
|
|
2009-02-11 06:52:01 +00:00
|
|
|
rhdVGASaveFB(rhdPtr);
|
2008-12-13 07:51:28 +00:00
|
|
|
VGA->Stored = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
RHDVGARestore(RHDPtr rhdPtr)
|
|
|
|
{
|
|
|
|
struct rhdVGA *VGA = rhdPtr->VGA;
|
|
|
|
|
|
|
|
RHDFUNC(rhdPtr);
|
|
|
|
|
|
|
|
if (!VGA)
|
|
|
|
return; /* We don't need to warn , this is intended use */
|
|
|
|
|
|
|
|
if (!VGA->Stored) {
|
2009-02-11 06:52:01 +00:00
|
|
|
xf86DrvMsg(rhdPtr->scrnIndex, X_ERROR,
|
|
|
|
"%s: trying to restore uninitialized values.\n", __func__);
|
2008-12-13 07:51:28 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (VGA->FB)
|
2009-02-11 06:52:01 +00:00
|
|
|
memcpy(((CARD8 *) rhdPtr->FbBase) + VGA->FBOffset, VGA->FB,
|
|
|
|
VGA->FBSize);
|
2008-12-13 07:51:28 +00:00
|
|
|
|
|
|
|
RHDRegWrite(rhdPtr, VGA_RENDER_CONTROL, VGA->Render_Control);
|
|
|
|
RHDRegWrite(rhdPtr, VGA_MODE_CONTROL, VGA->Mode_Control);
|
|
|
|
RHDRegWrite(rhdPtr, VGA_HDP_CONTROL, VGA->HDP_Control);
|
|
|
|
RHDRegWrite(rhdPtr, D1VGA_CONTROL, VGA->D1_Control);
|
|
|
|
RHDRegWrite(rhdPtr, D2VGA_CONTROL, VGA->D2_Control);
|
2009-02-11 06:52:01 +00:00
|
|
|
RHD_UNSETDEBUGFLAG(rhdPtr, VGA_SETUP);
|
2008-12-13 07:51:28 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
RHDVGADisable(RHDPtr rhdPtr)
|
|
|
|
{
|
|
|
|
RHDFUNC(rhdPtr);
|
|
|
|
|
|
|
|
_RHDRegMask(rhdPtr, VGA_RENDER_CONTROL, 0, 0x00030000);
|
|
|
|
_RHDRegMask(rhdPtr, VGA_MODE_CONTROL, 0, 0x00000030);
|
|
|
|
_RHDRegMask(rhdPtr, VGA_HDP_CONTROL, 0x00010010, 0x00010010);
|
2009-02-11 06:52:01 +00:00
|
|
|
RHDRegMask(rhdPtr, D1VGA_CONTROL, 0, D1VGA_MODE_ENABLE);
|
|
|
|
RHDRegMask(rhdPtr, D2VGA_CONTROL, 0, D2VGA_MODE_ENABLE);
|
2008-12-13 07:51:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
RHDVGADestroy(RHDPtr rhdPtr)
|
|
|
|
{
|
|
|
|
struct rhdVGA *VGA = rhdPtr->VGA;
|
|
|
|
|
|
|
|
RHDFUNC(rhdPtr);
|
|
|
|
|
|
|
|
if (!VGA)
|
|
|
|
return; /* We don't need to warn , this is intended use */
|
|
|
|
|
|
|
|
if (VGA->FB)
|
2009-02-11 06:52:01 +00:00
|
|
|
xfree(VGA->FB);
|
|
|
|
xfree(VGA);
|
2008-12-13 07:51:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|