2006-06-26 11:25:12 +04:00
/*
* linux / drivers / clocksource / acpi_pm . c
*
* This file contains the ACPI PM based clocksource .
*
* This code was largely moved from the i386 timer_pm . c file
* which was ( C ) Dominik Brodowski < linux @ brodo . de > 2003
* and contained the following comments :
*
* Driver to use the Power Management Timer ( PMTMR ) available in some
* southbridges as primary timing source for the Linux kernel .
*
* Based on parts of linux / drivers / acpi / hardware / hwtimer . c , timer_pit . c ,
* timer_hpet . c , and on Arjan van de Ven ' s implementation for 2.4 .
*
* This file is licensed under the GPL v2 .
*/
2007-02-16 12:27:57 +03:00
# include <linux/acpi_pmtmr.h>
2006-06-26 11:25:12 +04:00
# include <linux/clocksource.h>
# include <linux/errno.h>
# include <linux/init.h>
# include <linux/pci.h>
# include <asm/io.h>
/*
* The I / O port the PMTMR resides at .
* The location is detected during setup_arch ( ) ,
* in arch / i386 / acpi / boot . c
*/
2006-06-26 11:25:14 +04:00
u32 pmtmr_ioport __read_mostly ;
2006-06-26 11:25:12 +04:00
static inline u32 read_pmtmr ( void )
{
/* mask the output to 24 bits */
return inl ( pmtmr_ioport ) & ACPI_PM_MASK ;
}
2007-02-16 12:27:57 +03:00
u32 acpi_pm_read_verified ( void )
2006-06-26 11:25:12 +04:00
{
u32 v1 = 0 , v2 = 0 , v3 = 0 ;
/*
* It has been reported that because of various broken
* chipsets ( ICH4 , PIIX4 and PIIX4E ) where the ACPI PM clock
2006-06-26 11:25:14 +04:00
* source is not latched , you must read it multiple
2006-06-26 11:25:12 +04:00
* times to ensure a safe value is read :
*/
do {
v1 = read_pmtmr ( ) ;
v2 = read_pmtmr ( ) ;
v3 = read_pmtmr ( ) ;
2006-10-21 21:24:10 +04:00
} while ( unlikely ( ( v1 > v2 & & v1 < v3 ) | | ( v2 > v3 & & v2 < v1 )
| | ( v3 > v1 & & v3 < v2 ) ) ) ;
2006-06-26 11:25:12 +04:00
2007-02-16 12:27:57 +03:00
return v2 ;
}
static cycle_t acpi_pm_read_slow ( void )
{
return ( cycle_t ) acpi_pm_read_verified ( ) ;
2006-06-26 11:25:12 +04:00
}
static cycle_t acpi_pm_read ( void )
{
return ( cycle_t ) read_pmtmr ( ) ;
}
static struct clocksource clocksource_acpi_pm = {
. name = " acpi_pm " ,
. rating = 200 ,
. read = acpi_pm_read ,
. mask = ( cycle_t ) ACPI_PM_MASK ,
. mult = 0 , /*to be caluclated*/
. shift = 22 ,
2007-02-16 12:27:36 +03:00
. flags = CLOCK_SOURCE_IS_CONTINUOUS ,
2006-06-26 11:25:12 +04:00
} ;
# ifdef CONFIG_PCI
2006-12-10 13:21:33 +03:00
static int __devinitdata acpi_pm_good ;
2006-06-26 11:25:12 +04:00
static int __init acpi_pm_good_setup ( char * __str )
{
2006-12-10 13:21:33 +03:00
acpi_pm_good = 1 ;
return 1 ;
2006-06-26 11:25:12 +04:00
}
__setup ( " acpi_pm_good " , acpi_pm_good_setup ) ;
static inline void acpi_pm_need_workaround ( void )
{
2007-02-16 12:27:57 +03:00
clocksource_acpi_pm . read = acpi_pm_read_slow ;
2007-03-27 09:32:19 +04:00
clocksource_acpi_pm . rating = 120 ;
2006-06-26 11:25:12 +04:00
}
/*
* PIIX4 Errata :
*
* The power management timer may return improper results when read .
* Although the timer value settles properly after incrementing ,
* while incrementing there is a 3 ns window every 69.8 ns where the
* timer value is indeterminate ( a 4.2 % chance that the data will be
* incorrect when read ) . As a result , the ACPI free running count up
* timer specification is violated due to erroneous reads .
*/
static void __devinit acpi_pm_check_blacklist ( struct pci_dev * dev )
{
u8 rev ;
if ( acpi_pm_good )
return ;
pci_read_config_byte ( dev , PCI_REVISION_ID , & rev ) ;
/* the bug has been fixed in PIIX4M */
if ( rev < 3 ) {
printk ( KERN_WARNING " * Found PM-Timer Bug on the chipset. "
" Due to workarounds for a bug, \n "
" * this clock source is slow. Consider trying "
" other clock sources \n " ) ;
acpi_pm_need_workaround ( ) ;
}
}
DECLARE_PCI_FIXUP_EARLY ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_82371AB_3 ,
acpi_pm_check_blacklist ) ;
static void __devinit acpi_pm_check_graylist ( struct pci_dev * dev )
{
if ( acpi_pm_good )
return ;
printk ( KERN_WARNING " * The chipset may have PM-Timer Bug. Due to "
" workarounds for a bug, \n "
" * this clock source is slow. If you are sure your timer "
" does not have \n "
" * this bug, please use \" acpi_pm_good \" to disable the "
" workaround \n " ) ;
acpi_pm_need_workaround ( ) ;
}
DECLARE_PCI_FIXUP_EARLY ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_82801DB_0 ,
acpi_pm_check_graylist ) ;
2006-10-21 21:24:10 +04:00
DECLARE_PCI_FIXUP_EARLY ( PCI_VENDOR_ID_SERVERWORKS , PCI_DEVICE_ID_SERVERWORKS_LE ,
acpi_pm_check_graylist ) ;
2006-06-26 11:25:12 +04:00
# endif
2006-12-08 13:36:02 +03:00
# ifndef CONFIG_X86_64
# include "mach_timer.h"
# define PMTMR_EXPECTED_RATE \
( ( CALIBRATE_LATCH * ( PMTMR_TICKS_PER_SEC > > 10 ) ) / ( CLOCK_TICK_RATE > > 10 ) )
/*
* Some boards have the PMTMR running way too fast . We check
* the PMTMR rate against PIT channel 2 to catch these cases .
*/
static int verify_pmtmr_rate ( void )
{
u32 value1 , value2 ;
unsigned long count , delta ;
mach_prepare_counter ( ) ;
value1 = read_pmtmr ( ) ;
mach_countup ( & count ) ;
value2 = read_pmtmr ( ) ;
delta = ( value2 - value1 ) & ACPI_PM_MASK ;
/* Check that the PMTMR delta is within 5% of what we expect */
if ( delta < ( PMTMR_EXPECTED_RATE * 19 ) / 20 | |
delta > ( PMTMR_EXPECTED_RATE * 21 ) / 20 ) {
printk ( KERN_INFO " PM-Timer running at invalid rate: %lu%% "
" of normal - aborting. \n " ,
100UL * delta / PMTMR_EXPECTED_RATE ) ;
return - 1 ;
}
return 0 ;
}
# else
# define verify_pmtmr_rate() (0)
# endif
2006-06-26 11:25:12 +04:00
static int __init init_acpi_pm_clocksource ( void )
{
u32 value1 , value2 ;
unsigned int i ;
if ( ! pmtmr_ioport )
return - ENODEV ;
clocksource_acpi_pm . mult = clocksource_hz2mult ( PMTMR_TICKS_PER_SEC ,
clocksource_acpi_pm . shift ) ;
/* "verify" this timing source: */
value1 = read_pmtmr ( ) ;
for ( i = 0 ; i < 10000 ; i + + ) {
value2 = read_pmtmr ( ) ;
if ( value2 = = value1 )
continue ;
if ( value2 > value1 )
goto pm_good ;
if ( ( value2 < value1 ) & & ( ( value2 ) < 0xFFF ) )
goto pm_good ;
printk ( KERN_INFO " PM-Timer had inconsistent results: "
" 0x%#x, 0x%#x - aborting. \n " , value1 , value2 ) ;
return - EINVAL ;
}
printk ( KERN_INFO " PM-Timer had no reasonable result: "
" 0x%#x - aborting. \n " , value1 ) ;
return - ENODEV ;
pm_good :
2006-12-08 13:36:02 +03:00
if ( verify_pmtmr_rate ( ) ! = 0 )
return - ENODEV ;
2006-06-26 11:25:14 +04:00
return clocksource_register ( & clocksource_acpi_pm ) ;
2006-06-26 11:25:12 +04:00
}
2007-03-05 11:30:50 +03:00
/* We use fs_initcall because we want the PCI fixups to have run
* but we still need to load before device_initcall
*/
fs_initcall ( init_acpi_pm_clocksource ) ;