aboutsummaryrefslogtreecommitdiffstats
path: root/drivers
diff options
context:
space:
mode:
authorDaniel Mack <daniel@caiaq.de>2009-03-24 20:18:35 -0400
committerMike Frysinger <vapier@gentoo.org>2010-10-18 02:49:28 -0400
commitac01e97d644da8e947ffa1bde5083290fe2e36e7 (patch)
treec2a97d4c239afb65376dbaa81aca7610b0883369 /drivers
parent2b666ca4a68cbc22483b0f2e1ba3c0e59b01ae9e (diff)
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>
Diffstat (limited to 'drivers')
-rw-r--r--drivers/spi/spi_bfin5xx.c120
1 files changed, 75 insertions, 45 deletions
diff --git a/drivers/spi/spi_bfin5xx.c b/drivers/spi/spi_bfin5xx.c
index 10a6dc3d37ac..4f20b923a95c 100644
--- a/drivers/spi/spi_bfin5xx.c
+++ b/drivers/spi/spi_bfin5xx.c
@@ -1006,20 +1006,24 @@ static u16 ssel[][MAX_SPI_SSEL] = {
1006/* first setup for new devices */ 1006/* first setup for new devices */
1007static int bfin_spi_setup(struct spi_device *spi) 1007static int bfin_spi_setup(struct spi_device *spi)
1008{ 1008{
1009 struct bfin5xx_spi_chip *chip_info = NULL; 1009 struct bfin5xx_spi_chip *chip_info;
1010 struct chip_data *chip; 1010 struct chip_data *chip = NULL;
1011 struct driver_data *drv_data = spi_master_get_devdata(spi->master); 1011 struct driver_data *drv_data = spi_master_get_devdata(spi->master);
1012 int ret; 1012 int ret = -EINVAL;
1013 1013
1014 if (spi->bits_per_word != 8 && spi->bits_per_word != 16) 1014 if (spi->bits_per_word != 8 && spi->bits_per_word != 16)
1015 return -EINVAL; 1015 goto error;
1016 1016
1017 /* Only alloc (or use chip_info) on first setup */ 1017 /* Only alloc (or use chip_info) on first setup */
1018 chip_info = NULL;
1018 chip = spi_get_ctldata(spi); 1019 chip = spi_get_ctldata(spi);
1019 if (chip == NULL) { 1020 if (chip == NULL) {
1020 chip = kzalloc(sizeof(struct chip_data), GFP_KERNEL); 1021 chip = kzalloc(sizeof(*chip), GFP_KERNEL);
1021 if (!chip) 1022 if (!chip) {
1022 return -ENOMEM; 1023 dev_err(&spi->dev, "cannot allocate chip data\n");
1024 ret = -ENOMEM;
1025 goto error;
1026 }
1023 1027
1024 chip->enable_dma = 0; 1028 chip->enable_dma = 0;
1025 chip_info = spi->controller_data; 1029 chip_info = spi->controller_data;
@@ -1036,7 +1040,7 @@ static int bfin_spi_setup(struct spi_device *spi)
1036 if (chip_info->ctl_reg & (SPE|MSTR|CPOL|CPHA|LSBF|SIZE)) { 1040 if (chip_info->ctl_reg & (SPE|MSTR|CPOL|CPHA|LSBF|SIZE)) {
1037 dev_err(&spi->dev, "do not set bits in ctl_reg " 1041 dev_err(&spi->dev, "do not set bits in ctl_reg "
1038 "that the SPI framework manages\n"); 1042 "that the SPI framework manages\n");
1039 return -EINVAL; 1043 goto error;
1040 } 1044 }
1041 1045
1042 chip->enable_dma = chip_info->enable_dma != 0 1046 chip->enable_dma = chip_info->enable_dma != 0
@@ -1060,26 +1064,6 @@ static int bfin_spi_setup(struct spi_device *spi)
1060 chip->ctl_reg |= MSTR; 1064 chip->ctl_reg |= MSTR;
1061 1065
1062 /* 1066 /*
1063 * if any one SPI chip is registered and wants DMA, request the
1064 * DMA channel for it
1065 */
1066 if (chip->enable_dma && !drv_data->dma_requested) {
1067 /* register dma irq handler */
1068 if (request_dma(drv_data->dma_channel, "BFIN_SPI_DMA") < 0) {
1069 dev_dbg(&spi->dev,
1070 "Unable to request BlackFin SPI DMA channel\n");
1071 return -ENODEV;
1072 }
1073 if (set_dma_callback(drv_data->dma_channel,
1074 bfin_spi_dma_irq_handler, drv_data) < 0) {
1075 dev_dbg(&spi->dev, "Unable to set dma callback\n");
1076 return -EPERM;
1077 }
1078 dma_disable_irq(drv_data->dma_channel);
1079 drv_data->dma_requested = 1;
1080 }
1081
1082 /*
1083 * Notice: for blackfin, the speed_hz is the value of register 1067 * Notice: for blackfin, the speed_hz is the value of register
1084 * SPI_BAUD, not the real baudrate 1068 * SPI_BAUD, not the real baudrate
1085 */ 1069 */
@@ -1087,16 +1071,6 @@ static int bfin_spi_setup(struct spi_device *spi)
1087 chip->flag = 1 << (spi->chip_select); 1071 chip->flag = 1 << (spi->chip_select);
1088 chip->chip_select_num = spi->chip_select; 1072 chip->chip_select_num = spi->chip_select;
1089 1073
1090 if (chip->chip_select_num == 0) {
1091 ret = gpio_request(chip->cs_gpio, spi->modalias);
1092 if (ret) {
1093 if (drv_data->dma_requested)
1094 free_dma(drv_data->dma_channel);
1095 return ret;
1096 }
1097 gpio_direction_output(chip->cs_gpio, 1);
1098 }
1099
1100 switch (chip->bits_per_word) { 1074 switch (chip->bits_per_word) {
1101 case 8: 1075 case 8:
1102 chip->n_bytes = 1; 1076 chip->n_bytes = 1;
@@ -1123,9 +1097,39 @@ static int bfin_spi_setup(struct spi_device *spi)
1123 default: 1097 default:
1124 dev_err(&spi->dev, "%d bits_per_word is not supported\n", 1098 dev_err(&spi->dev, "%d bits_per_word is not supported\n",
1125 chip->bits_per_word); 1099 chip->bits_per_word);
1126 if (chip_info) 1100 goto error;
1127 kfree(chip); 1101 }
1128 return -ENODEV; 1102
1103 /*
1104 * if any one SPI chip is registered and wants DMA, request the
1105 * DMA channel for it
1106 */
1107 if (chip->enable_dma && !drv_data->dma_requested) {
1108 /* register dma irq handler */
1109 ret = request_dma(drv_data->dma_channel, "BFIN_SPI_DMA");
1110 if (ret) {
1111 dev_err(&spi->dev,
1112 "Unable to request BlackFin SPI DMA channel\n");
1113 goto error;
1114 }
1115 drv_data->dma_requested = 1;
1116
1117 ret = set_dma_callback(drv_data->dma_channel,
1118 bfin_spi_dma_irq_handler, drv_data);
1119 if (ret) {
1120 dev_err(&spi->dev, "Unable to set dma callback\n");
1121 goto error;
1122 }
1123 dma_disable_irq(drv_data->dma_channel);
1124 }
1125
1126 if (chip->chip_select_num == 0) {
1127 ret = gpio_request(chip->cs_gpio, spi->modalias);
1128 if (ret) {
1129 dev_err(&spi->dev, "gpio_request() error\n");
1130 goto pin_error;
1131 }
1132 gpio_direction_output(chip->cs_gpio, 1);
1129 } 1133 }
1130 1134
1131 dev_dbg(&spi->dev, "setup spi chip %s, width is %d, dma is %d\n", 1135 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)
1136 spi_set_ctldata(spi, chip); 1140 spi_set_ctldata(spi, chip);
1137 1141
1138 dev_dbg(&spi->dev, "chip select number is %d\n", chip->chip_select_num); 1142 dev_dbg(&spi->dev, "chip select number is %d\n", chip->chip_select_num);
1139 if ((chip->chip_select_num > 0) 1143 if (chip->chip_select_num > 0 &&
1140 && (chip->chip_select_num <= spi->master->num_chipselect)) 1144 chip->chip_select_num <= spi->master->num_chipselect) {
1141 peripheral_request(ssel[spi->master->bus_num] 1145 ret = peripheral_request(ssel[spi->master->bus_num]
1142 [chip->chip_select_num-1], spi->modalias); 1146 [chip->chip_select_num-1], spi->modalias);
1147 if (ret) {
1148 dev_err(&spi->dev, "peripheral_request() error\n");
1149 goto pin_error;
1150 }
1151 }
1143 1152
1144 bfin_spi_cs_deactive(drv_data, chip); 1153 bfin_spi_cs_deactive(drv_data, chip);
1145 1154
1146 return 0; 1155 return 0;
1156
1157 pin_error:
1158 if (chip->chip_select_num == 0)
1159 gpio_free(chip->cs_gpio);
1160 else
1161 peripheral_free(ssel[spi->master->bus_num]
1162 [chip->chip_select_num - 1]);
1163 error:
1164 if (chip) {
1165 if (drv_data->dma_requested)
1166 free_dma(drv_data->dma_channel);
1167 drv_data->dma_requested = 0;
1168
1169 kfree(chip);
1170 /* prevent free 'chip' twice */
1171 spi_set_ctldata(spi, NULL);
1172 }
1173
1174 return ret;
1147} 1175}
1148 1176
1149/* 1177/*
@@ -1166,6 +1194,8 @@ static void bfin_spi_cleanup(struct spi_device *spi)
1166 gpio_free(chip->cs_gpio); 1194 gpio_free(chip->cs_gpio);
1167 1195
1168 kfree(chip); 1196 kfree(chip);
1197 /* prevent free 'chip' twice */
1198 spi_set_ctldata(spi, NULL);
1169} 1199}
1170 1200
1171static inline int bfin_spi_init_queue(struct driver_data *drv_data) 1201static inline int bfin_spi_init_queue(struct driver_data *drv_data)