Skip to content

Commit 63c5395

Browse files
l1kbroonie
authored andcommitted
spi: bcm-qspi: Fix use-after-free on unbind
bcm_qspi_remove() calls spi_unregister_master() even though bcm_qspi_probe() calls devm_spi_register_master(). The spi_master is therefore unregistered and freed twice on unbind. Moreover, since commit 0392727 ("spi: bcm-qspi: Handle clock probe deferral"), bcm_qspi_probe() leaks the spi_master allocation if the call to devm_clk_get_optional() fails. Fix by switching over to the new devm_spi_alloc_master() helper which keeps the private data accessible until the driver has unbound and also avoids the spi_master leak on probe. While at it, fix an ordering issue in bcm_qspi_remove() wherein spi_unregister_master() is called after uninitializing the hardware, disabling the clock and freeing an IRQ data structure. The correct order is to call spi_unregister_master() *before* those teardown steps because bus accesses may still be ongoing until that function returns. Fixes: fa236a7 ("spi: bcm-qspi: Add Broadcom MSPI driver") Signed-off-by: Lukas Wunner <lukas@wunner.de> Cc: <stable@vger.kernel.org> # v4.9+: 123456789abc: spi: Introduce device-managed SPI controller allocation Cc: <stable@vger.kernel.org> # v4.9+ Cc: Kamal Dasu <kdasu.kdev@gmail.com> Acked-by: Florian Fainelli <f.fainelli@gmail.com> Tested-by: Florian Fainelli <f.fainelli@gmail.com> Link: https://lore.kernel.org/r/5e31a9a59fd1c0d0b795b2fe219f25e5ee855f9d.1605121038.git.lukas@wunner.de Signed-off-by: Mark Brown <broonie@kernel.org>
1 parent e13ee6c commit 63c5395

1 file changed

Lines changed: 12 additions & 22 deletions

File tree

drivers/spi/spi-bcm-qspi.c

Lines changed: 12 additions & 22 deletions
Original file line numberDiff line numberDiff line change
@@ -1327,7 +1327,7 @@ int bcm_qspi_probe(struct platform_device *pdev,
13271327

13281328
data = of_id->data;
13291329

1330-
master = spi_alloc_master(dev, sizeof(struct bcm_qspi));
1330+
master = devm_spi_alloc_master(dev, sizeof(struct bcm_qspi));
13311331
if (!master) {
13321332
dev_err(dev, "error allocating spi_master\n");
13331333
return -ENOMEM;
@@ -1367,21 +1367,17 @@ int bcm_qspi_probe(struct platform_device *pdev,
13671367

13681368
if (res) {
13691369
qspi->base[MSPI] = devm_ioremap_resource(dev, res);
1370-
if (IS_ERR(qspi->base[MSPI])) {
1371-
ret = PTR_ERR(qspi->base[MSPI]);
1372-
goto qspi_resource_err;
1373-
}
1370+
if (IS_ERR(qspi->base[MSPI]))
1371+
return PTR_ERR(qspi->base[MSPI]);
13741372
} else {
1375-
goto qspi_resource_err;
1373+
return 0;
13761374
}
13771375

13781376
res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "bspi");
13791377
if (res) {
13801378
qspi->base[BSPI] = devm_ioremap_resource(dev, res);
1381-
if (IS_ERR(qspi->base[BSPI])) {
1382-
ret = PTR_ERR(qspi->base[BSPI]);
1383-
goto qspi_resource_err;
1384-
}
1379+
if (IS_ERR(qspi->base[BSPI]))
1380+
return PTR_ERR(qspi->base[BSPI]);
13851381
qspi->bspi_mode = true;
13861382
} else {
13871383
qspi->bspi_mode = false;
@@ -1392,18 +1388,14 @@ int bcm_qspi_probe(struct platform_device *pdev,
13921388
res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "cs_reg");
13931389
if (res) {
13941390
qspi->base[CHIP_SELECT] = devm_ioremap_resource(dev, res);
1395-
if (IS_ERR(qspi->base[CHIP_SELECT])) {
1396-
ret = PTR_ERR(qspi->base[CHIP_SELECT]);
1397-
goto qspi_resource_err;
1398-
}
1391+
if (IS_ERR(qspi->base[CHIP_SELECT]))
1392+
return PTR_ERR(qspi->base[CHIP_SELECT]);
13991393
}
14001394

14011395
qspi->dev_ids = kcalloc(num_irqs, sizeof(struct bcm_qspi_dev_id),
14021396
GFP_KERNEL);
1403-
if (!qspi->dev_ids) {
1404-
ret = -ENOMEM;
1405-
goto qspi_resource_err;
1406-
}
1397+
if (!qspi->dev_ids)
1398+
return -ENOMEM;
14071399

14081400
for (val = 0; val < num_irqs; val++) {
14091401
irq = -1;
@@ -1484,7 +1476,7 @@ int bcm_qspi_probe(struct platform_device *pdev,
14841476
qspi->xfer_mode.addrlen = -1;
14851477
qspi->xfer_mode.hp = -1;
14861478

1487-
ret = devm_spi_register_master(&pdev->dev, master);
1479+
ret = spi_register_master(master);
14881480
if (ret < 0) {
14891481
dev_err(dev, "can't register master\n");
14901482
goto qspi_reg_err;
@@ -1497,8 +1489,6 @@ int bcm_qspi_probe(struct platform_device *pdev,
14971489
clk_disable_unprepare(qspi->clk);
14981490
qspi_probe_err:
14991491
kfree(qspi->dev_ids);
1500-
qspi_resource_err:
1501-
spi_master_put(master);
15021492
return ret;
15031493
}
15041494
/* probe function to be called by SoC specific platform driver probe */
@@ -1508,10 +1498,10 @@ int bcm_qspi_remove(struct platform_device *pdev)
15081498
{
15091499
struct bcm_qspi *qspi = platform_get_drvdata(pdev);
15101500

1501+
spi_unregister_master(qspi->master);
15111502
bcm_qspi_hw_uninit(qspi);
15121503
clk_disable_unprepare(qspi->clk);
15131504
kfree(qspi->dev_ids);
1514-
spi_unregister_master(qspi->master);
15151505

15161506
return 0;
15171507
}

0 commit comments

Comments
 (0)