Commit 018af9be authored by Christophe JAILLET's avatar Christophe JAILLET Committed by Vinod Koul

dmaengine: ti: k3-udma-glue: Fix an error handling path in 'k3_udma_glue_cfg_rx_flow()'

All but one error handling paths in the 'k3_udma_glue_cfg_rx_flow()'
function 'goto err' and call 'k3_udma_glue_release_rx_flow()'.

This not correct because this function has a 'channel->flows_ready--;' at
the end, but 'flows_ready' has not been incremented here, when we branch to
the error handling path.

In order to keep a correct value in 'flows_ready', un-roll
'k3_udma_glue_release_rx_flow()', simplify it, add some labels and branch
at the correct places when an error is detected.

Doing so, we also NULLify 'flow->udma_rflow' in a path that was lacking it.

Fixes: d7024191 ("dmaengine: ti: k3-udma: Add glue layer for non DMAengine user")
Signed-off-by: default avatarChristophe JAILLET <christophe.jaillet@wanadoo.fr>
Acked-by: default avatarPeter Ujfalusi <peter.ujfalusi@ti.com>
Link: https://lore.kernel.org/r/20200318191209.1267-1-christophe.jaillet@wanadoo.frSigned-off-by: default avatarVinod Koul <vkoul@kernel.org>
parent 01c4df39
...@@ -564,12 +564,12 @@ static int k3_udma_glue_cfg_rx_flow(struct k3_udma_glue_rx_channel *rx_chn, ...@@ -564,12 +564,12 @@ static int k3_udma_glue_cfg_rx_flow(struct k3_udma_glue_rx_channel *rx_chn,
if (IS_ERR(flow->udma_rflow)) { if (IS_ERR(flow->udma_rflow)) {
ret = PTR_ERR(flow->udma_rflow); ret = PTR_ERR(flow->udma_rflow);
dev_err(dev, "UDMAX rflow get err %d\n", ret); dev_err(dev, "UDMAX rflow get err %d\n", ret);
goto err; return ret;
} }
if (flow->udma_rflow_id != xudma_rflow_get_id(flow->udma_rflow)) { if (flow->udma_rflow_id != xudma_rflow_get_id(flow->udma_rflow)) {
xudma_rflow_put(rx_chn->common.udmax, flow->udma_rflow); ret = -ENODEV;
return -ENODEV; goto err_rflow_put;
} }
/* request and cfg rings */ /* request and cfg rings */
...@@ -578,7 +578,7 @@ static int k3_udma_glue_cfg_rx_flow(struct k3_udma_glue_rx_channel *rx_chn, ...@@ -578,7 +578,7 @@ static int k3_udma_glue_cfg_rx_flow(struct k3_udma_glue_rx_channel *rx_chn,
if (!flow->ringrx) { if (!flow->ringrx) {
ret = -ENODEV; ret = -ENODEV;
dev_err(dev, "Failed to get RX ring\n"); dev_err(dev, "Failed to get RX ring\n");
goto err; goto err_rflow_put;
} }
flow->ringrxfdq = k3_ringacc_request_ring(rx_chn->common.ringacc, flow->ringrxfdq = k3_ringacc_request_ring(rx_chn->common.ringacc,
...@@ -586,19 +586,19 @@ static int k3_udma_glue_cfg_rx_flow(struct k3_udma_glue_rx_channel *rx_chn, ...@@ -586,19 +586,19 @@ static int k3_udma_glue_cfg_rx_flow(struct k3_udma_glue_rx_channel *rx_chn,
if (!flow->ringrxfdq) { if (!flow->ringrxfdq) {
ret = -ENODEV; ret = -ENODEV;
dev_err(dev, "Failed to get RXFDQ ring\n"); dev_err(dev, "Failed to get RXFDQ ring\n");
goto err; goto err_ringrx_free;
} }
ret = k3_ringacc_ring_cfg(flow->ringrx, &flow_cfg->rx_cfg); ret = k3_ringacc_ring_cfg(flow->ringrx, &flow_cfg->rx_cfg);
if (ret) { if (ret) {
dev_err(dev, "Failed to cfg ringrx %d\n", ret); dev_err(dev, "Failed to cfg ringrx %d\n", ret);
goto err; goto err_ringrxfdq_free;
} }
ret = k3_ringacc_ring_cfg(flow->ringrxfdq, &flow_cfg->rxfdq_cfg); ret = k3_ringacc_ring_cfg(flow->ringrxfdq, &flow_cfg->rxfdq_cfg);
if (ret) { if (ret) {
dev_err(dev, "Failed to cfg ringrxfdq %d\n", ret); dev_err(dev, "Failed to cfg ringrxfdq %d\n", ret);
goto err; goto err_ringrxfdq_free;
} }
if (rx_chn->remote) { if (rx_chn->remote) {
...@@ -648,7 +648,7 @@ static int k3_udma_glue_cfg_rx_flow(struct k3_udma_glue_rx_channel *rx_chn, ...@@ -648,7 +648,7 @@ static int k3_udma_glue_cfg_rx_flow(struct k3_udma_glue_rx_channel *rx_chn,
if (ret) { if (ret) {
dev_err(dev, "flow%d config failed: %d\n", flow->udma_rflow_id, dev_err(dev, "flow%d config failed: %d\n", flow->udma_rflow_id,
ret); ret);
goto err; goto err_ringrxfdq_free;
} }
rx_chn->flows_ready++; rx_chn->flows_ready++;
...@@ -656,8 +656,17 @@ static int k3_udma_glue_cfg_rx_flow(struct k3_udma_glue_rx_channel *rx_chn, ...@@ -656,8 +656,17 @@ static int k3_udma_glue_cfg_rx_flow(struct k3_udma_glue_rx_channel *rx_chn,
flow->udma_rflow_id, rx_chn->flows_ready); flow->udma_rflow_id, rx_chn->flows_ready);
return 0; return 0;
err:
k3_udma_glue_release_rx_flow(rx_chn, flow_idx); err_ringrxfdq_free:
k3_ringacc_ring_free(flow->ringrxfdq);
err_ringrx_free:
k3_ringacc_ring_free(flow->ringrx);
err_rflow_put:
xudma_rflow_put(rx_chn->common.udmax, flow->udma_rflow);
flow->udma_rflow = NULL;
return ret; return ret;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment