| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * SCSI device handler infrastruture. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or modify it | 
					
						
							|  |  |  |  * under the terms of the GNU General Public License as published by the | 
					
						
							|  |  |  |  * Free Software Foundation; either version 2 of the License, or (at your | 
					
						
							|  |  |  |  * option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, but | 
					
						
							|  |  |  |  * WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | 
					
						
							|  |  |  |  * General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License along | 
					
						
							|  |  |  |  * with this program; if not, write to the Free Software Foundation, Inc., | 
					
						
							|  |  |  |  * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright IBM Corporation, 2007 | 
					
						
							|  |  |  |  *      Authors: | 
					
						
							|  |  |  |  *               Chandra Seetharaman <sekharan@us.ibm.com> | 
					
						
							|  |  |  |  *               Mike Anderson <andmike@linux.vnet.ibm.com> | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <scsi/scsi_dh.h>
 | 
					
						
							|  |  |  | #include "../scsi_priv.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-17 16:53:33 -07:00
										 |  |  | struct scsi_dh_devinfo_list { | 
					
						
							|  |  |  | 	struct list_head node; | 
					
						
							|  |  |  | 	char vendor[9]; | 
					
						
							|  |  |  | 	char model[17]; | 
					
						
							|  |  |  | 	struct scsi_device_handler *handler; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  | static DEFINE_SPINLOCK(list_lock); | 
					
						
							|  |  |  | static LIST_HEAD(scsi_dh_list); | 
					
						
							| 
									
										
										
										
											2008-07-17 16:53:33 -07:00
										 |  |  | static LIST_HEAD(scsi_dh_dev_list); | 
					
						
							| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | static struct scsi_device_handler *get_device_handler(const char *name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct scsi_device_handler *tmp, *found = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spin_lock(&list_lock); | 
					
						
							|  |  |  | 	list_for_each_entry(tmp, &scsi_dh_list, list) { | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  | 		if (!strncmp(tmp->name, name, strlen(tmp->name))) { | 
					
						
							| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  | 			found = tmp; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	spin_unlock(&list_lock); | 
					
						
							|  |  |  | 	return found; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-17 16:53:33 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | static struct scsi_device_handler * | 
					
						
							|  |  |  | scsi_dh_cache_lookup(struct scsi_device *sdev) | 
					
						
							| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2008-07-17 16:53:33 -07:00
										 |  |  | 	struct scsi_dh_devinfo_list *tmp; | 
					
						
							|  |  |  | 	struct scsi_device_handler *found_dh = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spin_lock(&list_lock); | 
					
						
							|  |  |  | 	list_for_each_entry(tmp, &scsi_dh_dev_list, node) { | 
					
						
							|  |  |  | 		if (!strncmp(sdev->vendor, tmp->vendor, strlen(tmp->vendor)) && | 
					
						
							|  |  |  | 		    !strncmp(sdev->model, tmp->model, strlen(tmp->model))) { | 
					
						
							|  |  |  | 			found_dh = tmp->handler; | 
					
						
							|  |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-07-17 16:53:33 -07:00
										 |  |  | 	spin_unlock(&list_lock); | 
					
						
							| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-17 16:53:33 -07:00
										 |  |  | 	return found_dh; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int scsi_dh_handler_lookup(struct scsi_device_handler *scsi_dh, | 
					
						
							|  |  |  | 				  struct scsi_device *sdev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int i, found = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for(i = 0; scsi_dh->devlist[i].vendor; i++) { | 
					
						
							|  |  |  | 		if (!strncmp(sdev->vendor, scsi_dh->devlist[i].vendor, | 
					
						
							|  |  |  | 			     strlen(scsi_dh->devlist[i].vendor)) && | 
					
						
							|  |  |  | 		    !strncmp(sdev->model, scsi_dh->devlist[i].model, | 
					
						
							|  |  |  | 			     strlen(scsi_dh->devlist[i].model))) { | 
					
						
							|  |  |  | 			found = 1; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return found; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * device_handler_match - Attach a device handler to a device | 
					
						
							|  |  |  |  * @scsi_dh - The device handler to match against or NULL | 
					
						
							|  |  |  |  * @sdev - SCSI device to be tested against @scsi_dh | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Tests @sdev against the device handler @scsi_dh or against | 
					
						
							|  |  |  |  * all registered device_handler if @scsi_dh == NULL. | 
					
						
							|  |  |  |  * Returns the found device handler or NULL if not found. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static struct scsi_device_handler * | 
					
						
							|  |  |  | device_handler_match(struct scsi_device_handler *scsi_dh, | 
					
						
							|  |  |  | 		     struct scsi_device *sdev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct scsi_device_handler *found_dh = NULL; | 
					
						
							|  |  |  | 	struct scsi_dh_devinfo_list *tmp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	found_dh = scsi_dh_cache_lookup(sdev); | 
					
						
							|  |  |  | 	if (found_dh) | 
					
						
							|  |  |  | 		return found_dh; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (scsi_dh) { | 
					
						
							|  |  |  | 		if (scsi_dh_handler_lookup(scsi_dh, sdev)) | 
					
						
							|  |  |  | 			found_dh = scsi_dh; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		struct scsi_device_handler *tmp_dh; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		spin_lock(&list_lock); | 
					
						
							|  |  |  | 		list_for_each_entry(tmp_dh, &scsi_dh_list, list) { | 
					
						
							|  |  |  | 			if (scsi_dh_handler_lookup(tmp_dh, sdev)) | 
					
						
							|  |  |  | 				found_dh = tmp_dh; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		spin_unlock(&list_lock); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (found_dh) { /* If device is found, add it to the cache */ | 
					
						
							|  |  |  | 		tmp = kmalloc(sizeof(*tmp), GFP_KERNEL); | 
					
						
							|  |  |  | 		if (tmp) { | 
					
						
							|  |  |  | 			strncpy(tmp->vendor, sdev->vendor, 8); | 
					
						
							|  |  |  | 			strncpy(tmp->model, sdev->model, 16); | 
					
						
							|  |  |  | 			tmp->vendor[8] = '\0'; | 
					
						
							|  |  |  | 			tmp->model[16] = '\0'; | 
					
						
							|  |  |  | 			tmp->handler = found_dh; | 
					
						
							|  |  |  | 			spin_lock(&list_lock); | 
					
						
							|  |  |  | 			list_add(&tmp->node, &scsi_dh_dev_list); | 
					
						
							|  |  |  | 			spin_unlock(&list_lock); | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			found_dh = NULL; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return found_dh; | 
					
						
							| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  |  * scsi_dh_handler_attach - Attach a device handler to a device | 
					
						
							|  |  |  |  * @sdev - SCSI device the device handler should attach to | 
					
						
							|  |  |  |  * @scsi_dh - The device handler to attach | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static int scsi_dh_handler_attach(struct scsi_device *sdev, | 
					
						
							|  |  |  | 				  struct scsi_device_handler *scsi_dh) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int err = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (sdev->scsi_dh_data) { | 
					
						
							|  |  |  | 		if (sdev->scsi_dh_data->scsi_dh != scsi_dh) | 
					
						
							|  |  |  | 			err = -EBUSY; | 
					
						
							| 
									
										
										
										
											2009-06-26 19:30:06 -07:00
										 |  |  | 		else | 
					
						
							|  |  |  | 			kref_get(&sdev->scsi_dh_data->kref); | 
					
						
							|  |  |  | 	} else if (scsi_dh->attach) { | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  | 		err = scsi_dh->attach(sdev); | 
					
						
							| 
									
										
										
										
											2009-06-26 19:30:06 -07:00
										 |  |  | 		if (!err) { | 
					
						
							|  |  |  | 			kref_init(&sdev->scsi_dh_data->kref); | 
					
						
							|  |  |  | 			sdev->scsi_dh_data->sdev = sdev; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  | 	return err; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-26 19:30:06 -07:00
										 |  |  | static void __detach_handler (struct kref *kref) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct scsi_dh_data *scsi_dh_data = container_of(kref, struct scsi_dh_data, kref); | 
					
						
							|  |  |  | 	scsi_dh_data->scsi_dh->detach(scsi_dh_data->sdev); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * scsi_dh_handler_detach - Detach a device handler from a device | 
					
						
							|  |  |  |  * @sdev - SCSI device the device handler should be detached from | 
					
						
							|  |  |  |  * @scsi_dh - Device handler to be detached | 
					
						
							| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  |  * Detach from a device handler. If a device handler is specified, | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:57 -07:00
										 |  |  |  * only detach if the currently attached handler matches @scsi_dh. | 
					
						
							| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  | static void scsi_dh_handler_detach(struct scsi_device *sdev, | 
					
						
							|  |  |  | 				   struct scsi_device_handler *scsi_dh) | 
					
						
							| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  | 	if (!sdev->scsi_dh_data) | 
					
						
							|  |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  | 	if (scsi_dh && scsi_dh != sdev->scsi_dh_data->scsi_dh) | 
					
						
							|  |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  | 	if (!scsi_dh) | 
					
						
							|  |  |  | 		scsi_dh = sdev->scsi_dh_data->scsi_dh; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (scsi_dh && scsi_dh->detach) | 
					
						
							| 
									
										
										
										
											2009-06-26 19:30:06 -07:00
										 |  |  | 		kref_put(&sdev->scsi_dh_data->kref, __detach_handler); | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:57 -07:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Functions for sysfs attribute 'dh_state' | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static ssize_t | 
					
						
							|  |  |  | store_dh_state(struct device *dev, struct device_attribute *attr, | 
					
						
							|  |  |  | 	       const char *buf, size_t count) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct scsi_device *sdev = to_scsi_device(dev); | 
					
						
							|  |  |  | 	struct scsi_device_handler *scsi_dh; | 
					
						
							|  |  |  | 	int err = -EINVAL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!sdev->scsi_dh_data) { | 
					
						
							|  |  |  | 		/*
 | 
					
						
							|  |  |  | 		 * Attach to a device handler | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 		if (!(scsi_dh = get_device_handler(buf))) | 
					
						
							|  |  |  | 			return err; | 
					
						
							|  |  |  | 		err = scsi_dh_handler_attach(sdev, scsi_dh); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		scsi_dh = sdev->scsi_dh_data->scsi_dh; | 
					
						
							|  |  |  | 		if (!strncmp(buf, "detach", 6)) { | 
					
						
							|  |  |  | 			/*
 | 
					
						
							|  |  |  | 			 * Detach from a device handler | 
					
						
							|  |  |  | 			 */ | 
					
						
							|  |  |  | 			scsi_dh_handler_detach(sdev, scsi_dh); | 
					
						
							|  |  |  | 			err = 0; | 
					
						
							|  |  |  | 		} else if (!strncmp(buf, "activate", 8)) { | 
					
						
							|  |  |  | 			/*
 | 
					
						
							|  |  |  | 			 * Activate a device handler | 
					
						
							|  |  |  | 			 */ | 
					
						
							|  |  |  | 			if (scsi_dh->activate) | 
					
						
							| 
									
										
										
										
											2009-10-21 09:22:46 -07:00
										 |  |  | 				err = scsi_dh->activate(sdev, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:57 -07:00
										 |  |  | 			else | 
					
						
							|  |  |  | 				err = 0; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return err<0?err:count; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static ssize_t | 
					
						
							|  |  |  | show_dh_state(struct device *dev, struct device_attribute *attr, char *buf) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct scsi_device *sdev = to_scsi_device(dev); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!sdev->scsi_dh_data) | 
					
						
							|  |  |  | 		return snprintf(buf, 20, "detached\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return snprintf(buf, 20, "%s\n", sdev->scsi_dh_data->scsi_dh->name); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct device_attribute scsi_dh_state_attr = | 
					
						
							|  |  |  | 	__ATTR(dh_state, S_IRUGO | S_IWUSR, show_dh_state, | 
					
						
							|  |  |  | 	       store_dh_state); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * scsi_dh_sysfs_attr_add - Callback for scsi_init_dh | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static int scsi_dh_sysfs_attr_add(struct device *dev, void *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct scsi_device *sdev; | 
					
						
							|  |  |  | 	int err; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!scsi_is_sdev_device(dev)) | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	sdev = to_scsi_device(dev); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	err = device_create_file(&sdev->sdev_gendev, | 
					
						
							|  |  |  | 				 &scsi_dh_state_attr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * scsi_dh_sysfs_attr_remove - Callback for scsi_exit_dh | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static int scsi_dh_sysfs_attr_remove(struct device *dev, void *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct scsi_device *sdev; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!scsi_is_sdev_device(dev)) | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	sdev = to_scsi_device(dev); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	device_remove_file(&sdev->sdev_gendev, | 
					
						
							|  |  |  | 			   &scsi_dh_state_attr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * scsi_dh_notifier - notifier chain callback | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static int scsi_dh_notifier(struct notifier_block *nb, | 
					
						
							|  |  |  | 			    unsigned long action, void *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct device *dev = data; | 
					
						
							|  |  |  | 	struct scsi_device *sdev; | 
					
						
							|  |  |  | 	int err = 0; | 
					
						
							| 
									
										
										
										
											2008-07-17 16:53:33 -07:00
										 |  |  | 	struct scsi_device_handler *devinfo = NULL; | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (!scsi_is_sdev_device(dev)) | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	sdev = to_scsi_device(dev); | 
					
						
							| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  | 	if (action == BUS_NOTIFY_ADD_DEVICE) { | 
					
						
							| 
									
										
										
										
											2009-09-11 10:20:35 -07:00
										 |  |  | 		err = device_create_file(dev, &scsi_dh_state_attr); | 
					
						
							|  |  |  | 		/* don't care about err */ | 
					
						
							| 
									
										
										
										
											2008-07-17 16:53:33 -07:00
										 |  |  | 		devinfo = device_handler_match(NULL, sdev); | 
					
						
							| 
									
										
										
										
											2009-09-11 10:20:35 -07:00
										 |  |  | 		if (devinfo) | 
					
						
							|  |  |  | 			err = scsi_dh_handler_attach(sdev, devinfo); | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  | 	} else if (action == BUS_NOTIFY_DEL_DEVICE) { | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:57 -07:00
										 |  |  | 		device_remove_file(dev, &scsi_dh_state_attr); | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  | 		scsi_dh_handler_detach(sdev, NULL); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return err; | 
					
						
							| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * scsi_dh_notifier_add - Callback for scsi_register_device_handler | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static int scsi_dh_notifier_add(struct device *dev, void *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct scsi_device_handler *scsi_dh = data; | 
					
						
							|  |  |  | 	struct scsi_device *sdev; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!scsi_is_sdev_device(dev)) | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!get_device(dev)) | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	sdev = to_scsi_device(dev); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (device_handler_match(scsi_dh, sdev)) | 
					
						
							|  |  |  | 		scsi_dh_handler_attach(sdev, scsi_dh); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	put_device(dev); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * scsi_dh_notifier_remove - Callback for scsi_unregister_device_handler | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  | static int scsi_dh_notifier_remove(struct device *dev, void *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct scsi_device_handler *scsi_dh = data; | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  | 	struct scsi_device *sdev; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!scsi_is_sdev_device(dev)) | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!get_device(dev)) | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	sdev = to_scsi_device(dev); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	scsi_dh_handler_detach(sdev, scsi_dh); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	put_device(dev); | 
					
						
							| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * scsi_register_device_handler - register a device handler personality | 
					
						
							|  |  |  |  *      module. | 
					
						
							|  |  |  |  * @scsi_dh - device handler to be registered. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Returns 0 on success, -EBUSY if handler already registered. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | int scsi_register_device_handler(struct scsi_device_handler *scsi_dh) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (get_device_handler(scsi_dh->name)) | 
					
						
							|  |  |  | 		return -EBUSY; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spin_lock(&list_lock); | 
					
						
							|  |  |  | 	list_add(&scsi_dh->list, &scsi_dh_list); | 
					
						
							|  |  |  | 	spin_unlock(&list_lock); | 
					
						
							|  |  |  | 	bus_for_each_dev(&scsi_bus_type, NULL, scsi_dh, scsi_dh_notifier_add); | 
					
						
							|  |  |  | 	printk(KERN_INFO "%s: device handler registered\n", scsi_dh->name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return SCSI_DH_OK; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | EXPORT_SYMBOL_GPL(scsi_register_device_handler); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * scsi_unregister_device_handler - register a device handler personality | 
					
						
							|  |  |  |  *      module. | 
					
						
							|  |  |  |  * @scsi_dh - device handler to be unregistered. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Returns 0 on success, -ENODEV if handler not registered. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | int scsi_unregister_device_handler(struct scsi_device_handler *scsi_dh) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2008-07-17 16:53:33 -07:00
										 |  |  | 	struct scsi_dh_devinfo_list *tmp, *pos; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  | 	if (!get_device_handler(scsi_dh->name)) | 
					
						
							|  |  |  | 		return -ENODEV; | 
					
						
							| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	bus_for_each_dev(&scsi_bus_type, NULL, scsi_dh, | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  | 			 scsi_dh_notifier_remove); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  | 	spin_lock(&list_lock); | 
					
						
							|  |  |  | 	list_del(&scsi_dh->list); | 
					
						
							| 
									
										
										
										
											2008-07-17 16:53:33 -07:00
										 |  |  | 	list_for_each_entry_safe(pos, tmp, &scsi_dh_dev_list, node) { | 
					
						
							|  |  |  | 		if (pos->handler == scsi_dh) { | 
					
						
							|  |  |  | 			list_del(&pos->node); | 
					
						
							|  |  |  | 			kfree(pos); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  | 	spin_unlock(&list_lock); | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  | 	printk(KERN_INFO "%s: device handler unregistered\n", scsi_dh->name); | 
					
						
							| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  | 	return SCSI_DH_OK; | 
					
						
							| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  | } | 
					
						
							|  |  |  | EXPORT_SYMBOL_GPL(scsi_unregister_device_handler); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * scsi_dh_activate - activate the path associated with the scsi_device | 
					
						
							|  |  |  |  *      corresponding to the given request queue. | 
					
						
							| 
									
										
										
										
											2009-10-21 09:22:46 -07:00
										 |  |  |  *     Returns immediately without waiting for activation to be completed. | 
					
						
							|  |  |  |  * @q    - Request queue that is associated with the scsi_device to be | 
					
						
							|  |  |  |  *         activated. | 
					
						
							|  |  |  |  * @fn   - Function to be called upon completion of the activation. | 
					
						
							|  |  |  |  *         Function fn is called with data (below) and the error code. | 
					
						
							|  |  |  |  *         Function fn may be called from the same calling context. So, | 
					
						
							|  |  |  |  *         do not hold the lock in the caller which may be needed in fn. | 
					
						
							|  |  |  |  * @data - data passed to the function fn upon completion. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-10-21 09:22:46 -07:00
										 |  |  | int scsi_dh_activate(struct request_queue *q, activate_complete fn, void *data) | 
					
						
							| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  | { | 
					
						
							|  |  |  | 	int err = 0; | 
					
						
							|  |  |  | 	unsigned long flags; | 
					
						
							|  |  |  | 	struct scsi_device *sdev; | 
					
						
							|  |  |  | 	struct scsi_device_handler *scsi_dh = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spin_lock_irqsave(q->queue_lock, flags); | 
					
						
							|  |  |  | 	sdev = q->queuedata; | 
					
						
							|  |  |  | 	if (sdev && sdev->scsi_dh_data) | 
					
						
							|  |  |  | 		scsi_dh = sdev->scsi_dh_data->scsi_dh; | 
					
						
							|  |  |  | 	if (!scsi_dh || !get_device(&sdev->sdev_gendev)) | 
					
						
							|  |  |  | 		err = SCSI_DH_NOSYS; | 
					
						
							|  |  |  | 	spin_unlock_irqrestore(q->queue_lock, flags); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (err) | 
					
						
							|  |  |  | 		return err; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (scsi_dh->activate) | 
					
						
							| 
									
										
										
										
											2009-10-21 09:22:46 -07:00
										 |  |  | 		err = scsi_dh->activate(sdev, fn, data); | 
					
						
							| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  | 	put_device(&sdev->sdev_gendev); | 
					
						
							|  |  |  | 	return err; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | EXPORT_SYMBOL_GPL(scsi_dh_activate); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-03 12:42:33 -07:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * scsi_dh_set_params - set the parameters for the device as per the | 
					
						
							|  |  |  |  *      string specified in params. | 
					
						
							|  |  |  |  * @q - Request queue that is associated with the scsi_device for | 
					
						
							|  |  |  |  *      which the parameters to be set. | 
					
						
							|  |  |  |  * @params - parameters in the following format | 
					
						
							|  |  |  |  *      "no_of_params\0param1\0param2\0param3\0...\0" | 
					
						
							|  |  |  |  *      for example, string for 2 parameters with value 10 and 21 | 
					
						
							|  |  |  |  *      is specified as "2\010\021\0". | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | int scsi_dh_set_params(struct request_queue *q, const char *params) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int err = -SCSI_DH_NOSYS; | 
					
						
							|  |  |  | 	unsigned long flags; | 
					
						
							|  |  |  | 	struct scsi_device *sdev; | 
					
						
							|  |  |  | 	struct scsi_device_handler *scsi_dh = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spin_lock_irqsave(q->queue_lock, flags); | 
					
						
							|  |  |  | 	sdev = q->queuedata; | 
					
						
							|  |  |  | 	if (sdev && sdev->scsi_dh_data) | 
					
						
							|  |  |  | 		scsi_dh = sdev->scsi_dh_data->scsi_dh; | 
					
						
							|  |  |  | 	if (scsi_dh && scsi_dh->set_params && get_device(&sdev->sdev_gendev)) | 
					
						
							|  |  |  | 		err = 0; | 
					
						
							|  |  |  | 	spin_unlock_irqrestore(q->queue_lock, flags); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (err) | 
					
						
							|  |  |  | 		return err; | 
					
						
							|  |  |  | 	err = scsi_dh->set_params(sdev, params); | 
					
						
							|  |  |  | 	put_device(&sdev->sdev_gendev); | 
					
						
							|  |  |  | 	return err; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | EXPORT_SYMBOL_GPL(scsi_dh_set_params); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * scsi_dh_handler_exist - Return TRUE(1) if a device handler exists for | 
					
						
							|  |  |  |  *	the given name. FALSE(0) otherwise. | 
					
						
							|  |  |  |  * @name - name of the device handler. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | int scsi_dh_handler_exist(const char *name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return (get_device_handler(name) != NULL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | EXPORT_SYMBOL_GPL(scsi_dh_handler_exist); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-17 17:49:02 -07:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * scsi_dh_handler_attach - Attach device handler | 
					
						
							|  |  |  |  * @sdev - sdev the handler should be attached to | 
					
						
							|  |  |  |  * @name - name of the handler to attach | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | int scsi_dh_attach(struct request_queue *q, const char *name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unsigned long flags; | 
					
						
							|  |  |  | 	struct scsi_device *sdev; | 
					
						
							|  |  |  | 	struct scsi_device_handler *scsi_dh; | 
					
						
							|  |  |  | 	int err = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	scsi_dh = get_device_handler(name); | 
					
						
							|  |  |  | 	if (!scsi_dh) | 
					
						
							|  |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spin_lock_irqsave(q->queue_lock, flags); | 
					
						
							|  |  |  | 	sdev = q->queuedata; | 
					
						
							|  |  |  | 	if (!sdev || !get_device(&sdev->sdev_gendev)) | 
					
						
							|  |  |  | 		err = -ENODEV; | 
					
						
							|  |  |  | 	spin_unlock_irqrestore(q->queue_lock, flags); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!err) { | 
					
						
							|  |  |  | 		err = scsi_dh_handler_attach(sdev, scsi_dh); | 
					
						
							|  |  |  | 		put_device(&sdev->sdev_gendev); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return err; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | EXPORT_SYMBOL_GPL(scsi_dh_attach); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * scsi_dh_handler_detach - Detach device handler | 
					
						
							|  |  |  |  * @sdev - sdev the handler should be detached from | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This function will detach the device handler only | 
					
						
							|  |  |  |  * if the sdev is not part of the internal list, ie | 
					
						
							|  |  |  |  * if it has been attached manually. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void scsi_dh_detach(struct request_queue *q) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unsigned long flags; | 
					
						
							|  |  |  | 	struct scsi_device *sdev; | 
					
						
							|  |  |  | 	struct scsi_device_handler *scsi_dh = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spin_lock_irqsave(q->queue_lock, flags); | 
					
						
							|  |  |  | 	sdev = q->queuedata; | 
					
						
							|  |  |  | 	if (!sdev || !get_device(&sdev->sdev_gendev)) | 
					
						
							|  |  |  | 		sdev = NULL; | 
					
						
							|  |  |  | 	spin_unlock_irqrestore(q->queue_lock, flags); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!sdev) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (sdev->scsi_dh_data) { | 
					
						
							|  |  |  | 		scsi_dh = sdev->scsi_dh_data->scsi_dh; | 
					
						
							| 
									
										
										
										
											2009-06-26 19:30:06 -07:00
										 |  |  | 		scsi_dh_handler_detach(sdev, scsi_dh); | 
					
						
							| 
									
										
										
										
											2008-07-17 17:49:02 -07:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	put_device(&sdev->sdev_gendev); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | EXPORT_SYMBOL_GPL(scsi_dh_detach); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  | static struct notifier_block scsi_dh_nb = { | 
					
						
							|  |  |  | 	.notifier_call = scsi_dh_notifier | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int __init scsi_dh_init(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int r; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	r = bus_register_notifier(&scsi_bus_type, &scsi_dh_nb); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:57 -07:00
										 |  |  | 	if (!r) | 
					
						
							|  |  |  | 		bus_for_each_dev(&scsi_bus_type, NULL, NULL, | 
					
						
							|  |  |  | 				 scsi_dh_sysfs_attr_add); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  | 	return r; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void __exit scsi_dh_exit(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:57 -07:00
										 |  |  | 	bus_for_each_dev(&scsi_bus_type, NULL, NULL, | 
					
						
							|  |  |  | 			 scsi_dh_sysfs_attr_remove); | 
					
						
							| 
									
										
										
										
											2008-07-17 16:52:51 -07:00
										 |  |  | 	bus_unregister_notifier(&scsi_bus_type, &scsi_dh_nb); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | module_init(scsi_dh_init); | 
					
						
							|  |  |  | module_exit(scsi_dh_exit); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-01 14:49:46 -07:00
										 |  |  | MODULE_DESCRIPTION("SCSI device handler"); | 
					
						
							|  |  |  | MODULE_AUTHOR("Chandra Seetharaman <sekharan@us.ibm.com>"); | 
					
						
							|  |  |  | MODULE_LICENSE("GPL"); |