#include #include #include "edac_mc.h" #include "edac_module.h" #define EDAC_MC_VERSION "Ver: 2.0.3" __DATE__ #ifdef CONFIG_EDAC_DEBUG /* Values of 0 to 4 will generate output */ int edac_debug_level = 1; EXPORT_SYMBOL_GPL(edac_debug_level); #endif /* scope is to module level only */ struct workqueue_struct *edac_workqueue; /* private to this file */ static struct task_struct *edac_thread; /* * sysfs object: /sys/devices/system/edac * need to export to other files in this modules */ static struct sysdev_class edac_class = { set_kset_name("edac"), }; static int edac_class_valid = 0; /* * edac_get_edac_class() * * return pointer to the edac class of 'edac' */ struct sysdev_class *edac_get_edac_class(void) { struct sysdev_class *classptr=NULL; if (edac_class_valid) classptr = &edac_class; return classptr; } /* * edac_register_sysfs_edac_name() * * register the 'edac' into /sys/devices/system * * return: * 0 success * !0 error */ static int edac_register_sysfs_edac_name(void) { int err; /* create the /sys/devices/system/edac directory */ err = sysdev_class_register(&edac_class); if (err) { debugf1("%s() error=%d\n", __func__, err); return err; } edac_class_valid = 1; return 0; } /* * sysdev_class_unregister() * * unregister the 'edac' from /sys/devices/system */ static void edac_unregister_sysfs_edac_name(void) { /* only if currently registered, then unregister it */ if (edac_class_valid) sysdev_class_unregister(&edac_class); edac_class_valid = 0; } /* * Check MC status every edac_get_poll_msec(). * Check PCI status every edac_get_poll_msec() as well. * * This where the work gets done for edac. * * SMP safe, doesn't use NMI, and auto-rate-limits. */ static void do_edac_check(void) { debugf3("%s()\n", __func__); /* perform the poll activities */ edac_check_mc_devices(); edac_pci_do_parity_check(); } /* * Action thread for EDAC to perform the POLL operations */ static int edac_kernel_thread(void *arg) { int msec; while (!kthread_should_stop()) { do_edac_check(); /* goto sleep for the interval */ msec = (HZ * edac_get_poll_msec()) / 1000; schedule_timeout_interruptible(msec); try_to_freeze(); } return 0; } /* * edac_workqueue_setup * initialize the edac work queue for polling operations */ static int edac_workqueue_setup(void) { edac_workqueue = create_singlethread_workqueue("edac-poller"); if (edac_workqueue == NULL) return -ENODEV; else return 0; } /* * edac_workqueue_teardown * teardown the edac workqueue */ static void edac_workqueue_teardown(void) { if (edac_workqueue) { flush_workqueue(edac_workqueue); destroy_workqueue(edac_workqueue); edac_workqueue = NULL; } } /* * edac_init * module initialization entry point */ static int __init edac_init(void) { int err = 0; edac_printk(KERN_INFO, EDAC_MC, EDAC_MC_VERSION "\n"); /* * Harvest and clear any boot/initialization PCI parity errors * * FIXME: This only clears errors logged by devices present at time of * module initialization. We should also do an initial clear * of each newly hotplugged device. */ edac_pci_clear_parity_errors(); /* * perform the registration of the /sys/devices/system/edac object */ if (edac_register_sysfs_edac_name()) { edac_printk(KERN_ERR, EDAC_MC, "Error initializing 'edac' kobject\n"); err = -ENODEV; goto error; } /* Create the MC sysfs entries, must be first */ if (edac_sysfs_memctrl_setup()) { edac_printk(KERN_ERR, EDAC_MC, "Error initializing sysfs code\n"); err = -ENODEV; goto error_sysfs; } /* Create the PCI parity sysfs entries */ if (edac_sysfs_pci_setup()) { edac_printk(KERN_ERR, EDAC_MC, "PCI: Error initializing sysfs code\n"); err = -ENODEV; goto error_mem; } /* Setup/Initialize the edac_device system */ err = edac_workqueue_setup(); if (err) { edac_printk(KERN_ERR, EDAC_MC, "init WorkQueue failure\n"); goto error_pci; } /* create our kernel thread */ edac_thread = kthread_run(edac_kernel_thread, NULL, "kedac"); if (IS_ERR(edac_thread)) { err = PTR_ERR(edac_thread); goto error_work; } return 0; /* Error teardown stack */ error_work: edac_workqueue_teardown(); error_pci: edac_sysfs_pci_teardown(); error_mem: edac_sysfs_memctrl_teardown(); error_sysfs: edac_unregister_sysfs_edac_name(); error: return err; } /* * edac_exit() * module exit/termination function */ static void __exit edac_exit(void) { debugf0("%s()\n", __func__); kthread_stop(edac_thread); /* tear down the various subsystems*/ edac_workqueue_teardown(); edac_sysfs_memctrl_teardown(); edac_sysfs_pci_teardown(); edac_unregister_sysfs_edac_name(); } /* * Inform the kernel of our entry and exit points */ module_init(edac_init); module_exit(edac_exit); MODULE_LICENSE("GPL"); MODULE_AUTHOR("Doug Thompson www.softwarebitmaker.com, et al"); MODULE_DESCRIPTION("Core library routines for EDAC reporting"); /* refer to *_sysfs.c files for parameters that are exported via sysfs */ #ifdef CONFIG_EDAC_DEBUG module_param(edac_debug_level, int, 0644); MODULE_PARM_DESC(edac_debug_level, "Debug level"); #endif