2005-04-17 02:20:36 +04:00
/*
2006-06-05 23:25:20 +04:00
* ( C ) 2004 - 2006 Sebastian Witt < se . witt @ gmx . net >
2005-04-17 02:20:36 +04:00
*
* Licensed under the terms of the GNU GPL License version 2.
* Based upon reverse engineered information
*
* BIG FAT DISCLAIMER : Work in progress code . Possibly * dangerous *
*/
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/moduleparam.h>
# include <linux/init.h>
# include <linux/cpufreq.h>
# include <linux/pci.h>
# include <linux/delay.h>
# define NFORCE2_XTAL 25
# define NFORCE2_BOOTFSB 0x48
# define NFORCE2_PLLENABLE 0xa8
# define NFORCE2_PLLREG 0xa4
# define NFORCE2_PLLADR 0xa0
# define NFORCE2_PLL(mul, div) (0x100000 | (mul << 8) | div)
# define NFORCE2_MIN_FSB 50
# define NFORCE2_SAFE_DISTANCE 50
/* Delay in ms between FSB changes */
2008-06-14 23:11:39 +04:00
/* #define NFORCE2_DELAY 10 */
2005-04-17 02:20:36 +04:00
2008-06-14 23:11:39 +04:00
/*
* nforce2_chipset :
2005-04-17 02:20:36 +04:00
* FSB is changed using the chipset
*/
2009-01-18 06:39:47 +03:00
static struct pci_dev * nforce2_dev ;
2005-04-17 02:20:36 +04:00
/* fid:
* multiplier * 10
*/
2008-06-14 23:11:39 +04:00
static int fid ;
2005-04-17 02:20:36 +04:00
/* min_fsb, max_fsb:
2006-02-28 08:43:23 +03:00
* minimum and maximum FSB ( = FSB at boot time )
2005-04-17 02:20:36 +04:00
*/
2008-06-14 23:11:39 +04:00
static int min_fsb ;
static int max_fsb ;
2005-04-17 02:20:36 +04:00
MODULE_AUTHOR ( " Sebastian Witt <se.witt@gmx.net> " ) ;
MODULE_DESCRIPTION ( " nForce2 FSB changing cpufreq driver " ) ;
MODULE_LICENSE ( " GPL " ) ;
module_param ( fid , int , 0444 ) ;
module_param ( min_fsb , int , 0444 ) ;
MODULE_PARM_DESC ( fid , " CPU multiplier to use (11.5 = 115) " ) ;
MODULE_PARM_DESC ( min_fsb ,
2008-06-14 23:11:39 +04:00
" Minimum FSB to use, if not defined: current FSB - 50 " ) ;
2005-04-17 02:20:36 +04:00
2009-01-18 06:42:19 +03:00
# define PFX "cpufreq-nforce2: "
2005-04-17 02:20:36 +04:00
2006-02-28 08:43:23 +03:00
/**
2005-04-17 02:20:36 +04:00
* nforce2_calc_fsb - calculate FSB
* @ pll : PLL value
2006-02-28 08:43:23 +03:00
*
2005-04-17 02:20:36 +04:00
* Calculates FSB from PLL value
*/
static int nforce2_calc_fsb ( int pll )
{
unsigned char mul , div ;
mul = ( pll > > 8 ) & 0xff ;
div = pll & 0xff ;
if ( div > 0 )
return NFORCE2_XTAL * mul / div ;
return 0 ;
}
2006-02-28 08:43:23 +03:00
/**
2005-04-17 02:20:36 +04:00
* nforce2_calc_pll - calculate PLL value
* @ fsb : FSB
2006-02-28 08:43:23 +03:00
*
2005-04-17 02:20:36 +04:00
* Calculate PLL value for given FSB
*/
static int nforce2_calc_pll ( unsigned int fsb )
{
unsigned char xmul , xdiv ;
unsigned char mul = 0 , div = 0 ;
int tried = 0 ;
/* Try to calculate multiplier and divider up to 4 times */
while ( ( ( mul = = 0 ) | | ( div = = 0 ) ) & & ( tried < = 3 ) ) {
2006-06-05 23:25:20 +04:00
for ( xdiv = 2 ; xdiv < = 0x80 ; xdiv + + )
2005-04-17 02:20:36 +04:00
for ( xmul = 1 ; xmul < = 0xfe ; xmul + + )
if ( nforce2_calc_fsb ( NFORCE2_PLL ( xmul , xdiv ) ) = =
fsb + tried ) {
mul = xmul ;
div = xdiv ;
}
tried + + ;
}
if ( ( mul = = 0 ) | | ( div = = 0 ) )
return - 1 ;
return NFORCE2_PLL ( mul , div ) ;
}
2006-02-28 08:43:23 +03:00
/**
2005-04-17 02:20:36 +04:00
* nforce2_write_pll - write PLL value to chipset
* @ pll : PLL value
2006-02-28 08:43:23 +03:00
*
2005-04-17 02:20:36 +04:00
* Writes new FSB PLL value to chipset
*/
static void nforce2_write_pll ( int pll )
{
int temp ;
/* Set the pll addr. to 0x00 */
2009-01-18 06:39:47 +03:00
pci_write_config_dword ( nforce2_dev , NFORCE2_PLLADR , 0 ) ;
2005-04-17 02:20:36 +04:00
/* Now write the value in all 64 registers */
2006-02-28 08:43:23 +03:00
for ( temp = 0 ; temp < = 0x3f ; temp + + )
2009-01-18 06:39:47 +03:00
pci_write_config_dword ( nforce2_dev , NFORCE2_PLLREG , pll ) ;
2005-04-17 02:20:36 +04:00
return ;
}
2006-02-28 08:43:23 +03:00
/**
2005-04-17 02:20:36 +04:00
* nforce2_fsb_read - Read FSB
*
* Read FSB from chipset
* If bootfsb ! = 0 , return FSB at boot - time
*/
static unsigned int nforce2_fsb_read ( int bootfsb )
{
struct pci_dev * nforce2_sub5 ;
u32 fsb , temp = 0 ;
/* Get chipset boot FSB from subdevice 5 (FSB at boot-time) */
2009-01-18 06:39:47 +03:00
nforce2_sub5 = pci_get_subsys ( PCI_VENDOR_ID_NVIDIA , 0x01EF ,
PCI_ANY_ID , PCI_ANY_ID , NULL ) ;
2005-04-17 02:20:36 +04:00
if ( ! nforce2_sub5 )
return 0 ;
pci_read_config_dword ( nforce2_sub5 , NFORCE2_BOOTFSB , & fsb ) ;
fsb / = 1000000 ;
2006-02-28 08:43:23 +03:00
2005-04-17 02:20:36 +04:00
/* Check if PLL register is already set */
2009-01-18 06:39:47 +03:00
pci_read_config_byte ( nforce2_dev , NFORCE2_PLLENABLE , ( u8 * ) & temp ) ;
2006-02-28 08:43:23 +03:00
2008-06-14 23:11:39 +04:00
if ( bootfsb | | ! temp )
2005-04-17 02:20:36 +04:00
return fsb ;
2008-06-14 23:11:39 +04:00
2005-04-17 02:20:36 +04:00
/* Use PLL register FSB value */
2009-01-18 06:39:47 +03:00
pci_read_config_dword ( nforce2_dev , NFORCE2_PLLREG , & temp ) ;
2005-04-17 02:20:36 +04:00
fsb = nforce2_calc_fsb ( temp ) ;
return fsb ;
}
2006-02-28 08:43:23 +03:00
/**
2005-04-17 02:20:36 +04:00
* nforce2_set_fsb - set new FSB
* @ fsb : New FSB
2006-02-28 08:43:23 +03:00
*
2005-04-17 02:20:36 +04:00
* Sets new FSB
*/
static int nforce2_set_fsb ( unsigned int fsb )
{
2005-12-01 12:09:22 +03:00
u32 temp = 0 ;
2005-04-17 02:20:36 +04:00
unsigned int tfsb ;
int diff ;
2005-12-01 12:09:22 +03:00
int pll = 0 ;
2005-04-17 02:20:36 +04:00
if ( ( fsb > max_fsb ) | | ( fsb < NFORCE2_MIN_FSB ) ) {
2009-01-18 06:42:19 +03:00
printk ( KERN_ERR PFX " FSB %d is out of range! \n " , fsb ) ;
2005-04-17 02:20:36 +04:00
return - EINVAL ;
}
2006-02-28 08:43:23 +03:00
2005-04-17 02:20:36 +04:00
tfsb = nforce2_fsb_read ( 0 ) ;
if ( ! tfsb ) {
2009-01-18 06:42:19 +03:00
printk ( KERN_ERR PFX " Error while reading the FSB \n " ) ;
2005-04-17 02:20:36 +04:00
return - EINVAL ;
}
/* First write? Then set actual value */
2009-01-18 06:39:47 +03:00
pci_read_config_byte ( nforce2_dev , NFORCE2_PLLENABLE , ( u8 * ) & temp ) ;
2005-04-17 02:20:36 +04:00
if ( ! temp ) {
pll = nforce2_calc_pll ( tfsb ) ;
if ( pll < 0 )
return - EINVAL ;
nforce2_write_pll ( pll ) ;
}
/* Enable write access */
temp = 0x01 ;
2009-01-18 06:39:47 +03:00
pci_write_config_byte ( nforce2_dev , NFORCE2_PLLENABLE , ( u8 ) temp ) ;
2005-04-17 02:20:36 +04:00
diff = tfsb - fsb ;
if ( ! diff )
return 0 ;
while ( ( tfsb ! = fsb ) & & ( tfsb < = max_fsb ) & & ( tfsb > = min_fsb ) ) {
if ( diff < 0 )
tfsb + + ;
else
tfsb - - ;
/* Calculate the PLL reg. value */
2008-06-14 23:11:39 +04:00
pll = nforce2_calc_pll ( tfsb ) ;
if ( pll = = - 1 )
2005-04-17 02:20:36 +04:00
return - EINVAL ;
2006-02-28 08:43:23 +03:00
2005-04-17 02:20:36 +04:00
nforce2_write_pll ( pll ) ;
# ifdef NFORCE2_DELAY
mdelay ( NFORCE2_DELAY ) ;
# endif
}
temp = 0x40 ;
2009-01-18 06:39:47 +03:00
pci_write_config_byte ( nforce2_dev , NFORCE2_PLLADR , ( u8 ) temp ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
/**
* nforce2_get - get the CPU frequency
* @ cpu : CPU number
2006-02-28 08:43:23 +03:00
*
2005-04-17 02:20:36 +04:00
* Returns the CPU frequency
*/
static unsigned int nforce2_get ( unsigned int cpu )
{
if ( cpu )
return 0 ;
return nforce2_fsb_read ( 0 ) * fid * 100 ;
}
/**
* nforce2_target - set a new CPUFreq policy
* @ policy : new policy
* @ target_freq : the target frequency
2009-01-18 06:39:47 +03:00
* @ relation : how that frequency relates to achieved frequency
* ( CPUFREQ_RELATION_L or CPUFREQ_RELATION_H )
2005-04-17 02:20:36 +04:00
*
* Sets a new CPUFreq policy .
*/
static int nforce2_target ( struct cpufreq_policy * policy ,
unsigned int target_freq , unsigned int relation )
{
2008-06-14 23:11:39 +04:00
/* unsigned long flags; */
2005-04-17 02:20:36 +04:00
struct cpufreq_freqs freqs ;
unsigned int target_fsb ;
if ( ( target_freq > policy - > max ) | | ( target_freq < policy - > min ) )
return - EINVAL ;
target_fsb = target_freq / ( fid * 100 ) ;
freqs . old = nforce2_get ( policy - > cpu ) ;
freqs . new = target_fsb * fid * 100 ;
if ( freqs . old = = freqs . new )
return 0 ;
2011-03-27 17:04:46 +04:00
pr_debug ( " Old CPU frequency %d kHz, new %d kHz \n " ,
2005-04-17 02:20:36 +04:00
freqs . old , freqs . new ) ;
2014-03-24 12:05:45 +04:00
cpufreq_freq_transition_begin ( policy , & freqs ) ;
2005-04-17 02:20:36 +04:00
/* Disable IRQs */
2008-06-14 23:11:39 +04:00
/* local_irq_save(flags); */
2005-04-17 02:20:36 +04:00
if ( nforce2_set_fsb ( target_fsb ) < 0 )
2009-01-18 06:42:19 +03:00
printk ( KERN_ERR PFX " Changing FSB to %d failed \n " ,
2008-06-14 23:11:39 +04:00
target_fsb ) ;
2005-04-17 02:20:36 +04:00
else
2011-03-27 17:04:46 +04:00
pr_debug ( " Changed FSB successfully to %d \n " ,
2008-06-14 23:11:39 +04:00
target_fsb ) ;
2005-04-17 02:20:36 +04:00
/* Enable IRQs */
2008-06-14 23:11:39 +04:00
/* local_irq_restore(flags); */
2005-04-17 02:20:36 +04:00
2014-03-24 12:05:45 +04:00
cpufreq_freq_transition_end ( policy , & freqs , 0 ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
/**
* nforce2_verify - verifies a new CPUFreq policy
* @ policy : new policy
*/
static int nforce2_verify ( struct cpufreq_policy * policy )
{
unsigned int fsb_pol_max ;
fsb_pol_max = policy - > max / ( fid * 100 ) ;
if ( policy - > min < ( fsb_pol_max * fid * 100 ) )
policy - > max = ( fsb_pol_max + 1 ) * fid * 100 ;
2013-10-02 12:43:19 +04:00
cpufreq_verify_within_cpu_limits ( policy ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
static int nforce2_cpu_init ( struct cpufreq_policy * policy )
{
unsigned int fsb ;
unsigned int rfid ;
/* capability check */
if ( policy - > cpu ! = 0 )
return - ENODEV ;
/* Get current FSB */
fsb = nforce2_fsb_read ( 0 ) ;
if ( ! fsb )
return - EIO ;
/* FIX: Get FID from CPU */
if ( ! fid ) {
if ( ! cpu_khz ) {
2009-01-18 06:42:19 +03:00
printk ( KERN_WARNING PFX
" cpu_khz not set, can't calculate multiplier! \n " ) ;
2005-04-17 02:20:36 +04:00
return - ENODEV ;
}
fid = cpu_khz / ( fsb * 100 ) ;
rfid = fid % 5 ;
if ( rfid ) {
if ( rfid > 2 )
fid + = 5 - rfid ;
else
fid - = rfid ;
}
}
2009-01-18 06:42:19 +03:00
printk ( KERN_INFO PFX " FSB currently at %i MHz, FID %d.%d \n " , fsb ,
2005-04-17 02:20:36 +04:00
fid / 10 , fid % 10 ) ;
2006-02-28 08:43:23 +03:00
2005-04-17 02:20:36 +04:00
/* Set maximum FSB to FSB at boot time */
max_fsb = nforce2_fsb_read ( 1 ) ;
2006-02-28 08:43:23 +03:00
2008-06-14 23:11:39 +04:00
if ( ! max_fsb )
2005-04-17 02:20:36 +04:00
return - EIO ;
if ( ! min_fsb )
min_fsb = max_fsb - NFORCE2_SAFE_DISTANCE ;
if ( min_fsb < NFORCE2_MIN_FSB )
min_fsb = NFORCE2_MIN_FSB ;
/* cpuinfo and default policy values */
2013-04-01 16:57:48 +04:00
policy - > min = policy - > cpuinfo . min_freq = min_fsb * fid * 100 ;
policy - > max = policy - > cpuinfo . max_freq = max_fsb * fid * 100 ;
2005-04-17 02:20:36 +04:00
policy - > cpuinfo . transition_latency = CPUFREQ_ETERNAL ;
return 0 ;
}
static int nforce2_cpu_exit ( struct cpufreq_policy * policy )
{
return 0 ;
}
2007-02-27 01:55:48 +03:00
static struct cpufreq_driver nforce2_driver = {
2005-04-17 02:20:36 +04:00
. name = " nforce2 " ,
. verify = nforce2_verify ,
. target = nforce2_target ,
. get = nforce2_get ,
. init = nforce2_cpu_init ,
. exit = nforce2_cpu_exit ,
} ;
2012-01-26 03:09:12 +04:00
# ifdef MODULE
2014-05-09 10:08:29 +04:00
static const struct pci_device_id nforce2_ids [ ] = {
2012-01-26 03:09:12 +04:00
{ PCI_VENDOR_ID_NVIDIA , PCI_DEVICE_ID_NVIDIA_NFORCE2 } ,
{ }
} ;
MODULE_DEVICE_TABLE ( pci , nforce2_ids ) ;
# endif
2005-04-17 02:20:36 +04:00
/**
* nforce2_detect_chipset - detect the Southbridge which contains FSB PLL logic
*
* Detects nForce2 A2 and C1 stepping
2006-02-28 08:43:23 +03:00
*
2005-04-17 02:20:36 +04:00
*/
2010-09-05 23:00:20 +04:00
static int nforce2_detect_chipset ( void )
2005-04-17 02:20:36 +04:00
{
2009-01-18 06:39:47 +03:00
nforce2_dev = pci_get_subsys ( PCI_VENDOR_ID_NVIDIA ,
2006-02-28 08:43:23 +03:00
PCI_DEVICE_ID_NVIDIA_NFORCE2 ,
PCI_ANY_ID , PCI_ANY_ID , NULL ) ;
2005-04-17 02:20:36 +04:00
2009-01-18 06:39:47 +03:00
if ( nforce2_dev = = NULL )
2005-04-17 02:20:36 +04:00
return - ENODEV ;
2009-01-18 06:42:19 +03:00
printk ( KERN_INFO PFX " Detected nForce2 chipset revision %X \n " ,
2009-01-18 06:39:47 +03:00
nforce2_dev - > revision ) ;
2009-01-18 06:42:19 +03:00
printk ( KERN_INFO PFX
" FSB changing is maybe unstable and can lead to "
2009-01-18 06:39:47 +03:00
" crashes and data loss. \n " ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
/**
* nforce2_init - initializes the nForce2 CPUFreq driver
*
* Initializes the nForce2 FSB support . Returns - ENODEV on unsupported
2015-05-23 08:06:18 +03:00
* devices , - EINVAL on problems during initialization , and zero on
2005-04-17 02:20:36 +04:00
* success .
*/
static int __init nforce2_init ( void )
{
/* TODO: do we need to detect the processor? */
/* detect chipset */
if ( nforce2_detect_chipset ( ) ) {
2009-02-21 04:58:47 +03:00
printk ( KERN_INFO PFX " No nForce2 chipset. \n " ) ;
2005-04-17 02:20:36 +04:00
return - ENODEV ;
}
return cpufreq_register_driver ( & nforce2_driver ) ;
}
/**
* nforce2_exit - unregisters cpufreq module
*
* Unregisters nForce2 FSB change support .
*/
static void __exit nforce2_exit ( void )
{
cpufreq_unregister_driver ( & nforce2_driver ) ;
}
module_init ( nforce2_init ) ;
module_exit ( nforce2_exit ) ;