Release 4.7 drivers/scsi/scsi_dh.c
/*
* 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 <linux/slab.h>
#include <linux/module.h>
#include <scsi/scsi_dh.h>
#include "scsi_priv.h"
static DEFINE_SPINLOCK(list_lock);
static LIST_HEAD(scsi_dh_list);
struct scsi_dh_blist {
const char *vendor;
const char *model;
const char *driver;
};
static const struct scsi_dh_blist scsi_dh_blist[] = {
{"DGC", "RAID", "clariion" },
{"DGC", "DISK", "clariion" },
{"DGC", "VRAID", "clariion" },
{"COMPAQ", "MSA1000 VOLUME", "hp_sw" },
{"COMPAQ", "HSV110", "hp_sw" },
{"HP", "HSV100", "hp_sw"},
{"DEC", "HSG80", "hp_sw"},
{"IBM", "1722", "rdac", },
{"IBM", "1724", "rdac", },
{"IBM", "1726", "rdac", },
{"IBM", "1742", "rdac", },
{"IBM", "1745", "rdac", },
{"IBM", "1746", "rdac", },
{"IBM", "1813", "rdac", },
{"IBM", "1814", "rdac", },
{"IBM", "1815", "rdac", },
{"IBM", "1818", "rdac", },
{"IBM", "3526", "rdac", },
{"SGI", "TP9", "rdac", },
{"SGI", "IS", "rdac", },
{"STK", "OPENstorage D280", "rdac", },
{"STK", "FLEXLINE 380", "rdac", },
{"SUN", "CSM", "rdac", },
{"SUN", "LCSM100", "rdac", },
{"SUN", "STK6580_6780", "rdac", },
{"SUN", "SUN_6180", "rdac", },
{"SUN", "ArrayStorage", "rdac", },
{"DELL", "MD3", "rdac", },
{"NETAPP", "INF-01-00", "rdac", },
{"LSI", "INF-01-00", "rdac", },
{"ENGENIO", "INF-01-00", "rdac", },
{NULL, NULL, NULL },
};
static const char *
scsi_dh_find_driver(struct scsi_device *sdev)
{
const struct scsi_dh_blist *b;
if (scsi_device_tpgs(sdev))
return "alua";
for (b = scsi_dh_blist; b->vendor; b++) {
if (!strncmp(sdev->vendor, b->vendor, strlen(b->vendor)) &&
!strncmp(sdev->model, b->model, strlen(b->model))) {
return b->driver;
}
}
return NULL;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
christoph hellwig | christoph hellwig | 94 | 100.00% | 1 | 100.00% |
| Total | 94 | 100.00% | 1 | 100.00% |
static struct scsi_device_handler *__scsi_dh_lookup(const char *name)
{
struct scsi_device_handler *tmp, *found = NULL;
spin_lock(&list_lock);
list_for_each_entry(tmp, &scsi_dh_list, list) {
if (!strncmp(tmp->name, name, strlen(tmp->name))) {
found = tmp;
break;
}
}
spin_unlock(&list_lock);
return found;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
chandra seetharaman | chandra seetharaman | 65 | 87.84% | 1 | 33.33% |
hannes reinecke | hannes reinecke | 8 | 10.81% | 1 | 33.33% |
christoph hellwig | christoph hellwig | 1 | 1.35% | 1 | 33.33% |
| Total | 74 | 100.00% | 3 | 100.00% |
static struct scsi_device_handler *scsi_dh_lookup(const char *name)
{
struct scsi_device_handler *dh;
dh = __scsi_dh_lookup(name);
if (!dh) {
request_module("scsi_dh_%s", name);
dh = __scsi_dh_lookup(name);
}
return dh;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
christoph hellwig | christoph hellwig | 47 | 95.92% | 1 | 50.00% |
paul mackerras | paul mackerras | 2 | 4.08% | 1 | 50.00% |
| Total | 49 | 100.00% | 2 | 100.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 error;
if (!try_module_get(scsi_dh->module))
return -EINVAL;
error = scsi_dh->attach(sdev);
if (error) {
sdev_printk(KERN_ERR, sdev, "%s: Attach failed (%d)\n",
scsi_dh->name, error);
module_put(scsi_dh->module);
} else
sdev->handler = scsi_dh;
return error;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
christoph hellwig | christoph hellwig | 44 | 55.00% | 3 | 60.00% |
hannes reinecke | hannes reinecke | 29 | 36.25% | 1 | 20.00% |
chandra seetharaman | chandra seetharaman | 7 | 8.75% | 1 | 20.00% |
| Total | 80 | 100.00% | 5 | 100.00% |
/*
* scsi_dh_handler_detach - Detach a device handler from a device
* @sdev - SCSI device the device handler should be detached from
*/
static void scsi_dh_handler_detach(struct scsi_device *sdev)
{
sdev->handler->detach(sdev);
sdev_printk(KERN_NOTICE, sdev, "%s: Detached\n", sdev->handler->name);
module_put(sdev->handler->module);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
christoph hellwig | christoph hellwig | 30 | 68.18% | 4 | 66.67% |
chandra seetharaman | chandra seetharaman | 8 | 18.18% | 1 | 16.67% |
mike christie | mike christie | 6 | 13.64% | 1 | 16.67% |
| Total | 44 | 100.00% | 6 | 100.00% |
int scsi_dh_add_device(struct scsi_device *sdev)
{
struct scsi_device_handler *devinfo = NULL;
const char *drv;
int err = 0;
drv = scsi_dh_find_driver(sdev);
if (drv)
devinfo = __scsi_dh_lookup(drv);
if (devinfo)
err = scsi_dh_handler_attach(sdev, devinfo);
return err;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
hannes reinecke | hannes reinecke | 36 | 59.02% | 4 | 57.14% |
christoph hellwig | christoph hellwig | 25 | 40.98% | 3 | 42.86% |
| Total | 61 | 100.00% | 7 | 100.00% |
void scsi_dh_release_device(struct scsi_device *sdev)
{
if (sdev->handler)
scsi_dh_handler_detach(sdev);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
hannes reinecke | hannes reinecke | 12 | 57.14% | 1 | 20.00% |
christoph hellwig | christoph hellwig | 7 | 33.33% | 3 | 60.00% |
jun'ichi nomura | jun'ichi nomura | 2 | 9.52% | 1 | 20.00% |
| Total | 21 | 100.00% | 5 | 100.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 (__scsi_dh_lookup(scsi_dh->name))
return -EBUSY;
if (!scsi_dh->attach || !scsi_dh->detach)
return -EINVAL;
spin_lock(&list_lock);
list_add(&scsi_dh->list, &scsi_dh_list);
spin_unlock(&list_lock);
printk(KERN_INFO "%s: device handler registered\n", scsi_dh->name);
return SCSI_DH_OK;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
chandra seetharaman | chandra seetharaman | 36 | 48.00% | 1 | 25.00% |
hannes reinecke | hannes reinecke | 22 | 29.33% | 1 | 25.00% |
christoph hellwig | christoph hellwig | 17 | 22.67% | 2 | 50.00% |
| Total | 75 | 100.00% | 4 | 100.00% |
EXPORT_SYMBOL_GPL(scsi_register_device_handler);
/*
* 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)
{
if (!__scsi_dh_lookup(scsi_dh->name))
return -ENODEV;
spin_lock(&list_lock);
list_del(&scsi_dh->list);
spin_unlock(&list_lock);
printk(KERN_INFO "%s: device handler unregistered\n", scsi_dh->name);
return SCSI_DH_OK;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
chandra seetharaman | chandra seetharaman | 39 | 68.42% | 1 | 33.33% |
hannes reinecke | hannes reinecke | 17 | 29.82% | 1 | 33.33% |
christoph hellwig | christoph hellwig | 1 | 1.75% | 1 | 33.33% |
| Total | 57 | 100.00% | 3 | 100.00% |
EXPORT_SYMBOL_GPL(scsi_unregister_device_handler);
static struct scsi_device *get_sdev_from_queue(struct request_queue *q)
{
struct scsi_device *sdev;
unsigned long flags;
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);
return sdev;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
chandra seetharaman | chandra seetharaman | 30 | 44.78% | 1 | 25.00% |
christoph hellwig | christoph hellwig | 25 | 37.31% | 1 | 25.00% |
babu moger | babu moger | 11 | 16.42% | 1 | 25.00% |
mike snitzer | mike snitzer | 1 | 1.49% | 1 | 25.00% |
| Total | 67 | 100.00% | 4 | 100.00% |
/*
* scsi_dh_activate - activate the path associated with the scsi_device
* corresponding to the given request queue.
* 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.
*
*/
int scsi_dh_activate(struct request_queue *q, activate_complete fn, void *data)
{
struct scsi_device *sdev;
int err = SCSI_DH_NOSYS;
sdev = get_sdev_from_queue(q);
if (!sdev) {
if (fn)
fn(data, err);
return err;
}
if (!sdev->handler)
goto out_fn;
err = SCSI_DH_NOTCONN;
if (sdev->sdev_state == SDEV_CANCEL ||
sdev->sdev_state == SDEV_DEL)
goto out_fn;
err = SCSI_DH_DEV_OFFLINED;
if (sdev->sdev_state == SDEV_OFFLINE)
goto out_fn;
if (sdev->handler->activate)
err = sdev->handler->activate(sdev, fn, data);
out_put_device:
put_device(&sdev->sdev_gendev);
return err;
out_fn:
if (fn)
fn(data, err);
goto out_put_device;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
christoph hellwig | christoph hellwig | 77 | 50.33% | 2 | 25.00% |
chandra seetharaman | chandra seetharaman | 29 | 18.95% | 2 | 25.00% |
menny hamburger | menny hamburger | 19 | 12.42% | 1 | 12.50% |
babu moger | babu moger | 18 | 11.76% | 1 | 12.50% |
mike snitzer | mike snitzer | 6 | 3.92% | 1 | 12.50% |
hannes reinecke | hannes reinecke | 4 | 2.61% | 1 | 12.50% |
| Total | 153 | 100.00% | 8 | 100.00% |
EXPORT_SYMBOL_GPL(scsi_dh_activate);
/*
* 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)
{
struct scsi_device *sdev;
int err = -SCSI_DH_NOSYS;
sdev = get_sdev_from_queue(q);
if (!sdev)
return err;
if (sdev->handler && sdev->handler->set_params)
err = sdev->handler->set_params(sdev, params);
put_device(&sdev->sdev_gendev);
return err;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
chandra seetharaman | chandra seetharaman | 57 | 74.03% | 1 | 33.33% |
christoph hellwig | christoph hellwig | 20 | 25.97% | 2 | 66.67% |
| Total | 77 | 100.00% | 3 | 100.00% |
EXPORT_SYMBOL_GPL(scsi_dh_set_params);
/*
* scsi_dh_attach - Attach device handler
* @q - Request queue that is associated with the scsi_device
* the handler should be attached to
* @name - name of the handler to attach
*/
int scsi_dh_attach(struct request_queue *q, const char *name)
{
struct scsi_device *sdev;
struct scsi_device_handler *scsi_dh;
int err = 0;
sdev = get_sdev_from_queue(q);
if (!sdev)
return -ENODEV;
scsi_dh = scsi_dh_lookup(name);
if (!scsi_dh) {
err = -EINVAL;
goto out_put_device;
}
if (sdev->handler) {
if (sdev->handler != scsi_dh)
err = -EBUSY;
goto out_put_device;
}
err = scsi_dh_handler_attach(sdev, scsi_dh);
out_put_device:
put_device(&sdev->sdev_gendev);
return err;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
hannes reinecke | hannes reinecke | 75 | 65.79% | 1 | 25.00% |
christoph hellwig | christoph hellwig | 39 | 34.21% | 3 | 75.00% |
| Total | 114 | 100.00% | 4 | 100.00% |
EXPORT_SYMBOL_GPL(scsi_dh_attach);
/*
* scsi_dh_attached_handler_name - Get attached device handler's name
* @q - Request queue that is associated with the scsi_device
* that may have a device handler attached
* @gfp - the GFP mask used in the kmalloc() call when allocating memory
*
* Returns name of attached handler, NULL if no handler is attached.
* Caller must take care to free the returned string.
*/
const char *scsi_dh_attached_handler_name(struct request_queue *q, gfp_t gfp)
{
struct scsi_device *sdev;
const char *handler_name = NULL;
sdev = get_sdev_from_queue(q);
if (!sdev)
return NULL;
if (sdev->handler)
handler_name = kstrdup(sdev->handler->name, gfp);
put_device(&sdev->sdev_gendev);
return handler_name;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
mike snitzer | mike snitzer | 69 | 95.83% | 1 | 33.33% |
christoph hellwig | christoph hellwig | 3 | 4.17% | 2 | 66.67% |
| Total | 72 | 100.00% | 3 | 100.00% |
EXPORT_SYMBOL_GPL(scsi_dh_attached_handler_name);
Overall Contributors
| Person | Tokens | Prop | Commits | CommitProp |
christoph hellwig | christoph hellwig | 719 | 51.88% | 11 | 36.67% |
chandra seetharaman | chandra seetharaman | 312 | 22.51% | 4 | 13.33% |
hannes reinecke | hannes reinecke | 208 | 15.01% | 6 | 20.00% |
mike snitzer | mike snitzer | 83 | 5.99% | 2 | 6.67% |
babu moger | babu moger | 29 | 2.09% | 1 | 3.33% |
menny hamburger | menny hamburger | 19 | 1.37% | 1 | 3.33% |
mike christie | mike christie | 6 | 0.43% | 1 | 3.33% |
tejun heo | tejun heo | 5 | 0.36% | 1 | 3.33% |
jun'ichi nomura | jun'ichi nomura | 2 | 0.14% | 1 | 3.33% |
paul mackerras | paul mackerras | 2 | 0.14% | 1 | 3.33% |
paul gortmaker | paul gortmaker | 1 | 0.07% | 1 | 3.33% |
| Total | 1386 | 100.00% | 30 | 100.00% |
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.