2010-02-04 10:59:27 -08:00
/*
2013-10-17 15:35:27 -07:00
* Intel MID PCI support
2010-02-04 10:59:27 -08:00
* Copyright ( c ) 2008 Intel Corporation
* Jesse Barnes < jesse . barnes @ intel . com >
*
* Moorestown has an interesting PCI implementation :
* - configuration space is memory mapped ( as defined by MCFG )
* - Lincroft devices also have a real , type 1 configuration space
* - Early Lincroft silicon has a type 1 access bug that will cause
* a hang if non - existent devices are accessed
* - some devices have the " fixed BAR " capability , which means
* they can ' t be relocated or modified ; check for that during
* BAR sizing
*
* So , we use the MCFG space for all reads and writes , but also send
* Lincroft writes to type 1 space . But only read / write if the device
* actually exists , otherwise return all 1 s for reads and bit bucket
* the writes .
*/
# include <linux/sched.h>
# include <linux/pci.h>
# include <linux/ioport.h>
# include <linux/init.h>
# include <linux/dmi.h>
2013-07-15 10:40:48 +03:00
# include <linux/acpi.h>
# include <linux/io.h>
# include <linux/smp.h>
2010-02-04 10:59:27 -08:00
# include <asm/segment.h>
# include <asm/pci_x86.h>
# include <asm/hw_irq.h>
# include <asm/io_apic.h>
2013-12-16 12:07:38 -08:00
# include <asm/intel-mid.h>
2010-02-04 10:59:27 -08:00
# define PCIE_CAP_OFFSET 0x100
2015-07-29 12:16:47 +03:00
/* Quirks for the listed devices */
# define PCI_DEVICE_ID_INTEL_MRFL_MMC 0x1190
2010-02-04 10:59:27 -08:00
/* Fixed BAR fields */
# define PCIE_VNDR_CAP_ID_FIXED_BAR 0x00 /* Fixed BAR (TBD) */
# define PCI_FIXED_BAR_0_SIZE 0x04
# define PCI_FIXED_BAR_1_SIZE 0x08
# define PCI_FIXED_BAR_2_SIZE 0x0c
# define PCI_FIXED_BAR_3_SIZE 0x10
# define PCI_FIXED_BAR_4_SIZE 0x14
# define PCI_FIXED_BAR_5_SIZE 0x1c
2013-07-15 10:40:48 +03:00
static int pci_soc_mode ;
2012-02-13 12:59:37 +00:00
2010-02-04 10:59:27 -08:00
/**
* fixed_bar_cap - return the offset of the fixed BAR cap if found
* @ bus : PCI bus
* @ devfn : device in question
*
* Look for the fixed BAR cap on @ bus and @ devfn , returning its offset
* if found or 0 otherwise .
*/
static int fixed_bar_cap ( struct pci_bus * bus , unsigned int devfn )
{
int pos ;
u32 pcie_cap = 0 , cap_data ;
pos = PCIE_CAP_OFFSET ;
2010-02-24 09:42:50 -08:00
if ( ! raw_pci_ext_ops )
return 0 ;
2010-02-04 10:59:27 -08:00
while ( pos ) {
if ( raw_pci_ext_ops - > read ( pci_domain_nr ( bus ) , bus - > number ,
devfn , pos , 4 , & pcie_cap ) )
return 0 ;
2010-07-16 11:58:26 -07:00
if ( PCI_EXT_CAP_ID ( pcie_cap ) = = 0x0000 | |
PCI_EXT_CAP_ID ( pcie_cap ) = = 0xffff )
break ;
2010-02-04 10:59:27 -08:00
if ( PCI_EXT_CAP_ID ( pcie_cap ) = = PCI_EXT_CAP_ID_VNDR ) {
raw_pci_ext_ops - > read ( pci_domain_nr ( bus ) , bus - > number ,
devfn , pos + 4 , 4 , & cap_data ) ;
if ( ( cap_data & 0xffff ) = = PCIE_VNDR_CAP_ID_FIXED_BAR )
return pos ;
}
2010-07-16 11:58:26 -07:00
pos = PCI_EXT_CAP_NEXT ( pcie_cap ) ;
2010-02-04 10:59:27 -08:00
}
return 0 ;
}
static int pci_device_update_fixed ( struct pci_bus * bus , unsigned int devfn ,
int reg , int len , u32 val , int offset )
{
u32 size ;
unsigned int domain , busnum ;
int bar = ( reg - PCI_BASE_ADDRESS_0 ) > > 2 ;
domain = pci_domain_nr ( bus ) ;
busnum = bus - > number ;
if ( val = = ~ 0 & & len = = 4 ) {
unsigned long decode ;
raw_pci_ext_ops - > read ( domain , busnum , devfn ,
offset + 8 + ( bar * 4 ) , 4 , & size ) ;
/* Turn the size into a decode pattern for the sizing code */
if ( size ) {
decode = size - 1 ;
decode | = decode > > 1 ;
decode | = decode > > 2 ;
decode | = decode > > 4 ;
decode | = decode > > 8 ;
decode | = decode > > 16 ;
decode + + ;
decode = ~ ( decode - 1 ) ;
} else {
2010-05-14 14:41:14 -07:00
decode = 0 ;
2010-02-04 10:59:27 -08:00
}
/*
* If val is all ones , the core code is trying to size the reg ,
* so update the mmconfig space with the real size .
*
* Note : this assumes the fixed size we got is a power of two .
*/
return raw_pci_ext_ops - > write ( domain , busnum , devfn , reg , 4 ,
decode ) ;
}
/* This is some other kind of BAR write, so just do it. */
return raw_pci_ext_ops - > write ( domain , busnum , devfn , reg , len , val ) ;
}
/**
* type1_access_ok - check whether to use type 1
* @ bus : bus number
* @ devfn : device & function in question
*
* If the bus is on a Lincroft chip and it exists , or is not on a Lincroft at
* all , the we can go ahead with any reads & writes . If it ' s on a Lincroft ,
* but doesn ' t exist , avoid the access altogether to keep the chip from
* hanging .
*/
static bool type1_access_ok ( unsigned int bus , unsigned int devfn , int reg )
{
2013-07-15 10:40:48 +03:00
/*
* This is a workaround for A0 LNC bug where PCI status register does
2010-02-04 10:59:27 -08:00
* not have new CAP bit set . can not be written by SW either .
*
* PCI header type in real LNC indicates a single function device , this
* will prevent probing other devices under the same function in PCI
* shim . Therefore , use the header type in shim instead .
*/
if ( reg > = 0x100 | | reg = = PCI_STATUS | | reg = = PCI_HEADER_TYPE )
2013-10-17 15:35:28 -07:00
return false ;
2013-05-20 10:20:21 -06:00
if ( bus = = 0 & & ( devfn = = PCI_DEVFN ( 2 , 0 )
| | devfn = = PCI_DEVFN ( 0 , 0 )
| | devfn = = PCI_DEVFN ( 3 , 0 ) ) )
2013-10-17 15:35:28 -07:00
return true ;
return false ; /* Langwell on others */
2010-02-04 10:59:27 -08:00
}
static int pci_read ( struct pci_bus * bus , unsigned int devfn , int where ,
int size , u32 * value )
{
if ( type1_access_ok ( bus - > number , devfn , where ) )
return pci_direct_conf1 . read ( pci_domain_nr ( bus ) , bus - > number ,
devfn , where , size , value ) ;
return raw_pci_ext_ops - > read ( pci_domain_nr ( bus ) , bus - > number ,
devfn , where , size , value ) ;
}
static int pci_write ( struct pci_bus * bus , unsigned int devfn , int where ,
int size , u32 value )
{
int offset ;
2013-07-15 10:40:48 +03:00
/*
* On MRST , there is no PCI ROM BAR , this will cause a subsequent read
2010-02-04 10:59:27 -08:00
* to ROM BAR return 0 then being ignored .
*/
if ( where = = PCI_ROM_ADDRESS )
return 0 ;
/*
* Devices with fixed BARs need special handling :
* - BAR sizing code will save , write ~ 0 , read size , restore
* - so writes to fixed BARs need special handling
* - other writes to fixed BAR devices should go through mmconfig
*/
offset = fixed_bar_cap ( bus , devfn ) ;
if ( offset & &
( where > = PCI_BASE_ADDRESS_0 & & where < = PCI_BASE_ADDRESS_5 ) ) {
return pci_device_update_fixed ( bus , devfn , where , size , value ,
offset ) ;
}
/*
* On Moorestown update both real & mmconfig space
* Note : early Lincroft silicon can ' t handle type 1 accesses to
* non - existent devices , so just eat the write in that case .
*/
if ( type1_access_ok ( bus - > number , devfn , where ) )
return pci_direct_conf1 . write ( pci_domain_nr ( bus ) , bus - > number ,
devfn , where , size , value ) ;
return raw_pci_ext_ops - > write ( pci_domain_nr ( bus ) , bus - > number , devfn ,
where , size , value ) ;
}
2013-10-17 15:35:29 -07:00
static int intel_mid_pci_irq_enable ( struct pci_dev * dev )
2010-02-04 10:59:27 -08:00
{
2015-04-13 14:11:54 +08:00
struct irq_alloc_info info ;
2014-06-09 16:20:01 +08:00
int polarity ;
2015-07-29 12:16:48 +03:00
int ret ;
2010-02-04 10:59:27 -08:00
2014-10-27 13:21:42 +08:00
if ( dev - > irq_managed & & dev - > irq > 0 )
return 0 ;
2015-07-29 12:16:47 +03:00
switch ( intel_mid_identify_cpu ( ) ) {
case INTEL_MID_CPU_CHIP_TANGIER :
2014-06-09 16:20:01 +08:00
polarity = 0 ; /* active high */
2015-07-29 12:16:47 +03:00
/* Special treatment for IRQ0 */
if ( dev - > irq = = 0 ) {
/*
* TNG has IRQ0 assigned to eMMC controller . But there
* are also other devices with bogus PCI configuration
* that have IRQ0 assigned . This check ensures that
* eMMC gets it .
*/
if ( dev - > device ! = PCI_DEVICE_ID_INTEL_MRFL_MMC )
return - EBUSY ;
}
break ;
default :
2014-06-09 16:20:01 +08:00
polarity = 1 ; /* active low */
2015-07-29 12:16:47 +03:00
break ;
}
2015-04-13 14:11:54 +08:00
ioapic_set_alloc_attr ( & info , dev_to_node ( & dev - > dev ) , 1 , polarity ) ;
2010-02-04 10:59:27 -08:00
2013-07-15 10:40:48 +03:00
/*
* MRST only have IOAPIC , the PCI irq lines are 1 : 1 mapped to
2010-02-04 10:59:27 -08:00
* IOAPIC RTE entries , so we just enable RTE for the device .
*/
2015-07-29 12:16:48 +03:00
ret = mp_map_gsi_to_irq ( dev - > irq , IOAPIC_MAP_ALLOC , & info ) ;
if ( ret < 0 )
return ret ;
2014-06-09 16:19:56 +08:00
2014-10-27 13:21:42 +08:00
dev - > irq_managed = 1 ;
2010-02-04 10:59:27 -08:00
return 0 ;
}
2014-06-09 16:20:09 +08:00
static void intel_mid_pci_irq_disable ( struct pci_dev * dev )
{
2015-03-20 14:56:19 +01:00
if ( ! mp_should_keep_irq ( & dev - > dev ) & & dev - > irq_managed & &
dev - > irq > 0 ) {
2014-06-09 16:20:09 +08:00
mp_unmap_irq ( dev - > irq ) ;
2014-10-27 13:21:42 +08:00
dev - > irq_managed = 0 ;
}
2014-06-09 16:20:09 +08:00
}
2015-07-29 12:16:49 +03:00
static struct pci_ops intel_mid_pci_ops = {
2010-02-04 10:59:27 -08:00
. read = pci_read ,
. write = pci_write ,
} ;
/**
2013-10-17 15:35:29 -07:00
* intel_mid_pci_init - installs intel_mid_pci_ops
2010-02-04 10:59:27 -08:00
*
* Moorestown has an interesting PCI implementation ( see above ) .
* Called when the early platform detection installs it .
*/
2013-10-17 15:35:29 -07:00
int __init intel_mid_pci_init ( void )
2010-02-04 10:59:27 -08:00
{
2013-07-15 10:40:48 +03:00
pr_info ( " Intel MID platform detected, using MID PCI ops \n " ) ;
2010-02-04 10:59:27 -08:00
pci_mmcfg_late_init ( ) ;
2013-10-17 15:35:29 -07:00
pcibios_enable_irq = intel_mid_pci_irq_enable ;
2014-06-09 16:20:09 +08:00
pcibios_disable_irq = intel_mid_pci_irq_disable ;
2013-10-17 15:35:29 -07:00
pci_root_ops = intel_mid_pci_ops ;
2012-02-13 12:59:37 +00:00
pci_soc_mode = 1 ;
2010-02-04 10:59:27 -08:00
/* Continue with standard init */
return 1 ;
}
2013-07-15 10:40:48 +03:00
/*
* Langwell devices are not true PCI devices ; they are not subject to 10 ms
* d3 to d0 delay required by PCI spec .
2012-02-13 12:59:00 +00:00
*/
2012-12-21 14:02:53 -08:00
static void pci_d3delay_fixup ( struct pci_dev * dev )
2012-02-13 12:59:00 +00:00
{
2013-07-15 10:40:48 +03:00
/*
* PCI fixups are effectively decided compile time . If we have a dual
* SoC / non - SoC kernel we don ' t want to mangle d3 on non - SoC devices .
*/
if ( ! pci_soc_mode )
return ;
/*
* True PCI devices in Lincroft should allow type 1 access , the rest
* are Langwell fake PCI devices .
2012-02-13 12:59:00 +00:00
*/
if ( type1_access_ok ( dev - > bus - > number , dev - > devfn , PCI_DEVICE_ID ) )
return ;
dev - > d3_delay = 0 ;
}
DECLARE_PCI_FIXUP_FINAL ( PCI_VENDOR_ID_INTEL , PCI_ANY_ID , pci_d3delay_fixup ) ;
2012-12-21 14:02:53 -08:00
static void mrst_power_off_unused_dev ( struct pci_dev * dev )
2012-02-13 12:59:00 +00:00
{
2012-06-23 10:23:50 +08:00
pci_set_power_state ( dev , PCI_D3hot ) ;
2012-02-13 12:59:00 +00:00
}
DECLARE_PCI_FIXUP_FINAL ( PCI_VENDOR_ID_INTEL , 0x0801 , mrst_power_off_unused_dev ) ;
DECLARE_PCI_FIXUP_FINAL ( PCI_VENDOR_ID_INTEL , 0x0809 , mrst_power_off_unused_dev ) ;
DECLARE_PCI_FIXUP_FINAL ( PCI_VENDOR_ID_INTEL , 0x080C , mrst_power_off_unused_dev ) ;
DECLARE_PCI_FIXUP_FINAL ( PCI_VENDOR_ID_INTEL , 0x0815 , mrst_power_off_unused_dev ) ;
2010-02-04 10:59:27 -08:00
/*
* Langwell devices reside at fixed offsets , don ' t try to move them .
*/
2012-12-21 14:02:53 -08:00
static void pci_fixed_bar_fixup ( struct pci_dev * dev )
2010-02-04 10:59:27 -08:00
{
unsigned long offset ;
u32 size ;
int i ;
2012-02-13 12:59:37 +00:00
if ( ! pci_soc_mode )
return ;
2010-05-14 13:55:57 -07:00
/* Must have extended configuration space */
if ( dev - > cfg_size < PCIE_CAP_OFFSET + 4 )
return ;
2010-02-04 10:59:27 -08:00
/* Fixup the BAR sizes for fixed BAR devices and make them unmoveable */
offset = fixed_bar_cap ( dev - > bus , dev - > devfn ) ;
if ( ! offset | | PCI_DEVFN ( 2 , 0 ) = = dev - > devfn | |
PCI_DEVFN ( 2 , 2 ) = = dev - > devfn )
return ;
for ( i = 0 ; i < PCI_ROM_RESOURCE ; i + + ) {
pci_read_config_dword ( dev , offset + 8 + ( i * 4 ) , & size ) ;
dev - > resource [ i ] . end = dev - > resource [ i ] . start + size - 1 ;
dev - > resource [ i ] . flags | = IORESOURCE_PCI_FIXED ;
}
}
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_INTEL , PCI_ANY_ID , pci_fixed_bar_fixup ) ;