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>
2012-06-22 14:55:12 +08:00
# include <linux/rcupdate.h>
2006-04-07 19:49:30 +02:00
# include <asm/e820.h>
2008-12-27 18:32:28 +05:30
# include <asm/pci_x86.h>
2009-08-14 15:37:50 -04:00
# include <acpi/acpi.h>
2005-04-16 15:20:36 -07:00
2006-04-07 19:50:12 +02:00
/* Assume systems with more busses have correct MCFG */
2005-04-16 15:20:36 -07:00
# 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 ;
2006-12-23 10:00:43 +09:00
static int mmcfg_last_accessed_cpu ;
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
{
2009-11-13 17:35:04 -07:00
struct pci_mmcfg_region * cfg = pci_mmconfig_lookup ( seg , bus ) ;
2005-06-23 17:35:56 -07:00
2009-11-13 17:35:04 -07:00
if ( cfg )
return cfg - > address ;
2006-01-27 02:03:50 +01:00
return 0 ;
2005-06-23 17:35:56 -07:00
}
2005-04-16 15:20:36 -07:00
2006-09-30 23:27:10 -07:00
/*
* This is always called under pci_config_lock
*/
static void pci_exp_set_dev_base ( unsigned int base , int bus , int devfn )
2005-04-16 15:20:36 -07:00
{
2009-11-13 17:34:08 -07:00
u32 dev_base = base | PCI_MMCFG_BUS_OFFSET ( bus ) | ( devfn < < 12 ) ;
2006-12-23 10:00:43 +09:00
int cpu = smp_processor_id ( ) ;
if ( dev_base ! = mmcfg_last_accessed_device | |
cpu ! = mmcfg_last_accessed_cpu ) {
2005-04-16 15:20:36 -07:00
mmcfg_last_accessed_device = dev_base ;
2006-12-23 10:00:43 +09:00
mmcfg_last_accessed_cpu = cpu ;
2005-04-16 15:20:36 -07:00
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
2006-04-11 12:54:48 +02:00
if ( ( bus > 255 ) | | ( devfn > 255 ) | | ( reg > 4095 ) ) {
2008-01-14 17:31:09 -05:00
err : * value = - 1 ;
2005-04-16 15:20:36 -07:00
return - EINVAL ;
2006-04-07 19:50:15 +02:00
}
2005-04-16 15:20:36 -07:00
2012-06-22 14:55:12 +08:00
rcu_read_lock ( ) ;
2005-12-12 22:17:11 -08:00
base = get_base_addr ( seg , bus , devfn ) ;
2012-06-22 14:55:12 +08:00
if ( ! base ) {
rcu_read_unlock ( ) ;
2008-01-14 17:31:09 -05:00
goto err ;
2012-06-22 14:55:12 +08:00
}
2005-12-12 22:17:10 -08:00
2010-02-17 14:35:25 +00:00
raw_spin_lock_irqsave ( & pci_config_lock , flags ) ;
2005-04-16 15:20:36 -07:00
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 :
2007-08-10 22:30:59 +02:00
* value = mmio_config_readb ( mmcfg_virt_addr + reg ) ;
2005-04-16 15:20:36 -07:00
break ;
case 2 :
2007-08-10 22:30:59 +02:00
* value = mmio_config_readw ( mmcfg_virt_addr + reg ) ;
2005-04-16 15:20:36 -07:00
break ;
case 4 :
2007-08-10 22:30:59 +02:00
* value = mmio_config_readl ( mmcfg_virt_addr + reg ) ;
2005-04-16 15:20:36 -07:00
break ;
}
2010-02-17 14:35:25 +00:00
raw_spin_unlock_irqrestore ( & pci_config_lock , flags ) ;
2012-06-22 14:55:12 +08:00
rcu_read_unlock ( ) ;
2005-04-16 15:20:36 -07:00
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
2007-02-02 19:48:22 +03:00
if ( ( bus > 255 ) | | ( devfn > 255 ) | | ( reg > 4095 ) )
2005-04-16 15:20:36 -07:00
return - EINVAL ;
2012-06-22 14:55:12 +08:00
rcu_read_lock ( ) ;
2005-12-12 22:17:11 -08:00
base = get_base_addr ( seg , bus , devfn ) ;
2012-06-22 14:55:12 +08:00
if ( ! base ) {
rcu_read_unlock ( ) ;
2008-01-14 17:31:09 -05:00
return - EINVAL ;
2012-06-22 14:55:12 +08:00
}
2005-12-12 22:17:10 -08:00
2010-02-17 14:35:25 +00:00
raw_spin_lock_irqsave ( & pci_config_lock , flags ) ;
2005-04-16 15:20:36 -07:00
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 :
2007-08-12 02:23:16 -07:00
mmio_config_writeb ( mmcfg_virt_addr + reg , value ) ;
2005-04-16 15:20:36 -07:00
break ;
case 2 :
2007-08-12 02:23:16 -07:00
mmio_config_writew ( mmcfg_virt_addr + reg , value ) ;
2005-04-16 15:20:36 -07:00
break ;
case 4 :
2007-08-12 02:23:16 -07:00
mmio_config_writel ( mmcfg_virt_addr + reg , value ) ;
2005-04-16 15:20:36 -07:00
break ;
}
2010-02-17 14:35:25 +00:00
raw_spin_unlock_irqrestore ( & pci_config_lock , flags ) ;
2012-06-22 14:55:12 +08:00
rcu_read_unlock ( ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
2012-06-22 14:55:17 +08:00
const struct pci_raw_ops pci_mmcfg = {
2005-04-16 15:20:36 -07:00
. read = pci_mmcfg_read ,
. write = pci_mmcfg_write ,
} ;
2007-02-13 13:26:20 +01:00
int __init pci_mmcfg_arch_init ( void )
2005-12-12 22:17:11 -08:00
{
2008-02-10 09:45:28 -05:00
printk ( KERN_INFO " PCI: Using MMCONFIG for extended config space \n " ) ;
raw_pci_ext_ops = & pci_mmcfg ;
2007-02-13 13:26:20 +01:00
return 1 ;
2005-04-16 15:20:36 -07:00
}
2008-02-15 01:28:41 -08:00
void __init pci_mmcfg_arch_free ( void )
{
}
2012-06-22 14:55:13 +08:00
2012-12-21 14:02:53 -08:00
int pci_mmcfg_arch_map ( struct pci_mmcfg_region * cfg )
2012-06-22 14:55:13 +08:00
{
return 0 ;
}
void pci_mmcfg_arch_unmap ( struct pci_mmcfg_region * cfg )
{
unsigned long flags ;
/* Invalidate the cached mmcfg map entry. */
raw_spin_lock_irqsave ( & pci_config_lock , flags ) ;
mmcfg_last_accessed_device = 0 ;
raw_spin_unlock_irqrestore ( & pci_config_lock , flags ) ;
}