aboutsummaryrefslogtreecommitdiffstats
path: root/arch/ppc64/kernel
diff options
context:
space:
mode:
Diffstat (limited to 'arch/ppc64/kernel')
-rw-r--r--arch/ppc64/kernel/eeh.c46
1 files changed, 30 insertions, 16 deletions
diff --git a/arch/ppc64/kernel/eeh.c b/arch/ppc64/kernel/eeh.c
index 926d3110dfd7..b23159464abb 100644
--- a/arch/ppc64/kernel/eeh.c
+++ b/arch/ppc64/kernel/eeh.c
@@ -397,6 +397,28 @@ void __init pci_addr_cache_build(void)
397/* --------------------------------------------------------------- */ 397/* --------------------------------------------------------------- */
398/* Above lies the PCI Address Cache. Below lies the EEH event infrastructure */ 398/* Above lies the PCI Address Cache. Below lies the EEH event infrastructure */
399 399
400void eeh_slot_error_detail (struct pci_dn *pdn, int severity)
401{
402 unsigned long flags;
403 int rc;
404
405 /* Log the error with the rtas logger */
406 spin_lock_irqsave(&slot_errbuf_lock, flags);
407 memset(slot_errbuf, 0, eeh_error_buf_size);
408
409 rc = rtas_call(ibm_slot_error_detail,
410 8, 1, NULL, pdn->eeh_config_addr,
411 BUID_HI(pdn->phb->buid),
412 BUID_LO(pdn->phb->buid), NULL, 0,
413 virt_to_phys(slot_errbuf),
414 eeh_error_buf_size,
415 severity);
416
417 if (rc == 0)
418 log_error(slot_errbuf, ERR_TYPE_RTAS_LOG, 0);
419 spin_unlock_irqrestore(&slot_errbuf_lock, flags);
420}
421
400/** 422/**
401 * eeh_register_notifier - Register to find out about EEH events. 423 * eeh_register_notifier - Register to find out about EEH events.
402 * @nb: notifier block to callback on events 424 * @nb: notifier block to callback on events
@@ -454,9 +476,12 @@ static void eeh_panic(struct pci_dev *dev, int reset_state)
454 * Since the panic_on_oops sysctl is used to halt the system 476 * Since the panic_on_oops sysctl is used to halt the system
455 * in light of potential corruption, we can use it here. 477 * in light of potential corruption, we can use it here.
456 */ 478 */
457 if (panic_on_oops) 479 if (panic_on_oops) {
480 struct device_node *dn = pci_device_to_OF_node(dev);
481 eeh_slot_error_detail (PCI_DN(dn), 2 /* Permanent Error */);
458 panic("EEH: MMIO failure (%d) on device:%s\n", reset_state, 482 panic("EEH: MMIO failure (%d) on device:%s\n", reset_state,
459 pci_name(dev)); 483 pci_name(dev));
484 }
460 else { 485 else {
461 __get_cpu_var(ignored_failures)++; 486 __get_cpu_var(ignored_failures)++;
462 printk(KERN_INFO "EEH: Ignored MMIO failure (%d) on device:%s\n", 487 printk(KERN_INFO "EEH: Ignored MMIO failure (%d) on device:%s\n",
@@ -539,7 +564,7 @@ int eeh_dn_check_failure(struct device_node *dn, struct pci_dev *dev)
539 int ret; 564 int ret;
540 int rets[3]; 565 int rets[3];
541 unsigned long flags; 566 unsigned long flags;
542 int rc, reset_state; 567 int reset_state;
543 struct eeh_event *event; 568 struct eeh_event *event;
544 struct pci_dn *pdn; 569 struct pci_dn *pdn;
545 570
@@ -603,20 +628,7 @@ int eeh_dn_check_failure(struct device_node *dn, struct pci_dev *dev)
603 628
604 reset_state = rets[0]; 629 reset_state = rets[0];
605 630
606 spin_lock_irqsave(&slot_errbuf_lock, flags); 631 eeh_slot_error_detail (pdn, 1 /* Temporary Error */);
607 memset(slot_errbuf, 0, eeh_error_buf_size);
608
609 rc = rtas_call(ibm_slot_error_detail,
610 8, 1, NULL, pdn->eeh_config_addr,
611 BUID_HI(pdn->phb->buid),
612 BUID_LO(pdn->phb->buid), NULL, 0,
613 virt_to_phys(slot_errbuf),
614 eeh_error_buf_size,
615 1 /* Temporary Error */);
616
617 if (rc == 0)
618 log_error(slot_errbuf, ERR_TYPE_RTAS_LOG, 0);
619 spin_unlock_irqrestore(&slot_errbuf_lock, flags);
620 632
621 printk(KERN_INFO "EEH: MMIO failure (%d) on device: %s %s\n", 633 printk(KERN_INFO "EEH: MMIO failure (%d) on device: %s %s\n",
622 rets[0], dn->name, dn->full_name); 634 rets[0], dn->name, dn->full_name);
@@ -783,6 +795,8 @@ void __init eeh_init(void)
783 struct device_node *phb, *np; 795 struct device_node *phb, *np;
784 struct eeh_early_enable_info info; 796 struct eeh_early_enable_info info;
785 797
798 spin_lock_init(&slot_errbuf_lock);
799
786 np = of_find_node_by_path("/rtas"); 800 np = of_find_node_by_path("/rtas");
787 if (np == NULL) 801 if (np == NULL)
788 return; 802 return;