2005-04-16 15:20:36 -07:00
/*
* mmconfig . c - Low - level direct PCI config space access via MMCONFIG
2007-02-02 19:48:22 +03:00
*
2005-04-16 15:20:36 -07:00
* This is an 64 bit optimized version that always keeps the full mmconfig
* space mapped . This allows lockless config space operation .
*/
# include <linux/pci.h>
# include <linux/init.h>
2005-06-23 17:35:56 -07:00
# include <linux/acpi.h>
2005-12-12 22:17:11 -08:00
# include <linux/bitmap.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>
2005-04-16 15:20:36 -07:00
2009-11-13 17:34:59 -07:00
# define PREFIX "PCI: "
2005-12-15 09:17:44 +00:00
static char __iomem * pci_dev_base ( unsigned int seg , unsigned int bus , unsigned int 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 ) ;
2008-01-14 17:31:09 -05:00
2009-11-13 17:35:04 -07:00
if ( cfg & & cfg - > virt )
return cfg - > virt + ( PCI_MMCFG_BUS_OFFSET ( bus ) | ( devfn < < 12 ) ) ;
return NULL ;
2005-04-16 15:20:36 -07:00
}
static int pci_mmcfg_read ( unsigned int seg , unsigned int bus ,
unsigned int devfn , int reg , int len , u32 * value )
{
2005-12-15 09:17:44 +00:00
char __iomem * addr ;
2005-04-16 15:20:36 -07:00
2005-12-12 22:17:10 -08:00
/* Why do we have this when nobody checks it. How about a BUG()!? -AK */
2006-04-11 12:54:48 +02:00
if ( unlikely ( ( 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:10 -08:00
addr = pci_dev_base ( seg , bus , devfn ) ;
2012-06-22 14:55:12 +08:00
if ( ! addr ) {
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
2005-04-16 15:20:36 -07:00
switch ( len ) {
case 1 :
2007-08-10 22:30:59 +02:00
* value = mmio_config_readb ( 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 ( 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 ( addr + reg ) ;
2005-04-16 15:20:36 -07:00
break ;
}
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 )
{
2005-12-15 09:17:44 +00:00
char __iomem * addr ;
2005-04-16 15:20:36 -07:00
2005-12-12 22:17:10 -08:00
/* Why do we have this when nobody checks it. How about a BUG()!? -AK */
2005-04-16 15:20:36 -07:00
if ( unlikely ( ( bus > 255 ) | | ( devfn > 255 ) | | ( reg > 4095 ) ) )
return - EINVAL ;
2012-06-22 14:55:12 +08:00
rcu_read_lock ( ) ;
2005-12-12 22:17:10 -08:00
addr = pci_dev_base ( seg , bus , devfn ) ;
2012-06-22 14:55:12 +08:00
if ( ! addr ) {
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
2005-04-16 15:20:36 -07:00
switch ( len ) {
case 1 :
2007-08-10 22:30:59 +02:00
mmio_config_writeb ( addr + reg , value ) ;
2005-04-16 15:20:36 -07:00
break ;
case 2 :
2007-08-10 22:30:59 +02:00
mmio_config_writew ( addr + reg , value ) ;
2005-04-16 15:20:36 -07:00
break ;
case 4 :
2007-08-10 22:30:59 +02:00
mmio_config_writel ( addr + reg , value ) ;
2005-04-16 15:20:36 -07:00
break ;
}
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 ,
} ;
2012-12-21 14:02:53 -08:00
static void __iomem * mcfg_ioremap ( struct pci_mmcfg_region * cfg )
2007-02-13 13:26:20 +01:00
{
void __iomem * addr ;
2009-03-19 20:55:35 -07:00
u64 start , size ;
2009-11-13 17:34:08 -07:00
int num_buses ;
2009-03-19 20:55:35 -07:00
2009-11-13 17:34:18 -07:00
start = cfg - > address + PCI_MMCFG_BUS_OFFSET ( cfg - > start_bus ) ;
num_buses = cfg - > end_bus - cfg - > start_bus + 1 ;
2009-11-13 17:34:08 -07:00
size = PCI_MMCFG_BUS_OFFSET ( num_buses ) ;
2009-03-19 20:55:35 -07:00
addr = ioremap_nocache ( start , size ) ;
2009-11-13 17:34:59 -07:00
if ( addr )
2009-11-13 17:34:18 -07:00
addr - = PCI_MMCFG_BUS_OFFSET ( cfg - > start_bus ) ;
2007-02-13 13:26:20 +01:00
return addr ;
}
2007-02-13 13:26:20 +01:00
int __init pci_mmcfg_arch_init ( void )
2005-04-16 15:20:36 -07:00
{
2009-11-13 17:34:39 -07:00
struct pci_mmcfg_region * cfg ;
2007-02-13 13:26:20 +01:00
2012-06-22 14:55:13 +08:00
list_for_each_entry ( cfg , & pci_mmcfg_list , list )
if ( pci_mmcfg_arch_map ( cfg ) ) {
2008-02-15 01:28:41 -08:00
pci_mmcfg_arch_free ( ) ;
2007-02-13 13:26:20 +01:00
return 0 ;
2005-06-23 17:35:56 -07:00
}
2012-06-22 14:55:13 +08:00
2008-02-10 09:45:28 -05:00
raw_pci_ext_ops = & pci_mmcfg ;
2012-06-22 14:55:13 +08:00
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 )
{
2009-11-13 17:34:39 -07:00
struct pci_mmcfg_region * cfg ;
2008-02-15 01:28:41 -08:00
2012-06-22 14:55:13 +08:00
list_for_each_entry ( cfg , & pci_mmcfg_list , list )
pci_mmcfg_arch_unmap ( cfg ) ;
}
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
{
cfg - > virt = mcfg_ioremap ( cfg ) ;
if ( ! cfg - > virt ) {
2012-06-22 14:55:22 +08:00
pr_err ( PREFIX " can't map MMCONFIG at %pR \n " , & cfg - > res ) ;
2012-06-22 14:55:13 +08:00
return - ENOMEM ;
}
return 0 ;
}
void pci_mmcfg_arch_unmap ( struct pci_mmcfg_region * cfg )
{
if ( cfg & & cfg - > virt ) {
iounmap ( cfg - > virt + PCI_MMCFG_BUS_OFFSET ( cfg - > start_bus ) ) ;
cfg - > virt = NULL ;
2008-02-15 01:28:41 -08:00
}
}