diff options
Diffstat (limited to 'drivers/block/cciss.c')
-rw-r--r-- | drivers/block/cciss.c | 70 |
1 files changed, 65 insertions, 5 deletions
diff --git a/drivers/block/cciss.c b/drivers/block/cciss.c index 3c6a6a21d540..a12b95eef18e 100644 --- a/drivers/block/cciss.c +++ b/drivers/block/cciss.c | |||
@@ -174,6 +174,8 @@ static int sendcmd_withirq(__u8 cmd, int ctlr, void *buff, size_t size, | |||
174 | unsigned int use_unit_num, unsigned int log_unit, __u8 page_code, | 174 | unsigned int use_unit_num, unsigned int log_unit, __u8 page_code, |
175 | int cmd_type); | 175 | int cmd_type); |
176 | 176 | ||
177 | static void fail_all_cmds(unsigned long ctlr); | ||
178 | |||
177 | #ifdef CONFIG_PROC_FS | 179 | #ifdef CONFIG_PROC_FS |
178 | static int cciss_proc_get_info(char *buffer, char **start, off_t offset, | 180 | static int cciss_proc_get_info(char *buffer, char **start, off_t offset, |
179 | int length, int *eof, void *data); | 181 | int length, int *eof, void *data); |
@@ -387,6 +389,8 @@ static CommandList_struct * cmd_alloc(ctlr_info_t *h, int get_from_pool) | |||
387 | return NULL; | 389 | return NULL; |
388 | memset(c, 0, sizeof(CommandList_struct)); | 390 | memset(c, 0, sizeof(CommandList_struct)); |
389 | 391 | ||
392 | c->cmdindex = -1; | ||
393 | |||
390 | c->err_info = (ErrorInfo_struct *)pci_alloc_consistent( | 394 | c->err_info = (ErrorInfo_struct *)pci_alloc_consistent( |
391 | h->pdev, sizeof(ErrorInfo_struct), | 395 | h->pdev, sizeof(ErrorInfo_struct), |
392 | &err_dma_handle); | 396 | &err_dma_handle); |
@@ -417,6 +421,8 @@ static CommandList_struct * cmd_alloc(ctlr_info_t *h, int get_from_pool) | |||
417 | err_dma_handle = h->errinfo_pool_dhandle | 421 | err_dma_handle = h->errinfo_pool_dhandle |
418 | + i*sizeof(ErrorInfo_struct); | 422 | + i*sizeof(ErrorInfo_struct); |
419 | h->nr_allocs++; | 423 | h->nr_allocs++; |
424 | |||
425 | c->cmdindex = i; | ||
420 | } | 426 | } |
421 | 427 | ||
422 | c->busaddr = (__u32) cmd_dma_handle; | 428 | c->busaddr = (__u32) cmd_dma_handle; |
@@ -2257,7 +2263,11 @@ queue: | |||
2257 | /* fill in the request */ | 2263 | /* fill in the request */ |
2258 | drv = creq->rq_disk->private_data; | 2264 | drv = creq->rq_disk->private_data; |
2259 | c->Header.ReplyQueue = 0; // unused in simple mode | 2265 | c->Header.ReplyQueue = 0; // unused in simple mode |
2260 | c->Header.Tag.lower = c->busaddr; // use the physical address the cmd block for tag | 2266 | /* got command from pool, so use the command block index instead */ |
2267 | /* for direct lookups. */ | ||
2268 | /* The first 2 bits are reserved for controller error reporting. */ | ||
2269 | c->Header.Tag.lower = (c->cmdindex << 3); | ||
2270 | c->Header.Tag.lower |= 0x04; /* flag for direct lookup. */ | ||
2261 | c->Header.LUN.LogDev.VolId= drv->LunID; | 2271 | c->Header.LUN.LogDev.VolId= drv->LunID; |
2262 | c->Header.LUN.LogDev.Mode = 1; | 2272 | c->Header.LUN.LogDev.Mode = 1; |
2263 | c->Request.CDBLen = 10; // 12 byte commands not in FW yet; | 2273 | c->Request.CDBLen = 10; // 12 byte commands not in FW yet; |
@@ -2332,7 +2342,7 @@ static irqreturn_t do_cciss_intr(int irq, void *dev_id, struct pt_regs *regs) | |||
2332 | ctlr_info_t *h = dev_id; | 2342 | ctlr_info_t *h = dev_id; |
2333 | CommandList_struct *c; | 2343 | CommandList_struct *c; |
2334 | unsigned long flags; | 2344 | unsigned long flags; |
2335 | __u32 a, a1; | 2345 | __u32 a, a1, a2; |
2336 | int j; | 2346 | int j; |
2337 | int start_queue = h->next_to_run; | 2347 | int start_queue = h->next_to_run; |
2338 | 2348 | ||
@@ -2350,10 +2360,21 @@ static irqreturn_t do_cciss_intr(int irq, void *dev_id, struct pt_regs *regs) | |||
2350 | while((a = h->access.command_completed(h)) != FIFO_EMPTY) | 2360 | while((a = h->access.command_completed(h)) != FIFO_EMPTY) |
2351 | { | 2361 | { |
2352 | a1 = a; | 2362 | a1 = a; |
2363 | if ((a & 0x04)) { | ||
2364 | a2 = (a >> 3); | ||
2365 | if (a2 >= NR_CMDS) { | ||
2366 | printk(KERN_WARNING "cciss: controller cciss%d failed, stopping.\n", h->ctlr); | ||
2367 | fail_all_cmds(h->ctlr); | ||
2368 | return IRQ_HANDLED; | ||
2369 | } | ||
2370 | |||
2371 | c = h->cmd_pool + a2; | ||
2372 | a = c->busaddr; | ||
2373 | |||
2374 | } else { | ||
2353 | a &= ~3; | 2375 | a &= ~3; |
2354 | if ((c = h->cmpQ) == NULL) | 2376 | if ((c = h->cmpQ) == NULL) { |
2355 | { | 2377 | printk(KERN_WARNING "cciss: Completion of %08x ignored\n", a1); |
2356 | printk(KERN_WARNING "cciss: Completion of %08lx ignored\n", (unsigned long)a1); | ||
2357 | continue; | 2378 | continue; |
2358 | } | 2379 | } |
2359 | while(c->busaddr != a) { | 2380 | while(c->busaddr != a) { |
@@ -2361,6 +2382,7 @@ static irqreturn_t do_cciss_intr(int irq, void *dev_id, struct pt_regs *regs) | |||
2361 | if (c == h->cmpQ) | 2382 | if (c == h->cmpQ) |
2362 | break; | 2383 | break; |
2363 | } | 2384 | } |
2385 | } | ||
2364 | /* | 2386 | /* |
2365 | * If we've found the command, take it off the | 2387 | * If we've found the command, take it off the |
2366 | * completion Q and free it | 2388 | * completion Q and free it |
@@ -3124,5 +3146,43 @@ static void __exit cciss_cleanup(void) | |||
3124 | remove_proc_entry("cciss", proc_root_driver); | 3146 | remove_proc_entry("cciss", proc_root_driver); |
3125 | } | 3147 | } |
3126 | 3148 | ||
3149 | static void fail_all_cmds(unsigned long ctlr) | ||
3150 | { | ||
3151 | /* If we get here, the board is apparently dead. */ | ||
3152 | ctlr_info_t *h = hba[ctlr]; | ||
3153 | CommandList_struct *c; | ||
3154 | unsigned long flags; | ||
3155 | |||
3156 | printk(KERN_WARNING "cciss%d: controller not responding.\n", h->ctlr); | ||
3157 | h->alive = 0; /* the controller apparently died... */ | ||
3158 | |||
3159 | spin_lock_irqsave(CCISS_LOCK(ctlr), flags); | ||
3160 | |||
3161 | pci_disable_device(h->pdev); /* Make sure it is really dead. */ | ||
3162 | |||
3163 | /* move everything off the request queue onto the completed queue */ | ||
3164 | while( (c = h->reqQ) != NULL ) { | ||
3165 | removeQ(&(h->reqQ), c); | ||
3166 | h->Qdepth--; | ||
3167 | addQ (&(h->cmpQ), c); | ||
3168 | } | ||
3169 | |||
3170 | /* Now, fail everything on the completed queue with a HW error */ | ||
3171 | while( (c = h->cmpQ) != NULL ) { | ||
3172 | removeQ(&h->cmpQ, c); | ||
3173 | c->err_info->CommandStatus = CMD_HARDWARE_ERR; | ||
3174 | if (c->cmd_type == CMD_RWREQ) { | ||
3175 | complete_command(h, c, 0); | ||
3176 | } else if (c->cmd_type == CMD_IOCTL_PEND) | ||
3177 | complete(c->waiting); | ||
3178 | #ifdef CONFIG_CISS_SCSI_TAPE | ||
3179 | else if (c->cmd_type == CMD_SCSI) | ||
3180 | complete_scsi_command(c, 0, 0); | ||
3181 | #endif | ||
3182 | } | ||
3183 | spin_unlock_irqrestore(CCISS_LOCK(ctlr), flags); | ||
3184 | return; | ||
3185 | } | ||
3186 | |||
3127 | module_init(cciss_init); | 3187 | module_init(cciss_init); |
3128 | module_exit(cciss_cleanup); | 3188 | module_exit(cciss_cleanup); |