The patch implements PowerNV backends to support set/get settings. Also, we needn't maintain multiple fields in "struct pnv_phb" to trace different EEH states. The patch merges all EEH states to one field "eeh_state".
Signed-off-by: Gavin Shan <sha...@linux.vnet.ibm.com> --- arch/powerpc/platforms/powernv/eeh-ioda.c | 82 ++++++++++++++++++++++++- arch/powerpc/platforms/powernv/eeh-powernv.c | 34 +++++++++++ arch/powerpc/platforms/powernv/pci.c | 4 +- arch/powerpc/platforms/powernv/pci.h | 12 +++- 4 files changed, 124 insertions(+), 8 deletions(-) diff --git a/arch/powerpc/platforms/powernv/eeh-ioda.c b/arch/powerpc/platforms/powernv/eeh-ioda.c index 84f3036..64c3d1e 100644 --- a/arch/powerpc/platforms/powernv/eeh-ioda.c +++ b/arch/powerpc/platforms/powernv/eeh-ioda.c @@ -132,7 +132,7 @@ static int ioda_eeh_post_init(struct pci_controller *hose) &ioda_eeh_dbgfs_ops); #endif - phb->eeh_enabled = 1; + phb->eeh_state |= PNV_EEH_STATE_ENABLED; } return 0; @@ -583,6 +583,78 @@ static int ioda_eeh_configure_bridge(struct eeh_pe *pe) return 0; } +/** + * ioda_eeh_set_setting - Configure the settings to affect EEH core + * @option: option + * @value: value + * @data: dependent data + * + * Configure the settings to affect EEH core. + */ +static int ioda_eeh_set_setting(int option, int value, void *data) +{ + struct pci_controller *hose = (struct pci_controller *)data; + struct pnv_phb *phb = hose->private_data; + int ret = 0; + + switch (option) { + case EEH_SETTING_BLOCK_CFG: + if (value) + phb->eeh_state |= PNV_EEH_STATE_CFG_BLOCKED; + else + phb->eeh_state &= ~PNV_EEH_STATE_CFG_BLOCKED; + break; + case EEH_SETTING_BLOCK_IO: + if (value) + phb->eeh_state |= PNV_EEH_STATE_IO_BLOCKED; + else + phb->eeh_state &= ~PNV_EEH_STATE_IO_BLOCKED; + break; + default: + pr_warning("%s: Unrecognized option (%d)\n", + __func__, option); + ret = -EINVAL; + } + + return ret; +} + +/** + * ioda_eeh_get_setting - Retrieve the settings to affect EEH core + * @option: option + * @value: value + * @data: dependent data + * + * EEH core retrieves the settings and utilize them. + */ +static int ioda_eeh_get_setting(int option, int *value, void *data) +{ + struct pci_controller *hose = (struct pci_controller *)data; + struct pnv_phb *phb = hose->private_data; + int ret = 0; + + switch (option) { + case EEH_SETTING_BLOCK_CFG: + if (phb->eeh_state & PNV_EEH_STATE_CFG_BLOCKED) + *value = 1; + else + *value = 0; + break; + case EEH_SETTING_BLOCK_IO: + if (phb->eeh_state & PNV_EEH_STATE_IO_BLOCKED) + *value = 1; + else + *value = 0; + break; + default: + pr_warning("%s: Unrecognized option (%d)\n", + __func__, option); + ret = -EINVAL; + } + + return ret; +} + static void ioda_eeh_hub_diag_common(struct OpalIoP7IOCErrorData *data) { /* GEM */ @@ -815,7 +887,7 @@ static int ioda_eeh_next_error(struct eeh_pe **pe) * removed, we needn't take care of it any more. */ phb = hose->private_data; - if (phb->removed) + if (phb->eeh_state & PNV_EEH_STATE_REMOVED) continue; rc = opal_pci_next_error(phb->opal_id, @@ -850,7 +922,7 @@ static int ioda_eeh_next_error(struct eeh_pe **pe) list_for_each_entry_safe(hose, tmp, &hose_list, list_node) { phb = hose->private_data; - phb->removed = 1; + phb->eeh_state |= PNV_EEH_STATE_REMOVED; } WARN(1, "EEH: dead IOC detected\n"); @@ -867,7 +939,7 @@ static int ioda_eeh_next_error(struct eeh_pe **pe) WARN(1, "EEH: dead PHB#%x detected\n", hose->global_number); - phb->removed = 1; + phb->eeh_state |= PNV_EEH_STATE_REMOVED; ret = 3; goto out; } else if (severity == OPAL_EEH_SEV_PHB_FENCED) { @@ -905,5 +977,7 @@ struct pnv_eeh_ops ioda_eeh_ops = { .reset = ioda_eeh_reset, .get_log = ioda_eeh_get_log, .configure_bridge = ioda_eeh_configure_bridge, + .set_setting = ioda_eeh_set_setting, + .get_setting = ioda_eeh_get_setting, .next_error = ioda_eeh_next_error }; diff --git a/arch/powerpc/platforms/powernv/eeh-powernv.c b/arch/powerpc/platforms/powernv/eeh-powernv.c index 9559115..cac5e18 100644 --- a/arch/powerpc/platforms/powernv/eeh-powernv.c +++ b/arch/powerpc/platforms/powernv/eeh-powernv.c @@ -355,6 +355,38 @@ static int powernv_eeh_write_config(struct device_node *dn, int where, } /** + * powernv_eeh_set_setting - Configure setting to affect EEH core + * @option: option + * @value: value + * @data: option dependent data + * + * Configure setting to affect the behaviour of EEH core. + */ +static int powernv_eeh_set_setting(int option, int value, void *data) +{ + struct pci_controller *hose = data; + struct pnv_phb *phb = hose->private_data; + + return phb->set_setting(option, value, data); +} + +/** + * powernv_eeh_get_setting - Retrieve settings to affect EEH core + * @option: option + * @value: value + * @data: option dependent data + * + * Retrieve setting to affect the behaviour of EEH core + */ +static int powernv_eeh_get_setting(int option, int *value, void *data) +{ + struct pci_controller *hose = data; + struct pnv_phb *phb = hose->private_data; + + return phb->get_setting(option, value, data); +} + +/** * powernv_eeh_next_error - Retrieve next EEH error to handle * @pe: Affected PE * @@ -391,6 +423,8 @@ static struct eeh_ops powernv_eeh_ops = { .configure_bridge = powernv_eeh_configure_bridge, .read_config = powernv_eeh_read_config, .write_config = powernv_eeh_write_config, + .set_setting = powernv_eeh_set_setting, + .get_setting = powernv_eeh_get_setting, .next_error = powernv_eeh_next_error }; diff --git a/arch/powerpc/platforms/powernv/pci.c b/arch/powerpc/platforms/powernv/pci.c index 6d9a506..1f31826 100644 --- a/arch/powerpc/platforms/powernv/pci.c +++ b/arch/powerpc/platforms/powernv/pci.c @@ -308,7 +308,7 @@ static int pnv_pci_read_config(struct pci_bus *bus, if (phb_pe && (phb_pe->state & EEH_PE_ISOLATED)) return PCIBIOS_SUCCESSFUL; - if (phb->eeh_enabled) { + if (phb->eeh_state & PNV_EEH_STATE_ENABLED) { if (*val == EEH_IO_ERROR_VALUE(size)) { busdn = pci_bus_to_OF_node(bus); for (dn = busdn->child; dn; dn = dn->sibling) { @@ -358,7 +358,7 @@ static int pnv_pci_write_config(struct pci_bus *bus, /* Check if the PHB got frozen due to an error (no response) */ #ifdef CONFIG_EEH - if (!phb->eeh_enabled) + if (!(phb->eeh_state & PNV_EEH_STATE_ENABLED)) pnv_pci_config_check_eeh(phb, bus, bdfn); #else pnv_pci_config_check_eeh(phb, bus, bdfn); diff --git a/arch/powerpc/platforms/powernv/pci.h b/arch/powerpc/platforms/powernv/pci.h index 43906e3..a281a1c 100644 --- a/arch/powerpc/platforms/powernv/pci.h +++ b/arch/powerpc/platforms/powernv/pci.h @@ -76,8 +76,17 @@ struct pnv_eeh_ops { int (*get_log)(struct eeh_pe *pe, int severity, char *drv_log, unsigned long len); int (*configure_bridge)(struct eeh_pe *pe); + int (*set_setting)(int option, int value, void *data); + int (*get_setting)(int option, int *value, void *data); int (*next_error)(struct eeh_pe **pe); }; + +/* EEH states maintained by PCI hose */ +#define PNV_EEH_STATE_ENABLED (1 << 0) /* EEH enabled */ +#define PNV_EEH_STATE_REMOVED (1 << 1) /* PHB removed */ +#define PNV_EEH_STATE_CFG_BLOCKED (1 << 2) /* PHB PCI-CFG blocked */ +#define PNV_EEH_STATE_IO_BLOCKED (1 << 3) /* PHB MMIO blocked */ + #endif /* CONFIG_EEH */ struct pnv_phb { @@ -92,8 +101,7 @@ struct pnv_phb { #ifdef CONFIG_EEH struct pnv_eeh_ops *eeh_ops; - int eeh_enabled; - int removed; + int eeh_state; #endif #ifdef CONFIG_DEBUG_FS -- 1.7.5.4 _______________________________________________ Linuxppc-dev mailing list Linuxppc-dev@lists.ozlabs.org https://lists.ozlabs.org/listinfo/linuxppc-dev