blob: 0aacafc96f21b5b1a895727b11ee453129664ce0 [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
24#include <scsi/scsi.h>
25#include <scsi/scsi_dbg.h>
26#include <scsi/scsi_eh.h>
27#include <scsi/scsi_dh.h>
28
Hannes Reinecke2aef6d52008-07-17 16:53:09 -070029#define HP_SW_NAME "hp_sw"
Mike Christief6dd3372008-05-01 14:49:59 -070030
Hannes Reinecke2aef6d52008-07-17 16:53:09 -070031#define HP_SW_TIMEOUT (60 * HZ)
32#define HP_SW_RETRIES 3
33
34#define HP_SW_PATH_UNINITIALIZED -1
35#define HP_SW_PATH_ACTIVE 0
36#define HP_SW_PATH_PASSIVE 1
Mike Christief6dd3372008-05-01 14:49:59 -070037
38struct hp_sw_dh_data {
39 unsigned char sense[SCSI_SENSE_BUFFERSIZE];
Hannes Reinecke2aef6d52008-07-17 16:53:09 -070040 int path_state;
Mike Christief6dd3372008-05-01 14:49:59 -070041 int retries;
42};
43
44static inline struct hp_sw_dh_data *get_hp_sw_data(struct scsi_device *sdev)
45{
46 struct scsi_dh_data *scsi_dh_data = sdev->scsi_dh_data;
47 BUG_ON(scsi_dh_data == NULL);
48 return ((struct hp_sw_dh_data *) scsi_dh_data->buf);
49}
50
Hannes Reinecke2aef6d52008-07-17 16:53:09 -070051/*
52 * tur_done - Handle TEST UNIT READY return status
53 * @sdev: sdev the command has been sent to
54 * @errors: blk error code
55 *
56 * Returns SCSI_DH_DEV_OFFLINED if the sdev is on the passive path
57 */
58static int tur_done(struct scsi_device *sdev, unsigned char *sense)
Mike Christief6dd3372008-05-01 14:49:59 -070059{
Hannes Reinecke2aef6d52008-07-17 16:53:09 -070060 struct scsi_sense_hdr sshdr;
61 int ret;
62
63 ret = scsi_normalize_sense(sense, SCSI_SENSE_BUFFERSIZE, &sshdr);
64 if (!ret) {
65 sdev_printk(KERN_WARNING, sdev,
66 "%s: sending tur failed, no sense available\n",
67 HP_SW_NAME);
68 ret = SCSI_DH_IO;
69 goto done;
70 }
71 switch (sshdr.sense_key) {
72 case UNIT_ATTENTION:
73 ret = SCSI_DH_IMM_RETRY;
74 break;
75 case NOT_READY:
76 if ((sshdr.asc == 0x04) && (sshdr.ascq == 2)) {
77 /*
78 * LUN not ready - Initialization command required
79 *
80 * This is the passive path
81 */
82 ret = SCSI_DH_DEV_OFFLINED;
83 break;
84 }
85 /* Fallthrough */
86 default:
87 sdev_printk(KERN_WARNING, sdev,
88 "%s: sending tur failed, sense %x/%x/%x\n",
89 HP_SW_NAME, sshdr.sense_key, sshdr.asc,
90 sshdr.ascq);
91 break;
92 }
93
94done:
95 return ret;
96}
97
98/*
99 * hp_sw_tur - Send TEST UNIT READY
100 * @sdev: sdev command should be sent to
101 *
102 * Use the TEST UNIT READY command to determine
103 * the path state.
104 */
105static int hp_sw_tur(struct scsi_device *sdev, struct hp_sw_dh_data *h)
106{
107 struct request *req;
108 int ret;
109
Alan D. Brunellefebd7a52008-12-09 15:52:15 +0100110retry:
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700111 req = blk_get_request(sdev->request_queue, WRITE, GFP_NOIO);
112 if (!req)
113 return SCSI_DH_RES_TEMP_UNAVAIL;
114
115 req->cmd_type = REQ_TYPE_BLOCK_PC;
Mike Christie6000a362008-08-19 18:45:30 -0500116 req->cmd_flags |= REQ_FAILFAST_DEV | REQ_FAILFAST_TRANSPORT |
117 REQ_FAILFAST_DRIVER;
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700118 req->cmd_len = COMMAND_SIZE(TEST_UNIT_READY);
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700119 req->cmd[0] = TEST_UNIT_READY;
120 req->timeout = HP_SW_TIMEOUT;
121 req->sense = h->sense;
122 memset(req->sense, 0, SCSI_SENSE_BUFFERSIZE);
123 req->sense_len = 0;
124
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700125 ret = blk_execute_rq(req->q, NULL, req, 1);
126 if (ret == -EIO) {
127 if (req->sense_len > 0) {
128 ret = tur_done(sdev, h->sense);
129 } else {
130 sdev_printk(KERN_WARNING, sdev,
131 "%s: sending tur failed with %x\n",
132 HP_SW_NAME, req->errors);
133 ret = SCSI_DH_IO;
134 }
135 } else {
136 h->path_state = HP_SW_PATH_ACTIVE;
137 ret = SCSI_DH_OK;
138 }
Alan D. Brunellefebd7a52008-12-09 15:52:15 +0100139 if (ret == SCSI_DH_IMM_RETRY) {
140 blk_put_request(req);
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700141 goto retry;
Alan D. Brunellefebd7a52008-12-09 15:52:15 +0100142 }
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700143 if (ret == SCSI_DH_DEV_OFFLINED) {
144 h->path_state = HP_SW_PATH_PASSIVE;
145 ret = SCSI_DH_OK;
146 }
147
148 blk_put_request(req);
149
150 return ret;
151}
152
153/*
154 * start_done - Handle START STOP UNIT return status
155 * @sdev: sdev the command has been sent to
156 * @errors: blk error code
157 */
158static int start_done(struct scsi_device *sdev, unsigned char *sense)
159{
Mike Christief6dd3372008-05-01 14:49:59 -0700160 struct scsi_sense_hdr sshdr;
161 int rc;
162
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700163 rc = scsi_normalize_sense(sense, SCSI_SENSE_BUFFERSIZE, &sshdr);
164 if (!rc) {
165 sdev_printk(KERN_WARNING, sdev,
166 "%s: sending start_stop_unit failed, "
167 "no sense available\n",
168 HP_SW_NAME);
169 return SCSI_DH_IO;
170 }
Mike Christief6dd3372008-05-01 14:49:59 -0700171 switch (sshdr.sense_key) {
172 case NOT_READY:
173 if ((sshdr.asc == 0x04) && (sshdr.ascq == 3)) {
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700174 /*
175 * LUN not ready - manual intervention required
176 *
177 * Switch-over in progress, retry.
178 */
Mike Christief6dd3372008-05-01 14:49:59 -0700179 rc = SCSI_DH_RETRY;
Mike Christief6dd3372008-05-01 14:49:59 -0700180 break;
181 }
182 /* fall through */
183 default:
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700184 sdev_printk(KERN_WARNING, sdev,
185 "%s: sending start_stop_unit failed, sense %x/%x/%x\n",
186 HP_SW_NAME, sshdr.sense_key, sshdr.asc,
187 sshdr.ascq);
Mike Christief6dd3372008-05-01 14:49:59 -0700188 rc = SCSI_DH_IO;
189 }
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700190
Mike Christief6dd3372008-05-01 14:49:59 -0700191 return rc;
192}
193
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700194/*
195 * hp_sw_start_stop - Send START STOP UNIT command
196 * @sdev: sdev command should be sent to
197 *
198 * Sending START STOP UNIT activates the SP.
199 */
200static int hp_sw_start_stop(struct scsi_device *sdev, struct hp_sw_dh_data *h)
Mike Christief6dd3372008-05-01 14:49:59 -0700201{
Mike Christief6dd3372008-05-01 14:49:59 -0700202 struct request *req;
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700203 int ret, retry;
Mike Christief6dd3372008-05-01 14:49:59 -0700204
Alan D. Brunellefebd7a52008-12-09 15:52:15 +0100205retry:
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700206 req = blk_get_request(sdev->request_queue, WRITE, GFP_NOIO);
Mike Christief6dd3372008-05-01 14:49:59 -0700207 if (!req)
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700208 return SCSI_DH_RES_TEMP_UNAVAIL;
Mike Christief6dd3372008-05-01 14:49:59 -0700209
210 req->cmd_type = REQ_TYPE_BLOCK_PC;
Mike Christie6000a362008-08-19 18:45:30 -0500211 req->cmd_flags |= REQ_FAILFAST_DEV | REQ_FAILFAST_TRANSPORT |
212 REQ_FAILFAST_DRIVER;
Mike Christief6dd3372008-05-01 14:49:59 -0700213 req->cmd_len = COMMAND_SIZE(START_STOP);
Mike Christief6dd3372008-05-01 14:49:59 -0700214 req->cmd[0] = START_STOP;
215 req->cmd[4] = 1; /* Start spin cycle */
216 req->timeout = HP_SW_TIMEOUT;
217 req->sense = h->sense;
218 memset(req->sense, 0, SCSI_SENSE_BUFFERSIZE);
219 req->sense_len = 0;
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700220 retry = h->retries;
Mike Christief6dd3372008-05-01 14:49:59 -0700221
222 ret = blk_execute_rq(req->q, NULL, req, 1);
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700223 if (ret == -EIO) {
224 if (req->sense_len > 0) {
225 ret = start_done(sdev, h->sense);
226 } else {
227 sdev_printk(KERN_WARNING, sdev,
228 "%s: sending start_stop_unit failed with %x\n",
229 HP_SW_NAME, req->errors);
230 ret = SCSI_DH_IO;
231 }
232 } else
233 ret = SCSI_DH_OK;
234
235 if (ret == SCSI_DH_RETRY) {
Alan D. Brunellefebd7a52008-12-09 15:52:15 +0100236 if (--retry) {
237 blk_put_request(req);
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700238 goto retry;
Alan D. Brunellefebd7a52008-12-09 15:52:15 +0100239 }
Mike Christief6dd3372008-05-01 14:49:59 -0700240 ret = SCSI_DH_IO;
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700241 }
242
243 blk_put_request(req);
244
245 return ret;
246}
247
248static int hp_sw_prep_fn(struct scsi_device *sdev, struct request *req)
249{
250 struct hp_sw_dh_data *h = get_hp_sw_data(sdev);
251 int ret = BLKPREP_OK;
252
253 if (h->path_state != HP_SW_PATH_ACTIVE) {
254 ret = BLKPREP_KILL;
255 req->cmd_flags |= REQ_QUIET;
256 }
257 return ret;
258
259}
260
261/*
262 * hp_sw_activate - Activate a path
263 * @sdev: sdev on the path to be activated
264 *
265 * The HP Active/Passive firmware is pretty simple;
266 * the passive path reports NOT READY with sense codes
267 * 0x04/0x02; a START STOP UNIT command will then
268 * activate the passive path (and deactivate the
269 * previously active one).
270 */
Chandra Seetharaman3ae31f62009-10-21 09:22:46 -0700271static int hp_sw_activate(struct scsi_device *sdev,
272 activate_complete fn, void *data)
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700273{
274 int ret = SCSI_DH_OK;
275 struct hp_sw_dh_data *h = get_hp_sw_data(sdev);
276
277 ret = hp_sw_tur(sdev, h);
278
279 if (ret == SCSI_DH_OK && h->path_state == HP_SW_PATH_PASSIVE) {
280 ret = hp_sw_start_stop(sdev, h);
281 if (ret == SCSI_DH_OK)
282 sdev_printk(KERN_INFO, sdev,
283 "%s: activated path\n",
284 HP_SW_NAME);
285 }
286
Chandra Seetharaman3ae31f62009-10-21 09:22:46 -0700287 if (fn)
288 fn(data, ret);
289 return 0;
Mike Christief6dd3372008-05-01 14:49:59 -0700290}
291
Adrian Bunkf08c0762008-08-11 11:59:21 -0700292static const struct scsi_dh_devlist hp_sw_dh_data_list[] = {
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700293 {"COMPAQ", "MSA1000 VOLUME"},
294 {"COMPAQ", "HSV110"},
295 {"HP", "HSV100"},
Mike Christief6dd3372008-05-01 14:49:59 -0700296 {"DEC", "HSG80"},
297 {NULL, NULL},
298};
299
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700300static int hp_sw_bus_attach(struct scsi_device *sdev);
301static void hp_sw_bus_detach(struct scsi_device *sdev);
Mike Christief6dd3372008-05-01 14:49:59 -0700302
303static struct scsi_device_handler hp_sw_dh = {
304 .name = HP_SW_NAME,
305 .module = THIS_MODULE,
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700306 .devlist = hp_sw_dh_data_list,
307 .attach = hp_sw_bus_attach,
308 .detach = hp_sw_bus_detach,
Mike Christief6dd3372008-05-01 14:49:59 -0700309 .activate = hp_sw_activate,
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700310 .prep_fn = hp_sw_prep_fn,
Mike Christief6dd3372008-05-01 14:49:59 -0700311};
312
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700313static int hp_sw_bus_attach(struct scsi_device *sdev)
Mike Christief6dd3372008-05-01 14:49:59 -0700314{
Mike Christief6dd3372008-05-01 14:49:59 -0700315 struct scsi_dh_data *scsi_dh_data;
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700316 struct hp_sw_dh_data *h;
Mike Christief6dd3372008-05-01 14:49:59 -0700317 unsigned long flags;
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700318 int ret;
Mike Christief6dd3372008-05-01 14:49:59 -0700319
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700320 scsi_dh_data = kzalloc(sizeof(struct scsi_device_handler *)
321 + sizeof(struct hp_sw_dh_data) , GFP_KERNEL);
322 if (!scsi_dh_data) {
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700323 sdev_printk(KERN_ERR, sdev, "%s: Attach Failed\n",
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700324 HP_SW_NAME);
Chandra Seetharaman33af79d2008-07-16 17:35:08 -0700325 return 0;
Mike Christief6dd3372008-05-01 14:49:59 -0700326 }
327
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700328 scsi_dh_data->scsi_dh = &hp_sw_dh;
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700329 h = (struct hp_sw_dh_data *) scsi_dh_data->buf;
330 h->path_state = HP_SW_PATH_UNINITIALIZED;
331 h->retries = HP_SW_RETRIES;
332
333 ret = hp_sw_tur(sdev, h);
334 if (ret != SCSI_DH_OK || h->path_state == HP_SW_PATH_UNINITIALIZED)
335 goto failed;
336
337 if (!try_module_get(THIS_MODULE))
338 goto failed;
339
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700340 spin_lock_irqsave(sdev->request_queue->queue_lock, flags);
341 sdev->scsi_dh_data = scsi_dh_data;
342 spin_unlock_irqrestore(sdev->request_queue->queue_lock, flags);
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700343
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700344 sdev_printk(KERN_INFO, sdev, "%s: attached to %s path\n",
345 HP_SW_NAME, h->path_state == HP_SW_PATH_ACTIVE?
346 "active":"passive");
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700347
Mike Christief6dd3372008-05-01 14:49:59 -0700348 return 0;
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700349
350failed:
351 kfree(scsi_dh_data);
352 sdev_printk(KERN_ERR, sdev, "%s: not attached\n",
353 HP_SW_NAME);
354 return -EINVAL;
Mike Christief6dd3372008-05-01 14:49:59 -0700355}
356
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700357static void hp_sw_bus_detach( struct scsi_device *sdev )
358{
359 struct scsi_dh_data *scsi_dh_data;
360 unsigned long flags;
361
362 spin_lock_irqsave(sdev->request_queue->queue_lock, flags);
363 scsi_dh_data = sdev->scsi_dh_data;
364 sdev->scsi_dh_data = NULL;
365 spin_unlock_irqrestore(sdev->request_queue->queue_lock, flags);
366 module_put(THIS_MODULE);
367
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700368 sdev_printk(KERN_NOTICE, sdev, "%s: Detached\n", HP_SW_NAME);
Hannes Reinecke765cbc62008-07-17 16:52:51 -0700369
370 kfree(scsi_dh_data);
371}
372
Mike Christief6dd3372008-05-01 14:49:59 -0700373static int __init hp_sw_init(void)
374{
375 return scsi_register_device_handler(&hp_sw_dh);
376}
377
378static void __exit hp_sw_exit(void)
379{
380 scsi_unregister_device_handler(&hp_sw_dh);
381}
382
383module_init(hp_sw_init);
384module_exit(hp_sw_exit);
385
Hannes Reinecke2aef6d52008-07-17 16:53:09 -0700386MODULE_DESCRIPTION("HP Active/Passive driver");
Mike Christief6dd3372008-05-01 14:49:59 -0700387MODULE_AUTHOR("Mike Christie <michaelc@cs.wisc.edu");
388MODULE_LICENSE("GPL");