2005-04-16 15:20:36 -07:00
/*
* This program is free software ; you can redistribute it and / or modify it
* under the terms of the GNU General Public License as published by the
* Free Software Foundation ; either version 2 of the License , or ( at your
* option ) any later version .
*
* Copyright ( C ) 2003 , 04 Ralf Baechle ( ralf @ linux - mips . org )
*/
# include <linux/kernel.h>
# include <linux/mm.h>
# include <linux/bootmem.h>
# include <linux/init.h>
# include <linux/types.h>
# include <linux/pci.h>
/*
* Indicate whether we respect the PCI setup left by the firmware .
*
* Make this long - lived so that we know when shutting down
* whether we probed only or not .
*/
int pci_probe_only ;
# define PCI_ASSIGN_ALL_BUSSES 1
unsigned int pci_probe = PCI_ASSIGN_ALL_BUSSES ;
/*
* The PCI controller list .
*/
2008-06-18 10:18:20 +03:00
static struct pci_controller * hose_head , * * hose_tail = & hose_head ;
2005-04-16 15:20:36 -07:00
2009-09-17 02:25:07 +02:00
unsigned long PCIBIOS_MIN_IO ;
unsigned long PCIBIOS_MIN_MEM ;
2005-04-16 15:20:36 -07:00
2008-10-14 11:45:09 +02:00
static int pci_initialized ;
2005-04-16 15:20:36 -07:00
/*
* We need to avoid collisions with ` mirrored ' VGA ports
* and other strange ISA hardware , so we always want the
* addresses to be allocated in the 0x000 - 0x0ff region
* modulo 0x400 .
*
* Why ? Because some silly external IO cards only decode
* the low 10 bits of the IO address . The 0x00 - 0xff region
* is reserved for motherboard devices that decode all 16
* bits , so it ' s ok to allocate at , say , 0x2800 - 0x28ff ,
* but we want to try to avoid allocating at 0x2900 - 0x2bff
* which might have be mirrored at 0x0100 - 0x03ff . .
*/
2010-01-01 17:40:49 +01:00
resource_size_t
2010-01-01 17:40:50 +01:00
pcibios_align_resource ( void * data , const struct resource * res ,
2006-06-12 17:06:02 -07:00
resource_size_t size , resource_size_t align )
2005-04-16 15:20:36 -07:00
{
struct pci_dev * dev = data ;
struct pci_controller * hose = dev - > sysdata ;
2006-06-12 17:06:02 -07:00
resource_size_t start = res - > start ;
2005-04-16 15:20:36 -07:00
if ( res - > flags & IORESOURCE_IO ) {
/* Make sure we start at our min on all hoses */
if ( start < PCIBIOS_MIN_IO + hose - > io_resource - > start )
start = PCIBIOS_MIN_IO + hose - > io_resource - > start ;
/*
* Put everything into 0x00 - 0xff region modulo 0x400
*/
if ( start & 0x300 )
start = ( start + 0x3ff ) & ~ 0x3ff ;
} else if ( res - > flags & IORESOURCE_MEM ) {
/* Make sure we start at our min on all hoses */
if ( start < PCIBIOS_MIN_MEM + hose - > mem_resource - > start )
start = PCIBIOS_MIN_MEM + hose - > mem_resource - > start ;
}
2010-01-01 17:40:49 +01:00
return start ;
2005-04-16 15:20:36 -07:00
}
2008-10-14 11:45:09 +02:00
static void __devinit pcibios_scanbus ( struct pci_controller * hose )
{
static int next_busno ;
static int need_domain_info ;
struct pci_bus * bus ;
if ( ! hose - > iommu )
PCI_DMA_BUS_IS_PHYS = 1 ;
if ( hose - > get_busno & & pci_probe_only )
next_busno = ( * hose - > get_busno ) ( ) ;
bus = pci_scan_bus ( next_busno , hose - > pci_ops , hose ) ;
hose - > bus = bus ;
need_domain_info = need_domain_info | | hose - > index ;
hose - > need_domain_info = need_domain_info ;
if ( bus ) {
next_busno = bus - > subordinate + 1 ;
/* Don't allow 8-bit bus number overflow inside the hose -
reserve some space for bridges . */
if ( next_busno > 224 ) {
next_busno = 0 ;
need_domain_info = 1 ;
}
if ( ! pci_probe_only ) {
pci_bus_size_bridges ( bus ) ;
pci_bus_assign_resources ( bus ) ;
pci_enable_bridges ( bus ) ;
}
}
}
static DEFINE_MUTEX ( pci_scan_mutex ) ;
2007-08-24 02:13:33 +01:00
void __devinit register_pci_controller ( struct pci_controller * hose )
2005-04-16 15:20:36 -07:00
{
2007-04-08 13:28:44 +02:00
if ( request_resource ( & iomem_resource , hose - > mem_resource ) < 0 )
goto out ;
if ( request_resource ( & ioport_resource , hose - > io_resource ) < 0 ) {
release_resource ( hose - > mem_resource ) ;
goto out ;
}
2005-04-16 15:20:36 -07:00
* hose_tail = hose ;
hose_tail = & hose - > next ;
2006-12-07 15:35:43 +01:00
/*
* Do not panic here but later - this might hapen before console init .
*/
if ( ! hose - > io_map_base ) {
printk ( KERN_WARNING
" registering PCI controller with io_map_base unset \n " ) ;
}
2008-10-14 11:45:09 +02:00
/*
* Scan the bus if it is register after the PCI subsystem
* initialization .
*/
if ( pci_initialized ) {
mutex_lock ( & pci_scan_mutex ) ;
pcibios_scanbus ( hose ) ;
mutex_unlock ( & pci_scan_mutex ) ;
}
2007-04-08 13:28:44 +02:00
return ;
out :
printk ( KERN_WARNING
" Skipping PCI bus scan due to resource conflict \n " ) ;
2005-04-16 15:20:36 -07:00
}
static int __init pcibios_init ( void )
{
struct pci_controller * hose ;
/* Scan all of the recorded PCI controllers. */
2008-10-14 11:45:09 +02:00
for ( hose = hose_head ; hose ; hose = hose - > next )
pcibios_scanbus ( hose ) ;
2005-04-16 15:20:36 -07:00
2008-12-16 21:37:10 -07:00
pci_fixup_irqs ( pci_common_swizzle , pcibios_map_irq ) ;
2005-04-16 15:20:36 -07:00
2008-10-14 11:45:09 +02:00
pci_initialized = 1 ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
subsys_initcall ( pcibios_init ) ;
static int pcibios_enable_resources ( struct pci_dev * dev , int mask )
{
u16 cmd , old_cmd ;
int idx ;
struct resource * r ;
pci_read_config_word ( dev , PCI_COMMAND , & cmd ) ;
old_cmd = cmd ;
2005-07-12 09:18:53 +00:00
for ( idx = 0 ; idx < PCI_NUM_RESOURCES ; idx + + ) {
2005-04-16 15:20:36 -07:00
/* Only set up the requested stuff */
if ( ! ( mask & ( 1 < < idx ) ) )
continue ;
r = & dev - > resource [ idx ] ;
2008-02-19 15:59:33 +00:00
if ( ! ( r - > flags & ( IORESOURCE_IO | IORESOURCE_MEM ) ) )
continue ;
if ( ( idx = = PCI_ROM_RESOURCE ) & &
( ! ( r - > flags & IORESOURCE_ROM_ENABLE ) ) )
continue ;
2005-04-16 15:20:36 -07:00
if ( ! r - > start & & r - > end ) {
2008-02-19 16:01:20 +00:00
printk ( KERN_ERR " PCI: Device %s not available "
" because of resource collisions \n " ,
pci_name ( dev ) ) ;
2005-04-16 15:20:36 -07:00
return - EINVAL ;
}
if ( r - > flags & IORESOURCE_IO )
cmd | = PCI_COMMAND_IO ;
if ( r - > flags & IORESOURCE_MEM )
cmd | = PCI_COMMAND_MEMORY ;
}
if ( cmd ! = old_cmd ) {
2008-02-19 16:01:20 +00:00
printk ( " PCI: Enabling device %s (%04x -> %04x) \n " ,
pci_name ( dev ) , old_cmd , cmd ) ;
2005-04-16 15:20:36 -07:00
pci_write_config_word ( dev , PCI_COMMAND , cmd ) ;
}
return 0 ;
}
/*
* If we set up a device for bus mastering , we need to check the latency
* timer as certain crappy BIOSes forget to set it properly .
*/
2008-07-15 19:57:30 +03:00
static unsigned int pcibios_max_latency = 255 ;
2005-04-16 15:20:36 -07:00
void pcibios_set_master ( struct pci_dev * dev )
{
u8 lat ;
pci_read_config_byte ( dev , PCI_LATENCY_TIMER , & lat ) ;
if ( lat < 16 )
lat = ( 64 < = pcibios_max_latency ) ? 64 : pcibios_max_latency ;
else if ( lat > pcibios_max_latency )
lat = pcibios_max_latency ;
else
return ;
printk ( KERN_DEBUG " PCI: Setting latency timer of device %s to %d \n " ,
pci_name ( dev ) , lat ) ;
pci_write_config_byte ( dev , PCI_LATENCY_TIMER , lat ) ;
}
unsigned int pcibios_assign_all_busses ( void )
{
return ( pci_probe & PCI_ASSIGN_ALL_BUSSES ) ? 1 : 0 ;
}
int pcibios_enable_device ( struct pci_dev * dev , int mask )
{
int err ;
if ( ( err = pcibios_enable_resources ( dev , mask ) ) < 0 )
return err ;
return pcibios_plat_dev_init ( dev ) ;
}
2007-08-23 14:17:14 +01:00
static void pcibios_fixup_device_resources ( struct pci_dev * dev ,
2005-04-16 15:20:36 -07:00
struct pci_bus * bus )
{
/* Update device resources. */
struct pci_controller * hose = ( struct pci_controller * ) bus - > sysdata ;
unsigned long offset = 0 ;
int i ;
for ( i = 0 ; i < PCI_NUM_RESOURCES ; i + + ) {
if ( ! dev - > resource [ i ] . start )
continue ;
if ( dev - > resource [ i ] . flags & IORESOURCE_IO )
offset = hose - > io_offset ;
else if ( dev - > resource [ i ] . flags & IORESOURCE_MEM )
offset = hose - > mem_offset ;
dev - > resource [ i ] . start + = offset ;
dev - > resource [ i ] . end + = offset ;
}
}
2008-03-08 09:56:28 +00:00
void __devinit pcibios_fixup_bus ( struct pci_bus * bus )
2005-04-16 15:20:36 -07:00
{
/* Propagate hose info into the subordinate devices. */
struct pci_controller * hose = bus - > sysdata ;
struct list_head * ln ;
struct pci_dev * dev = bus - > self ;
if ( ! dev ) {
bus - > resource [ 0 ] = hose - > io_resource ;
bus - > resource [ 1 ] = hose - > mem_resource ;
} else if ( pci_probe_only & &
( dev - > class > > 8 ) = = PCI_CLASS_BRIDGE_PCI ) {
pci_read_bridge_bases ( bus ) ;
pcibios_fixup_device_resources ( dev , bus ) ;
2005-09-03 15:56:17 -07:00
}
2005-04-16 15:20:36 -07:00
for ( ln = bus - > devices . next ; ln ! = & bus - > devices ; ln = ln - > next ) {
2007-07-13 01:26:52 +09:00
dev = pci_dev_b ( ln ) ;
2005-04-16 15:20:36 -07:00
if ( ( dev - > class > > 8 ) ! = PCI_CLASS_BRIDGE_PCI )
pcibios_fixup_device_resources ( dev , bus ) ;
}
}
void __init
pcibios_update_irq ( struct pci_dev * dev , int irq )
{
pci_write_config_byte ( dev , PCI_INTERRUPT_LINE , irq ) ;
}
2007-08-23 14:17:14 +01:00
void pcibios_resource_to_bus ( struct pci_dev * dev , struct pci_bus_region * region ,
2005-04-16 15:20:36 -07:00
struct resource * res )
{
struct pci_controller * hose = ( struct pci_controller * ) dev - > sysdata ;
unsigned long offset = 0 ;
if ( res - > flags & IORESOURCE_IO )
offset = hose - > io_offset ;
else if ( res - > flags & IORESOURCE_MEM )
offset = hose - > mem_offset ;
region - > start = res - > start - offset ;
region - > end = res - > end - offset ;
}
2005-09-03 15:56:20 -07:00
void __devinit
pcibios_bus_to_resource ( struct pci_dev * dev , struct resource * res ,
struct pci_bus_region * region )
{
struct pci_controller * hose = ( struct pci_controller * ) dev - > sysdata ;
unsigned long offset = 0 ;
if ( res - > flags & IORESOURCE_IO )
offset = hose - > io_offset ;
else if ( res - > flags & IORESOURCE_MEM )
offset = hose - > mem_offset ;
res - > start = region - > start + offset ;
res - > end = region - > end + offset ;
}
2005-04-16 15:20:36 -07:00
# ifdef CONFIG_HOTPLUG
EXPORT_SYMBOL ( pcibios_resource_to_bus ) ;
2005-09-03 15:56:20 -07:00
EXPORT_SYMBOL ( pcibios_bus_to_resource ) ;
2005-04-16 15:20:36 -07:00
EXPORT_SYMBOL ( PCIBIOS_MIN_IO ) ;
EXPORT_SYMBOL ( PCIBIOS_MIN_MEM ) ;
# endif
2008-12-09 17:58:46 +00:00
int pci_mmap_page_range ( struct pci_dev * dev , struct vm_area_struct * vma ,
enum pci_mmap_state mmap_state , int write_combine )
{
unsigned long prot ;
/*
* I / O space can be accessed via normal processor loads and stores on
* this platform but for now we elect not to do this and portable
* drivers should not do this anyway .
*/
if ( mmap_state = = pci_mmap_io )
return - EINVAL ;
/*
* Ignore write - combine ; for now only return uncached mappings .
*/
prot = pgprot_val ( vma - > vm_page_prot ) ;
prot = ( prot & ~ _CACHE_MASK ) | _CACHE_UNCACHED ;
vma - > vm_page_prot = __pgprot ( prot ) ;
return remap_pfn_range ( vma , vma - > vm_start , vma - > vm_pgoff ,
vma - > vm_end - vma - > vm_start , vma - > vm_page_prot ) ;
}
2008-07-24 00:25:14 +09:00
char * ( * pcibios_plat_setup ) ( char * str ) __devinitdata ;
char * __devinit pcibios_setup ( char * str )
2005-04-16 15:20:36 -07:00
{
2008-07-24 00:25:14 +09:00
if ( pcibios_plat_setup )
return pcibios_plat_setup ( str ) ;
2005-04-16 15:20:36 -07:00
return str ;
}