2018-01-26 20:45:16 +03:00
// SPDX-License-Identifier: GPL-2.0
2005-04-17 02:20:36 +04:00
/*
2018-03-10 01:36:33 +03:00
* Support routines for initializing a PCI subsystem
2005-04-17 02:20:36 +04:00
*
* Extruded from code written by
* Dave Rusling ( david . rusling @ reo . mts . dec . com )
* David Mosberger ( davidm @ cs . arizona . edu )
* David Miller ( davem @ redhat . com )
*/
# include <linux/kernel.h>
# include <linux/pci.h>
# include <linux/errno.h>
# include <linux/ioport.h>
# include <linux/cache.h>
2017-06-28 23:14:02 +03:00
# include "pci.h"
2005-04-17 02:20:36 +04:00
2017-06-28 23:14:02 +03:00
void pci_assign_irq ( struct pci_dev * dev )
2005-04-17 02:20:36 +04:00
{
2017-06-28 23:14:02 +03:00
u8 pin ;
u8 slot = - 1 ;
2007-02-06 03:36:07 +03:00
int irq = 0 ;
2017-06-28 23:14:02 +03:00
struct pci_host_bridge * hbrg = pci_find_host_bridge ( dev - > bus ) ;
if ( ! ( hbrg - > map_irq ) ) {
2018-01-18 21:55:24 +03:00
pci_dbg ( dev , " runtime IRQ mapping not provided by arch \n " ) ;
2017-06-28 23:14:02 +03:00
return ;
}
2005-04-17 02:20:36 +04:00
2021-09-18 23:18:02 +03:00
/*
* If this device is not on the primary bus , we need to figure out
* which interrupt pin it will come in on . We know which slot it
* will come in on because that slot is where the bridge is . Each
* time the interrupt line passes through a PCI - PCI bridge we must
* apply the swizzle function .
*/
2005-04-17 02:20:36 +04:00
pci_read_config_byte ( dev , PCI_INTERRUPT_PIN , & pin ) ;
2007-02-06 03:36:07 +03:00
/* Cope with illegal. */
if ( pin > 4 )
2005-04-17 02:20:36 +04:00
pin = 1 ;
2017-06-28 23:14:02 +03:00
if ( pin ) {
2021-09-18 23:18:02 +03:00
/* Follow the chain of bridges, swizzling as we go. */
2017-06-28 23:14:02 +03:00
if ( hbrg - > swizzle_irq )
slot = ( * ( hbrg - > swizzle_irq ) ) ( dev , & pin ) ;
2005-04-17 02:20:36 +04:00
2017-06-28 23:14:02 +03:00
/*
2021-09-18 23:18:02 +03:00
* If a swizzling function is not used , map_irq ( ) must
* ignore slot .
2017-06-28 23:14:02 +03:00
*/
irq = ( * ( hbrg - > map_irq ) ) ( dev , slot , pin ) ;
2007-02-06 03:36:07 +03:00
if ( irq = = - 1 )
irq = 0 ;
}
2005-04-17 02:20:36 +04:00
dev - > irq = irq ;
2018-01-18 21:55:24 +03:00
pci_dbg ( dev , " assign IRQ: got %d \n " , dev - > irq ) ;
2005-04-17 02:20:36 +04:00
2021-09-18 23:18:02 +03:00
/*
* Always tell the device , so the driver knows what is the real IRQ
* to use ; the device does not use it .
*/
2017-08-10 20:49:57 +03:00
pci_write_config_byte ( dev , PCI_INTERRUPT_LINE , irq ) ;
2005-04-17 02:20:36 +04:00
}