blob: 084062bb8ee9de341ded2d0bb2d1b4b2ea897ecb [file] [log] [blame]
Mike Christief6dd3372008-05-01 14:49:59 -07001/*
2 * Basic HP/COMPAQ MSA 1000 support. This is only needed if your HW cannot be
3 * upgraded.
4 *
5 * Copyright (C) 2006 Red Hat, Inc. All rights reserved.
6 * Copyright (C) 2006 Mike Christie
Hannes Reinecke2aef6d52008-07-17 16:53:09 -07007 * Copyright (C) 2008 Hannes Reinecke <hare@suse.de>
Mike Christief6dd3372008-05-01 14:49:59 -07008 *
9 * This program is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; either version 2, or (at your option)
12 * any later version.
13 *
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
18 *
19 * You should have received a copy of the GNU General Public License
20 * along with this program; see the file COPYING. If not, write to
21 * the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
22 */
23
Tejun Heo5a0e3ad2010-03-24 17:04:11 +090024#include <linux/slab.h>
Paul Gortmakeracf3368f2011-05-27 09:47:43 -040025#include <linux/module.h>
Mike Christief6dd3372008-05-01 14:49:59 -070026#include <scsi/scsi.h>
27#include <scsi/scsi_dbg.h>
28#include <scsi/scsi_eh.h>
29#include <scsi/scsi_dh.h>
30
Hannes Reinecke2aef6d52008-07-17 16:53:09 -070031#define HP_SW_NAME "hp_sw"
Mike Christief6dd3372008-05-01 14:49:59 -070032
Hannes Reinecke2aef6d52008-07-17 16:53:09 -070033#define HP_SW_TIMEOUT (60 * HZ)
34#define HP_SW_RETRIES 3
35
36#define HP_SW_PATH_UNINITIALIZED -1
37#define HP_SW_PATH_ACTIVE 0
38#define HP_SW_PATH_PASSIVE 1
Mike Christief6dd3372008-05-01 14:49:59 -070039
40struct hp_sw_dh_data {
41 unsigned char sense[SCSI_SENSE_BUFFERSIZE];
Hannes Reinecke2aef6d52008-07-17 16:53:09 -070042 int path_state;
Mike Christief6dd3372008-05-01 14:49:59 -070043 int retries;
Chandra Seetharaman4e2ef862009-10-21 09:22:58 -070044 int retry_cnt;
45 struct scsi_device *sdev;
46 activate_complete callback_fn;
47 void *callback_data;
Mike Christief6dd3372008-05-01 14:49:59 -070048};
49
Chandra Seetharaman4e2ef862009-10-21 09:22:58 -070050static int hp_sw_start_stop(struct hp_sw_dh_data *);
51
Mike Christief6dd3372008-05-01 14:49:59 -070052static inline struct hp_sw_dh_data *get_hp_sw_data(struct scsi_device *sdev)
53{
54 struct scsi_dh_data *scsi_dh_data = sdev->scsi_dh_data;
55 BUG_ON(scsi_dh_data == NULL);
56 return ((struct hp_sw_dh_data *) scsi_dh_data->buf);
57}
58
Hannes Reinecke2aef6d52008-07-17 16:53:09 -070059/*
60 * tur_done - Handle TEST UNIT READY return status
61 * @sdev: sdev the command has been sent to
62 * @errors: blk error code
63 *
64 * Returns SCSI_DH_DEV_OFFLINED if the sdev is on the passive path
65 */
66static int tur_done(struct scsi_device *sdev, unsigned char *sense)
Mike Christief6dd3372008-05-01 14:49:59 -070067{
Hannes Reinecke2aef6d52008-07-17 16:53:09 -070068 struct scsi_sense_hdr sshdr;
69 int ret;
70
71 ret = scsi_normalize_sense(sense, SCSI_SENSE_BUFFERSIZE, &sshdr);
72 if (!ret) {
73 sdev_printk(KERN_WARNING, sdev,
74 "%s: sending tur failed, no sense available\n",
75 HP_SW_NAME);
76 ret = SCSI_DH_IO;
77 goto done;
78 }
79 switch (sshdr.sense_key) {
80 case UNIT_ATTENTION:
81 ret = SCSI_DH_IMM_RETRY;
82 break;
83 case NOT_READY:
84 if ((sshdr.asc == 0x04) && (sshdr.ascq == 2)) {
85 /*
86 * LUN not ready - Initialization command required
87 *
88 * This is the passive path
89 */
90 ret = SCSI_DH_DEV_OFFLINED;
91 break;
92 }
93 /* Fallthrough */
94 default:
95 sdev_printk(KERN_WARNING, sdev,
96 "%s: sending tur failed, sense %x/%x/%x\n",
97 HP_SW_NAME, sshdr.sense_key, sshdr.asc,
98 sshdr.ascq);
99 break;
100 }
101
102done:
103 return ret;
104}
105
106/*
107 * hp_sw_tur - Send TEST UNIT READY
108 * @sdev: sdev command should be sent to
109 *
110 * Use the TEST UNIT READY command to determine
111 * the path state.
112 */
113static int hp_sw_tur(struct scsi_device *sdev, struct hp_sw_dh_data *h)
114{
115 struct request *req;
116 int ret;
117
Alan D. Brunellefebd7a52008-12-09 15:52:15 +0100118retry:
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700119 req = blk_get_request(sdev->request_queue, WRITE, GFP_NOIO);
120 if (!req)
121 return SCSI_DH_RES_TEMP_UNAVAIL;
122
123 req->cmd_type = REQ_TYPE_BLOCK_PC;
Mike Christie6000a362008-08-19 18:45:30 -0500124 req->cmd_flags |= REQ_FAILFAST_DEV | REQ_FAILFAST_TRANSPORT |
125 REQ_FAILFAST_DRIVER;
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700126 req->cmd_len = COMMAND_SIZE(TEST_UNIT_READY);
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700127 req->cmd[0] = TEST_UNIT_READY;
128 req->timeout = HP_SW_TIMEOUT;
129 req->sense = h->sense;
130 memset(req->sense, 0, SCSI_SENSE_BUFFERSIZE);
131 req->sense_len = 0;
132
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700133 ret = blk_execute_rq(req->q, NULL, req, 1);
134 if (ret == -EIO) {
135 if (req->sense_len > 0) {
136 ret = tur_done(sdev, h->sense);
137 } else {
138 sdev_printk(KERN_WARNING, sdev,
139 "%s: sending tur failed with %x\n",
140 HP_SW_NAME, req->errors);
141 ret = SCSI_DH_IO;
142 }
143 } else {
144 h->path_state = HP_SW_PATH_ACTIVE;
145 ret = SCSI_DH_OK;
146 }
Alan D. Brunellefebd7a52008-12-09 15:52:15 +0100147 if (ret == SCSI_DH_IMM_RETRY) {
148 blk_put_request(req);
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700149 goto retry;
Alan D. Brunellefebd7a52008-12-09 15:52:15 +0100150 }
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700151 if (ret == SCSI_DH_DEV_OFFLINED) {
152 h->path_state = HP_SW_PATH_PASSIVE;
153 ret = SCSI_DH_OK;
154 }
155
156 blk_put_request(req);
157
158 return ret;
159}
160
161/*
162 * start_done - Handle START STOP UNIT return status
163 * @sdev: sdev the command has been sent to
164 * @errors: blk error code
165 */
166static int start_done(struct scsi_device *sdev, unsigned char *sense)
167{
Mike Christief6dd3372008-05-01 14:49:59 -0700168 struct scsi_sense_hdr sshdr;
169 int rc;
170
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700171 rc = scsi_normalize_sense(sense, SCSI_SENSE_BUFFERSIZE, &sshdr);
172 if (!rc) {
173 sdev_printk(KERN_WARNING, sdev,
174 "%s: sending start_stop_unit failed, "
175 "no sense available\n",
176 HP_SW_NAME);
177 return SCSI_DH_IO;
178 }
Mike Christief6dd3372008-05-01 14:49:59 -0700179 switch (sshdr.sense_key) {
180 case NOT_READY:
181 if ((sshdr.asc == 0x04) && (sshdr.ascq == 3)) {
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700182 /*
183 * LUN not ready - manual intervention required
184 *
185 * Switch-over in progress, retry.
186 */
Mike Christief6dd3372008-05-01 14:49:59 -0700187 rc = SCSI_DH_RETRY;
Mike Christief6dd3372008-05-01 14:49:59 -0700188 break;
189 }
190 /* fall through */
191 default:
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700192 sdev_printk(KERN_WARNING, sdev,
193 "%s: sending start_stop_unit failed, sense %x/%x/%x\n",
194 HP_SW_NAME, sshdr.sense_key, sshdr.asc,
195 sshdr.ascq);
Mike Christief6dd3372008-05-01 14:49:59 -0700196 rc = SCSI_DH_IO;
197 }
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700198
Mike Christief6dd3372008-05-01 14:49:59 -0700199 return rc;
200}
201
Chandra Seetharaman4e2ef862009-10-21 09:22:58 -0700202static void start_stop_endio(struct request *req, int error)
203{
204 struct hp_sw_dh_data *h = req->end_io_data;
205 unsigned err = SCSI_DH_OK;
206
207 if (error || host_byte(req->errors) != DID_OK ||
208 msg_byte(req->errors) != COMMAND_COMPLETE) {
209 sdev_printk(KERN_WARNING, h->sdev,
210 "%s: sending start_stop_unit failed with %x\n",
211 HP_SW_NAME, req->errors);
212 err = SCSI_DH_IO;
213 goto done;
214 }
215
216 if (req->sense_len > 0) {
217 err = start_done(h->sdev, h->sense);
218 if (err == SCSI_DH_RETRY) {
219 err = SCSI_DH_IO;
220 if (--h->retry_cnt) {
221 blk_put_request(req);
222 err = hp_sw_start_stop(h);
223 if (err == SCSI_DH_OK)
224 return;
225 }
226 }
227 }
228done:
Mike Snitzer7a1e9d82011-01-25 11:52:17 -0500229 req->end_io_data = NULL;
230 __blk_put_request(req->q, req);
Chandra Seetharaman4e2ef862009-10-21 09:22:58 -0700231 if (h->callback_fn) {
232 h->callback_fn(h->callback_data, err);
233 h->callback_fn = h->callback_data = NULL;
234 }
235 return;
236
237}
238
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700239/*
240 * hp_sw_start_stop - Send START STOP UNIT command
241 * @sdev: sdev command should be sent to
242 *
243 * Sending START STOP UNIT activates the SP.
244 */
Chandra Seetharaman4e2ef862009-10-21 09:22:58 -0700245static int hp_sw_start_stop(struct hp_sw_dh_data *h)
Mike Christief6dd3372008-05-01 14:49:59 -0700246{
Mike Christief6dd3372008-05-01 14:49:59 -0700247 struct request *req;
Mike Christief6dd3372008-05-01 14:49:59 -0700248
Chandra Seetharaman4e2ef862009-10-21 09:22:58 -0700249 req = blk_get_request(h->sdev->request_queue, WRITE, GFP_ATOMIC);
Mike Christief6dd3372008-05-01 14:49:59 -0700250 if (!req)
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700251 return SCSI_DH_RES_TEMP_UNAVAIL;
Mike Christief6dd3372008-05-01 14:49:59 -0700252
253 req->cmd_type = REQ_TYPE_BLOCK_PC;
Mike Christie6000a362008-08-19 18:45:30 -0500254 req->cmd_flags |= REQ_FAILFAST_DEV | REQ_FAILFAST_TRANSPORT |
255 REQ_FAILFAST_DRIVER;
Mike Christief6dd3372008-05-01 14:49:59 -0700256 req->cmd_len = COMMAND_SIZE(START_STOP);
Mike Christief6dd3372008-05-01 14:49:59 -0700257 req->cmd[0] = START_STOP;
258 req->cmd[4] = 1; /* Start spin cycle */
259 req->timeout = HP_SW_TIMEOUT;
260 req->sense = h->sense;
261 memset(req->sense, 0, SCSI_SENSE_BUFFERSIZE);
262 req->sense_len = 0;
Chandra Seetharaman4e2ef862009-10-21 09:22:58 -0700263 req->end_io_data = h;
Mike Christief6dd3372008-05-01 14:49:59 -0700264
Chandra Seetharaman4e2ef862009-10-21 09:22:58 -0700265 blk_execute_rq_nowait(req->q, NULL, req, 1, start_stop_endio);
266 return SCSI_DH_OK;
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700267}
268
269static int hp_sw_prep_fn(struct scsi_device *sdev, struct request *req)
270{
271 struct hp_sw_dh_data *h = get_hp_sw_data(sdev);
272 int ret = BLKPREP_OK;
273
274 if (h->path_state != HP_SW_PATH_ACTIVE) {
275 ret = BLKPREP_KILL;
276 req->cmd_flags |= REQ_QUIET;
277 }
278 return ret;
279
280}
281
282/*
283 * hp_sw_activate - Activate a path
284 * @sdev: sdev on the path to be activated
285 *
286 * The HP Active/Passive firmware is pretty simple;
287 * the passive path reports NOT READY with sense codes
288 * 0x04/0x02; a START STOP UNIT command will then
289 * activate the passive path (and deactivate the
290 * previously active one).
291 */
Chandra Seetharaman3ae31f62009-10-21 09:22:46 -0700292static int hp_sw_activate(struct scsi_device *sdev,
293 activate_complete fn, void *data)
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700294{
295 int ret = SCSI_DH_OK;
296 struct hp_sw_dh_data *h = get_hp_sw_data(sdev);
297
298 ret = hp_sw_tur(sdev, h);
299
300 if (ret == SCSI_DH_OK && h->path_state == HP_SW_PATH_PASSIVE) {
Chandra Seetharaman4e2ef862009-10-21 09:22:58 -0700301 h->retry_cnt = h->retries;
302 h->callback_fn = fn;
303 h->callback_data = data;
304 ret = hp_sw_start_stop(h);
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700305 if (ret == SCSI_DH_OK)
Chandra Seetharaman4e2ef862009-10-21 09:22:58 -0700306 return 0;
307 h->callback_fn = h->callback_data = NULL;
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700308 }
309
Chandra Seetharaman3ae31f62009-10-21 09:22:46 -0700310 if (fn)
311 fn(data, ret);
312 return 0;
Mike Christief6dd3372008-05-01 14:49:59 -0700313}
314
Adrian Bunkf08c0762008-08-11 11:59:21 -0700315static const struct scsi_dh_devlist hp_sw_dh_data_list[] = {
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700316 {"COMPAQ", "MSA1000 VOLUME"},
317 {"COMPAQ", "HSV110"},
318 {"HP", "HSV100"},
Mike Christief6dd3372008-05-01 14:49:59 -0700319 {"DEC", "HSG80"},
320 {NULL, NULL},
321};
322
Moger, Babua3159692011-12-01 15:01:28 -0500323static bool hp_sw_match(struct scsi_device *sdev)
324{
325 int i;
326
327 if (scsi_device_tpgs(sdev))
328 return false;
329
330 for (i = 0; hp_sw_dh_data_list[i].vendor; i++) {
331 if (!strncmp(sdev->vendor, hp_sw_dh_data_list[i].vendor,
332 strlen(hp_sw_dh_data_list[i].vendor)) &&
333 !strncmp(sdev->model, hp_sw_dh_data_list[i].model,
334 strlen(hp_sw_dh_data_list[i].model))) {
335 return true;
336 }
337 }
338 return false;
339}
340
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700341static int hp_sw_bus_attach(struct scsi_device *sdev);
342static void hp_sw_bus_detach(struct scsi_device *sdev);
Mike Christief6dd3372008-05-01 14:49:59 -0700343
344static struct scsi_device_handler hp_sw_dh = {
345 .name = HP_SW_NAME,
346 .module = THIS_MODULE,
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700347 .devlist = hp_sw_dh_data_list,
348 .attach = hp_sw_bus_attach,
349 .detach = hp_sw_bus_detach,
Mike Christief6dd3372008-05-01 14:49:59 -0700350 .activate = hp_sw_activate,
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700351 .prep_fn = hp_sw_prep_fn,
Moger, Babua3159692011-12-01 15:01:28 -0500352 .match = hp_sw_match,
Mike Christief6dd3372008-05-01 14:49:59 -0700353};
354
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700355static int hp_sw_bus_attach(struct scsi_device *sdev)
Mike Christief6dd3372008-05-01 14:49:59 -0700356{
Mike Christief6dd3372008-05-01 14:49:59 -0700357 struct scsi_dh_data *scsi_dh_data;
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700358 struct hp_sw_dh_data *h;
Mike Christief6dd3372008-05-01 14:49:59 -0700359 unsigned long flags;
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700360 int ret;
Mike Christief6dd3372008-05-01 14:49:59 -0700361
Hillf Danton9dfeb312011-02-11 15:17:33 -0700362 scsi_dh_data = kzalloc(sizeof(*scsi_dh_data)
363 + sizeof(*h) , GFP_KERNEL);
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700364 if (!scsi_dh_data) {
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700365 sdev_printk(KERN_ERR, sdev, "%s: Attach Failed\n",
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700366 HP_SW_NAME);
Chandra Seetharaman33af79d2008-07-16 17:35:08 -0700367 return 0;
Mike Christief6dd3372008-05-01 14:49:59 -0700368 }
369
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700370 scsi_dh_data->scsi_dh = &hp_sw_dh;
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700371 h = (struct hp_sw_dh_data *) scsi_dh_data->buf;
372 h->path_state = HP_SW_PATH_UNINITIALIZED;
373 h->retries = HP_SW_RETRIES;
Chandra Seetharaman4e2ef862009-10-21 09:22:58 -0700374 h->sdev = sdev;
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700375
376 ret = hp_sw_tur(sdev, h);
377 if (ret != SCSI_DH_OK || h->path_state == HP_SW_PATH_UNINITIALIZED)
378 goto failed;
379
380 if (!try_module_get(THIS_MODULE))
381 goto failed;
382
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700383 spin_lock_irqsave(sdev->request_queue->queue_lock, flags);
384 sdev->scsi_dh_data = scsi_dh_data;
385 spin_unlock_irqrestore(sdev->request_queue->queue_lock, flags);
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700386
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700387 sdev_printk(KERN_INFO, sdev, "%s: attached to %s path\n",
388 HP_SW_NAME, h->path_state == HP_SW_PATH_ACTIVE?
389 "active":"passive");
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700390
Mike Christief6dd3372008-05-01 14:49:59 -0700391 return 0;
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700392
393failed:
394 kfree(scsi_dh_data);
395 sdev_printk(KERN_ERR, sdev, "%s: not attached\n",
396 HP_SW_NAME);
397 return -EINVAL;
Mike Christief6dd3372008-05-01 14:49:59 -0700398}
399
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700400static void hp_sw_bus_detach( struct scsi_device *sdev )
401{
402 struct scsi_dh_data *scsi_dh_data;
403 unsigned long flags;
404
405 spin_lock_irqsave(sdev->request_queue->queue_lock, flags);
406 scsi_dh_data = sdev->scsi_dh_data;
407 sdev->scsi_dh_data = NULL;
408 spin_unlock_irqrestore(sdev->request_queue->queue_lock, flags);
409 module_put(THIS_MODULE);
410
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700411 sdev_printk(KERN_NOTICE, sdev, "%s: Detached\n", HP_SW_NAME);
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700412
413 kfree(scsi_dh_data);
414}
415
Mike Christief6dd3372008-05-01 14:49:59 -0700416static int __init hp_sw_init(void)
417{
418 return scsi_register_device_handler(&hp_sw_dh);
419}
420
421static void __exit hp_sw_exit(void)
422{
423 scsi_unregister_device_handler(&hp_sw_dh);
424}
425
426module_init(hp_sw_init);
427module_exit(hp_sw_exit);
428
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700429MODULE_DESCRIPTION("HP Active/Passive driver");
Mike Christief6dd3372008-05-01 14:49:59 -0700430MODULE_AUTHOR("Mike Christie <michaelc@cs.wisc.edu");
431MODULE_LICENSE("GPL");