2005-04-17 02:20:36 +04:00
/*
* This file contains work - arounds for x86 and x86_64 platform bugs .
*/
# include <linux/pci.h>
# include <linux/irq.h>
2007-10-13 01:04:23 +04:00
# include <asm/hpet.h>
2005-04-17 02:20:36 +04:00
# if defined(CONFIG_X86_IO_APIC) && defined(CONFIG_SMP) && defined(CONFIG_PCI)
2012-12-22 02:02:53 +04:00
static void quirk_intel_irqbalance ( struct pci_dev * dev )
2005-04-17 02:20:36 +04:00
{
2011-07-11 19:01:38 +04:00
u8 config ;
2008-02-11 07:18:15 +03:00
u16 word ;
2005-04-17 02:20:36 +04:00
/* BIOS may enable hardware IRQ balancing for
* E7520 / E7320 / E7525 ( revision ID 0x9 and below )
* based platforms .
* Disable SW irqbalance / affinity on those platforms .
*/
2011-07-11 19:01:38 +04:00
if ( dev - > revision > 0x9 )
2005-04-17 02:20:36 +04:00
return ;
2007-05-02 21:27:04 +04:00
/* enable access to config space*/
pci_read_config_byte ( dev , 0xf4 , & config ) ;
pci_write_config_byte ( dev , 0xf4 , config | 0x2 ) ;
2005-04-17 02:20:36 +04:00
2008-02-11 07:18:15 +03:00
/*
* read xTPR register . We may not have a pci_dev for device 8
* because it might be hidden until the above write .
*/
pci_bus_read_config_word ( dev - > bus , PCI_DEVFN ( 8 , 0 ) , 0x4c , & word ) ;
2005-04-17 02:20:36 +04:00
if ( ! ( word & ( 1 < < 13 ) ) ) {
2007-12-18 00:09:40 +03:00
dev_info ( & dev - > dev , " Intel E7520/7320/7525 detected; "
" disabling irq balancing and affinity \n " ) ;
2005-04-17 02:20:36 +04:00
noirqdebug_setup ( " " ) ;
# ifdef CONFIG_PROC_FS
no_irq_affinity = 1 ;
# endif
}
2007-05-02 21:27:04 +04:00
/* put back the original value for config space*/
2006-01-19 04:44:13 +03:00
if ( ! ( config & 0x2 ) )
2007-05-02 21:27:04 +04:00
pci_write_config_byte ( dev , 0xf4 , config ) ;
2005-04-17 02:20:36 +04:00
}
2007-10-19 22:35:02 +04:00
DECLARE_PCI_FIXUP_FINAL ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_E7320_MCH ,
quirk_intel_irqbalance ) ;
DECLARE_PCI_FIXUP_FINAL ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_E7525_MCH ,
quirk_intel_irqbalance ) ;
DECLARE_PCI_FIXUP_FINAL ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_E7520_MCH ,
quirk_intel_irqbalance ) ;
2005-04-17 02:20:36 +04:00
# endif
2007-10-13 01:04:23 +04:00
# if defined(CONFIG_HPET_TIMER)
unsigned long force_hpet_address ;
2007-10-13 01:04:24 +04:00
static enum {
NONE_FORCE_HPET_RESUME ,
OLD_ICH_FORCE_HPET_RESUME ,
2007-10-19 22:35:02 +04:00
ICH_FORCE_HPET_RESUME ,
2007-10-19 21:51:27 +04:00
VT8237_FORCE_HPET_RESUME ,
NVIDIA_FORCE_HPET_RESUME ,
2008-05-09 13:49:11 +04:00
ATI_FORCE_HPET_RESUME ,
2007-10-13 01:04:24 +04:00
} force_hpet_resume_type ;
2007-10-13 01:04:23 +04:00
static void __iomem * rcba_base ;
2007-10-13 01:04:24 +04:00
static void ich_force_hpet_resume ( void )
2007-10-13 01:04:23 +04:00
{
u32 val ;
if ( ! force_hpet_address )
return ;
2009-03-10 08:10:32 +03:00
BUG_ON ( rcba_base = = NULL ) ;
2007-10-13 01:04:23 +04:00
/* read the Function Disable register, dword mode only */
val = readl ( rcba_base + 0x3404 ) ;
if ( ! ( val & 0x80 ) ) {
/* HPET disabled in HPTC. Trying to enable */
writel ( val | 0x80 , rcba_base + 0x3404 ) ;
}
val = readl ( rcba_base + 0x3404 ) ;
if ( ! ( val & 0x80 ) )
BUG ( ) ;
else
printk ( KERN_DEBUG " Force enabled HPET at resume \n " ) ;
return ;
}
static void ich_force_enable_hpet ( struct pci_dev * dev )
{
u32 val ;
u32 uninitialized_var ( rcba ) ;
int err = 0 ;
if ( hpet_address | | force_hpet_address )
return ;
pci_read_config_dword ( dev , 0xF0 , & rcba ) ;
rcba & = 0xFFFFC000 ;
if ( rcba = = 0 ) {
2007-12-18 00:09:40 +03:00
dev_printk ( KERN_DEBUG , & dev - > dev , " RCBA disabled; "
" cannot force enable HPET \n " ) ;
2007-10-13 01:04:23 +04:00
return ;
}
/* use bits 31:14, 16 kB aligned */
rcba_base = ioremap_nocache ( rcba , 0x4000 ) ;
if ( rcba_base = = NULL ) {
2007-12-18 00:09:40 +03:00
dev_printk ( KERN_DEBUG , & dev - > dev , " ioremap failed; "
" cannot force enable HPET \n " ) ;
2007-10-13 01:04:23 +04:00
return ;
}
/* read the Function Disable register, dword mode only */
val = readl ( rcba_base + 0x3404 ) ;
if ( val & 0x80 ) {
/* HPET is enabled in HPTC. Just not reported by BIOS */
val = val & 0x3 ;
force_hpet_address = 0xFED00000 | ( val < < 12 ) ;
2007-12-18 00:09:40 +03:00
dev_printk ( KERN_DEBUG , & dev - > dev , " Force enabled HPET at "
" 0x%lx \n " , force_hpet_address ) ;
2007-10-13 01:04:23 +04:00
iounmap ( rcba_base ) ;
return ;
}
/* HPET disabled in HPTC. Trying to enable */
writel ( val | 0x80 , rcba_base + 0x3404 ) ;
val = readl ( rcba_base + 0x3404 ) ;
if ( ! ( val & 0x80 ) ) {
err = 1 ;
} else {
val = val & 0x3 ;
force_hpet_address = 0xFED00000 | ( val < < 12 ) ;
}
if ( err ) {
force_hpet_address = 0 ;
iounmap ( rcba_base ) ;
2007-12-18 00:09:40 +03:00
dev_printk ( KERN_DEBUG , & dev - > dev ,
" Failed to force enable HPET \n " ) ;
2007-10-13 01:04:23 +04:00
} else {
2007-10-13 01:04:24 +04:00
force_hpet_resume_type = ICH_FORCE_HPET_RESUME ;
2007-12-18 00:09:40 +03:00
dev_printk ( KERN_DEBUG , & dev - > dev , " Force enabled HPET at "
" 0x%lx \n " , force_hpet_address ) ;
2007-10-13 01:04:23 +04:00
}
}
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_ESB2_0 ,
2007-10-19 22:35:02 +04:00
ich_force_enable_hpet ) ;
2008-06-04 05:40:17 +04:00
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_ICH6_0 ,
ich_force_enable_hpet ) ;
2007-10-13 01:04:23 +04:00
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_ICH6_1 ,
2007-10-19 22:35:02 +04:00
ich_force_enable_hpet ) ;
2007-10-13 01:04:24 +04:00
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_ICH7_0 ,
2007-10-19 22:35:02 +04:00
ich_force_enable_hpet ) ;
2007-10-13 01:04:23 +04:00
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_ICH7_1 ,
2007-10-19 22:35:02 +04:00
ich_force_enable_hpet ) ;
2007-10-13 01:04:23 +04:00
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_ICH7_31 ,
2007-10-19 22:35:02 +04:00
ich_force_enable_hpet ) ;
2007-10-13 01:04:23 +04:00
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_ICH8_1 ,
2007-10-19 22:35:02 +04:00
ich_force_enable_hpet ) ;
2008-12-16 14:39:57 +03:00
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_ICH8_4 ,
ich_force_enable_hpet ) ;
2008-01-30 15:33:39 +03:00
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_ICH9_7 ,
ich_force_enable_hpet ) ;
2009-01-09 23:17:40 +03:00
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_INTEL , 0x3a16 , /* ICH10 */
ich_force_enable_hpet ) ;
2007-10-13 01:04:24 +04:00
static struct pci_dev * cached_dev ;
2008-05-10 23:42:14 +04:00
static void hpet_print_force_info ( void )
{
printk ( KERN_INFO " HPET not enabled in BIOS. "
" You might try hpet=force boot option \n " ) ;
}
2007-10-13 01:04:24 +04:00
static void old_ich_force_hpet_resume ( void )
{
u32 val ;
u32 uninitialized_var ( gen_cntl ) ;
if ( ! force_hpet_address | | ! cached_dev )
return ;
pci_read_config_dword ( cached_dev , 0xD0 , & gen_cntl ) ;
gen_cntl & = ( ~ ( 0x7 < < 15 ) ) ;
gen_cntl | = ( 0x4 < < 15 ) ;
pci_write_config_dword ( cached_dev , 0xD0 , gen_cntl ) ;
pci_read_config_dword ( cached_dev , 0xD0 , & gen_cntl ) ;
val = gen_cntl > > 15 ;
val & = 0x7 ;
if ( val = = 0x4 )
printk ( KERN_DEBUG " Force enabled HPET at resume \n " ) ;
else
BUG ( ) ;
}
static void old_ich_force_enable_hpet ( struct pci_dev * dev )
{
u32 val ;
u32 uninitialized_var ( gen_cntl ) ;
if ( hpet_address | | force_hpet_address )
return ;
pci_read_config_dword ( dev , 0xD0 , & gen_cntl ) ;
/*
* Bit 17 is HPET enable bit .
* Bit 16 : 15 control the HPET base address .
*/
val = gen_cntl > > 15 ;
val & = 0x7 ;
if ( val & 0x4 ) {
val & = 0x3 ;
force_hpet_address = 0xFED00000 | ( val < < 12 ) ;
2007-12-18 00:09:40 +03:00
dev_printk ( KERN_DEBUG , & dev - > dev , " HPET at 0x%lx \n " ,
force_hpet_address ) ;
2007-10-13 01:04:24 +04:00
return ;
}
/*
* HPET is disabled . Trying enabling at FED00000 and check
* whether it sticks
*/
gen_cntl & = ( ~ ( 0x7 < < 15 ) ) ;
gen_cntl | = ( 0x4 < < 15 ) ;
pci_write_config_dword ( dev , 0xD0 , gen_cntl ) ;
pci_read_config_dword ( dev , 0xD0 , & gen_cntl ) ;
val = gen_cntl > > 15 ;
val & = 0x7 ;
if ( val & 0x4 ) {
/* HPET is enabled in HPTC. Just not reported by BIOS */
val & = 0x3 ;
force_hpet_address = 0xFED00000 | ( val < < 12 ) ;
2007-12-18 00:09:40 +03:00
dev_printk ( KERN_DEBUG , & dev - > dev , " Force enabled HPET at "
" 0x%lx \n " , force_hpet_address ) ;
2007-10-13 01:04:24 +04:00
cached_dev = dev ;
2007-10-13 01:04:24 +04:00
force_hpet_resume_type = OLD_ICH_FORCE_HPET_RESUME ;
return ;
}
2007-12-18 00:09:40 +03:00
dev_printk ( KERN_DEBUG , & dev - > dev , " Failed to force enable HPET \n " ) ;
2007-10-13 01:04:24 +04:00
}
2007-10-19 22:35:02 +04:00
/*
* Undocumented chipset features . Make sure that the user enforced
* this .
*/
static void old_ich_force_enable_hpet_user ( struct pci_dev * dev )
{
if ( hpet_force_user )
old_ich_force_enable_hpet ( dev ) ;
}
2008-06-09 16:55:20 +04:00
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_ESB_1 ,
old_ich_force_enable_hpet_user ) ;
2007-10-19 22:35:02 +04:00
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_82801CA_0 ,
old_ich_force_enable_hpet_user ) ;
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_82801CA_12 ,
old_ich_force_enable_hpet_user ) ;
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_82801DB_0 ,
old_ich_force_enable_hpet_user ) ;
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_82801DB_12 ,
old_ich_force_enable_hpet_user ) ;
2007-10-13 01:04:24 +04:00
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_82801EB_0 ,
2007-10-19 22:35:02 +04:00
old_ich_force_enable_hpet ) ;
2007-10-13 01:04:24 +04:00
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_82801EB_12 ,
2007-10-19 22:35:02 +04:00
old_ich_force_enable_hpet ) ;
2007-10-13 01:04:24 +04:00
2007-10-19 22:35:02 +04:00
static void vt8237_force_hpet_resume ( void )
{
u32 val ;
if ( ! force_hpet_address | | ! cached_dev )
return ;
val = 0xfed00000 | 0x80 ;
pci_write_config_dword ( cached_dev , 0x68 , val ) ;
pci_read_config_dword ( cached_dev , 0x68 , & val ) ;
if ( val & 0x80 )
printk ( KERN_DEBUG " Force enabled HPET at resume \n " ) ;
else
BUG ( ) ;
}
static void vt8237_force_enable_hpet ( struct pci_dev * dev )
{
u32 uninitialized_var ( val ) ;
2008-05-10 23:42:14 +04:00
if ( hpet_address | | force_hpet_address )
return ;
if ( ! hpet_force_user ) {
hpet_print_force_info ( ) ;
2007-10-19 22:35:02 +04:00
return ;
2008-05-10 23:42:14 +04:00
}
2007-10-19 22:35:02 +04:00
pci_read_config_dword ( dev , 0x68 , & val ) ;
/*
* Bit 7 is HPET enable bit .
* Bit 31 : 10 is HPET base address ( contrary to what datasheet claims )
*/
if ( val & 0x80 ) {
force_hpet_address = ( val & ~ 0x3ff ) ;
2007-12-18 00:09:40 +03:00
dev_printk ( KERN_DEBUG , & dev - > dev , " HPET at 0x%lx \n " ,
force_hpet_address ) ;
2007-10-19 22:35:02 +04:00
return ;
}
/*
* HPET is disabled . Trying enabling at FED00000 and check
* whether it sticks
*/
val = 0xfed00000 | 0x80 ;
pci_write_config_dword ( dev , 0x68 , val ) ;
pci_read_config_dword ( dev , 0x68 , & val ) ;
if ( val & 0x80 ) {
force_hpet_address = ( val & ~ 0x3ff ) ;
2007-12-18 00:09:40 +03:00
dev_printk ( KERN_DEBUG , & dev - > dev , " Force enabled HPET at "
" 0x%lx \n " , force_hpet_address ) ;
2007-10-19 22:35:02 +04:00
cached_dev = dev ;
force_hpet_resume_type = VT8237_FORCE_HPET_RESUME ;
return ;
}
2007-12-18 00:09:40 +03:00
dev_printk ( KERN_DEBUG , & dev - > dev , " Failed to force enable HPET \n " ) ;
2007-10-19 22:35:02 +04:00
}
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_VIA , PCI_DEVICE_ID_VIA_8235 ,
vt8237_force_enable_hpet ) ;
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_VIA , PCI_DEVICE_ID_VIA_8237 ,
vt8237_force_enable_hpet ) ;
2010-09-14 09:15:08 +04:00
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_VIA , PCI_DEVICE_ID_VIA_CX700 ,
vt8237_force_enable_hpet ) ;
2007-10-19 22:35:02 +04:00
2008-05-09 13:49:11 +04:00
static void ati_force_hpet_resume ( void )
{
pci_write_config_dword ( cached_dev , 0x14 , 0xfed00000 ) ;
printk ( KERN_DEBUG " Force enabled HPET at resume \n " ) ;
}
2008-09-05 20:33:26 +04:00
static u32 ati_ixp4x0_rev ( struct pci_dev * dev )
{
2013-03-05 00:16:20 +04:00
int err = 0 ;
u32 d = 0 ;
u8 b = 0 ;
2008-09-05 20:33:26 +04:00
2013-03-05 00:16:20 +04:00
err = pci_read_config_byte ( dev , 0xac , & b ) ;
2008-09-05 20:33:26 +04:00
b & = ~ ( 1 < < 5 ) ;
2013-03-05 00:16:20 +04:00
err | = pci_write_config_byte ( dev , 0xac , b ) ;
err | = pci_read_config_dword ( dev , 0x70 , & d ) ;
2008-09-05 20:33:26 +04:00
d | = 1 < < 8 ;
2013-03-05 00:16:20 +04:00
err | = pci_write_config_dword ( dev , 0x70 , d ) ;
err | = pci_read_config_dword ( dev , 0x8 , & d ) ;
2008-09-05 20:33:26 +04:00
d & = 0xff ;
dev_printk ( KERN_DEBUG , & dev - > dev , " SB4X0 revision 0x%x \n " , d ) ;
2013-03-05 00:16:20 +04:00
WARN_ON_ONCE ( err ) ;
2008-09-05 20:33:26 +04:00
return d ;
}
2008-05-09 13:49:11 +04:00
static void ati_force_enable_hpet ( struct pci_dev * dev )
{
2008-09-05 20:33:26 +04:00
u32 d , val ;
u8 b ;
2008-05-09 13:49:11 +04:00
2008-05-10 23:42:14 +04:00
if ( hpet_address | | force_hpet_address )
return ;
if ( ! hpet_force_user ) {
hpet_print_force_info ( ) ;
2008-05-09 13:49:11 +04:00
return ;
2008-05-10 23:42:14 +04:00
}
2008-05-09 13:49:11 +04:00
2008-09-05 20:33:26 +04:00
d = ati_ixp4x0_rev ( dev ) ;
if ( d < 0x82 )
return ;
/* base address */
2008-05-09 13:49:11 +04:00
pci_write_config_dword ( dev , 0x14 , 0xfed00000 ) ;
pci_read_config_dword ( dev , 0x14 , & val ) ;
2008-09-05 20:33:26 +04:00
/* enable interrupt */
outb ( 0x72 , 0xcd6 ) ; b = inb ( 0xcd7 ) ;
b | = 0x1 ;
outb ( 0x72 , 0xcd6 ) ; outb ( b , 0xcd7 ) ;
outb ( 0x72 , 0xcd6 ) ; b = inb ( 0xcd7 ) ;
if ( ! ( b & 0x1 ) )
return ;
pci_read_config_dword ( dev , 0x64 , & d ) ;
d | = ( 1 < < 10 ) ;
pci_write_config_dword ( dev , 0x64 , d ) ;
pci_read_config_dword ( dev , 0x64 , & d ) ;
if ( ! ( d & ( 1 < < 10 ) ) )
return ;
2008-05-09 13:49:11 +04:00
force_hpet_address = val ;
force_hpet_resume_type = ATI_FORCE_HPET_RESUME ;
dev_printk ( KERN_DEBUG , & dev - > dev , " Force enabled HPET at 0x%lx \n " ,
force_hpet_address ) ;
cached_dev = dev ;
}
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_ATI , PCI_DEVICE_ID_ATI_IXP400_SMBUS ,
ati_force_enable_hpet ) ;
2007-10-19 21:51:27 +04:00
/*
* Undocumented chipset feature taken from LinuxBIOS .
*/
static void nvidia_force_hpet_resume ( void )
{
pci_write_config_dword ( cached_dev , 0x44 , 0xfed00001 ) ;
printk ( KERN_DEBUG " Force enabled HPET at resume \n " ) ;
}
static void nvidia_force_enable_hpet ( struct pci_dev * dev )
{
u32 uninitialized_var ( val ) ;
2008-05-10 23:42:14 +04:00
if ( hpet_address | | force_hpet_address )
return ;
if ( ! hpet_force_user ) {
hpet_print_force_info ( ) ;
2007-10-19 21:51:27 +04:00
return ;
2008-05-10 23:42:14 +04:00
}
2007-10-19 21:51:27 +04:00
pci_write_config_dword ( dev , 0x44 , 0xfed00001 ) ;
pci_read_config_dword ( dev , 0x44 , & val ) ;
force_hpet_address = val & 0xfffffffe ;
force_hpet_resume_type = NVIDIA_FORCE_HPET_RESUME ;
2007-12-18 00:09:40 +03:00
dev_printk ( KERN_DEBUG , & dev - > dev , " Force enabled HPET at 0x%lx \n " ,
2007-10-19 21:51:27 +04:00
force_hpet_address ) ;
cached_dev = dev ;
return ;
}
/* ISA Bridges */
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_NVIDIA , 0x0050 ,
nvidia_force_enable_hpet ) ;
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_NVIDIA , 0x0051 ,
nvidia_force_enable_hpet ) ;
2007-10-19 22:35:02 +04:00
2007-10-19 22:34:15 +04:00
/* LPC bridges */
2008-03-19 17:51:50 +03:00
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_NVIDIA , 0x0260 ,
nvidia_force_enable_hpet ) ;
2007-10-19 22:34:15 +04:00
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_NVIDIA , 0x0360 ,
nvidia_force_enable_hpet ) ;
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_NVIDIA , 0x0361 ,
nvidia_force_enable_hpet ) ;
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_NVIDIA , 0x0362 ,
nvidia_force_enable_hpet ) ;
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_NVIDIA , 0x0363 ,
nvidia_force_enable_hpet ) ;
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_NVIDIA , 0x0364 ,
nvidia_force_enable_hpet ) ;
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_NVIDIA , 0x0365 ,
nvidia_force_enable_hpet ) ;
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_NVIDIA , 0x0366 ,
nvidia_force_enable_hpet ) ;
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_NVIDIA , 0x0367 ,
nvidia_force_enable_hpet ) ;
2007-10-13 01:04:24 +04:00
void force_hpet_resume ( void )
{
switch ( force_hpet_resume_type ) {
2008-02-07 00:39:44 +03:00
case ICH_FORCE_HPET_RESUME :
ich_force_hpet_resume ( ) ;
return ;
case OLD_ICH_FORCE_HPET_RESUME :
old_ich_force_hpet_resume ( ) ;
return ;
case VT8237_FORCE_HPET_RESUME :
vt8237_force_hpet_resume ( ) ;
return ;
case NVIDIA_FORCE_HPET_RESUME :
nvidia_force_hpet_resume ( ) ;
return ;
2008-05-09 13:49:11 +04:00
case ATI_FORCE_HPET_RESUME :
ati_force_hpet_resume ( ) ;
return ;
2008-02-07 00:39:44 +03:00
default :
2007-10-13 01:04:24 +04:00
break ;
}
}
2010-01-21 22:09:52 +03:00
/*
* HPET MSI on some boards ( ATI SB700 / SB800 ) has side effect on
* floppy DMA . Disable HPET MSI on such platforms .
2010-05-17 20:43:24 +04:00
* See erratum # 27 ( Misinterpreted MSI Requests May Result in
* Corrupted LPC DMA Data ) in AMD Publication # 46837 ,
* " SB700 Family Product Errata " , Rev . 1.0 , March 2010.
2010-01-21 22:09:52 +03:00
*/
static void force_disable_hpet_msi ( struct pci_dev * unused )
{
hpet_msi_disable = 1 ;
}
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_ATI , PCI_DEVICE_ID_ATI_SBX00_SMBUS ,
force_disable_hpet_msi ) ;
2009-04-17 14:07:46 +04:00
# endif
# if defined(CONFIG_PCI) && defined(CONFIG_NUMA)
/* Set correct numa_node information for AMD NB functions */
2012-12-22 02:02:53 +04:00
static void quirk_amd_nb_node ( struct pci_dev * dev )
2009-04-17 14:07:46 +04:00
{
struct pci_dev * nb_ht ;
unsigned int devfn ;
2009-11-12 21:09:31 +03:00
u32 node ;
2009-04-17 14:07:46 +04:00
u32 val ;
devfn = PCI_DEVFN ( PCI_SLOT ( dev - > devfn ) , 0 ) ;
nb_ht = pci_get_slot ( dev - > bus , devfn ) ;
if ( ! nb_ht )
return ;
pci_read_config_dword ( nb_ht , 0x60 , & val ) ;
2014-03-13 15:43:01 +04:00
node = pcibus_to_node ( dev - > bus ) | ( val & 7 ) ;
2009-11-12 21:09:31 +03:00
/*
* Some hardware may return an invalid node ID ,
* so check it first :
*/
if ( node_online ( node ) )
set_dev_node ( & dev - > dev , node ) ;
2009-09-08 14:16:18 +04:00
pci_dev_put ( nb_ht ) ;
2009-04-17 14:07:46 +04:00
}
2007-10-13 01:04:24 +04:00
2009-04-17 14:07:46 +04:00
DECLARE_PCI_FIXUP_FINAL ( PCI_VENDOR_ID_AMD , PCI_DEVICE_ID_AMD_K8_NB ,
quirk_amd_nb_node ) ;
DECLARE_PCI_FIXUP_FINAL ( PCI_VENDOR_ID_AMD , PCI_DEVICE_ID_AMD_K8_NB_ADDRMAP ,
quirk_amd_nb_node ) ;
DECLARE_PCI_FIXUP_FINAL ( PCI_VENDOR_ID_AMD , PCI_DEVICE_ID_AMD_K8_NB_MEMCTL ,
quirk_amd_nb_node ) ;
DECLARE_PCI_FIXUP_FINAL ( PCI_VENDOR_ID_AMD , PCI_DEVICE_ID_AMD_K8_NB_MISC ,
quirk_amd_nb_node ) ;
DECLARE_PCI_FIXUP_FINAL ( PCI_VENDOR_ID_AMD , PCI_DEVICE_ID_AMD_10H_NB_HT ,
quirk_amd_nb_node ) ;
DECLARE_PCI_FIXUP_FINAL ( PCI_VENDOR_ID_AMD , PCI_DEVICE_ID_AMD_10H_NB_MAP ,
quirk_amd_nb_node ) ;
DECLARE_PCI_FIXUP_FINAL ( PCI_VENDOR_ID_AMD , PCI_DEVICE_ID_AMD_10H_NB_DRAM ,
quirk_amd_nb_node ) ;
DECLARE_PCI_FIXUP_FINAL ( PCI_VENDOR_ID_AMD , PCI_DEVICE_ID_AMD_10H_NB_MISC ,
quirk_amd_nb_node ) ;
DECLARE_PCI_FIXUP_FINAL ( PCI_VENDOR_ID_AMD , PCI_DEVICE_ID_AMD_10H_NB_LINK ,
quirk_amd_nb_node ) ;
2011-12-02 11:21:43 +04:00
DECLARE_PCI_FIXUP_FINAL ( PCI_VENDOR_ID_AMD , PCI_DEVICE_ID_AMD_15H_NB_F0 ,
quirk_amd_nb_node ) ;
DECLARE_PCI_FIXUP_FINAL ( PCI_VENDOR_ID_AMD , PCI_DEVICE_ID_AMD_15H_NB_F1 ,
quirk_amd_nb_node ) ;
DECLARE_PCI_FIXUP_FINAL ( PCI_VENDOR_ID_AMD , PCI_DEVICE_ID_AMD_15H_NB_F2 ,
quirk_amd_nb_node ) ;
DECLARE_PCI_FIXUP_FINAL ( PCI_VENDOR_ID_AMD , PCI_DEVICE_ID_AMD_15H_NB_F3 ,
quirk_amd_nb_node ) ;
DECLARE_PCI_FIXUP_FINAL ( PCI_VENDOR_ID_AMD , PCI_DEVICE_ID_AMD_15H_NB_F4 ,
quirk_amd_nb_node ) ;
DECLARE_PCI_FIXUP_FINAL ( PCI_VENDOR_ID_AMD , PCI_DEVICE_ID_AMD_15H_NB_F5 ,
quirk_amd_nb_node ) ;
2007-10-13 01:04:23 +04:00
# endif
2014-01-24 02:13:32 +04:00
# ifdef CONFIG_PCI
/*
* Processor does not ensure DRAM scrub read / write sequence
* is atomic wrt accesses to CC6 save state area . Therefore
* if a concurrent scrub read / write access is to same address
* the entry may appear as if it is not written . This quirk
* applies to Fam16h models 00 h - 0F h
*
* See " Revision Guide " for AMD F16h models 00 h - 0f h ,
* document 51810 rev . 3.04 , Nov 2013
*/
static void amd_disable_seq_and_redirect_scrub ( struct pci_dev * dev )
{
u32 val ;
/*
* Suggested workaround :
* set D18F3x58 [ 4 : 0 ] = 00 h and set D18F3x5C [ 0 ] = 0 b
*/
pci_read_config_dword ( dev , 0x58 , & val ) ;
if ( val & 0x1F ) {
val & = ~ ( 0x1F ) ;
pci_write_config_dword ( dev , 0x58 , val ) ;
}
pci_read_config_dword ( dev , 0x5C , & val ) ;
if ( val & BIT ( 0 ) ) {
val & = ~ BIT ( 0 ) ;
pci_write_config_dword ( dev , 0x5c , val ) ;
}
}
DECLARE_PCI_FIXUP_FINAL ( PCI_VENDOR_ID_AMD , PCI_DEVICE_ID_AMD_16H_NB_F3 ,
amd_disable_seq_and_redirect_scrub ) ;
# endif