summaryrefslogtreecommitdiffstats
path: root/drivers/uio
diff options
context:
space:
mode:
authorDan Carpenter <dan.carpenter@oracle.com>2018-08-02 03:15:58 -0400
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>2018-08-02 04:14:26 -0400
commit95883676e34ab93f600787cc9831707bcdad4398 (patch)
tree5427e08e1a67dde399bf92390a19439c12a44b30 /drivers/uio
parent7ceb1c37533e2298797188087796dd44931d86af (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.c69
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
225out_free: 227err_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);
233err_free_ddr_vaddr:
234 dma_free_coherent(dev, extram_pool_sz, gdev->ddr_vaddr,
235 gdev->ddr_paddr);
236err_free_sram:
237 if (pdata->sram_pool)
238 gen_pool_free(gdev->sram_pool, gdev->sram_vaddr, sram_pool_sz);
239err_clk_disable:
240 clk_disable(gdev->pruss_clk);
241err_clk_put:
242 clk_put(gdev->pruss_clk);
243err_free_info:
244 kfree(gdev->info);
245err_free_gdev:
246 kfree(gdev);
247
227 return ret; 248 return ret;
228} 249}
229 250