aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/scsi/3w-9xxx.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/scsi/3w-9xxx.c')
-rw-r--r--drivers/scsi/3w-9xxx.c135
1 files changed, 37 insertions, 98 deletions
diff --git a/drivers/scsi/3w-9xxx.c b/drivers/scsi/3w-9xxx.c
index eb766c3af1c8..76c09097175f 100644
--- a/drivers/scsi/3w-9xxx.c
+++ b/drivers/scsi/3w-9xxx.c
@@ -1306,22 +1306,26 @@ static irqreturn_t twa_interrupt(int irq, void *dev_instance)
1306 wake_up(&tw_dev->ioctl_wqueue); 1306 wake_up(&tw_dev->ioctl_wqueue);
1307 } 1307 }
1308 } else { 1308 } else {
1309 struct scsi_cmnd *cmd;
1310
1311 cmd = tw_dev->srb[request_id];
1312
1309 twa_scsiop_execute_scsi_complete(tw_dev, request_id); 1313 twa_scsiop_execute_scsi_complete(tw_dev, request_id);
1310 /* If no error command was a success */ 1314 /* If no error command was a success */
1311 if (error == 0) { 1315 if (error == 0) {
1312 tw_dev->srb[request_id]->result = (DID_OK << 16); 1316 cmd->result = (DID_OK << 16);
1313 } 1317 }
1314 1318
1315 /* If error, command failed */ 1319 /* If error, command failed */
1316 if (error == 1) { 1320 if (error == 1) {
1317 /* Ask for a host reset */ 1321 /* Ask for a host reset */
1318 tw_dev->srb[request_id]->result = (DID_OK << 16) | (CHECK_CONDITION << 1); 1322 cmd->result = (DID_OK << 16) | (CHECK_CONDITION << 1);
1319 } 1323 }
1320 1324
1321 /* Report residual bytes for single sgl */ 1325 /* Report residual bytes for single sgl */
1322 if ((tw_dev->srb[request_id]->use_sg <= 1) && (full_command_packet->command.newcommand.status == 0)) { 1326 if ((scsi_sg_count(cmd) <= 1) && (full_command_packet->command.newcommand.status == 0)) {
1323 if (full_command_packet->command.newcommand.sg_list[0].length < tw_dev->srb[request_id]->request_bufflen) 1327 if (full_command_packet->command.newcommand.sg_list[0].length < scsi_bufflen(tw_dev->srb[request_id]))
1324 tw_dev->srb[request_id]->resid = tw_dev->srb[request_id]->request_bufflen - full_command_packet->command.newcommand.sg_list[0].length; 1328 scsi_set_resid(cmd, scsi_bufflen(cmd) - full_command_packet->command.newcommand.sg_list[0].length);
1325 } 1329 }
1326 1330
1327 /* Now complete the io */ 1331 /* Now complete the io */
@@ -1384,52 +1388,20 @@ static int twa_map_scsi_sg_data(TW_Device_Extension *tw_dev, int request_id)
1384{ 1388{
1385 int use_sg; 1389 int use_sg;
1386 struct scsi_cmnd *cmd = tw_dev->srb[request_id]; 1390 struct scsi_cmnd *cmd = tw_dev->srb[request_id];
1387 struct pci_dev *pdev = tw_dev->tw_pci_dev;
1388 int retval = 0;
1389
1390 if (cmd->use_sg == 0)
1391 goto out;
1392
1393 use_sg = pci_map_sg(pdev, cmd->request_buffer, cmd->use_sg, DMA_BIDIRECTIONAL);
1394 1391
1395 if (use_sg == 0) { 1392 use_sg = scsi_dma_map(cmd);
1393 if (!use_sg)
1394 return 0;
1395 else if (use_sg < 0) {
1396 TW_PRINTK(tw_dev->host, TW_DRIVER, 0x1c, "Failed to map scatter gather list"); 1396 TW_PRINTK(tw_dev->host, TW_DRIVER, 0x1c, "Failed to map scatter gather list");
1397 goto out; 1397 return 0;
1398 } 1398 }
1399 1399
1400 cmd->SCp.phase = TW_PHASE_SGLIST; 1400 cmd->SCp.phase = TW_PHASE_SGLIST;
1401 cmd->SCp.have_data_in = use_sg; 1401 cmd->SCp.have_data_in = use_sg;
1402 retval = use_sg;
1403out:
1404 return retval;
1405} /* End twa_map_scsi_sg_data() */
1406
1407/* This function will perform a pci-dma map for a single buffer */
1408static dma_addr_t twa_map_scsi_single_data(TW_Device_Extension *tw_dev, int request_id)
1409{
1410 dma_addr_t mapping;
1411 struct scsi_cmnd *cmd = tw_dev->srb[request_id];
1412 struct pci_dev *pdev = tw_dev->tw_pci_dev;
1413 dma_addr_t retval = 0;
1414
1415 if (cmd->request_bufflen == 0) {
1416 retval = 0;
1417 goto out;
1418 }
1419
1420 mapping = pci_map_single(pdev, cmd->request_buffer, cmd->request_bufflen, DMA_BIDIRECTIONAL);
1421
1422 if (mapping == 0) {
1423 TW_PRINTK(tw_dev->host, TW_DRIVER, 0x1d, "Failed to map page");
1424 goto out;
1425 }
1426 1402
1427 cmd->SCp.phase = TW_PHASE_SINGLE; 1403 return use_sg;
1428 cmd->SCp.have_data_in = mapping; 1404} /* End twa_map_scsi_sg_data() */
1429 retval = mapping;
1430out:
1431 return retval;
1432} /* End twa_map_scsi_single_data() */
1433 1405
1434/* This function will poll for a response interrupt of a request */ 1406/* This function will poll for a response interrupt of a request */
1435static int twa_poll_response(TW_Device_Extension *tw_dev, int request_id, int seconds) 1407static int twa_poll_response(TW_Device_Extension *tw_dev, int request_id, int seconds)
@@ -1815,15 +1787,13 @@ static int twa_scsiop_execute_scsi(TW_Device_Extension *tw_dev, int request_id,
1815 u32 num_sectors = 0x0; 1787 u32 num_sectors = 0x0;
1816 int i, sg_count; 1788 int i, sg_count;
1817 struct scsi_cmnd *srb = NULL; 1789 struct scsi_cmnd *srb = NULL;
1818 struct scatterlist *sglist = NULL; 1790 struct scatterlist *sglist = NULL, *sg;
1819 dma_addr_t buffaddr = 0x0;
1820 int retval = 1; 1791 int retval = 1;
1821 1792
1822 if (tw_dev->srb[request_id]) { 1793 if (tw_dev->srb[request_id]) {
1823 if (tw_dev->srb[request_id]->request_buffer) {
1824 sglist = (struct scatterlist *)tw_dev->srb[request_id]->request_buffer;
1825 }
1826 srb = tw_dev->srb[request_id]; 1794 srb = tw_dev->srb[request_id];
1795 if (scsi_sglist(srb))
1796 sglist = scsi_sglist(srb);
1827 } 1797 }
1828 1798
1829 /* Initialize command packet */ 1799 /* Initialize command packet */
@@ -1856,32 +1826,12 @@ static int twa_scsiop_execute_scsi(TW_Device_Extension *tw_dev, int request_id,
1856 1826
1857 if (!sglistarg) { 1827 if (!sglistarg) {
1858 /* Map sglist from scsi layer to cmd packet */ 1828 /* Map sglist from scsi layer to cmd packet */
1859 if (tw_dev->srb[request_id]->use_sg == 0) {
1860 if (tw_dev->srb[request_id]->request_bufflen < TW_MIN_SGL_LENGTH) {
1861 command_packet->sg_list[0].address = TW_CPU_TO_SGL(tw_dev->generic_buffer_phys[request_id]);
1862 command_packet->sg_list[0].length = cpu_to_le32(TW_MIN_SGL_LENGTH);
1863 if (tw_dev->srb[request_id]->sc_data_direction == DMA_TO_DEVICE || tw_dev->srb[request_id]->sc_data_direction == DMA_BIDIRECTIONAL)
1864 memcpy(tw_dev->generic_buffer_virt[request_id], tw_dev->srb[request_id]->request_buffer, tw_dev->srb[request_id]->request_bufflen);
1865 } else {
1866 buffaddr = twa_map_scsi_single_data(tw_dev, request_id);
1867 if (buffaddr == 0)
1868 goto out;
1869
1870 command_packet->sg_list[0].address = TW_CPU_TO_SGL(buffaddr);
1871 command_packet->sg_list[0].length = cpu_to_le32(tw_dev->srb[request_id]->request_bufflen);
1872 }
1873 command_packet->sgl_entries__lunh = cpu_to_le16(TW_REQ_LUN_IN((srb->device->lun >> 4), 1));
1874 1829
1875 if (command_packet->sg_list[0].address & TW_CPU_TO_SGL(TW_ALIGNMENT_9000_SGL)) { 1830 if (scsi_sg_count(srb)) {
1876 TW_PRINTK(tw_dev->host, TW_DRIVER, 0x2d, "Found unaligned address during execute scsi"); 1831 if ((scsi_sg_count(srb) == 1) &&
1877 goto out; 1832 (scsi_bufflen(srb) < TW_MIN_SGL_LENGTH)) {
1878 } 1833 if (srb->sc_data_direction == DMA_TO_DEVICE || srb->sc_data_direction == DMA_BIDIRECTIONAL) {
1879 } 1834 struct scatterlist *sg = scsi_sglist(srb);
1880
1881 if (tw_dev->srb[request_id]->use_sg > 0) {
1882 if ((tw_dev->srb[request_id]->use_sg == 1) && (tw_dev->srb[request_id]->request_bufflen < TW_MIN_SGL_LENGTH)) {
1883 if (tw_dev->srb[request_id]->sc_data_direction == DMA_TO_DEVICE || tw_dev->srb[request_id]->sc_data_direction == DMA_BIDIRECTIONAL) {
1884 struct scatterlist *sg = (struct scatterlist *)tw_dev->srb[request_id]->request_buffer;
1885 char *buf = kmap_atomic(sg->page, KM_IRQ0) + sg->offset; 1835 char *buf = kmap_atomic(sg->page, KM_IRQ0) + sg->offset;
1886 memcpy(tw_dev->generic_buffer_virt[request_id], buf, sg->length); 1836 memcpy(tw_dev->generic_buffer_virt[request_id], buf, sg->length);
1887 kunmap_atomic(buf - sg->offset, KM_IRQ0); 1837 kunmap_atomic(buf - sg->offset, KM_IRQ0);
@@ -1893,16 +1843,16 @@ static int twa_scsiop_execute_scsi(TW_Device_Extension *tw_dev, int request_id,
1893 if (sg_count == 0) 1843 if (sg_count == 0)
1894 goto out; 1844 goto out;
1895 1845
1896 for (i = 0; i < sg_count; i++) { 1846 scsi_for_each_sg(srb, sg, sg_count, i) {
1897 command_packet->sg_list[i].address = TW_CPU_TO_SGL(sg_dma_address(&sglist[i])); 1847 command_packet->sg_list[i].address = TW_CPU_TO_SGL(sg_dma_address(sg));
1898 command_packet->sg_list[i].length = cpu_to_le32(sg_dma_len(&sglist[i])); 1848 command_packet->sg_list[i].length = cpu_to_le32(sg_dma_len(sg));
1899 if (command_packet->sg_list[i].address & TW_CPU_TO_SGL(TW_ALIGNMENT_9000_SGL)) { 1849 if (command_packet->sg_list[i].address & TW_CPU_TO_SGL(TW_ALIGNMENT_9000_SGL)) {
1900 TW_PRINTK(tw_dev->host, TW_DRIVER, 0x2e, "Found unaligned sgl address during execute scsi"); 1850 TW_PRINTK(tw_dev->host, TW_DRIVER, 0x2e, "Found unaligned sgl address during execute scsi");
1901 goto out; 1851 goto out;
1902 } 1852 }
1903 } 1853 }
1904 } 1854 }
1905 command_packet->sgl_entries__lunh = cpu_to_le16(TW_REQ_LUN_IN((srb->device->lun >> 4), tw_dev->srb[request_id]->use_sg)); 1855 command_packet->sgl_entries__lunh = cpu_to_le16(TW_REQ_LUN_IN((srb->device->lun >> 4), scsi_sg_count(tw_dev->srb[request_id])));
1906 } 1856 }
1907 } else { 1857 } else {
1908 /* Internal cdb post */ 1858 /* Internal cdb post */
@@ -1932,7 +1882,7 @@ static int twa_scsiop_execute_scsi(TW_Device_Extension *tw_dev, int request_id,
1932 1882
1933 /* Update SG statistics */ 1883 /* Update SG statistics */
1934 if (srb) { 1884 if (srb) {
1935 tw_dev->sgl_entries = tw_dev->srb[request_id]->use_sg; 1885 tw_dev->sgl_entries = scsi_sg_count(tw_dev->srb[request_id]);
1936 if (tw_dev->sgl_entries > tw_dev->max_sgl_entries) 1886 if (tw_dev->sgl_entries > tw_dev->max_sgl_entries)
1937 tw_dev->max_sgl_entries = tw_dev->sgl_entries; 1887 tw_dev->max_sgl_entries = tw_dev->sgl_entries;
1938 } 1888 }
@@ -1951,16 +1901,13 @@ out:
1951/* This function completes an execute scsi operation */ 1901/* This function completes an execute scsi operation */
1952static void twa_scsiop_execute_scsi_complete(TW_Device_Extension *tw_dev, int request_id) 1902static void twa_scsiop_execute_scsi_complete(TW_Device_Extension *tw_dev, int request_id)
1953{ 1903{
1954 if (tw_dev->srb[request_id]->request_bufflen < TW_MIN_SGL_LENGTH && 1904 struct scsi_cmnd *cmd = tw_dev->srb[request_id];
1955 (tw_dev->srb[request_id]->sc_data_direction == DMA_FROM_DEVICE || 1905
1956 tw_dev->srb[request_id]->sc_data_direction == DMA_BIDIRECTIONAL)) { 1906 if (scsi_bufflen(cmd) < TW_MIN_SGL_LENGTH &&
1957 if (tw_dev->srb[request_id]->use_sg == 0) { 1907 (cmd->sc_data_direction == DMA_FROM_DEVICE ||
1958 memcpy(tw_dev->srb[request_id]->request_buffer, 1908 cmd->sc_data_direction == DMA_BIDIRECTIONAL)) {
1959 tw_dev->generic_buffer_virt[request_id], 1909 if (scsi_sg_count(cmd) == 1) {
1960 tw_dev->srb[request_id]->request_bufflen); 1910 struct scatterlist *sg = scsi_sglist(tw_dev->srb[request_id]);
1961 }
1962 if (tw_dev->srb[request_id]->use_sg == 1) {
1963 struct scatterlist *sg = (struct scatterlist *)tw_dev->srb[request_id]->request_buffer;
1964 char *buf; 1911 char *buf;
1965 unsigned long flags = 0; 1912 unsigned long flags = 0;
1966 local_irq_save(flags); 1913 local_irq_save(flags);
@@ -2017,16 +1964,8 @@ static char *twa_string_lookup(twa_message_type *table, unsigned int code)
2017static void twa_unmap_scsi_data(TW_Device_Extension *tw_dev, int request_id) 1964static void twa_unmap_scsi_data(TW_Device_Extension *tw_dev, int request_id)
2018{ 1965{
2019 struct scsi_cmnd *cmd = tw_dev->srb[request_id]; 1966 struct scsi_cmnd *cmd = tw_dev->srb[request_id];
2020 struct pci_dev *pdev = tw_dev->tw_pci_dev;
2021 1967
2022 switch(cmd->SCp.phase) { 1968 scsi_dma_unmap(cmd);
2023 case TW_PHASE_SINGLE:
2024 pci_unmap_single(pdev, cmd->SCp.have_data_in, cmd->request_bufflen, DMA_BIDIRECTIONAL);
2025 break;
2026 case TW_PHASE_SGLIST:
2027 pci_unmap_sg(pdev, cmd->request_buffer, cmd->use_sg, DMA_BIDIRECTIONAL);
2028 break;
2029 }
2030} /* End twa_unmap_scsi_data() */ 1969} /* End twa_unmap_scsi_data() */
2031 1970
2032/* scsi_host_template initializer */ 1971/* scsi_host_template initializer */