diff options
| author | Dan Carpenter <dan.carpenter@oracle.com> | 2018-08-02 03:15:58 -0400 |
|---|---|---|
| committer | Greg Kroah-Hartman <gregkh@linuxfoundation.org> | 2018-08-02 04:14:26 -0400 |
| commit | 95883676e34ab93f600787cc9831707bcdad4398 (patch) | |
| tree | 5427e08e1a67dde399bf92390a19439c12a44b30 /drivers/uio | |
| parent | 7ceb1c37533e2298797188087796dd44931d86af (diff) | |
uio: pruss: fix error handling in probe
There are two bugs here. First the error codes weren't set on several
paths. And second, if the call to request_threaded_irq() inside
uio_register_device() fails then it would lead to a double free when
we call uio_unregister_device() inside pruss_cleanup().
Signed-off-by: Dan Carpenter <dan.carpenter@oracle.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Diffstat (limited to 'drivers/uio')
| -rw-r--r-- | drivers/uio/uio_pruss.c | 69 |
1 files changed, 45 insertions, 24 deletions
diff --git a/drivers/uio/uio_pruss.c b/drivers/uio/uio_pruss.c index 91aea8823af5..1cc175d3c25c 100644 --- a/drivers/uio/uio_pruss.c +++ b/drivers/uio/uio_pruss.c | |||
| @@ -122,7 +122,7 @@ static int pruss_probe(struct platform_device *pdev) | |||
| 122 | struct uio_pruss_dev *gdev; | 122 | struct uio_pruss_dev *gdev; |
| 123 | struct resource *regs_prussio; | 123 | struct resource *regs_prussio; |
| 124 | struct device *dev = &pdev->dev; | 124 | struct device *dev = &pdev->dev; |
| 125 | int ret = -ENODEV, cnt = 0, len; | 125 | int ret, cnt, i, len; |
| 126 | struct uio_pruss_pdata *pdata = dev_get_platdata(dev); | 126 | struct uio_pruss_pdata *pdata = dev_get_platdata(dev); |
| 127 | 127 | ||
| 128 | gdev = kzalloc(sizeof(struct uio_pruss_dev), GFP_KERNEL); | 128 | gdev = kzalloc(sizeof(struct uio_pruss_dev), GFP_KERNEL); |
| @@ -131,8 +131,8 @@ static int pruss_probe(struct platform_device *pdev) | |||
| 131 | 131 | ||
| 132 | gdev->info = kcalloc(MAX_PRUSS_EVT, sizeof(*p), GFP_KERNEL); | 132 | gdev->info = kcalloc(MAX_PRUSS_EVT, sizeof(*p), GFP_KERNEL); |
| 133 | if (!gdev->info) { | 133 | if (!gdev->info) { |
| 134 | kfree(gdev); | 134 | ret = -ENOMEM; |
| 135 | return -ENOMEM; | 135 | goto err_free_gdev; |
| 136 | } | 136 | } |
| 137 | 137 | ||
| 138 | /* Power on PRU in case its not done as part of boot-loader */ | 138 | /* Power on PRU in case its not done as part of boot-loader */ |
| @@ -140,29 +140,26 @@ static int pruss_probe(struct platform_device *pdev) | |||
| 140 | if (IS_ERR(gdev->pruss_clk)) { | 140 | if (IS_ERR(gdev->pruss_clk)) { |
| 141 | dev_err(dev, "Failed to get clock\n"); | 141 | dev_err(dev, "Failed to get clock\n"); |
| 142 | ret = PTR_ERR(gdev->pruss_clk); | 142 | ret = PTR_ERR(gdev->pruss_clk); |
| 143 | kfree(gdev->info); | 143 | goto err_free_info; |
| 144 | kfree(gdev); | 144 | } |
| 145 | return ret; | 145 | |
| 146 | } else { | 146 | ret = clk_enable(gdev->pruss_clk); |
| 147 | ret = clk_enable(gdev->pruss_clk); | 147 | if (ret) { |
| 148 | if (ret) { | 148 | dev_err(dev, "Failed to enable clock\n"); |
| 149 | dev_err(dev, "Failed to enable clock\n"); | 149 | goto err_clk_put; |
| 150 | clk_put(gdev->pruss_clk); | ||
| 151 | kfree(gdev->info); | ||
| 152 | kfree(gdev); | ||
| 153 | return ret; | ||
| 154 | } | ||
| 155 | } | 150 | } |
| 156 | 151 | ||
| 157 | regs_prussio = platform_get_resource(pdev, IORESOURCE_MEM, 0); | 152 | regs_prussio = platform_get_resource(pdev, IORESOURCE_MEM, 0); |
| 158 | if (!regs_prussio) { | 153 | if (!regs_prussio) { |
| 159 | dev_err(dev, "No PRUSS I/O resource specified\n"); | 154 | dev_err(dev, "No PRUSS I/O resource specified\n"); |
| 160 | goto out_free; | 155 | ret = -EIO; |
| 156 | goto err_clk_disable; | ||
| 161 | } | 157 | } |
| 162 | 158 | ||
| 163 | if (!regs_prussio->start) { | 159 | if (!regs_prussio->start) { |
| 164 | dev_err(dev, "Invalid memory resource\n"); | 160 | dev_err(dev, "Invalid memory resource\n"); |
| 165 | goto out_free; | 161 | ret = -EIO; |
| 162 | goto err_clk_disable; | ||
| 166 | } | 163 | } |
| 167 | 164 | ||
| 168 | if (pdata->sram_pool) { | 165 | if (pdata->sram_pool) { |
| @@ -172,7 +169,8 @@ static int pruss_probe(struct platform_device *pdev) | |||
| 172 | sram_pool_sz, &gdev->sram_paddr); | 169 | sram_pool_sz, &gdev->sram_paddr); |
| 173 | if (!gdev->sram_vaddr) { | 170 | if (!gdev->sram_vaddr) { |
| 174 | dev_err(dev, "Could not allocate SRAM pool\n"); | 171 | dev_err(dev, "Could not allocate SRAM pool\n"); |
| 175 | goto out_free; | 172 | ret = -ENOMEM; |
| 173 | goto err_clk_disable; | ||
| 176 | } | 174 | } |
| 177 | } | 175 | } |
| 178 | 176 | ||
| @@ -180,14 +178,16 @@ static int pruss_probe(struct platform_device *pdev) | |||
| 180 | &(gdev->ddr_paddr), GFP_KERNEL | GFP_DMA); | 178 | &(gdev->ddr_paddr), GFP_KERNEL | GFP_DMA); |
| 181 | if (!gdev->ddr_vaddr) { | 179 | if (!gdev->ddr_vaddr) { |
| 182 | dev_err(dev, "Could not allocate external memory\n"); | 180 | dev_err(dev, "Could not allocate external memory\n"); |
| 183 | goto out_free; | 181 | ret = -ENOMEM; |
| 182 | goto err_free_sram; | ||
| 184 | } | 183 | } |
| 185 | 184 | ||
| 186 | len = resource_size(regs_prussio); | 185 | len = resource_size(regs_prussio); |
| 187 | gdev->prussio_vaddr = ioremap(regs_prussio->start, len); | 186 | gdev->prussio_vaddr = ioremap(regs_prussio->start, len); |
| 188 | if (!gdev->prussio_vaddr) { | 187 | if (!gdev->prussio_vaddr) { |
| 189 | dev_err(dev, "Can't remap PRUSS I/O address range\n"); | 188 | dev_err(dev, "Can't remap PRUSS I/O address range\n"); |
| 190 | goto out_free; | 189 | ret = -ENOMEM; |
| 190 | goto err_free_ddr_vaddr; | ||
| 191 | } | 191 | } |
| 192 | 192 | ||
| 193 | gdev->pintc_base = pdata->pintc_base; | 193 | gdev->pintc_base = pdata->pintc_base; |
| @@ -215,15 +215,36 @@ static int pruss_probe(struct platform_device *pdev) | |||
| 215 | p->priv = gdev; | 215 | p->priv = gdev; |
| 216 | 216 | ||
| 217 | ret = uio_register_device(dev, p); | 217 | ret = uio_register_device(dev, p); |
| 218 | if (ret < 0) | 218 | if (ret < 0) { |
| 219 | goto out_free; | 219 | kfree(p->name); |
| 220 | goto err_unloop; | ||
| 221 | } | ||
| 220 | } | 222 | } |
| 221 | 223 | ||
| 222 | platform_set_drvdata(pdev, gdev); | 224 | platform_set_drvdata(pdev, gdev); |
| 223 | return 0; | 225 | return 0; |
| 224 | 226 | ||
| 225 | out_free: | 227 | err_unloop: |
| 226 | pruss_cleanup(dev, gdev); | 228 | for (i = 0, p = gdev->info; i < cnt; i++, p++) { |
| 229 | uio_unregister_device(p); | ||
| 230 | kfree(p->name); | ||
| 231 | } | ||
| 232 | iounmap(gdev->prussio_vaddr); | ||
| 233 | err_free_ddr_vaddr: | ||
| 234 | dma_free_coherent(dev, extram_pool_sz, gdev->ddr_vaddr, | ||
| 235 | gdev->ddr_paddr); | ||
| 236 | err_free_sram: | ||
| 237 | if (pdata->sram_pool) | ||
| 238 | gen_pool_free(gdev->sram_pool, gdev->sram_vaddr, sram_pool_sz); | ||
| 239 | err_clk_disable: | ||
| 240 | clk_disable(gdev->pruss_clk); | ||
| 241 | err_clk_put: | ||
| 242 | clk_put(gdev->pruss_clk); | ||
| 243 | err_free_info: | ||
| 244 | kfree(gdev->info); | ||
| 245 | err_free_gdev: | ||
| 246 | kfree(gdev); | ||
| 247 | |||
| 227 | return ret; | 248 | return ret; |
| 228 | } | 249 | } |
| 229 | 250 | ||
