spi/bfin_spi: fix resources leakage
Re-order setup() a bit so we don't leak memory/dma/gpio resources upon errors. Also make sure we don't call kfree() twice on the same object. Signed-off-by: Daniel Mack <daniel@caiaq.de> Signed-off-by: Bryan Wu <cooloney@kernel.org> Signed-off-by: Yi Li <yi.li@analog.com> Signed-off-by: Mike Frysinger <vapier@gentoo.org>
This commit is contained in:
parent
2b666ca4a6
commit
ac01e97d64
|
@ -1006,20 +1006,24 @@ static u16 ssel[][MAX_SPI_SSEL] = {
|
||||||
/* first setup for new devices */
|
/* first setup for new devices */
|
||||||
static int bfin_spi_setup(struct spi_device *spi)
|
static int bfin_spi_setup(struct spi_device *spi)
|
||||||
{
|
{
|
||||||
struct bfin5xx_spi_chip *chip_info = NULL;
|
struct bfin5xx_spi_chip *chip_info;
|
||||||
struct chip_data *chip;
|
struct chip_data *chip = NULL;
|
||||||
struct driver_data *drv_data = spi_master_get_devdata(spi->master);
|
struct driver_data *drv_data = spi_master_get_devdata(spi->master);
|
||||||
int ret;
|
int ret = -EINVAL;
|
||||||
|
|
||||||
if (spi->bits_per_word != 8 && spi->bits_per_word != 16)
|
if (spi->bits_per_word != 8 && spi->bits_per_word != 16)
|
||||||
return -EINVAL;
|
goto error;
|
||||||
|
|
||||||
/* Only alloc (or use chip_info) on first setup */
|
/* Only alloc (or use chip_info) on first setup */
|
||||||
|
chip_info = NULL;
|
||||||
chip = spi_get_ctldata(spi);
|
chip = spi_get_ctldata(spi);
|
||||||
if (chip == NULL) {
|
if (chip == NULL) {
|
||||||
chip = kzalloc(sizeof(struct chip_data), GFP_KERNEL);
|
chip = kzalloc(sizeof(*chip), GFP_KERNEL);
|
||||||
if (!chip)
|
if (!chip) {
|
||||||
return -ENOMEM;
|
dev_err(&spi->dev, "cannot allocate chip data\n");
|
||||||
|
ret = -ENOMEM;
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
|
||||||
chip->enable_dma = 0;
|
chip->enable_dma = 0;
|
||||||
chip_info = spi->controller_data;
|
chip_info = spi->controller_data;
|
||||||
|
@ -1036,7 +1040,7 @@ static int bfin_spi_setup(struct spi_device *spi)
|
||||||
if (chip_info->ctl_reg & (SPE|MSTR|CPOL|CPHA|LSBF|SIZE)) {
|
if (chip_info->ctl_reg & (SPE|MSTR|CPOL|CPHA|LSBF|SIZE)) {
|
||||||
dev_err(&spi->dev, "do not set bits in ctl_reg "
|
dev_err(&spi->dev, "do not set bits in ctl_reg "
|
||||||
"that the SPI framework manages\n");
|
"that the SPI framework manages\n");
|
||||||
return -EINVAL;
|
goto error;
|
||||||
}
|
}
|
||||||
|
|
||||||
chip->enable_dma = chip_info->enable_dma != 0
|
chip->enable_dma = chip_info->enable_dma != 0
|
||||||
|
@ -1059,26 +1063,6 @@ static int bfin_spi_setup(struct spi_device *spi)
|
||||||
/* we dont support running in slave mode (yet?) */
|
/* we dont support running in slave mode (yet?) */
|
||||||
chip->ctl_reg |= MSTR;
|
chip->ctl_reg |= MSTR;
|
||||||
|
|
||||||
/*
|
|
||||||
* if any one SPI chip is registered and wants DMA, request the
|
|
||||||
* DMA channel for it
|
|
||||||
*/
|
|
||||||
if (chip->enable_dma && !drv_data->dma_requested) {
|
|
||||||
/* register dma irq handler */
|
|
||||||
if (request_dma(drv_data->dma_channel, "BFIN_SPI_DMA") < 0) {
|
|
||||||
dev_dbg(&spi->dev,
|
|
||||||
"Unable to request BlackFin SPI DMA channel\n");
|
|
||||||
return -ENODEV;
|
|
||||||
}
|
|
||||||
if (set_dma_callback(drv_data->dma_channel,
|
|
||||||
bfin_spi_dma_irq_handler, drv_data) < 0) {
|
|
||||||
dev_dbg(&spi->dev, "Unable to set dma callback\n");
|
|
||||||
return -EPERM;
|
|
||||||
}
|
|
||||||
dma_disable_irq(drv_data->dma_channel);
|
|
||||||
drv_data->dma_requested = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Notice: for blackfin, the speed_hz is the value of register
|
* Notice: for blackfin, the speed_hz is the value of register
|
||||||
* SPI_BAUD, not the real baudrate
|
* SPI_BAUD, not the real baudrate
|
||||||
|
@ -1087,16 +1071,6 @@ static int bfin_spi_setup(struct spi_device *spi)
|
||||||
chip->flag = 1 << (spi->chip_select);
|
chip->flag = 1 << (spi->chip_select);
|
||||||
chip->chip_select_num = spi->chip_select;
|
chip->chip_select_num = spi->chip_select;
|
||||||
|
|
||||||
if (chip->chip_select_num == 0) {
|
|
||||||
ret = gpio_request(chip->cs_gpio, spi->modalias);
|
|
||||||
if (ret) {
|
|
||||||
if (drv_data->dma_requested)
|
|
||||||
free_dma(drv_data->dma_channel);
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
gpio_direction_output(chip->cs_gpio, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (chip->bits_per_word) {
|
switch (chip->bits_per_word) {
|
||||||
case 8:
|
case 8:
|
||||||
chip->n_bytes = 1;
|
chip->n_bytes = 1;
|
||||||
|
@ -1123,9 +1097,39 @@ static int bfin_spi_setup(struct spi_device *spi)
|
||||||
default:
|
default:
|
||||||
dev_err(&spi->dev, "%d bits_per_word is not supported\n",
|
dev_err(&spi->dev, "%d bits_per_word is not supported\n",
|
||||||
chip->bits_per_word);
|
chip->bits_per_word);
|
||||||
if (chip_info)
|
goto error;
|
||||||
kfree(chip);
|
}
|
||||||
return -ENODEV;
|
|
||||||
|
/*
|
||||||
|
* if any one SPI chip is registered and wants DMA, request the
|
||||||
|
* DMA channel for it
|
||||||
|
*/
|
||||||
|
if (chip->enable_dma && !drv_data->dma_requested) {
|
||||||
|
/* register dma irq handler */
|
||||||
|
ret = request_dma(drv_data->dma_channel, "BFIN_SPI_DMA");
|
||||||
|
if (ret) {
|
||||||
|
dev_err(&spi->dev,
|
||||||
|
"Unable to request BlackFin SPI DMA channel\n");
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
drv_data->dma_requested = 1;
|
||||||
|
|
||||||
|
ret = set_dma_callback(drv_data->dma_channel,
|
||||||
|
bfin_spi_dma_irq_handler, drv_data);
|
||||||
|
if (ret) {
|
||||||
|
dev_err(&spi->dev, "Unable to set dma callback\n");
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
dma_disable_irq(drv_data->dma_channel);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (chip->chip_select_num == 0) {
|
||||||
|
ret = gpio_request(chip->cs_gpio, spi->modalias);
|
||||||
|
if (ret) {
|
||||||
|
dev_err(&spi->dev, "gpio_request() error\n");
|
||||||
|
goto pin_error;
|
||||||
|
}
|
||||||
|
gpio_direction_output(chip->cs_gpio, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
dev_dbg(&spi->dev, "setup spi chip %s, width is %d, dma is %d\n",
|
dev_dbg(&spi->dev, "setup spi chip %s, width is %d, dma is %d\n",
|
||||||
|
@ -1136,14 +1140,38 @@ static int bfin_spi_setup(struct spi_device *spi)
|
||||||
spi_set_ctldata(spi, chip);
|
spi_set_ctldata(spi, chip);
|
||||||
|
|
||||||
dev_dbg(&spi->dev, "chip select number is %d\n", chip->chip_select_num);
|
dev_dbg(&spi->dev, "chip select number is %d\n", chip->chip_select_num);
|
||||||
if ((chip->chip_select_num > 0)
|
if (chip->chip_select_num > 0 &&
|
||||||
&& (chip->chip_select_num <= spi->master->num_chipselect))
|
chip->chip_select_num <= spi->master->num_chipselect) {
|
||||||
peripheral_request(ssel[spi->master->bus_num]
|
ret = peripheral_request(ssel[spi->master->bus_num]
|
||||||
[chip->chip_select_num-1], spi->modalias);
|
[chip->chip_select_num-1], spi->modalias);
|
||||||
|
if (ret) {
|
||||||
|
dev_err(&spi->dev, "peripheral_request() error\n");
|
||||||
|
goto pin_error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
bfin_spi_cs_deactive(drv_data, chip);
|
bfin_spi_cs_deactive(drv_data, chip);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
pin_error:
|
||||||
|
if (chip->chip_select_num == 0)
|
||||||
|
gpio_free(chip->cs_gpio);
|
||||||
|
else
|
||||||
|
peripheral_free(ssel[spi->master->bus_num]
|
||||||
|
[chip->chip_select_num - 1]);
|
||||||
|
error:
|
||||||
|
if (chip) {
|
||||||
|
if (drv_data->dma_requested)
|
||||||
|
free_dma(drv_data->dma_channel);
|
||||||
|
drv_data->dma_requested = 0;
|
||||||
|
|
||||||
|
kfree(chip);
|
||||||
|
/* prevent free 'chip' twice */
|
||||||
|
spi_set_ctldata(spi, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -1166,6 +1194,8 @@ static void bfin_spi_cleanup(struct spi_device *spi)
|
||||||
gpio_free(chip->cs_gpio);
|
gpio_free(chip->cs_gpio);
|
||||||
|
|
||||||
kfree(chip);
|
kfree(chip);
|
||||||
|
/* prevent free 'chip' twice */
|
||||||
|
spi_set_ctldata(spi, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int bfin_spi_init_queue(struct driver_data *drv_data)
|
static inline int bfin_spi_init_queue(struct driver_data *drv_data)
|
||||||
|
|
Loading…
Reference in New Issue