2005-04-16 15:20:36 -07:00
/*
* Copyright ( C ) 2004 Matthew Wilcox < matthew @ wil . cx >
* Copyright ( C ) 2004 Intel Corp .
*
* This code is released under the GNU General Public License version 2.
*/
/*
* mmconfig . c - Low - level direct PCI config space access via MMCONFIG
*/
# include <linux/pci.h>
# include <linux/init.h>
2005-06-23 17:35:56 -07:00
# include <linux/acpi.h>
2005-04-16 15:20:36 -07:00
# include "pci.h"
# define mmcfg_virt_addr ((void __iomem *) fix_to_virt(FIX_PCIE_MCFG))
/* The base address of the last MMCONFIG device accessed */
static u32 mmcfg_last_accessed_device ;
2005-12-12 22:17:11 -08:00
static DECLARE_BITMAP ( fallback_slots , 32 ) ;
2005-04-16 15:20:36 -07:00
/*
* Functions for accessing PCI configuration space with MMCONFIG accesses
*/
2005-12-12 22:17:11 -08:00
static u32 get_base_addr ( unsigned int seg , int bus , unsigned devfn )
2005-06-23 17:35:56 -07:00
{
int cfg_num = - 1 ;
struct acpi_table_mcfg_config * cfg ;
2005-12-12 22:17:11 -08:00
if ( seg = = 0 & & bus = = 0 & &
test_bit ( PCI_SLOT ( devfn ) , fallback_slots ) )
return 0 ;
2005-06-23 17:35:56 -07:00
while ( 1 ) {
+ + cfg_num ;
if ( cfg_num > = pci_mmcfg_config_num ) {
2006-01-27 02:03:50 +01:00
break ;
2005-06-23 17:35:56 -07:00
}
cfg = & pci_mmcfg_config [ cfg_num ] ;
if ( cfg - > pci_segment_group_number ! = seg )
continue ;
if ( ( cfg - > start_bus_number < = bus ) & &
( cfg - > end_bus_number > = bus ) )
return cfg - > base_address ;
}
2006-01-27 02:03:50 +01:00
/* Handle more broken MCFG tables on Asus etc.
They only contain a single entry for bus 0 - 0. Assume
this applies to all busses . */
cfg = & pci_mmcfg_config [ 0 ] ;
if ( pci_mmcfg_config_num = = 1 & &
cfg - > pci_segment_group_number = = 0 & &
( cfg - > start_bus_number | cfg - > end_bus_number ) = = 0 )
return cfg - > base_address ;
/* Fall back to type 0 */
return 0 ;
2005-06-23 17:35:56 -07:00
}
2005-04-16 15:20:36 -07:00
2005-12-12 22:17:10 -08:00
static inline void pci_exp_set_dev_base ( unsigned int base , int bus , int devfn )
2005-04-16 15:20:36 -07:00
{
2005-12-12 22:17:10 -08:00
u32 dev_base = base | ( bus < < 20 ) | ( devfn < < 12 ) ;
2005-04-16 15:20:36 -07:00
if ( dev_base ! = mmcfg_last_accessed_device ) {
mmcfg_last_accessed_device = dev_base ;
set_fixmap_nocache ( FIX_PCIE_MCFG , dev_base ) ;
}
}
static int pci_mmcfg_read ( unsigned int seg , unsigned int bus ,
unsigned int devfn , int reg , int len , u32 * value )
{
unsigned long flags ;
2005-12-12 22:17:10 -08:00
u32 base ;
2005-04-16 15:20:36 -07:00
if ( ! value | | ( bus > 255 ) | | ( devfn > 255 ) | | ( reg > 4095 ) )
return - EINVAL ;
2005-12-12 22:17:11 -08:00
base = get_base_addr ( seg , bus , devfn ) ;
2005-12-12 22:17:10 -08:00
if ( ! base )
return pci_conf1_read ( seg , bus , devfn , reg , len , value ) ;
2005-04-16 15:20:36 -07:00
spin_lock_irqsave ( & pci_config_lock , flags ) ;
2005-12-12 22:17:10 -08:00
pci_exp_set_dev_base ( base , bus , devfn ) ;
2005-04-16 15:20:36 -07:00
switch ( len ) {
case 1 :
* value = readb ( mmcfg_virt_addr + reg ) ;
break ;
case 2 :
* value = readw ( mmcfg_virt_addr + reg ) ;
break ;
case 4 :
* value = readl ( mmcfg_virt_addr + reg ) ;
break ;
}
spin_unlock_irqrestore ( & pci_config_lock , flags ) ;
return 0 ;
}
static int pci_mmcfg_write ( unsigned int seg , unsigned int bus ,
unsigned int devfn , int reg , int len , u32 value )
{
unsigned long flags ;
2005-12-12 22:17:10 -08:00
u32 base ;
2005-04-16 15:20:36 -07:00
if ( ( bus > 255 ) | | ( devfn > 255 ) | | ( reg > 4095 ) )
return - EINVAL ;
2005-12-12 22:17:11 -08:00
base = get_base_addr ( seg , bus , devfn ) ;
2005-12-12 22:17:10 -08:00
if ( ! base )
return pci_conf1_write ( seg , bus , devfn , reg , len , value ) ;
2005-04-16 15:20:36 -07:00
spin_lock_irqsave ( & pci_config_lock , flags ) ;
2005-12-12 22:17:10 -08:00
pci_exp_set_dev_base ( base , bus , devfn ) ;
2005-04-16 15:20:36 -07:00
switch ( len ) {
case 1 :
writeb ( value , mmcfg_virt_addr + reg ) ;
break ;
case 2 :
writew ( value , mmcfg_virt_addr + reg ) ;
break ;
case 4 :
writel ( value , mmcfg_virt_addr + reg ) ;
break ;
}
spin_unlock_irqrestore ( & pci_config_lock , flags ) ;
return 0 ;
}
static struct pci_raw_ops pci_mmcfg = {
. read = pci_mmcfg_read ,
. write = pci_mmcfg_write ,
} ;
2005-12-12 22:17:11 -08:00
/* K8 systems have some devices (typically in the builtin northbridge)
that are only accessible using type1
Normally this can be expressed in the MCFG by not listing them
and assigning suitable _SEGs , but this isn ' t implemented in some BIOS .
Instead try to discover all devices on bus 0 that are unreachable using MM
and fallback for them .
We only do this for bus 0 / seg 0 */
static __init void unreachable_devices ( void )
{
int i ;
unsigned long flags ;
for ( i = 0 ; i < 32 ; i + + ) {
u32 val1 ;
u32 addr ;
pci_conf1_read ( 0 , 0 , PCI_DEVFN ( i , 0 ) , 0 , 4 , & val1 ) ;
if ( val1 = = 0xffffffff )
continue ;
/* Locking probably not needed, but safer */
spin_lock_irqsave ( & pci_config_lock , flags ) ;
addr = get_base_addr ( 0 , 0 , PCI_DEVFN ( i , 0 ) ) ;
if ( addr ! = 0 )
pci_exp_set_dev_base ( addr , 0 , PCI_DEVFN ( i , 0 ) ) ;
2005-12-16 11:08:55 -08:00
if ( addr = = 0 | | readl ( ( u32 __iomem * ) mmcfg_virt_addr ) ! = val1 )
2005-12-12 22:17:11 -08:00
set_bit ( i , fallback_slots ) ;
spin_unlock_irqrestore ( & pci_config_lock , flags ) ;
}
}
2005-04-16 15:20:36 -07:00
static int __init pci_mmcfg_init ( void )
{
if ( ( pci_probe & PCI_PROBE_MMCONF ) = = 0 )
goto out ;
2005-06-23 17:35:56 -07:00
acpi_table_parse ( ACPI_MCFG , acpi_parse_mcfg ) ;
if ( ( pci_mmcfg_config_num = = 0 ) | |
( pci_mmcfg_config = = NULL ) | |
( pci_mmcfg_config [ 0 ] . base_address = = 0 ) )
2005-04-16 15:20:36 -07:00
goto out ;
printk ( KERN_INFO " PCI: Using MMCONFIG \n " ) ;
raw_pci_ops = & pci_mmcfg ;
pci_probe = ( pci_probe & ~ PCI_PROBE_MASK ) | PCI_PROBE_MMCONF ;
2005-12-12 22:17:11 -08:00
unreachable_devices ( ) ;
2005-04-16 15:20:36 -07:00
out :
return 0 ;
}
arch_initcall ( pci_mmcfg_init ) ;