Skip to content

Commit 23261f0

Browse files
jhovoldbroonie
authored andcommitted
ASoC: stm32: sai: fix OF node leak on probe
The reference taken to the sync provider OF node when probing the platform device is currently only dropped if the set_sync() callback fails during DAI probe. Make sure to drop the reference on platform probe failures (e.g. probe deferral) and on driver unbind. This also avoids a potential use-after-free in case the DAI is ever reprobed without first rebinding the platform driver. Fixes: 5914d28 ("ASoC: stm32: sai: Add synchronization support") Fixes: d4180b4 ("ASoC: stm32: sai: fix set_sync service") Cc: Olivier Moysan <olivier.moysan@st.com> Cc: stable@vger.kernel.org # 4.16: d4180b4 Signed-off-by: Johan Hovold <johan@kernel.org> Reviewed-by: olivier moysan <olivier.moysan@foss.st.com> Link: https://patch.msgid.link/20251124104908.15754-4-johan@kernel.org Signed-off-by: Mark Brown <broonie@kernel.org>
1 parent 312ec2f commit 23261f0

2 files changed

Lines changed: 19 additions & 16 deletions

File tree

sound/soc/stm/stm32_sai.c

Lines changed: 3 additions & 9 deletions
Original file line numberDiff line numberDiff line change
@@ -138,7 +138,6 @@ static int stm32_sai_set_sync(struct stm32_sai_data *sai_client,
138138
if (!pdev) {
139139
dev_err(&sai_client->pdev->dev,
140140
"Device not found for node %pOFn\n", np_provider);
141-
of_node_put(np_provider);
142141
return -ENODEV;
143142
}
144143

@@ -147,21 +146,16 @@ static int stm32_sai_set_sync(struct stm32_sai_data *sai_client,
147146
if (!sai_provider) {
148147
dev_err(&sai_client->pdev->dev,
149148
"SAI sync provider data not found\n");
150-
ret = -EINVAL;
151-
goto error;
149+
return -EINVAL;
152150
}
153151

154152
/* Configure sync client */
155153
ret = stm32_sai_sync_conf_client(sai_client, synci);
156154
if (ret < 0)
157-
goto error;
155+
return ret;
158156

159157
/* Configure sync provider */
160-
ret = stm32_sai_sync_conf_provider(sai_provider, synco);
161-
162-
error:
163-
of_node_put(np_provider);
164-
return ret;
158+
return stm32_sai_sync_conf_provider(sai_provider, synco);
165159
}
166160

167161
static int stm32_sai_get_parent_clk(struct stm32_sai_data *sai)

sound/soc/stm/stm32_sai_sub.c

Lines changed: 16 additions & 7 deletions
Original file line numberDiff line numberDiff line change
@@ -1586,7 +1586,8 @@ static int stm32_sai_sub_parse_of(struct platform_device *pdev,
15861586
dev_err(&pdev->dev,
15871587
"External synchro not supported\n");
15881588
of_node_put(args.np);
1589-
return -EINVAL;
1589+
ret = -EINVAL;
1590+
goto err_put_sync_provider;
15901591
}
15911592
sai->sync = SAI_SYNC_EXTERNAL;
15921593

@@ -1595,7 +1596,8 @@ static int stm32_sai_sub_parse_of(struct platform_device *pdev,
15951596
(sai->synci > (SAI_GCR_SYNCIN_MAX + 1))) {
15961597
dev_err(&pdev->dev, "Wrong SAI index\n");
15971598
of_node_put(args.np);
1598-
return -EINVAL;
1599+
ret = -EINVAL;
1600+
goto err_put_sync_provider;
15991601
}
16001602

16011603
if (of_property_match_string(args.np, "compatible",
@@ -1609,7 +1611,8 @@ static int stm32_sai_sub_parse_of(struct platform_device *pdev,
16091611
if (!sai->synco) {
16101612
dev_err(&pdev->dev, "Unknown SAI sub-block\n");
16111613
of_node_put(args.np);
1612-
return -EINVAL;
1614+
ret = -EINVAL;
1615+
goto err_put_sync_provider;
16131616
}
16141617
}
16151618

@@ -1619,13 +1622,15 @@ static int stm32_sai_sub_parse_of(struct platform_device *pdev,
16191622

16201623
of_node_put(args.np);
16211624
sai->sai_ck = devm_clk_get(&pdev->dev, "sai_ck");
1622-
if (IS_ERR(sai->sai_ck))
1623-
return dev_err_probe(&pdev->dev, PTR_ERR(sai->sai_ck),
1624-
"Missing kernel clock sai_ck\n");
1625+
if (IS_ERR(sai->sai_ck)) {
1626+
ret = dev_err_probe(&pdev->dev, PTR_ERR(sai->sai_ck),
1627+
"Missing kernel clock sai_ck\n");
1628+
goto err_put_sync_provider;
1629+
}
16251630

16261631
ret = clk_prepare(sai->pdata->pclk);
16271632
if (ret < 0)
1628-
return ret;
1633+
goto err_put_sync_provider;
16291634

16301635
if (STM_SAI_IS_F4(sai->pdata))
16311636
return 0;
@@ -1647,6 +1652,8 @@ static int stm32_sai_sub_parse_of(struct platform_device *pdev,
16471652

16481653
err_unprepare_pclk:
16491654
clk_unprepare(sai->pdata->pclk);
1655+
err_put_sync_provider:
1656+
of_node_put(sai->np_sync_provider);
16501657

16511658
return ret;
16521659
}
@@ -1720,6 +1727,7 @@ static int stm32_sai_sub_probe(struct platform_device *pdev)
17201727

17211728
err_unprepare_pclk:
17221729
clk_unprepare(sai->pdata->pclk);
1730+
of_node_put(sai->np_sync_provider);
17231731

17241732
return ret;
17251733
}
@@ -1732,6 +1740,7 @@ static void stm32_sai_sub_remove(struct platform_device *pdev)
17321740
snd_dmaengine_pcm_unregister(&pdev->dev);
17331741
snd_soc_unregister_component(&pdev->dev);
17341742
pm_runtime_disable(&pdev->dev);
1743+
of_node_put(sai->np_sync_provider);
17351744
}
17361745

17371746
static int stm32_sai_sub_suspend(struct device *dev)

0 commit comments

Comments
 (0)