diff mbox

ASoC: dapm: Fix kcontrol widget name memory management

Message ID 1436385599-12403-1-git-send-email-lars@metafoo.de (mailing list archive)
State Accepted
Commit e18077b6e5dfe26e9fbbdc1fd1085a1701c24bea
Headers show

Commit Message

Lars-Peter Clausen July 8, 2015, 7:59 p.m. UTC
The name field of the widget template is only used inside
snd_soc_dapm_new_control_unlocked() which allocates a copy for the actual
widget. This means we need to free the name allocated for the template in
dapm_kcontrol_data_alloc() and not the name of the actual widget in
dapm_kcontrol_free(). Otherwise we get a double free on the widget name and
a memory leak on the template name.

Fixes: 773da9b358bf ("ASoC: dapm: Append "Autodisable" to autodisable widget names")
Signed-off-by: Lars-Peter Clausen <lars@metafoo.de>
---
 sound/soc/soc-dapm.c | 10 ++++------
 1 file changed, 4 insertions(+), 6 deletions(-)

Comments

Charles Keepax July 9, 2015, 8:06 a.m. UTC | #1
On Wed, Jul 08, 2015 at 09:59:59PM +0200, Lars-Peter Clausen wrote:
> The name field of the widget template is only used inside
> snd_soc_dapm_new_control_unlocked() which allocates a copy for the actual
> widget. This means we need to free the name allocated for the template in
> dapm_kcontrol_data_alloc() and not the name of the actual widget in
> dapm_kcontrol_free(). Otherwise we get a double free on the widget name and
> a memory leak on the template name.
> 
> Fixes: 773da9b358bf ("ASoC: dapm: Append "Autodisable" to autodisable widget names")
> Signed-off-by: Lars-Peter Clausen <lars@metafoo.de>
> ---

Oops, thanks for fixing that up.

Acked-by: Charles Keepax <ckeepax@opensource.wolfsonmicro.com>

Thanks,
Charles
diff mbox

Patch

diff --git a/sound/soc/soc-dapm.c b/sound/soc/soc-dapm.c
index 6bac491..1628f0c 100644
--- a/sound/soc/soc-dapm.c
+++ b/sound/soc/soc-dapm.c
@@ -358,9 +358,10 @@  static int dapm_kcontrol_data_alloc(struct snd_soc_dapm_widget *widget,
 			data->widget =
 				snd_soc_dapm_new_control_unlocked(widget->dapm,
 				&template);
+			kfree(name);
 			if (!data->widget) {
 				ret = -ENOMEM;
-				goto err_name;
+				goto err_data;
 			}
 		}
 		break;
@@ -391,9 +392,10 @@  static int dapm_kcontrol_data_alloc(struct snd_soc_dapm_widget *widget,
 
 			data->widget = snd_soc_dapm_new_control_unlocked(
 						widget->dapm, &template);
+			kfree(name);
 			if (!data->widget) {
 				ret = -ENOMEM;
-				goto err_name;
+				goto err_data;
 			}
 
 			snd_soc_dapm_add_path(widget->dapm, data->widget,
@@ -408,8 +410,6 @@  static int dapm_kcontrol_data_alloc(struct snd_soc_dapm_widget *widget,
 
 	return 0;
 
-err_name:
-	kfree(name);
 err_data:
 	kfree(data);
 	return ret;
@@ -418,8 +418,6 @@  err_data:
 static void dapm_kcontrol_free(struct snd_kcontrol *kctl)
 {
 	struct dapm_kcontrol_data *data = snd_kcontrol_chip(kctl);
-	if (data->widget)
-		kfree(data->widget->name);
 	kfree(data->wlist);
 	kfree(data);
 }