| 
									
										
										
										
											2012-07-30 14:43:02 -07:00
										 |  |  | #include <linux/module.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "notifier-error-inject.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int debugfs_errno_set(void *data, u64 val) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	*(int *)data = clamp_t(int, val, -MAX_ERRNO, 0); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int debugfs_errno_get(void *data, u64 *val) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	*val = *(int *)data; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DEFINE_SIMPLE_ATTRIBUTE(fops_errno, debugfs_errno_get, debugfs_errno_set, | 
					
						
							|  |  |  | 			"%lld\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-29 12:23:28 -04:00
										 |  |  | static struct dentry *debugfs_create_errno(const char *name, umode_t mode, | 
					
						
							| 
									
										
										
										
											2012-07-30 14:43:02 -07:00
										 |  |  | 				struct dentry *parent, int *value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return debugfs_create_file(name, mode, parent, value, &fops_errno); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int notifier_err_inject_callback(struct notifier_block *nb, | 
					
						
							|  |  |  | 				unsigned long val, void *p) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int err = 0; | 
					
						
							|  |  |  | 	struct notifier_err_inject *err_inject = | 
					
						
							|  |  |  | 		container_of(nb, struct notifier_err_inject, nb); | 
					
						
							|  |  |  | 	struct notifier_err_inject_action *action; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (action = err_inject->actions; action->name; action++) { | 
					
						
							|  |  |  | 		if (action->val == val) { | 
					
						
							|  |  |  | 			err = action->error; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (err) | 
					
						
							|  |  |  | 		pr_info("Injecting error (%d) to %s\n", err, action->name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return notifier_from_errno(err); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct dentry *notifier_err_inject_dir; | 
					
						
							|  |  |  | EXPORT_SYMBOL_GPL(notifier_err_inject_dir); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct dentry *notifier_err_inject_init(const char *name, struct dentry *parent, | 
					
						
							|  |  |  | 			struct notifier_err_inject *err_inject, int priority) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct notifier_err_inject_action *action; | 
					
						
							| 
									
										
										
										
											2013-03-29 12:23:28 -04:00
										 |  |  | 	umode_t mode = S_IFREG | S_IRUSR | S_IWUSR; | 
					
						
							| 
									
										
										
										
											2012-07-30 14:43:02 -07:00
										 |  |  | 	struct dentry *dir; | 
					
						
							|  |  |  | 	struct dentry *actions_dir; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	err_inject->nb.notifier_call = notifier_err_inject_callback; | 
					
						
							|  |  |  | 	err_inject->nb.priority = priority; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	dir = debugfs_create_dir(name, parent); | 
					
						
							|  |  |  | 	if (!dir) | 
					
						
							|  |  |  | 		return ERR_PTR(-ENOMEM); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	actions_dir = debugfs_create_dir("actions", dir); | 
					
						
							|  |  |  | 	if (!actions_dir) | 
					
						
							|  |  |  | 		goto fail; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (action = err_inject->actions; action->name; action++) { | 
					
						
							|  |  |  | 		struct dentry *action_dir; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		action_dir = debugfs_create_dir(action->name, actions_dir); | 
					
						
							|  |  |  | 		if (!action_dir) | 
					
						
							|  |  |  | 			goto fail; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/*
 | 
					
						
							|  |  |  | 		 * Create debugfs r/w file containing action->error. If | 
					
						
							|  |  |  | 		 * notifier call chain is called with action->val, it will | 
					
						
							|  |  |  | 		 * fail with the error code | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 		if (!debugfs_create_errno("error", mode, action_dir, | 
					
						
							|  |  |  | 					&action->error)) | 
					
						
							|  |  |  | 			goto fail; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return dir; | 
					
						
							|  |  |  | fail: | 
					
						
							|  |  |  | 	debugfs_remove_recursive(dir); | 
					
						
							|  |  |  | 	return ERR_PTR(-ENOMEM); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | EXPORT_SYMBOL_GPL(notifier_err_inject_init); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int __init err_inject_init(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	notifier_err_inject_dir = | 
					
						
							|  |  |  | 		debugfs_create_dir("notifier-error-inject", NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!notifier_err_inject_dir) | 
					
						
							|  |  |  | 		return -ENOMEM; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void __exit err_inject_exit(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	debugfs_remove_recursive(notifier_err_inject_dir); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | module_init(err_inject_init); | 
					
						
							|  |  |  | module_exit(err_inject_exit); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | MODULE_DESCRIPTION("Notifier error injection module"); | 
					
						
							|  |  |  | MODULE_LICENSE("GPL"); | 
					
						
							|  |  |  | MODULE_AUTHOR("Akinobu Mita <akinobu.mita@gmail.com>"); |