2 * Basic HP/COMPAQ MSA 1000 support. This is only needed if your HW cannot be
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>
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)
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.
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.
24 #include <scsi/scsi.h>
25 #include <scsi/scsi_dbg.h>
26 #include <scsi/scsi_eh.h>
27 #include <scsi/scsi_dh.h>
29 #define HP_SW_NAME "hp_sw"
31 #define HP_SW_TIMEOUT (60 * HZ)
32 #define HP_SW_RETRIES 3
34 #define HP_SW_PATH_UNINITIALIZED -1
35 #define HP_SW_PATH_ACTIVE 0
36 #define HP_SW_PATH_PASSIVE 1
38 struct hp_sw_dh_data
{
39 unsigned char sense
[SCSI_SENSE_BUFFERSIZE
];
44 static inline struct hp_sw_dh_data
*get_hp_sw_data(struct scsi_device
*sdev
)
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
);
52 * tur_done - Handle TEST UNIT READY return status
53 * @sdev: sdev the command has been sent to
54 * @errors: blk error code
56 * Returns SCSI_DH_DEV_OFFLINED if the sdev is on the passive path
58 static int tur_done(struct scsi_device
*sdev
, unsigned char *sense
)
60 struct scsi_sense_hdr sshdr
;
63 ret
= scsi_normalize_sense(sense
, SCSI_SENSE_BUFFERSIZE
, &sshdr
);
65 sdev_printk(KERN_WARNING
, sdev
,
66 "%s: sending tur failed, no sense available\n",
71 switch (sshdr
.sense_key
) {
73 ret
= SCSI_DH_IMM_RETRY
;
76 if ((sshdr
.asc
== 0x04) && (sshdr
.ascq
== 2)) {
78 * LUN not ready - Initialization command required
80 * This is the passive path
82 ret
= SCSI_DH_DEV_OFFLINED
;
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
,
99 * hp_sw_tur - Send TEST UNIT READY
100 * @sdev: sdev command should be sent to
102 * Use the TEST UNIT READY command to determine
105 static int hp_sw_tur(struct scsi_device
*sdev
, struct hp_sw_dh_data
*h
)
111 req
= blk_get_request(sdev
->request_queue
, WRITE
, GFP_NOIO
);
113 return SCSI_DH_RES_TEMP_UNAVAIL
;
115 req
->cmd_type
= REQ_TYPE_BLOCK_PC
;
116 req
->cmd_flags
|= REQ_FAILFAST_DEV
| REQ_FAILFAST_TRANSPORT
|
118 req
->cmd_len
= COMMAND_SIZE(TEST_UNIT_READY
);
119 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
);
125 ret
= blk_execute_rq(req
->q
, NULL
, req
, 1);
127 if (req
->sense_len
> 0) {
128 ret
= tur_done(sdev
, h
->sense
);
130 sdev_printk(KERN_WARNING
, sdev
,
131 "%s: sending tur failed with %x\n",
132 HP_SW_NAME
, req
->errors
);
136 h
->path_state
= HP_SW_PATH_ACTIVE
;
139 if (ret
== SCSI_DH_IMM_RETRY
) {
140 blk_put_request(req
);
143 if (ret
== SCSI_DH_DEV_OFFLINED
) {
144 h
->path_state
= HP_SW_PATH_PASSIVE
;
148 blk_put_request(req
);
154 * start_done - Handle START STOP UNIT return status
155 * @sdev: sdev the command has been sent to
156 * @errors: blk error code
158 static int start_done(struct scsi_device
*sdev
, unsigned char *sense
)
160 struct scsi_sense_hdr sshdr
;
163 rc
= scsi_normalize_sense(sense
, SCSI_SENSE_BUFFERSIZE
, &sshdr
);
165 sdev_printk(KERN_WARNING
, sdev
,
166 "%s: sending start_stop_unit failed, "
167 "no sense available\n",
171 switch (sshdr
.sense_key
) {
173 if ((sshdr
.asc
== 0x04) && (sshdr
.ascq
== 3)) {
175 * LUN not ready - manual intervention required
177 * Switch-over in progress, retry.
184 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
,
195 * hp_sw_start_stop - Send START STOP UNIT command
196 * @sdev: sdev command should be sent to
198 * Sending START STOP UNIT activates the SP.
200 static int hp_sw_start_stop(struct scsi_device
*sdev
, struct hp_sw_dh_data
*h
)
206 req
= blk_get_request(sdev
->request_queue
, WRITE
, GFP_NOIO
);
208 return SCSI_DH_RES_TEMP_UNAVAIL
;
210 req
->cmd_type
= REQ_TYPE_BLOCK_PC
;
211 req
->cmd_flags
|= REQ_FAILFAST_DEV
| REQ_FAILFAST_TRANSPORT
|
213 req
->cmd_len
= COMMAND_SIZE(START_STOP
);
214 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
);
222 ret
= blk_execute_rq(req
->q
, NULL
, req
, 1);
224 if (req
->sense_len
> 0) {
225 ret
= start_done(sdev
, h
->sense
);
227 sdev_printk(KERN_WARNING
, sdev
,
228 "%s: sending start_stop_unit failed with %x\n",
229 HP_SW_NAME
, req
->errors
);
235 if (ret
== SCSI_DH_RETRY
) {
237 blk_put_request(req
);
243 blk_put_request(req
);
248 static int hp_sw_prep_fn(struct scsi_device
*sdev
, struct request
*req
)
250 struct hp_sw_dh_data
*h
= get_hp_sw_data(sdev
);
251 int ret
= BLKPREP_OK
;
253 if (h
->path_state
!= HP_SW_PATH_ACTIVE
) {
255 req
->cmd_flags
|= REQ_QUIET
;
262 * hp_sw_activate - Activate a path
263 * @sdev: sdev on the path to be activated
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).
271 static int hp_sw_activate(struct scsi_device
*sdev
)
273 int ret
= SCSI_DH_OK
;
274 struct hp_sw_dh_data
*h
= get_hp_sw_data(sdev
);
276 ret
= hp_sw_tur(sdev
, h
);
278 if (ret
== SCSI_DH_OK
&& h
->path_state
== HP_SW_PATH_PASSIVE
) {
279 ret
= hp_sw_start_stop(sdev
, h
);
280 if (ret
== SCSI_DH_OK
)
281 sdev_printk(KERN_INFO
, sdev
,
282 "%s: activated path\n",
289 static const struct scsi_dh_devlist hp_sw_dh_data_list
[] = {
290 {"COMPAQ", "MSA1000 VOLUME"},
291 {"COMPAQ", "HSV110"},
297 static int hp_sw_bus_attach(struct scsi_device
*sdev
);
298 static void hp_sw_bus_detach(struct scsi_device
*sdev
);
300 static struct scsi_device_handler hp_sw_dh
= {
302 .module
= THIS_MODULE
,
303 .devlist
= hp_sw_dh_data_list
,
304 .attach
= hp_sw_bus_attach
,
305 .detach
= hp_sw_bus_detach
,
306 .activate
= hp_sw_activate
,
307 .prep_fn
= hp_sw_prep_fn
,
310 static int hp_sw_bus_attach(struct scsi_device
*sdev
)
312 struct scsi_dh_data
*scsi_dh_data
;
313 struct hp_sw_dh_data
*h
;
317 scsi_dh_data
= kzalloc(sizeof(struct scsi_device_handler
*)
318 + sizeof(struct hp_sw_dh_data
) , GFP_KERNEL
);
320 sdev_printk(KERN_ERR
, sdev
, "%s: Attach Failed\n",
325 scsi_dh_data
->scsi_dh
= &hp_sw_dh
;
326 h
= (struct hp_sw_dh_data
*) scsi_dh_data
->buf
;
327 h
->path_state
= HP_SW_PATH_UNINITIALIZED
;
328 h
->retries
= HP_SW_RETRIES
;
330 ret
= hp_sw_tur(sdev
, h
);
331 if (ret
!= SCSI_DH_OK
|| h
->path_state
== HP_SW_PATH_UNINITIALIZED
)
334 if (!try_module_get(THIS_MODULE
))
337 spin_lock_irqsave(sdev
->request_queue
->queue_lock
, flags
);
338 sdev
->scsi_dh_data
= scsi_dh_data
;
339 spin_unlock_irqrestore(sdev
->request_queue
->queue_lock
, flags
);
341 sdev_printk(KERN_INFO
, sdev
, "%s: attached to %s path\n",
342 HP_SW_NAME
, h
->path_state
== HP_SW_PATH_ACTIVE
?
349 sdev_printk(KERN_ERR
, sdev
, "%s: not attached\n",
354 static void hp_sw_bus_detach( struct scsi_device
*sdev
)
356 struct scsi_dh_data
*scsi_dh_data
;
359 spin_lock_irqsave(sdev
->request_queue
->queue_lock
, flags
);
360 scsi_dh_data
= sdev
->scsi_dh_data
;
361 sdev
->scsi_dh_data
= NULL
;
362 spin_unlock_irqrestore(sdev
->request_queue
->queue_lock
, flags
);
363 module_put(THIS_MODULE
);
365 sdev_printk(KERN_NOTICE
, sdev
, "%s: Detached\n", HP_SW_NAME
);
370 static int __init
hp_sw_init(void)
372 return scsi_register_device_handler(&hp_sw_dh
);
375 static void __exit
hp_sw_exit(void)
377 scsi_unregister_device_handler(&hp_sw_dh
);
380 module_init(hp_sw_init
);
381 module_exit(hp_sw_exit
);
383 MODULE_DESCRIPTION("HP Active/Passive driver");
384 MODULE_AUTHOR("Mike Christie <michaelc@cs.wisc.edu");
385 MODULE_LICENSE("GPL");