mirror of
https://github.com/openwrt/openwrt.git
synced 2025-01-10 06:52:53 +00:00
335 lines
9.3 KiB
Diff
335 lines
9.3 KiB
Diff
|
From 1ffba0c4d1122688268e59832a5e2bbc0917cac7 Mon Sep 17 00:00:00 2001
|
||
|
From: Camelia Groza <camelia.groza@nxp.com>
|
||
|
Date: Wed, 3 Oct 2018 16:37:06 +0300
|
||
|
Subject: [PATCH] sdk_dpaa: ceetm: avoid double frees on error paths
|
||
|
|
||
|
The stack calls the destroy() callback when a qdisc init() fails.
|
||
|
We stop calling it ourselves and trust the stack do the cleanup.
|
||
|
|
||
|
Signed-off-by: Camelia Groza <camelia.groza@nxp.com>
|
||
|
---
|
||
|
.../ethernet/freescale/sdk_dpaa/dpaa_eth_ceetm.c | 94 ++++++++--------------
|
||
|
1 file changed, 34 insertions(+), 60 deletions(-)
|
||
|
|
||
|
--- a/drivers/net/ethernet/freescale/sdk_dpaa/dpaa_eth_ceetm.c
|
||
|
+++ b/drivers/net/ethernet/freescale/sdk_dpaa/dpaa_eth_ceetm.c
|
||
|
@@ -687,16 +687,13 @@ static int ceetm_init_root(struct Qdisc
|
||
|
|
||
|
/* Validate inputs */
|
||
|
if (sch->parent != TC_H_ROOT) {
|
||
|
- pr_err("CEETM: a root ceetm qdisc can not be attached to a class\n");
|
||
|
- tcf_block_put(priv->block);
|
||
|
- qdisc_class_hash_destroy(&priv->clhash);
|
||
|
+ pr_err("CEETM: a root ceetm qdisc must be root\n");
|
||
|
return -EINVAL;
|
||
|
}
|
||
|
|
||
|
if (!mac_dev) {
|
||
|
pr_err("CEETM: the interface is lacking a mac\n");
|
||
|
- err = -EINVAL;
|
||
|
- goto err_init_root;
|
||
|
+ return -EINVAL;
|
||
|
}
|
||
|
|
||
|
/* Pre-allocate underlying pfifo qdiscs.
|
||
|
@@ -713,8 +710,7 @@ static int ceetm_init_root(struct Qdisc
|
||
|
sizeof(priv->root.qdiscs[0]),
|
||
|
GFP_KERNEL);
|
||
|
if (!priv->root.qdiscs) {
|
||
|
- err = -ENOMEM;
|
||
|
- goto err_init_root;
|
||
|
+ return -ENOMEM;
|
||
|
}
|
||
|
|
||
|
for (i = 0; i < dev->num_tx_queues; i++) {
|
||
|
@@ -724,10 +720,8 @@ static int ceetm_init_root(struct Qdisc
|
||
|
|
||
|
qdisc = qdisc_create_dflt(dev_queue, &pfifo_qdisc_ops,
|
||
|
parent_id, extack);
|
||
|
- if (!qdisc) {
|
||
|
- err = -ENOMEM;
|
||
|
- goto err_init_root;
|
||
|
- }
|
||
|
+ if (!qdisc)
|
||
|
+ return -ENOMEM;
|
||
|
|
||
|
priv->root.qdiscs[i] = qdisc;
|
||
|
qdisc->flags |= TCQ_F_ONETXQUEUE;
|
||
|
@@ -739,8 +733,7 @@ static int ceetm_init_root(struct Qdisc
|
||
|
if (!priv->root.qstats) {
|
||
|
pr_err(KBUILD_BASENAME " : %s : alloc_percpu() failed\n",
|
||
|
__func__);
|
||
|
- err = -ENOMEM;
|
||
|
- goto err_init_root;
|
||
|
+ return -ENOMEM;
|
||
|
}
|
||
|
|
||
|
priv->shaped = qopt->shaped;
|
||
|
@@ -754,7 +747,7 @@ static int ceetm_init_root(struct Qdisc
|
||
|
if (err) {
|
||
|
pr_err(KBUILD_BASENAME " : %s : failed to claim the SP\n",
|
||
|
__func__);
|
||
|
- goto err_init_root;
|
||
|
+ return err;
|
||
|
}
|
||
|
|
||
|
priv->root.sp = sp;
|
||
|
@@ -766,7 +759,7 @@ static int ceetm_init_root(struct Qdisc
|
||
|
if (err) {
|
||
|
pr_err(KBUILD_BASENAME " : %s : failed to claim the LNI\n",
|
||
|
__func__);
|
||
|
- goto err_init_root;
|
||
|
+ return err;
|
||
|
}
|
||
|
|
||
|
priv->root.lni = lni;
|
||
|
@@ -775,7 +768,7 @@ static int ceetm_init_root(struct Qdisc
|
||
|
if (err) {
|
||
|
pr_err(KBUILD_BASENAME " : %s : failed to link the SP and LNI\n",
|
||
|
__func__);
|
||
|
- goto err_init_root;
|
||
|
+ return err;
|
||
|
}
|
||
|
|
||
|
lni->sp = sp;
|
||
|
@@ -786,7 +779,7 @@ static int ceetm_init_root(struct Qdisc
|
||
|
if (err) {
|
||
|
pr_err(KBUILD_BASENAME " : %s : failed to configure the LNI shaper\n",
|
||
|
__func__);
|
||
|
- goto err_init_root;
|
||
|
+ return err;
|
||
|
}
|
||
|
|
||
|
bps = priv->root.rate << 3; /* Bps -> bps */
|
||
|
@@ -794,7 +787,7 @@ static int ceetm_init_root(struct Qdisc
|
||
|
if (err) {
|
||
|
pr_err(KBUILD_BASENAME " : %s : failed to configure the LNI shaper\n",
|
||
|
__func__);
|
||
|
- goto err_init_root;
|
||
|
+ return err;
|
||
|
}
|
||
|
|
||
|
bps = priv->root.ceil << 3; /* Bps -> bps */
|
||
|
@@ -802,7 +795,7 @@ static int ceetm_init_root(struct Qdisc
|
||
|
if (err) {
|
||
|
pr_err(KBUILD_BASENAME " : %s : failed to configure the LNI shaper\n",
|
||
|
__func__);
|
||
|
- goto err_init_root;
|
||
|
+ return err;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
@@ -810,10 +803,6 @@ static int ceetm_init_root(struct Qdisc
|
||
|
|
||
|
dpa_enable_ceetm(dev);
|
||
|
return 0;
|
||
|
-
|
||
|
-err_init_root:
|
||
|
- ceetm_destroy(sch);
|
||
|
- return err;
|
||
|
}
|
||
|
|
||
|
/* Configure a prio ceetm qdisc */
|
||
|
@@ -830,15 +819,13 @@ static int ceetm_init_prio(struct Qdisc
|
||
|
|
||
|
if (sch->parent == TC_H_ROOT) {
|
||
|
pr_err("CEETM: a prio ceetm qdisc can not be root\n");
|
||
|
- err = -EINVAL;
|
||
|
- goto err_init_prio;
|
||
|
+ return -EINVAL;
|
||
|
}
|
||
|
|
||
|
parent_qdisc = qdisc_lookup(dev, TC_H_MAJ(sch->parent));
|
||
|
if (strcmp(parent_qdisc->ops->id, ceetm_qdisc_ops.id)) {
|
||
|
pr_err("CEETM: a ceetm qdisc can not be attached to other qdisc/class types\n");
|
||
|
- err = -EINVAL;
|
||
|
- goto err_init_prio;
|
||
|
+ return -EINVAL;
|
||
|
}
|
||
|
|
||
|
/* Obtain the parent root ceetm_class */
|
||
|
@@ -846,8 +833,7 @@ static int ceetm_init_prio(struct Qdisc
|
||
|
|
||
|
if (!parent_cl || parent_cl->type != CEETM_ROOT) {
|
||
|
pr_err("CEETM: a prio ceetm qdiscs can be added only under a root ceetm class\n");
|
||
|
- err = -EINVAL;
|
||
|
- goto err_init_prio;
|
||
|
+ return -EINVAL;
|
||
|
}
|
||
|
|
||
|
priv->prio.parent = parent_cl;
|
||
|
@@ -863,8 +849,7 @@ static int ceetm_init_prio(struct Qdisc
|
||
|
if (!child_cl) {
|
||
|
pr_err(KBUILD_BASENAME " : %s : kzalloc() failed\n",
|
||
|
__func__);
|
||
|
- err = -ENOMEM;
|
||
|
- goto err_init_prio;
|
||
|
+ return -ENOMEM;
|
||
|
}
|
||
|
|
||
|
child_cl->prio.cstats = alloc_percpu(struct ceetm_class_stats);
|
||
|
@@ -907,8 +892,7 @@ static int ceetm_init_prio(struct Qdisc
|
||
|
|
||
|
err_init_prio_cls:
|
||
|
ceetm_cls_destroy(sch, child_cl);
|
||
|
-err_init_prio:
|
||
|
- ceetm_destroy(sch);
|
||
|
+ /* Note: ceetm_destroy() will be called by our caller */
|
||
|
return err;
|
||
|
}
|
||
|
|
||
|
@@ -928,16 +912,14 @@ static int ceetm_init_wbfs(struct Qdisc
|
||
|
/* Validate inputs */
|
||
|
if (sch->parent == TC_H_ROOT) {
|
||
|
pr_err("CEETM: a wbfs ceetm qdiscs can not be root\n");
|
||
|
- err = -EINVAL;
|
||
|
- goto err_init_wbfs;
|
||
|
+ return -EINVAL;
|
||
|
}
|
||
|
|
||
|
/* Obtain the parent prio ceetm qdisc */
|
||
|
parent_qdisc = qdisc_lookup(dev, TC_H_MAJ(sch->parent));
|
||
|
if (strcmp(parent_qdisc->ops->id, ceetm_qdisc_ops.id)) {
|
||
|
pr_err("CEETM: a ceetm qdisc can not be attached to other qdisc/class types\n");
|
||
|
- err = -EINVAL;
|
||
|
- goto err_init_wbfs;
|
||
|
+ return -EINVAL;
|
||
|
}
|
||
|
|
||
|
/* Obtain the parent prio ceetm class */
|
||
|
@@ -946,28 +928,24 @@ static int ceetm_init_wbfs(struct Qdisc
|
||
|
|
||
|
if (!parent_cl || parent_cl->type != CEETM_PRIO) {
|
||
|
pr_err("CEETM: a wbfs ceetm qdiscs can be added only under a prio ceetm class\n");
|
||
|
- err = -EINVAL;
|
||
|
- goto err_init_wbfs;
|
||
|
+ return -EINVAL;
|
||
|
}
|
||
|
|
||
|
if (!qopt->qcount || !qopt->qweight[0]) {
|
||
|
pr_err("CEETM: qcount and qweight are mandatory for a wbfs ceetm qdisc\n");
|
||
|
- err = -EINVAL;
|
||
|
- goto err_init_wbfs;
|
||
|
+ return -EINVAL;
|
||
|
}
|
||
|
|
||
|
priv->shaped = parent_cl->shaped;
|
||
|
|
||
|
if (!priv->shaped && (qopt->cr || qopt->er)) {
|
||
|
pr_err("CEETM: CR/ER can be enabled only for shaped wbfs ceetm qdiscs\n");
|
||
|
- err = -EINVAL;
|
||
|
- goto err_init_wbfs;
|
||
|
+ return -EINVAL;
|
||
|
}
|
||
|
|
||
|
if (priv->shaped && !(qopt->cr || qopt->er)) {
|
||
|
pr_err("CEETM: either CR or ER must be enabled for shaped wbfs ceetm qdiscs\n");
|
||
|
- err = -EINVAL;
|
||
|
- goto err_init_wbfs;
|
||
|
+ return -EINVAL;
|
||
|
}
|
||
|
|
||
|
/* Obtain the parent root ceetm class */
|
||
|
@@ -975,16 +953,14 @@ static int ceetm_init_wbfs(struct Qdisc
|
||
|
if ((root_cl->root.wbfs_grp_a && root_cl->root.wbfs_grp_b) ||
|
||
|
root_cl->root.wbfs_grp_large) {
|
||
|
pr_err("CEETM: no more wbfs classes are available\n");
|
||
|
- err = -EINVAL;
|
||
|
- goto err_init_wbfs;
|
||
|
+ return -EINVAL;
|
||
|
}
|
||
|
|
||
|
if ((root_cl->root.wbfs_grp_a || root_cl->root.wbfs_grp_b) &&
|
||
|
qopt->qcount == CEETM_MAX_WBFS_QCOUNT) {
|
||
|
pr_err("CEETM: only %d wbfs classes are available\n",
|
||
|
CEETM_MIN_WBFS_QCOUNT);
|
||
|
- err = -EINVAL;
|
||
|
- goto err_init_wbfs;
|
||
|
+ return -EINVAL;
|
||
|
}
|
||
|
|
||
|
priv->wbfs.parent = parent_cl;
|
||
|
@@ -1013,7 +989,7 @@ static int ceetm_init_wbfs(struct Qdisc
|
||
|
if (err) {
|
||
|
pr_err(KBUILD_BASENAME " : %s : failed to get group details\n",
|
||
|
__func__);
|
||
|
- goto err_init_wbfs;
|
||
|
+ return err;
|
||
|
}
|
||
|
|
||
|
small_group = true;
|
||
|
@@ -1031,7 +1007,7 @@ static int ceetm_init_wbfs(struct Qdisc
|
||
|
if (err) {
|
||
|
pr_err(KBUILD_BASENAME " : %s : failed to get group details\n",
|
||
|
__func__);
|
||
|
- goto err_init_wbfs;
|
||
|
+ return err;
|
||
|
}
|
||
|
|
||
|
small_group = true;
|
||
|
@@ -1044,7 +1020,7 @@ static int ceetm_init_wbfs(struct Qdisc
|
||
|
err = qman_ceetm_channel_set_group(priv->wbfs.ch, small_group, prio_a,
|
||
|
prio_b);
|
||
|
if (err)
|
||
|
- goto err_init_wbfs;
|
||
|
+ return err;
|
||
|
|
||
|
if (priv->shaped) {
|
||
|
err = qman_ceetm_channel_set_group_cr_eligibility(priv->wbfs.ch,
|
||
|
@@ -1053,7 +1029,7 @@ static int ceetm_init_wbfs(struct Qdisc
|
||
|
if (err) {
|
||
|
pr_err(KBUILD_BASENAME " : %s : failed to set group CR eligibility\n",
|
||
|
__func__);
|
||
|
- goto err_init_wbfs;
|
||
|
+ return err;
|
||
|
}
|
||
|
|
||
|
err = qman_ceetm_channel_set_group_er_eligibility(priv->wbfs.ch,
|
||
|
@@ -1062,7 +1038,7 @@ static int ceetm_init_wbfs(struct Qdisc
|
||
|
if (err) {
|
||
|
pr_err(KBUILD_BASENAME " : %s : failed to set group ER eligibility\n",
|
||
|
__func__);
|
||
|
- goto err_init_wbfs;
|
||
|
+ return err;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
@@ -1072,8 +1048,7 @@ static int ceetm_init_wbfs(struct Qdisc
|
||
|
if (!child_cl) {
|
||
|
pr_err(KBUILD_BASENAME " : %s : kzalloc() failed\n",
|
||
|
__func__);
|
||
|
- err = -ENOMEM;
|
||
|
- goto err_init_wbfs;
|
||
|
+ return -ENOMEM;
|
||
|
}
|
||
|
|
||
|
child_cl->wbfs.cstats = alloc_percpu(struct ceetm_class_stats);
|
||
|
@@ -1130,8 +1105,7 @@ static int ceetm_init_wbfs(struct Qdisc
|
||
|
|
||
|
err_init_wbfs_cls:
|
||
|
ceetm_cls_destroy(sch, child_cl);
|
||
|
-err_init_wbfs:
|
||
|
- ceetm_destroy(sch);
|
||
|
+ /* Note: ceetm_destroy() will be called by our caller */
|
||
|
return err;
|
||
|
}
|
||
|
|
||
|
@@ -1202,7 +1176,7 @@ static int ceetm_init(struct Qdisc *sch,
|
||
|
break;
|
||
|
default:
|
||
|
pr_err(KBUILD_BASENAME " : %s : invalid qdisc\n", __func__);
|
||
|
- ceetm_destroy(sch);
|
||
|
+ /* Note: ceetm_destroy() will be called by our caller */
|
||
|
ret = -EINVAL;
|
||
|
}
|
||
|
|
||
|
@@ -1549,7 +1523,7 @@ static int ceetm_cls_change(struct Qdisc
|
||
|
}
|
||
|
|
||
|
if (!cl && priv->type != CEETM_ROOT) {
|
||
|
- pr_err("CEETM: only root ceetm classes can be attached to the root ceetm qdisc\n");
|
||
|
+ pr_err("CEETM: root ceetm classes can be attached to the root ceetm qdisc only\n");
|
||
|
return -EINVAL;
|
||
|
}
|
||
|
|