hpsa: clean up error handling
refactor error cleanup and shutdown disable interrupts and pci_disable_device on critical failures add hpsa_free_cfgtables function Reviewed-by: Scott Teel <scott.teel@pmcs.com> Reviewed-by: Kevin Barnett <kevin.barnett@pmcs.com> Reviewed-by: Tomas Henzl <thenzl@redhat.com> Reviewed-by: Hannes Reinecke <hare@Suse.de> Signed-off-by: Robert Elliott <elliott@hp.com> Signed-off-by: Don Brace <don.brace@pmcs.com> Reviewed-by: Christoph Hellwig <hch@lst.de> Signed-off-by: James Bottomley <JBottomley@Odin.com>
This commit is contained in:
parent
cc64c817a8
commit
195f2c65f9
|
@ -6495,6 +6495,17 @@ static int hpsa_find_cfg_addrs(struct pci_dev *pdev, void __iomem *vaddr,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void hpsa_free_cfgtables(struct ctlr_info *h)
|
||||||
|
{
|
||||||
|
if (h->transtable)
|
||||||
|
iounmap(h->transtable);
|
||||||
|
if (h->cfgtable)
|
||||||
|
iounmap(h->cfgtable);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Find and map CISS config table and transfer table
|
||||||
|
+ * several items must be unmapped (freed) later
|
||||||
|
+ * */
|
||||||
static int hpsa_find_cfgtables(struct ctlr_info *h)
|
static int hpsa_find_cfgtables(struct ctlr_info *h)
|
||||||
{
|
{
|
||||||
u64 cfg_offset;
|
u64 cfg_offset;
|
||||||
|
@ -6521,8 +6532,11 @@ static int hpsa_find_cfgtables(struct ctlr_info *h)
|
||||||
h->transtable = remap_pci_mem(pci_resource_start(h->pdev,
|
h->transtable = remap_pci_mem(pci_resource_start(h->pdev,
|
||||||
cfg_base_addr_index)+cfg_offset+trans_offset,
|
cfg_base_addr_index)+cfg_offset+trans_offset,
|
||||||
sizeof(*h->transtable));
|
sizeof(*h->transtable));
|
||||||
if (!h->transtable)
|
if (!h->transtable) {
|
||||||
|
dev_err(&h->pdev->dev, "Failed mapping transfer table\n");
|
||||||
|
hpsa_free_cfgtables(h);
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -6698,6 +6712,17 @@ error:
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* free items allocated or mapped by hpsa_pci_init */
|
||||||
|
static void hpsa_free_pci_init(struct ctlr_info *h)
|
||||||
|
{
|
||||||
|
hpsa_free_cfgtables(h); /* pci_init 4 */
|
||||||
|
iounmap(h->vaddr); /* pci_init 3 */
|
||||||
|
hpsa_disable_interrupt_mode(h); /* pci_init 2 */
|
||||||
|
pci_release_regions(h->pdev); /* pci_init 2 */
|
||||||
|
pci_disable_device(h->pdev); /* pci_init 1 */
|
||||||
|
}
|
||||||
|
|
||||||
|
/* several items must be freed later */
|
||||||
static int hpsa_pci_init(struct ctlr_info *h)
|
static int hpsa_pci_init(struct ctlr_info *h)
|
||||||
{
|
{
|
||||||
int prod_index, err;
|
int prod_index, err;
|
||||||
|
@ -6716,15 +6741,15 @@ static int hpsa_pci_init(struct ctlr_info *h)
|
||||||
|
|
||||||
err = pci_enable_device(h->pdev);
|
err = pci_enable_device(h->pdev);
|
||||||
if (err) {
|
if (err) {
|
||||||
dev_warn(&h->pdev->dev, "unable to enable PCI device\n");
|
dev_err(&h->pdev->dev, "failed to enable PCI device\n");
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
err = pci_request_regions(h->pdev, HPSA);
|
err = pci_request_regions(h->pdev, HPSA);
|
||||||
if (err) {
|
if (err) {
|
||||||
dev_err(&h->pdev->dev,
|
dev_err(&h->pdev->dev,
|
||||||
"cannot obtain PCI resources, aborting\n");
|
"failed to obtain PCI resources\n");
|
||||||
return err;
|
goto clean1; /* pci */
|
||||||
}
|
}
|
||||||
|
|
||||||
pci_set_master(h->pdev);
|
pci_set_master(h->pdev);
|
||||||
|
@ -6732,40 +6757,41 @@ static int hpsa_pci_init(struct ctlr_info *h)
|
||||||
hpsa_interrupt_mode(h);
|
hpsa_interrupt_mode(h);
|
||||||
err = hpsa_pci_find_memory_BAR(h->pdev, &h->paddr);
|
err = hpsa_pci_find_memory_BAR(h->pdev, &h->paddr);
|
||||||
if (err)
|
if (err)
|
||||||
goto err_out_free_res;
|
goto clean2; /* intmode+region, pci */
|
||||||
h->vaddr = remap_pci_mem(h->paddr, 0x250);
|
h->vaddr = remap_pci_mem(h->paddr, 0x250);
|
||||||
if (!h->vaddr) {
|
if (!h->vaddr) {
|
||||||
|
dev_err(&h->pdev->dev, "failed to remap PCI mem\n");
|
||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
goto err_out_free_res;
|
goto clean2; /* intmode+region, pci */
|
||||||
}
|
}
|
||||||
err = hpsa_wait_for_board_state(h->pdev, h->vaddr, BOARD_READY);
|
err = hpsa_wait_for_board_state(h->pdev, h->vaddr, BOARD_READY);
|
||||||
if (err)
|
if (err)
|
||||||
goto err_out_free_res;
|
goto clean3; /* vaddr, intmode+region, pci */
|
||||||
err = hpsa_find_cfgtables(h);
|
err = hpsa_find_cfgtables(h);
|
||||||
if (err)
|
if (err)
|
||||||
goto err_out_free_res;
|
goto clean3; /* vaddr, intmode+region, pci */
|
||||||
hpsa_find_board_params(h);
|
hpsa_find_board_params(h);
|
||||||
|
|
||||||
if (!hpsa_CISS_signature_present(h)) {
|
if (!hpsa_CISS_signature_present(h)) {
|
||||||
err = -ENODEV;
|
err = -ENODEV;
|
||||||
goto err_out_free_res;
|
goto clean4; /* cfgtables, vaddr, intmode+region, pci */
|
||||||
}
|
}
|
||||||
hpsa_set_driver_support_bits(h);
|
hpsa_set_driver_support_bits(h);
|
||||||
hpsa_p600_dma_prefetch_quirk(h);
|
hpsa_p600_dma_prefetch_quirk(h);
|
||||||
err = hpsa_enter_simple_mode(h);
|
err = hpsa_enter_simple_mode(h);
|
||||||
if (err)
|
if (err)
|
||||||
goto err_out_free_res;
|
goto clean4; /* cfgtables, vaddr, intmode+region, pci */
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
err_out_free_res:
|
clean4: /* cfgtables, vaddr, intmode+region, pci */
|
||||||
if (h->transtable)
|
hpsa_free_cfgtables(h);
|
||||||
iounmap(h->transtable);
|
clean3: /* vaddr, intmode+region, pci */
|
||||||
if (h->cfgtable)
|
iounmap(h->vaddr);
|
||||||
iounmap(h->cfgtable);
|
clean2: /* intmode+region, pci */
|
||||||
if (h->vaddr)
|
hpsa_disable_interrupt_mode(h);
|
||||||
iounmap(h->vaddr);
|
|
||||||
pci_disable_device(h->pdev);
|
|
||||||
pci_release_regions(h->pdev);
|
pci_release_regions(h->pdev);
|
||||||
|
clean1: /* pci */
|
||||||
|
pci_disable_device(h->pdev);
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -6974,8 +7000,9 @@ static int hpsa_request_irqs(struct ctlr_info *h,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (rc) {
|
if (rc) {
|
||||||
dev_err(&h->pdev->dev, "unable to get irq %d for %s\n",
|
dev_err(&h->pdev->dev, "failed to get irq %d for %s\n",
|
||||||
h->intr[h->intr_mode], h->devname);
|
h->intr[h->intr_mode], h->devname);
|
||||||
|
hpsa_free_irqs(h);
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -7027,15 +7054,11 @@ static void hpsa_undo_allocations_after_kdump_soft_reset(struct ctlr_info *h)
|
||||||
kfree(h->ioaccel1_blockFetchTable);
|
kfree(h->ioaccel1_blockFetchTable);
|
||||||
kfree(h->blockFetchTable);
|
kfree(h->blockFetchTable);
|
||||||
hpsa_free_reply_queues(h);
|
hpsa_free_reply_queues(h);
|
||||||
if (h->vaddr)
|
hpsa_free_cfgtables(h); /* pci_init 4 */
|
||||||
iounmap(h->vaddr);
|
iounmap(h->vaddr); /* pci_init 3 */
|
||||||
if (h->transtable)
|
hpsa_disable_interrupt_mode(h); /* pci_init 2 */
|
||||||
iounmap(h->transtable);
|
|
||||||
if (h->cfgtable)
|
|
||||||
iounmap(h->cfgtable);
|
|
||||||
hpsa_disable_interrupt_mode(h);
|
|
||||||
pci_disable_device(h->pdev);
|
pci_disable_device(h->pdev);
|
||||||
pci_release_regions(h->pdev);
|
pci_release_regions(h->pdev); /* pci_init 2 */
|
||||||
kfree(h);
|
kfree(h);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -7359,7 +7382,7 @@ reinit_after_soft_reset:
|
||||||
dac = 0;
|
dac = 0;
|
||||||
} else {
|
} else {
|
||||||
dev_err(&pdev->dev, "no suitable DMA available\n");
|
dev_err(&pdev->dev, "no suitable DMA available\n");
|
||||||
goto clean1;
|
goto clean2;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -7470,6 +7493,7 @@ clean4:
|
||||||
clean2_and_free_irqs:
|
clean2_and_free_irqs:
|
||||||
hpsa_free_irqs(h);
|
hpsa_free_irqs(h);
|
||||||
clean2:
|
clean2:
|
||||||
|
hpsa_free_pci_init(h);
|
||||||
clean1:
|
clean1:
|
||||||
if (h->resubmit_wq)
|
if (h->resubmit_wq)
|
||||||
destroy_workqueue(h->resubmit_wq);
|
destroy_workqueue(h->resubmit_wq);
|
||||||
|
@ -7561,12 +7585,10 @@ static void hpsa_remove_one(struct pci_dev *pdev)
|
||||||
destroy_workqueue(h->resubmit_wq);
|
destroy_workqueue(h->resubmit_wq);
|
||||||
hpsa_unregister_scsi(h); /* unhook from SCSI subsystem */
|
hpsa_unregister_scsi(h); /* unhook from SCSI subsystem */
|
||||||
|
|
||||||
/* includes hpsa_free_irqs and hpsa_disable_interrupt_mode */
|
/* includes hpsa_free_irqs */
|
||||||
|
/* includes hpsa_disable_interrupt_mode - pci_init 2 */
|
||||||
hpsa_shutdown(pdev);
|
hpsa_shutdown(pdev);
|
||||||
|
|
||||||
iounmap(h->vaddr);
|
|
||||||
iounmap(h->transtable);
|
|
||||||
iounmap(h->cfgtable);
|
|
||||||
hpsa_free_device_info(h);
|
hpsa_free_device_info(h);
|
||||||
hpsa_free_sg_chain_blocks(h);
|
hpsa_free_sg_chain_blocks(h);
|
||||||
pci_free_consistent(h->pdev,
|
pci_free_consistent(h->pdev,
|
||||||
|
@ -7581,8 +7603,10 @@ static void hpsa_remove_one(struct pci_dev *pdev)
|
||||||
kfree(h->ioaccel1_blockFetchTable);
|
kfree(h->ioaccel1_blockFetchTable);
|
||||||
kfree(h->ioaccel2_blockFetchTable);
|
kfree(h->ioaccel2_blockFetchTable);
|
||||||
kfree(h->hba_inquiry_data);
|
kfree(h->hba_inquiry_data);
|
||||||
pci_disable_device(pdev);
|
|
||||||
pci_release_regions(pdev);
|
/* includes hpsa_disable_interrupt_mode - pci_init 2 */
|
||||||
|
hpsa_free_pci_init(h);
|
||||||
|
|
||||||
free_percpu(h->lockup_detected);
|
free_percpu(h->lockup_detected);
|
||||||
kfree(h);
|
kfree(h);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue