|  | /* | 
|  | * edac_module.c | 
|  | * | 
|  | * (C) 2007 www.softwarebitmaker.com | 
|  | * | 
|  | * This file is licensed under the terms of the GNU General Public | 
|  | * License version 2. This program is licensed "as is" without any | 
|  | * warranty of any kind, whether express or implied. | 
|  | * | 
|  | * Author: Doug Thompson <dougthompson@xmission.com> | 
|  | * | 
|  | */ | 
|  | #include <linux/edac.h> | 
|  |  | 
|  | #include "edac_core.h" | 
|  | #include "edac_module.h" | 
|  |  | 
|  | #define EDAC_VERSION "Ver: 2.1.0 " __DATE__ | 
|  |  | 
|  | #ifdef CONFIG_EDAC_DEBUG | 
|  | /* Values of 0 to 4 will generate output */ | 
|  | int edac_debug_level = 2; | 
|  | EXPORT_SYMBOL_GPL(edac_debug_level); | 
|  | #endif | 
|  |  | 
|  | /* scope is to module level only */ | 
|  | struct workqueue_struct *edac_workqueue; | 
|  |  | 
|  | /* | 
|  | * edac_op_state_to_string() | 
|  | */ | 
|  | char *edac_op_state_to_string(int opstate) | 
|  | { | 
|  | if (opstate == OP_RUNNING_POLL) | 
|  | return "POLLED"; | 
|  | else if (opstate == OP_RUNNING_INTERRUPT) | 
|  | return "INTERRUPT"; | 
|  | else if (opstate == OP_RUNNING_POLL_INTR) | 
|  | return "POLL-INTR"; | 
|  | else if (opstate == OP_ALLOC) | 
|  | return "ALLOC"; | 
|  | else if (opstate == OP_OFFLINE) | 
|  | return "OFFLINE"; | 
|  |  | 
|  | return "UNKNOWN"; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * 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_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(); | 
|  |  | 
|  | /* | 
|  | * now set up the mc_kset under the edac class object | 
|  | */ | 
|  | err = edac_sysfs_setup_mc_kset(); | 
|  | if (err) | 
|  | goto error; | 
|  |  | 
|  | /* Setup/Initialize the workq for this core */ | 
|  | err = edac_workqueue_setup(); | 
|  | if (err) { | 
|  | edac_printk(KERN_ERR, EDAC_MC, "init WorkQueue failure\n"); | 
|  | goto workq_fail; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  |  | 
|  | /* Error teardown stack */ | 
|  | workq_fail: | 
|  | edac_sysfs_teardown_mc_kset(); | 
|  |  | 
|  | error: | 
|  | return err; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * edac_exit() | 
|  | *      module exit/termination function | 
|  | */ | 
|  | static void __exit edac_exit(void) | 
|  | { | 
|  | debugf0("%s()\n", __func__); | 
|  |  | 
|  | /* tear down the various subsystems */ | 
|  | edac_workqueue_teardown(); | 
|  | edac_sysfs_teardown_mc_kset(); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * 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 |