提交 df480518 编写于 作者: J Jiri Slaby 提交者: Greg Kroah-Hartman

Char: mxser, call pci_disable_device from probe/remove

Vasiliy found that pci_disable_device is not called on fail paths in
mxser_probe. Actually, it is called from nowhere in the driver.

There are three changes needed:
1) don't use pseudo-generic mxser_release_res. Let's use it only from
   ISA paths from now on. All the pci stuff is moved to probe and
   remove PCI-related functions.
2) reorder fail-paths in the probe function so that it makes sense and
   we can call them from the sequential code naturally (the further we
   are the earlier label we go to).
3) add pci_disable_device both to mxser_probe and mxser_remove.

There is a nit of adding CONFIG_PCI ifdef to mxser_remove. it is
because this driver supports ISA-only compilations and it would choke
up on the newly added calls now.
Signed-off-by: NJiri Slaby <jslaby@suse.cz>
Cc: Kulikov Vasiliy <segooon@gmail.com>
Signed-off-by: NGreg Kroah-Hartman <gregkh@suse.de>
上级 24b4b67d
...@@ -2339,20 +2339,11 @@ struct tty_port_operations mxser_port_ops = { ...@@ -2339,20 +2339,11 @@ struct tty_port_operations mxser_port_ops = {
* The MOXA Smartio/Industio serial driver boot-time initialization code! * The MOXA Smartio/Industio serial driver boot-time initialization code!
*/ */
static void mxser_release_res(struct mxser_board *brd, struct pci_dev *pdev, static void mxser_release_ISA_res(struct mxser_board *brd)
unsigned int irq)
{ {
if (irq) free_irq(brd->irq, brd);
free_irq(brd->irq, brd); release_region(brd->ports[0].ioaddr, 8 * brd->info->nports);
if (pdev != NULL) { /* PCI */ release_region(brd->vector, 1);
#ifdef CONFIG_PCI
pci_release_region(pdev, 2);
pci_release_region(pdev, 3);
#endif
} else {
release_region(brd->ports[0].ioaddr, 8 * brd->info->nports);
release_region(brd->vector, 1);
}
} }
static int __devinit mxser_initbrd(struct mxser_board *brd, static int __devinit mxser_initbrd(struct mxser_board *brd,
...@@ -2397,13 +2388,11 @@ static int __devinit mxser_initbrd(struct mxser_board *brd, ...@@ -2397,13 +2388,11 @@ static int __devinit mxser_initbrd(struct mxser_board *brd,
retval = request_irq(brd->irq, mxser_interrupt, IRQF_SHARED, "mxser", retval = request_irq(brd->irq, mxser_interrupt, IRQF_SHARED, "mxser",
brd); brd);
if (retval) { if (retval)
printk(KERN_ERR "Board %s: Request irq failed, IRQ (%d) may " printk(KERN_ERR "Board %s: Request irq failed, IRQ (%d) may "
"conflict with another device.\n", "conflict with another device.\n",
brd->info->name, brd->irq); brd->info->name, brd->irq);
/* We hold resources, we need to release them. */
mxser_release_res(brd, pdev, 0);
}
return retval; return retval;
} }
...@@ -2555,7 +2544,7 @@ static int __devinit mxser_probe(struct pci_dev *pdev, ...@@ -2555,7 +2544,7 @@ static int __devinit mxser_probe(struct pci_dev *pdev,
ioaddress = pci_resource_start(pdev, 2); ioaddress = pci_resource_start(pdev, 2);
retval = pci_request_region(pdev, 2, "mxser(IO)"); retval = pci_request_region(pdev, 2, "mxser(IO)");
if (retval) if (retval)
goto err; goto err_dis;
brd->info = &mxser_cards[ent->driver_data]; brd->info = &mxser_cards[ent->driver_data];
for (i = 0; i < brd->info->nports; i++) for (i = 0; i < brd->info->nports; i++)
...@@ -2565,7 +2554,7 @@ static int __devinit mxser_probe(struct pci_dev *pdev, ...@@ -2565,7 +2554,7 @@ static int __devinit mxser_probe(struct pci_dev *pdev,
ioaddress = pci_resource_start(pdev, 3); ioaddress = pci_resource_start(pdev, 3);
retval = pci_request_region(pdev, 3, "mxser(vector)"); retval = pci_request_region(pdev, 3, "mxser(vector)");
if (retval) if (retval)
goto err_relio; goto err_zero;
brd->vector = ioaddress; brd->vector = ioaddress;
/* irq */ /* irq */
...@@ -2608,7 +2597,7 @@ static int __devinit mxser_probe(struct pci_dev *pdev, ...@@ -2608,7 +2597,7 @@ static int __devinit mxser_probe(struct pci_dev *pdev,
/* mxser_initbrd will hook ISR. */ /* mxser_initbrd will hook ISR. */
retval = mxser_initbrd(brd, pdev); retval = mxser_initbrd(brd, pdev);
if (retval) if (retval)
goto err_null; goto err_rel3;
for (i = 0; i < brd->info->nports; i++) for (i = 0; i < brd->info->nports; i++)
tty_register_device(mxvar_sdriver, brd->idx + i, &pdev->dev); tty_register_device(mxvar_sdriver, brd->idx + i, &pdev->dev);
...@@ -2616,10 +2605,13 @@ static int __devinit mxser_probe(struct pci_dev *pdev, ...@@ -2616,10 +2605,13 @@ static int __devinit mxser_probe(struct pci_dev *pdev,
pci_set_drvdata(pdev, brd); pci_set_drvdata(pdev, brd);
return 0; return 0;
err_relio: err_rel3:
pci_release_region(pdev, 2); pci_release_region(pdev, 3);
err_null: err_zero:
brd->info = NULL; brd->info = NULL;
pci_release_region(pdev, 2);
err_dis:
pci_disable_device(pdev);
err: err:
return retval; return retval;
#else #else
...@@ -2629,14 +2621,19 @@ static int __devinit mxser_probe(struct pci_dev *pdev, ...@@ -2629,14 +2621,19 @@ static int __devinit mxser_probe(struct pci_dev *pdev,
static void __devexit mxser_remove(struct pci_dev *pdev) static void __devexit mxser_remove(struct pci_dev *pdev)
{ {
#ifdef CONFIG_PCI
struct mxser_board *brd = pci_get_drvdata(pdev); struct mxser_board *brd = pci_get_drvdata(pdev);
unsigned int i; unsigned int i;
for (i = 0; i < brd->info->nports; i++) for (i = 0; i < brd->info->nports; i++)
tty_unregister_device(mxvar_sdriver, brd->idx + i); tty_unregister_device(mxvar_sdriver, brd->idx + i);
mxser_release_res(brd, pdev, 1); free_irq(pdev->irq, brd);
pci_release_region(pdev, 2);
pci_release_region(pdev, 3);
pci_disable_device(pdev);
brd->info = NULL; brd->info = NULL;
#endif
} }
static struct pci_driver mxser_driver = { static struct pci_driver mxser_driver = {
...@@ -2741,7 +2738,7 @@ static void __exit mxser_module_exit(void) ...@@ -2741,7 +2738,7 @@ static void __exit mxser_module_exit(void)
for (i = 0; i < MXSER_BOARDS; i++) for (i = 0; i < MXSER_BOARDS; i++)
if (mxser_boards[i].info != NULL) if (mxser_boards[i].info != NULL)
mxser_release_res(&mxser_boards[i], NULL, 1); mxser_release_ISA_res(&mxser_boards[i]);
} }
module_init(mxser_module_init); module_init(mxser_module_init);
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册