scsi: smartpqi: enhance BMIC cache flush

- distinguish between shutdown and non-shutdown.

Reviewed-by: Scott Benesh <scott.benesh@microsemi.com>
Signed-off-by: Kevin Barnett <kevin.barnett@microsemi.com>
Signed-off-by: Don Brace <don.brace@microsemi.com>
Signed-off-by: Martin K. Petersen <martin.petersen@oracle.com>
This commit is contained in:
Kevin Barnett 2017-08-10 13:46:45 -05:00 committed by Martin K. Petersen
parent 336b681931
commit 58322fe006
2 changed files with 33 additions and 15 deletions

View File

@ -1079,9 +1079,9 @@ enum pqi_ctrl_mode {
#define BMIC_SENSE_CONTROLLER_PARAMETERS 0x64 #define BMIC_SENSE_CONTROLLER_PARAMETERS 0x64
#define BMIC_SENSE_SUBSYSTEM_INFORMATION 0x66 #define BMIC_SENSE_SUBSYSTEM_INFORMATION 0x66
#define BMIC_WRITE_HOST_WELLNESS 0xa5 #define BMIC_WRITE_HOST_WELLNESS 0xa5
#define BMIC_CACHE_FLUSH 0xc2 #define BMIC_FLUSH_CACHE 0xc2
#define SA_CACHE_FLUSH 0x1 #define SA_FLUSH_CACHE 0x1
#define MASKED_DEVICE(lunid) ((lunid)[3] & 0xc0) #define MASKED_DEVICE(lunid) ((lunid)[3] & 0xc0)
#define CISS_GET_LEVEL_2_BUS(lunid) ((lunid)[7] & 0x3f) #define CISS_GET_LEVEL_2_BUS(lunid) ((lunid)[7] & 0x3f)
@ -1187,6 +1187,23 @@ struct bmic_identify_physical_device {
u8 padding_to_multiple_of_512[9]; u8 padding_to_multiple_of_512[9];
}; };
struct bmic_flush_cache {
u8 disable_flag;
u8 system_power_action;
u8 ndu_flush;
u8 shutdown_event;
u8 reserved[28];
};
/* for shutdown_event member of struct bmic_flush_cache */
enum bmic_flush_cache_shutdown_event {
NONE_CACHE_FLUSH_ONLY = 0,
SHUTDOWN = 1,
HIBERNATE = 2,
SUSPEND = 3,
RESTART = 4
};
#pragma pack() #pragma pack()
int pqi_add_sas_host(struct Scsi_Host *shost, struct pqi_ctrl_info *ctrl_info); int pqi_add_sas_host(struct Scsi_Host *shost, struct pqi_ctrl_info *ctrl_info);

View File

@ -431,10 +431,10 @@ static int pqi_build_raid_path_request(struct pqi_ctrl_info *ctrl_info,
cdb[1] = CISS_GET_RAID_MAP; cdb[1] = CISS_GET_RAID_MAP;
put_unaligned_be32(buffer_length, &cdb[6]); put_unaligned_be32(buffer_length, &cdb[6]);
break; break;
case SA_CACHE_FLUSH: case SA_FLUSH_CACHE:
request->data_direction = SOP_WRITE_FLAG; request->data_direction = SOP_WRITE_FLAG;
cdb[0] = BMIC_WRITE; cdb[0] = BMIC_WRITE;
cdb[6] = BMIC_CACHE_FLUSH; cdb[6] = BMIC_FLUSH_CACHE;
put_unaligned_be16(buffer_length, &cdb[7]); put_unaligned_be16(buffer_length, &cdb[7]);
break; break;
case BMIC_IDENTIFY_CONTROLLER: case BMIC_IDENTIFY_CONTROLLER:
@ -585,14 +585,13 @@ static int pqi_identify_physical_device(struct pqi_ctrl_info *ctrl_info,
return rc; return rc;
} }
#define SA_CACHE_FLUSH_BUFFER_LENGTH 4 static int pqi_flush_cache(struct pqi_ctrl_info *ctrl_info,
enum bmic_flush_cache_shutdown_event shutdown_event)
static int pqi_flush_cache(struct pqi_ctrl_info *ctrl_info)
{ {
int rc; int rc;
struct pqi_raid_path_request request; struct pqi_raid_path_request request;
int pci_direction; int pci_direction;
u8 *buffer; struct bmic_flush_cache *flush_cache;
/* /*
* Don't bother trying to flush the cache if the controller is * Don't bother trying to flush the cache if the controller is
@ -601,13 +600,15 @@ static int pqi_flush_cache(struct pqi_ctrl_info *ctrl_info)
if (pqi_ctrl_offline(ctrl_info)) if (pqi_ctrl_offline(ctrl_info))
return -ENXIO; return -ENXIO;
buffer = kzalloc(SA_CACHE_FLUSH_BUFFER_LENGTH, GFP_KERNEL); flush_cache = kzalloc(sizeof(*flush_cache), GFP_KERNEL);
if (!buffer) if (!flush_cache)
return -ENOMEM; return -ENOMEM;
flush_cache->shutdown_event = shutdown_event;
rc = pqi_build_raid_path_request(ctrl_info, &request, rc = pqi_build_raid_path_request(ctrl_info, &request,
SA_CACHE_FLUSH, RAID_CTLR_LUNID, buffer, SA_FLUSH_CACHE, RAID_CTLR_LUNID, flush_cache,
SA_CACHE_FLUSH_BUFFER_LENGTH, 0, &pci_direction); sizeof(*flush_cache), 0, &pci_direction);
if (rc) if (rc)
goto out; goto out;
@ -618,7 +619,7 @@ static int pqi_flush_cache(struct pqi_ctrl_info *ctrl_info)
pci_direction); pci_direction);
out: out:
kfree(buffer); kfree(flush_cache);
return rc; return rc;
} }
@ -6693,7 +6694,7 @@ static void pqi_shutdown(struct pci_dev *pci_dev)
* Write all data in the controller's battery-backed cache to * Write all data in the controller's battery-backed cache to
* storage. * storage.
*/ */
rc = pqi_flush_cache(ctrl_info); rc = pqi_flush_cache(ctrl_info, SHUTDOWN);
if (rc == 0) if (rc == 0)
return; return;
@ -6737,7 +6738,7 @@ static __maybe_unused int pqi_suspend(struct pci_dev *pci_dev, pm_message_t stat
pqi_cancel_rescan_worker(ctrl_info); pqi_cancel_rescan_worker(ctrl_info);
pqi_wait_until_scan_finished(ctrl_info); pqi_wait_until_scan_finished(ctrl_info);
pqi_wait_until_lun_reset_finished(ctrl_info); pqi_wait_until_lun_reset_finished(ctrl_info);
pqi_flush_cache(ctrl_info); pqi_flush_cache(ctrl_info, SUSPEND);
pqi_ctrl_block_requests(ctrl_info); pqi_ctrl_block_requests(ctrl_info);
pqi_ctrl_wait_until_quiesced(ctrl_info); pqi_ctrl_wait_until_quiesced(ctrl_info);
pqi_wait_until_inbound_queues_empty(ctrl_info); pqi_wait_until_inbound_queues_empty(ctrl_info);