xref: /openbmc/linux/drivers/scsi/device_handler/scsi_dh_hp_sw.c (revision f15cbe6f1a4b4d9df59142fc8e4abb973302cf44)
1 /*
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
7  * Copyright (C) 2008 Hannes Reinecke <hare@suse.de>
8  *
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 
29 #define HP_SW_NAME			"hp_sw"
30 
31 #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
37 
38 struct hp_sw_dh_data {
39 	unsigned char sense[SCSI_SENSE_BUFFERSIZE];
40 	int path_state;
41 	int retries;
42 };
43 
44 static 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 
51 /*
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  */
58 static int tur_done(struct scsi_device *sdev, unsigned char *sense)
59 {
60 	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 
94 done:
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  */
105 static int hp_sw_tur(struct scsi_device *sdev, struct hp_sw_dh_data *h)
106 {
107 	struct request *req;
108 	int ret;
109 
110 	req = blk_get_request(sdev->request_queue, WRITE, GFP_NOIO);
111 	if (!req)
112 		return SCSI_DH_RES_TEMP_UNAVAIL;
113 
114 	req->cmd_type = REQ_TYPE_BLOCK_PC;
115 	req->cmd_flags |= REQ_FAILFAST;
116 	req->cmd_len = COMMAND_SIZE(TEST_UNIT_READY);
117 	memset(req->cmd, 0, MAX_COMMAND_SIZE);
118 	req->cmd[0] = TEST_UNIT_READY;
119 	req->timeout = HP_SW_TIMEOUT;
120 	req->sense = h->sense;
121 	memset(req->sense, 0, SCSI_SENSE_BUFFERSIZE);
122 	req->sense_len = 0;
123 
124 retry:
125 	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 	}
139 	if (ret == SCSI_DH_IMM_RETRY)
140 		goto retry;
141 	if (ret == SCSI_DH_DEV_OFFLINED) {
142 		h->path_state = HP_SW_PATH_PASSIVE;
143 		ret = SCSI_DH_OK;
144 	}
145 
146 	blk_put_request(req);
147 
148 	return ret;
149 }
150 
151 /*
152  * start_done - Handle START STOP UNIT return status
153  * @sdev: sdev the command has been sent to
154  * @errors: blk error code
155  */
156 static int start_done(struct scsi_device *sdev, unsigned char *sense)
157 {
158 	struct scsi_sense_hdr sshdr;
159 	int rc;
160 
161 	rc = scsi_normalize_sense(sense, SCSI_SENSE_BUFFERSIZE, &sshdr);
162 	if (!rc) {
163 		sdev_printk(KERN_WARNING, sdev,
164 			    "%s: sending start_stop_unit failed, "
165 			    "no sense available\n",
166 			    HP_SW_NAME);
167 		return SCSI_DH_IO;
168 	}
169 	switch (sshdr.sense_key) {
170 	case NOT_READY:
171 		if ((sshdr.asc == 0x04) && (sshdr.ascq == 3)) {
172 			/*
173 			 * LUN not ready - manual intervention required
174 			 *
175 			 * Switch-over in progress, retry.
176 			 */
177 			rc = SCSI_DH_RETRY;
178 			break;
179 		}
180 		/* fall through */
181 	default:
182 		sdev_printk(KERN_WARNING, sdev,
183 			   "%s: sending start_stop_unit failed, sense %x/%x/%x\n",
184 			   HP_SW_NAME, sshdr.sense_key, sshdr.asc,
185 			   sshdr.ascq);
186 		rc = SCSI_DH_IO;
187 	}
188 
189 	return rc;
190 }
191 
192 /*
193  * hp_sw_start_stop - Send START STOP UNIT command
194  * @sdev: sdev command should be sent to
195  *
196  * Sending START STOP UNIT activates the SP.
197  */
198 static int hp_sw_start_stop(struct scsi_device *sdev, struct hp_sw_dh_data *h)
199 {
200 	struct request *req;
201 	int ret, retry;
202 
203 	req = blk_get_request(sdev->request_queue, WRITE, GFP_NOIO);
204 	if (!req)
205 		return SCSI_DH_RES_TEMP_UNAVAIL;
206 
207 	req->cmd_type = REQ_TYPE_BLOCK_PC;
208 	req->cmd_flags |= REQ_FAILFAST;
209 	req->cmd_len = COMMAND_SIZE(START_STOP);
210 	memset(req->cmd, 0, MAX_COMMAND_SIZE);
211 	req->cmd[0] = START_STOP;
212 	req->cmd[4] = 1;	/* Start spin cycle */
213 	req->timeout = HP_SW_TIMEOUT;
214 	req->sense = h->sense;
215 	memset(req->sense, 0, SCSI_SENSE_BUFFERSIZE);
216 	req->sense_len = 0;
217 	retry = h->retries;
218 
219 retry:
220 	ret = blk_execute_rq(req->q, NULL, req, 1);
221 	if (ret == -EIO) {
222 		if (req->sense_len > 0) {
223 			ret = start_done(sdev, h->sense);
224 		} else {
225 			sdev_printk(KERN_WARNING, sdev,
226 				    "%s: sending start_stop_unit failed with %x\n",
227 				    HP_SW_NAME, req->errors);
228 			ret = SCSI_DH_IO;
229 		}
230 	} else
231 		ret = SCSI_DH_OK;
232 
233 	if (ret == SCSI_DH_RETRY) {
234 		if (--retry)
235 			goto retry;
236 		ret = SCSI_DH_IO;
237 	}
238 
239 	blk_put_request(req);
240 
241 	return ret;
242 }
243 
244 static int hp_sw_prep_fn(struct scsi_device *sdev, struct request *req)
245 {
246 	struct hp_sw_dh_data *h = get_hp_sw_data(sdev);
247 	int ret = BLKPREP_OK;
248 
249 	if (h->path_state != HP_SW_PATH_ACTIVE) {
250 		ret = BLKPREP_KILL;
251 		req->cmd_flags |= REQ_QUIET;
252 	}
253 	return ret;
254 
255 }
256 
257 /*
258  * hp_sw_activate - Activate a path
259  * @sdev: sdev on the path to be activated
260  *
261  * The HP Active/Passive firmware is pretty simple;
262  * the passive path reports NOT READY with sense codes
263  * 0x04/0x02; a START STOP UNIT command will then
264  * activate the passive path (and deactivate the
265  * previously active one).
266  */
267 static int hp_sw_activate(struct scsi_device *sdev)
268 {
269 	int ret = SCSI_DH_OK;
270 	struct hp_sw_dh_data *h = get_hp_sw_data(sdev);
271 
272 	ret = hp_sw_tur(sdev, h);
273 
274 	if (ret == SCSI_DH_OK && h->path_state == HP_SW_PATH_PASSIVE) {
275 		ret = hp_sw_start_stop(sdev, h);
276 		if (ret == SCSI_DH_OK)
277 			sdev_printk(KERN_INFO, sdev,
278 				    "%s: activated path\n",
279 				    HP_SW_NAME);
280 	}
281 
282 	return ret;
283 }
284 
285 const struct scsi_dh_devlist hp_sw_dh_data_list[] = {
286 	{"COMPAQ", "MSA1000 VOLUME"},
287 	{"COMPAQ", "HSV110"},
288 	{"HP", "HSV100"},
289 	{"DEC", "HSG80"},
290 	{NULL, NULL},
291 };
292 
293 static int hp_sw_bus_attach(struct scsi_device *sdev);
294 static void hp_sw_bus_detach(struct scsi_device *sdev);
295 
296 static struct scsi_device_handler hp_sw_dh = {
297 	.name		= HP_SW_NAME,
298 	.module		= THIS_MODULE,
299 	.devlist	= hp_sw_dh_data_list,
300 	.attach		= hp_sw_bus_attach,
301 	.detach		= hp_sw_bus_detach,
302 	.activate	= hp_sw_activate,
303 	.prep_fn	= hp_sw_prep_fn,
304 };
305 
306 static int hp_sw_bus_attach(struct scsi_device *sdev)
307 {
308 	struct scsi_dh_data *scsi_dh_data;
309 	struct hp_sw_dh_data *h;
310 	unsigned long flags;
311 	int ret;
312 
313 	scsi_dh_data = kzalloc(sizeof(struct scsi_device_handler *)
314 			       + sizeof(struct hp_sw_dh_data) , GFP_KERNEL);
315 	if (!scsi_dh_data) {
316 		sdev_printk(KERN_ERR, sdev, "%s: Attach Failed\n",
317 			    HP_SW_NAME);
318 		return 0;
319 	}
320 
321 	scsi_dh_data->scsi_dh = &hp_sw_dh;
322 	h = (struct hp_sw_dh_data *) scsi_dh_data->buf;
323 	h->path_state = HP_SW_PATH_UNINITIALIZED;
324 	h->retries = HP_SW_RETRIES;
325 
326 	ret = hp_sw_tur(sdev, h);
327 	if (ret != SCSI_DH_OK || h->path_state == HP_SW_PATH_UNINITIALIZED)
328 		goto failed;
329 
330 	if (!try_module_get(THIS_MODULE))
331 		goto failed;
332 
333 	spin_lock_irqsave(sdev->request_queue->queue_lock, flags);
334 	sdev->scsi_dh_data = scsi_dh_data;
335 	spin_unlock_irqrestore(sdev->request_queue->queue_lock, flags);
336 
337 	sdev_printk(KERN_INFO, sdev, "%s: attached to %s path\n",
338 		    HP_SW_NAME, h->path_state == HP_SW_PATH_ACTIVE?
339 		    "active":"passive");
340 
341 	return 0;
342 
343 failed:
344 	kfree(scsi_dh_data);
345 	sdev_printk(KERN_ERR, sdev, "%s: not attached\n",
346 		    HP_SW_NAME);
347 	return -EINVAL;
348 }
349 
350 static void hp_sw_bus_detach( struct scsi_device *sdev )
351 {
352 	struct scsi_dh_data *scsi_dh_data;
353 	unsigned long flags;
354 
355 	spin_lock_irqsave(sdev->request_queue->queue_lock, flags);
356 	scsi_dh_data = sdev->scsi_dh_data;
357 	sdev->scsi_dh_data = NULL;
358 	spin_unlock_irqrestore(sdev->request_queue->queue_lock, flags);
359 	module_put(THIS_MODULE);
360 
361 	sdev_printk(KERN_NOTICE, sdev, "%s: Detached\n", HP_SW_NAME);
362 
363 	kfree(scsi_dh_data);
364 }
365 
366 static int __init hp_sw_init(void)
367 {
368 	return scsi_register_device_handler(&hp_sw_dh);
369 }
370 
371 static void __exit hp_sw_exit(void)
372 {
373 	scsi_unregister_device_handler(&hp_sw_dh);
374 }
375 
376 module_init(hp_sw_init);
377 module_exit(hp_sw_exit);
378 
379 MODULE_DESCRIPTION("HP Active/Passive driver");
380 MODULE_AUTHOR("Mike Christie <michaelc@cs.wisc.edu");
381 MODULE_LICENSE("GPL");
382