* [PATCH 5/5] arcmsr: Modify ARC-1214 Inband Messages Behavior
@ 2012-12-04 12:02 NickCheng
0 siblings, 0 replies; 4+ messages in thread
From: NickCheng @ 2012-12-04 12:02 UTC (permalink / raw)
To: linux-scsi; +Cc: linux-kernel, jejb
[-- Attachment #1: Type: text/plain, Size: 179 bytes --]
From: Nick Cheng <nick.cheng@areca.com.tw>
Modify ARC-1214 inband messages behavior to make up for HW seldom malfunction.
Signed-off-by: Nick Cheng <nick.cheng@areca.com.tw>
---
[-- Attachment #2: patch5 --]
[-- Type: application/octet-stream, Size: 46846 bytes --]
diff -uprN a/drivers/scsi/arcmsr/arcmsr_attr.c b/drivers/scsi/arcmsr/arcmsr_attr.c
--- a/drivers/scsi/arcmsr/arcmsr_attr.c 2012-12-04 19:21:52.495082862 +0800
+++ b/drivers/scsi/arcmsr/arcmsr_attr.c 2012-12-04 19:21:58.767084375 +0800
@@ -72,40 +72,82 @@ arcmsr_sysfs_iop_message_read(struct fil
(struct AdapterControlBlock *)host->hostdata;
uint8_t *pQbuffer,*ptmpQbuffer;
int32_t allxfer_len = 0;
+ unsigned long flags;
if (!capable(CAP_SYS_ADMIN))
return -EACCES;
/* do message unit read. */
ptmpQbuffer = (uint8_t *)buf;
- while ((acb->rqbuf_firstindex != acb->rqbuf_lastindex)
- && (allxfer_len < 1031)) {
+ spin_lock_irqsave(&acb->rqbuffer_lock, flags);
+ if (acb->rqbuf_firstindex != acb->rqbuf_lastindex) {
pQbuffer = &acb->rqbuffer[acb->rqbuf_firstindex];
- memcpy(ptmpQbuffer, pQbuffer, 1);
- acb->rqbuf_firstindex++;
- acb->rqbuf_firstindex %= ARCMSR_MAX_QBUFFER;
- ptmpQbuffer++;
- allxfer_len++;
+ if (acb->rqbuf_firstindex > acb->rqbuf_lastindex) {
+ if ((ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex) >= 1032) {
+ memcpy(ptmpQbuffer, pQbuffer, 1032);
+ acb->rqbuf_firstindex += 1032;
+ allxfer_len = 1032;
+ } else {
+ if (((ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex) + acb->rqbuf_lastindex) > 1032) {
+ memcpy(ptmpQbuffer, pQbuffer, ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex);
+ ptmpQbuffer += ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex;
+ memcpy(ptmpQbuffer, acb->rqbuffer, 1032 - (ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex));
+ acb->rqbuf_firstindex = 1032 - (ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex);
+ allxfer_len = 1032;
+ } else {
+ memcpy(ptmpQbuffer, pQbuffer, ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex);
+ ptmpQbuffer += ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex;
+ memcpy(ptmpQbuffer, acb->rqbuffer, acb->rqbuf_lastindex);
+ allxfer_len = ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex + acb->rqbuf_lastindex;
+ acb->rqbuf_firstindex = acb->rqbuf_lastindex;
+ }
+ }
+ } else {
+ if ((acb->rqbuf_lastindex - acb->rqbuf_firstindex) > 1032) {
+ memcpy(ptmpQbuffer, pQbuffer, 1032);
+ acb->rqbuf_firstindex += 1032;
+ allxfer_len = 1032;
+ } else {
+ memcpy(ptmpQbuffer, pQbuffer, acb->rqbuf_lastindex - acb->rqbuf_firstindex);
+ allxfer_len = acb->rqbuf_lastindex - acb->rqbuf_firstindex;
+ acb->rqbuf_firstindex = acb->rqbuf_lastindex;
+ }
+ }
}
if (acb->acb_flags & ACB_F_IOPDATA_OVERFLOW) {
struct QBUFFER __iomem *prbuffer;
- uint8_t __iomem *iop_data;
- int32_t iop_len;
-
- acb->acb_flags &= ~ACB_F_IOPDATA_OVERFLOW;
+ uint8_t __iomem *iop_data, *vaddr, *temp;
+ int32_t data_len_residual, data_len, rqbuf_lastindex;
+ acb->acb_flags &= ~ACB_F_IOPDATA_OVERFLOW;
+ rqbuf_lastindex = acb->rqbuf_lastindex;
prbuffer = arcmsr_get_iop_rqbuffer(acb);
- iop_data = prbuffer->data;
- iop_len = readl(&prbuffer->data_len);
- while (iop_len > 0) {
- acb->rqbuffer[acb->rqbuf_lastindex] =
- readb(iop_data);
- acb->rqbuf_lastindex++;
- acb->rqbuf_lastindex %= ARCMSR_MAX_QBUFFER;
- iop_data++;
- iop_len--;
+ iop_data = (uint8_t __iomem *)prbuffer->data;
+ data_len_residual = data_len = readl(&prbuffer->data_len);
+ if (data_len > 0) {
+ temp = vaddr = kmalloc(data_len, GFP_ATOMIC);
+ do {
+ memcpy(temp, iop_data, 4);
+ temp += 4;
+ iop_data += 4;
+ data_len_residual -= 4;
+ } while (data_len_residual > 0);
+ pQbuffer = &acb->rqbuffer[acb->rqbuf_lastindex];
+ temp = vaddr;
+ if ((rqbuf_lastindex + data_len) > ARCMSR_MAX_QBUFFER) {
+ memcpy(pQbuffer, temp, ARCMSR_MAX_QBUFFER - rqbuf_lastindex);
+ temp += (ARCMSR_MAX_QBUFFER - rqbuf_lastindex);
+ rqbuf_lastindex = (rqbuf_lastindex + data_len) % ARCMSR_MAX_QBUFFER;
+ memcpy(&acb->rqbuffer[0], temp, rqbuf_lastindex);
+ } else {
+ memcpy(pQbuffer, temp, data_len);
+ rqbuf_lastindex = (rqbuf_lastindex + data_len) % ARCMSR_MAX_QBUFFER;
+ }
+ kfree(vaddr);
}
+ acb->rqbuf_lastindex = rqbuf_lastindex;
arcmsr_iop_message_read(acb);
}
+ spin_unlock_irqrestore(&acb->rqbuffer_lock, flags);
return (allxfer_len);
}
@@ -122,6 +164,7 @@ arcmsr_sysfs_iop_message_write(struct fi
(struct AdapterControlBlock *)host->hostdata;
int32_t my_empty_len, user_len, wqbuf_firstindex, wqbuf_lastindex;
uint8_t *pQbuffer, *ptmpuserbuffer;
+ unsigned long flags;
if (!capable(CAP_SYS_ADMIN))
return -EACCES;
@@ -130,10 +173,12 @@ arcmsr_sysfs_iop_message_write(struct fi
/* do message unit write. */
ptmpuserbuffer = (uint8_t *)buf;
user_len = (int32_t)count;
+ spin_lock_irqsave(&acb->wqbuffer_lock, flags);
wqbuf_lastindex = acb->wqbuf_lastindex;
wqbuf_firstindex = acb->wqbuf_firstindex;
if (wqbuf_lastindex != wqbuf_firstindex) {
arcmsr_post_ioctldata2iop(acb);
+ spin_unlock_irqrestore(&acb->wqbuffer_lock, flags);
return 0; /*need retry*/
} else {
my_empty_len = (wqbuf_firstindex-wqbuf_lastindex - 1)
@@ -155,8 +200,10 @@ arcmsr_sysfs_iop_message_write(struct fi
~ACB_F_MESSAGE_WQBUFFER_CLEARED;
arcmsr_post_ioctldata2iop(acb);
}
+ spin_unlock_irqrestore(&acb->wqbuffer_lock, flags);
return count;
} else {
+ spin_unlock_irqrestore(&acb->wqbuffer_lock, flags);
return 0; /*need retry*/
}
}
@@ -174,6 +221,7 @@ arcmsr_sysfs_iop_message_clear(struct fi
struct AdapterControlBlock *acb =
(struct AdapterControlBlock *)host->hostdata;
uint8_t *pQbuffer;
+ unsigned long flags;
if (!capable(CAP_SYS_ADMIN))
return -EACCES;
@@ -186,10 +234,14 @@ arcmsr_sysfs_iop_message_clear(struct fi
(ACB_F_MESSAGE_WQBUFFER_CLEARED
| ACB_F_MESSAGE_RQBUFFER_CLEARED
| ACB_F_MESSAGE_WQBUFFER_READED);
+ spin_lock_irqsave(&acb->rqbuffer_lock, flags);
acb->rqbuf_firstindex = 0;
acb->rqbuf_lastindex = 0;
+ spin_unlock_irqrestore(&acb->rqbuffer_lock, flags);
+ spin_lock_irqsave(&acb->wqbuffer_lock, flags);
acb->wqbuf_firstindex = 0;
acb->wqbuf_lastindex = 0;
+ spin_unlock_irqrestore(&acb->wqbuffer_lock, flags);
pQbuffer = acb->rqbuffer;
memset(pQbuffer, 0, sizeof (struct QBUFFER));
pQbuffer = acb->wqbuffer;
diff -uprN a/drivers/scsi/arcmsr/arcmsr.h b/drivers/scsi/arcmsr/arcmsr.h
--- a/drivers/scsi/arcmsr/arcmsr.h 2012-12-04 19:21:52.495082862 +0800
+++ b/drivers/scsi/arcmsr/arcmsr.h 2012-12-04 19:21:58.767084375 +0800
@@ -63,6 +63,7 @@ struct device_attribute;
#define ARCMSR_DEFAULT_SG_ENTRIES 38
#define ARCMSR_MAX_HBB_POSTQUEUE 264
#define ARCMSR_MAX_ARC1214_POSTQUEUE 256
+#define ARCMSR_MAX_ARC1214_DONEQUEUE 257
#define ARCMSR_MAX_XFER_LEN 0x26000
#define ARCMSR_CDB_SG_PAGE_LENGTH 256
#define ARCMST_NUM_MSIX_VECTORS 4
@@ -564,7 +565,7 @@ struct OutBound_SRB {
struct MessageUnit_D {
struct InBound_SRB post_qbuffer[ARCMSR_MAX_ARC1214_POSTQUEUE];
- struct OutBound_SRB done_qbuffer[ARCMSR_MAX_ARC1214_POSTQUEUE];
+ struct OutBound_SRB done_qbuffer[ARCMSR_MAX_ARC1214_DONEQUEUE];
u16 postq_index;
u16 doneq_index;
u32 __iomem *chip_id; /*0x00004*/
@@ -617,6 +618,7 @@ struct AdapterControlBlock
spinlock_t eh_lock;
spinlock_t ccblist_lock;
spinlock_t postq_lock;
+ spinlock_t doneq_lock;
spinlock_t rqbuffer_lock;
spinlock_t wqbuffer_lock;
union {
@@ -670,15 +672,15 @@ struct AdapterControlBlock
unsigned int uncache_size;
uint8_t rqbuffer[ARCMSR_MAX_QBUFFER];
/* data collection buffer for read from 80331 */
- int32_t rqbuf_firstindex;
+ uint32_t rqbuf_firstindex;
/* first of read buffer */
- int32_t rqbuf_lastindex;
+ uint32_t rqbuf_lastindex;
/* last of read buffer */
uint8_t wqbuffer[ARCMSR_MAX_QBUFFER];
/* data collection buffer for write to 80331 */
- int32_t wqbuf_firstindex;
+ uint32_t wqbuf_firstindex;
/* first of write buffer */
- int32_t wqbuf_lastindex;
+ uint32_t wqbuf_lastindex;
/* last of write buffer */
uint8_t devstate[ARCMSR_MAX_TARGETID][ARCMSR_MAX_TARGETLUN];
/* id0 ..... id15, lun0...lun7 */
diff -uprN a/drivers/scsi/arcmsr/arcmsr_hba.c b/drivers/scsi/arcmsr/arcmsr_hba.c
--- a/drivers/scsi/arcmsr/arcmsr_hba.c 2012-12-04 19:21:52.499082863 +0800
+++ b/drivers/scsi/arcmsr/arcmsr_hba.c 2012-12-04 19:21:58.771084153 +0800
@@ -768,18 +768,18 @@ arcmsr_message_isr_bh_fn(struct work_str
(diff & 0x01) == 1) {
scsi_add_device(acb->host,
0, target, lun);
- } else if ((temp & 0x01) == 0
- && (diff & 0x01) == 1) {
- psdev =
- scsi_device_lookup(acb->host,
- 0, target, lun);
- if (psdev != NULL) {
- scsi_remove_device(psdev);
- scsi_device_put(psdev);
+ } else if ((temp & 0x01) == 0
+ && (diff & 0x01) == 1) {
+ psdev =
+ scsi_device_lookup(acb->host,
+ 0, target, lun);
+ if (psdev != NULL) {
+ scsi_remove_device(psdev);
+ scsi_device_put(psdev);
+ }
}
- }
- temp >>= 1;
- diff >>= 1;
+ temp >>= 1;
+ diff >>= 1;
}
}
devicemap++;
@@ -813,20 +813,20 @@ arcmsr_message_isr_bh_fn(struct work_str
lun < ARCMSR_MAX_TARGETLUN;
lun++) {
if ((temp & 0x01) == 1 &&
- (diff & 0x01) == 1) {
+ (diff & 0x01) == 1) {
scsi_add_device(acb->host,
0, target, lun);
} else if ((temp & 0x01) == 0
- && (diff & 0x01) == 1) {
+ && (diff & 0x01) == 1) {
psdev = scsi_device_lookup(acb->host,
0, target, lun);
if (psdev != NULL) {
scsi_remove_device(psdev);
scsi_device_put(psdev);
}
- }
- temp >>= 1;
- diff >>= 1;
+ }
+ temp >>= 1;
+ diff >>= 1;
}
}
devicemap++;
@@ -869,10 +869,10 @@ arcmsr_message_isr_bh_fn(struct work_str
if (psdev != NULL) {
scsi_remove_device(psdev);
scsi_device_put(psdev);
+ }
}
- }
- temp >>= 1;
- diff >>= 1;
+ temp >>= 1;
+ diff >>= 1;
}
}
devicemap++;
@@ -915,10 +915,10 @@ arcmsr_message_isr_bh_fn(struct work_str
if (psdev != NULL) {
scsi_remove_device(psdev);
scsi_device_put(psdev);
+ }
}
- }
- temp >>= 1;
- diff >>= 1;
+ temp >>= 1;
+ diff >>= 1;
}
}
devicemap++;
@@ -981,8 +981,8 @@ arcmsr_resume(struct pci_dev *pdev)
error = pci_set_dma_mask(pdev, DMA_BIT_MASK(32));
if (error) {
pr_warn("scsi%d: No suitable DMA mask available\n",
- host->host_no);
- goto controller_unregister;
+ host->host_no);
+ goto controller_unregister;
}
}
pci_set_master(pdev);
@@ -1097,6 +1097,7 @@ static int arcmsr_probe(struct pci_dev *
spin_lock_init(&acb->eh_lock);
spin_lock_init(&acb->ccblist_lock);
spin_lock_init(&acb->postq_lock);
+ spin_lock_init(&acb->doneq_lock);
spin_lock_init(&acb->rqbuffer_lock);
spin_lock_init(&acb->wqbuffer_lock);
acb->acb_flags |= (ACB_F_MESSAGE_WQBUFFER_CLEARED |
@@ -1541,33 +1542,29 @@ arcmsr_done4abort_postqueue(struct Adapt
uint32_t doneq_index, index_stripped, addressLow, residual;
bool error;
struct CommandControlBlock *pCCB;
- outbound_write_pointer =
- readl(pmu->outboundlist_copy_pointer);
+ outbound_write_pointer = pmu->done_qbuffer[0].addressLow + 1;
doneq_index = pmu->doneq_index;
residual = atomic_read(&acb->ccboutstandingcount);
for (i = 0; i < residual; i++) {
- while ((doneq_index & 0xFF) !=
- (outbound_write_pointer & 0xFF)) {
+ while ((doneq_index & 0xFFF) != (outbound_write_pointer & 0xFFF)) {
if (doneq_index & 0x4000) {
- index_stripped = doneq_index & 0xFF;
+ index_stripped = doneq_index & 0xFFF;
index_stripped += 1;
index_stripped %=
- ARCMSR_MAX_ARC1214_POSTQUEUE;
+ ARCMSR_MAX_ARC1214_DONEQUEUE;
pmu->doneq_index = index_stripped ?
- (index_stripped | 0x4000) :
- index_stripped;
+ (index_stripped | 0x4000) : (index_stripped + 1);
} else {
index_stripped = doneq_index;
index_stripped += 1;
index_stripped %=
- ARCMSR_MAX_ARC1214_POSTQUEUE;
+ ARCMSR_MAX_ARC1214_DONEQUEUE;
pmu->doneq_index =
- index_stripped ? index_stripped :
- (index_stripped | 0x4000);
+ index_stripped ? index_stripped : ((index_stripped | 0x4000) + 1);
}
doneq_index = pmu->doneq_index;
addressLow =
- pmu->done_qbuffer[doneq_index & 0xFF].addressLow;
+ pmu->done_qbuffer[doneq_index & 0xFFF].addressLow;
ccb_cdb_phy = (addressLow & 0xFFFFFFF0);
pARCMSR_CDB = (struct ARCMSR_CDB *)
(acb->vir2phy_offset + ccb_cdb_phy);
@@ -1582,7 +1579,7 @@ arcmsr_done4abort_postqueue(struct Adapt
}
mdelay(10);
outbound_write_pointer =
- readl(pmu->outboundlist_copy_pointer);
+ pmu->done_qbuffer[0].addressLow + 1;
doneq_index = pmu->doneq_index;
}
pmu->postq_index = 0;
@@ -2004,12 +2001,14 @@ arcmsr_iop_message_read(struct AdapterCo
case ACB_ADAPTER_TYPE_C: {
struct MessageUnit_C __iomem *reg = acb->pmuC;
writel(ARCMSR_HBCMU_DRV2IOP_DATA_READ_OK, ®->inbound_doorbell);
+ /*readl(®->inbound_doorbell);*/
}
break;
case ACB_ADAPTER_TYPE_D: {
struct MessageUnit_D __iomem *reg = acb->pmuD;
writel(ARCMSR_ARC1214_DRV2IOP_DATA_OUT_READ,
reg->inbound_doorbell);
+ /*readl(reg->inbound_doorbell);*/
break;
}
}
@@ -2127,36 +2126,61 @@ struct QBUFFER __iomem
void
arcmsr_iop2drv_data_wrote_handle(struct AdapterControlBlock *acb)
{
- uint8_t __iomem *iop_data;
- struct QBUFFER __iomem *prbuffer;
- struct QBUFFER *pQbuffer;
- int32_t my_empty_len, iop_len, rqbuf_firstindex, rqbuf_lastindex;
+ uint8_t __iomem *iop_data, *pQbuffer, *vaddr, *temp;
+ int32_t buf_empty_len, data_len, data_len_residual;
+ uint32_t rqbuf_firstindex, rqbuf_lastindex;
unsigned long flags;
-
+ struct QBUFFER __iomem *prbuffer;
spin_lock_irqsave(&acb->rqbuffer_lock, flags);
rqbuf_lastindex = acb->rqbuf_lastindex;
rqbuf_firstindex = acb->rqbuf_firstindex;
prbuffer = arcmsr_get_iop_rqbuffer(acb);
iop_data = (uint8_t __iomem *)prbuffer->data;
- iop_len = prbuffer->data_len;
- my_empty_len = (rqbuf_firstindex - rqbuf_lastindex - 1) &
- (ARCMSR_MAX_QBUFFER - 1);
-
- if (my_empty_len >= iop_len) {
- while (iop_len > 0) {
- pQbuffer = (struct QBUFFER *)
- &acb->rqbuffer[rqbuf_lastindex];
- memcpy(pQbuffer, iop_data, 1);
- rqbuf_lastindex++;
- rqbuf_lastindex %= ARCMSR_MAX_QBUFFER;
- iop_data++;
- iop_len--;
+ data_len_residual = data_len = readl(&prbuffer->data_len);
+ buf_empty_len = (rqbuf_firstindex - rqbuf_lastindex - 1) &
+ (ARCMSR_MAX_QBUFFER - 1);
+ if (buf_empty_len >= data_len) {
+ if (data_len > 0) {
+ temp = vaddr = kmalloc(data_len, GFP_ATOMIC);
+ if (!vaddr) {
+ goto leave;
+ }
+ acb->acb_flags &= ~ACB_F_IOPDATA_OVERFLOW;
+ while (data_len_residual >= 4) {
+ memcpy(temp, iop_data, 4);
+ temp += 4;
+ iop_data += 4;
+ data_len_residual -= 4;
+ }
+ if ((data_len_residual > 0) &&
+ (data_len_residual < 4)) {
+ memcpy(temp, iop_data, data_len_residual);
+ }
+ pQbuffer = &acb->rqbuffer[rqbuf_lastindex];
+ temp = vaddr;
+ if ((rqbuf_lastindex + data_len) >
+ ARCMSR_MAX_QBUFFER) {
+ memcpy(pQbuffer, temp,
+ ARCMSR_MAX_QBUFFER - rqbuf_lastindex);
+ temp += (ARCMSR_MAX_QBUFFER -
+ rqbuf_lastindex);
+ rqbuf_lastindex = (rqbuf_lastindex + data_len)
+ % ARCMSR_MAX_QBUFFER;
+ memcpy(&acb->rqbuffer[0], temp,
+ rqbuf_lastindex);
+ } else {
+ memcpy(pQbuffer, temp, data_len);
+ rqbuf_lastindex = (rqbuf_lastindex + data_len)
+ % ARCMSR_MAX_QBUFFER;
+ }
+ kfree(vaddr);
+ acb->rqbuf_lastindex = rqbuf_lastindex;
+ arcmsr_iop_message_read(acb);
}
- acb->rqbuf_lastindex = rqbuf_lastindex;
- arcmsr_iop_message_read(acb);
} else {
acb->acb_flags |= ACB_F_IOPDATA_OVERFLOW;
}
+ leave:
spin_unlock_irqrestore(&acb->rqbuffer_lock, flags);
}
@@ -2164,7 +2188,6 @@ void
arcmsr_iop2drv_data_read_handle(struct AdapterControlBlock *acb)
{
unsigned long flags;
-
spin_lock_irqsave(&acb->wqbuffer_lock, flags);
acb->acb_flags |= ACB_F_MESSAGE_WQBUFFER_READED;
if (acb->wqbuf_firstindex != acb->wqbuf_lastindex) {
@@ -2176,18 +2199,113 @@ arcmsr_iop2drv_data_read_handle(struct A
acb->acb_flags &= (~ACB_F_MESSAGE_WQBUFFER_READED);
pwbuffer = arcmsr_get_iop_wqbuffer(acb);
iop_data = (uint8_t __iomem *)pwbuffer->data;
-
- while ((acb->wqbuf_firstindex != acb->wqbuf_lastindex) &&
- (allxfer_len < 124)) {
- pQbuffer = &acb->wqbuffer[acb->wqbuf_firstindex];
- memcpy(iop_data, pQbuffer, 1);
- acb->wqbuf_firstindex++;
- acb->wqbuf_firstindex %= ARCMSR_MAX_QBUFFER;
- iop_data++;
- allxfer_len++;
+ if (acb->wqbuf_firstindex > acb->wqbuf_lastindex) {
+ if ((ARCMSR_MAX_QBUFFER - acb->wqbuf_firstindex) >= 4) {
+ do {
+ pQbuffer =
+ &acb->wqbuffer[acb->wqbuf_firstindex];
+ if (acb->wqbuf_firstindex + 4
+ > ARCMSR_MAX_QBUFFER) {
+ memcpy(iop_data, pQbuffer,
+ ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_firstindex);
+ iop_data += ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex += 4;
+ acb->wqbuf_firstindex %=
+ ARCMSR_MAX_QBUFFER;
+ memcpy(iop_data,
+ &acb->wqbuffer[0],
+ acb->wqbuf_firstindex);
+ iop_data +=
+ acb->wqbuf_firstindex;
+ } else {
+ if ((acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex) > 4) {
+ memcpy(iop_data,
+ pQbuffer, 4);
+ acb->wqbuf_firstindex
+ += 4;
+ acb->wqbuf_firstindex
+ %= ARCMSR_MAX_QBUFFER;
+ iop_data += 4;
+ } else {
+ memcpy(iop_data, pQbuffer,
+ acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex);
+ allxfer_len +=
+ acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex =
+ acb->wqbuf_lastindex;
+ break;
+ }
+ }
+ allxfer_len += 4;
+ } while ((acb->wqbuf_firstindex !=
+ acb->wqbuf_lastindex) && (allxfer_len < 124));
+ } else {
+ pQbuffer =
+ &acb->wqbuffer[acb->wqbuf_firstindex];
+ memcpy(iop_data, pQbuffer, ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_firstindex);
+ iop_data += ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_firstindex;
+ allxfer_len = ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex = 0;
+ do {
+ pQbuffer =
+ &acb->wqbuffer[acb->wqbuf_firstindex];
+ if ((acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex) > 4) {
+ memcpy(iop_data, pQbuffer, 4);
+ acb->wqbuf_firstindex += 4;
+ acb->wqbuf_firstindex %=
+ ARCMSR_MAX_QBUFFER;
+ iop_data += 4;
+ } else {
+ memcpy(iop_data, pQbuffer,
+ acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex);
+ allxfer_len +=
+ acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex =
+ acb->wqbuf_lastindex;
+ break;
+ }
+ allxfer_len += 4;
+ } while ((acb->wqbuf_firstindex !=
+ acb->wqbuf_lastindex) && (allxfer_len < 124));
+ }
+ } else {
+ do {
+ pQbuffer =
+ &acb->wqbuffer[acb->wqbuf_firstindex];
+ if ((acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex) > 4) {
+ memcpy(iop_data, pQbuffer, 4);
+ acb->wqbuf_firstindex += 4;
+ acb->wqbuf_firstindex %=
+ ARCMSR_MAX_QBUFFER;
+ iop_data += 4;
+ } else {
+ memcpy(iop_data, pQbuffer,
+ acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex);
+ allxfer_len += acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex =
+ acb->wqbuf_lastindex;
+ break;
+ }
+ allxfer_len += 4;
+ } while ((acb->wqbuf_firstindex !=
+ acb->wqbuf_lastindex) && (allxfer_len < 124));
}
- pwbuffer->data_len = allxfer_len;
- arcmsr_iop_message_wrote(acb);
+ writel(allxfer_len, &pwbuffer->data_len);
+ arcmsr_iop_message_wrote(acb);/*notice IOP the message has been written*/
}
if (acb->wqbuf_firstindex == acb->wqbuf_lastindex) {
acb->acb_flags |= ACB_F_MESSAGE_WQBUFFER_CLEARED;
@@ -2200,7 +2318,6 @@ arcmsr_hbaA_doorbell_isr(struct AdapterC
{
uint32_t outbound_doorbell;
struct MessageUnit_A __iomem *reg = acb->pmuA;
-
outbound_doorbell = readl(®->outbound_doorbell);
do {
writel(outbound_doorbell, ®->outbound_doorbell);
@@ -2223,7 +2340,6 @@ arcmsr_hbaC_doorbell_isr(struct AdapterC
uint32_t outbound_doorbell;
struct MessageUnit_C __iomem *reg =
(struct MessageUnit_C *)pACB->pmuC;
-
outbound_doorbell = readl(®->outbound_doorbell);
do {
if (outbound_doorbell &
@@ -2255,10 +2371,6 @@ arcmsr_hbaD_doorbell_isr(struct AdapterC
(struct MessageUnit_D *)pACB->pmuD;
outbound_doorbell = readl(pmu->outbound_doorbell);
- if (unlikely(!outbound_doorbell)) {
- arcmsr_iop2drv_data_wrote_handle(pACB);
- arcmsr_iop2drv_data_read_handle(pACB);
- }
do {
writel(outbound_doorbell, pmu->outbound_doorbell);
if (outbound_doorbell &
@@ -2367,28 +2479,30 @@ arcmsr_hbaD_postqueue_isr(struct Adapter
struct MessageUnit_D __iomem *pmu;
struct ARCMSR_CDB *arcmsr_cdb;
struct CommandControlBlock *ccb;
+ unsigned long flags;
+ spin_lock_irqsave(&acb->doneq_lock, flags);
pmu = (struct MessageUnit_D *)acb->pmuD;
- outbound_write_pointer = readl(pmu->outboundlist_copy_pointer);
+ outbound_write_pointer = pmu->done_qbuffer[0].addressLow + 1;
doneq_index = pmu->doneq_index;
- if ((doneq_index & 0xFF) != (outbound_write_pointer & 0xFF)) {
+ if ((doneq_index & 0xFFF) != (outbound_write_pointer & 0xFFF)) {
do {
if (doneq_index & 0x4000) {
- index_stripped = doneq_index & 0xFF;
+ index_stripped = doneq_index & 0xFFF;
index_stripped += 1;
- index_stripped %= ARCMSR_MAX_ARC1214_POSTQUEUE;
+ index_stripped %= ARCMSR_MAX_ARC1214_DONEQUEUE;
pmu->doneq_index = index_stripped
- ? (index_stripped | 0x4000) : index_stripped;
+ ? (index_stripped | 0x4000) : (index_stripped + 1);
} else {
index_stripped = doneq_index;
index_stripped += 1;
- index_stripped %= ARCMSR_MAX_ARC1214_POSTQUEUE;
+ index_stripped %= ARCMSR_MAX_ARC1214_DONEQUEUE;
pmu->doneq_index = index_stripped
- ? index_stripped : (index_stripped | 0x4000);
+ ? index_stripped : ((index_stripped | 0x4000) + 1);
}
doneq_index = pmu->doneq_index;
addressLow =
- pmu->done_qbuffer[doneq_index & 0xFF].addressLow;
+ pmu->done_qbuffer[doneq_index & 0xFFF].addressLow;
ccb_cdb_phy = (addressLow & 0xFFFFFFF0);
arcmsr_cdb = (struct ARCMSR_CDB *)(acb->vir2phy_offset
+ ccb_cdb_phy);
@@ -2399,12 +2513,13 @@ arcmsr_hbaD_postqueue_isr(struct Adapter
arcmsr_drain_donequeue(acb, ccb, error);
writel(doneq_index,
pmu->outboundlist_read_pointer);
- } while ((doneq_index & 0xFF) !=
- (outbound_write_pointer & 0xFF));
+ } while ((doneq_index & 0xFFF) !=
+ (outbound_write_pointer & 0xFFF));
}
writel(ARCMSR_ARC1214_OUTBOUND_LIST_INTERRUPT_CLEAR,
pmu->outboundlist_interrupt_cause);
readl(pmu->outboundlist_interrupt_cause);
+ spin_unlock_irqrestore(&acb->doneq_lock, flags);
}
static void
@@ -2457,7 +2572,6 @@ arcmsr_hbaA_handle_isr(struct AdapterCon
}
do {
writel(outbound_intstatus, ®->outbound_intstatus);
- readl(®->outbound_intstatus);
if (outbound_intstatus & ARCMSR_MU_OUTBOUND_DOORBELL_INT)
arcmsr_hbaA_doorbell_isr(acb);
if (outbound_intstatus & ARCMSR_MU_OUTBOUND_POSTQUEUE_INT)
@@ -2597,29 +2711,131 @@ arcmsr_iop_parking(struct AdapterControl
void
arcmsr_post_ioctldata2iop(struct AdapterControlBlock *acb)
{
- uint8_t __iomem *iop_data;
- uint8_t *pQbuffer;
- int32_t wqbuf_firstindex, wqbuf_lastindex;
- int32_t allxfer_len = 0;
- struct QBUFFER __iomem *pwbuffer;
- pwbuffer = arcmsr_get_iop_wqbuffer(acb);
- iop_data = (uint8_t __iomem *)pwbuffer->data;
if (acb->acb_flags & ACB_F_MESSAGE_WQBUFFER_READED) {
- acb->acb_flags &= (~ACB_F_MESSAGE_WQBUFFER_READED);
- wqbuf_firstindex = acb->wqbuf_firstindex;
- wqbuf_lastindex = acb->wqbuf_lastindex;
- while ((wqbuf_firstindex != wqbuf_lastindex)
- && (allxfer_len < 124)) {
- pQbuffer = &acb->wqbuffer[wqbuf_firstindex];
- memcpy(iop_data, pQbuffer, 1);
- wqbuf_firstindex++;
- wqbuf_firstindex %= ARCMSR_MAX_QBUFFER;
- iop_data++;
- allxfer_len++;
- }
- acb->wqbuf_firstindex = wqbuf_firstindex;
- pwbuffer->data_len = allxfer_len;
- arcmsr_iop_message_wrote(acb);
+ if (acb->wqbuf_firstindex != acb->wqbuf_lastindex) {
+ uint8_t *pQbuffer;
+ uint8_t __iomem *iop_data;
+ int32_t allxfer_len = 0;
+ struct QBUFFER __iomem *pwbuffer;
+ acb->acb_flags &= (~ACB_F_MESSAGE_WQBUFFER_READED);
+ pwbuffer = arcmsr_get_iop_wqbuffer(acb);
+ iop_data = (uint8_t __iomem *)pwbuffer->data;
+ if (acb->wqbuf_firstindex > acb->wqbuf_lastindex) {
+ if ((ARCMSR_MAX_QBUFFER -
+ acb->wqbuf_firstindex) >= 4) {
+ do {
+ pQbuffer =
+ &acb->wqbuffer[acb->wqbuf_firstindex];
+ if (acb->wqbuf_firstindex + 4 >
+ ARCMSR_MAX_QBUFFER) {
+ memcpy(iop_data, pQbuffer,
+ ARCMSR_MAX_QBUFFER - acb->wqbuf_firstindex);
+ iop_data += ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex += 4;
+ acb->wqbuf_firstindex %=
+ ARCMSR_MAX_QBUFFER;
+ memcpy(iop_data,
+ &acb->wqbuffer[0],
+ acb->wqbuf_firstindex);
+ iop_data +=
+ acb->wqbuf_firstindex;
+ } else {
+ if ((acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex) > 4) {
+ memcpy(iop_data,
+ pQbuffer, 4);
+ acb->wqbuf_firstindex
+ += 4;
+ acb->wqbuf_firstindex
+ %= ARCMSR_MAX_QBUFFER;
+ iop_data += 4;
+ } else {
+ memcpy(iop_data,
+ pQbuffer,
+ acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex);
+ allxfer_len +=
+ acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex
+ = acb->wqbuf_lastindex;
+ break;
+ }
+ }
+ allxfer_len += 4;
+ } while ((acb->wqbuf_firstindex !=
+ acb->wqbuf_lastindex) &&
+ (allxfer_len < 124));
+ } else {
+ pQbuffer =
+ &acb->wqbuffer[acb->wqbuf_firstindex];
+ memcpy(iop_data, pQbuffer,
+ ARCMSR_MAX_QBUFFER -
+ acb->wqbuf_firstindex);
+ iop_data += ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_firstindex;
+ allxfer_len = ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex = 0;
+ do {
+ pQbuffer =
+ &acb->wqbuffer[acb->wqbuf_firstindex];
+ if ((acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex) > 4) {
+ memcpy(iop_data,
+ pQbuffer, 4);
+ acb->wqbuf_firstindex
+ += 4;
+ acb->wqbuf_firstindex
+ %= ARCMSR_MAX_QBUFFER;
+ iop_data += 4;
+ } else {
+ memcpy(iop_data, pQbuffer,
+ acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex);
+ allxfer_len +=
+ acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex =
+ acb->wqbuf_lastindex;
+ break;
+ }
+ allxfer_len += 4;
+ } while ((acb->wqbuf_firstindex
+ != acb->wqbuf_lastindex) &&
+ (allxfer_len < 124));
+ }
+ } else {
+ do {
+ pQbuffer =
+ &acb->wqbuffer[acb->wqbuf_firstindex];
+ if ((acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex) > 4) {
+ memcpy(iop_data,
+ pQbuffer, 4);
+ acb->wqbuf_firstindex
+ += 4;
+ acb->wqbuf_firstindex
+ %= ARCMSR_MAX_QBUFFER;
+ iop_data += 4;
+ } else {
+ memcpy(iop_data, pQbuffer,
+ acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex);
+ allxfer_len += acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex =
+ acb->wqbuf_lastindex;
+ break;
+ }
+ allxfer_len += 4;
+ } while ((acb->wqbuf_firstindex !=
+ acb->wqbuf_lastindex) && (allxfer_len < 124));
+ }
+ writel(allxfer_len, &pwbuffer->data_len);
+ arcmsr_iop_message_wrote(acb);
+ }
}
}
@@ -2627,74 +2843,155 @@ static int
arcmsr_iop_message_xfer(struct AdapterControlBlock *acb,
struct scsi_cmnd *cmd)
{
- struct CMD_MESSAGE_FIELD *pcmdmessagefld;
- int retvalue = 0, transfer_len = 0;
char *buffer;
+ unsigned short use_sg;
+ int retvalue = 0, transfer_len = 0;
+ unsigned long flags;
+ struct CMD_MESSAGE_FIELD *pcmdmessagefld;
+ uint32_t controlcode = (uint32_t)cmd->cmnd[5] << 24 |
+ (uint32_t)cmd->cmnd[6] << 16 |
+ (uint32_t)cmd->cmnd[7] << 8 |
+ (uint32_t)cmd->cmnd[8];
struct scatterlist *sg;
- uint32_t controlcode = (uint32_t)cmd->cmnd[5] << 24
- |(uint32_t)cmd->cmnd[6] << 16
- |(uint32_t)cmd->cmnd[7] << 8
- | (uint32_t)cmd->cmnd[8];
+
+ use_sg = scsi_sg_count(cmd);
sg = scsi_sglist(cmd);
buffer = kmap_atomic(sg_page(sg), KM_IRQ0) + sg->offset;
- if (scsi_sg_count(cmd) > 1) {
+ if (use_sg > 1) {
retvalue = ARCMSR_MESSAGE_FAIL;
goto message_out;
}
transfer_len += sg->length;
-
if (transfer_len > sizeof(struct CMD_MESSAGE_FIELD)) {
retvalue = ARCMSR_MESSAGE_FAIL;
+ printk("%s: ARCMSR_MESSAGE_FAIL!\n", __func__);
goto message_out;
}
pcmdmessagefld = (struct CMD_MESSAGE_FIELD *)buffer;
switch (controlcode) {
-
case ARCMSR_MESSAGE_READ_RQBUFFER: {
unsigned char *ver_addr;
uint8_t *pQbuffer, *ptmpQbuffer;
- int32_t allxfer_len = 0;
- unsigned long flags;
-
+ uint32_t allxfer_len = 0;
ver_addr = kmalloc(1032, GFP_ATOMIC);
if (!ver_addr) {
retvalue = ARCMSR_MESSAGE_FAIL;
+ printk("%s: memory not enough!\n", __func__);
goto message_out;
}
ptmpQbuffer = ver_addr;
spin_lock_irqsave(&acb->rqbuffer_lock, flags);
- while ((acb->rqbuf_firstindex != acb->rqbuf_lastindex)
- && (allxfer_len < 1031)) {
+ if (acb->rqbuf_firstindex != acb->rqbuf_lastindex) {
pQbuffer = &acb->rqbuffer[acb->rqbuf_firstindex];
- memcpy(ptmpQbuffer, pQbuffer, 1);
- acb->rqbuf_firstindex++;
- acb->rqbuf_firstindex %= ARCMSR_MAX_QBUFFER;
- ptmpQbuffer++;
- allxfer_len++;
+ if (acb->rqbuf_firstindex > acb->rqbuf_lastindex) {
+ if ((ARCMSR_MAX_QBUFFER -
+ acb->rqbuf_firstindex) >= 1032) {
+ memcpy(ptmpQbuffer, pQbuffer, 1032);
+ acb->rqbuf_firstindex += 1032;
+ allxfer_len = 1032;
+ } else {
+ if (((ARCMSR_MAX_QBUFFER -
+ acb->rqbuf_firstindex) +
+ acb->rqbuf_lastindex) > 1032) {
+ memcpy(ptmpQbuffer,
+ pQbuffer, ARCMSR_MAX_QBUFFER
+ - acb->rqbuf_firstindex);
+ ptmpQbuffer +=
+ ARCMSR_MAX_QBUFFER -
+ acb->rqbuf_firstindex;
+ memcpy(ptmpQbuffer,
+ acb->rqbuffer, 1032 -
+ (ARCMSR_MAX_QBUFFER
+ - acb->rqbuf_firstindex));
+ acb->rqbuf_firstindex =
+ 1032 - (ARCMSR_MAX_QBUFFER
+ - acb->rqbuf_firstindex);
+ allxfer_len = 1032;
+ } else {
+ memcpy(ptmpQbuffer,
+ pQbuffer, ARCMSR_MAX_QBUFFER
+ - acb->rqbuf_firstindex);
+ ptmpQbuffer +=
+ ARCMSR_MAX_QBUFFER -
+ acb->rqbuf_firstindex;
+ memcpy(ptmpQbuffer,
+ acb->rqbuffer,
+ acb->rqbuf_lastindex);
+ allxfer_len = ARCMSR_MAX_QBUFFER
+ - acb->rqbuf_firstindex +
+ acb->rqbuf_lastindex;
+ acb->rqbuf_firstindex =
+ acb->rqbuf_lastindex;
+ }
+ }
+ } else {
+ if ((acb->rqbuf_lastindex -
+ acb->rqbuf_firstindex) > 1032) {
+ memcpy(ptmpQbuffer, pQbuffer, 1032);
+ acb->rqbuf_firstindex += 1032;
+ allxfer_len = 1032;
+ } else {
+ memcpy(ptmpQbuffer, pQbuffer,
+ acb->rqbuf_lastindex - acb->rqbuf_firstindex);
+ allxfer_len = acb->rqbuf_lastindex
+ - acb->rqbuf_firstindex;
+ acb->rqbuf_firstindex =
+ acb->rqbuf_lastindex;
+ }
+ }
}
if (acb->acb_flags & ACB_F_IOPDATA_OVERFLOW) {
-
struct QBUFFER __iomem *prbuffer;
- uint8_t __iomem *iop_data;
- int32_t iop_len;
-
- acb->acb_flags &= ~ACB_F_IOPDATA_OVERFLOW;
+ uint8_t __iomem *iop_data, *vaddr, *temp;
+ uint32_t data_len_residual, data_len, rqbuf_lastindex;
+ rqbuf_lastindex = acb->rqbuf_lastindex;
prbuffer = arcmsr_get_iop_rqbuffer(acb);
- iop_data = prbuffer->data;
- iop_len = readl(&prbuffer->data_len);
- while (iop_len > 0) {
- acb->rqbuffer[acb->rqbuf_lastindex] =
- readb(iop_data);
- acb->rqbuf_lastindex++;
- acb->rqbuf_lastindex %= ARCMSR_MAX_QBUFFER;
- iop_data++;
- iop_len--;
+ iop_data = (uint8_t __iomem *)prbuffer->data;
+ data_len_residual = data_len = readl(&prbuffer->data_len);
+ if (data_len > 0) {
+ temp = vaddr = kmalloc(data_len, GFP_ATOMIC);
+ if (!vaddr) {
+ goto leave;
+ }
+ acb->acb_flags &= ~ACB_F_IOPDATA_OVERFLOW;
+ while (data_len_residual >= 4) {
+ memcpy(temp, iop_data, 4);
+ temp += 4;
+ iop_data += 4;
+ data_len_residual -= 4;
+ }
+ if ((data_len_residual > 0) &&
+ (data_len_residual < 4)) {
+ memcpy(temp, iop_data,
+ data_len_residual);
+ }
+ pQbuffer =
+ &acb->rqbuffer[acb->rqbuf_lastindex];
+ temp = vaddr;
+ if ((rqbuf_lastindex + data_len) >
+ ARCMSR_MAX_QBUFFER) {
+ memcpy(pQbuffer, temp,
+ ARCMSR_MAX_QBUFFER - rqbuf_lastindex);
+ temp += (ARCMSR_MAX_QBUFFER -
+ rqbuf_lastindex);
+ rqbuf_lastindex = (rqbuf_lastindex +
+ data_len) % ARCMSR_MAX_QBUFFER;
+ memcpy(&acb->rqbuffer[0],
+ temp, rqbuf_lastindex);
+ } else {
+ memcpy(pQbuffer, temp, data_len);
+ rqbuf_lastindex =
+ (rqbuf_lastindex + data_len) %
+ ARCMSR_MAX_QBUFFER;
+ }
+ kfree(vaddr);
+ acb->rqbuf_lastindex = rqbuf_lastindex;
+ arcmsr_iop_message_read(acb);
}
- arcmsr_iop_message_read(acb);
}
+ leave:
spin_unlock_irqrestore(&acb->rqbuffer_lock, flags);
- memcpy(pcmdmessagefld->messagedatabuffer,
- ver_addr, allxfer_len);
+ memcpy(pcmdmessagefld->messagedatabuffer, ver_addr, allxfer_len);
pcmdmessagefld->cmdmessage.Length = allxfer_len;
if (acb->fw_flag == FW_DEADLOCK) {
pcmdmessagefld->cmdmessage.ReturnCode =
@@ -2704,26 +3001,23 @@ arcmsr_iop_message_xfer(struct AdapterCo
ARCMSR_MESSAGE_RETURNCODE_OK;
}
kfree(ver_addr);
- }
break;
-
+ }
case ARCMSR_MESSAGE_WRITE_WQBUFFER: {
unsigned char *ver_addr;
int32_t my_empty_len, user_len, wqbuf_firstindex,
wqbuf_lastindex;
uint8_t *pQbuffer, *ptmpuserbuffer;
- unsigned long flags;
-
ver_addr = kmalloc(1032, GFP_ATOMIC);
if (!ver_addr) {
retvalue = ARCMSR_MESSAGE_FAIL;
goto message_out;
}
if (acb->fw_flag == FW_DEADLOCK) {
- pcmdmessagefld->cmdmessage.ReturnCode =
+ pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_BUS_HANG_ON;
} else {
- pcmdmessagefld->cmdmessage.ReturnCode =
+ pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_OK;
}
ptmpuserbuffer = ver_addr;
@@ -2735,28 +3029,39 @@ arcmsr_iop_message_xfer(struct AdapterCo
wqbuf_firstindex = acb->wqbuf_firstindex;
if (wqbuf_lastindex != wqbuf_firstindex) {
struct SENSE_DATA *sensebuffer =
- (struct SENSE_DATA *)cmd->sense_buffer;
+ (struct SENSE_DATA *)cmd->sense_buffer;
arcmsr_post_ioctldata2iop(acb);
/* has error report sensedata */
- sensebuffer->ErrorCode = 0x70;
+ sensebuffer->ErrorCode = SCSI_SENSE_CURRENT_ERRORS;
sensebuffer->SenseKey = ILLEGAL_REQUEST;
sensebuffer->AdditionalSenseLength = 0x0A;
sensebuffer->AdditionalSenseCode = 0x20;
sensebuffer->Valid = 1;
retvalue = ARCMSR_MESSAGE_FAIL;
} else {
- my_empty_len = (wqbuf_firstindex-wqbuf_lastindex - 1)
- &(ARCMSR_MAX_QBUFFER - 1);
+ my_empty_len = (wqbuf_firstindex - wqbuf_lastindex - 1)
+ & (ARCMSR_MAX_QBUFFER - 1);
if (my_empty_len >= user_len) {
while (user_len > 0) {
- pQbuffer =
- &acb->wqbuffer[acb->wqbuf_lastindex];
- memcpy(pQbuffer, ptmpuserbuffer, 1);
- acb->wqbuf_lastindex++;
- acb->wqbuf_lastindex %=
- ARCMSR_MAX_QBUFFER;
- ptmpuserbuffer++;
- user_len--;
+ pQbuffer = &acb->wqbuffer[acb->wqbuf_lastindex];
+ if ((acb->wqbuf_lastindex + user_len)
+ > ARCMSR_MAX_QBUFFER) {
+ memcpy(pQbuffer, ptmpuserbuffer,
+ ARCMSR_MAX_QBUFFER -
+ acb->wqbuf_lastindex);
+ ptmpuserbuffer += (ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_lastindex);
+ user_len -= (ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_lastindex);
+ acb->wqbuf_lastindex = 0;
+ } else {
+ memcpy(pQbuffer, ptmpuserbuffer,
+ user_len);
+ acb->wqbuf_lastindex += user_len;
+ acb->wqbuf_lastindex %=
+ ARCMSR_MAX_QBUFFER;
+ user_len = 0;
+ }
}
if (acb->acb_flags &
ACB_F_MESSAGE_WQBUFFER_CLEARED) {
@@ -2765,24 +3070,26 @@ arcmsr_iop_message_xfer(struct AdapterCo
arcmsr_post_ioctldata2iop(acb);
}
} else {
- /* has error report sensedata */
struct SENSE_DATA *sensebuffer =
- (struct SENSE_DATA *)cmd->sense_buffer;
- sensebuffer->ErrorCode = 0x70;
+ (struct SENSE_DATA *)cmd->sense_buffer;
+ /* has error report sensedata */
+ sensebuffer->ErrorCode =
+ SCSI_SENSE_CURRENT_ERRORS;
sensebuffer->SenseKey = ILLEGAL_REQUEST;
sensebuffer->AdditionalSenseLength = 0x0A;
sensebuffer->AdditionalSenseCode = 0x20;
sensebuffer->Valid = 1;
retvalue = ARCMSR_MESSAGE_FAIL;
}
- }
- spin_unlock_irqrestore(&acb->wqbuffer_lock, flags);
- kfree(ver_addr);
}
+ spin_unlock_irqrestore(&acb->wqbuffer_lock, flags);
+ kfree(ver_addr);
break;
-
+ }
case ARCMSR_MESSAGE_CLEAR_RQBUFFER: {
uint8_t *pQbuffer = acb->rqbuffer;
+
+ spin_lock_irqsave(&acb->rqbuffer_lock, flags);
if (acb->acb_flags & ACB_F_IOPDATA_OVERFLOW) {
acb->acb_flags &= ~ACB_F_IOPDATA_OVERFLOW;
arcmsr_iop_message_read(acb);
@@ -2791,6 +3098,7 @@ arcmsr_iop_message_xfer(struct AdapterCo
acb->rqbuf_firstindex = 0;
acb->rqbuf_lastindex = 0;
memset(pQbuffer, 0, ARCMSR_MAX_QBUFFER);
+ spin_unlock_irqrestore(&acb->rqbuffer_lock, flags);
if (acb->fw_flag == FW_DEADLOCK) {
pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_BUS_HANG_ON;
@@ -2798,9 +3106,8 @@ arcmsr_iop_message_xfer(struct AdapterCo
pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_OK;
}
- }
break;
-
+ }
case ARCMSR_MESSAGE_CLEAR_WQBUFFER: {
uint8_t *pQbuffer = acb->wqbuffer;
if (acb->fw_flag == FW_DEADLOCK) {
@@ -2810,39 +3117,40 @@ arcmsr_iop_message_xfer(struct AdapterCo
pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_OK;
}
-
+ spin_lock_irqsave(&acb->wqbuffer_lock, flags);
if (acb->acb_flags & ACB_F_IOPDATA_OVERFLOW) {
acb->acb_flags &= ~ACB_F_IOPDATA_OVERFLOW;
arcmsr_iop_message_read(acb);
}
- acb->acb_flags |=
- (ACB_F_MESSAGE_WQBUFFER_CLEARED |
- ACB_F_MESSAGE_WQBUFFER_READED);
+ acb->acb_flags |= (ACB_F_MESSAGE_WQBUFFER_CLEARED |
+ ACB_F_MESSAGE_WQBUFFER_READED);
acb->wqbuf_firstindex = 0;
acb->wqbuf_lastindex = 0;
memset(pQbuffer, 0, ARCMSR_MAX_QBUFFER);
- }
+ spin_unlock_irqrestore(&acb->wqbuffer_lock, flags);
break;
-
+ }
case ARCMSR_MESSAGE_CLEAR_ALLQBUFFER: {
uint8_t *pQbuffer;
-
if (acb->acb_flags & ACB_F_IOPDATA_OVERFLOW) {
acb->acb_flags &= ~ACB_F_IOPDATA_OVERFLOW;
arcmsr_iop_message_read(acb);
}
- acb->acb_flags |=
- (ACB_F_MESSAGE_WQBUFFER_CLEARED
- | ACB_F_MESSAGE_RQBUFFER_CLEARED
- | ACB_F_MESSAGE_WQBUFFER_READED);
+ spin_lock_irqsave(&acb->rqbuffer_lock, flags);
+ acb->acb_flags |= ACB_F_MESSAGE_RQBUFFER_CLEARED;
acb->rqbuf_firstindex = 0;
acb->rqbuf_lastindex = 0;
- acb->wqbuf_firstindex = 0;
- acb->wqbuf_lastindex = 0;
pQbuffer = acb->rqbuffer;
memset(pQbuffer, 0, sizeof(struct QBUFFER));
+ spin_unlock_irqrestore(&acb->rqbuffer_lock, flags);
+ spin_lock_irqsave(&acb->wqbuffer_lock, flags);
+ acb->acb_flags |= (ACB_F_MESSAGE_WQBUFFER_CLEARED |
+ ACB_F_MESSAGE_WQBUFFER_READED);
+ acb->wqbuf_firstindex = 0;
+ acb->wqbuf_lastindex = 0;
pQbuffer = acb->wqbuffer;
memset(pQbuffer, 0, sizeof(struct QBUFFER));
+ spin_unlock_irqrestore(&acb->wqbuffer_lock, flags);
if (acb->fw_flag == FW_DEADLOCK) {
pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_BUS_HANG_ON;
@@ -2850,9 +3158,8 @@ arcmsr_iop_message_xfer(struct AdapterCo
pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_OK;
}
- }
break;
-
+ }
case ARCMSR_MESSAGE_RETURN_CODE_3F: {
if (acb->fw_flag == FW_DEADLOCK) {
pcmdmessagefld->cmdmessage.ReturnCode =
@@ -2862,7 +3169,7 @@ arcmsr_iop_message_xfer(struct AdapterCo
ARCMSR_MESSAGE_RETURNCODE_3F;
}
break;
- }
+ }
case ARCMSR_MESSAGE_SAY_HELLO: {
int8_t *hello_string = "Hello! I am ARCMSR";
if (acb->fw_flag == FW_DEADLOCK) {
@@ -2872,33 +3179,42 @@ arcmsr_iop_message_xfer(struct AdapterCo
pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_OK;
}
- memcpy(pcmdmessagefld->messagedatabuffer, hello_string
- , (int16_t)strlen(hello_string));
- }
+ memcpy(pcmdmessagefld->messagedatabuffer,
+ hello_string, (int16_t)strlen(hello_string));
break;
-
- case ARCMSR_MESSAGE_SAY_GOODBYE:
+ }
+ case ARCMSR_MESSAGE_SAY_GOODBYE: {
if (acb->fw_flag == FW_DEADLOCK) {
pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_BUS_HANG_ON;
+ } else {
+ pcmdmessagefld->cmdmessage.ReturnCode =
+ ARCMSR_MESSAGE_RETURNCODE_OK;
}
arcmsr_iop_parking(acb);
break;
-
- case ARCMSR_MESSAGE_FLUSH_ADAPTER_CACHE:
+ }
+ case ARCMSR_MESSAGE_FLUSH_ADAPTER_CACHE: {
if (acb->fw_flag == FW_DEADLOCK) {
pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_BUS_HANG_ON;
+ } else {
+ pcmdmessagefld->cmdmessage.ReturnCode =
+ ARCMSR_MESSAGE_RETURNCODE_OK;
}
arcmsr_flush_adapter_cache(acb);
break;
-
+ }
default:
retvalue = ARCMSR_MESSAGE_FAIL;
+ printk("unknown controlcode(%d)\n", __LINE__);
}
message_out:
- sg = scsi_sglist(cmd);
- kunmap_atomic(buffer - sg->offset, KM_IRQ0);
+ if (use_sg) {
+ struct scatterlist *sg;
+ sg = scsi_sglist(cmd);
+ kunmap_atomic(buffer - sg->offset, KM_IRQ0);
+ }
return retvalue;
}
@@ -2930,7 +3246,6 @@ arcmsr_handle_virtual_command(struct Ada
unsigned char inqdata[36];
char *buffer;
struct scatterlist *sg;
-
if (cmd->device->lun) {
cmd->result = (DID_TIME_OUT << 16);
cmd->scsi_done(cmd);
@@ -2949,14 +3264,11 @@ arcmsr_handle_virtual_command(struct Ada
strncpy(&inqdata[16], "RAID controller ", 16);
/* Product Identification */
strncpy(&inqdata[32], "R001", 4); /* Product Revision */
-
sg = scsi_sglist(cmd);
buffer = kmap_atomic(sg_page(sg), KM_IRQ0) + sg->offset;
-
memcpy(buffer, inqdata, sizeof(inqdata));
sg = scsi_sglist(cmd);
kunmap_atomic(buffer - sg->offset, KM_IRQ0);
-
cmd->scsi_done(cmd);
}
break;
@@ -3611,19 +3923,21 @@ arcmsr_hbaD_polling_ccbdone(struct Adapt
{
bool error;
uint32_t poll_ccb_done = 0, poll_count = 0, flag_ccb, ccb_cdb_phy;
- int rtn, index, outbound_write_pointer;
+ int rtn, doneq_index, index_stripped, outbound_write_pointer;
+ unsigned long flags;
struct ARCMSR_CDB *arcmsr_cdb;
struct CommandControlBlock *pCCB;
- struct MessageUnit_D __iomem *reg =
+ struct MessageUnit_D __iomem *pmu =
(struct MessageUnit_D *)acb->pmuD;
+ spin_lock_irqsave(&acb->doneq_lock, flags);
polling_hbaD_ccb_retry:
poll_count++;
while (1) {
outbound_write_pointer =
- readl(reg->outboundlist_copy_pointer);
- index = reg->doneq_index;
- if ((outbound_write_pointer & 0xFF) == index) {
+ pmu->done_qbuffer[0].addressLow + 1;
+ doneq_index = pmu->doneq_index;
+ if ((outbound_write_pointer & 0xFFF) == (doneq_index & 0xFFF)) {
if (poll_ccb_done) {
rtn = SUCCESS;
break;
@@ -3636,17 +3950,27 @@ arcmsr_hbaD_polling_ccbdone(struct Adapt
goto polling_hbaD_ccb_retry;
}
}
- flag_ccb = reg->done_qbuffer[index].addressLow;
+ if (doneq_index & 0x4000) {
+ index_stripped = doneq_index & 0xFFF;
+ index_stripped += 1;
+ index_stripped %= ARCMSR_MAX_ARC1214_DONEQUEUE;
+ pmu->doneq_index = index_stripped ? (index_stripped | 0x4000)
+ : (index_stripped + 1);
+ } else {
+ index_stripped = doneq_index;
+ index_stripped += 1;
+ index_stripped %= ARCMSR_MAX_ARC1214_DONEQUEUE;
+ pmu->doneq_index = index_stripped ? index_stripped :
+ ((index_stripped | 0x4000) + 1);
+ }
+ doneq_index = pmu->doneq_index;
+ flag_ccb = pmu->done_qbuffer[doneq_index & 0xFFF].addressLow;
ccb_cdb_phy = (flag_ccb & 0xFFFFFFF0);
arcmsr_cdb = (struct ARCMSR_CDB *)(acb->vir2phy_offset +
ccb_cdb_phy);
pCCB = container_of(arcmsr_cdb,
struct CommandControlBlock, arcmsr_cdb);
poll_ccb_done = (pCCB == poll_ccb) ? 1 : 0;
- index++;
- index %= ARCMSR_MAX_ARC1214_POSTQUEUE;
- reg->doneq_index = index;
- /* check if command done with no error*/
if ((pCCB->acb != acb) ||
(pCCB->startdone != ARCMSR_CCB_START)) {
if (pCCB->startdone == ARCMSR_CCB_ABORTED) {
@@ -3673,6 +3997,7 @@ arcmsr_hbaD_polling_ccbdone(struct Adapt
? true : false;
arcmsr_report_ccb_state(acb, pCCB, error);
}
+ spin_unlock_irqrestore(&acb->doneq_lock, flags);
return rtn;
}
@@ -3805,7 +4130,7 @@ arcmsr_iop_confirm(struct AdapterControl
struct MessageUnit_D *reg =
(struct MessageUnit_D *)acb->pmuD;
reg->postq_index = 0;
- reg->doneq_index = 0x40FF;
+ reg->doneq_index = 0;
rwbuffer = reg->msgcode_rwbuffer;
writel(ARCMSR_SIGNATURE_SET_CONFIG, rwbuffer++);
writel(cdb_phyaddr_hi32, rwbuffer++);
@@ -3849,7 +4174,8 @@ arcmsr_wait_firmware_ready(struct Adapte
}
break;
case ACB_ADAPTER_TYPE_C: {
- struct MessageUnit_C *reg = (struct MessageUnit_C *)acb->pmuC;
+ struct MessageUnit_C *reg =
+ (struct MessageUnit_C *)acb->pmuC;
do {
firmware_state = readl(®->outbound_msgaddr1);
} while ((firmware_state &
^ permalink raw reply [flat|nested] 4+ messages in thread
* [PATCH 5/5] arcmsr: Modify ARC-1214 Inband Messages Behavior
@ 2013-02-08 6:04 NickCheng
0 siblings, 0 replies; 4+ messages in thread
From: NickCheng @ 2013-02-08 6:04 UTC (permalink / raw)
To: linux-scsi; +Cc: linux-kernel, jejb, ???
[-- Attachment #1: Type: text/plain, Size: 179 bytes --]
From: Nick Cheng <nick.cheng@areca.com.tw>
Modify ARC-1214 inband messages behavior to make up for HW seldom malfunction.
Signed-off-by: Nick Cheng <nick.cheng@areca.com.tw>
---
[-- Attachment #2: patch5 --]
[-- Type: application/octet-stream, Size: 46801 bytes --]
diff -uprN a/drivers/scsi/arcmsr/arcmsr_attr.c b/drivers/scsi/arcmsr/arcmsr_attr.c
--- a/drivers/scsi/arcmsr/arcmsr_attr.c 2013-02-08 13:57:56.091899027 +0800
+++ b/drivers/scsi/arcmsr/arcmsr_attr.c 2013-02-08 13:58:11.070196747 +0800
@@ -72,40 +72,82 @@ arcmsr_sysfs_iop_message_read(struct fil
(struct AdapterControlBlock *)host->hostdata;
uint8_t *pQbuffer,*ptmpQbuffer;
int32_t allxfer_len = 0;
+ unsigned long flags;
if (!capable(CAP_SYS_ADMIN))
return -EACCES;
/* do message unit read. */
ptmpQbuffer = (uint8_t *)buf;
- while ((acb->rqbuf_firstindex != acb->rqbuf_lastindex)
- && (allxfer_len < 1031)) {
+ spin_lock_irqsave(&acb->rqbuffer_lock, flags);
+ if (acb->rqbuf_firstindex != acb->rqbuf_lastindex) {
pQbuffer = &acb->rqbuffer[acb->rqbuf_firstindex];
- memcpy(ptmpQbuffer, pQbuffer, 1);
- acb->rqbuf_firstindex++;
- acb->rqbuf_firstindex %= ARCMSR_MAX_QBUFFER;
- ptmpQbuffer++;
- allxfer_len++;
+ if (acb->rqbuf_firstindex > acb->rqbuf_lastindex) {
+ if ((ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex) >= 1032) {
+ memcpy(ptmpQbuffer, pQbuffer, 1032);
+ acb->rqbuf_firstindex += 1032;
+ allxfer_len = 1032;
+ } else {
+ if (((ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex) + acb->rqbuf_lastindex) > 1032) {
+ memcpy(ptmpQbuffer, pQbuffer, ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex);
+ ptmpQbuffer += ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex;
+ memcpy(ptmpQbuffer, acb->rqbuffer, 1032 - (ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex));
+ acb->rqbuf_firstindex = 1032 - (ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex);
+ allxfer_len = 1032;
+ } else {
+ memcpy(ptmpQbuffer, pQbuffer, ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex);
+ ptmpQbuffer += ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex;
+ memcpy(ptmpQbuffer, acb->rqbuffer, acb->rqbuf_lastindex);
+ allxfer_len = ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex + acb->rqbuf_lastindex;
+ acb->rqbuf_firstindex = acb->rqbuf_lastindex;
+ }
+ }
+ } else {
+ if ((acb->rqbuf_lastindex - acb->rqbuf_firstindex) > 1032) {
+ memcpy(ptmpQbuffer, pQbuffer, 1032);
+ acb->rqbuf_firstindex += 1032;
+ allxfer_len = 1032;
+ } else {
+ memcpy(ptmpQbuffer, pQbuffer, acb->rqbuf_lastindex - acb->rqbuf_firstindex);
+ allxfer_len = acb->rqbuf_lastindex - acb->rqbuf_firstindex;
+ acb->rqbuf_firstindex = acb->rqbuf_lastindex;
+ }
+ }
}
if (acb->acb_flags & ACB_F_IOPDATA_OVERFLOW) {
struct QBUFFER __iomem *prbuffer;
- uint8_t __iomem *iop_data;
- int32_t iop_len;
-
- acb->acb_flags &= ~ACB_F_IOPDATA_OVERFLOW;
+ uint8_t __iomem *iop_data, *vaddr, *temp;
+ int32_t data_len_residual, data_len, rqbuf_lastindex;
+ acb->acb_flags &= ~ACB_F_IOPDATA_OVERFLOW;
+ rqbuf_lastindex = acb->rqbuf_lastindex;
prbuffer = arcmsr_get_iop_rqbuffer(acb);
- iop_data = prbuffer->data;
- iop_len = readl(&prbuffer->data_len);
- while (iop_len > 0) {
- acb->rqbuffer[acb->rqbuf_lastindex] =
- readb(iop_data);
- acb->rqbuf_lastindex++;
- acb->rqbuf_lastindex %= ARCMSR_MAX_QBUFFER;
- iop_data++;
- iop_len--;
+ iop_data = (uint8_t __iomem *)prbuffer->data;
+ data_len_residual = data_len = readl(&prbuffer->data_len);
+ if (data_len > 0) {
+ temp = vaddr = kmalloc(data_len, GFP_ATOMIC);
+ do {
+ memcpy(temp, iop_data, 4);
+ temp += 4;
+ iop_data += 4;
+ data_len_residual -= 4;
+ } while (data_len_residual > 0);
+ pQbuffer = &acb->rqbuffer[acb->rqbuf_lastindex];
+ temp = vaddr;
+ if ((rqbuf_lastindex + data_len) > ARCMSR_MAX_QBUFFER) {
+ memcpy(pQbuffer, temp, ARCMSR_MAX_QBUFFER - rqbuf_lastindex);
+ temp += (ARCMSR_MAX_QBUFFER - rqbuf_lastindex);
+ rqbuf_lastindex = (rqbuf_lastindex + data_len) % ARCMSR_MAX_QBUFFER;
+ memcpy(&acb->rqbuffer[0], temp, rqbuf_lastindex);
+ } else {
+ memcpy(pQbuffer, temp, data_len);
+ rqbuf_lastindex = (rqbuf_lastindex + data_len) % ARCMSR_MAX_QBUFFER;
+ }
+ kfree(vaddr);
}
+ acb->rqbuf_lastindex = rqbuf_lastindex;
arcmsr_iop_message_read(acb);
}
+ spin_unlock_irqrestore(&acb->rqbuffer_lock, flags);
return (allxfer_len);
}
@@ -122,6 +164,7 @@ arcmsr_sysfs_iop_message_write(struct fi
(struct AdapterControlBlock *)host->hostdata;
int32_t my_empty_len, user_len, wqbuf_firstindex, wqbuf_lastindex;
uint8_t *pQbuffer, *ptmpuserbuffer;
+ unsigned long flags;
if (!capable(CAP_SYS_ADMIN))
return -EACCES;
@@ -130,10 +173,12 @@ arcmsr_sysfs_iop_message_write(struct fi
/* do message unit write. */
ptmpuserbuffer = (uint8_t *)buf;
user_len = (int32_t)count;
+ spin_lock_irqsave(&acb->wqbuffer_lock, flags);
wqbuf_lastindex = acb->wqbuf_lastindex;
wqbuf_firstindex = acb->wqbuf_firstindex;
if (wqbuf_lastindex != wqbuf_firstindex) {
arcmsr_post_ioctldata2iop(acb);
+ spin_unlock_irqrestore(&acb->wqbuffer_lock, flags);
return 0; /*need retry*/
} else {
my_empty_len = (wqbuf_firstindex-wqbuf_lastindex - 1)
@@ -155,8 +200,10 @@ arcmsr_sysfs_iop_message_write(struct fi
~ACB_F_MESSAGE_WQBUFFER_CLEARED;
arcmsr_post_ioctldata2iop(acb);
}
+ spin_unlock_irqrestore(&acb->wqbuffer_lock, flags);
return count;
} else {
+ spin_unlock_irqrestore(&acb->wqbuffer_lock, flags);
return 0; /*need retry*/
}
}
@@ -174,6 +221,7 @@ arcmsr_sysfs_iop_message_clear(struct fi
struct AdapterControlBlock *acb =
(struct AdapterControlBlock *)host->hostdata;
uint8_t *pQbuffer;
+ unsigned long flags;
if (!capable(CAP_SYS_ADMIN))
return -EACCES;
@@ -186,10 +234,14 @@ arcmsr_sysfs_iop_message_clear(struct fi
(ACB_F_MESSAGE_WQBUFFER_CLEARED
| ACB_F_MESSAGE_RQBUFFER_CLEARED
| ACB_F_MESSAGE_WQBUFFER_READED);
+ spin_lock_irqsave(&acb->rqbuffer_lock, flags);
acb->rqbuf_firstindex = 0;
acb->rqbuf_lastindex = 0;
+ spin_unlock_irqrestore(&acb->rqbuffer_lock, flags);
+ spin_lock_irqsave(&acb->wqbuffer_lock, flags);
acb->wqbuf_firstindex = 0;
acb->wqbuf_lastindex = 0;
+ spin_unlock_irqrestore(&acb->wqbuffer_lock, flags);
pQbuffer = acb->rqbuffer;
memset(pQbuffer, 0, sizeof (struct QBUFFER));
pQbuffer = acb->wqbuffer;
diff -uprN a/drivers/scsi/arcmsr/arcmsr.h b/drivers/scsi/arcmsr/arcmsr.h
--- a/drivers/scsi/arcmsr/arcmsr.h 2013-02-08 13:57:56.460908463 +0800
+++ b/drivers/scsi/arcmsr/arcmsr.h 2013-02-08 13:58:11.430206075 +0800
@@ -63,6 +63,7 @@ struct device_attribute;
#define ARCMSR_DEFAULT_SG_ENTRIES 38
#define ARCMSR_MAX_HBB_POSTQUEUE 264
#define ARCMSR_MAX_ARC1214_POSTQUEUE 256
+#define ARCMSR_MAX_ARC1214_DONEQUEUE 257
#define ARCMSR_MAX_XFER_LEN 0x26000
#define ARCMSR_CDB_SG_PAGE_LENGTH 256
#define ARCMST_NUM_MSIX_VECTORS 4
@@ -564,7 +565,7 @@ struct OutBound_SRB {
struct MessageUnit_D {
struct InBound_SRB post_qbuffer[ARCMSR_MAX_ARC1214_POSTQUEUE];
- struct OutBound_SRB done_qbuffer[ARCMSR_MAX_ARC1214_POSTQUEUE];
+ struct OutBound_SRB done_qbuffer[ARCMSR_MAX_ARC1214_DONEQUEUE];
u16 postq_index;
u16 doneq_index;
u32 __iomem *chip_id; /*0x00004*/
@@ -617,6 +618,7 @@ struct AdapterControlBlock
spinlock_t eh_lock;
spinlock_t ccblist_lock;
spinlock_t postq_lock;
+ spinlock_t doneq_lock;
spinlock_t rqbuffer_lock;
spinlock_t wqbuffer_lock;
union {
@@ -670,15 +672,15 @@ struct AdapterControlBlock
unsigned int uncache_size;
uint8_t rqbuffer[ARCMSR_MAX_QBUFFER];
/* data collection buffer for read from 80331 */
- int32_t rqbuf_firstindex;
+ uint32_t rqbuf_firstindex;
/* first of read buffer */
- int32_t rqbuf_lastindex;
+ uint32_t rqbuf_lastindex;
/* last of read buffer */
uint8_t wqbuffer[ARCMSR_MAX_QBUFFER];
/* data collection buffer for write to 80331 */
- int32_t wqbuf_firstindex;
+ uint32_t wqbuf_firstindex;
/* first of write buffer */
- int32_t wqbuf_lastindex;
+ uint32_t wqbuf_lastindex;
/* last of write buffer */
uint8_t devstate[ARCMSR_MAX_TARGETID][ARCMSR_MAX_TARGETLUN];
/* id0 ..... id15, lun0...lun7 */
diff -uprN a/drivers/scsi/arcmsr/arcmsr_hba.c b/drivers/scsi/arcmsr/arcmsr_hba.c
--- a/drivers/scsi/arcmsr/arcmsr_hba.c 2013-02-08 13:57:56.797912198 +0800
+++ b/drivers/scsi/arcmsr/arcmsr_hba.c 2013-02-08 13:58:11.776204212 +0800
@@ -768,18 +768,18 @@ arcmsr_message_isr_bh_fn(struct work_str
(diff & 0x01) == 1) {
scsi_add_device(acb->host,
0, target, lun);
- } else if ((temp & 0x01) == 0
- && (diff & 0x01) == 1) {
- psdev =
- scsi_device_lookup(acb->host,
- 0, target, lun);
- if (psdev != NULL) {
- scsi_remove_device(psdev);
- scsi_device_put(psdev);
+ } else if ((temp & 0x01) == 0
+ && (diff & 0x01) == 1) {
+ psdev =
+ scsi_device_lookup(acb->host,
+ 0, target, lun);
+ if (psdev != NULL) {
+ scsi_remove_device(psdev);
+ scsi_device_put(psdev);
+ }
}
- }
- temp >>= 1;
- diff >>= 1;
+ temp >>= 1;
+ diff >>= 1;
}
}
devicemap++;
@@ -813,20 +813,20 @@ arcmsr_message_isr_bh_fn(struct work_str
lun < ARCMSR_MAX_TARGETLUN;
lun++) {
if ((temp & 0x01) == 1 &&
- (diff & 0x01) == 1) {
+ (diff & 0x01) == 1) {
scsi_add_device(acb->host,
0, target, lun);
} else if ((temp & 0x01) == 0
- && (diff & 0x01) == 1) {
+ && (diff & 0x01) == 1) {
psdev = scsi_device_lookup(acb->host,
0, target, lun);
if (psdev != NULL) {
scsi_remove_device(psdev);
scsi_device_put(psdev);
}
- }
- temp >>= 1;
- diff >>= 1;
+ }
+ temp >>= 1;
+ diff >>= 1;
}
}
devicemap++;
@@ -869,10 +869,10 @@ arcmsr_message_isr_bh_fn(struct work_str
if (psdev != NULL) {
scsi_remove_device(psdev);
scsi_device_put(psdev);
+ }
}
- }
- temp >>= 1;
- diff >>= 1;
+ temp >>= 1;
+ diff >>= 1;
}
}
devicemap++;
@@ -915,10 +915,10 @@ arcmsr_message_isr_bh_fn(struct work_str
if (psdev != NULL) {
scsi_remove_device(psdev);
scsi_device_put(psdev);
+ }
}
- }
- temp >>= 1;
- diff >>= 1;
+ temp >>= 1;
+ diff >>= 1;
}
}
devicemap++;
@@ -981,8 +981,8 @@ arcmsr_resume(struct pci_dev *pdev)
error = pci_set_dma_mask(pdev, DMA_BIT_MASK(32));
if (error) {
pr_warn("scsi%d: No suitable DMA mask available\n",
- host->host_no);
- goto controller_unregister;
+ host->host_no);
+ goto controller_unregister;
}
}
pci_set_master(pdev);
@@ -1097,6 +1097,7 @@ static int arcmsr_probe(struct pci_dev *
spin_lock_init(&acb->eh_lock);
spin_lock_init(&acb->ccblist_lock);
spin_lock_init(&acb->postq_lock);
+ spin_lock_init(&acb->doneq_lock);
spin_lock_init(&acb->rqbuffer_lock);
spin_lock_init(&acb->wqbuffer_lock);
acb->acb_flags |= (ACB_F_MESSAGE_WQBUFFER_CLEARED |
@@ -1541,33 +1542,29 @@ arcmsr_done4abort_postqueue(struct Adapt
uint32_t doneq_index, index_stripped, addressLow, residual;
bool error;
struct CommandControlBlock *pCCB;
- outbound_write_pointer =
- readl(pmu->outboundlist_copy_pointer);
+ outbound_write_pointer = pmu->done_qbuffer[0].addressLow + 1;
doneq_index = pmu->doneq_index;
residual = atomic_read(&acb->ccboutstandingcount);
for (i = 0; i < residual; i++) {
- while ((doneq_index & 0xFF) !=
- (outbound_write_pointer & 0xFF)) {
+ while ((doneq_index & 0xFFF) != (outbound_write_pointer & 0xFFF)) {
if (doneq_index & 0x4000) {
- index_stripped = doneq_index & 0xFF;
+ index_stripped = doneq_index & 0xFFF;
index_stripped += 1;
index_stripped %=
- ARCMSR_MAX_ARC1214_POSTQUEUE;
+ ARCMSR_MAX_ARC1214_DONEQUEUE;
pmu->doneq_index = index_stripped ?
- (index_stripped | 0x4000) :
- index_stripped;
+ (index_stripped | 0x4000) : (index_stripped + 1);
} else {
index_stripped = doneq_index;
index_stripped += 1;
index_stripped %=
- ARCMSR_MAX_ARC1214_POSTQUEUE;
+ ARCMSR_MAX_ARC1214_DONEQUEUE;
pmu->doneq_index =
- index_stripped ? index_stripped :
- (index_stripped | 0x4000);
+ index_stripped ? index_stripped : ((index_stripped | 0x4000) + 1);
}
doneq_index = pmu->doneq_index;
addressLow =
- pmu->done_qbuffer[doneq_index & 0xFF].addressLow;
+ pmu->done_qbuffer[doneq_index & 0xFFF].addressLow;
ccb_cdb_phy = (addressLow & 0xFFFFFFF0);
pARCMSR_CDB = (struct ARCMSR_CDB *)
(acb->vir2phy_offset + ccb_cdb_phy);
@@ -1582,7 +1579,7 @@ arcmsr_done4abort_postqueue(struct Adapt
}
mdelay(10);
outbound_write_pointer =
- readl(pmu->outboundlist_copy_pointer);
+ pmu->done_qbuffer[0].addressLow + 1;
doneq_index = pmu->doneq_index;
}
pmu->postq_index = 0;
@@ -2004,12 +2001,14 @@ arcmsr_iop_message_read(struct AdapterCo
case ACB_ADAPTER_TYPE_C: {
struct MessageUnit_C __iomem *reg = acb->pmuC;
writel(ARCMSR_HBCMU_DRV2IOP_DATA_READ_OK, ®->inbound_doorbell);
+ /*readl(®->inbound_doorbell);*/
}
break;
case ACB_ADAPTER_TYPE_D: {
struct MessageUnit_D __iomem *reg = acb->pmuD;
writel(ARCMSR_ARC1214_DRV2IOP_DATA_OUT_READ,
reg->inbound_doorbell);
+ /*readl(reg->inbound_doorbell);*/
break;
}
}
@@ -2127,36 +2126,61 @@ struct QBUFFER __iomem
void
arcmsr_iop2drv_data_wrote_handle(struct AdapterControlBlock *acb)
{
- uint8_t __iomem *iop_data;
- struct QBUFFER __iomem *prbuffer;
- struct QBUFFER *pQbuffer;
- int32_t my_empty_len, iop_len, rqbuf_firstindex, rqbuf_lastindex;
+ uint8_t __iomem *iop_data, *pQbuffer, *vaddr, *temp;
+ int32_t buf_empty_len, data_len, data_len_residual;
+ uint32_t rqbuf_firstindex, rqbuf_lastindex;
unsigned long flags;
-
+ struct QBUFFER __iomem *prbuffer;
spin_lock_irqsave(&acb->rqbuffer_lock, flags);
rqbuf_lastindex = acb->rqbuf_lastindex;
rqbuf_firstindex = acb->rqbuf_firstindex;
prbuffer = arcmsr_get_iop_rqbuffer(acb);
iop_data = (uint8_t __iomem *)prbuffer->data;
- iop_len = prbuffer->data_len;
- my_empty_len = (rqbuf_firstindex - rqbuf_lastindex - 1) &
- (ARCMSR_MAX_QBUFFER - 1);
-
- if (my_empty_len >= iop_len) {
- while (iop_len > 0) {
- pQbuffer = (struct QBUFFER *)
- &acb->rqbuffer[rqbuf_lastindex];
- memcpy(pQbuffer, iop_data, 1);
- rqbuf_lastindex++;
- rqbuf_lastindex %= ARCMSR_MAX_QBUFFER;
- iop_data++;
- iop_len--;
+ data_len_residual = data_len = readl(&prbuffer->data_len);
+ buf_empty_len = (rqbuf_firstindex - rqbuf_lastindex - 1) &
+ (ARCMSR_MAX_QBUFFER - 1);
+ if (buf_empty_len >= data_len) {
+ if (data_len > 0) {
+ temp = vaddr = kmalloc(data_len, GFP_ATOMIC);
+ if (!vaddr) {
+ goto leave;
+ }
+ acb->acb_flags &= ~ACB_F_IOPDATA_OVERFLOW;
+ while (data_len_residual >= 4) {
+ memcpy(temp, iop_data, 4);
+ temp += 4;
+ iop_data += 4;
+ data_len_residual -= 4;
+ }
+ if ((data_len_residual > 0) &&
+ (data_len_residual < 4)) {
+ memcpy(temp, iop_data, data_len_residual);
+ }
+ pQbuffer = &acb->rqbuffer[rqbuf_lastindex];
+ temp = vaddr;
+ if ((rqbuf_lastindex + data_len) >
+ ARCMSR_MAX_QBUFFER) {
+ memcpy(pQbuffer, temp,
+ ARCMSR_MAX_QBUFFER - rqbuf_lastindex);
+ temp += (ARCMSR_MAX_QBUFFER -
+ rqbuf_lastindex);
+ rqbuf_lastindex = (rqbuf_lastindex + data_len)
+ % ARCMSR_MAX_QBUFFER;
+ memcpy(&acb->rqbuffer[0], temp,
+ rqbuf_lastindex);
+ } else {
+ memcpy(pQbuffer, temp, data_len);
+ rqbuf_lastindex = (rqbuf_lastindex + data_len)
+ % ARCMSR_MAX_QBUFFER;
+ }
+ kfree(vaddr);
+ acb->rqbuf_lastindex = rqbuf_lastindex;
+ arcmsr_iop_message_read(acb);
}
- acb->rqbuf_lastindex = rqbuf_lastindex;
- arcmsr_iop_message_read(acb);
} else {
acb->acb_flags |= ACB_F_IOPDATA_OVERFLOW;
}
+ leave:
spin_unlock_irqrestore(&acb->rqbuffer_lock, flags);
}
@@ -2164,7 +2188,6 @@ void
arcmsr_iop2drv_data_read_handle(struct AdapterControlBlock *acb)
{
unsigned long flags;
-
spin_lock_irqsave(&acb->wqbuffer_lock, flags);
acb->acb_flags |= ACB_F_MESSAGE_WQBUFFER_READED;
if (acb->wqbuf_firstindex != acb->wqbuf_lastindex) {
@@ -2176,18 +2199,113 @@ arcmsr_iop2drv_data_read_handle(struct A
acb->acb_flags &= (~ACB_F_MESSAGE_WQBUFFER_READED);
pwbuffer = arcmsr_get_iop_wqbuffer(acb);
iop_data = (uint8_t __iomem *)pwbuffer->data;
-
- while ((acb->wqbuf_firstindex != acb->wqbuf_lastindex) &&
- (allxfer_len < 124)) {
- pQbuffer = &acb->wqbuffer[acb->wqbuf_firstindex];
- memcpy(iop_data, pQbuffer, 1);
- acb->wqbuf_firstindex++;
- acb->wqbuf_firstindex %= ARCMSR_MAX_QBUFFER;
- iop_data++;
- allxfer_len++;
+ if (acb->wqbuf_firstindex > acb->wqbuf_lastindex) {
+ if ((ARCMSR_MAX_QBUFFER - acb->wqbuf_firstindex) >= 4) {
+ do {
+ pQbuffer =
+ &acb->wqbuffer[acb->wqbuf_firstindex];
+ if (acb->wqbuf_firstindex + 4
+ > ARCMSR_MAX_QBUFFER) {
+ memcpy(iop_data, pQbuffer,
+ ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_firstindex);
+ iop_data += ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex += 4;
+ acb->wqbuf_firstindex %=
+ ARCMSR_MAX_QBUFFER;
+ memcpy(iop_data,
+ &acb->wqbuffer[0],
+ acb->wqbuf_firstindex);
+ iop_data +=
+ acb->wqbuf_firstindex;
+ } else {
+ if ((acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex) > 4) {
+ memcpy(iop_data,
+ pQbuffer, 4);
+ acb->wqbuf_firstindex
+ += 4;
+ acb->wqbuf_firstindex
+ %= ARCMSR_MAX_QBUFFER;
+ iop_data += 4;
+ } else {
+ memcpy(iop_data, pQbuffer,
+ acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex);
+ allxfer_len +=
+ acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex =
+ acb->wqbuf_lastindex;
+ break;
+ }
+ }
+ allxfer_len += 4;
+ } while ((acb->wqbuf_firstindex !=
+ acb->wqbuf_lastindex) && (allxfer_len < 124));
+ } else {
+ pQbuffer =
+ &acb->wqbuffer[acb->wqbuf_firstindex];
+ memcpy(iop_data, pQbuffer, ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_firstindex);
+ iop_data += ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_firstindex;
+ allxfer_len = ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex = 0;
+ do {
+ pQbuffer =
+ &acb->wqbuffer[acb->wqbuf_firstindex];
+ if ((acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex) > 4) {
+ memcpy(iop_data, pQbuffer, 4);
+ acb->wqbuf_firstindex += 4;
+ acb->wqbuf_firstindex %=
+ ARCMSR_MAX_QBUFFER;
+ iop_data += 4;
+ } else {
+ memcpy(iop_data, pQbuffer,
+ acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex);
+ allxfer_len +=
+ acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex =
+ acb->wqbuf_lastindex;
+ break;
+ }
+ allxfer_len += 4;
+ } while ((acb->wqbuf_firstindex !=
+ acb->wqbuf_lastindex) && (allxfer_len < 124));
+ }
+ } else {
+ do {
+ pQbuffer =
+ &acb->wqbuffer[acb->wqbuf_firstindex];
+ if ((acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex) > 4) {
+ memcpy(iop_data, pQbuffer, 4);
+ acb->wqbuf_firstindex += 4;
+ acb->wqbuf_firstindex %=
+ ARCMSR_MAX_QBUFFER;
+ iop_data += 4;
+ } else {
+ memcpy(iop_data, pQbuffer,
+ acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex);
+ allxfer_len += acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex =
+ acb->wqbuf_lastindex;
+ break;
+ }
+ allxfer_len += 4;
+ } while ((acb->wqbuf_firstindex !=
+ acb->wqbuf_lastindex) && (allxfer_len < 124));
}
- pwbuffer->data_len = allxfer_len;
- arcmsr_iop_message_wrote(acb);
+ writel(allxfer_len, &pwbuffer->data_len);
+ arcmsr_iop_message_wrote(acb);/*notice IOP the message has been written*/
}
if (acb->wqbuf_firstindex == acb->wqbuf_lastindex) {
acb->acb_flags |= ACB_F_MESSAGE_WQBUFFER_CLEARED;
@@ -2200,7 +2318,6 @@ arcmsr_hbaA_doorbell_isr(struct AdapterC
{
uint32_t outbound_doorbell;
struct MessageUnit_A __iomem *reg = acb->pmuA;
-
outbound_doorbell = readl(®->outbound_doorbell);
do {
writel(outbound_doorbell, ®->outbound_doorbell);
@@ -2223,7 +2340,6 @@ arcmsr_hbaC_doorbell_isr(struct AdapterC
uint32_t outbound_doorbell;
struct MessageUnit_C __iomem *reg =
(struct MessageUnit_C *)pACB->pmuC;
-
outbound_doorbell = readl(®->outbound_doorbell);
do {
if (outbound_doorbell &
@@ -2255,10 +2371,6 @@ arcmsr_hbaD_doorbell_isr(struct AdapterC
(struct MessageUnit_D *)pACB->pmuD;
outbound_doorbell = readl(pmu->outbound_doorbell);
- if (unlikely(!outbound_doorbell)) {
- arcmsr_iop2drv_data_wrote_handle(pACB);
- arcmsr_iop2drv_data_read_handle(pACB);
- }
do {
writel(outbound_doorbell, pmu->outbound_doorbell);
if (outbound_doorbell &
@@ -2367,28 +2479,30 @@ arcmsr_hbaD_postqueue_isr(struct Adapter
struct MessageUnit_D __iomem *pmu;
struct ARCMSR_CDB *arcmsr_cdb;
struct CommandControlBlock *ccb;
+ unsigned long flags;
+ spin_lock_irqsave(&acb->doneq_lock, flags);
pmu = (struct MessageUnit_D *)acb->pmuD;
- outbound_write_pointer = readl(pmu->outboundlist_copy_pointer);
+ outbound_write_pointer = pmu->done_qbuffer[0].addressLow + 1;
doneq_index = pmu->doneq_index;
- if ((doneq_index & 0xFF) != (outbound_write_pointer & 0xFF)) {
+ if ((doneq_index & 0xFFF) != (outbound_write_pointer & 0xFFF)) {
do {
if (doneq_index & 0x4000) {
- index_stripped = doneq_index & 0xFF;
+ index_stripped = doneq_index & 0xFFF;
index_stripped += 1;
- index_stripped %= ARCMSR_MAX_ARC1214_POSTQUEUE;
+ index_stripped %= ARCMSR_MAX_ARC1214_DONEQUEUE;
pmu->doneq_index = index_stripped
- ? (index_stripped | 0x4000) : index_stripped;
+ ? (index_stripped | 0x4000) : (index_stripped + 1);
} else {
index_stripped = doneq_index;
index_stripped += 1;
- index_stripped %= ARCMSR_MAX_ARC1214_POSTQUEUE;
+ index_stripped %= ARCMSR_MAX_ARC1214_DONEQUEUE;
pmu->doneq_index = index_stripped
- ? index_stripped : (index_stripped | 0x4000);
+ ? index_stripped : ((index_stripped | 0x4000) + 1);
}
doneq_index = pmu->doneq_index;
addressLow =
- pmu->done_qbuffer[doneq_index & 0xFF].addressLow;
+ pmu->done_qbuffer[doneq_index & 0xFFF].addressLow;
ccb_cdb_phy = (addressLow & 0xFFFFFFF0);
arcmsr_cdb = (struct ARCMSR_CDB *)(acb->vir2phy_offset
+ ccb_cdb_phy);
@@ -2399,12 +2513,13 @@ arcmsr_hbaD_postqueue_isr(struct Adapter
arcmsr_drain_donequeue(acb, ccb, error);
writel(doneq_index,
pmu->outboundlist_read_pointer);
- } while ((doneq_index & 0xFF) !=
- (outbound_write_pointer & 0xFF));
+ } while ((doneq_index & 0xFFF) !=
+ (outbound_write_pointer & 0xFFF));
}
writel(ARCMSR_ARC1214_OUTBOUND_LIST_INTERRUPT_CLEAR,
pmu->outboundlist_interrupt_cause);
readl(pmu->outboundlist_interrupt_cause);
+ spin_unlock_irqrestore(&acb->doneq_lock, flags);
}
static void
@@ -2457,7 +2572,6 @@ arcmsr_hbaA_handle_isr(struct AdapterCon
}
do {
writel(outbound_intstatus, ®->outbound_intstatus);
- readl(®->outbound_intstatus);
if (outbound_intstatus & ARCMSR_MU_OUTBOUND_DOORBELL_INT)
arcmsr_hbaA_doorbell_isr(acb);
if (outbound_intstatus & ARCMSR_MU_OUTBOUND_POSTQUEUE_INT)
@@ -2597,29 +2711,131 @@ arcmsr_iop_parking(struct AdapterControl
void
arcmsr_post_ioctldata2iop(struct AdapterControlBlock *acb)
{
- uint8_t __iomem *iop_data;
- uint8_t *pQbuffer;
- int32_t wqbuf_firstindex, wqbuf_lastindex;
- int32_t allxfer_len = 0;
- struct QBUFFER __iomem *pwbuffer;
- pwbuffer = arcmsr_get_iop_wqbuffer(acb);
- iop_data = (uint8_t __iomem *)pwbuffer->data;
if (acb->acb_flags & ACB_F_MESSAGE_WQBUFFER_READED) {
- acb->acb_flags &= (~ACB_F_MESSAGE_WQBUFFER_READED);
- wqbuf_firstindex = acb->wqbuf_firstindex;
- wqbuf_lastindex = acb->wqbuf_lastindex;
- while ((wqbuf_firstindex != wqbuf_lastindex)
- && (allxfer_len < 124)) {
- pQbuffer = &acb->wqbuffer[wqbuf_firstindex];
- memcpy(iop_data, pQbuffer, 1);
- wqbuf_firstindex++;
- wqbuf_firstindex %= ARCMSR_MAX_QBUFFER;
- iop_data++;
- allxfer_len++;
- }
- acb->wqbuf_firstindex = wqbuf_firstindex;
- pwbuffer->data_len = allxfer_len;
- arcmsr_iop_message_wrote(acb);
+ if (acb->wqbuf_firstindex != acb->wqbuf_lastindex) {
+ uint8_t *pQbuffer;
+ uint8_t __iomem *iop_data;
+ int32_t allxfer_len = 0;
+ struct QBUFFER __iomem *pwbuffer;
+ acb->acb_flags &= (~ACB_F_MESSAGE_WQBUFFER_READED);
+ pwbuffer = arcmsr_get_iop_wqbuffer(acb);
+ iop_data = (uint8_t __iomem *)pwbuffer->data;
+ if (acb->wqbuf_firstindex > acb->wqbuf_lastindex) {
+ if ((ARCMSR_MAX_QBUFFER -
+ acb->wqbuf_firstindex) >= 4) {
+ do {
+ pQbuffer =
+ &acb->wqbuffer[acb->wqbuf_firstindex];
+ if (acb->wqbuf_firstindex + 4 >
+ ARCMSR_MAX_QBUFFER) {
+ memcpy(iop_data, pQbuffer,
+ ARCMSR_MAX_QBUFFER - acb->wqbuf_firstindex);
+ iop_data += ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex += 4;
+ acb->wqbuf_firstindex %=
+ ARCMSR_MAX_QBUFFER;
+ memcpy(iop_data,
+ &acb->wqbuffer[0],
+ acb->wqbuf_firstindex);
+ iop_data +=
+ acb->wqbuf_firstindex;
+ } else {
+ if ((acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex) > 4) {
+ memcpy(iop_data,
+ pQbuffer, 4);
+ acb->wqbuf_firstindex
+ += 4;
+ acb->wqbuf_firstindex
+ %= ARCMSR_MAX_QBUFFER;
+ iop_data += 4;
+ } else {
+ memcpy(iop_data,
+ pQbuffer,
+ acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex);
+ allxfer_len +=
+ acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex
+ = acb->wqbuf_lastindex;
+ break;
+ }
+ }
+ allxfer_len += 4;
+ } while ((acb->wqbuf_firstindex !=
+ acb->wqbuf_lastindex) &&
+ (allxfer_len < 124));
+ } else {
+ pQbuffer =
+ &acb->wqbuffer[acb->wqbuf_firstindex];
+ memcpy(iop_data, pQbuffer,
+ ARCMSR_MAX_QBUFFER -
+ acb->wqbuf_firstindex);
+ iop_data += ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_firstindex;
+ allxfer_len = ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex = 0;
+ do {
+ pQbuffer =
+ &acb->wqbuffer[acb->wqbuf_firstindex];
+ if ((acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex) > 4) {
+ memcpy(iop_data,
+ pQbuffer, 4);
+ acb->wqbuf_firstindex
+ += 4;
+ acb->wqbuf_firstindex
+ %= ARCMSR_MAX_QBUFFER;
+ iop_data += 4;
+ } else {
+ memcpy(iop_data, pQbuffer,
+ acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex);
+ allxfer_len +=
+ acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex =
+ acb->wqbuf_lastindex;
+ break;
+ }
+ allxfer_len += 4;
+ } while ((acb->wqbuf_firstindex
+ != acb->wqbuf_lastindex) &&
+ (allxfer_len < 124));
+ }
+ } else {
+ do {
+ pQbuffer =
+ &acb->wqbuffer[acb->wqbuf_firstindex];
+ if ((acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex) > 4) {
+ memcpy(iop_data,
+ pQbuffer, 4);
+ acb->wqbuf_firstindex
+ += 4;
+ acb->wqbuf_firstindex
+ %= ARCMSR_MAX_QBUFFER;
+ iop_data += 4;
+ } else {
+ memcpy(iop_data, pQbuffer,
+ acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex);
+ allxfer_len += acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex =
+ acb->wqbuf_lastindex;
+ break;
+ }
+ allxfer_len += 4;
+ } while ((acb->wqbuf_firstindex !=
+ acb->wqbuf_lastindex) && (allxfer_len < 124));
+ }
+ writel(allxfer_len, &pwbuffer->data_len);
+ arcmsr_iop_message_wrote(acb);
+ }
}
}
@@ -2627,74 +2843,155 @@ static int
arcmsr_iop_message_xfer(struct AdapterControlBlock *acb,
struct scsi_cmnd *cmd)
{
- struct CMD_MESSAGE_FIELD *pcmdmessagefld;
- int retvalue = 0, transfer_len = 0;
char *buffer;
+ unsigned short use_sg;
+ int retvalue = 0, transfer_len = 0;
+ unsigned long flags;
+ struct CMD_MESSAGE_FIELD *pcmdmessagefld;
+ uint32_t controlcode = (uint32_t)cmd->cmnd[5] << 24 |
+ (uint32_t)cmd->cmnd[6] << 16 |
+ (uint32_t)cmd->cmnd[7] << 8 |
+ (uint32_t)cmd->cmnd[8];
struct scatterlist *sg;
- uint32_t controlcode = (uint32_t)cmd->cmnd[5] << 24
- |(uint32_t)cmd->cmnd[6] << 16
- |(uint32_t)cmd->cmnd[7] << 8
- | (uint32_t)cmd->cmnd[8];
+
+ use_sg = scsi_sg_count(cmd);
sg = scsi_sglist(cmd);
buffer = kmap_atomic(sg_page(sg)) + sg->offset;
- if (scsi_sg_count(cmd) > 1) {
+ if (use_sg > 1) {
retvalue = ARCMSR_MESSAGE_FAIL;
goto message_out;
}
transfer_len += sg->length;
-
if (transfer_len > sizeof(struct CMD_MESSAGE_FIELD)) {
retvalue = ARCMSR_MESSAGE_FAIL;
+ printk("%s: ARCMSR_MESSAGE_FAIL!\n", __func__);
goto message_out;
}
pcmdmessagefld = (struct CMD_MESSAGE_FIELD *)buffer;
switch (controlcode) {
-
case ARCMSR_MESSAGE_READ_RQBUFFER: {
unsigned char *ver_addr;
uint8_t *pQbuffer, *ptmpQbuffer;
- int32_t allxfer_len = 0;
- unsigned long flags;
-
+ uint32_t allxfer_len = 0;
ver_addr = kmalloc(1032, GFP_ATOMIC);
if (!ver_addr) {
retvalue = ARCMSR_MESSAGE_FAIL;
+ printk("%s: memory not enough!\n", __func__);
goto message_out;
}
ptmpQbuffer = ver_addr;
spin_lock_irqsave(&acb->rqbuffer_lock, flags);
- while ((acb->rqbuf_firstindex != acb->rqbuf_lastindex)
- && (allxfer_len < 1031)) {
+ if (acb->rqbuf_firstindex != acb->rqbuf_lastindex) {
pQbuffer = &acb->rqbuffer[acb->rqbuf_firstindex];
- memcpy(ptmpQbuffer, pQbuffer, 1);
- acb->rqbuf_firstindex++;
- acb->rqbuf_firstindex %= ARCMSR_MAX_QBUFFER;
- ptmpQbuffer++;
- allxfer_len++;
+ if (acb->rqbuf_firstindex > acb->rqbuf_lastindex) {
+ if ((ARCMSR_MAX_QBUFFER -
+ acb->rqbuf_firstindex) >= 1032) {
+ memcpy(ptmpQbuffer, pQbuffer, 1032);
+ acb->rqbuf_firstindex += 1032;
+ allxfer_len = 1032;
+ } else {
+ if (((ARCMSR_MAX_QBUFFER -
+ acb->rqbuf_firstindex) +
+ acb->rqbuf_lastindex) > 1032) {
+ memcpy(ptmpQbuffer,
+ pQbuffer, ARCMSR_MAX_QBUFFER
+ - acb->rqbuf_firstindex);
+ ptmpQbuffer +=
+ ARCMSR_MAX_QBUFFER -
+ acb->rqbuf_firstindex;
+ memcpy(ptmpQbuffer,
+ acb->rqbuffer, 1032 -
+ (ARCMSR_MAX_QBUFFER
+ - acb->rqbuf_firstindex));
+ acb->rqbuf_firstindex =
+ 1032 - (ARCMSR_MAX_QBUFFER
+ - acb->rqbuf_firstindex);
+ allxfer_len = 1032;
+ } else {
+ memcpy(ptmpQbuffer,
+ pQbuffer, ARCMSR_MAX_QBUFFER
+ - acb->rqbuf_firstindex);
+ ptmpQbuffer +=
+ ARCMSR_MAX_QBUFFER -
+ acb->rqbuf_firstindex;
+ memcpy(ptmpQbuffer,
+ acb->rqbuffer,
+ acb->rqbuf_lastindex);
+ allxfer_len = ARCMSR_MAX_QBUFFER
+ - acb->rqbuf_firstindex +
+ acb->rqbuf_lastindex;
+ acb->rqbuf_firstindex =
+ acb->rqbuf_lastindex;
+ }
+ }
+ } else {
+ if ((acb->rqbuf_lastindex -
+ acb->rqbuf_firstindex) > 1032) {
+ memcpy(ptmpQbuffer, pQbuffer, 1032);
+ acb->rqbuf_firstindex += 1032;
+ allxfer_len = 1032;
+ } else {
+ memcpy(ptmpQbuffer, pQbuffer,
+ acb->rqbuf_lastindex - acb->rqbuf_firstindex);
+ allxfer_len = acb->rqbuf_lastindex
+ - acb->rqbuf_firstindex;
+ acb->rqbuf_firstindex =
+ acb->rqbuf_lastindex;
+ }
+ }
}
if (acb->acb_flags & ACB_F_IOPDATA_OVERFLOW) {
-
struct QBUFFER __iomem *prbuffer;
- uint8_t __iomem *iop_data;
- int32_t iop_len;
-
- acb->acb_flags &= ~ACB_F_IOPDATA_OVERFLOW;
+ uint8_t __iomem *iop_data, *vaddr, *temp;
+ uint32_t data_len_residual, data_len, rqbuf_lastindex;
+ rqbuf_lastindex = acb->rqbuf_lastindex;
prbuffer = arcmsr_get_iop_rqbuffer(acb);
- iop_data = prbuffer->data;
- iop_len = readl(&prbuffer->data_len);
- while (iop_len > 0) {
- acb->rqbuffer[acb->rqbuf_lastindex] =
- readb(iop_data);
- acb->rqbuf_lastindex++;
- acb->rqbuf_lastindex %= ARCMSR_MAX_QBUFFER;
- iop_data++;
- iop_len--;
+ iop_data = (uint8_t __iomem *)prbuffer->data;
+ data_len_residual = data_len = readl(&prbuffer->data_len);
+ if (data_len > 0) {
+ temp = vaddr = kmalloc(data_len, GFP_ATOMIC);
+ if (!vaddr) {
+ goto leave;
+ }
+ acb->acb_flags &= ~ACB_F_IOPDATA_OVERFLOW;
+ while (data_len_residual >= 4) {
+ memcpy(temp, iop_data, 4);
+ temp += 4;
+ iop_data += 4;
+ data_len_residual -= 4;
+ }
+ if ((data_len_residual > 0) &&
+ (data_len_residual < 4)) {
+ memcpy(temp, iop_data,
+ data_len_residual);
+ }
+ pQbuffer =
+ &acb->rqbuffer[acb->rqbuf_lastindex];
+ temp = vaddr;
+ if ((rqbuf_lastindex + data_len) >
+ ARCMSR_MAX_QBUFFER) {
+ memcpy(pQbuffer, temp,
+ ARCMSR_MAX_QBUFFER - rqbuf_lastindex);
+ temp += (ARCMSR_MAX_QBUFFER -
+ rqbuf_lastindex);
+ rqbuf_lastindex = (rqbuf_lastindex +
+ data_len) % ARCMSR_MAX_QBUFFER;
+ memcpy(&acb->rqbuffer[0],
+ temp, rqbuf_lastindex);
+ } else {
+ memcpy(pQbuffer, temp, data_len);
+ rqbuf_lastindex =
+ (rqbuf_lastindex + data_len) %
+ ARCMSR_MAX_QBUFFER;
+ }
+ kfree(vaddr);
+ acb->rqbuf_lastindex = rqbuf_lastindex;
+ arcmsr_iop_message_read(acb);
}
- arcmsr_iop_message_read(acb);
}
+ leave:
spin_unlock_irqrestore(&acb->rqbuffer_lock, flags);
- memcpy(pcmdmessagefld->messagedatabuffer,
- ver_addr, allxfer_len);
+ memcpy(pcmdmessagefld->messagedatabuffer, ver_addr, allxfer_len);
pcmdmessagefld->cmdmessage.Length = allxfer_len;
if (acb->fw_flag == FW_DEADLOCK) {
pcmdmessagefld->cmdmessage.ReturnCode =
@@ -2704,26 +3001,23 @@ arcmsr_iop_message_xfer(struct AdapterCo
ARCMSR_MESSAGE_RETURNCODE_OK;
}
kfree(ver_addr);
- }
break;
-
+ }
case ARCMSR_MESSAGE_WRITE_WQBUFFER: {
unsigned char *ver_addr;
int32_t my_empty_len, user_len, wqbuf_firstindex,
wqbuf_lastindex;
uint8_t *pQbuffer, *ptmpuserbuffer;
- unsigned long flags;
-
ver_addr = kmalloc(1032, GFP_ATOMIC);
if (!ver_addr) {
retvalue = ARCMSR_MESSAGE_FAIL;
goto message_out;
}
if (acb->fw_flag == FW_DEADLOCK) {
- pcmdmessagefld->cmdmessage.ReturnCode =
+ pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_BUS_HANG_ON;
} else {
- pcmdmessagefld->cmdmessage.ReturnCode =
+ pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_OK;
}
ptmpuserbuffer = ver_addr;
@@ -2735,28 +3029,39 @@ arcmsr_iop_message_xfer(struct AdapterCo
wqbuf_firstindex = acb->wqbuf_firstindex;
if (wqbuf_lastindex != wqbuf_firstindex) {
struct SENSE_DATA *sensebuffer =
- (struct SENSE_DATA *)cmd->sense_buffer;
+ (struct SENSE_DATA *)cmd->sense_buffer;
arcmsr_post_ioctldata2iop(acb);
/* has error report sensedata */
- sensebuffer->ErrorCode = 0x70;
+ sensebuffer->ErrorCode = SCSI_SENSE_CURRENT_ERRORS;
sensebuffer->SenseKey = ILLEGAL_REQUEST;
sensebuffer->AdditionalSenseLength = 0x0A;
sensebuffer->AdditionalSenseCode = 0x20;
sensebuffer->Valid = 1;
retvalue = ARCMSR_MESSAGE_FAIL;
} else {
- my_empty_len = (wqbuf_firstindex-wqbuf_lastindex - 1)
- &(ARCMSR_MAX_QBUFFER - 1);
+ my_empty_len = (wqbuf_firstindex - wqbuf_lastindex - 1)
+ & (ARCMSR_MAX_QBUFFER - 1);
if (my_empty_len >= user_len) {
while (user_len > 0) {
- pQbuffer =
- &acb->wqbuffer[acb->wqbuf_lastindex];
- memcpy(pQbuffer, ptmpuserbuffer, 1);
- acb->wqbuf_lastindex++;
- acb->wqbuf_lastindex %=
- ARCMSR_MAX_QBUFFER;
- ptmpuserbuffer++;
- user_len--;
+ pQbuffer = &acb->wqbuffer[acb->wqbuf_lastindex];
+ if ((acb->wqbuf_lastindex + user_len)
+ > ARCMSR_MAX_QBUFFER) {
+ memcpy(pQbuffer, ptmpuserbuffer,
+ ARCMSR_MAX_QBUFFER -
+ acb->wqbuf_lastindex);
+ ptmpuserbuffer += (ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_lastindex);
+ user_len -= (ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_lastindex);
+ acb->wqbuf_lastindex = 0;
+ } else {
+ memcpy(pQbuffer, ptmpuserbuffer,
+ user_len);
+ acb->wqbuf_lastindex += user_len;
+ acb->wqbuf_lastindex %=
+ ARCMSR_MAX_QBUFFER;
+ user_len = 0;
+ }
}
if (acb->acb_flags &
ACB_F_MESSAGE_WQBUFFER_CLEARED) {
@@ -2765,24 +3070,26 @@ arcmsr_iop_message_xfer(struct AdapterCo
arcmsr_post_ioctldata2iop(acb);
}
} else {
- /* has error report sensedata */
struct SENSE_DATA *sensebuffer =
- (struct SENSE_DATA *)cmd->sense_buffer;
- sensebuffer->ErrorCode = 0x70;
+ (struct SENSE_DATA *)cmd->sense_buffer;
+ /* has error report sensedata */
+ sensebuffer->ErrorCode =
+ SCSI_SENSE_CURRENT_ERRORS;
sensebuffer->SenseKey = ILLEGAL_REQUEST;
sensebuffer->AdditionalSenseLength = 0x0A;
sensebuffer->AdditionalSenseCode = 0x20;
sensebuffer->Valid = 1;
retvalue = ARCMSR_MESSAGE_FAIL;
}
- }
- spin_unlock_irqrestore(&acb->wqbuffer_lock, flags);
- kfree(ver_addr);
}
+ spin_unlock_irqrestore(&acb->wqbuffer_lock, flags);
+ kfree(ver_addr);
break;
-
+ }
case ARCMSR_MESSAGE_CLEAR_RQBUFFER: {
uint8_t *pQbuffer = acb->rqbuffer;
+
+ spin_lock_irqsave(&acb->rqbuffer_lock, flags);
if (acb->acb_flags & ACB_F_IOPDATA_OVERFLOW) {
acb->acb_flags &= ~ACB_F_IOPDATA_OVERFLOW;
arcmsr_iop_message_read(acb);
@@ -2791,6 +3098,7 @@ arcmsr_iop_message_xfer(struct AdapterCo
acb->rqbuf_firstindex = 0;
acb->rqbuf_lastindex = 0;
memset(pQbuffer, 0, ARCMSR_MAX_QBUFFER);
+ spin_unlock_irqrestore(&acb->rqbuffer_lock, flags);
if (acb->fw_flag == FW_DEADLOCK) {
pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_BUS_HANG_ON;
@@ -2798,9 +3106,8 @@ arcmsr_iop_message_xfer(struct AdapterCo
pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_OK;
}
- }
break;
-
+ }
case ARCMSR_MESSAGE_CLEAR_WQBUFFER: {
uint8_t *pQbuffer = acb->wqbuffer;
if (acb->fw_flag == FW_DEADLOCK) {
@@ -2810,39 +3117,40 @@ arcmsr_iop_message_xfer(struct AdapterCo
pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_OK;
}
-
+ spin_lock_irqsave(&acb->wqbuffer_lock, flags);
if (acb->acb_flags & ACB_F_IOPDATA_OVERFLOW) {
acb->acb_flags &= ~ACB_F_IOPDATA_OVERFLOW;
arcmsr_iop_message_read(acb);
}
- acb->acb_flags |=
- (ACB_F_MESSAGE_WQBUFFER_CLEARED |
- ACB_F_MESSAGE_WQBUFFER_READED);
+ acb->acb_flags |= (ACB_F_MESSAGE_WQBUFFER_CLEARED |
+ ACB_F_MESSAGE_WQBUFFER_READED);
acb->wqbuf_firstindex = 0;
acb->wqbuf_lastindex = 0;
memset(pQbuffer, 0, ARCMSR_MAX_QBUFFER);
- }
+ spin_unlock_irqrestore(&acb->wqbuffer_lock, flags);
break;
-
+ }
case ARCMSR_MESSAGE_CLEAR_ALLQBUFFER: {
uint8_t *pQbuffer;
-
if (acb->acb_flags & ACB_F_IOPDATA_OVERFLOW) {
acb->acb_flags &= ~ACB_F_IOPDATA_OVERFLOW;
arcmsr_iop_message_read(acb);
}
- acb->acb_flags |=
- (ACB_F_MESSAGE_WQBUFFER_CLEARED
- | ACB_F_MESSAGE_RQBUFFER_CLEARED
- | ACB_F_MESSAGE_WQBUFFER_READED);
+ spin_lock_irqsave(&acb->rqbuffer_lock, flags);
+ acb->acb_flags |= ACB_F_MESSAGE_RQBUFFER_CLEARED;
acb->rqbuf_firstindex = 0;
acb->rqbuf_lastindex = 0;
- acb->wqbuf_firstindex = 0;
- acb->wqbuf_lastindex = 0;
pQbuffer = acb->rqbuffer;
memset(pQbuffer, 0, sizeof(struct QBUFFER));
+ spin_unlock_irqrestore(&acb->rqbuffer_lock, flags);
+ spin_lock_irqsave(&acb->wqbuffer_lock, flags);
+ acb->acb_flags |= (ACB_F_MESSAGE_WQBUFFER_CLEARED |
+ ACB_F_MESSAGE_WQBUFFER_READED);
+ acb->wqbuf_firstindex = 0;
+ acb->wqbuf_lastindex = 0;
pQbuffer = acb->wqbuffer;
memset(pQbuffer, 0, sizeof(struct QBUFFER));
+ spin_unlock_irqrestore(&acb->wqbuffer_lock, flags);
if (acb->fw_flag == FW_DEADLOCK) {
pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_BUS_HANG_ON;
@@ -2850,9 +3158,8 @@ arcmsr_iop_message_xfer(struct AdapterCo
pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_OK;
}
- }
break;
-
+ }
case ARCMSR_MESSAGE_RETURN_CODE_3F: {
if (acb->fw_flag == FW_DEADLOCK) {
pcmdmessagefld->cmdmessage.ReturnCode =
@@ -2862,7 +3169,7 @@ arcmsr_iop_message_xfer(struct AdapterCo
ARCMSR_MESSAGE_RETURNCODE_3F;
}
break;
- }
+ }
case ARCMSR_MESSAGE_SAY_HELLO: {
int8_t *hello_string = "Hello! I am ARCMSR";
if (acb->fw_flag == FW_DEADLOCK) {
@@ -2872,33 +3179,42 @@ arcmsr_iop_message_xfer(struct AdapterCo
pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_OK;
}
- memcpy(pcmdmessagefld->messagedatabuffer, hello_string
- , (int16_t)strlen(hello_string));
- }
+ memcpy(pcmdmessagefld->messagedatabuffer,
+ hello_string, (int16_t)strlen(hello_string));
break;
-
- case ARCMSR_MESSAGE_SAY_GOODBYE:
+ }
+ case ARCMSR_MESSAGE_SAY_GOODBYE: {
if (acb->fw_flag == FW_DEADLOCK) {
pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_BUS_HANG_ON;
+ } else {
+ pcmdmessagefld->cmdmessage.ReturnCode =
+ ARCMSR_MESSAGE_RETURNCODE_OK;
}
arcmsr_iop_parking(acb);
break;
-
- case ARCMSR_MESSAGE_FLUSH_ADAPTER_CACHE:
+ }
+ case ARCMSR_MESSAGE_FLUSH_ADAPTER_CACHE: {
if (acb->fw_flag == FW_DEADLOCK) {
pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_BUS_HANG_ON;
+ } else {
+ pcmdmessagefld->cmdmessage.ReturnCode =
+ ARCMSR_MESSAGE_RETURNCODE_OK;
}
arcmsr_flush_adapter_cache(acb);
break;
-
+ }
default:
retvalue = ARCMSR_MESSAGE_FAIL;
+ printk("unknown controlcode(%d)\n", __LINE__);
}
message_out:
- sg = scsi_sglist(cmd);
- kunmap_atomic(buffer - sg->offset);
+ if (use_sg) {
+ struct scatterlist *sg;
+ sg = scsi_sglist(cmd);
+ kunmap_atomic(buffer - sg->offset);
+ }
return retvalue;
}
@@ -2930,7 +3246,6 @@ arcmsr_handle_virtual_command(struct Ada
unsigned char inqdata[36];
char *buffer;
struct scatterlist *sg;
-
if (cmd->device->lun) {
cmd->result = (DID_TIME_OUT << 16);
cmd->scsi_done(cmd);
@@ -2949,14 +3264,11 @@ arcmsr_handle_virtual_command(struct Ada
strncpy(&inqdata[16], "RAID controller ", 16);
/* Product Identification */
strncpy(&inqdata[32], "R001", 4); /* Product Revision */
-
sg = scsi_sglist(cmd);
buffer = kmap_atomic(sg_page(sg)) + sg->offset;
-
memcpy(buffer, inqdata, sizeof(inqdata));
sg = scsi_sglist(cmd);
kunmap_atomic(buffer - sg->offset);
-
cmd->scsi_done(cmd);
}
break;
@@ -3611,19 +3923,21 @@ arcmsr_hbaD_polling_ccbdone(struct Adapt
{
bool error;
uint32_t poll_ccb_done = 0, poll_count = 0, flag_ccb, ccb_cdb_phy;
- int rtn, index, outbound_write_pointer;
+ int rtn, doneq_index, index_stripped, outbound_write_pointer;
+ unsigned long flags;
struct ARCMSR_CDB *arcmsr_cdb;
struct CommandControlBlock *pCCB;
- struct MessageUnit_D __iomem *reg =
+ struct MessageUnit_D __iomem *pmu =
(struct MessageUnit_D *)acb->pmuD;
+ spin_lock_irqsave(&acb->doneq_lock, flags);
polling_hbaD_ccb_retry:
poll_count++;
while (1) {
outbound_write_pointer =
- readl(reg->outboundlist_copy_pointer);
- index = reg->doneq_index;
- if ((outbound_write_pointer & 0xFF) == index) {
+ pmu->done_qbuffer[0].addressLow + 1;
+ doneq_index = pmu->doneq_index;
+ if ((outbound_write_pointer & 0xFFF) == (doneq_index & 0xFFF)) {
if (poll_ccb_done) {
rtn = SUCCESS;
break;
@@ -3636,17 +3950,27 @@ arcmsr_hbaD_polling_ccbdone(struct Adapt
goto polling_hbaD_ccb_retry;
}
}
- flag_ccb = reg->done_qbuffer[index].addressLow;
+ if (doneq_index & 0x4000) {
+ index_stripped = doneq_index & 0xFFF;
+ index_stripped += 1;
+ index_stripped %= ARCMSR_MAX_ARC1214_DONEQUEUE;
+ pmu->doneq_index = index_stripped ? (index_stripped | 0x4000)
+ : (index_stripped + 1);
+ } else {
+ index_stripped = doneq_index;
+ index_stripped += 1;
+ index_stripped %= ARCMSR_MAX_ARC1214_DONEQUEUE;
+ pmu->doneq_index = index_stripped ? index_stripped :
+ ((index_stripped | 0x4000) + 1);
+ }
+ doneq_index = pmu->doneq_index;
+ flag_ccb = pmu->done_qbuffer[doneq_index & 0xFFF].addressLow;
ccb_cdb_phy = (flag_ccb & 0xFFFFFFF0);
arcmsr_cdb = (struct ARCMSR_CDB *)(acb->vir2phy_offset +
ccb_cdb_phy);
pCCB = container_of(arcmsr_cdb,
struct CommandControlBlock, arcmsr_cdb);
poll_ccb_done = (pCCB == poll_ccb) ? 1 : 0;
- index++;
- index %= ARCMSR_MAX_ARC1214_POSTQUEUE;
- reg->doneq_index = index;
- /* check if command done with no error*/
if ((pCCB->acb != acb) ||
(pCCB->startdone != ARCMSR_CCB_START)) {
if (pCCB->startdone == ARCMSR_CCB_ABORTED) {
@@ -3673,6 +3997,7 @@ arcmsr_hbaD_polling_ccbdone(struct Adapt
? true : false;
arcmsr_report_ccb_state(acb, pCCB, error);
}
+ spin_unlock_irqrestore(&acb->doneq_lock, flags);
return rtn;
}
@@ -3805,7 +4130,7 @@ arcmsr_iop_confirm(struct AdapterControl
struct MessageUnit_D *reg =
(struct MessageUnit_D *)acb->pmuD;
reg->postq_index = 0;
- reg->doneq_index = 0x40FF;
+ reg->doneq_index = 0;
rwbuffer = reg->msgcode_rwbuffer;
writel(ARCMSR_SIGNATURE_SET_CONFIG, rwbuffer++);
writel(cdb_phyaddr_hi32, rwbuffer++);
@@ -3849,7 +4174,8 @@ arcmsr_wait_firmware_ready(struct Adapte
}
break;
case ACB_ADAPTER_TYPE_C: {
- struct MessageUnit_C *reg = (struct MessageUnit_C *)acb->pmuC;
+ struct MessageUnit_C *reg =
+ (struct MessageUnit_C *)acb->pmuC;
do {
firmware_state = readl(®->outbound_msgaddr1);
} while ((firmware_state &
^ permalink raw reply [flat|nested] 4+ messages in thread
* [PATCH 5/5] arcmsr: Modify ARC-1214 Inband Messages Behavior
@ 2013-02-08 6:03 NickCheng
0 siblings, 0 replies; 4+ messages in thread
From: NickCheng @ 2013-02-08 6:03 UTC (permalink / raw)
To: linux-scsi; +Cc: linux-kernel, jejb, ???
[-- Attachment #1: Type: text/plain, Size: 179 bytes --]
From: Nick Cheng <nick.cheng@areca.com.tw>
Modify ARC-1214 inband messages behavior to make up for HW seldom malfunction.
Signed-off-by: Nick Cheng <nick.cheng@areca.com.tw>
---
[-- Attachment #2: checkpatch5 --]
[-- Type: application/octet-stream, Size: 10802 bytes --]
WARNING: line over 80 characters
#26: FILE: drivers/scsi/arcmsr/arcmsr_attr.c:86:
+ if ((ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex) >= 1032) {
WARNING: line over 80 characters
#31: FILE: drivers/scsi/arcmsr/arcmsr_attr.c:91:
+ if (((ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex) + acb->rqbuf_lastindex) > 1032) {
WARNING: line over 80 characters
#32: FILE: drivers/scsi/arcmsr/arcmsr_attr.c:92:
+ memcpy(ptmpQbuffer, pQbuffer, ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex);
WARNING: line over 80 characters
#33: FILE: drivers/scsi/arcmsr/arcmsr_attr.c:93:
+ ptmpQbuffer += ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex;
WARNING: line over 80 characters
#34: FILE: drivers/scsi/arcmsr/arcmsr_attr.c:94:
+ memcpy(ptmpQbuffer, acb->rqbuffer, 1032 - (ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex));
WARNING: line over 80 characters
#35: FILE: drivers/scsi/arcmsr/arcmsr_attr.c:95:
+ acb->rqbuf_firstindex = 1032 - (ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex);
WARNING: line over 80 characters
#38: FILE: drivers/scsi/arcmsr/arcmsr_attr.c:98:
+ memcpy(ptmpQbuffer, pQbuffer, ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex);
WARNING: line over 80 characters
#39: FILE: drivers/scsi/arcmsr/arcmsr_attr.c:99:
+ ptmpQbuffer += ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex;
WARNING: line over 80 characters
#40: FILE: drivers/scsi/arcmsr/arcmsr_attr.c:100:
+ memcpy(ptmpQbuffer, acb->rqbuffer, acb->rqbuf_lastindex);
WARNING: line over 80 characters
#41: FILE: drivers/scsi/arcmsr/arcmsr_attr.c:101:
+ allxfer_len = ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex + acb->rqbuf_lastindex;
WARNING: line over 80 characters
#42: FILE: drivers/scsi/arcmsr/arcmsr_attr.c:102:
+ acb->rqbuf_firstindex = acb->rqbuf_lastindex;
WARNING: line over 80 characters
#46: FILE: drivers/scsi/arcmsr/arcmsr_attr.c:106:
+ if ((acb->rqbuf_lastindex - acb->rqbuf_firstindex) > 1032) {
WARNING: line over 80 characters
#51: FILE: drivers/scsi/arcmsr/arcmsr_attr.c:111:
+ memcpy(ptmpQbuffer, pQbuffer, acb->rqbuf_lastindex - acb->rqbuf_firstindex);
WARNING: line over 80 characters
#52: FILE: drivers/scsi/arcmsr/arcmsr_attr.c:112:
+ allxfer_len = acb->rqbuf_lastindex - acb->rqbuf_firstindex;
WARNING: line over 80 characters
#90: FILE: drivers/scsi/arcmsr/arcmsr_attr.c:137:
+ memcpy(pQbuffer, temp, ARCMSR_MAX_QBUFFER - rqbuf_lastindex);
WARNING: line over 80 characters
#92: FILE: drivers/scsi/arcmsr/arcmsr_attr.c:139:
+ rqbuf_lastindex = (rqbuf_lastindex + data_len) % ARCMSR_MAX_QBUFFER;
WARNING: line over 80 characters
#93: FILE: drivers/scsi/arcmsr/arcmsr_attr.c:140:
+ memcpy(&acb->rqbuffer[0], temp, rqbuf_lastindex);
WARNING: line over 80 characters
#96: FILE: drivers/scsi/arcmsr/arcmsr_attr.c:143:
+ rqbuf_lastindex = (rqbuf_lastindex + data_len) % ARCMSR_MAX_QBUFFER;
WARNING: Too many leading tabs - consider code refactoring
#225: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:771:
+ } else if ((temp & 0x01) == 0
WARNING: line over 80 characters
#228: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:774:
+ scsi_device_lookup(acb->host,
WARNING: Too many leading tabs - consider code refactoring
#230: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:776:
+ if (psdev != NULL) {
WARNING: line over 80 characters
#231: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:777:
+ scsi_remove_device(psdev);
WARNING: line over 80 characters
#232: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:778:
+ scsi_device_put(psdev);
WARNING: line over 80 characters
#328: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:1549:
+ while ((doneq_index & 0xFFF) != (outbound_write_pointer & 0xFFF)) {
WARNING: line over 80 characters
#339: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:1556:
+ (index_stripped | 0x4000) : (index_stripped + 1);
WARNING: line over 80 characters
#349: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:1563:
+ index_stripped ? index_stripped : ((index_stripped | 0x4000) + 1);
WARNING: line over 80 characters
#354: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:1567:
+ pmu->done_qbuffer[doneq_index & 0xFFF].addressLow;
WARNING: braces {} are not necessary for single statement blocks
#420: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2145:
+ if (!vaddr) {
+ goto leave;
+ }
WARNING: labels should not be indented
#460: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2183:
+ leave:
WARNING: Too many leading tabs - consider code refactoring
#506: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2223:
+ if ((acb->wqbuf_lastindex
WARNING: Too many leading tabs - consider code refactoring
#515: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2232:
+ } else {
WARNING: line over 80 characters
#516: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2233:
+ memcpy(iop_data, pQbuffer,
WARNING: line over 80 characters
#518: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2235:
+ - acb->wqbuf_firstindex);
WARNING: line over 80 characters
#593: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2308:
+ arcmsr_iop_message_wrote(acb);/*notice IOP the message has been written*/
WARNING: line over 80 characters
#646: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2495:
+ ? (index_stripped | 0x4000) : (index_stripped + 1);
WARNING: line over 80 characters
#654: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2501:
+ ? index_stripped : ((index_stripped | 0x4000) + 1);
WARNING: line over 80 characters
#727: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2728:
+ &acb->wqbuffer[acb->wqbuf_firstindex];
WARNING: Too many leading tabs - consider code refactoring
#728: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2729:
+ if (acb->wqbuf_firstindex + 4 >
WARNING: line over 80 characters
#730: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2731:
+ memcpy(iop_data, pQbuffer,
WARNING: line over 80 characters
#731: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2732:
+ ARCMSR_MAX_QBUFFER - acb->wqbuf_firstindex);
WARNING: line over 80 characters
#732: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2733:
+ iop_data += ARCMSR_MAX_QBUFFER
WARNING: line over 80 characters
#734: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2735:
+ acb->wqbuf_firstindex += 4;
WARNING: Too many leading tabs - consider code refactoring
#742: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2743:
+ } else {
WARNING: line over 80 characters
#743: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2744:
+ if ((acb->wqbuf_lastindex -
WARNING: Too many leading tabs - consider code refactoring
#743: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2744:
+ if ((acb->wqbuf_lastindex -
WARNING: line over 80 characters
#744: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2745:
+ acb->wqbuf_firstindex) > 4) {
WARNING: line over 80 characters
#747: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2748:
+ acb->wqbuf_firstindex
WARNING: line over 80 characters
#749: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2750:
+ acb->wqbuf_firstindex
WARNING: line over 80 characters
#750: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2751:
+ %= ARCMSR_MAX_QBUFFER;
WARNING: Too many leading tabs - consider code refactoring
#752: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2753:
+ } else {
WARNING: line over 80 characters
#755: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2756:
+ acb->wqbuf_lastindex
WARNING: line over 80 characters
#756: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2757:
+ - acb->wqbuf_firstindex);
WARNING: line over 80 characters
#758: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2759:
+ acb->wqbuf_lastindex
WARNING: line over 80 characters
#759: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2760:
+ - acb->wqbuf_firstindex;
WARNING: line over 80 characters
#760: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2761:
+ acb->wqbuf_firstindex
WARNING: line over 80 characters
#761: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2762:
+ = acb->wqbuf_lastindex;
WARNING: line over 80 characters
#782: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2783:
+ &acb->wqbuffer[acb->wqbuf_firstindex];
WARNING: Too many leading tabs - consider code refactoring
#783: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2784:
+ if ((acb->wqbuf_lastindex -
WARNING: Too many leading tabs - consider code refactoring
#792: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2793:
+ } else {
WARNING: line over 80 characters
#793: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2794:
+ memcpy(iop_data, pQbuffer,
WARNING: line over 80 characters
#795: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2796:
+ - acb->wqbuf_firstindex);
WARNING: line over 80 characters
#825: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2826:
+ allxfer_len += acb->wqbuf_lastindex
WARNING: printk() should include KERN_ facility level
#874: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2867:
+ printk("%s: ARCMSR_MESSAGE_FAIL!\n", __func__);
WARNING: printk() should include KERN_ facility level
#890: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2879:
+ printk("%s: memory not enough!\n", __func__);
WARNING: line over 80 characters
#953: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2935:
+ acb->rqbuf_lastindex - acb->rqbuf_firstindex);
WARNING: line over 80 characters
#982: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2950:
+ data_len_residual = data_len = readl(&prbuffer->data_len);
WARNING: braces {} are not necessary for single statement blocks
#985: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2953:
+ if (!vaddr) {
+ goto leave;
+ }
WARNING: labels should not be indented
#1025: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2992:
+ leave:
WARNING: line over 80 characters
#1029: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:2994:
+ memcpy(pcmdmessagefld->messagedatabuffer, ver_addr, allxfer_len);
WARNING: line over 80 characters
#1093: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:3046:
+ pQbuffer = &acb->wqbuffer[acb->wqbuf_lastindex];
WARNING: line over 80 characters
#1099: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:3052:
+ ptmpuserbuffer += (ARCMSR_MAX_QBUFFER
WARNING: line over 80 characters
#1107: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:3060:
+ acb->wqbuf_lastindex += user_len;
WARNING: printk() should include KERN_ facility level
#1282: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:3210:
+ printk("unknown controlcode(%d)\n", __LINE__);
WARNING: line over 80 characters
#1354: FILE: drivers/scsi/arcmsr/arcmsr_hba.c:3957:
+ pmu->doneq_index = index_stripped ? (index_stripped | 0x4000)
ERROR: Missing Signed-off-by: line(s)
total: 1 errors, 74 warnings, 1349 lines checked
../patch5 has style problems, please review.
If any of these errors are false positives, please report
them to the maintainer, see CHECKPATCH in MAINTAINERS.
^ permalink raw reply [flat|nested] 4+ messages in thread
* [PATCH 5/5] arcmsr: Modify ARC-1214 Inband Messages Behavior
@ 2013-02-06 8:38 NickCheng
0 siblings, 0 replies; 4+ messages in thread
From: NickCheng @ 2013-02-06 8:38 UTC (permalink / raw)
To: linux-scsi; +Cc: linux-kernel, jejb, 黃清隆
[-- Attachment #1: Type: text/plain, Size: 179 bytes --]
From: Nick Cheng <nick.cheng@areca.com.tw>
Modify ARC-1214 inband messages behavior to make up for HW seldom malfunction.
Signed-off-by: Nick Cheng <nick.cheng@areca.com.tw>
---
[-- Attachment #2: patch5 --]
[-- Type: application/octet-stream, Size: 46801 bytes --]
diff -uprN a/drivers/scsi/arcmsr/arcmsr_attr.c b/drivers/scsi/arcmsr/arcmsr_attr.c
--- a/drivers/scsi/arcmsr/arcmsr_attr.c 2013-02-06 16:29:11.434076347 +0800
+++ b/drivers/scsi/arcmsr/arcmsr_attr.c 2013-02-06 16:29:25.566410389 +0800
@@ -72,40 +72,82 @@ arcmsr_sysfs_iop_message_read(struct fil
(struct AdapterControlBlock *)host->hostdata;
uint8_t *pQbuffer,*ptmpQbuffer;
int32_t allxfer_len = 0;
+ unsigned long flags;
if (!capable(CAP_SYS_ADMIN))
return -EACCES;
/* do message unit read. */
ptmpQbuffer = (uint8_t *)buf;
- while ((acb->rqbuf_firstindex != acb->rqbuf_lastindex)
- && (allxfer_len < 1031)) {
+ spin_lock_irqsave(&acb->rqbuffer_lock, flags);
+ if (acb->rqbuf_firstindex != acb->rqbuf_lastindex) {
pQbuffer = &acb->rqbuffer[acb->rqbuf_firstindex];
- memcpy(ptmpQbuffer, pQbuffer, 1);
- acb->rqbuf_firstindex++;
- acb->rqbuf_firstindex %= ARCMSR_MAX_QBUFFER;
- ptmpQbuffer++;
- allxfer_len++;
+ if (acb->rqbuf_firstindex > acb->rqbuf_lastindex) {
+ if ((ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex) >= 1032) {
+ memcpy(ptmpQbuffer, pQbuffer, 1032);
+ acb->rqbuf_firstindex += 1032;
+ allxfer_len = 1032;
+ } else {
+ if (((ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex) + acb->rqbuf_lastindex) > 1032) {
+ memcpy(ptmpQbuffer, pQbuffer, ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex);
+ ptmpQbuffer += ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex;
+ memcpy(ptmpQbuffer, acb->rqbuffer, 1032 - (ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex));
+ acb->rqbuf_firstindex = 1032 - (ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex);
+ allxfer_len = 1032;
+ } else {
+ memcpy(ptmpQbuffer, pQbuffer, ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex);
+ ptmpQbuffer += ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex;
+ memcpy(ptmpQbuffer, acb->rqbuffer, acb->rqbuf_lastindex);
+ allxfer_len = ARCMSR_MAX_QBUFFER - acb->rqbuf_firstindex + acb->rqbuf_lastindex;
+ acb->rqbuf_firstindex = acb->rqbuf_lastindex;
+ }
+ }
+ } else {
+ if ((acb->rqbuf_lastindex - acb->rqbuf_firstindex) > 1032) {
+ memcpy(ptmpQbuffer, pQbuffer, 1032);
+ acb->rqbuf_firstindex += 1032;
+ allxfer_len = 1032;
+ } else {
+ memcpy(ptmpQbuffer, pQbuffer, acb->rqbuf_lastindex - acb->rqbuf_firstindex);
+ allxfer_len = acb->rqbuf_lastindex - acb->rqbuf_firstindex;
+ acb->rqbuf_firstindex = acb->rqbuf_lastindex;
+ }
+ }
}
if (acb->acb_flags & ACB_F_IOPDATA_OVERFLOW) {
struct QBUFFER __iomem *prbuffer;
- uint8_t __iomem *iop_data;
- int32_t iop_len;
-
- acb->acb_flags &= ~ACB_F_IOPDATA_OVERFLOW;
+ uint8_t __iomem *iop_data, *vaddr, *temp;
+ int32_t data_len_residual, data_len, rqbuf_lastindex;
+ acb->acb_flags &= ~ACB_F_IOPDATA_OVERFLOW;
+ rqbuf_lastindex = acb->rqbuf_lastindex;
prbuffer = arcmsr_get_iop_rqbuffer(acb);
- iop_data = prbuffer->data;
- iop_len = readl(&prbuffer->data_len);
- while (iop_len > 0) {
- acb->rqbuffer[acb->rqbuf_lastindex] =
- readb(iop_data);
- acb->rqbuf_lastindex++;
- acb->rqbuf_lastindex %= ARCMSR_MAX_QBUFFER;
- iop_data++;
- iop_len--;
+ iop_data = (uint8_t __iomem *)prbuffer->data;
+ data_len_residual = data_len = readl(&prbuffer->data_len);
+ if (data_len > 0) {
+ temp = vaddr = kmalloc(data_len, GFP_ATOMIC);
+ do {
+ memcpy(temp, iop_data, 4);
+ temp += 4;
+ iop_data += 4;
+ data_len_residual -= 4;
+ } while (data_len_residual > 0);
+ pQbuffer = &acb->rqbuffer[acb->rqbuf_lastindex];
+ temp = vaddr;
+ if ((rqbuf_lastindex + data_len) > ARCMSR_MAX_QBUFFER) {
+ memcpy(pQbuffer, temp, ARCMSR_MAX_QBUFFER - rqbuf_lastindex);
+ temp += (ARCMSR_MAX_QBUFFER - rqbuf_lastindex);
+ rqbuf_lastindex = (rqbuf_lastindex + data_len) % ARCMSR_MAX_QBUFFER;
+ memcpy(&acb->rqbuffer[0], temp, rqbuf_lastindex);
+ } else {
+ memcpy(pQbuffer, temp, data_len);
+ rqbuf_lastindex = (rqbuf_lastindex + data_len) % ARCMSR_MAX_QBUFFER;
+ }
+ kfree(vaddr);
}
+ acb->rqbuf_lastindex = rqbuf_lastindex;
arcmsr_iop_message_read(acb);
}
+ spin_unlock_irqrestore(&acb->rqbuffer_lock, flags);
return (allxfer_len);
}
@@ -122,6 +164,7 @@ arcmsr_sysfs_iop_message_write(struct fi
(struct AdapterControlBlock *)host->hostdata;
int32_t my_empty_len, user_len, wqbuf_firstindex, wqbuf_lastindex;
uint8_t *pQbuffer, *ptmpuserbuffer;
+ unsigned long flags;
if (!capable(CAP_SYS_ADMIN))
return -EACCES;
@@ -130,10 +173,12 @@ arcmsr_sysfs_iop_message_write(struct fi
/* do message unit write. */
ptmpuserbuffer = (uint8_t *)buf;
user_len = (int32_t)count;
+ spin_lock_irqsave(&acb->wqbuffer_lock, flags);
wqbuf_lastindex = acb->wqbuf_lastindex;
wqbuf_firstindex = acb->wqbuf_firstindex;
if (wqbuf_lastindex != wqbuf_firstindex) {
arcmsr_post_ioctldata2iop(acb);
+ spin_unlock_irqrestore(&acb->wqbuffer_lock, flags);
return 0; /*need retry*/
} else {
my_empty_len = (wqbuf_firstindex-wqbuf_lastindex - 1)
@@ -155,8 +200,10 @@ arcmsr_sysfs_iop_message_write(struct fi
~ACB_F_MESSAGE_WQBUFFER_CLEARED;
arcmsr_post_ioctldata2iop(acb);
}
+ spin_unlock_irqrestore(&acb->wqbuffer_lock, flags);
return count;
} else {
+ spin_unlock_irqrestore(&acb->wqbuffer_lock, flags);
return 0; /*need retry*/
}
}
@@ -174,6 +221,7 @@ arcmsr_sysfs_iop_message_clear(struct fi
struct AdapterControlBlock *acb =
(struct AdapterControlBlock *)host->hostdata;
uint8_t *pQbuffer;
+ unsigned long flags;
if (!capable(CAP_SYS_ADMIN))
return -EACCES;
@@ -186,10 +234,14 @@ arcmsr_sysfs_iop_message_clear(struct fi
(ACB_F_MESSAGE_WQBUFFER_CLEARED
| ACB_F_MESSAGE_RQBUFFER_CLEARED
| ACB_F_MESSAGE_WQBUFFER_READED);
+ spin_lock_irqsave(&acb->rqbuffer_lock, flags);
acb->rqbuf_firstindex = 0;
acb->rqbuf_lastindex = 0;
+ spin_unlock_irqrestore(&acb->rqbuffer_lock, flags);
+ spin_lock_irqsave(&acb->wqbuffer_lock, flags);
acb->wqbuf_firstindex = 0;
acb->wqbuf_lastindex = 0;
+ spin_unlock_irqrestore(&acb->wqbuffer_lock, flags);
pQbuffer = acb->rqbuffer;
memset(pQbuffer, 0, sizeof (struct QBUFFER));
pQbuffer = acb->wqbuffer;
diff -uprN a/drivers/scsi/arcmsr/arcmsr.h b/drivers/scsi/arcmsr/arcmsr.h
--- a/drivers/scsi/arcmsr/arcmsr.h 2013-02-06 16:29:12.026093684 +0800
+++ b/drivers/scsi/arcmsr/arcmsr.h 2013-02-06 16:29:25.869417446 +0800
@@ -63,6 +63,7 @@ struct device_attribute;
#define ARCMSR_DEFAULT_SG_ENTRIES 38
#define ARCMSR_MAX_HBB_POSTQUEUE 264
#define ARCMSR_MAX_ARC1214_POSTQUEUE 256
+#define ARCMSR_MAX_ARC1214_DONEQUEUE 257
#define ARCMSR_MAX_XFER_LEN 0x26000
#define ARCMSR_CDB_SG_PAGE_LENGTH 256
#define ARCMST_NUM_MSIX_VECTORS 4
@@ -564,7 +565,7 @@ struct OutBound_SRB {
struct MessageUnit_D {
struct InBound_SRB post_qbuffer[ARCMSR_MAX_ARC1214_POSTQUEUE];
- struct OutBound_SRB done_qbuffer[ARCMSR_MAX_ARC1214_POSTQUEUE];
+ struct OutBound_SRB done_qbuffer[ARCMSR_MAX_ARC1214_DONEQUEUE];
u16 postq_index;
u16 doneq_index;
u32 __iomem *chip_id; /*0x00004*/
@@ -617,6 +618,7 @@ struct AdapterControlBlock
spinlock_t eh_lock;
spinlock_t ccblist_lock;
spinlock_t postq_lock;
+ spinlock_t doneq_lock;
spinlock_t rqbuffer_lock;
spinlock_t wqbuffer_lock;
union {
@@ -670,15 +672,15 @@ struct AdapterControlBlock
unsigned int uncache_size;
uint8_t rqbuffer[ARCMSR_MAX_QBUFFER];
/* data collection buffer for read from 80331 */
- int32_t rqbuf_firstindex;
+ uint32_t rqbuf_firstindex;
/* first of read buffer */
- int32_t rqbuf_lastindex;
+ uint32_t rqbuf_lastindex;
/* last of read buffer */
uint8_t wqbuffer[ARCMSR_MAX_QBUFFER];
/* data collection buffer for write to 80331 */
- int32_t wqbuf_firstindex;
+ uint32_t wqbuf_firstindex;
/* first of write buffer */
- int32_t wqbuf_lastindex;
+ uint32_t wqbuf_lastindex;
/* last of write buffer */
uint8_t devstate[ARCMSR_MAX_TARGETID][ARCMSR_MAX_TARGETLUN];
/* id0 ..... id15, lun0...lun7 */
diff -uprN a/drivers/scsi/arcmsr/arcmsr_hba.c b/drivers/scsi/arcmsr/arcmsr_hba.c
--- a/drivers/scsi/arcmsr/arcmsr_hba.c 2013-02-06 16:29:12.453104788 +0800
+++ b/drivers/scsi/arcmsr/arcmsr_hba.c 2013-02-06 16:29:26.195427363 +0800
@@ -768,18 +768,18 @@ arcmsr_message_isr_bh_fn(struct work_str
(diff & 0x01) == 1) {
scsi_add_device(acb->host,
0, target, lun);
- } else if ((temp & 0x01) == 0
- && (diff & 0x01) == 1) {
- psdev =
- scsi_device_lookup(acb->host,
- 0, target, lun);
- if (psdev != NULL) {
- scsi_remove_device(psdev);
- scsi_device_put(psdev);
+ } else if ((temp & 0x01) == 0
+ && (diff & 0x01) == 1) {
+ psdev =
+ scsi_device_lookup(acb->host,
+ 0, target, lun);
+ if (psdev != NULL) {
+ scsi_remove_device(psdev);
+ scsi_device_put(psdev);
+ }
}
- }
- temp >>= 1;
- diff >>= 1;
+ temp >>= 1;
+ diff >>= 1;
}
}
devicemap++;
@@ -813,20 +813,20 @@ arcmsr_message_isr_bh_fn(struct work_str
lun < ARCMSR_MAX_TARGETLUN;
lun++) {
if ((temp & 0x01) == 1 &&
- (diff & 0x01) == 1) {
+ (diff & 0x01) == 1) {
scsi_add_device(acb->host,
0, target, lun);
} else if ((temp & 0x01) == 0
- && (diff & 0x01) == 1) {
+ && (diff & 0x01) == 1) {
psdev = scsi_device_lookup(acb->host,
0, target, lun);
if (psdev != NULL) {
scsi_remove_device(psdev);
scsi_device_put(psdev);
}
- }
- temp >>= 1;
- diff >>= 1;
+ }
+ temp >>= 1;
+ diff >>= 1;
}
}
devicemap++;
@@ -869,10 +869,10 @@ arcmsr_message_isr_bh_fn(struct work_str
if (psdev != NULL) {
scsi_remove_device(psdev);
scsi_device_put(psdev);
+ }
}
- }
- temp >>= 1;
- diff >>= 1;
+ temp >>= 1;
+ diff >>= 1;
}
}
devicemap++;
@@ -915,10 +915,10 @@ arcmsr_message_isr_bh_fn(struct work_str
if (psdev != NULL) {
scsi_remove_device(psdev);
scsi_device_put(psdev);
+ }
}
- }
- temp >>= 1;
- diff >>= 1;
+ temp >>= 1;
+ diff >>= 1;
}
}
devicemap++;
@@ -981,8 +981,8 @@ arcmsr_resume(struct pci_dev *pdev)
error = pci_set_dma_mask(pdev, DMA_BIT_MASK(32));
if (error) {
pr_warn("scsi%d: No suitable DMA mask available\n",
- host->host_no);
- goto controller_unregister;
+ host->host_no);
+ goto controller_unregister;
}
}
pci_set_master(pdev);
@@ -1097,6 +1097,7 @@ static int arcmsr_probe(struct pci_dev *
spin_lock_init(&acb->eh_lock);
spin_lock_init(&acb->ccblist_lock);
spin_lock_init(&acb->postq_lock);
+ spin_lock_init(&acb->doneq_lock);
spin_lock_init(&acb->rqbuffer_lock);
spin_lock_init(&acb->wqbuffer_lock);
acb->acb_flags |= (ACB_F_MESSAGE_WQBUFFER_CLEARED |
@@ -1541,33 +1542,29 @@ arcmsr_done4abort_postqueue(struct Adapt
uint32_t doneq_index, index_stripped, addressLow, residual;
bool error;
struct CommandControlBlock *pCCB;
- outbound_write_pointer =
- readl(pmu->outboundlist_copy_pointer);
+ outbound_write_pointer = pmu->done_qbuffer[0].addressLow + 1;
doneq_index = pmu->doneq_index;
residual = atomic_read(&acb->ccboutstandingcount);
for (i = 0; i < residual; i++) {
- while ((doneq_index & 0xFF) !=
- (outbound_write_pointer & 0xFF)) {
+ while ((doneq_index & 0xFFF) != (outbound_write_pointer & 0xFFF)) {
if (doneq_index & 0x4000) {
- index_stripped = doneq_index & 0xFF;
+ index_stripped = doneq_index & 0xFFF;
index_stripped += 1;
index_stripped %=
- ARCMSR_MAX_ARC1214_POSTQUEUE;
+ ARCMSR_MAX_ARC1214_DONEQUEUE;
pmu->doneq_index = index_stripped ?
- (index_stripped | 0x4000) :
- index_stripped;
+ (index_stripped | 0x4000) : (index_stripped + 1);
} else {
index_stripped = doneq_index;
index_stripped += 1;
index_stripped %=
- ARCMSR_MAX_ARC1214_POSTQUEUE;
+ ARCMSR_MAX_ARC1214_DONEQUEUE;
pmu->doneq_index =
- index_stripped ? index_stripped :
- (index_stripped | 0x4000);
+ index_stripped ? index_stripped : ((index_stripped | 0x4000) + 1);
}
doneq_index = pmu->doneq_index;
addressLow =
- pmu->done_qbuffer[doneq_index & 0xFF].addressLow;
+ pmu->done_qbuffer[doneq_index & 0xFFF].addressLow;
ccb_cdb_phy = (addressLow & 0xFFFFFFF0);
pARCMSR_CDB = (struct ARCMSR_CDB *)
(acb->vir2phy_offset + ccb_cdb_phy);
@@ -1582,7 +1579,7 @@ arcmsr_done4abort_postqueue(struct Adapt
}
mdelay(10);
outbound_write_pointer =
- readl(pmu->outboundlist_copy_pointer);
+ pmu->done_qbuffer[0].addressLow + 1;
doneq_index = pmu->doneq_index;
}
pmu->postq_index = 0;
@@ -2004,12 +2001,14 @@ arcmsr_iop_message_read(struct AdapterCo
case ACB_ADAPTER_TYPE_C: {
struct MessageUnit_C __iomem *reg = acb->pmuC;
writel(ARCMSR_HBCMU_DRV2IOP_DATA_READ_OK, ®->inbound_doorbell);
+ /*readl(®->inbound_doorbell);*/
}
break;
case ACB_ADAPTER_TYPE_D: {
struct MessageUnit_D __iomem *reg = acb->pmuD;
writel(ARCMSR_ARC1214_DRV2IOP_DATA_OUT_READ,
reg->inbound_doorbell);
+ /*readl(reg->inbound_doorbell);*/
break;
}
}
@@ -2127,36 +2126,61 @@ struct QBUFFER __iomem
void
arcmsr_iop2drv_data_wrote_handle(struct AdapterControlBlock *acb)
{
- uint8_t __iomem *iop_data;
- struct QBUFFER __iomem *prbuffer;
- struct QBUFFER *pQbuffer;
- int32_t my_empty_len, iop_len, rqbuf_firstindex, rqbuf_lastindex;
+ uint8_t __iomem *iop_data, *pQbuffer, *vaddr, *temp;
+ int32_t buf_empty_len, data_len, data_len_residual;
+ uint32_t rqbuf_firstindex, rqbuf_lastindex;
unsigned long flags;
-
+ struct QBUFFER __iomem *prbuffer;
spin_lock_irqsave(&acb->rqbuffer_lock, flags);
rqbuf_lastindex = acb->rqbuf_lastindex;
rqbuf_firstindex = acb->rqbuf_firstindex;
prbuffer = arcmsr_get_iop_rqbuffer(acb);
iop_data = (uint8_t __iomem *)prbuffer->data;
- iop_len = prbuffer->data_len;
- my_empty_len = (rqbuf_firstindex - rqbuf_lastindex - 1) &
- (ARCMSR_MAX_QBUFFER - 1);
-
- if (my_empty_len >= iop_len) {
- while (iop_len > 0) {
- pQbuffer = (struct QBUFFER *)
- &acb->rqbuffer[rqbuf_lastindex];
- memcpy(pQbuffer, iop_data, 1);
- rqbuf_lastindex++;
- rqbuf_lastindex %= ARCMSR_MAX_QBUFFER;
- iop_data++;
- iop_len--;
+ data_len_residual = data_len = readl(&prbuffer->data_len);
+ buf_empty_len = (rqbuf_firstindex - rqbuf_lastindex - 1) &
+ (ARCMSR_MAX_QBUFFER - 1);
+ if (buf_empty_len >= data_len) {
+ if (data_len > 0) {
+ temp = vaddr = kmalloc(data_len, GFP_ATOMIC);
+ if (!vaddr) {
+ goto leave;
+ }
+ acb->acb_flags &= ~ACB_F_IOPDATA_OVERFLOW;
+ while (data_len_residual >= 4) {
+ memcpy(temp, iop_data, 4);
+ temp += 4;
+ iop_data += 4;
+ data_len_residual -= 4;
+ }
+ if ((data_len_residual > 0) &&
+ (data_len_residual < 4)) {
+ memcpy(temp, iop_data, data_len_residual);
+ }
+ pQbuffer = &acb->rqbuffer[rqbuf_lastindex];
+ temp = vaddr;
+ if ((rqbuf_lastindex + data_len) >
+ ARCMSR_MAX_QBUFFER) {
+ memcpy(pQbuffer, temp,
+ ARCMSR_MAX_QBUFFER - rqbuf_lastindex);
+ temp += (ARCMSR_MAX_QBUFFER -
+ rqbuf_lastindex);
+ rqbuf_lastindex = (rqbuf_lastindex + data_len)
+ % ARCMSR_MAX_QBUFFER;
+ memcpy(&acb->rqbuffer[0], temp,
+ rqbuf_lastindex);
+ } else {
+ memcpy(pQbuffer, temp, data_len);
+ rqbuf_lastindex = (rqbuf_lastindex + data_len)
+ % ARCMSR_MAX_QBUFFER;
+ }
+ kfree(vaddr);
+ acb->rqbuf_lastindex = rqbuf_lastindex;
+ arcmsr_iop_message_read(acb);
}
- acb->rqbuf_lastindex = rqbuf_lastindex;
- arcmsr_iop_message_read(acb);
} else {
acb->acb_flags |= ACB_F_IOPDATA_OVERFLOW;
}
+ leave:
spin_unlock_irqrestore(&acb->rqbuffer_lock, flags);
}
@@ -2164,7 +2188,6 @@ void
arcmsr_iop2drv_data_read_handle(struct AdapterControlBlock *acb)
{
unsigned long flags;
-
spin_lock_irqsave(&acb->wqbuffer_lock, flags);
acb->acb_flags |= ACB_F_MESSAGE_WQBUFFER_READED;
if (acb->wqbuf_firstindex != acb->wqbuf_lastindex) {
@@ -2176,18 +2199,113 @@ arcmsr_iop2drv_data_read_handle(struct A
acb->acb_flags &= (~ACB_F_MESSAGE_WQBUFFER_READED);
pwbuffer = arcmsr_get_iop_wqbuffer(acb);
iop_data = (uint8_t __iomem *)pwbuffer->data;
-
- while ((acb->wqbuf_firstindex != acb->wqbuf_lastindex) &&
- (allxfer_len < 124)) {
- pQbuffer = &acb->wqbuffer[acb->wqbuf_firstindex];
- memcpy(iop_data, pQbuffer, 1);
- acb->wqbuf_firstindex++;
- acb->wqbuf_firstindex %= ARCMSR_MAX_QBUFFER;
- iop_data++;
- allxfer_len++;
+ if (acb->wqbuf_firstindex > acb->wqbuf_lastindex) {
+ if ((ARCMSR_MAX_QBUFFER - acb->wqbuf_firstindex) >= 4) {
+ do {
+ pQbuffer =
+ &acb->wqbuffer[acb->wqbuf_firstindex];
+ if (acb->wqbuf_firstindex + 4
+ > ARCMSR_MAX_QBUFFER) {
+ memcpy(iop_data, pQbuffer,
+ ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_firstindex);
+ iop_data += ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex += 4;
+ acb->wqbuf_firstindex %=
+ ARCMSR_MAX_QBUFFER;
+ memcpy(iop_data,
+ &acb->wqbuffer[0],
+ acb->wqbuf_firstindex);
+ iop_data +=
+ acb->wqbuf_firstindex;
+ } else {
+ if ((acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex) > 4) {
+ memcpy(iop_data,
+ pQbuffer, 4);
+ acb->wqbuf_firstindex
+ += 4;
+ acb->wqbuf_firstindex
+ %= ARCMSR_MAX_QBUFFER;
+ iop_data += 4;
+ } else {
+ memcpy(iop_data, pQbuffer,
+ acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex);
+ allxfer_len +=
+ acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex =
+ acb->wqbuf_lastindex;
+ break;
+ }
+ }
+ allxfer_len += 4;
+ } while ((acb->wqbuf_firstindex !=
+ acb->wqbuf_lastindex) && (allxfer_len < 124));
+ } else {
+ pQbuffer =
+ &acb->wqbuffer[acb->wqbuf_firstindex];
+ memcpy(iop_data, pQbuffer, ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_firstindex);
+ iop_data += ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_firstindex;
+ allxfer_len = ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex = 0;
+ do {
+ pQbuffer =
+ &acb->wqbuffer[acb->wqbuf_firstindex];
+ if ((acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex) > 4) {
+ memcpy(iop_data, pQbuffer, 4);
+ acb->wqbuf_firstindex += 4;
+ acb->wqbuf_firstindex %=
+ ARCMSR_MAX_QBUFFER;
+ iop_data += 4;
+ } else {
+ memcpy(iop_data, pQbuffer,
+ acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex);
+ allxfer_len +=
+ acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex =
+ acb->wqbuf_lastindex;
+ break;
+ }
+ allxfer_len += 4;
+ } while ((acb->wqbuf_firstindex !=
+ acb->wqbuf_lastindex) && (allxfer_len < 124));
+ }
+ } else {
+ do {
+ pQbuffer =
+ &acb->wqbuffer[acb->wqbuf_firstindex];
+ if ((acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex) > 4) {
+ memcpy(iop_data, pQbuffer, 4);
+ acb->wqbuf_firstindex += 4;
+ acb->wqbuf_firstindex %=
+ ARCMSR_MAX_QBUFFER;
+ iop_data += 4;
+ } else {
+ memcpy(iop_data, pQbuffer,
+ acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex);
+ allxfer_len += acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex =
+ acb->wqbuf_lastindex;
+ break;
+ }
+ allxfer_len += 4;
+ } while ((acb->wqbuf_firstindex !=
+ acb->wqbuf_lastindex) && (allxfer_len < 124));
}
- pwbuffer->data_len = allxfer_len;
- arcmsr_iop_message_wrote(acb);
+ writel(allxfer_len, &pwbuffer->data_len);
+ arcmsr_iop_message_wrote(acb);/*notice IOP the message has been written*/
}
if (acb->wqbuf_firstindex == acb->wqbuf_lastindex) {
acb->acb_flags |= ACB_F_MESSAGE_WQBUFFER_CLEARED;
@@ -2200,7 +2318,6 @@ arcmsr_hbaA_doorbell_isr(struct AdapterC
{
uint32_t outbound_doorbell;
struct MessageUnit_A __iomem *reg = acb->pmuA;
-
outbound_doorbell = readl(®->outbound_doorbell);
do {
writel(outbound_doorbell, ®->outbound_doorbell);
@@ -2223,7 +2340,6 @@ arcmsr_hbaC_doorbell_isr(struct AdapterC
uint32_t outbound_doorbell;
struct MessageUnit_C __iomem *reg =
(struct MessageUnit_C *)pACB->pmuC;
-
outbound_doorbell = readl(®->outbound_doorbell);
do {
if (outbound_doorbell &
@@ -2255,10 +2371,6 @@ arcmsr_hbaD_doorbell_isr(struct AdapterC
(struct MessageUnit_D *)pACB->pmuD;
outbound_doorbell = readl(pmu->outbound_doorbell);
- if (unlikely(!outbound_doorbell)) {
- arcmsr_iop2drv_data_wrote_handle(pACB);
- arcmsr_iop2drv_data_read_handle(pACB);
- }
do {
writel(outbound_doorbell, pmu->outbound_doorbell);
if (outbound_doorbell &
@@ -2367,28 +2479,30 @@ arcmsr_hbaD_postqueue_isr(struct Adapter
struct MessageUnit_D __iomem *pmu;
struct ARCMSR_CDB *arcmsr_cdb;
struct CommandControlBlock *ccb;
+ unsigned long flags;
+ spin_lock_irqsave(&acb->doneq_lock, flags);
pmu = (struct MessageUnit_D *)acb->pmuD;
- outbound_write_pointer = readl(pmu->outboundlist_copy_pointer);
+ outbound_write_pointer = pmu->done_qbuffer[0].addressLow + 1;
doneq_index = pmu->doneq_index;
- if ((doneq_index & 0xFF) != (outbound_write_pointer & 0xFF)) {
+ if ((doneq_index & 0xFFF) != (outbound_write_pointer & 0xFFF)) {
do {
if (doneq_index & 0x4000) {
- index_stripped = doneq_index & 0xFF;
+ index_stripped = doneq_index & 0xFFF;
index_stripped += 1;
- index_stripped %= ARCMSR_MAX_ARC1214_POSTQUEUE;
+ index_stripped %= ARCMSR_MAX_ARC1214_DONEQUEUE;
pmu->doneq_index = index_stripped
- ? (index_stripped | 0x4000) : index_stripped;
+ ? (index_stripped | 0x4000) : (index_stripped + 1);
} else {
index_stripped = doneq_index;
index_stripped += 1;
- index_stripped %= ARCMSR_MAX_ARC1214_POSTQUEUE;
+ index_stripped %= ARCMSR_MAX_ARC1214_DONEQUEUE;
pmu->doneq_index = index_stripped
- ? index_stripped : (index_stripped | 0x4000);
+ ? index_stripped : ((index_stripped | 0x4000) + 1);
}
doneq_index = pmu->doneq_index;
addressLow =
- pmu->done_qbuffer[doneq_index & 0xFF].addressLow;
+ pmu->done_qbuffer[doneq_index & 0xFFF].addressLow;
ccb_cdb_phy = (addressLow & 0xFFFFFFF0);
arcmsr_cdb = (struct ARCMSR_CDB *)(acb->vir2phy_offset
+ ccb_cdb_phy);
@@ -2399,12 +2513,13 @@ arcmsr_hbaD_postqueue_isr(struct Adapter
arcmsr_drain_donequeue(acb, ccb, error);
writel(doneq_index,
pmu->outboundlist_read_pointer);
- } while ((doneq_index & 0xFF) !=
- (outbound_write_pointer & 0xFF));
+ } while ((doneq_index & 0xFFF) !=
+ (outbound_write_pointer & 0xFFF));
}
writel(ARCMSR_ARC1214_OUTBOUND_LIST_INTERRUPT_CLEAR,
pmu->outboundlist_interrupt_cause);
readl(pmu->outboundlist_interrupt_cause);
+ spin_unlock_irqrestore(&acb->doneq_lock, flags);
}
static void
@@ -2457,7 +2572,6 @@ arcmsr_hbaA_handle_isr(struct AdapterCon
}
do {
writel(outbound_intstatus, ®->outbound_intstatus);
- readl(®->outbound_intstatus);
if (outbound_intstatus & ARCMSR_MU_OUTBOUND_DOORBELL_INT)
arcmsr_hbaA_doorbell_isr(acb);
if (outbound_intstatus & ARCMSR_MU_OUTBOUND_POSTQUEUE_INT)
@@ -2597,29 +2711,131 @@ arcmsr_iop_parking(struct AdapterControl
void
arcmsr_post_ioctldata2iop(struct AdapterControlBlock *acb)
{
- uint8_t __iomem *iop_data;
- uint8_t *pQbuffer;
- int32_t wqbuf_firstindex, wqbuf_lastindex;
- int32_t allxfer_len = 0;
- struct QBUFFER __iomem *pwbuffer;
- pwbuffer = arcmsr_get_iop_wqbuffer(acb);
- iop_data = (uint8_t __iomem *)pwbuffer->data;
if (acb->acb_flags & ACB_F_MESSAGE_WQBUFFER_READED) {
- acb->acb_flags &= (~ACB_F_MESSAGE_WQBUFFER_READED);
- wqbuf_firstindex = acb->wqbuf_firstindex;
- wqbuf_lastindex = acb->wqbuf_lastindex;
- while ((wqbuf_firstindex != wqbuf_lastindex)
- && (allxfer_len < 124)) {
- pQbuffer = &acb->wqbuffer[wqbuf_firstindex];
- memcpy(iop_data, pQbuffer, 1);
- wqbuf_firstindex++;
- wqbuf_firstindex %= ARCMSR_MAX_QBUFFER;
- iop_data++;
- allxfer_len++;
- }
- acb->wqbuf_firstindex = wqbuf_firstindex;
- pwbuffer->data_len = allxfer_len;
- arcmsr_iop_message_wrote(acb);
+ if (acb->wqbuf_firstindex != acb->wqbuf_lastindex) {
+ uint8_t *pQbuffer;
+ uint8_t __iomem *iop_data;
+ int32_t allxfer_len = 0;
+ struct QBUFFER __iomem *pwbuffer;
+ acb->acb_flags &= (~ACB_F_MESSAGE_WQBUFFER_READED);
+ pwbuffer = arcmsr_get_iop_wqbuffer(acb);
+ iop_data = (uint8_t __iomem *)pwbuffer->data;
+ if (acb->wqbuf_firstindex > acb->wqbuf_lastindex) {
+ if ((ARCMSR_MAX_QBUFFER -
+ acb->wqbuf_firstindex) >= 4) {
+ do {
+ pQbuffer =
+ &acb->wqbuffer[acb->wqbuf_firstindex];
+ if (acb->wqbuf_firstindex + 4 >
+ ARCMSR_MAX_QBUFFER) {
+ memcpy(iop_data, pQbuffer,
+ ARCMSR_MAX_QBUFFER - acb->wqbuf_firstindex);
+ iop_data += ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex += 4;
+ acb->wqbuf_firstindex %=
+ ARCMSR_MAX_QBUFFER;
+ memcpy(iop_data,
+ &acb->wqbuffer[0],
+ acb->wqbuf_firstindex);
+ iop_data +=
+ acb->wqbuf_firstindex;
+ } else {
+ if ((acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex) > 4) {
+ memcpy(iop_data,
+ pQbuffer, 4);
+ acb->wqbuf_firstindex
+ += 4;
+ acb->wqbuf_firstindex
+ %= ARCMSR_MAX_QBUFFER;
+ iop_data += 4;
+ } else {
+ memcpy(iop_data,
+ pQbuffer,
+ acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex);
+ allxfer_len +=
+ acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex
+ = acb->wqbuf_lastindex;
+ break;
+ }
+ }
+ allxfer_len += 4;
+ } while ((acb->wqbuf_firstindex !=
+ acb->wqbuf_lastindex) &&
+ (allxfer_len < 124));
+ } else {
+ pQbuffer =
+ &acb->wqbuffer[acb->wqbuf_firstindex];
+ memcpy(iop_data, pQbuffer,
+ ARCMSR_MAX_QBUFFER -
+ acb->wqbuf_firstindex);
+ iop_data += ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_firstindex;
+ allxfer_len = ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex = 0;
+ do {
+ pQbuffer =
+ &acb->wqbuffer[acb->wqbuf_firstindex];
+ if ((acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex) > 4) {
+ memcpy(iop_data,
+ pQbuffer, 4);
+ acb->wqbuf_firstindex
+ += 4;
+ acb->wqbuf_firstindex
+ %= ARCMSR_MAX_QBUFFER;
+ iop_data += 4;
+ } else {
+ memcpy(iop_data, pQbuffer,
+ acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex);
+ allxfer_len +=
+ acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex =
+ acb->wqbuf_lastindex;
+ break;
+ }
+ allxfer_len += 4;
+ } while ((acb->wqbuf_firstindex
+ != acb->wqbuf_lastindex) &&
+ (allxfer_len < 124));
+ }
+ } else {
+ do {
+ pQbuffer =
+ &acb->wqbuffer[acb->wqbuf_firstindex];
+ if ((acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex) > 4) {
+ memcpy(iop_data,
+ pQbuffer, 4);
+ acb->wqbuf_firstindex
+ += 4;
+ acb->wqbuf_firstindex
+ %= ARCMSR_MAX_QBUFFER;
+ iop_data += 4;
+ } else {
+ memcpy(iop_data, pQbuffer,
+ acb->wqbuf_lastindex -
+ acb->wqbuf_firstindex);
+ allxfer_len += acb->wqbuf_lastindex
+ - acb->wqbuf_firstindex;
+ acb->wqbuf_firstindex =
+ acb->wqbuf_lastindex;
+ break;
+ }
+ allxfer_len += 4;
+ } while ((acb->wqbuf_firstindex !=
+ acb->wqbuf_lastindex) && (allxfer_len < 124));
+ }
+ writel(allxfer_len, &pwbuffer->data_len);
+ arcmsr_iop_message_wrote(acb);
+ }
}
}
@@ -2627,74 +2843,155 @@ static int
arcmsr_iop_message_xfer(struct AdapterControlBlock *acb,
struct scsi_cmnd *cmd)
{
- struct CMD_MESSAGE_FIELD *pcmdmessagefld;
- int retvalue = 0, transfer_len = 0;
char *buffer;
+ unsigned short use_sg;
+ int retvalue = 0, transfer_len = 0;
+ unsigned long flags;
+ struct CMD_MESSAGE_FIELD *pcmdmessagefld;
+ uint32_t controlcode = (uint32_t)cmd->cmnd[5] << 24 |
+ (uint32_t)cmd->cmnd[6] << 16 |
+ (uint32_t)cmd->cmnd[7] << 8 |
+ (uint32_t)cmd->cmnd[8];
struct scatterlist *sg;
- uint32_t controlcode = (uint32_t)cmd->cmnd[5] << 24
- |(uint32_t)cmd->cmnd[6] << 16
- |(uint32_t)cmd->cmnd[7] << 8
- | (uint32_t)cmd->cmnd[8];
+
+ use_sg = scsi_sg_count(cmd);
sg = scsi_sglist(cmd);
buffer = kmap_atomic(sg_page(sg)) + sg->offset;
- if (scsi_sg_count(cmd) > 1) {
+ if (use_sg > 1) {
retvalue = ARCMSR_MESSAGE_FAIL;
goto message_out;
}
transfer_len += sg->length;
-
if (transfer_len > sizeof(struct CMD_MESSAGE_FIELD)) {
retvalue = ARCMSR_MESSAGE_FAIL;
+ printk("%s: ARCMSR_MESSAGE_FAIL!\n", __func__);
goto message_out;
}
pcmdmessagefld = (struct CMD_MESSAGE_FIELD *)buffer;
switch (controlcode) {
-
case ARCMSR_MESSAGE_READ_RQBUFFER: {
unsigned char *ver_addr;
uint8_t *pQbuffer, *ptmpQbuffer;
- int32_t allxfer_len = 0;
- unsigned long flags;
-
+ uint32_t allxfer_len = 0;
ver_addr = kmalloc(1032, GFP_ATOMIC);
if (!ver_addr) {
retvalue = ARCMSR_MESSAGE_FAIL;
+ printk("%s: memory not enough!\n", __func__);
goto message_out;
}
ptmpQbuffer = ver_addr;
spin_lock_irqsave(&acb->rqbuffer_lock, flags);
- while ((acb->rqbuf_firstindex != acb->rqbuf_lastindex)
- && (allxfer_len < 1031)) {
+ if (acb->rqbuf_firstindex != acb->rqbuf_lastindex) {
pQbuffer = &acb->rqbuffer[acb->rqbuf_firstindex];
- memcpy(ptmpQbuffer, pQbuffer, 1);
- acb->rqbuf_firstindex++;
- acb->rqbuf_firstindex %= ARCMSR_MAX_QBUFFER;
- ptmpQbuffer++;
- allxfer_len++;
+ if (acb->rqbuf_firstindex > acb->rqbuf_lastindex) {
+ if ((ARCMSR_MAX_QBUFFER -
+ acb->rqbuf_firstindex) >= 1032) {
+ memcpy(ptmpQbuffer, pQbuffer, 1032);
+ acb->rqbuf_firstindex += 1032;
+ allxfer_len = 1032;
+ } else {
+ if (((ARCMSR_MAX_QBUFFER -
+ acb->rqbuf_firstindex) +
+ acb->rqbuf_lastindex) > 1032) {
+ memcpy(ptmpQbuffer,
+ pQbuffer, ARCMSR_MAX_QBUFFER
+ - acb->rqbuf_firstindex);
+ ptmpQbuffer +=
+ ARCMSR_MAX_QBUFFER -
+ acb->rqbuf_firstindex;
+ memcpy(ptmpQbuffer,
+ acb->rqbuffer, 1032 -
+ (ARCMSR_MAX_QBUFFER
+ - acb->rqbuf_firstindex));
+ acb->rqbuf_firstindex =
+ 1032 - (ARCMSR_MAX_QBUFFER
+ - acb->rqbuf_firstindex);
+ allxfer_len = 1032;
+ } else {
+ memcpy(ptmpQbuffer,
+ pQbuffer, ARCMSR_MAX_QBUFFER
+ - acb->rqbuf_firstindex);
+ ptmpQbuffer +=
+ ARCMSR_MAX_QBUFFER -
+ acb->rqbuf_firstindex;
+ memcpy(ptmpQbuffer,
+ acb->rqbuffer,
+ acb->rqbuf_lastindex);
+ allxfer_len = ARCMSR_MAX_QBUFFER
+ - acb->rqbuf_firstindex +
+ acb->rqbuf_lastindex;
+ acb->rqbuf_firstindex =
+ acb->rqbuf_lastindex;
+ }
+ }
+ } else {
+ if ((acb->rqbuf_lastindex -
+ acb->rqbuf_firstindex) > 1032) {
+ memcpy(ptmpQbuffer, pQbuffer, 1032);
+ acb->rqbuf_firstindex += 1032;
+ allxfer_len = 1032;
+ } else {
+ memcpy(ptmpQbuffer, pQbuffer,
+ acb->rqbuf_lastindex - acb->rqbuf_firstindex);
+ allxfer_len = acb->rqbuf_lastindex
+ - acb->rqbuf_firstindex;
+ acb->rqbuf_firstindex =
+ acb->rqbuf_lastindex;
+ }
+ }
}
if (acb->acb_flags & ACB_F_IOPDATA_OVERFLOW) {
-
struct QBUFFER __iomem *prbuffer;
- uint8_t __iomem *iop_data;
- int32_t iop_len;
-
- acb->acb_flags &= ~ACB_F_IOPDATA_OVERFLOW;
+ uint8_t __iomem *iop_data, *vaddr, *temp;
+ uint32_t data_len_residual, data_len, rqbuf_lastindex;
+ rqbuf_lastindex = acb->rqbuf_lastindex;
prbuffer = arcmsr_get_iop_rqbuffer(acb);
- iop_data = prbuffer->data;
- iop_len = readl(&prbuffer->data_len);
- while (iop_len > 0) {
- acb->rqbuffer[acb->rqbuf_lastindex] =
- readb(iop_data);
- acb->rqbuf_lastindex++;
- acb->rqbuf_lastindex %= ARCMSR_MAX_QBUFFER;
- iop_data++;
- iop_len--;
+ iop_data = (uint8_t __iomem *)prbuffer->data;
+ data_len_residual = data_len = readl(&prbuffer->data_len);
+ if (data_len > 0) {
+ temp = vaddr = kmalloc(data_len, GFP_ATOMIC);
+ if (!vaddr) {
+ goto leave;
+ }
+ acb->acb_flags &= ~ACB_F_IOPDATA_OVERFLOW;
+ while (data_len_residual >= 4) {
+ memcpy(temp, iop_data, 4);
+ temp += 4;
+ iop_data += 4;
+ data_len_residual -= 4;
+ }
+ if ((data_len_residual > 0) &&
+ (data_len_residual < 4)) {
+ memcpy(temp, iop_data,
+ data_len_residual);
+ }
+ pQbuffer =
+ &acb->rqbuffer[acb->rqbuf_lastindex];
+ temp = vaddr;
+ if ((rqbuf_lastindex + data_len) >
+ ARCMSR_MAX_QBUFFER) {
+ memcpy(pQbuffer, temp,
+ ARCMSR_MAX_QBUFFER - rqbuf_lastindex);
+ temp += (ARCMSR_MAX_QBUFFER -
+ rqbuf_lastindex);
+ rqbuf_lastindex = (rqbuf_lastindex +
+ data_len) % ARCMSR_MAX_QBUFFER;
+ memcpy(&acb->rqbuffer[0],
+ temp, rqbuf_lastindex);
+ } else {
+ memcpy(pQbuffer, temp, data_len);
+ rqbuf_lastindex =
+ (rqbuf_lastindex + data_len) %
+ ARCMSR_MAX_QBUFFER;
+ }
+ kfree(vaddr);
+ acb->rqbuf_lastindex = rqbuf_lastindex;
+ arcmsr_iop_message_read(acb);
}
- arcmsr_iop_message_read(acb);
}
+ leave:
spin_unlock_irqrestore(&acb->rqbuffer_lock, flags);
- memcpy(pcmdmessagefld->messagedatabuffer,
- ver_addr, allxfer_len);
+ memcpy(pcmdmessagefld->messagedatabuffer, ver_addr, allxfer_len);
pcmdmessagefld->cmdmessage.Length = allxfer_len;
if (acb->fw_flag == FW_DEADLOCK) {
pcmdmessagefld->cmdmessage.ReturnCode =
@@ -2704,26 +3001,23 @@ arcmsr_iop_message_xfer(struct AdapterCo
ARCMSR_MESSAGE_RETURNCODE_OK;
}
kfree(ver_addr);
- }
break;
-
+ }
case ARCMSR_MESSAGE_WRITE_WQBUFFER: {
unsigned char *ver_addr;
int32_t my_empty_len, user_len, wqbuf_firstindex,
wqbuf_lastindex;
uint8_t *pQbuffer, *ptmpuserbuffer;
- unsigned long flags;
-
ver_addr = kmalloc(1032, GFP_ATOMIC);
if (!ver_addr) {
retvalue = ARCMSR_MESSAGE_FAIL;
goto message_out;
}
if (acb->fw_flag == FW_DEADLOCK) {
- pcmdmessagefld->cmdmessage.ReturnCode =
+ pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_BUS_HANG_ON;
} else {
- pcmdmessagefld->cmdmessage.ReturnCode =
+ pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_OK;
}
ptmpuserbuffer = ver_addr;
@@ -2735,28 +3029,39 @@ arcmsr_iop_message_xfer(struct AdapterCo
wqbuf_firstindex = acb->wqbuf_firstindex;
if (wqbuf_lastindex != wqbuf_firstindex) {
struct SENSE_DATA *sensebuffer =
- (struct SENSE_DATA *)cmd->sense_buffer;
+ (struct SENSE_DATA *)cmd->sense_buffer;
arcmsr_post_ioctldata2iop(acb);
/* has error report sensedata */
- sensebuffer->ErrorCode = 0x70;
+ sensebuffer->ErrorCode = SCSI_SENSE_CURRENT_ERRORS;
sensebuffer->SenseKey = ILLEGAL_REQUEST;
sensebuffer->AdditionalSenseLength = 0x0A;
sensebuffer->AdditionalSenseCode = 0x20;
sensebuffer->Valid = 1;
retvalue = ARCMSR_MESSAGE_FAIL;
} else {
- my_empty_len = (wqbuf_firstindex-wqbuf_lastindex - 1)
- &(ARCMSR_MAX_QBUFFER - 1);
+ my_empty_len = (wqbuf_firstindex - wqbuf_lastindex - 1)
+ & (ARCMSR_MAX_QBUFFER - 1);
if (my_empty_len >= user_len) {
while (user_len > 0) {
- pQbuffer =
- &acb->wqbuffer[acb->wqbuf_lastindex];
- memcpy(pQbuffer, ptmpuserbuffer, 1);
- acb->wqbuf_lastindex++;
- acb->wqbuf_lastindex %=
- ARCMSR_MAX_QBUFFER;
- ptmpuserbuffer++;
- user_len--;
+ pQbuffer = &acb->wqbuffer[acb->wqbuf_lastindex];
+ if ((acb->wqbuf_lastindex + user_len)
+ > ARCMSR_MAX_QBUFFER) {
+ memcpy(pQbuffer, ptmpuserbuffer,
+ ARCMSR_MAX_QBUFFER -
+ acb->wqbuf_lastindex);
+ ptmpuserbuffer += (ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_lastindex);
+ user_len -= (ARCMSR_MAX_QBUFFER
+ - acb->wqbuf_lastindex);
+ acb->wqbuf_lastindex = 0;
+ } else {
+ memcpy(pQbuffer, ptmpuserbuffer,
+ user_len);
+ acb->wqbuf_lastindex += user_len;
+ acb->wqbuf_lastindex %=
+ ARCMSR_MAX_QBUFFER;
+ user_len = 0;
+ }
}
if (acb->acb_flags &
ACB_F_MESSAGE_WQBUFFER_CLEARED) {
@@ -2765,24 +3070,26 @@ arcmsr_iop_message_xfer(struct AdapterCo
arcmsr_post_ioctldata2iop(acb);
}
} else {
- /* has error report sensedata */
struct SENSE_DATA *sensebuffer =
- (struct SENSE_DATA *)cmd->sense_buffer;
- sensebuffer->ErrorCode = 0x70;
+ (struct SENSE_DATA *)cmd->sense_buffer;
+ /* has error report sensedata */
+ sensebuffer->ErrorCode =
+ SCSI_SENSE_CURRENT_ERRORS;
sensebuffer->SenseKey = ILLEGAL_REQUEST;
sensebuffer->AdditionalSenseLength = 0x0A;
sensebuffer->AdditionalSenseCode = 0x20;
sensebuffer->Valid = 1;
retvalue = ARCMSR_MESSAGE_FAIL;
}
- }
- spin_unlock_irqrestore(&acb->wqbuffer_lock, flags);
- kfree(ver_addr);
}
+ spin_unlock_irqrestore(&acb->wqbuffer_lock, flags);
+ kfree(ver_addr);
break;
-
+ }
case ARCMSR_MESSAGE_CLEAR_RQBUFFER: {
uint8_t *pQbuffer = acb->rqbuffer;
+
+ spin_lock_irqsave(&acb->rqbuffer_lock, flags);
if (acb->acb_flags & ACB_F_IOPDATA_OVERFLOW) {
acb->acb_flags &= ~ACB_F_IOPDATA_OVERFLOW;
arcmsr_iop_message_read(acb);
@@ -2791,6 +3098,7 @@ arcmsr_iop_message_xfer(struct AdapterCo
acb->rqbuf_firstindex = 0;
acb->rqbuf_lastindex = 0;
memset(pQbuffer, 0, ARCMSR_MAX_QBUFFER);
+ spin_unlock_irqrestore(&acb->rqbuffer_lock, flags);
if (acb->fw_flag == FW_DEADLOCK) {
pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_BUS_HANG_ON;
@@ -2798,9 +3106,8 @@ arcmsr_iop_message_xfer(struct AdapterCo
pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_OK;
}
- }
break;
-
+ }
case ARCMSR_MESSAGE_CLEAR_WQBUFFER: {
uint8_t *pQbuffer = acb->wqbuffer;
if (acb->fw_flag == FW_DEADLOCK) {
@@ -2810,39 +3117,40 @@ arcmsr_iop_message_xfer(struct AdapterCo
pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_OK;
}
-
+ spin_lock_irqsave(&acb->wqbuffer_lock, flags);
if (acb->acb_flags & ACB_F_IOPDATA_OVERFLOW) {
acb->acb_flags &= ~ACB_F_IOPDATA_OVERFLOW;
arcmsr_iop_message_read(acb);
}
- acb->acb_flags |=
- (ACB_F_MESSAGE_WQBUFFER_CLEARED |
- ACB_F_MESSAGE_WQBUFFER_READED);
+ acb->acb_flags |= (ACB_F_MESSAGE_WQBUFFER_CLEARED |
+ ACB_F_MESSAGE_WQBUFFER_READED);
acb->wqbuf_firstindex = 0;
acb->wqbuf_lastindex = 0;
memset(pQbuffer, 0, ARCMSR_MAX_QBUFFER);
- }
+ spin_unlock_irqrestore(&acb->wqbuffer_lock, flags);
break;
-
+ }
case ARCMSR_MESSAGE_CLEAR_ALLQBUFFER: {
uint8_t *pQbuffer;
-
if (acb->acb_flags & ACB_F_IOPDATA_OVERFLOW) {
acb->acb_flags &= ~ACB_F_IOPDATA_OVERFLOW;
arcmsr_iop_message_read(acb);
}
- acb->acb_flags |=
- (ACB_F_MESSAGE_WQBUFFER_CLEARED
- | ACB_F_MESSAGE_RQBUFFER_CLEARED
- | ACB_F_MESSAGE_WQBUFFER_READED);
+ spin_lock_irqsave(&acb->rqbuffer_lock, flags);
+ acb->acb_flags |= ACB_F_MESSAGE_RQBUFFER_CLEARED;
acb->rqbuf_firstindex = 0;
acb->rqbuf_lastindex = 0;
- acb->wqbuf_firstindex = 0;
- acb->wqbuf_lastindex = 0;
pQbuffer = acb->rqbuffer;
memset(pQbuffer, 0, sizeof(struct QBUFFER));
+ spin_unlock_irqrestore(&acb->rqbuffer_lock, flags);
+ spin_lock_irqsave(&acb->wqbuffer_lock, flags);
+ acb->acb_flags |= (ACB_F_MESSAGE_WQBUFFER_CLEARED |
+ ACB_F_MESSAGE_WQBUFFER_READED);
+ acb->wqbuf_firstindex = 0;
+ acb->wqbuf_lastindex = 0;
pQbuffer = acb->wqbuffer;
memset(pQbuffer, 0, sizeof(struct QBUFFER));
+ spin_unlock_irqrestore(&acb->wqbuffer_lock, flags);
if (acb->fw_flag == FW_DEADLOCK) {
pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_BUS_HANG_ON;
@@ -2850,9 +3158,8 @@ arcmsr_iop_message_xfer(struct AdapterCo
pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_OK;
}
- }
break;
-
+ }
case ARCMSR_MESSAGE_RETURN_CODE_3F: {
if (acb->fw_flag == FW_DEADLOCK) {
pcmdmessagefld->cmdmessage.ReturnCode =
@@ -2862,7 +3169,7 @@ arcmsr_iop_message_xfer(struct AdapterCo
ARCMSR_MESSAGE_RETURNCODE_3F;
}
break;
- }
+ }
case ARCMSR_MESSAGE_SAY_HELLO: {
int8_t *hello_string = "Hello! I am ARCMSR";
if (acb->fw_flag == FW_DEADLOCK) {
@@ -2872,33 +3179,42 @@ arcmsr_iop_message_xfer(struct AdapterCo
pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_OK;
}
- memcpy(pcmdmessagefld->messagedatabuffer, hello_string
- , (int16_t)strlen(hello_string));
- }
+ memcpy(pcmdmessagefld->messagedatabuffer,
+ hello_string, (int16_t)strlen(hello_string));
break;
-
- case ARCMSR_MESSAGE_SAY_GOODBYE:
+ }
+ case ARCMSR_MESSAGE_SAY_GOODBYE: {
if (acb->fw_flag == FW_DEADLOCK) {
pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_BUS_HANG_ON;
+ } else {
+ pcmdmessagefld->cmdmessage.ReturnCode =
+ ARCMSR_MESSAGE_RETURNCODE_OK;
}
arcmsr_iop_parking(acb);
break;
-
- case ARCMSR_MESSAGE_FLUSH_ADAPTER_CACHE:
+ }
+ case ARCMSR_MESSAGE_FLUSH_ADAPTER_CACHE: {
if (acb->fw_flag == FW_DEADLOCK) {
pcmdmessagefld->cmdmessage.ReturnCode =
ARCMSR_MESSAGE_RETURNCODE_BUS_HANG_ON;
+ } else {
+ pcmdmessagefld->cmdmessage.ReturnCode =
+ ARCMSR_MESSAGE_RETURNCODE_OK;
}
arcmsr_flush_adapter_cache(acb);
break;
-
+ }
default:
retvalue = ARCMSR_MESSAGE_FAIL;
+ printk("unknown controlcode(%d)\n", __LINE__);
}
message_out:
- sg = scsi_sglist(cmd);
- kunmap_atomic(buffer - sg->offset);
+ if (use_sg) {
+ struct scatterlist *sg;
+ sg = scsi_sglist(cmd);
+ kunmap_atomic(buffer - sg->offset);
+ }
return retvalue;
}
@@ -2930,7 +3246,6 @@ arcmsr_handle_virtual_command(struct Ada
unsigned char inqdata[36];
char *buffer;
struct scatterlist *sg;
-
if (cmd->device->lun) {
cmd->result = (DID_TIME_OUT << 16);
cmd->scsi_done(cmd);
@@ -2949,14 +3264,11 @@ arcmsr_handle_virtual_command(struct Ada
strncpy(&inqdata[16], "RAID controller ", 16);
/* Product Identification */
strncpy(&inqdata[32], "R001", 4); /* Product Revision */
-
sg = scsi_sglist(cmd);
buffer = kmap_atomic(sg_page(sg)) + sg->offset;
-
memcpy(buffer, inqdata, sizeof(inqdata));
sg = scsi_sglist(cmd);
kunmap_atomic(buffer - sg->offset);
-
cmd->scsi_done(cmd);
}
break;
@@ -3611,19 +3923,21 @@ arcmsr_hbaD_polling_ccbdone(struct Adapt
{
bool error;
uint32_t poll_ccb_done = 0, poll_count = 0, flag_ccb, ccb_cdb_phy;
- int rtn, index, outbound_write_pointer;
+ int rtn, doneq_index, index_stripped, outbound_write_pointer;
+ unsigned long flags;
struct ARCMSR_CDB *arcmsr_cdb;
struct CommandControlBlock *pCCB;
- struct MessageUnit_D __iomem *reg =
+ struct MessageUnit_D __iomem *pmu =
(struct MessageUnit_D *)acb->pmuD;
+ spin_lock_irqsave(&acb->doneq_lock, flags);
polling_hbaD_ccb_retry:
poll_count++;
while (1) {
outbound_write_pointer =
- readl(reg->outboundlist_copy_pointer);
- index = reg->doneq_index;
- if ((outbound_write_pointer & 0xFF) == index) {
+ pmu->done_qbuffer[0].addressLow + 1;
+ doneq_index = pmu->doneq_index;
+ if ((outbound_write_pointer & 0xFFF) == (doneq_index & 0xFFF)) {
if (poll_ccb_done) {
rtn = SUCCESS;
break;
@@ -3636,17 +3950,27 @@ arcmsr_hbaD_polling_ccbdone(struct Adapt
goto polling_hbaD_ccb_retry;
}
}
- flag_ccb = reg->done_qbuffer[index].addressLow;
+ if (doneq_index & 0x4000) {
+ index_stripped = doneq_index & 0xFFF;
+ index_stripped += 1;
+ index_stripped %= ARCMSR_MAX_ARC1214_DONEQUEUE;
+ pmu->doneq_index = index_stripped ? (index_stripped | 0x4000)
+ : (index_stripped + 1);
+ } else {
+ index_stripped = doneq_index;
+ index_stripped += 1;
+ index_stripped %= ARCMSR_MAX_ARC1214_DONEQUEUE;
+ pmu->doneq_index = index_stripped ? index_stripped :
+ ((index_stripped | 0x4000) + 1);
+ }
+ doneq_index = pmu->doneq_index;
+ flag_ccb = pmu->done_qbuffer[doneq_index & 0xFFF].addressLow;
ccb_cdb_phy = (flag_ccb & 0xFFFFFFF0);
arcmsr_cdb = (struct ARCMSR_CDB *)(acb->vir2phy_offset +
ccb_cdb_phy);
pCCB = container_of(arcmsr_cdb,
struct CommandControlBlock, arcmsr_cdb);
poll_ccb_done = (pCCB == poll_ccb) ? 1 : 0;
- index++;
- index %= ARCMSR_MAX_ARC1214_POSTQUEUE;
- reg->doneq_index = index;
- /* check if command done with no error*/
if ((pCCB->acb != acb) ||
(pCCB->startdone != ARCMSR_CCB_START)) {
if (pCCB->startdone == ARCMSR_CCB_ABORTED) {
@@ -3673,6 +3997,7 @@ arcmsr_hbaD_polling_ccbdone(struct Adapt
? true : false;
arcmsr_report_ccb_state(acb, pCCB, error);
}
+ spin_unlock_irqrestore(&acb->doneq_lock, flags);
return rtn;
}
@@ -3805,7 +4130,7 @@ arcmsr_iop_confirm(struct AdapterControl
struct MessageUnit_D *reg =
(struct MessageUnit_D *)acb->pmuD;
reg->postq_index = 0;
- reg->doneq_index = 0x40FF;
+ reg->doneq_index = 0;
rwbuffer = reg->msgcode_rwbuffer;
writel(ARCMSR_SIGNATURE_SET_CONFIG, rwbuffer++);
writel(cdb_phyaddr_hi32, rwbuffer++);
@@ -3849,7 +4174,8 @@ arcmsr_wait_firmware_ready(struct Adapte
}
break;
case ACB_ADAPTER_TYPE_C: {
- struct MessageUnit_C *reg = (struct MessageUnit_C *)acb->pmuC;
+ struct MessageUnit_C *reg =
+ (struct MessageUnit_C *)acb->pmuC;
do {
firmware_state = readl(®->outbound_msgaddr1);
} while ((firmware_state &
^ permalink raw reply [flat|nested] 4+ messages in thread
end of thread, other threads:[~2013-02-08 6:05 UTC | newest]
Thread overview: 4+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2012-12-04 12:02 [PATCH 5/5] arcmsr: Modify ARC-1214 Inband Messages Behavior NickCheng
2013-02-06 8:38 NickCheng
2013-02-08 6:03 NickCheng
2013-02-08 6:04 NickCheng
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).