2019-05-27 08:55:05 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2012-02-27 20:03:53 +00:00
/*
* The file intends to implement the platform dependent EEH operations on pseries .
* Actually , the pseries platform is built based on RTAS heavily . That means the
* pseries platform dependent EEH operations will be built on RTAS calls . The functions
2016-06-01 16:34:37 +10:00
* are derived from arch / powerpc / platforms / pseries / eeh . c and necessary cleanup has
2012-02-27 20:03:53 +00:00
* been done .
*
* Copyright Benjamin Herrenschmidt & Gavin Shan , IBM Corporation 2011.
* Copyright IBM Corporation 2001 , 2005 , 2006
* Copyright Dave Engebretsen & Todd Inglett 2001
* Copyright Linas Vepstas 2005 , 2006
*/
# include <linux/atomic.h>
# include <linux/delay.h>
# include <linux/export.h>
# include <linux/init.h>
# include <linux/list.h>
# include <linux/of.h>
# include <linux/pci.h>
# include <linux/proc_fs.h>
# include <linux/rbtree.h>
# include <linux/sched.h>
# include <linux/seq_file.h>
# include <linux/spinlock.h>
2020-07-13 09:39:33 -05:00
# include <linux/crash_dump.h>
2012-02-27 20:03:53 +00:00
# include <asm/eeh.h>
# include <asm/eeh_event.h>
# include <asm/io.h>
# include <asm/machdep.h>
# include <asm/ppc-pci.h>
# include <asm/rtas.h>
2012-02-27 20:03:54 +00:00
/* RTAS tokens */
static int ibm_set_eeh_option ;
static int ibm_set_slot_reset ;
static int ibm_read_slot_reset_state ;
static int ibm_read_slot_reset_state2 ;
static int ibm_slot_error_detail ;
static int ibm_get_config_addr_info ;
static int ibm_get_config_addr_info2 ;
static int ibm_configure_pe ;
2017-11-09 08:00:34 -06:00
void pseries_pcibios_bus_add_device ( struct pci_dev * pdev )
{
struct pci_dn * pdn = pci_get_pdn ( pdev ) ;
2019-08-16 14:48:09 +10:00
if ( eeh_has_flag ( EEH_FORCE_DISABLED ) )
2017-11-09 08:00:34 -06:00
return ;
2019-08-16 14:48:13 +10:00
dev_dbg ( & pdev - > dev , " EEH: Setting up device \n " ) ;
2019-08-16 14:48:09 +10:00
# ifdef CONFIG_PCI_IOV
if ( pdev - > is_virtfn ) {
struct pci_dn * physfn_pdn ;
pdn - > device_id = pdev - > device ;
pdn - > vendor_id = pdev - > vendor ;
pdn - > class_code = pdev - > class ;
/*
* Last allow unfreeze return code used for retrieval
* by user space in eeh - sysfs to show the last command
* completion from platform .
*/
pdn - > last_allow_rc = 0 ;
physfn_pdn = pci_get_pdn ( pdev - > physfn ) ;
pdn - > pe_number = physfn_pdn - > pe_num_map [ pdn - > vf_index ] ;
}
# endif
2020-03-06 18:39:03 +11:00
pseries_eeh_init_edev ( pdn ) ;
2019-08-16 14:48:09 +10:00
# ifdef CONFIG_PCI_IOV
if ( pdev - > is_virtfn ) {
struct eeh_dev * edev = pdn_to_eeh_dev ( pdn ) ;
2017-11-09 08:00:34 -06:00
2019-08-16 14:48:09 +10:00
edev - > pe_config_addr = ( pdn - > busno < < 16 ) | ( pdn - > devfn < < 8 ) ;
eeh_rmv_from_parent_pe ( edev ) ; /* Remove as it is adding to bus pe */
eeh_add_to_parent_pe ( edev ) ; /* Add as VF PE type */
}
2018-01-05 10:45:48 -06:00
# endif
2020-03-06 18:39:04 +11:00
eeh_probe_device ( pdev ) ;
2019-08-16 14:48:09 +10:00
}
2017-11-09 08:00:34 -06:00
2020-07-13 09:39:33 -05:00
/**
* pseries_eeh_get_config_addr - Retrieve config address
*
* Retrieve the assocated config address . Actually , there ' re 2 RTAS
* function calls dedicated for the purpose . We need implement
* it through the new function and then the old one . Besides ,
* you should make sure the config address is figured out from
* FDT node before calling the function .
*
* It ' s notable that zero ' ed return value means invalid PE config
* address .
*/
static int pseries_eeh_get_config_addr ( struct pci_controller * phb , int config_addr )
{
int ret = 0 ;
int rets [ 3 ] ;
if ( ibm_get_config_addr_info2 ! = RTAS_UNKNOWN_SERVICE ) {
/*
* First of all , we need to make sure there has one PE
* associated with the device . Otherwise , PE address is
* meaningless .
*/
ret = rtas_call ( ibm_get_config_addr_info2 , 4 , 2 , rets ,
config_addr , BUID_HI ( phb - > buid ) ,
BUID_LO ( phb - > buid ) , 1 ) ;
if ( ret | | ( rets [ 0 ] = = 0 ) )
return 0 ;
/* Retrieve the associated PE config address */
ret = rtas_call ( ibm_get_config_addr_info2 , 4 , 2 , rets ,
config_addr , BUID_HI ( phb - > buid ) ,
BUID_LO ( phb - > buid ) , 0 ) ;
if ( ret ) {
pr_warn ( " %s: Failed to get address for PHB#%x-PE#%x \n " ,
__func__ , phb - > global_number , config_addr ) ;
return 0 ;
}
return rets [ 0 ] ;
}
if ( ibm_get_config_addr_info ! = RTAS_UNKNOWN_SERVICE ) {
ret = rtas_call ( ibm_get_config_addr_info , 4 , 2 , rets ,
config_addr , BUID_HI ( phb - > buid ) ,
BUID_LO ( phb - > buid ) , 0 ) ;
if ( ret ) {
pr_warn ( " %s: Failed to get address for PHB#%x-PE#%x \n " ,
__func__ , phb - > global_number , config_addr ) ;
return 0 ;
}
return rets [ 0 ] ;
}
return ret ;
}
/**
* pseries_eeh_phb_reset - Reset the specified PHB
* @ phb : PCI controller
* @ config_adddr : the associated config address
* @ option : reset option
*
* Reset the specified PHB / PE
*/
static int pseries_eeh_phb_reset ( struct pci_controller * phb , int config_addr , int option )
{
int ret ;
/* Reset PE through RTAS call */
ret = rtas_call ( ibm_set_slot_reset , 4 , 1 , NULL ,
config_addr , BUID_HI ( phb - > buid ) ,
BUID_LO ( phb - > buid ) , option ) ;
/* If fundamental-reset not supported, try hot-reset */
if ( option = = EEH_RESET_FUNDAMENTAL & &
ret = = - 8 ) {
option = EEH_RESET_HOT ;
ret = rtas_call ( ibm_set_slot_reset , 4 , 1 , NULL ,
config_addr , BUID_HI ( phb - > buid ) ,
BUID_LO ( phb - > buid ) , option ) ;
}
/* We need reset hold or settlement delay */
if ( option = = EEH_RESET_FUNDAMENTAL | |
option = = EEH_RESET_HOT )
msleep ( EEH_PE_RST_HOLD_TIME ) ;
else
msleep ( EEH_PE_RST_SETTLE_TIME ) ;
return ret ;
}
/**
* pseries_eeh_phb_configure_bridge - Configure PCI bridges in the indicated PE
* @ phb : PCI controller
* @ config_adddr : the associated config address
*
* The function will be called to reconfigure the bridges included
* in the specified PE so that the mulfunctional PE would be recovered
* again .
*/
static int pseries_eeh_phb_configure_bridge ( struct pci_controller * phb , int config_addr )
{
int ret ;
/* Waiting 0.2s maximum before skipping configuration */
int max_wait = 200 ;
while ( max_wait > 0 ) {
ret = rtas_call ( ibm_configure_pe , 3 , 1 , NULL ,
config_addr , BUID_HI ( phb - > buid ) ,
BUID_LO ( phb - > buid ) ) ;
if ( ! ret )
return ret ;
if ( ret < 0 )
break ;
/*
* If RTAS returns a delay value that ' s above 100 ms , cut it
* down to 100 ms in case firmware made a mistake . For more
* on how these delay values work see rtas_busy_delay_time
*/
if ( ret > RTAS_EXTENDED_DELAY_MIN + 2 & &
ret < = RTAS_EXTENDED_DELAY_MAX )
ret = RTAS_EXTENDED_DELAY_MIN + 2 ;
max_wait - = rtas_busy_delay_time ( ret ) ;
if ( max_wait < 0 )
break ;
rtas_busy_delay ( ret ) ;
}
pr_warn ( " %s: Unable to configure bridge PHB#%x-PE#%x (%d) \n " ,
__func__ , phb - > global_number , config_addr , ret ) ;
/* PAPR defines -3 as "Parameter Error" for this function: */
if ( ret = = - 3 )
return - EINVAL ;
else
return - EIO ;
}
2012-02-27 20:04:00 +00:00
/*
* Buffer for reporting slot - error - detail rtas calls . Its here
* in BSS , and not dynamically alloced , so that it ends up in
* RMO where RTAS can access it .
*/
static unsigned char slot_errbuf [ RTAS_ERROR_LOG_MAX ] ;
static DEFINE_SPINLOCK ( slot_errbuf_lock ) ;
static int eeh_error_buf_size ;
2012-02-27 20:03:53 +00:00
/**
* pseries_eeh_init - EEH platform dependent initialization
*
* EEH platform dependent initialization on pseries .
*/
static int pseries_eeh_init ( void )
{
2020-07-13 09:39:33 -05:00
struct pci_controller * phb ;
struct pci_dn * pdn ;
int addr , config_addr ;
2012-02-27 20:03:54 +00:00
/* figure out EEH RTAS function call tokens */
ibm_set_eeh_option = rtas_token ( " ibm,set-eeh-option " ) ;
ibm_set_slot_reset = rtas_token ( " ibm,set-slot-reset " ) ;
ibm_read_slot_reset_state2 = rtas_token ( " ibm,read-slot-reset-state2 " ) ;
ibm_read_slot_reset_state = rtas_token ( " ibm,read-slot-reset-state " ) ;
ibm_slot_error_detail = rtas_token ( " ibm,slot-error-detail " ) ;
ibm_get_config_addr_info2 = rtas_token ( " ibm,get-config-addr-info2 " ) ;
ibm_get_config_addr_info = rtas_token ( " ibm,get-config-addr-info " ) ;
ibm_configure_pe = rtas_token ( " ibm,configure-pe " ) ;
2016-04-07 16:28:27 +10:00
/*
* ibm , configure - pe and ibm , configure - bridge have the same semantics ,
* however ibm , configure - pe can be faster . If we can ' t find
* ibm , configure - pe then fall back to using ibm , configure - bridge .
*/
if ( ibm_configure_pe = = RTAS_UNKNOWN_SERVICE )
ibm_configure_pe = rtas_token ( " ibm,configure-bridge " ) ;
2012-02-27 20:03:54 +00:00
2013-06-05 14:25:50 +00:00
/*
* Necessary sanity check . We needn ' t check " get-config-addr-info "
* and its variant since the old firmware probably support address
* of domain / bus / slot / function for EEH RTAS operations .
*/
2014-09-30 12:39:04 +10:00
if ( ibm_set_eeh_option = = RTAS_UNKNOWN_SERVICE | |
ibm_set_slot_reset = = RTAS_UNKNOWN_SERVICE | |
( ibm_read_slot_reset_state2 = = RTAS_UNKNOWN_SERVICE & &
ibm_read_slot_reset_state = = RTAS_UNKNOWN_SERVICE ) | |
ibm_slot_error_detail = = RTAS_UNKNOWN_SERVICE | |
2016-04-07 16:28:27 +10:00
ibm_configure_pe = = RTAS_UNKNOWN_SERVICE ) {
2014-09-30 12:39:04 +10:00
pr_info ( " EEH functionality not supported \n " ) ;
2012-02-27 20:03:54 +00:00
return - EINVAL ;
}
2012-02-27 20:04:00 +00:00
/* Initialize error log lock and size */
spin_lock_init ( & slot_errbuf_lock ) ;
eeh_error_buf_size = rtas_token ( " rtas-error-log-max " ) ;
if ( eeh_error_buf_size = = RTAS_UNKNOWN_SERVICE ) {
2014-09-30 12:39:04 +10:00
pr_info ( " %s: unknown EEH error log size \n " ,
2012-02-27 20:04:00 +00:00
__func__ ) ;
eeh_error_buf_size = 1024 ;
} else if ( eeh_error_buf_size > RTAS_ERROR_LOG_MAX ) {
2014-09-30 12:39:04 +10:00
pr_info ( " %s: EEH error log size %d exceeds the maximal %d \n " ,
2012-02-27 20:04:00 +00:00
__func__ , eeh_error_buf_size , RTAS_ERROR_LOG_MAX ) ;
eeh_error_buf_size = RTAS_ERROR_LOG_MAX ;
}
2012-09-07 22:44:21 +00:00
/* Set EEH probe mode */
2014-07-17 14:41:39 +10:00
eeh_add_flag ( EEH_PROBE_MODE_DEVTREE | EEH_ENABLE_IO_FOR_LOG ) ;
2012-09-07 22:44:21 +00:00
2017-11-09 08:00:34 -06:00
/* Set EEH machine dependent code */
ppc_md . pcibios_bus_add_device = pseries_pcibios_bus_add_device ;
2020-07-13 09:39:33 -05:00
if ( is_kdump_kernel ( ) | | reset_devices ) {
pr_info ( " Issue PHB reset ... \n " ) ;
list_for_each_entry ( phb , & hose_list , list_node ) {
pdn = list_first_entry ( & PCI_DN ( phb - > dn ) - > child_list , struct pci_dn , list ) ;
addr = ( pdn - > busno < < 16 ) | ( pdn - > devfn < < 8 ) ;
config_addr = pseries_eeh_get_config_addr ( phb , addr ) ;
/* invalid PE config addr */
if ( config_addr = = 0 )
continue ;
pseries_eeh_phb_reset ( phb , config_addr , EEH_RESET_FUNDAMENTAL ) ;
pseries_eeh_phb_reset ( phb , config_addr , EEH_RESET_DEACTIVATE ) ;
pseries_eeh_phb_configure_bridge ( phb , config_addr ) ;
}
}
2012-02-27 20:03:53 +00:00
return 0 ;
}
2015-03-17 16:15:06 +11:00
static int pseries_eeh_cap_start ( struct pci_dn * pdn )
2013-07-24 10:24:59 +08:00
{
u32 status ;
if ( ! pdn )
return 0 ;
rtas_read_config ( pdn , PCI_STATUS , 2 , & status ) ;
if ( ! ( status & PCI_STATUS_CAP_LIST ) )
return 0 ;
return PCI_CAPABILITY_LIST ;
}
2015-03-17 16:15:06 +11:00
static int pseries_eeh_find_cap ( struct pci_dn * pdn , int cap )
2013-07-24 10:24:59 +08:00
{
2015-03-17 16:15:06 +11:00
int pos = pseries_eeh_cap_start ( pdn ) ;
2013-07-24 10:24:59 +08:00
int cnt = 48 ; /* Maximal number of capabilities */
u32 id ;
if ( ! pos )
return 0 ;
while ( cnt - - ) {
rtas_read_config ( pdn , pos , 1 , & pos ) ;
if ( pos < 0x40 )
break ;
pos & = ~ 3 ;
rtas_read_config ( pdn , pos + PCI_CAP_LIST_ID , 1 , & id ) ;
if ( id = = 0xff )
break ;
if ( id = = cap )
return pos ;
pos + = PCI_CAP_LIST_NEXT ;
}
return 0 ;
}
2015-03-17 16:15:06 +11:00
static int pseries_eeh_find_ecap ( struct pci_dn * pdn , int cap )
2014-04-24 18:00:16 +10:00
{
2015-03-17 16:15:06 +11:00
struct eeh_dev * edev = pdn_to_eeh_dev ( pdn ) ;
2014-04-24 18:00:16 +10:00
u32 header ;
int pos = 256 ;
int ttl = ( 4096 - 256 ) / 8 ;
if ( ! edev | | ! edev - > pcie_cap )
return 0 ;
if ( rtas_read_config ( pdn , pos , 4 , & header ) ! = PCIBIOS_SUCCESSFUL )
return 0 ;
else if ( ! header )
return 0 ;
while ( ttl - - > 0 ) {
if ( PCI_EXT_CAP_ID ( header ) = = cap & & pos )
return pos ;
pos = PCI_EXT_CAP_NEXT ( header ) ;
if ( pos < 256 )
break ;
if ( rtas_read_config ( pdn , pos , 4 , & header ) ! = PCIBIOS_SUCCESSFUL )
break ;
}
return 0 ;
}
2012-09-07 22:44:21 +00:00
/**
2020-03-06 18:39:03 +11:00
* pseries_eeh_init_edev - initialise the eeh_dev and eeh_pe for a pci_dn
*
2015-03-17 16:15:06 +11:00
* @ pdn : PCI device node
2012-09-07 22:44:21 +00:00
*
2020-03-06 18:39:03 +11:00
* When we discover a new PCI device via the device - tree we create a
* corresponding pci_dn and we allocate , but don ' t initialise , an eeh_dev .
* This function takes care of the initialisation and inserts the eeh_dev
* into the correct eeh_pe . If no eeh_pe exists we ' ll allocate one .
2012-09-07 22:44:21 +00:00
*/
2020-03-06 18:39:03 +11:00
void pseries_eeh_init_edev ( struct pci_dn * pdn )
2012-09-07 22:44:21 +00:00
{
struct eeh_dev * edev ;
struct eeh_pe pe ;
2013-07-24 10:24:59 +08:00
u32 pcie_flags ;
2012-09-07 22:44:21 +00:00
int enable = 0 ;
int ret ;
2020-03-06 18:39:03 +11:00
if ( WARN_ON_ONCE ( ! eeh_has_flag ( EEH_PROBE_MODE_DEVTREE ) ) )
return ;
/*
* Find the eeh_dev for this pdn . The storage for the eeh_dev was
* allocated at the same time as the pci_dn .
*
* XXX : We should probably re - visit that .
*/
2015-03-17 16:15:06 +11:00
edev = pdn_to_eeh_dev ( pdn ) ;
2020-03-06 18:39:03 +11:00
if ( ! edev )
return ;
/*
* If - > pe is set then we ' ve already probed this device . We hit
* this path when a pci_dev is removed and rescanned while recovering
* a PE ( i . e . for devices where the driver doesn ' t support error
* recovery ) .
*/
if ( edev - > pe )
return ;
2012-09-07 22:44:21 +00:00
2015-03-17 16:15:06 +11:00
/* Check class/vendor/device IDs */
if ( ! pdn - > vendor_id | | ! pdn - > device_id | | ! pdn - > class_code )
2020-03-06 18:39:03 +11:00
return ;
2012-09-07 22:44:21 +00:00
2015-03-17 16:15:06 +11:00
/* Skip for PCI-ISA bridge */
if ( ( pdn - > class_code > > 8 ) = = PCI_CLASS_BRIDGE_ISA )
2020-03-06 18:39:03 +11:00
return ;
2013-11-20 22:14:58 +11:00
2019-08-16 14:48:13 +10:00
eeh_edev_dbg ( edev , " Probing device \n " ) ;
2013-07-24 10:24:59 +08:00
/*
* Update class code and mode of eeh device . We need
* correctly reflects that current device is root port
* or PCIe switch downstream port .
*/
2015-03-17 16:15:06 +11:00
edev - > class_code = pdn - > class_code ;
edev - > pcix_cap = pseries_eeh_find_cap ( pdn , PCI_CAP_ID_PCIX ) ;
edev - > pcie_cap = pseries_eeh_find_cap ( pdn , PCI_CAP_ID_EXP ) ;
edev - > aer_cap = pseries_eeh_find_ecap ( pdn , PCI_EXT_CAP_ID_ERR ) ;
2013-07-24 10:25:01 +08:00
edev - > mode & = 0xFFFFFF00 ;
2013-07-24 10:24:59 +08:00
if ( ( edev - > class_code > > 8 ) = = PCI_CLASS_BRIDGE_PCI ) {
edev - > mode | = EEH_DEV_BRIDGE ;
if ( edev - > pcie_cap ) {
rtas_read_config ( pdn , edev - > pcie_cap + PCI_EXP_FLAGS ,
2 , & pcie_flags ) ;
pcie_flags = ( pcie_flags & PCI_EXP_FLAGS_TYPE ) > > 4 ;
if ( pcie_flags = = PCI_EXP_TYPE_ROOT_PORT )
edev - > mode | = EEH_DEV_ROOT_PORT ;
else if ( pcie_flags = = PCI_EXP_TYPE_DOWNSTREAM )
edev - > mode | = EEH_DEV_DS_PORT ;
}
}
2012-09-07 22:44:21 +00:00
/* Initialize the fake PE */
memset ( & pe , 0 , sizeof ( struct eeh_pe ) ) ;
2017-08-29 17:34:01 +10:00
pe . phb = pdn - > phb ;
2015-03-17 16:15:06 +11:00
pe . config_addr = ( pdn - > busno < < 16 ) | ( pdn - > devfn < < 8 ) ;
2012-09-07 22:44:21 +00:00
/* Enable EEH on the device */
2019-08-16 14:48:13 +10:00
eeh_edev_dbg ( edev , " Enabling EEH on device \n " ) ;
2012-09-07 22:44:21 +00:00
ret = eeh_ops - > set_option ( & pe , EEH_OPT_ENABLE ) ;
2019-08-16 14:48:07 +10:00
if ( ret ) {
2019-08-16 14:48:13 +10:00
eeh_edev_dbg ( edev , " EEH failed to enable on device (code %d) \n " , ret ) ;
2019-08-16 14:48:07 +10:00
} else {
2012-09-07 22:44:21 +00:00
/* Retrieve PE address */
edev - > pe_config_addr = eeh_ops - > get_pe_addr ( & pe ) ;
pe . addr = edev - > pe_config_addr ;
/* Some older systems (Power4) allow the ibm,set-eeh-option
* call to succeed even on nodes where EEH is not supported .
* Verify support explicitly .
*/
ret = eeh_ops - > get_state ( & pe , NULL ) ;
if ( ret > 0 & & ret ! = EEH_STATE_NOT_SUPPORT )
enable = 1 ;
if ( enable ) {
2014-07-17 14:41:38 +10:00
eeh_add_flag ( EEH_ENABLED ) ;
2012-09-07 22:44:21 +00:00
eeh_add_to_parent_pe ( edev ) ;
2015-03-17 16:15:06 +11:00
} else if ( pdn - > parent & & pdn_to_eeh_dev ( pdn - > parent ) & &
( pdn_to_eeh_dev ( pdn - > parent ) ) - > pe ) {
2012-09-07 22:44:21 +00:00
/* This device doesn't support EEH, but it may have an
* EEH parent , in which case we mark it as supported .
*/
2015-03-17 16:15:06 +11:00
edev - > pe_config_addr = pdn_to_eeh_dev ( pdn - > parent ) - > pe_config_addr ;
2012-09-07 22:44:21 +00:00
eeh_add_to_parent_pe ( edev ) ;
}
2019-08-16 14:48:13 +10:00
eeh_edev_dbg ( edev , " EEH is %s on device (code %d) \n " ,
( enable ? " enabled " : " unsupported " ) , ret ) ;
2012-09-07 22:44:21 +00:00
}
/* Save memory bars */
eeh_save_bars ( edev ) ;
2020-03-06 18:39:03 +11:00
}
2020-03-06 18:39:04 +11:00
static struct eeh_dev * pseries_eeh_probe ( struct pci_dev * pdev )
{
struct eeh_dev * edev ;
struct pci_dn * pdn ;
pdn = pci_get_pdn_by_devfn ( pdev - > bus , pdev - > devfn ) ;
if ( ! pdn )
return NULL ;
/*
* If the system supports EEH on this device then the eeh_dev was
* configured and inserted into a PE in pseries_eeh_init_edev ( )
*/
edev = pdn_to_eeh_dev ( pdn ) ;
if ( ! edev | | ! edev - > pe )
return NULL ;
return edev ;
}
2020-03-06 18:39:03 +11:00
/**
* pseries_eeh_init_edev_recursive - Enable EEH for the indicated device
* @ pdn : PCI device node
*
* This routine must be used to perform EEH initialization for the
* indicated PCI device that was added after system boot ( e . g .
* hotplug , dlpar ) .
*/
void pseries_eeh_init_edev_recursive ( struct pci_dn * pdn )
{
struct pci_dn * n ;
if ( ! pdn )
return ;
list_for_each_entry ( n , & pdn - > child_list , list )
pseries_eeh_init_edev_recursive ( n ) ;
2012-09-07 22:44:21 +00:00
2020-03-06 18:39:03 +11:00
pseries_eeh_init_edev ( pdn ) ;
2012-09-07 22:44:21 +00:00
}
2020-03-06 18:39:03 +11:00
EXPORT_SYMBOL_GPL ( pseries_eeh_init_edev_recursive ) ;
2012-09-07 22:44:21 +00:00
2012-02-27 20:03:53 +00:00
/**
* pseries_eeh_set_option - Initialize EEH or MMIO / DMA reenable
2012-09-07 22:44:14 +00:00
* @ pe : EEH PE
2012-02-27 20:03:53 +00:00
* @ option : operation to be issued
*
* The function is used to control the EEH functionality globally .
* Currently , following options are support according to PAPR :
* Enable EEH , Disable EEH , Enable MMIO and Enable DMA
*/
2012-09-07 22:44:14 +00:00
static int pseries_eeh_set_option ( struct eeh_pe * pe , int option )
2012-02-27 20:03:53 +00:00
{
2012-02-27 20:03:55 +00:00
int ret = 0 ;
int config_addr ;
/*
* When we ' re enabling or disabling EEH functioality on
* the particular PE , the PE config address is possibly
* unavailable . Therefore , we have to figure it out from
* the FDT node .
*/
switch ( option ) {
case EEH_OPT_DISABLE :
case EEH_OPT_ENABLE :
case EEH_OPT_THAW_MMIO :
case EEH_OPT_THAW_DMA :
2012-09-07 22:44:14 +00:00
config_addr = pe - > config_addr ;
if ( pe - > addr )
config_addr = pe - > addr ;
2012-02-27 20:03:55 +00:00
break ;
2014-09-30 12:38:52 +10:00
case EEH_OPT_FREEZE_PE :
/* Not support */
return 0 ;
2012-02-27 20:03:55 +00:00
default :
pr_err ( " %s: Invalid option %d \n " ,
__func__ , option ) ;
return - EINVAL ;
}
ret = rtas_call ( ibm_set_eeh_option , 4 , 1 , NULL ,
2012-09-07 22:44:14 +00:00
config_addr , BUID_HI ( pe - > phb - > buid ) ,
BUID_LO ( pe - > phb - > buid ) , option ) ;
2012-02-27 20:03:55 +00:00
return ret ;
2012-02-27 20:03:53 +00:00
}
/**
* pseries_eeh_get_pe_addr - Retrieve PE address
2012-09-07 22:44:14 +00:00
* @ pe : EEH PE
2012-02-27 20:03:53 +00:00
*
* Retrieve the assocated PE address . Actually , there ' re 2 RTAS
* function calls dedicated for the purpose . We need implement
* it through the new function and then the old one . Besides ,
* you should make sure the config address is figured out from
* FDT node before calling the function .
*
* It ' s notable that zero ' ed return value means invalid PE config
* address .
*/
2012-09-07 22:44:14 +00:00
static int pseries_eeh_get_pe_addr ( struct eeh_pe * pe )
2012-02-27 20:03:53 +00:00
{
2012-02-27 20:03:56 +00:00
int ret = 0 ;
int rets [ 3 ] ;
if ( ibm_get_config_addr_info2 ! = RTAS_UNKNOWN_SERVICE ) {
/*
* First of all , we need to make sure there has one PE
* associated with the device . Otherwise , PE address is
* meaningless .
*/
ret = rtas_call ( ibm_get_config_addr_info2 , 4 , 2 , rets ,
2012-09-07 22:44:14 +00:00
pe - > config_addr , BUID_HI ( pe - > phb - > buid ) ,
BUID_LO ( pe - > phb - > buid ) , 1 ) ;
2012-02-27 20:03:56 +00:00
if ( ret | | ( rets [ 0 ] = = 0 ) )
return 0 ;
/* Retrieve the associated PE config address */
ret = rtas_call ( ibm_get_config_addr_info2 , 4 , 2 , rets ,
2012-09-07 22:44:14 +00:00
pe - > config_addr , BUID_HI ( pe - > phb - > buid ) ,
BUID_LO ( pe - > phb - > buid ) , 0 ) ;
2012-02-27 20:03:56 +00:00
if ( ret ) {
2016-11-16 14:02:15 +11:00
pr_warn ( " %s: Failed to get address for PHB#%x-PE#%x \n " ,
2012-09-07 22:44:14 +00:00
__func__ , pe - > phb - > global_number , pe - > config_addr ) ;
2012-02-27 20:03:56 +00:00
return 0 ;
}
return rets [ 0 ] ;
}
if ( ibm_get_config_addr_info ! = RTAS_UNKNOWN_SERVICE ) {
ret = rtas_call ( ibm_get_config_addr_info , 4 , 2 , rets ,
2012-09-07 22:44:14 +00:00
pe - > config_addr , BUID_HI ( pe - > phb - > buid ) ,
BUID_LO ( pe - > phb - > buid ) , 0 ) ;
2012-02-27 20:03:56 +00:00
if ( ret ) {
2016-11-16 14:02:15 +11:00
pr_warn ( " %s: Failed to get address for PHB#%x-PE#%x \n " ,
2012-09-07 22:44:14 +00:00
__func__ , pe - > phb - > global_number , pe - > config_addr ) ;
2012-02-27 20:03:56 +00:00
return 0 ;
}
return rets [ 0 ] ;
}
return ret ;
2012-02-27 20:03:53 +00:00
}
/**
* pseries_eeh_get_state - Retrieve PE state
2012-09-07 22:44:14 +00:00
* @ pe : EEH PE
2018-09-12 11:23:32 +10:00
* @ delay : suggested time to wait if state is unavailable
2012-02-27 20:03:53 +00:00
*
* Retrieve the state of the specified PE . On RTAS compliant
* pseries platform , there already has one dedicated RTAS function
* for the purpose . It ' s notable that the associated PE config address
* might be ready when calling the function . Therefore , endeavour to
* use the PE config address if possible . Further more , there ' re 2
* RTAS calls for the purpose , we need to try the new one and back
* to the old one if the new one couldn ' t work properly .
*/
2018-09-12 11:23:32 +10:00
static int pseries_eeh_get_state ( struct eeh_pe * pe , int * delay )
2012-02-27 20:03:53 +00:00
{
2012-02-27 20:03:57 +00:00
int config_addr ;
int ret ;
int rets [ 4 ] ;
int result ;
/* Figure out PE config address if possible */
2012-09-07 22:44:14 +00:00
config_addr = pe - > config_addr ;
if ( pe - > addr )
config_addr = pe - > addr ;
2012-02-27 20:03:57 +00:00
if ( ibm_read_slot_reset_state2 ! = RTAS_UNKNOWN_SERVICE ) {
ret = rtas_call ( ibm_read_slot_reset_state2 , 3 , 4 , rets ,
2012-09-07 22:44:14 +00:00
config_addr , BUID_HI ( pe - > phb - > buid ) ,
BUID_LO ( pe - > phb - > buid ) ) ;
2012-02-27 20:03:57 +00:00
} else if ( ibm_read_slot_reset_state ! = RTAS_UNKNOWN_SERVICE ) {
/* Fake PE unavailable info */
rets [ 2 ] = 0 ;
ret = rtas_call ( ibm_read_slot_reset_state , 3 , 3 , rets ,
2012-09-07 22:44:14 +00:00
config_addr , BUID_HI ( pe - > phb - > buid ) ,
BUID_LO ( pe - > phb - > buid ) ) ;
2012-02-27 20:03:57 +00:00
} else {
return EEH_STATE_NOT_SUPPORT ;
}
if ( ret )
return ret ;
/* Parse the result out */
2015-10-08 14:58:56 +11:00
if ( ! rets [ 1 ] )
return EEH_STATE_NOT_SUPPORT ;
switch ( rets [ 0 ] ) {
case 0 :
result = EEH_STATE_MMIO_ACTIVE |
EEH_STATE_DMA_ACTIVE ;
break ;
case 1 :
result = EEH_STATE_RESET_ACTIVE |
EEH_STATE_MMIO_ACTIVE |
EEH_STATE_DMA_ACTIVE ;
break ;
case 2 :
result = 0 ;
break ;
case 4 :
result = EEH_STATE_MMIO_ENABLED ;
break ;
case 5 :
if ( rets [ 2 ] ) {
2018-09-12 11:23:32 +10:00
if ( delay )
* delay = rets [ 2 ] ;
2015-10-08 14:58:56 +11:00
result = EEH_STATE_UNAVAILABLE ;
} else {
2012-02-27 20:03:57 +00:00
result = EEH_STATE_NOT_SUPPORT ;
}
2015-10-08 14:58:56 +11:00
break ;
default :
2012-02-27 20:03:57 +00:00
result = EEH_STATE_NOT_SUPPORT ;
}
return result ;
2012-02-27 20:03:53 +00:00
}
/**
* pseries_eeh_reset - Reset the specified PE
2012-09-07 22:44:14 +00:00
* @ pe : EEH PE
2012-02-27 20:03:53 +00:00
* @ option : reset option
*
* Reset the specified PE
*/
2012-09-07 22:44:14 +00:00
static int pseries_eeh_reset ( struct eeh_pe * pe , int option )
2012-02-27 20:03:53 +00:00
{
2012-02-27 20:03:59 +00:00
int config_addr ;
/* Figure out PE address */
2012-09-07 22:44:14 +00:00
config_addr = pe - > config_addr ;
if ( pe - > addr )
config_addr = pe - > addr ;
2012-02-27 20:03:59 +00:00
2020-07-13 09:39:33 -05:00
return pseries_eeh_phb_reset ( pe - > phb , config_addr , option ) ;
2012-02-27 20:03:53 +00:00
}
/**
* pseries_eeh_get_log - Retrieve error log
2012-09-07 22:44:14 +00:00
* @ pe : EEH PE
2012-02-27 20:03:53 +00:00
* @ severity : temporary or permanent error log
* @ drv_log : driver log to be combined with retrieved error log
* @ len : length of driver log
*
* Retrieve the temporary or permanent error from the PE .
* Actually , the error will be retrieved through the dedicated
* RTAS call .
*/
2012-09-07 22:44:14 +00:00
static int pseries_eeh_get_log ( struct eeh_pe * pe , int severity , char * drv_log , unsigned long len )
2012-02-27 20:03:53 +00:00
{
2012-02-27 20:04:00 +00:00
int config_addr ;
unsigned long flags ;
int ret ;
spin_lock_irqsave ( & slot_errbuf_lock , flags ) ;
memset ( slot_errbuf , 0 , eeh_error_buf_size ) ;
/* Figure out the PE address */
2012-09-07 22:44:14 +00:00
config_addr = pe - > config_addr ;
if ( pe - > addr )
config_addr = pe - > addr ;
2012-02-27 20:04:00 +00:00
ret = rtas_call ( ibm_slot_error_detail , 8 , 1 , NULL , config_addr ,
2012-09-07 22:44:14 +00:00
BUID_HI ( pe - > phb - > buid ) , BUID_LO ( pe - > phb - > buid ) ,
2012-02-27 20:04:00 +00:00
virt_to_phys ( drv_log ) , len ,
virt_to_phys ( slot_errbuf ) , eeh_error_buf_size ,
severity ) ;
if ( ! ret )
log_error ( slot_errbuf , ERR_TYPE_RTAS_LOG , 0 ) ;
spin_unlock_irqrestore ( & slot_errbuf_lock , flags ) ;
return ret ;
2012-02-27 20:03:53 +00:00
}
/**
* pseries_eeh_configure_bridge - Configure PCI bridges in the indicated PE
2012-09-07 22:44:14 +00:00
* @ pe : EEH PE
2012-02-27 20:03:53 +00:00
*
*/
2012-09-07 22:44:14 +00:00
static int pseries_eeh_configure_bridge ( struct eeh_pe * pe )
2012-02-27 20:03:53 +00:00
{
2012-02-27 20:04:01 +00:00
int config_addr ;
/* Figure out the PE address */
2012-09-07 22:44:14 +00:00
config_addr = pe - > config_addr ;
if ( pe - > addr )
config_addr = pe - > addr ;
2012-02-27 20:04:01 +00:00
2020-07-13 09:39:33 -05:00
return pseries_eeh_phb_configure_bridge ( pe - > phb , config_addr ) ;
2012-02-27 20:03:53 +00:00
}
2012-02-27 20:04:11 +00:00
/**
* pseries_eeh_read_config - Read PCI config space
2015-03-17 16:15:07 +11:00
* @ pdn : PCI device node
2012-02-27 20:04:11 +00:00
* @ where : PCI address
* @ size : size to read
* @ val : return value
*
* Read config space from the speicifed device
*/
2015-03-17 16:15:07 +11:00
static int pseries_eeh_read_config ( struct pci_dn * pdn , int where , int size , u32 * val )
2012-02-27 20:04:11 +00:00
{
return rtas_read_config ( pdn , where , size , val ) ;
}
/**
* pseries_eeh_write_config - Write PCI config space
2015-03-17 16:15:07 +11:00
* @ pdn : PCI device node
2012-02-27 20:04:11 +00:00
* @ where : PCI address
* @ size : size to write
* @ val : value to be written
*
* Write config space to the specified device
*/
2015-03-17 16:15:07 +11:00
static int pseries_eeh_write_config ( struct pci_dn * pdn , int where , int size , u32 val )
2012-02-27 20:04:11 +00:00
{
return rtas_write_config ( pdn , where , size , val ) ;
}
2018-01-05 10:45:46 -06:00
static int pseries_eeh_restore_config ( struct pci_dn * pdn )
{
struct eeh_dev * edev = pdn_to_eeh_dev ( pdn ) ;
s64 ret = 0 ;
if ( ! edev )
return - EEXIST ;
/*
* FIXME : The MPS , error routing rules , timeout setting are worthy
* to be exported by firmware in extendible way .
*/
if ( edev - > physfn )
ret = eeh_restore_vf_config ( pdn ) ;
if ( ret ) {
pr_warn ( " %s: Can't reinit PCI dev 0x%x (%lld) \n " ,
__func__ , edev - > pe_config_addr , ret ) ;
return - EIO ;
}
return ret ;
}
2018-01-05 10:45:49 -06:00
# ifdef CONFIG_PCI_IOV
int pseries_send_allow_unfreeze ( struct pci_dn * pdn ,
u16 * vf_pe_array , int cur_vfs )
{
int rc ;
int ibm_allow_unfreeze = rtas_token ( " ibm,open-sriov-allow-unfreeze " ) ;
unsigned long buid , addr ;
addr = rtas_config_addr ( pdn - > busno , pdn - > devfn , 0 ) ;
buid = pdn - > phb - > buid ;
spin_lock ( & rtas_data_buf_lock ) ;
memcpy ( rtas_data_buf , vf_pe_array , RTAS_DATA_BUF_SIZE ) ;
rc = rtas_call ( ibm_allow_unfreeze , 5 , 1 , NULL ,
addr ,
BUID_HI ( buid ) ,
BUID_LO ( buid ) ,
rtas_data_buf , cur_vfs * sizeof ( u16 ) ) ;
spin_unlock ( & rtas_data_buf_lock ) ;
if ( rc )
pr_warn ( " %s: Failed to allow unfreeze for PHB#%x-PE#%lx, rc=%x \n " ,
__func__ ,
pdn - > phb - > global_number , addr , rc ) ;
return rc ;
}
static int pseries_call_allow_unfreeze ( struct eeh_dev * edev )
{
struct pci_dn * pdn , * tmp , * parent , * physfn_pdn ;
int cur_vfs = 0 , rc = 0 , vf_index , bus , devfn ;
u16 * vf_pe_array ;
vf_pe_array = kzalloc ( RTAS_DATA_BUF_SIZE , GFP_KERNEL ) ;
if ( ! vf_pe_array )
return - ENOMEM ;
if ( pci_num_vf ( edev - > physfn ? edev - > physfn : edev - > pdev ) ) {
if ( edev - > pdev - > is_physfn ) {
cur_vfs = pci_num_vf ( edev - > pdev ) ;
pdn = eeh_dev_to_pdn ( edev ) ;
parent = pdn - > parent ;
for ( vf_index = 0 ; vf_index < cur_vfs ; vf_index + + )
vf_pe_array [ vf_index ] =
cpu_to_be16 ( pdn - > pe_num_map [ vf_index ] ) ;
rc = pseries_send_allow_unfreeze ( pdn , vf_pe_array ,
cur_vfs ) ;
pdn - > last_allow_rc = rc ;
for ( vf_index = 0 ; vf_index < cur_vfs ; vf_index + + ) {
list_for_each_entry_safe ( pdn , tmp ,
& parent - > child_list ,
list ) {
bus = pci_iov_virtfn_bus ( edev - > pdev ,
vf_index ) ;
devfn = pci_iov_virtfn_devfn ( edev - > pdev ,
vf_index ) ;
if ( pdn - > busno ! = bus | |
pdn - > devfn ! = devfn )
continue ;
pdn - > last_allow_rc = rc ;
}
}
} else {
pdn = pci_get_pdn ( edev - > pdev ) ;
vf_pe_array [ 0 ] = cpu_to_be16 ( pdn - > pe_number ) ;
physfn_pdn = pci_get_pdn ( edev - > physfn ) ;
rc = pseries_send_allow_unfreeze ( physfn_pdn ,
vf_pe_array , 1 ) ;
pdn - > last_allow_rc = rc ;
}
}
kfree ( vf_pe_array ) ;
return rc ;
}
static int pseries_notify_resume ( struct pci_dn * pdn )
{
struct eeh_dev * edev = pdn_to_eeh_dev ( pdn ) ;
if ( ! edev )
return - EEXIST ;
if ( rtas_token ( " ibm,open-sriov-allow-unfreeze " )
= = RTAS_UNKNOWN_SERVICE )
return - EINVAL ;
if ( edev - > pdev - > is_physfn | | edev - > pdev - > is_virtfn )
return pseries_call_allow_unfreeze ( edev ) ;
return 0 ;
}
# endif
2012-02-27 20:03:53 +00:00
static struct eeh_ops pseries_eeh_ops = {
. name = " pseries " ,
. init = pseries_eeh_init ,
2020-03-06 18:39:04 +11:00
. probe = pseries_eeh_probe ,
2012-02-27 20:03:53 +00:00
. set_option = pseries_eeh_set_option ,
. get_pe_addr = pseries_eeh_get_pe_addr ,
. get_state = pseries_eeh_get_state ,
. reset = pseries_eeh_reset ,
. get_log = pseries_eeh_get_log ,
2012-02-27 20:04:11 +00:00
. configure_bridge = pseries_eeh_configure_bridge ,
2014-09-30 12:38:56 +10:00
. err_inject = NULL ,
2012-02-27 20:04:11 +00:00
. read_config = pseries_eeh_read_config ,
2014-01-03 17:47:12 +08:00
. write_config = pseries_eeh_write_config ,
. next_error = NULL ,
2018-01-05 10:45:49 -06:00
. restore_config = pseries_eeh_restore_config ,
# ifdef CONFIG_PCI_IOV
. notify_resume = pseries_notify_resume
# endif
2012-02-27 20:03:53 +00:00
} ;
/**
* eeh_pseries_init - Register platform dependent EEH operations
*
* EEH initialization on pseries platform . This function should be
* called before any EEH related functions .
*/
2012-09-07 22:44:02 +00:00
static int __init eeh_pseries_init ( void )
2012-02-27 20:03:53 +00:00
{
2014-07-16 12:02:43 +10:00
int ret ;
2012-09-07 22:44:04 +00:00
ret = eeh_ops_register ( & pseries_eeh_ops ) ;
if ( ! ret )
pr_info ( " EEH: pSeries platform initialized \n " ) ;
else
pr_info ( " EEH: pSeries platform initialization failure (%d) \n " ,
ret ) ;
return ret ;
2012-02-27 20:03:53 +00:00
}
2014-07-16 12:02:43 +10:00
machine_early_initcall ( pseries , eeh_pseries_init ) ;