2  * Non Fatal Machine Check Exception Reporting
 
   4  * (C) Copyright 2002 Dave Jones. <davej@codemonkey.org.uk>
 
   6  * This file contains routines to check for non-fatal MCEs every 15s
 
  10 #include <linux/init.h>
 
  11 #include <linux/types.h>
 
  12 #include <linux/kernel.h>
 
  13 #include <linux/jiffies.h>
 
  14 #include <linux/workqueue.h>
 
  15 #include <linux/interrupt.h>
 
  16 #include <linux/smp.h>
 
  17 #include <linux/module.h>
 
  19 #include <asm/processor.h>
 
  20 #include <asm/system.h>
 
  27 #define MCE_RATE        15*HZ   /* timer rate is 15s */
 
  29 static void mce_checkregs(void *info)
 
  34         for (i = firstbank; i < nr_mce_banks; i++) {
 
  35                 rdmsr(MSR_IA32_MC0_STATUS+i*4, low, high);
 
  38                         printk(KERN_INFO "MCE: The hardware reports a non "
 
  39                                 "fatal, correctable incident occurred on "
 
  42                         printk(KERN_INFO "Bank %d: %08x%08x\n", i, high, low);
 
  45                          * Scrub the error so we don't pick it up in MCE_RATE
 
  48                         wrmsr(MSR_IA32_MC0_STATUS+i*4, 0UL, 0UL);
 
  52                         add_taint(TAINT_MACHINE_CHECK);
 
  57 static void mce_work_fn(struct work_struct *work);
 
  58 static DECLARE_DELAYED_WORK(mce_work, mce_work_fn);
 
  60 static void mce_work_fn(struct work_struct *work)
 
  62         on_each_cpu(mce_checkregs, NULL, 1);
 
  63         schedule_delayed_work(&mce_work, round_jiffies_relative(MCE_RATE));
 
  66 static int __init init_nonfatal_mce_checker(void)
 
  68         struct cpuinfo_x86 *c = &boot_cpu_data;
 
  70         /* Check for MCE support */
 
  71         if (!cpu_has(c, X86_FEATURE_MCE))
 
  74         /* Check for PPro style MCA */
 
  75         if (!cpu_has(c, X86_FEATURE_MCA))
 
  78         /* Some Athlons misbehave when we frob bank 0 */
 
  79         if (boot_cpu_data.x86_vendor == X86_VENDOR_AMD &&
 
  80                 boot_cpu_data.x86 == 6)
 
  86          * Check for non-fatal errors every MCE_RATE s
 
  88         schedule_delayed_work(&mce_work, round_jiffies_relative(MCE_RATE));
 
  89         printk(KERN_INFO "Machine check exception polling timer started.\n");
 
  92 module_init(init_nonfatal_mce_checker);
 
  94 MODULE_LICENSE("GPL");