1548 lines
39 KiB
C
Raw Normal View History

/*
* Copyright 2007 Luc Verhaegen <lverhaegen@novell.com>
* Copyright 2007 Matthias Hopf <mhopf@novell.com>
* Copyright 2007 Egbert Eich <eich@novell.com>
* Copyright 2007 Advanced Micro Devices, Inc.
*
* 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"
/* for usleep */
#if HAVE_XF86_ANSIC_H
# include "xf86_ansic.h"
#else
# include <unistd.h>
#endif
#include "rhd.h"
#include "rhd_crtc.h"
#include "rhd_pll.h"
#include "rhd_regs.h"
#include "rhd_atombios.h"
#define PLL_CALIBRATE_WAIT 0x100000
/*
* Get gain, charge pump, loop filter and current bias.
* For R500, this is done in atombios by ASIC_RegistersInit
* Some data table in atom should've provided this information.
*/
struct PLL_Control {
CARD16 FeedbackDivider; /* 0xFFFF/-1 is the endmarker here */
CARD32 Control;
};
/* From hardcoded values. */
static struct PLL_Control RV610PLLControl[] =
{
{ 0x0049, 0x159F8704 },
{ 0x006C, 0x159B8704 },
{ 0xFFFF, 0x159EC704 }
};
/* Some tables are provided by atombios,
* it's just that they are hidden away deliberately and not exposed */
static struct PLL_Control RV670PLLControl[] =
{
{ 0x004A, 0x159FC704 },
{ 0x0067, 0x159BC704 },
{ 0x00C4, 0x159EC704 },
{ 0x00F4, 0x1593A704 },
{ 0x0136, 0x1595A704 },
{ 0x01A4, 0x1596A704 },
{ 0x022C, 0x159CE504 },
{ 0xFFFF, 0x1591E404 }
};
/*
* Used by PLLElectrical() for r5xx+ and by rv620/35 code.
*/
static CARD32
PLLControlTableRetrieve(struct PLL_Control *Table, CARD16 FeedbackDivider)
{
int i;
for (i = 0; Table[i].FeedbackDivider < 0xFFFF ; i++)
if (Table[i].FeedbackDivider >= FeedbackDivider)
break;
return Table[i].Control;
}
/*
* Not used by rv620/35 code.
*/
static CARD32
PLLElectrical(RHDPtr rhdPtr, CARD16 FeedbackDivider)
{
switch (rhdPtr->ChipSet) {
case RHD_RV515:
if (rhdPtr->PciDeviceID == 0x7146)
return 0x00120704;
else
return 0;
case RHD_RV535:
if (rhdPtr->PciDeviceID == 0x71C1)
return 0x00230704;
else
return 0;
case RHD_RS600:
case RHD_RS690:
case RHD_RS740:
/* depending on MiscInfo also 0x00120004 */
return 0x00120704;
case RHD_R600:
return 0x01130704;
case RHD_RV610:
case RHD_RV630:
case RHD_M72:
case RHD_M74:
case RHD_M76:
return PLLControlTableRetrieve(RV610PLLControl, FeedbackDivider);
case RHD_RV670:
case RHD_R680:
return PLLControlTableRetrieve(RV670PLLControl, FeedbackDivider);
default:
return 0;
}
}
/*
* All R500s, RS6x0, R600, RV610 and RV630.
*/
/*
*
*/
static void
PLL1Calibrate(struct rhdPLL *PLL)
{
int i;
RHDFUNC(PLL);
RHDRegMask(PLL, P1PLL_CNTL, 1, 0x01); /* Reset */
usleep(2);
RHDRegMask(PLL, P1PLL_CNTL, 0, 0x01); /* Set */
for (i = 0; i < PLL_CALIBRATE_WAIT; i++)
if (((RHDRegRead(PLL, P1PLL_CNTL) >> 20) & 0x03) == 0x03)
break;
if (i == PLL_CALIBRATE_WAIT) {
if (RHDRegRead(PLL, P1PLL_CNTL) & 0x00100000) /* Calibration done? */
xf86DrvMsg(PLL->scrnIndex, X_ERROR,
"%s: Calibration failed.\n", __func__);
if (RHDRegRead(PLL, P1PLL_CNTL) & 0x00200000) /* PLL locked? */
xf86DrvMsg(PLL->scrnIndex, X_ERROR,
"%s: Locking failed.\n", __func__);
} else
RHDDebug(PLL->scrnIndex, "%s: lock in %d loops\n", __func__, i);
}
/*
*
*/
static void
PLL2Calibrate(struct rhdPLL *PLL)
{
int i;
RHDFUNC(PLL);
RHDRegMask(PLL, P2PLL_CNTL, 1, 0x01); /* Reset */
usleep(2);
RHDRegMask(PLL, P2PLL_CNTL, 0, 0x01); /* Set */
for (i = 0; i < PLL_CALIBRATE_WAIT; i++)
if (((RHDRegRead(PLL, P2PLL_CNTL) >> 20) & 0x03) == 0x03)
break;
if (i == PLL_CALIBRATE_WAIT) {
if (RHDRegRead(PLL, P2PLL_CNTL) & 0x00100000) /* Calibration done? */
xf86DrvMsg(PLL->scrnIndex, X_ERROR,
"%s: Calibration failed.\n", __func__);
if (RHDRegRead(PLL, P2PLL_CNTL) & 0x00200000) /* PLL locked? */
xf86DrvMsg(PLL->scrnIndex, X_ERROR,
"%s: Locking failed.\n", __func__);
} else
RHDDebug(PLL->scrnIndex, "%s: lock in %d loops\n", __func__, i);
}
/*
*
*/
static void
R500PLL1Power(struct rhdPLL *PLL, int Power)
{
RHDFUNC(PLL);
switch (Power) {
case RHD_POWER_ON:
RHDRegMask(PLL, P1PLL_CNTL, 0, 0x02); /* Powah */
usleep(2);
PLL1Calibrate(PLL);
return;
case RHD_POWER_RESET:
RHDRegMask(PLL, P1PLL_CNTL, 0x01, 0x01); /* Reset */
usleep(2);
RHDRegMask(PLL, P1PLL_CNTL, 0, 0x02); /* Powah */
usleep(2);
return;
case RHD_POWER_SHUTDOWN:
default:
RHDRegMask(PLL, P1PLL_CNTL, 0x01, 0x01); /* Reset */
usleep(2);
RHDRegMask(PLL, P1PLL_CNTL, 0x02, 0x02); /* Power down */
usleep(200);
return;
}
}
/*
*
*/
static void
R500PLL2Power(struct rhdPLL *PLL, int Power)
{
RHDFUNC(PLL);
switch (Power) {
case RHD_POWER_ON:
RHDRegMask(PLL, P2PLL_CNTL, 0, 0x02); /* Powah */
usleep(2);
PLL2Calibrate(PLL);
return;
case RHD_POWER_RESET:
RHDRegMask(PLL, P2PLL_CNTL, 0x01, 0x01); /* Reset */
usleep(2);
RHDRegMask(PLL, P2PLL_CNTL, 0, 0x02); /* Powah */
usleep(2);
return;
case RHD_POWER_SHUTDOWN:
default:
RHDRegMask(PLL, P2PLL_CNTL, 0x01, 0x01); /* Reset */
usleep(2);
RHDRegMask(PLL, P2PLL_CNTL, 0x02, 0x02); /* Power down */
usleep(200);
return;
}
}
/*
*
*/
static void
R500PLL1SetLow(struct rhdPLL *PLL, CARD32 RefDiv, CARD32 FBDiv, CARD32 PostDiv,
CARD32 Control)
{
RHDFUNC(PLL);
RHDRegWrite(PLL, EXT1_PPLL_REF_DIV_SRC, 0x01); /* XTAL */
RHDRegWrite(PLL, EXT1_PPLL_POST_DIV_SRC, 0x00); /* source = reference */
RHDRegWrite(PLL, EXT1_PPLL_UPDATE_LOCK, 0x01); /* lock */
RHDRegWrite(PLL, EXT1_PPLL_REF_DIV, RefDiv);
RHDRegWrite(PLL, EXT1_PPLL_FB_DIV, FBDiv);
RHDRegWrite(PLL, EXT1_PPLL_POST_DIV, PostDiv);
RHDRegWrite(PLL, EXT1_PPLL_CNTL, Control);
RHDRegMask(PLL, EXT1_PPLL_UPDATE_CNTL, 0x00010000, 0x00010000); /* no autoreset */
RHDRegMask(PLL, P1PLL_CNTL, 0, 0x04); /* don't bypass calibration */
/* We need to reset the anti glitch logic */
RHDRegMask(PLL, P1PLL_CNTL, 0, 0x00000002); /* power up */
/* reset anti glitch logic */
RHDRegMask(PLL, P1PLL_CNTL, 0x00002000, 0x00002000);
usleep(2);
RHDRegMask(PLL, P1PLL_CNTL, 0, 0x00002000);
/* powerdown and reset */
RHDRegMask(PLL, P1PLL_CNTL, 0x00000003, 0x00000003);
usleep(2);
RHDRegWrite(PLL, EXT1_PPLL_UPDATE_LOCK, 0); /* unlock */
RHDRegMask(PLL, EXT1_PPLL_UPDATE_CNTL, 0, 0x01); /* we're done updating! */
RHDRegMask(PLL, P1PLL_CNTL, 0, 0x02); /* Powah */
usleep(2);
PLL1Calibrate(PLL);
RHDRegWrite(PLL, EXT1_PPLL_POST_DIV_SRC, 0x01); /* source is PLL itself */
}
/*
*
*/
static void
R500PLL2SetLow(struct rhdPLL *PLL, CARD32 RefDiv, CARD32 FBDiv, CARD32 PostDiv,
CARD32 Control)
{
RHDRegWrite(PLL, EXT2_PPLL_REF_DIV_SRC, 0x01); /* XTAL */
RHDRegWrite(PLL, EXT2_PPLL_POST_DIV_SRC, 0x00); /* source = reference */
RHDRegWrite(PLL, EXT2_PPLL_UPDATE_LOCK, 0x01); /* lock */
RHDRegWrite(PLL, EXT2_PPLL_REF_DIV, RefDiv);
RHDRegWrite(PLL, EXT2_PPLL_FB_DIV, FBDiv);
RHDRegWrite(PLL, EXT2_PPLL_POST_DIV, PostDiv);
RHDRegWrite(PLL, EXT2_PPLL_CNTL, Control);
RHDRegMask(PLL, EXT2_PPLL_UPDATE_CNTL, 0x00010000, 0x00010000); /* no autoreset */
RHDRegMask(PLL, P2PLL_CNTL, 0, 0x04); /* don't bypass calibration */
/* We need to reset the anti glitch logic */
RHDRegMask(PLL, P2PLL_CNTL, 0, 0x00000002); /* power up */
/* reset anti glitch logic */
RHDRegMask(PLL, P2PLL_CNTL, 0x00002000, 0x00002000);
usleep(2);
RHDRegMask(PLL, P2PLL_CNTL, 0, 0x00002000);
/* powerdown and reset */
RHDRegMask(PLL, P2PLL_CNTL, 0x00000003, 0x00000003);
usleep(2);
RHDRegWrite(PLL, EXT2_PPLL_UPDATE_LOCK, 0); /* unlock */
RHDRegMask(PLL, EXT2_PPLL_UPDATE_CNTL, 0, 0x01); /* we're done updating! */
RHDRegMask(PLL, P2PLL_CNTL, 0, 0x02); /* Powah */
usleep(2);
PLL2Calibrate(PLL);
RHDRegWrite(PLL, EXT2_PPLL_POST_DIV_SRC, 0x01); /* source is PLL itself */
}
/*
* The CRTC ownership of each PLL is multiplexed on the PLL blocks, and the
* ownership can only be switched when the currently referenced PLL is active.
* This makes handling a slight bit more complex.
*/
static void
R500PLLCRTCGrab(struct rhdPLL *PLL, Bool Crtc2)
{
CARD32 Stored;
Bool PLL2IsCurrent;
if (!Crtc2) {
PLL2IsCurrent = RHDRegRead(PLL, PCLK_CRTC1_CNTL) & 0x00010000;
if (PLL->Id == PLL_ID_PLL1)
RHDRegMask(PLL, PCLK_CRTC1_CNTL, 0, 0x00010000);
else
RHDRegMask(PLL, PCLK_CRTC1_CNTL, 0x00010000, 0x00010000);
} else {
PLL2IsCurrent = RHDRegRead(PLL, PCLK_CRTC2_CNTL) & 0x00010000;
if (PLL->Id == PLL_ID_PLL1)
RHDRegMask(PLL, PCLK_CRTC2_CNTL, 0, 0x00010000);
else
RHDRegMask(PLL, PCLK_CRTC2_CNTL, 0x00010000, 0x00010000);
}
/* if the current pll is not active, then poke it just enough to flip
* owners */
if (!PLL2IsCurrent) {
Stored = RHDRegRead(PLL, P1PLL_CNTL);
if (Stored & 0x03) {
RHDRegMask(PLL, P1PLL_CNTL, 0, 0x03);
usleep(10);
RHDRegMask(PLL, P1PLL_CNTL, Stored, 0x03);
}
} else {
Stored = RHDRegRead(PLL, P2PLL_CNTL);
if (Stored & 0x03) {
RHDRegMask(PLL, P2PLL_CNTL, 0, 0x03);
usleep(10);
RHDRegMask(PLL, P2PLL_CNTL, Stored, 0x03);
}
}
}
/*
*
*/
static void
R500PLL1Set(struct rhdPLL *PLL, int PixelClock, CARD16 ReferenceDivider,
CARD16 FeedbackDivider, CARD8 PostDivider)
{
RHDPtr rhdPtr = RHDPTRI(PLL);
CARD32 RefDiv, FBDiv, PostDiv, Control;
RHDFUNC(PLL);
RefDiv = ReferenceDivider;
FBDiv = FeedbackDivider << 16;
if (rhdPtr->ChipSet > RHD_R600) { /* set up Feedbackdivider slip */
if (FeedbackDivider <= 0x24)
FBDiv |= 0x00000030;
else if (FeedbackDivider <= 0x3F)
FBDiv |= 0x00000020;
} else if (rhdPtr->ChipSet >= RHD_RS600) /* RS600, RS690, R600 */
FBDiv |= 0x00000030;
else
FBDiv |= RHDRegRead(PLL, EXT1_PPLL_FB_DIV) & 0x00000030;
PostDiv = RHDRegRead(PLL, EXT1_PPLL_POST_DIV) & ~0x0000007F;
PostDiv |= PostDivider & 0x0000007F;
Control = PLLElectrical(rhdPtr, FeedbackDivider);
if (!Control)
Control = RHDRegRead(PLL, EXT1_PPLL_CNTL);
/* Disable Spread Spectrum */
RHDRegMask(PLL, P1PLL_INT_SS_CNTL, 0, 0x00000001);
R500PLL1SetLow(PLL, RefDiv, FBDiv, PostDiv, Control);
if (rhdPtr->Crtc[0]->PLL == PLL)
R500PLLCRTCGrab(PLL, FALSE);
if (rhdPtr->Crtc[1]->PLL == PLL)
R500PLLCRTCGrab(PLL, TRUE);
}
/*
*
*/
static void
R500PLL2Set(struct rhdPLL *PLL, int PixelClock, CARD16 ReferenceDivider,
CARD16 FeedbackDivider, CARD8 PostDivider)
{
RHDPtr rhdPtr = RHDPTRI(PLL);
CARD32 RefDiv, FBDiv, PostDiv, Control;
RHDFUNC(PLL);
RefDiv = ReferenceDivider;
FBDiv = FeedbackDivider << 16;
if (rhdPtr->ChipSet > RHD_R600) { /* set up Feedbackdivider slip */
if (FeedbackDivider <= 0x24)
FBDiv |= 0x00000030;
else if (FeedbackDivider <= 0x3F)
FBDiv |= 0x00000020;
} else if (rhdPtr->ChipSet >= RHD_RS600) /* RS600, RS690, R600 */
FBDiv |= 0x00000030;
else
FBDiv |= RHDRegRead(PLL, EXT2_PPLL_FB_DIV) & 0x00000030;
PostDiv = RHDRegRead(PLL, EXT2_PPLL_POST_DIV) & ~0x0000007F;
PostDiv |= PostDivider & 0x0000007F;
Control = PLLElectrical(rhdPtr, FeedbackDivider);
if (!Control)
Control = RHDRegRead(PLL, EXT2_PPLL_CNTL);
/* Disable Spread Spectrum */
RHDRegMask(PLL, P2PLL_INT_SS_CNTL, 0, 0x00000001);
R500PLL2SetLow(PLL, RefDiv, FBDiv, PostDiv, Control);
if (rhdPtr->Crtc[0]->PLL == PLL)
R500PLLCRTCGrab(PLL, FALSE);
if (rhdPtr->Crtc[1]->PLL == PLL)
R500PLLCRTCGrab(PLL, TRUE);
}
/*
*
*/
static void
R500PLL1Save(struct rhdPLL *PLL)
{
RHDFUNC(PLL);
PLL->StoreActive = !(RHDRegRead(PLL, P1PLL_CNTL) & 0x03);
PLL->StoreRefDiv = RHDRegRead(PLL, EXT1_PPLL_REF_DIV);
PLL->StoreFBDiv = RHDRegRead(PLL, EXT1_PPLL_FB_DIV);
PLL->StorePostDiv = RHDRegRead(PLL, EXT1_PPLL_POST_DIV);
PLL->StoreControl = RHDRegRead(PLL, EXT1_PPLL_CNTL);
PLL->StoreSpreadSpectrum = RHDRegRead(PLL, P1PLL_INT_SS_CNTL);
PLL->StoreCrtc1Owner = !(RHDRegRead(PLL, PCLK_CRTC1_CNTL) & 0x00010000);
PLL->StoreCrtc2Owner = !(RHDRegRead(PLL, PCLK_CRTC2_CNTL) & 0x00010000);
PLL->Stored = TRUE;
}
/*
*
*/
static void
R500PLL2Save(struct rhdPLL *PLL)
{
RHDFUNC(PLL);
PLL->StoreActive = !(RHDRegRead(PLL, P2PLL_CNTL) & 0x03);
PLL->StoreRefDiv = RHDRegRead(PLL, EXT2_PPLL_REF_DIV);
PLL->StoreFBDiv = RHDRegRead(PLL, EXT2_PPLL_FB_DIV);
PLL->StorePostDiv = RHDRegRead(PLL, EXT2_PPLL_POST_DIV);
PLL->StoreControl = RHDRegRead(PLL, EXT2_PPLL_CNTL);
PLL->StoreSpreadSpectrum = RHDRegRead(PLL, P2PLL_INT_SS_CNTL);
PLL->StoreCrtc1Owner = RHDRegRead(PLL, PCLK_CRTC1_CNTL) & 0x00010000;
PLL->StoreCrtc2Owner = RHDRegRead(PLL, PCLK_CRTC2_CNTL) & 0x00010000;
PLL->Stored = TRUE;
}
/*
*
*/
static void
R500PLL1Restore(struct rhdPLL *PLL)
{
RHDFUNC(PLL);
if (!PLL->Stored) {
xf86DrvMsg(PLL->scrnIndex, X_ERROR, "%s: %s: trying to restore "
"uninitialized values.\n", __func__, PLL->Name);
return;
}
if (PLL->StoreActive) {
R500PLL1SetLow(PLL, PLL->StoreRefDiv, PLL->StoreFBDiv,
PLL->StorePostDiv, PLL->StoreControl);
/* HotFix: always keep spread spectrum disabled on restore */
if (0 && RHDPTRI(PLL)->ChipSet != RHD_M54)
RHDRegMask(PLL, P1PLL_INT_SS_CNTL,
PLL->StoreSpreadSpectrum, 0x00000001);
} else {
PLL->Power(PLL, RHD_POWER_SHUTDOWN);
/* lame attempt at at least restoring the old values */
RHDRegWrite(PLL, EXT1_PPLL_REF_DIV, PLL->StoreRefDiv);
RHDRegWrite(PLL, EXT1_PPLL_FB_DIV, PLL->StoreFBDiv);
RHDRegWrite(PLL, EXT1_PPLL_POST_DIV, PLL->StorePostDiv);
RHDRegWrite(PLL, EXT1_PPLL_CNTL, PLL->StoreControl);
RHDRegWrite(PLL, P1PLL_INT_SS_CNTL, PLL->StoreSpreadSpectrum);
}
if (PLL->StoreCrtc1Owner)
R500PLLCRTCGrab(PLL, FALSE);
if (PLL->StoreCrtc2Owner)
R500PLLCRTCGrab(PLL, TRUE);
}
/*
*
*/
static void
R500PLL2Restore(struct rhdPLL *PLL)
{
RHDFUNC(PLL);
if (!PLL->Stored) {
xf86DrvMsg(PLL->scrnIndex, X_ERROR, "%s: %s: trying to restore "
"uninitialized values.\n", __func__, PLL->Name);
return;
}
if (PLL->StoreActive) {
R500PLL2SetLow(PLL, PLL->StoreRefDiv, PLL->StoreFBDiv,
PLL->StorePostDiv, PLL->StoreControl);
if (RHDPTRI(PLL)->ChipSet != RHD_M54)
RHDRegMask(PLL, P2PLL_INT_SS_CNTL,
PLL->StoreSpreadSpectrum, 0x00000001);
} else {
PLL->Power(PLL, RHD_POWER_SHUTDOWN);
/* lame attempt at at least restoring the old values */
RHDRegWrite(PLL, EXT2_PPLL_REF_DIV, PLL->StoreRefDiv);
RHDRegWrite(PLL, EXT2_PPLL_FB_DIV, PLL->StoreFBDiv);
RHDRegWrite(PLL, EXT2_PPLL_POST_DIV, PLL->StorePostDiv);
RHDRegWrite(PLL, EXT2_PPLL_CNTL, PLL->StoreControl);
RHDRegWrite(PLL, P2PLL_INT_SS_CNTL, PLL->StoreSpreadSpectrum);
}
if (PLL->StoreCrtc1Owner)
R500PLLCRTCGrab(PLL, FALSE);
if (PLL->StoreCrtc2Owner)
R500PLLCRTCGrab(PLL, TRUE);
}
/*
* RV620 and up
*/
/*
*
*/
#define RV620_DCCGCLK_RESET 0
#define RV620_DCCGCLK_GRAB 1
#define RV620_DCCGCLK_RELEASE 2
/*
* I still have no idea what DCCG stands for and why it needs to hook off some
* pixelclock...
*/
static void
RV620DCCGCLKSet(struct rhdPLL *PLL, int set)
{
CARD32 tmp;
RHDFUNC(PLL);
switch(set) {
case RV620_DCCGCLK_GRAB:
if (PLL->Id == PLL_ID_PLL1)
RHDRegMask(PLL, DCCG_DISP_CLK_SRCSEL, 0, 0x00000003);
else if (PLL->Id == PLL_ID_PLL2)
RHDRegMask(PLL, DCCG_DISP_CLK_SRCSEL, 1, 0x00000003);
else
RHDRegMask(PLL, DCCG_DISP_CLK_SRCSEL, 3, 0x00000003);
break;
case RV620_DCCGCLK_RELEASE:
tmp = RHDRegRead(PLL, DCCG_DISP_CLK_SRCSEL) & 0x03;
if ((PLL->Id == PLL_ID_PLL1) && (tmp == 0)) {
/* set to other PLL or external */
tmp = RHDRegRead(PLL, P2PLL_CNTL);
if (!(tmp & 0x03) && /* powered and not in reset */
((tmp & 0x00300000) == 0x00300000)) /* calibrated and locked */
RHDRegMask(PLL, DCCG_DISP_CLK_SRCSEL, 1, 0x00000003);
else
RHDRegMask(PLL, DCCG_DISP_CLK_SRCSEL, 3, 0x00000003);
} else if ((PLL->Id == PLL_ID_PLL2) && (tmp == 1)) {
/* set to other PLL or external */
tmp = RHDRegRead(PLL, P1PLL_CNTL);
if (!(tmp & 0x03) && /* powered and not in reset */
((tmp & 0x00300000) == 0x00300000)) /* calibrated and locked */
RHDRegMask(PLL, DCCG_DISP_CLK_SRCSEL, 0, 0x00000003);
else
RHDRegMask(PLL, DCCG_DISP_CLK_SRCSEL, 3, 0x00000003);
} /* no other action needs to be taken */
break;
case RV620_DCCGCLK_RESET:
tmp = RHDRegRead(PLL, DCCG_DISP_CLK_SRCSEL) & 0x03;
if (((PLL->Id == PLL_ID_PLL1) && (tmp == 0)) ||
((PLL->Id == PLL_ID_PLL2) && (tmp == 1)))
RHDRegMask(PLL, DCCG_DISP_CLK_SRCSEL, 3, 0x00000003);
break;
default:
break;
}
}
/*
*
*/
static Bool
RV620DCCGCLKAvailable(struct rhdPLL *PLL)
{
CARD32 Dccg = RHDRegRead(PLL, DCCG_DISP_CLK_SRCSEL) & 0x03;
RHDFUNC(PLL);
if (Dccg & 0x02)
return TRUE;
if ((PLL->Id == PLL_ID_PLL1) && (Dccg == 0))
return TRUE;
if ((PLL->Id == PLL_ID_PLL2) && (Dccg == 1))
return TRUE;
return FALSE;
}
/*
*
*/
static void
RV620PLL1Power(struct rhdPLL *PLL, int Power)
{
RHDFUNC(PLL);
switch (Power) {
case RHD_POWER_ON:
{
Bool HasDccg = RV620DCCGCLKAvailable(PLL);
if (HasDccg)
RV620DCCGCLKSet(PLL, RV620_DCCGCLK_RESET);
RHDRegMask(PLL, P1PLL_CNTL, 0, 0x02); /* Powah */
usleep(2);
PLL1Calibrate(PLL);
if (HasDccg)
RV620DCCGCLKSet(PLL, RV620_DCCGCLK_GRAB);
return;
}
case RHD_POWER_RESET:
RV620DCCGCLKSet(PLL, RV620_DCCGCLK_RELEASE);
RHDRegMask(PLL, P1PLL_CNTL, 0x01, 0x01); /* Reset */
usleep(2);
RHDRegMask(PLL, P1PLL_CNTL, 0, 0x02); /* Powah */
usleep(2);
return;
case RHD_POWER_SHUTDOWN:
default:
RV620DCCGCLKSet(PLL, RV620_DCCGCLK_RELEASE);
RHDRegMask(PLL, P1PLL_CNTL, 0x01, 0x01); /* Reset */
usleep(2);
RHDRegMask(PLL, P1PLL_CNTL, 0x02, 0x02); /* Power down */
usleep(200);
return;
}
}
/*
*
*/
static void
RV620PLL2Power(struct rhdPLL *PLL, int Power)
{
RHDFUNC(PLL);
switch (Power) {
case RHD_POWER_ON:
{
Bool HasDccg = RV620DCCGCLKAvailable(PLL);
if (HasDccg)
RV620DCCGCLKSet(PLL, RV620_DCCGCLK_RESET);
RHDRegMask(PLL, P2PLL_CNTL, 0, 0x02); /* Powah */
usleep(2);
PLL2Calibrate(PLL);
if (HasDccg)
RV620DCCGCLKSet(PLL, RV620_DCCGCLK_GRAB);
return;
}
case RHD_POWER_RESET:
RV620DCCGCLKSet(PLL, RV620_DCCGCLK_RELEASE);
RHDRegMask(PLL, P2PLL_CNTL, 0x01, 0x01); /* Reset */
usleep(2);
RHDRegMask(PLL, P2PLL_CNTL, 0, 0x02); /* Powah */
usleep(2);
return;
case RHD_POWER_SHUTDOWN:
default:
RV620DCCGCLKSet(PLL, RV620_DCCGCLK_RELEASE);
RHDRegMask(PLL, P2PLL_CNTL, 0x01, 0x01); /* Reset */
usleep(2);
RHDRegMask(PLL, P2PLL_CNTL, 0x02, 0x02); /* Power down */
usleep(200);
return;
}
}
/*
*
*/
static void
RV620PLL1SetLow(struct rhdPLL *PLL, CARD32 RefDiv, CARD32 FBDiv, CARD32 PostDiv,
CARD8 ScalerDiv, CARD8 SymPostDiv, CARD32 Control)
{
RHDFUNC(PLL);
/* switch to external */
RHDRegWrite(PLL, EXT1_PPLL_POST_DIV_SRC, 0);
RHDRegMask(PLL, P1PLL_DISP_CLK_CNTL, 0x00000200, 0x00000300);
RHDRegMask(PLL, EXT1_SYM_PPLL_POST_DIV, 0, 0x00000100);
RHDRegMask(PLL, P1PLL_CNTL, 0x00000001, 0x00000001); /* reset */
usleep(2);
RHDRegMask(PLL, P1PLL_CNTL, 0x00000002, 0x00000002); /* power down */
usleep(10);
RHDRegMask(PLL, P1PLL_CNTL, 0x00002000, 0x00002000); /* reset anti-glitch */
RHDRegWrite(PLL, EXT1_PPLL_CNTL, Control);
RHDRegMask(PLL, P1PLL_DISP_CLK_CNTL, ScalerDiv, 0x0000003F);
RHDRegWrite(PLL, EXT1_PPLL_UPDATE_LOCK, 1); /* lock */
RHDRegWrite(PLL, EXT1_PPLL_POST_DIV_SRC, 0x00000001);
RHDRegWrite(PLL, EXT1_PPLL_REF_DIV, RefDiv);
RHDRegWrite(PLL, EXT1_PPLL_FB_DIV, FBDiv);
RHDRegMask(PLL, EXT1_PPLL_POST_DIV, PostDiv, 0x0000007F);
RHDRegMask(PLL, EXT1_SYM_PPLL_POST_DIV, SymPostDiv, 0x0000007F);
usleep(10);
RHDRegWrite(PLL, EXT1_PPLL_UPDATE_LOCK, 0); /* unlock */
RHDRegMask(PLL, P1PLL_CNTL, 0, 0x00000002); /* power up */
usleep(10);
RHDRegMask(PLL, P1PLL_CNTL, 0, 0x00002000); /* undo reset anti-glitch */
PLL1Calibrate(PLL);
/* switch back to the pll */
RHDRegMask(PLL, P1PLL_DISP_CLK_CNTL, 0, 0x00000300);
RHDRegMask(PLL, EXT1_SYM_PPLL_POST_DIV, 0x00000100, 0x00000100);
RHDRegWrite(PLL, EXT1_PPLL_POST_DIV_SRC, 0x00000001);
RHDRegMask(PLL, P1PLL_CNTL, 0, 0x80000000); /* new and undocumented */
}
/*
*
*/
static void
RV620PLL2SetLow(struct rhdPLL *PLL, CARD32 RefDiv, CARD32 FBDiv, CARD32 PostDiv,
CARD8 ScalerDiv, CARD8 SymPostDiv, CARD32 Control)
{
RHDFUNC(PLL);
/* switch to external */
RHDRegWrite(PLL, EXT2_PPLL_POST_DIV_SRC, 0);
RHDRegMask(PLL, P2PLL_DISP_CLK_CNTL, 0x00000200, 0x00000300);
RHDRegMask(PLL, EXT2_SYM_PPLL_POST_DIV, 0, 0x00000100);
RHDRegMask(PLL, P2PLL_CNTL, 0x00000001, 0x00000001); /* reset */
usleep(2);
RHDRegMask(PLL, P2PLL_CNTL, 0x00000002, 0x00000002); /* power down */
usleep(10);
RHDRegMask(PLL, P2PLL_CNTL, 0x00002000, 0x00002000); /* reset anti-glitch */
RHDRegWrite(PLL, EXT2_PPLL_CNTL, Control);
RHDRegMask(PLL, P2PLL_DISP_CLK_CNTL, ScalerDiv, 0x0000003F);
RHDRegWrite(PLL, EXT2_PPLL_UPDATE_LOCK, 1); /* lock */
RHDRegWrite(PLL, EXT2_PPLL_POST_DIV_SRC, 0x00000001);
RHDRegWrite(PLL, EXT2_PPLL_REF_DIV, RefDiv);
RHDRegWrite(PLL, EXT2_PPLL_FB_DIV, FBDiv);
RHDRegMask(PLL, EXT2_PPLL_POST_DIV, PostDiv, 0x0000007F);
RHDRegMask(PLL, EXT2_SYM_PPLL_POST_DIV, SymPostDiv, 0x0000007F);
usleep(10);
RHDRegWrite(PLL, EXT2_PPLL_UPDATE_LOCK, 0); /* unlock */
RHDRegMask(PLL, P2PLL_CNTL, 0, 0x00000002); /* power up */
usleep(10);
RHDRegMask(PLL, P2PLL_CNTL, 0, 0x00002000); /* undo reset anti-glitch */
PLL2Calibrate(PLL);
/* switch back to the pll */
RHDRegMask(PLL, P2PLL_DISP_CLK_CNTL, 0, 0x00000300);
RHDRegMask(PLL, EXT2_SYM_PPLL_POST_DIV, 0x00000100, 0x00000100);
RHDRegWrite(PLL, EXT2_PPLL_POST_DIV_SRC, 0x00000001);
RHDRegMask(PLL, P2PLL_CNTL, 0, 0x80000000); /* new and undocumented */
}
/*
*
*/
static void
RV620PLL1Set(struct rhdPLL *PLL, int PixelClock, CARD16 ReferenceDivider,
CARD16 FeedbackDivider, CARD8 PostDivider)
{
RHDPtr rhdPtr = RHDPTRI(PLL);
Bool HasDccg = RV620DCCGCLKAvailable(PLL);
CARD32 RefDiv, FBDiv, PostDiv, Control;
CARD8 ScalerDiv, SymPostDiv;
RHDFUNC(PLL);
if (HasDccg)
RV620DCCGCLKSet(PLL, RV620_DCCGCLK_RESET);
/* Disable Spread Spectrum */
RHDRegMask(PLL, P1PLL_INT_SS_CNTL, 0, 0x00000001);
RefDiv = ReferenceDivider;
FBDiv = RHDRegRead(PLL, EXT1_PPLL_FB_DIV) & ~0x07FF003F;
FBDiv |= ((FeedbackDivider << 16) | 0x0030) & 0x07FF003F;
PostDiv = RHDRegRead(PLL, EXT1_PPLL_POST_DIV) & ~0x0000007F;
PostDiv |= PostDivider & 0x0000007F;
/* introduce flags for this, like on unichrome */
ScalerDiv = 2; /* scaler post divider, 4 for UPDP */
SymPostDiv = PostDivider & 0x0000007F;
Control = PLLControlTableRetrieve(RV670PLLControl, FeedbackDivider);
RV620PLL1SetLow(PLL, RefDiv, FBDiv, PostDiv, ScalerDiv, SymPostDiv,
Control);
if (rhdPtr->Crtc[0]->PLL == PLL)
R500PLLCRTCGrab(PLL, FALSE);
if (rhdPtr->Crtc[1]->PLL == PLL)
R500PLLCRTCGrab(PLL, TRUE);
if (HasDccg)
RV620DCCGCLKSet(PLL, RV620_DCCGCLK_GRAB);
}
/*
*
*/
static void
RV620PLL2Set(struct rhdPLL *PLL, int PixelClock, CARD16 ReferenceDivider,
CARD16 FeedbackDivider, CARD8 PostDivider)
{
RHDPtr rhdPtr = RHDPTRI(PLL);
Bool HasDccg = RV620DCCGCLKAvailable(PLL);
CARD32 RefDiv, FBDiv, PostDiv, Control;
CARD8 ScalerDiv, SymPostDiv;
RHDFUNC(PLL);
if (HasDccg)
RV620DCCGCLKSet(PLL, RV620_DCCGCLK_RESET);
/* Disable Spread Spectrum */
RHDRegMask(PLL, P2PLL_INT_SS_CNTL, 0, 0x00000001);
RefDiv = ReferenceDivider;
FBDiv = RHDRegRead(PLL, EXT2_PPLL_FB_DIV) & ~0x07FF003F;
FBDiv |= ((FeedbackDivider << 16) | 0x0030) & 0x07FF003F;
PostDiv = RHDRegRead(PLL, EXT2_PPLL_POST_DIV) & ~0x0000007F;
PostDiv |= PostDivider & 0x0000007F;
/* introduce flags for this, like on unichrome */
ScalerDiv = 2; /* scaler post divider, 4 for UPDP */
SymPostDiv = PostDivider & 0x0000007F;
Control = PLLControlTableRetrieve(RV670PLLControl, FeedbackDivider);
RV620PLL2SetLow(PLL, RefDiv, FBDiv, PostDiv, ScalerDiv, SymPostDiv,
Control);
if (rhdPtr->Crtc[0]->PLL == PLL)
R500PLLCRTCGrab(PLL, FALSE);
if (rhdPtr->Crtc[1]->PLL == PLL)
R500PLLCRTCGrab(PLL, TRUE);
if (HasDccg)
RV620DCCGCLKSet(PLL, RV620_DCCGCLK_GRAB);
}
/*
*
*/
static void
RV620PLL1Save(struct rhdPLL *PLL)
{
RHDFUNC(PLL);
PLL->StoreActive = !(RHDRegRead(PLL, P1PLL_CNTL) & 0x03);
PLL->StoreRefDiv = RHDRegRead(PLL, EXT1_PPLL_REF_DIV);
PLL->StoreFBDiv = RHDRegRead(PLL, EXT1_PPLL_FB_DIV);
PLL->StorePostDiv = RHDRegRead(PLL, EXT1_PPLL_POST_DIV);
PLL->StorePostDivSrc = RHDRegRead(PLL, EXT1_PPLL_POST_DIV_SRC);
PLL->StoreControl = RHDRegRead(PLL, EXT1_PPLL_CNTL);
PLL->StoreSpreadSpectrum = RHDRegRead(PLL, P1PLL_INT_SS_CNTL);
PLL->StoreGlitchReset = RHDRegRead(PLL, P1PLL_CNTL) & 0x00002000;
PLL->StoreScalerPostDiv = RHDRegRead(PLL, P1PLL_DISP_CLK_CNTL) & 0x003F;
PLL->StoreSymPostDiv = RHDRegRead(PLL, EXT1_SYM_PPLL_POST_DIV) & 0x007F;
PLL->StoreCrtc1Owner = !(RHDRegRead(PLL, PCLK_CRTC1_CNTL) & 0x00010000);
PLL->StoreCrtc2Owner = !(RHDRegRead(PLL, PCLK_CRTC2_CNTL) & 0x00010000);
PLL->StoreDCCGCLKOwner = RV620DCCGCLKAvailable(PLL);
if (PLL->StoreDCCGCLKOwner)
PLL->StoreDCCGCLK = RHDRegRead(PLL, DCCG_DISP_CLK_SRCSEL);
else
PLL->StoreDCCGCLK = 0;
PLL->Stored = TRUE;
}
/*
*
*/
static void
RV620PLL2Save(struct rhdPLL *PLL)
{
RHDFUNC(PLL);
PLL->StoreActive = !(RHDRegRead(PLL, P2PLL_CNTL) & 0x03);
PLL->StoreRefDiv = RHDRegRead(PLL, EXT2_PPLL_REF_DIV);
PLL->StoreFBDiv = RHDRegRead(PLL, EXT2_PPLL_FB_DIV);
PLL->StorePostDiv = RHDRegRead(PLL, EXT2_PPLL_POST_DIV);
PLL->StorePostDivSrc = RHDRegRead(PLL, EXT2_PPLL_POST_DIV_SRC);
PLL->StoreControl = RHDRegRead(PLL, EXT2_PPLL_CNTL);
PLL->StoreSpreadSpectrum = RHDRegRead(PLL, P2PLL_INT_SS_CNTL);
PLL->StoreGlitchReset = RHDRegRead(PLL, P2PLL_CNTL) & 0x00002000;
PLL->StoreScalerPostDiv = RHDRegRead(PLL, P2PLL_DISP_CLK_CNTL) & 0x003F;
PLL->StoreSymPostDiv = RHDRegRead(PLL, EXT2_SYM_PPLL_POST_DIV) & 0x007F;
PLL->StoreCrtc1Owner = RHDRegRead(PLL, PCLK_CRTC1_CNTL) & 0x00010000;
PLL->StoreCrtc2Owner = RHDRegRead(PLL, PCLK_CRTC2_CNTL) & 0x00010000;
PLL->StoreDCCGCLKOwner = RV620DCCGCLKAvailable(PLL);
if (PLL->StoreDCCGCLKOwner)
PLL->StoreDCCGCLK = RHDRegRead(PLL, DCCG_DISP_CLK_SRCSEL);
else
PLL->StoreDCCGCLK = 0;
PLL->Stored = TRUE;
}
/*
* Notice how we handle the DCCG ownership here. There is a difference between
* currently holding the DCCG and what was held when in the VT. With the
* solution here we no longer hardlock, but we do have the danger of keeping
* the DCCG in external mode for too long a time, if both PLL restores are
* too far apart. This is currently not an issue as VT restoration goes over
* the whole device in one go anyway; no partial restoration going on
*/
static void
RV620PLL1Restore(struct rhdPLL *PLL)
{
RHDFUNC(PLL);
if (RV620DCCGCLKAvailable(PLL))
RHDRegMask(PLL, DCCG_DISP_CLK_SRCSEL, 0x03, 0x00000003);
if (PLL->StoreActive) {
RV620PLL1SetLow(PLL, PLL->StoreRefDiv, PLL->StoreFBDiv,
PLL->StorePostDiv, PLL->StoreScalerPostDiv,
PLL->StoreSymPostDiv, PLL->StoreControl);
RHDRegMask(PLL, P1PLL_INT_SS_CNTL,
PLL->StoreSpreadSpectrum, 0x00000001);
if (PLL->StoreDCCGCLKOwner)
RHDRegWrite(PLL, DCCG_DISP_CLK_SRCSEL, PLL->StoreDCCGCLK);
} else {
PLL->Power(PLL, RHD_POWER_SHUTDOWN);
/* lame attempt at at least restoring the old values */
RHDRegWrite(PLL, EXT1_PPLL_REF_DIV, PLL->StoreRefDiv);
RHDRegWrite(PLL, EXT1_PPLL_FB_DIV, PLL->StoreFBDiv);
RHDRegWrite(PLL, EXT1_PPLL_POST_DIV, PLL->StorePostDiv);
RHDRegWrite(PLL, EXT1_PPLL_POST_DIV_SRC, PLL->StorePostDivSrc);
RHDRegWrite(PLL, EXT1_PPLL_CNTL, PLL->StoreControl);
RHDRegMask(PLL, P1PLL_DISP_CLK_CNTL, PLL->StoreScalerPostDiv, 0x003F);
RHDRegMask(PLL, EXT1_SYM_PPLL_POST_DIV, PLL->StoreSymPostDiv, 0x007F);
RHDRegWrite(PLL, P1PLL_INT_SS_CNTL, PLL->StoreSpreadSpectrum);
if (PLL->StoreGlitchReset)
RHDRegMask(PLL, P1PLL_CNTL, 0x00002000, 0x00002000);
else
RHDRegMask(PLL, P1PLL_CNTL, 0, 0x00002000);
}
if (PLL->StoreCrtc1Owner)
R500PLLCRTCGrab(PLL, FALSE);
if (PLL->StoreCrtc2Owner)
R500PLLCRTCGrab(PLL, TRUE);
if (PLL->StoreDCCGCLKOwner)
RHDRegWrite(PLL, DCCG_DISP_CLK_SRCSEL, PLL->StoreDCCGCLK);
}
/*
*
*/
static void
RV620PLL2Restore(struct rhdPLL *PLL)
{
RHDFUNC(PLL);
if (RV620DCCGCLKAvailable(PLL))
RHDRegMask(PLL, DCCG_DISP_CLK_SRCSEL, 0x03, 0x00000003);
if (PLL->StoreActive) {
RV620PLL2SetLow(PLL, PLL->StoreRefDiv, PLL->StoreFBDiv,
PLL->StorePostDiv, PLL->StoreScalerPostDiv,
PLL->StoreSymPostDiv, PLL->StoreControl);
RHDRegMask(PLL, P2PLL_INT_SS_CNTL,
PLL->StoreSpreadSpectrum, 0x00000001);
} else {
PLL->Power(PLL, RHD_POWER_SHUTDOWN);
/* lame attempt at at least restoring the old values */
RHDRegWrite(PLL, EXT2_PPLL_REF_DIV, PLL->StoreRefDiv);
RHDRegWrite(PLL, EXT2_PPLL_FB_DIV, PLL->StoreFBDiv);
RHDRegWrite(PLL, EXT2_PPLL_POST_DIV, PLL->StorePostDiv);
RHDRegWrite(PLL, EXT2_PPLL_POST_DIV_SRC, PLL->StorePostDivSrc);
RHDRegWrite(PLL, EXT2_PPLL_CNTL, PLL->StoreControl);
RHDRegMask(PLL, P2PLL_DISP_CLK_CNTL, PLL->StoreScalerPostDiv, 0x003F);
RHDRegMask(PLL, EXT2_SYM_PPLL_POST_DIV, PLL->StoreSymPostDiv, 0x007F);
RHDRegWrite(PLL, P2PLL_INT_SS_CNTL, PLL->StoreSpreadSpectrum);
if (PLL->StoreGlitchReset)
RHDRegMask(PLL, P2PLL_CNTL, 0x00002000, 0x00002000);
else
RHDRegMask(PLL, P2PLL_CNTL, 0, 0x00002000);
}
if (PLL->StoreCrtc1Owner)
R500PLLCRTCGrab(PLL, FALSE);
if (PLL->StoreCrtc2Owner)
R500PLLCRTCGrab(PLL, TRUE);
if (PLL->StoreDCCGCLKOwner)
RHDRegWrite(PLL, DCCG_DISP_CLK_SRCSEL, PLL->StoreDCCGCLK);
}
/* Some defaults for when we don't have this info */
/* XTAL is visible on the cards */
#define RHD_PLL_REFERENCE_DEFAULT 27000
/* these required quite some testing */
#define RHD_R500_PLL_INTERNAL_MIN_DEFAULT 648000
#define RHD_RV620_PLL_INTERNAL_MIN_DEFAULT 702000
/* Lowest value seen so far */
#define RHD_PLL_INTERNAL_MAX_DEFAULT 1100000
#define RHD_PLL_MIN_DEFAULT 16000 /* guess */
#define RHD_PLL_MAX_DEFAULT 400000 /* 400Mhz modes... hrm */
enum pllComp {
PLL_NONE,
PLL_MIN,
PLL_MAX
};
/*
*
*/
#ifdef ATOM_BIOS
static Bool
getPLLValuesFromAtomBIOS(RHDPtr rhdPtr,
AtomBiosRequestID func, char *msg, CARD32 *val, enum pllComp comp)
{
AtomBiosArgRec arg;
AtomBiosResult ret;
if (rhdPtr->atomBIOS) {
ret = RHDAtomBiosFunc(rhdPtr->scrnIndex, rhdPtr->atomBIOS,
func, &arg);
if (ret == ATOM_SUCCESS) {
if (arg.val) {
switch (comp) {
case PLL_MAX:
if (arg.val < *val)
xf86DrvMsg(rhdPtr->scrnIndex, X_WARNING,
"Lower %s detected than the default: %lu %lu.\n"
"Please contact the authors ASAP.\n", msg,
(unsigned long)*val, (unsigned long)arg.val * 10);
break;
case PLL_MIN:
if (arg.val > *val)
xf86DrvMsg(rhdPtr->scrnIndex, X_WARNING,
"Higher %s detected than the default: %lu %lu.\n"
"Please contact the authors ASAP.\n", msg,
(unsigned long)*val, (unsigned long)arg.val * 10);
break;
default:
break;
}
*val = arg.val;
}
}
return TRUE;
} else
xf86DrvMsg(rhdPtr->scrnIndex, X_ERROR, "Failed to retrieve the %s"
" clock from ATOM.\n",msg);
return FALSE;
}
#endif
/*
*
*/
void
RHDSetupLimits(RHDPtr rhdPtr, CARD32 *RefClock,
CARD32 *IntMin, CARD32 *IntMax,
CARD32 *PixMin, CARD32 *PixMax)
{
/* Retrieve the internal PLL frequency limits*/
*RefClock = RHD_PLL_REFERENCE_DEFAULT;
if (rhdPtr->ChipSet < RHD_RV620)
*IntMin = RHD_R500_PLL_INTERNAL_MIN_DEFAULT;
else
*IntMin = RHD_RV620_PLL_INTERNAL_MIN_DEFAULT;
*IntMax = RHD_PLL_INTERNAL_MAX_DEFAULT;
/* keep the defaults */
*PixMin = RHD_PLL_MIN_DEFAULT;
*PixMax = RHD_PLL_MAX_DEFAULT;
#ifdef ATOM_BIOS
getPLLValuesFromAtomBIOS(rhdPtr, GET_MIN_PIXEL_CLOCK_PLL_OUTPUT, "minimum PLL output",
IntMin, PLL_MIN);
getPLLValuesFromAtomBIOS(rhdPtr, GET_MAX_PIXEL_CLOCK_PLL_OUTPUT, "maximum PLL output",
IntMax, PLL_MAX);
getPLLValuesFromAtomBIOS(rhdPtr, GET_MAX_PIXEL_CLK, "Pixel Clock",
PixMax, PLL_MAX);
getPLLValuesFromAtomBIOS(rhdPtr, GET_REF_CLOCK, "reference clock",
RefClock, PLL_NONE);
if (*IntMax == 0) {
if (rhdPtr->ChipSet < RHD_RV620)
*IntMax = RHD_R500_PLL_INTERNAL_MIN_DEFAULT;
else
*IntMax = RHD_RV620_PLL_INTERNAL_MIN_DEFAULT;
xf86DrvMsg(rhdPtr->scrnIndex, X_WARNING, "AtomBIOS reports maximum VCO freq 0. "
"Using %lu instead\n",(unsigned long)*IntMax);
}
#endif
}
/*
*
*/
Bool
RHDPLLsInit(RHDPtr rhdPtr)
{
struct rhdPLL *PLL;
CARD32 RefClock, IntMin, IntMax, PixMin, PixMax;
RHDFUNC(rhdPtr);
if (RHDUseAtom(rhdPtr, NULL, atomUsagePLL))
return FALSE;
RHDSetupLimits(rhdPtr, &RefClock, &IntMin, &IntMax, &PixMin, &PixMax);
/* PLL1 */
PLL = (struct rhdPLL *) xnfcalloc(sizeof(struct rhdPLL), 1);
PLL->scrnIndex = rhdPtr->scrnIndex;
PLL->Name = PLL_NAME_PLL1;
PLL->Id = PLL_ID_PLL1;
PLL->RefClock = RefClock;
PLL->IntMin = IntMin;
PLL->IntMax = IntMax;
PLL->PixMin = PixMin;
PLL->PixMax = PixMax;
PLL->Valid = NULL;
if (rhdPtr->ChipSet < RHD_RV620) {
PLL->Set = R500PLL1Set;
PLL->Power = R500PLL1Power;
PLL->Save = R500PLL1Save;
PLL->Restore = R500PLL1Restore;
} else {
PLL->Set = RV620PLL1Set;
PLL->Power = RV620PLL1Power;
PLL->Save = RV620PLL1Save;
PLL->Restore = RV620PLL1Restore;
}
rhdPtr->PLLs[0] = PLL;
/* PLL2 */
PLL = (struct rhdPLL *) xnfcalloc(sizeof(struct rhdPLL), 1);
PLL->scrnIndex = rhdPtr->scrnIndex;
PLL->Name = PLL_NAME_PLL2;
PLL->Id = PLL_ID_PLL2;
PLL->RefClock = RefClock;
PLL->IntMin = IntMin;
PLL->IntMax = IntMax;
PLL->PixMin = PixMin;
PLL->PixMax = PixMax;
PLL->Valid = NULL;
if (rhdPtr->ChipSet < RHD_RV620) {
PLL->Set = R500PLL2Set;
PLL->Power = R500PLL2Power;
PLL->Save = R500PLL2Save;
PLL->Restore = R500PLL2Restore;
} else {
PLL->Set = RV620PLL2Set;
PLL->Power = RV620PLL2Power;
PLL->Save = RV620PLL2Save;
PLL->Restore = RV620PLL2Restore;
}
rhdPtr->PLLs[1] = PLL;
return TRUE;
}
/*
*
*/
ModeStatus
RHDPLLValid(struct rhdPLL *PLL, CARD32 Clock)
{
RHDFUNC(PLL);
if (Clock < PLL->PixMin)
return MODE_CLOCK_LOW;
if (Clock > PLL->PixMax)
return MODE_CLOCK_HIGH;
if (PLL->Valid)
return PLL->Valid(PLL, Clock);
else
return MODE_OK;
}
/*
* Calculate the PLL parameters for a given dotclock.
*
* This calculation uses a linear approximation of an experimentally found
* curve that delimits reference versus feedback dividers on rv610. This curve
* can be shifted towards higher feedback divider through increasing the gain
* control, but the effect of this is rather limited.
*
* Since this upper limit still provides a wide enough range with enough
* granularity, we use it for all r5xx and r6xx devices.
*/
static Bool
PLLCalculate(struct rhdPLL *PLL, CARD32 PixelClock,
CARD16 *RefDivider, CARD16 *FBDivider, CARD8 *PostDivider)
{
/* limited by the number of bits available */
#define FB_DIV_LIMIT 2048
#define REF_DIV_LIMIT 1024
#define POST_DIV_LIMIT 128
CARD32 FBDiv, RefDiv, PostDiv, BestDiff = 0xFFFFFFFF;
float Ratio;
Ratio = ((float) PixelClock) / ((float) PLL->RefClock);
for (PostDiv = 2; PostDiv < POST_DIV_LIMIT; PostDiv++) {
CARD32 VCOOut = PixelClock * PostDiv;
/* we are conservative and avoid the limits */
if (VCOOut <= PLL->IntMin)
continue;
if (VCOOut >= PLL->IntMax)
break;
for (RefDiv = 1; RefDiv <= REF_DIV_LIMIT; RefDiv++) {
CARD32 Diff;
FBDiv = (CARD32) ((Ratio * PostDiv * RefDiv) + 0.5);
if (FBDiv >= FB_DIV_LIMIT)
break;
if (FBDiv > (500 + (13 * RefDiv))) /* rv6x0 limit */
break;
Diff = abs( PixelClock - (FBDiv * PLL->RefClock) / (PostDiv * RefDiv) );
if (Diff < BestDiff) {
*FBDivider = FBDiv;
*RefDivider = RefDiv;
*PostDivider = PostDiv;
BestDiff = Diff;
}
if (BestDiff == 0)
break;
}
if (BestDiff == 0)
break;
}
if (BestDiff != 0xFFFFFFFF) {
RHDDebug(PLL->scrnIndex, "PLL Calculation: %dkHz = "
"(((%i / 0x%X) * 0x%X) / 0x%X) (%dkHz off)\n",
(int) PixelClock, (unsigned int) PLL->RefClock, *RefDivider,
*FBDivider, *PostDivider, (int) BestDiff);
return TRUE;
} else { /* Should never happen */
xf86DrvMsg(PLL->scrnIndex, X_ERROR,
"%s: Failed to get a valid PLL setting for %dkHz\n",
__func__, (int) PixelClock);
return FALSE;
}
}
/*
*
*/
void
RHDPLLSet(struct rhdPLL *PLL, CARD32 Clock)
{
CARD16 RefDivider = 0, FBDivider = 0;
CARD8 PostDivider = 0;
RHDDebug(PLL->scrnIndex, "%s: Setting %s to %dkHz\n", __func__,
PLL->Name, Clock);
if (PLLCalculate(PLL, Clock, &RefDivider, &FBDivider, &PostDivider)) {
PLL->Set(PLL, Clock, RefDivider, FBDivider, PostDivider);
PLL->CurrentClock = Clock;
PLL->Active = TRUE;
} else
xf86DrvMsg(PLL->scrnIndex, X_WARNING,
"%s: Not altering any settings.\n", __func__);
}
/*
*
*/
void
RHDPLLPower(struct rhdPLL *PLL, int Power)
{
RHDFUNC(PLL);
if (PLL->Power)
PLL->Power(PLL, Power);
}
/*
*
*/
void
RHDPLLsPowerAll(RHDPtr rhdPtr, int Power)
{
struct rhdPLL *PLL;
RHDFUNC(rhdPtr);
PLL = rhdPtr->PLLs[0];
if (PLL->Power)
PLL->Power(PLL, Power);
PLL = rhdPtr->PLLs[1];
if (PLL->Power)
PLL->Power(PLL, Power);
}
/*
*
*/
void
RHDPLLsShutdownInactive(RHDPtr rhdPtr)
{
struct rhdPLL *PLL;
RHDFUNC(rhdPtr);
PLL = rhdPtr->PLLs[0];
if (PLL->Power && !PLL->Active)
PLL->Power(PLL, RHD_POWER_SHUTDOWN);
PLL = rhdPtr->PLLs[1];
if (PLL->Power && !PLL->Active)
PLL->Power(PLL, RHD_POWER_SHUTDOWN);
}
/*
*
*/
void
RHDPLLsSave(RHDPtr rhdPtr)
{
struct rhdPLL *PLL;
RHDFUNC(rhdPtr);
PLL = rhdPtr->PLLs[0];
if (PLL->Save)
PLL->Save(PLL);
PLL = rhdPtr->PLLs[1];
if (PLL->Save)
PLL->Save(PLL);
}
/*
*
*/
void
RHDPLLsRestore(RHDPtr rhdPtr)
{
struct rhdPLL *PLL;
RHDFUNC(rhdPtr);
PLL = rhdPtr->PLLs[0];
if (PLL->Restore)
PLL->Restore(PLL);
PLL = rhdPtr->PLLs[1];
if (PLL->Restore)
PLL->Restore(PLL);
}
/*
*
*/
void
RHDPLLsDestroy(RHDPtr rhdPtr)
{
RHDFUNC(rhdPtr);
if (rhdPtr->PLLs[0] && rhdPtr->PLLs[0]->Private)
xfree(rhdPtr->PLLs[0]->Private);
xfree(rhdPtr->PLLs[0]);
if (rhdPtr->PLLs[1] && rhdPtr->PLLs[1]->Private)
xfree(rhdPtr->PLLs[1]->Private);
xfree(rhdPtr->PLLs[1]);
}