blob: e3916641e627883418d18b4eeb94bcbe1ddec018 [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>
Mike Christief6dd3372008-05-01 14:49:59 -070025#include <scsi/scsi.h>
26#include <scsi/scsi_dbg.h>
27#include <scsi/scsi_eh.h>
28#include <scsi/scsi_dh.h>
29
Hannes Reinecke2aef6d52008-07-17 16:53:09 -070030#define HP_SW_NAME "hp_sw"
Mike Christief6dd3372008-05-01 14:49:59 -070031
Hannes Reinecke2aef6d52008-07-17 16:53:09 -070032#define HP_SW_TIMEOUT (60 * HZ)
33#define HP_SW_RETRIES 3
34
35#define HP_SW_PATH_UNINITIALIZED -1
36#define HP_SW_PATH_ACTIVE 0
37#define HP_SW_PATH_PASSIVE 1
Mike Christief6dd3372008-05-01 14:49:59 -070038
39struct hp_sw_dh_data {
40 unsigned char sense[SCSI_SENSE_BUFFERSIZE];
Hannes Reinecke2aef6d52008-07-17 16:53:09 -070041 int path_state;
Mike Christief6dd3372008-05-01 14:49:59 -070042 int retries;
Chandra Seetharaman4e2ef862009-10-21 09:22:58 -070043 int retry_cnt;
44 struct scsi_device *sdev;
45 activate_complete callback_fn;
46 void *callback_data;
Mike Christief6dd3372008-05-01 14:49:59 -070047};
48
Chandra Seetharaman4e2ef862009-10-21 09:22:58 -070049static int hp_sw_start_stop(struct hp_sw_dh_data *);
50
Mike Christief6dd3372008-05-01 14:49:59 -070051static inline struct hp_sw_dh_data *get_hp_sw_data(struct scsi_device *sdev)
52{
53 struct scsi_dh_data *scsi_dh_data = sdev->scsi_dh_data;
54 BUG_ON(scsi_dh_data == NULL);
55 return ((struct hp_sw_dh_data *) scsi_dh_data->buf);
56}
57
Hannes Reinecke2aef6d52008-07-17 16:53:09 -070058/*
59 * tur_done - Handle TEST UNIT READY return status
60 * @sdev: sdev the command has been sent to
61 * @errors: blk error code
62 *
63 * Returns SCSI_DH_DEV_OFFLINED if the sdev is on the passive path
64 */
65static int tur_done(struct scsi_device *sdev, unsigned char *sense)
Mike Christief6dd3372008-05-01 14:49:59 -070066{
Hannes Reinecke2aef6d52008-07-17 16:53:09 -070067 struct scsi_sense_hdr sshdr;
68 int ret;
69
70 ret = scsi_normalize_sense(sense, SCSI_SENSE_BUFFERSIZE, &sshdr);
71 if (!ret) {
72 sdev_printk(KERN_WARNING, sdev,
73 "%s: sending tur failed, no sense available\n",
74 HP_SW_NAME);
75 ret = SCSI_DH_IO;
76 goto done;
77 }
78 switch (sshdr.sense_key) {
79 case UNIT_ATTENTION:
80 ret = SCSI_DH_IMM_RETRY;
81 break;
82 case NOT_READY:
83 if ((sshdr.asc == 0x04) && (sshdr.ascq == 2)) {
84 /*
85 * LUN not ready - Initialization command required
86 *
87 * This is the passive path
88 */
89 ret = SCSI_DH_DEV_OFFLINED;
90 break;
91 }
92 /* Fallthrough */
93 default:
94 sdev_printk(KERN_WARNING, sdev,
95 "%s: sending tur failed, sense %x/%x/%x\n",
96 HP_SW_NAME, sshdr.sense_key, sshdr.asc,
97 sshdr.ascq);
98 break;
99 }
100
101done:
102 return ret;
103}
104
105/*
106 * hp_sw_tur - Send TEST UNIT READY
107 * @sdev: sdev command should be sent to
108 *
109 * Use the TEST UNIT READY command to determine
110 * the path state.
111 */
112static int hp_sw_tur(struct scsi_device *sdev, struct hp_sw_dh_data *h)
113{
114 struct request *req;
115 int ret;
116
Alan D. Brunellefebd7a52008-12-09 15:52:15 +0100117retry:
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700118 req = blk_get_request(sdev->request_queue, WRITE, GFP_NOIO);
119 if (!req)
120 return SCSI_DH_RES_TEMP_UNAVAIL;
121
122 req->cmd_type = REQ_TYPE_BLOCK_PC;
Mike Christie6000a362008-08-19 18:45:30 -0500123 req->cmd_flags |= REQ_FAILFAST_DEV | REQ_FAILFAST_TRANSPORT |
124 REQ_FAILFAST_DRIVER;
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700125 req->cmd_len = COMMAND_SIZE(TEST_UNIT_READY);
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700126 req->cmd[0] = TEST_UNIT_READY;
127 req->timeout = HP_SW_TIMEOUT;
128 req->sense = h->sense;
129 memset(req->sense, 0, SCSI_SENSE_BUFFERSIZE);
130 req->sense_len = 0;
131
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700132 ret = blk_execute_rq(req->q, NULL, req, 1);
133 if (ret == -EIO) {
134 if (req->sense_len > 0) {
135 ret = tur_done(sdev, h->sense);
136 } else {
137 sdev_printk(KERN_WARNING, sdev,
138 "%s: sending tur failed with %x\n",
139 HP_SW_NAME, req->errors);
140 ret = SCSI_DH_IO;
141 }
142 } else {
143 h->path_state = HP_SW_PATH_ACTIVE;
144 ret = SCSI_DH_OK;
145 }
Alan D. Brunellefebd7a52008-12-09 15:52:15 +0100146 if (ret == SCSI_DH_IMM_RETRY) {
147 blk_put_request(req);
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700148 goto retry;
Alan D. Brunellefebd7a52008-12-09 15:52:15 +0100149 }
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700150 if (ret == SCSI_DH_DEV_OFFLINED) {
151 h->path_state = HP_SW_PATH_PASSIVE;
152 ret = SCSI_DH_OK;
153 }
154
155 blk_put_request(req);
156
157 return ret;
158}
159
160/*
161 * start_done - Handle START STOP UNIT return status
162 * @sdev: sdev the command has been sent to
163 * @errors: blk error code
164 */
165static int start_done(struct scsi_device *sdev, unsigned char *sense)
166{
Mike Christief6dd3372008-05-01 14:49:59 -0700167 struct scsi_sense_hdr sshdr;
168 int rc;
169
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700170 rc = scsi_normalize_sense(sense, SCSI_SENSE_BUFFERSIZE, &sshdr);
171 if (!rc) {
172 sdev_printk(KERN_WARNING, sdev,
173 "%s: sending start_stop_unit failed, "
174 "no sense available\n",
175 HP_SW_NAME);
176 return SCSI_DH_IO;
177 }
Mike Christief6dd3372008-05-01 14:49:59 -0700178 switch (sshdr.sense_key) {
179 case NOT_READY:
180 if ((sshdr.asc == 0x04) && (sshdr.ascq == 3)) {
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700181 /*
182 * LUN not ready - manual intervention required
183 *
184 * Switch-over in progress, retry.
185 */
Mike Christief6dd3372008-05-01 14:49:59 -0700186 rc = SCSI_DH_RETRY;
Mike Christief6dd3372008-05-01 14:49:59 -0700187 break;
188 }
189 /* fall through */
190 default:
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700191 sdev_printk(KERN_WARNING, sdev,
192 "%s: sending start_stop_unit failed, sense %x/%x/%x\n",
193 HP_SW_NAME, sshdr.sense_key, sshdr.asc,
194 sshdr.ascq);
Mike Christief6dd3372008-05-01 14:49:59 -0700195 rc = SCSI_DH_IO;
196 }
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700197
Mike Christief6dd3372008-05-01 14:49:59 -0700198 return rc;
199}
200
Chandra Seetharaman4e2ef862009-10-21 09:22:58 -0700201static void start_stop_endio(struct request *req, int error)
202{
203 struct hp_sw_dh_data *h = req->end_io_data;
204 unsigned err = SCSI_DH_OK;
205
206 if (error || host_byte(req->errors) != DID_OK ||
207 msg_byte(req->errors) != COMMAND_COMPLETE) {
208 sdev_printk(KERN_WARNING, h->sdev,
209 "%s: sending start_stop_unit failed with %x\n",
210 HP_SW_NAME, req->errors);
211 err = SCSI_DH_IO;
212 goto done;
213 }
214
215 if (req->sense_len > 0) {
216 err = start_done(h->sdev, h->sense);
217 if (err == SCSI_DH_RETRY) {
218 err = SCSI_DH_IO;
219 if (--h->retry_cnt) {
220 blk_put_request(req);
221 err = hp_sw_start_stop(h);
222 if (err == SCSI_DH_OK)
223 return;
224 }
225 }
226 }
227done:
228 blk_put_request(req);
229 if (h->callback_fn) {
230 h->callback_fn(h->callback_data, err);
231 h->callback_fn = h->callback_data = NULL;
232 }
233 return;
234
235}
236
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700237/*
238 * hp_sw_start_stop - Send START STOP UNIT command
239 * @sdev: sdev command should be sent to
240 *
241 * Sending START STOP UNIT activates the SP.
242 */
Chandra Seetharaman4e2ef862009-10-21 09:22:58 -0700243static int hp_sw_start_stop(struct hp_sw_dh_data *h)
Mike Christief6dd3372008-05-01 14:49:59 -0700244{
Mike Christief6dd3372008-05-01 14:49:59 -0700245 struct request *req;
Mike Christief6dd3372008-05-01 14:49:59 -0700246
Chandra Seetharaman4e2ef862009-10-21 09:22:58 -0700247 req = blk_get_request(h->sdev->request_queue, WRITE, GFP_ATOMIC);
Mike Christief6dd3372008-05-01 14:49:59 -0700248 if (!req)
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700249 return SCSI_DH_RES_TEMP_UNAVAIL;
Mike Christief6dd3372008-05-01 14:49:59 -0700250
251 req->cmd_type = REQ_TYPE_BLOCK_PC;
Mike Christie6000a362008-08-19 18:45:30 -0500252 req->cmd_flags |= REQ_FAILFAST_DEV | REQ_FAILFAST_TRANSPORT |
253 REQ_FAILFAST_DRIVER;
Mike Christief6dd3372008-05-01 14:49:59 -0700254 req->cmd_len = COMMAND_SIZE(START_STOP);
Mike Christief6dd3372008-05-01 14:49:59 -0700255 req->cmd[0] = START_STOP;
256 req->cmd[4] = 1; /* Start spin cycle */
257 req->timeout = HP_SW_TIMEOUT;
258 req->sense = h->sense;
259 memset(req->sense, 0, SCSI_SENSE_BUFFERSIZE);
260 req->sense_len = 0;
Chandra Seetharaman4e2ef862009-10-21 09:22:58 -0700261 req->end_io_data = h;
Mike Christief6dd3372008-05-01 14:49:59 -0700262
Chandra Seetharaman4e2ef862009-10-21 09:22:58 -0700263 blk_execute_rq_nowait(req->q, NULL, req, 1, start_stop_endio);
264 return SCSI_DH_OK;
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700265}
266
267static int hp_sw_prep_fn(struct scsi_device *sdev, struct request *req)
268{
269 struct hp_sw_dh_data *h = get_hp_sw_data(sdev);
270 int ret = BLKPREP_OK;
271
272 if (h->path_state != HP_SW_PATH_ACTIVE) {
273 ret = BLKPREP_KILL;
274 req->cmd_flags |= REQ_QUIET;
275 }
276 return ret;
277
278}
279
280/*
281 * hp_sw_activate - Activate a path
282 * @sdev: sdev on the path to be activated
283 *
284 * The HP Active/Passive firmware is pretty simple;
285 * the passive path reports NOT READY with sense codes
286 * 0x04/0x02; a START STOP UNIT command will then
287 * activate the passive path (and deactivate the
288 * previously active one).
289 */
Chandra Seetharaman3ae31f62009-10-21 09:22:46 -0700290static int hp_sw_activate(struct scsi_device *sdev,
291 activate_complete fn, void *data)
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700292{
293 int ret = SCSI_DH_OK;
294 struct hp_sw_dh_data *h = get_hp_sw_data(sdev);
295
296 ret = hp_sw_tur(sdev, h);
297
298 if (ret == SCSI_DH_OK && h->path_state == HP_SW_PATH_PASSIVE) {
Chandra Seetharaman4e2ef862009-10-21 09:22:58 -0700299 h->retry_cnt = h->retries;
300 h->callback_fn = fn;
301 h->callback_data = data;
302 ret = hp_sw_start_stop(h);
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700303 if (ret == SCSI_DH_OK)
Chandra Seetharaman4e2ef862009-10-21 09:22:58 -0700304 return 0;
305 h->callback_fn = h->callback_data = NULL;
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700306 }
307
Chandra Seetharaman3ae31f62009-10-21 09:22:46 -0700308 if (fn)
309 fn(data, ret);
310 return 0;
Mike Christief6dd3372008-05-01 14:49:59 -0700311}
312
Adrian Bunkf08c0762008-08-11 11:59:21 -0700313static const struct scsi_dh_devlist hp_sw_dh_data_list[] = {
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700314 {"COMPAQ", "MSA1000 VOLUME"},
315 {"COMPAQ", "HSV110"},
316 {"HP", "HSV100"},
Mike Christief6dd3372008-05-01 14:49:59 -0700317 {"DEC", "HSG80"},
318 {NULL, NULL},
319};
320
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700321static int hp_sw_bus_attach(struct scsi_device *sdev);
322static void hp_sw_bus_detach(struct scsi_device *sdev);
Mike Christief6dd3372008-05-01 14:49:59 -0700323
324static struct scsi_device_handler hp_sw_dh = {
325 .name = HP_SW_NAME,
326 .module = THIS_MODULE,
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700327 .devlist = hp_sw_dh_data_list,
328 .attach = hp_sw_bus_attach,
329 .detach = hp_sw_bus_detach,
Mike Christief6dd3372008-05-01 14:49:59 -0700330 .activate = hp_sw_activate,
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700331 .prep_fn = hp_sw_prep_fn,
Mike Christief6dd3372008-05-01 14:49:59 -0700332};
333
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700334static int hp_sw_bus_attach(struct scsi_device *sdev)
Mike Christief6dd3372008-05-01 14:49:59 -0700335{
Mike Christief6dd3372008-05-01 14:49:59 -0700336 struct scsi_dh_data *scsi_dh_data;
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700337 struct hp_sw_dh_data *h;
Mike Christief6dd3372008-05-01 14:49:59 -0700338 unsigned long flags;
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700339 int ret;
Mike Christief6dd3372008-05-01 14:49:59 -0700340
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700341 scsi_dh_data = kzalloc(sizeof(struct scsi_device_handler *)
342 + sizeof(struct hp_sw_dh_data) , GFP_KERNEL);
343 if (!scsi_dh_data) {
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700344 sdev_printk(KERN_ERR, sdev, "%s: Attach Failed\n",
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700345 HP_SW_NAME);
Chandra Seetharaman33af79d2008-07-16 17:35:08 -0700346 return 0;
Mike Christief6dd3372008-05-01 14:49:59 -0700347 }
348
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700349 scsi_dh_data->scsi_dh = &hp_sw_dh;
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700350 h = (struct hp_sw_dh_data *) scsi_dh_data->buf;
351 h->path_state = HP_SW_PATH_UNINITIALIZED;
352 h->retries = HP_SW_RETRIES;
Chandra Seetharaman4e2ef862009-10-21 09:22:58 -0700353 h->sdev = sdev;
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700354
355 ret = hp_sw_tur(sdev, h);
356 if (ret != SCSI_DH_OK || h->path_state == HP_SW_PATH_UNINITIALIZED)
357 goto failed;
358
359 if (!try_module_get(THIS_MODULE))
360 goto failed;
361
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700362 spin_lock_irqsave(sdev->request_queue->queue_lock, flags);
363 sdev->scsi_dh_data = scsi_dh_data;
364 spin_unlock_irqrestore(sdev->request_queue->queue_lock, flags);
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700365
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700366 sdev_printk(KERN_INFO, sdev, "%s: attached to %s path\n",
367 HP_SW_NAME, h->path_state == HP_SW_PATH_ACTIVE?
368 "active":"passive");
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700369
Mike Christief6dd3372008-05-01 14:49:59 -0700370 return 0;
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700371
372failed:
373 kfree(scsi_dh_data);
374 sdev_printk(KERN_ERR, sdev, "%s: not attached\n",
375 HP_SW_NAME);
376 return -EINVAL;
Mike Christief6dd3372008-05-01 14:49:59 -0700377}
378
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700379static void hp_sw_bus_detach( struct scsi_device *sdev )
380{
381 struct scsi_dh_data *scsi_dh_data;
382 unsigned long flags;
383
384 spin_lock_irqsave(sdev->request_queue->queue_lock, flags);
385 scsi_dh_data = sdev->scsi_dh_data;
386 sdev->scsi_dh_data = NULL;
387 spin_unlock_irqrestore(sdev->request_queue->queue_lock, flags);
388 module_put(THIS_MODULE);
389
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700390 sdev_printk(KERN_NOTICE, sdev, "%s: Detached\n", HP_SW_NAME);
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700391
392 kfree(scsi_dh_data);
393}
394
Mike Christief6dd3372008-05-01 14:49:59 -0700395static int __init hp_sw_init(void)
396{
397 return scsi_register_device_handler(&hp_sw_dh);
398}
399
400static void __exit hp_sw_exit(void)
401{
402 scsi_unregister_device_handler(&hp_sw_dh);
403}
404
405module_init(hp_sw_init);
406module_exit(hp_sw_exit);
407
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700408MODULE_DESCRIPTION("HP Active/Passive driver");
Mike Christief6dd3372008-05-01 14:49:59 -0700409MODULE_AUTHOR("Mike Christie <michaelc@cs.wisc.edu");
410MODULE_LICENSE("GPL");