Commit 6bf3b630 authored by Raghava Aditya Renukunta's avatar Raghava Aditya Renukunta Committed by Martin K. Petersen

aacraid: SCSI blk tag support

The method to allocate and free FIB's in the present code utilizes
spinlocks. Multiple IO's have to wait on the spinlock to acquire or free
fibs creating a performance bottleneck.

An alternative solution would be to use block layer tags to keep track
of the fibs allocated and freed. To this end aac_fib_alloc_tag was
created to utilize the blk layer tags to plug into the Fib pool.These
functions are used exclusively in the IO path. 8 fibs are reserved for
the use of AIF management software and utilize the previous spinlock
based implementations.
Signed-off-by: default avatarRaghava Aditya Renukunta <RaghavaAditya.Renukunta@pmcs.com>
Reviewed-by: default avatarShane Seymour <shane.seymour@hpe.com>
Reviewed-by: default avatarJohannes Thumshirn <jthumshirn@suse.de>
Reviewed-by: default avatarTomas Henzl <thenzl@redhat.com>
Signed-off-by: default avatarMartin K. Petersen <martin.petersen@oracle.com>
parent bd8d859a
...@@ -323,7 +323,6 @@ static inline int aac_valid_context(struct scsi_cmnd *scsicmd, ...@@ -323,7 +323,6 @@ static inline int aac_valid_context(struct scsi_cmnd *scsicmd,
if (unlikely(!scsicmd || !scsicmd->scsi_done)) { if (unlikely(!scsicmd || !scsicmd->scsi_done)) {
dprintk((KERN_WARNING "aac_valid_context: scsi command corrupt\n")); dprintk((KERN_WARNING "aac_valid_context: scsi command corrupt\n"));
aac_fib_complete(fibptr); aac_fib_complete(fibptr);
aac_fib_free(fibptr);
return 0; return 0;
} }
scsicmd->SCp.phase = AAC_OWNER_MIDLEVEL; scsicmd->SCp.phase = AAC_OWNER_MIDLEVEL;
...@@ -331,7 +330,6 @@ static inline int aac_valid_context(struct scsi_cmnd *scsicmd, ...@@ -331,7 +330,6 @@ static inline int aac_valid_context(struct scsi_cmnd *scsicmd,
if (unlikely(!device || !scsi_device_online(device))) { if (unlikely(!device || !scsi_device_online(device))) {
dprintk((KERN_WARNING "aac_valid_context: scsi device corrupt\n")); dprintk((KERN_WARNING "aac_valid_context: scsi device corrupt\n"));
aac_fib_complete(fibptr); aac_fib_complete(fibptr);
aac_fib_free(fibptr);
return 0; return 0;
} }
return 1; return 1;
...@@ -541,7 +539,6 @@ static void get_container_name_callback(void *context, struct fib * fibptr) ...@@ -541,7 +539,6 @@ static void get_container_name_callback(void *context, struct fib * fibptr)
scsicmd->result = DID_OK << 16 | COMMAND_COMPLETE << 8 | SAM_STAT_GOOD; scsicmd->result = DID_OK << 16 | COMMAND_COMPLETE << 8 | SAM_STAT_GOOD;
aac_fib_complete(fibptr); aac_fib_complete(fibptr);
aac_fib_free(fibptr);
scsicmd->scsi_done(scsicmd); scsicmd->scsi_done(scsicmd);
} }
...@@ -557,7 +554,8 @@ static int aac_get_container_name(struct scsi_cmnd * scsicmd) ...@@ -557,7 +554,8 @@ static int aac_get_container_name(struct scsi_cmnd * scsicmd)
dev = (struct aac_dev *)scsicmd->device->host->hostdata; dev = (struct aac_dev *)scsicmd->device->host->hostdata;
if (!(cmd_fibcontext = aac_fib_alloc(dev))) cmd_fibcontext = aac_fib_alloc_tag(dev, scsicmd);
if (!cmd_fibcontext)
return -ENOMEM; return -ENOMEM;
aac_fib_init(cmd_fibcontext); aac_fib_init(cmd_fibcontext);
...@@ -586,7 +584,6 @@ static int aac_get_container_name(struct scsi_cmnd * scsicmd) ...@@ -586,7 +584,6 @@ static int aac_get_container_name(struct scsi_cmnd * scsicmd)
printk(KERN_WARNING "aac_get_container_name: aac_fib_send failed with status: %d.\n", status); printk(KERN_WARNING "aac_get_container_name: aac_fib_send failed with status: %d.\n", status);
aac_fib_complete(cmd_fibcontext); aac_fib_complete(cmd_fibcontext);
aac_fib_free(cmd_fibcontext);
return -1; return -1;
} }
...@@ -1024,7 +1021,6 @@ static void get_container_serial_callback(void *context, struct fib * fibptr) ...@@ -1024,7 +1021,6 @@ static void get_container_serial_callback(void *context, struct fib * fibptr)
scsicmd->result = DID_OK << 16 | COMMAND_COMPLETE << 8 | SAM_STAT_GOOD; scsicmd->result = DID_OK << 16 | COMMAND_COMPLETE << 8 | SAM_STAT_GOOD;
aac_fib_complete(fibptr); aac_fib_complete(fibptr);
aac_fib_free(fibptr);
scsicmd->scsi_done(scsicmd); scsicmd->scsi_done(scsicmd);
} }
...@@ -1040,7 +1036,8 @@ static int aac_get_container_serial(struct scsi_cmnd * scsicmd) ...@@ -1040,7 +1036,8 @@ static int aac_get_container_serial(struct scsi_cmnd * scsicmd)
dev = (struct aac_dev *)scsicmd->device->host->hostdata; dev = (struct aac_dev *)scsicmd->device->host->hostdata;
if (!(cmd_fibcontext = aac_fib_alloc(dev))) cmd_fibcontext = aac_fib_alloc_tag(dev, scsicmd);
if (!cmd_fibcontext)
return -ENOMEM; return -ENOMEM;
aac_fib_init(cmd_fibcontext); aac_fib_init(cmd_fibcontext);
...@@ -1068,7 +1065,6 @@ static int aac_get_container_serial(struct scsi_cmnd * scsicmd) ...@@ -1068,7 +1065,6 @@ static int aac_get_container_serial(struct scsi_cmnd * scsicmd)
printk(KERN_WARNING "aac_get_container_serial: aac_fib_send failed with status: %d.\n", status); printk(KERN_WARNING "aac_get_container_serial: aac_fib_send failed with status: %d.\n", status);
aac_fib_complete(cmd_fibcontext); aac_fib_complete(cmd_fibcontext);
aac_fib_free(cmd_fibcontext);
return -1; return -1;
} }
...@@ -1869,7 +1865,6 @@ static void io_callback(void *context, struct fib * fibptr) ...@@ -1869,7 +1865,6 @@ static void io_callback(void *context, struct fib * fibptr)
break; break;
} }
aac_fib_complete(fibptr); aac_fib_complete(fibptr);
aac_fib_free(fibptr);
scsicmd->scsi_done(scsicmd); scsicmd->scsi_done(scsicmd);
} }
...@@ -1954,7 +1949,8 @@ static int aac_read(struct scsi_cmnd * scsicmd) ...@@ -1954,7 +1949,8 @@ static int aac_read(struct scsi_cmnd * scsicmd)
/* /*
* Alocate and initialize a Fib * Alocate and initialize a Fib
*/ */
if (!(cmd_fibcontext = aac_fib_alloc(dev))) { cmd_fibcontext = aac_fib_alloc_tag(dev, scsicmd);
if (!cmd_fibcontext) {
printk(KERN_WARNING "aac_read: fib allocation failed\n"); printk(KERN_WARNING "aac_read: fib allocation failed\n");
return -1; return -1;
} }
...@@ -2051,7 +2047,8 @@ static int aac_write(struct scsi_cmnd * scsicmd) ...@@ -2051,7 +2047,8 @@ static int aac_write(struct scsi_cmnd * scsicmd)
/* /*
* Allocate and initialize a Fib then setup a BlockWrite command * Allocate and initialize a Fib then setup a BlockWrite command
*/ */
if (!(cmd_fibcontext = aac_fib_alloc(dev))) { cmd_fibcontext = aac_fib_alloc_tag(dev, scsicmd);
if (!cmd_fibcontext) {
/* FIB temporarily unavailable,not catastrophic failure */ /* FIB temporarily unavailable,not catastrophic failure */
/* scsicmd->result = DID_ERROR << 16; /* scsicmd->result = DID_ERROR << 16;
...@@ -2285,7 +2282,7 @@ static int aac_start_stop(struct scsi_cmnd *scsicmd) ...@@ -2285,7 +2282,7 @@ static int aac_start_stop(struct scsi_cmnd *scsicmd)
/* /*
* Allocate and initialize a Fib * Allocate and initialize a Fib
*/ */
cmd_fibcontext = aac_fib_alloc(aac); cmd_fibcontext = aac_fib_alloc_tag(aac, scsicmd);
if (!cmd_fibcontext) if (!cmd_fibcontext)
return SCSI_MLQUEUE_HOST_BUSY; return SCSI_MLQUEUE_HOST_BUSY;
...@@ -3157,7 +3154,6 @@ static void aac_srb_callback(void *context, struct fib * fibptr) ...@@ -3157,7 +3154,6 @@ static void aac_srb_callback(void *context, struct fib * fibptr)
scsicmd->result |= le32_to_cpu(srbreply->scsi_status); scsicmd->result |= le32_to_cpu(srbreply->scsi_status);
aac_fib_complete(fibptr); aac_fib_complete(fibptr);
aac_fib_free(fibptr);
scsicmd->scsi_done(scsicmd); scsicmd->scsi_done(scsicmd);
} }
...@@ -3187,9 +3183,10 @@ static int aac_send_srb_fib(struct scsi_cmnd* scsicmd) ...@@ -3187,9 +3183,10 @@ static int aac_send_srb_fib(struct scsi_cmnd* scsicmd)
/* /*
* Allocate and initialize a Fib then setup a BlockWrite command * Allocate and initialize a Fib then setup a BlockWrite command
*/ */
if (!(cmd_fibcontext = aac_fib_alloc(dev))) { cmd_fibcontext = aac_fib_alloc_tag(dev, scsicmd);
if (!cmd_fibcontext)
return -1; return -1;
}
status = aac_adapter_scsi(cmd_fibcontext, scsicmd); status = aac_adapter_scsi(cmd_fibcontext, scsicmd);
/* /*
......
...@@ -2114,6 +2114,7 @@ int aac_acquire_irq(struct aac_dev *dev); ...@@ -2114,6 +2114,7 @@ int aac_acquire_irq(struct aac_dev *dev);
void aac_free_irq(struct aac_dev *dev); void aac_free_irq(struct aac_dev *dev);
const char *aac_driverinfo(struct Scsi_Host *); const char *aac_driverinfo(struct Scsi_Host *);
struct fib *aac_fib_alloc(struct aac_dev *dev); struct fib *aac_fib_alloc(struct aac_dev *dev);
struct fib *aac_fib_alloc_tag(struct aac_dev *dev, struct scsi_cmnd *scmd);
int aac_fib_setup(struct aac_dev *dev); int aac_fib_setup(struct aac_dev *dev);
void aac_fib_map_free(struct aac_dev *dev); void aac_fib_map_free(struct aac_dev *dev);
void aac_fib_free(struct fib * context); void aac_fib_free(struct fib * context);
......
...@@ -137,6 +137,7 @@ int aac_fib_setup(struct aac_dev * dev) ...@@ -137,6 +137,7 @@ int aac_fib_setup(struct aac_dev * dev)
i++, fibptr++) i++, fibptr++)
{ {
fibptr->flags = 0; fibptr->flags = 0;
fibptr->size = sizeof(struct fib);
fibptr->dev = dev; fibptr->dev = dev;
fibptr->hw_fib_va = hw_fib; fibptr->hw_fib_va = hw_fib;
fibptr->data = (void *) fibptr->hw_fib_va->data; fibptr->data = (void *) fibptr->hw_fib_va->data;
...@@ -156,12 +157,37 @@ int aac_fib_setup(struct aac_dev * dev) ...@@ -156,12 +157,37 @@ int aac_fib_setup(struct aac_dev * dev)
*/ */
dev->fibs[dev->scsi_host_ptr->can_queue + AAC_NUM_MGT_FIB - 1].next = NULL; dev->fibs[dev->scsi_host_ptr->can_queue + AAC_NUM_MGT_FIB - 1].next = NULL;
/* /*
* Enable this to debug out of queue space * Set 8 fibs aside for management tools
*/ */
dev->free_fib = &dev->fibs[0]; dev->free_fib = &dev->fibs[dev->scsi_host_ptr->can_queue];
return 0; return 0;
} }
/**
* aac_fib_alloc_tag-allocate a fib using tags
* @dev: Adapter to allocate the fib for
*
* Allocate a fib from the adapter fib pool using tags
* from the blk layer.
*/
struct fib *aac_fib_alloc_tag(struct aac_dev *dev, struct scsi_cmnd *scmd)
{
struct fib *fibptr;
fibptr = &dev->fibs[scmd->request->tag];
/*
* Null out fields that depend on being zero at the start of
* each I/O
*/
fibptr->hw_fib_va->header.XferState = 0;
fibptr->type = FSAFS_NTC_FIB_CONTEXT;
fibptr->callback_data = NULL;
fibptr->callback = NULL;
return fibptr;
}
/** /**
* aac_fib_alloc - allocate a fib * aac_fib_alloc - allocate a fib
* @dev: Adapter to allocate the fib for * @dev: Adapter to allocate the fib for
......
...@@ -394,7 +394,6 @@ unsigned int aac_intr_normal(struct aac_dev *dev, u32 index, ...@@ -394,7 +394,6 @@ unsigned int aac_intr_normal(struct aac_dev *dev, u32 index,
fib->callback(fib->callback_data, fib); fib->callback(fib->callback_data, fib);
} else { } else {
aac_fib_complete(fib); aac_fib_complete(fib);
aac_fib_free(fib);
} }
} else { } else {
unsigned long flagv; unsigned long flagv;
...@@ -416,7 +415,6 @@ unsigned int aac_intr_normal(struct aac_dev *dev, u32 index, ...@@ -416,7 +415,6 @@ unsigned int aac_intr_normal(struct aac_dev *dev, u32 index,
fib->done = 0; fib->done = 0;
spin_unlock_irqrestore(&fib->event_lock, flagv); spin_unlock_irqrestore(&fib->event_lock, flagv);
aac_fib_complete(fib); aac_fib_complete(fib);
aac_fib_free(fib);
} }
} }
......
...@@ -454,6 +454,8 @@ static int aac_slave_configure(struct scsi_device *sdev) ...@@ -454,6 +454,8 @@ static int aac_slave_configure(struct scsi_device *sdev)
} else } else
scsi_change_queue_depth(sdev, 1); scsi_change_queue_depth(sdev, 1);
sdev->tagged_supported = 1;
return 0; return 0;
} }
......
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