Commit c49c8345 authored by Hannes Reinecke's avatar Hannes Reinecke Committed by Martin K. Petersen

scsi_dh_alua: allocate RTPG buffer separately

The RTPG buffer will only evaluated within alua_rtpg(),
so we can allocate it locally there and avoid having to
put it into the global structure.
Reviewed-by: default avatarEwan Milne <emilne@redhat.com>
Reviewed-by: default avatarChristoph Hellwig <hch@lst.de>
Signed-off-by: default avatarHannes Reinecke <hare@suse.de>
Signed-off-by: default avatarMartin K. Petersen <martin.petersen@oracle.com>
parent 40bb61a7
...@@ -56,7 +56,7 @@ ...@@ -56,7 +56,7 @@
#define TPGS_MODE_IMPLICIT 0x1 #define TPGS_MODE_IMPLICIT 0x1
#define TPGS_MODE_EXPLICIT 0x2 #define TPGS_MODE_EXPLICIT 0x2
#define ALUA_INQUIRY_SIZE 36 #define ALUA_RTPG_SIZE 128
#define ALUA_FAILOVER_TIMEOUT 60 #define ALUA_FAILOVER_TIMEOUT 60
#define ALUA_FAILOVER_RETRIES 5 #define ALUA_FAILOVER_RETRIES 5
...@@ -71,9 +71,6 @@ struct alua_dh_data { ...@@ -71,9 +71,6 @@ struct alua_dh_data {
int state; int state;
int pref; int pref;
unsigned flags; /* used for optimizing STPG */ unsigned flags; /* used for optimizing STPG */
unsigned char inq[ALUA_INQUIRY_SIZE];
unsigned char *buff;
int bufflen;
unsigned char transition_tmo; unsigned char transition_tmo;
struct scsi_device *sdev; struct scsi_device *sdev;
activate_complete callback_fn; activate_complete callback_fn;
...@@ -85,21 +82,6 @@ struct alua_dh_data { ...@@ -85,21 +82,6 @@ struct alua_dh_data {
static char print_alua_state(int); static char print_alua_state(int);
static int realloc_buffer(struct alua_dh_data *h, unsigned len)
{
if (h->buff && h->buff != h->inq)
kfree(h->buff);
h->buff = kmalloc(len, GFP_NOIO);
if (!h->buff) {
h->buff = h->inq;
h->bufflen = ALUA_INQUIRY_SIZE;
return 1;
}
h->bufflen = len;
return 0;
}
/* /*
* submit_rtpg - Issue a REPORT TARGET GROUP STATES command * submit_rtpg - Issue a REPORT TARGET GROUP STATES command
* @sdev: sdev the command should be sent to * @sdev: sdev the command should be sent to
...@@ -333,8 +315,8 @@ static int alua_check_sense(struct scsi_device *sdev, ...@@ -333,8 +315,8 @@ static int alua_check_sense(struct scsi_device *sdev,
static int alua_rtpg(struct scsi_device *sdev, struct alua_dh_data *h, int wait_for_transition) static int alua_rtpg(struct scsi_device *sdev, struct alua_dh_data *h, int wait_for_transition)
{ {
struct scsi_sense_hdr sense_hdr; struct scsi_sense_hdr sense_hdr;
int len, k, off, valid_states = 0; int len, k, off, valid_states = 0, bufflen = ALUA_RTPG_SIZE;
unsigned char *ucp; unsigned char *ucp, *buff;
unsigned err, retval; unsigned err, retval;
unsigned long expiry, interval = 0; unsigned long expiry, interval = 0;
unsigned int tpg_desc_tbl_off; unsigned int tpg_desc_tbl_off;
...@@ -345,14 +327,19 @@ static int alua_rtpg(struct scsi_device *sdev, struct alua_dh_data *h, int wait_ ...@@ -345,14 +327,19 @@ static int alua_rtpg(struct scsi_device *sdev, struct alua_dh_data *h, int wait_
else else
expiry = round_jiffies_up(jiffies + h->transition_tmo * HZ); expiry = round_jiffies_up(jiffies + h->transition_tmo * HZ);
buff = kzalloc(bufflen, GFP_KERNEL);
if (!buff)
return SCSI_DH_DEV_TEMP_BUSY;
retry: retry:
retval = submit_rtpg(sdev, h->buff, h->bufflen, &sense_hdr, h->flags); retval = submit_rtpg(sdev, buff, bufflen, &sense_hdr, h->flags);
if (retval) { if (retval) {
if (!scsi_sense_valid(&sense_hdr)) { if (!scsi_sense_valid(&sense_hdr)) {
sdev_printk(KERN_INFO, sdev, sdev_printk(KERN_INFO, sdev,
"%s: rtpg failed, result %d\n", "%s: rtpg failed, result %d\n",
ALUA_DH_NAME, retval); ALUA_DH_NAME, retval);
kfree(buff);
if (driver_byte(retval) == DRIVER_ERROR) if (driver_byte(retval) == DRIVER_ERROR)
return SCSI_DH_DEV_TEMP_BUSY; return SCSI_DH_DEV_TEMP_BUSY;
return SCSI_DH_IO; return SCSI_DH_IO;
...@@ -390,14 +377,18 @@ static int alua_rtpg(struct scsi_device *sdev, struct alua_dh_data *h, int wait_ ...@@ -390,14 +377,18 @@ static int alua_rtpg(struct scsi_device *sdev, struct alua_dh_data *h, int wait_
sdev_printk(KERN_ERR, sdev, "%s: rtpg failed\n", sdev_printk(KERN_ERR, sdev, "%s: rtpg failed\n",
ALUA_DH_NAME); ALUA_DH_NAME);
scsi_print_sense_hdr(sdev, ALUA_DH_NAME, &sense_hdr); scsi_print_sense_hdr(sdev, ALUA_DH_NAME, &sense_hdr);
kfree(buff);
return SCSI_DH_IO; return SCSI_DH_IO;
} }
len = get_unaligned_be32(&h->buff[0]) + 4; len = get_unaligned_be32(&buff[0]) + 4;
if (len > h->bufflen) { if (len > bufflen) {
/* Resubmit with the correct length */ /* Resubmit with the correct length */
if (realloc_buffer(h, len)) { kfree(buff);
bufflen = len;
buff = kmalloc(bufflen, GFP_KERNEL);
if (!buff) {
sdev_printk(KERN_WARNING, sdev, sdev_printk(KERN_WARNING, sdev,
"%s: kmalloc buffer failed\n",__func__); "%s: kmalloc buffer failed\n",__func__);
/* Temporary failure, bypass */ /* Temporary failure, bypass */
...@@ -407,24 +398,25 @@ static int alua_rtpg(struct scsi_device *sdev, struct alua_dh_data *h, int wait_ ...@@ -407,24 +398,25 @@ static int alua_rtpg(struct scsi_device *sdev, struct alua_dh_data *h, int wait_
} }
orig_transition_tmo = h->transition_tmo; orig_transition_tmo = h->transition_tmo;
if ((h->buff[4] & RTPG_FMT_MASK) == RTPG_FMT_EXT_HDR && h->buff[5] != 0) if ((buff[4] & RTPG_FMT_MASK) == RTPG_FMT_EXT_HDR && buff[5] != 0)
h->transition_tmo = h->buff[5]; h->transition_tmo = buff[5];
else else
h->transition_tmo = ALUA_FAILOVER_TIMEOUT; h->transition_tmo = ALUA_FAILOVER_TIMEOUT;
if (wait_for_transition && (orig_transition_tmo != h->transition_tmo)) { if (wait_for_transition &&
(orig_transition_tmo != h->transition_tmo)) {
sdev_printk(KERN_INFO, sdev, sdev_printk(KERN_INFO, sdev,
"%s: transition timeout set to %d seconds\n", "%s: transition timeout set to %d seconds\n",
ALUA_DH_NAME, h->transition_tmo); ALUA_DH_NAME, h->transition_tmo);
expiry = jiffies + h->transition_tmo * HZ; expiry = jiffies + h->transition_tmo * HZ;
} }
if ((h->buff[4] & RTPG_FMT_MASK) == RTPG_FMT_EXT_HDR) if ((buff[4] & RTPG_FMT_MASK) == RTPG_FMT_EXT_HDR)
tpg_desc_tbl_off = 8; tpg_desc_tbl_off = 8;
else else
tpg_desc_tbl_off = 4; tpg_desc_tbl_off = 4;
for (k = tpg_desc_tbl_off, ucp = h->buff + tpg_desc_tbl_off; for (k = tpg_desc_tbl_off, ucp = buff + tpg_desc_tbl_off;
k < len; k < len;
k += off, ucp += off) { k += off, ucp += off) {
...@@ -474,6 +466,7 @@ static int alua_rtpg(struct scsi_device *sdev, struct alua_dh_data *h, int wait_ ...@@ -474,6 +466,7 @@ static int alua_rtpg(struct scsi_device *sdev, struct alua_dh_data *h, int wait_
err = SCSI_DH_OK; err = SCSI_DH_OK;
break; break;
} }
kfree(buff);
return err; return err;
} }
...@@ -668,8 +661,6 @@ static int alua_bus_attach(struct scsi_device *sdev) ...@@ -668,8 +661,6 @@ static int alua_bus_attach(struct scsi_device *sdev)
h->state = TPGS_STATE_OPTIMIZED; h->state = TPGS_STATE_OPTIMIZED;
h->group_id = -1; h->group_id = -1;
h->rel_port = -1; h->rel_port = -1;
h->buff = h->inq;
h->bufflen = ALUA_INQUIRY_SIZE;
h->sdev = sdev; h->sdev = sdev;
err = alua_initialize(sdev, h); err = alua_initialize(sdev, h);
...@@ -691,8 +682,6 @@ static void alua_bus_detach(struct scsi_device *sdev) ...@@ -691,8 +682,6 @@ static void alua_bus_detach(struct scsi_device *sdev)
{ {
struct alua_dh_data *h = sdev->handler_data; struct alua_dh_data *h = sdev->handler_data;
if (h->buff && h->inq != h->buff)
kfree(h->buff);
sdev->handler_data = NULL; sdev->handler_data = NULL;
kfree(h); kfree(h);
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment