* [PATCH net] sch_netem: fix issues in netem_change() vs get_dist_table()
@ 2023-06-22 18:15 Eric Dumazet
2023-06-22 18:56 ` Jamal Hadi Salim
` (2 more replies)
0 siblings, 3 replies; 4+ messages in thread
From: Eric Dumazet @ 2023-06-22 18:15 UTC (permalink / raw)
To: David S . Miller, Jakub Kicinski, Paolo Abeni
Cc: netdev, eric.dumazet, Eric Dumazet, syzbot, Stephen Hemminger,
Jamal Hadi Salim, Simon Horman
In blamed commit, I missed that get_dist_table() was allocating
memory using GFP_KERNEL, and acquiring qdisc lock to perform
the swap of newly allocated table with current one.
In this patch, get_dist_table() is allocating memory and
copy user data before we acquire the qdisc lock.
Then we perform swap operations while being protected by the lock.
Note that after this patch netem_change() no longer can do partial changes.
If an error is returned, qdisc conf is left unchanged.
Fixes: 2174a08db80d ("sch_netem: acquire qdisc lock in netem_change()")
Reported-by: syzbot <syzkaller@googlegroups.com>
Signed-off-by: Eric Dumazet <edumazet@google.com>
Cc: Stephen Hemminger <stephen@networkplumber.org>
Cc: Jamal Hadi Salim <jhs@mojatatu.com>
Cc: Simon Horman <simon.horman@corigine.com>
---
net/sched/sch_netem.c | 59 ++++++++++++++++++-------------------------
1 file changed, 25 insertions(+), 34 deletions(-)
diff --git a/net/sched/sch_netem.c b/net/sched/sch_netem.c
index e79be1b3e74da3c154f7ee23e16cc9e8da8f7106..b93ec2a3454ebceea559299f90533cbf1c0f7c26 100644
--- a/net/sched/sch_netem.c
+++ b/net/sched/sch_netem.c
@@ -773,12 +773,10 @@ static void dist_free(struct disttable *d)
* signed 16 bit values.
*/
-static int get_dist_table(struct Qdisc *sch, struct disttable **tbl,
- const struct nlattr *attr)
+static int get_dist_table(struct disttable **tbl, const struct nlattr *attr)
{
size_t n = nla_len(attr)/sizeof(__s16);
const __s16 *data = nla_data(attr);
- spinlock_t *root_lock;
struct disttable *d;
int i;
@@ -793,13 +791,7 @@ static int get_dist_table(struct Qdisc *sch, struct disttable **tbl,
for (i = 0; i < n; i++)
d->table[i] = data[i];
- root_lock = qdisc_root_sleeping_lock(sch);
-
- spin_lock_bh(root_lock);
- swap(*tbl, d);
- spin_unlock_bh(root_lock);
-
- dist_free(d);
+ *tbl = d;
return 0;
}
@@ -956,6 +948,8 @@ static int netem_change(struct Qdisc *sch, struct nlattr *opt,
{
struct netem_sched_data *q = qdisc_priv(sch);
struct nlattr *tb[TCA_NETEM_MAX + 1];
+ struct disttable *delay_dist = NULL;
+ struct disttable *slot_dist = NULL;
struct tc_netem_qopt *qopt;
struct clgstate old_clg;
int old_loss_model = CLG_RANDOM;
@@ -966,6 +960,18 @@ static int netem_change(struct Qdisc *sch, struct nlattr *opt,
if (ret < 0)
return ret;
+ if (tb[TCA_NETEM_DELAY_DIST]) {
+ ret = get_dist_table(&delay_dist, tb[TCA_NETEM_DELAY_DIST]);
+ if (ret)
+ goto table_free;
+ }
+
+ if (tb[TCA_NETEM_SLOT_DIST]) {
+ ret = get_dist_table(&slot_dist, tb[TCA_NETEM_SLOT_DIST]);
+ if (ret)
+ goto table_free;
+ }
+
sch_tree_lock(sch);
/* backup q->clg and q->loss_model */
old_clg = q->clg;
@@ -975,26 +981,17 @@ static int netem_change(struct Qdisc *sch, struct nlattr *opt,
ret = get_loss_clg(q, tb[TCA_NETEM_LOSS]);
if (ret) {
q->loss_model = old_loss_model;
+ q->clg = old_clg;
goto unlock;
}
} else {
q->loss_model = CLG_RANDOM;
}
- if (tb[TCA_NETEM_DELAY_DIST]) {
- ret = get_dist_table(sch, &q->delay_dist,
- tb[TCA_NETEM_DELAY_DIST]);
- if (ret)
- goto get_table_failure;
- }
-
- if (tb[TCA_NETEM_SLOT_DIST]) {
- ret = get_dist_table(sch, &q->slot_dist,
- tb[TCA_NETEM_SLOT_DIST]);
- if (ret)
- goto get_table_failure;
- }
-
+ if (delay_dist)
+ swap(q->delay_dist, delay_dist);
+ if (slot_dist)
+ swap(q->slot_dist, slot_dist);
sch->limit = qopt->limit;
q->latency = PSCHED_TICKS2NS(qopt->latency);
@@ -1044,17 +1041,11 @@ static int netem_change(struct Qdisc *sch, struct nlattr *opt,
unlock:
sch_tree_unlock(sch);
- return ret;
-get_table_failure:
- /* recover clg and loss_model, in case of
- * q->clg and q->loss_model were modified
- * in get_loss_clg()
- */
- q->clg = old_clg;
- q->loss_model = old_loss_model;
-
- goto unlock;
+table_free:
+ dist_free(delay_dist);
+ dist_free(slot_dist);
+ return ret;
}
static int netem_init(struct Qdisc *sch, struct nlattr *opt,
--
2.41.0.178.g377b9f9a00-goog
^ permalink raw reply related [flat|nested] 4+ messages in thread
* Re: [PATCH net] sch_netem: fix issues in netem_change() vs get_dist_table()
2023-06-22 18:15 [PATCH net] sch_netem: fix issues in netem_change() vs get_dist_table() Eric Dumazet
@ 2023-06-22 18:56 ` Jamal Hadi Salim
2023-06-23 7:54 ` Simon Horman
2023-06-24 22:20 ` patchwork-bot+netdevbpf
2 siblings, 0 replies; 4+ messages in thread
From: Jamal Hadi Salim @ 2023-06-22 18:56 UTC (permalink / raw)
To: Eric Dumazet
Cc: David S . Miller, Jakub Kicinski, Paolo Abeni, netdev,
eric.dumazet, syzbot, Stephen Hemminger, Simon Horman
On Thu, Jun 22, 2023 at 2:15 PM Eric Dumazet <edumazet@google.com> wrote:
>
> In blamed commit, I missed that get_dist_table() was allocating
> memory using GFP_KERNEL, and acquiring qdisc lock to perform
> the swap of newly allocated table with current one.
>
> In this patch, get_dist_table() is allocating memory and
> copy user data before we acquire the qdisc lock.
>
> Then we perform swap operations while being protected by the lock.
>
> Note that after this patch netem_change() no longer can do partial changes.
> If an error is returned, qdisc conf is left unchanged.
>
> Fixes: 2174a08db80d ("sch_netem: acquire qdisc lock in netem_change()")
> Reported-by: syzbot <syzkaller@googlegroups.com>
> Signed-off-by: Eric Dumazet <edumazet@google.com>
> Cc: Stephen Hemminger <stephen@networkplumber.org>
> Cc: Jamal Hadi Salim <jhs@mojatatu.com>
> Cc: Simon Horman <simon.horman@corigine.com>
LGTM.
Acked-by: Jamal Hadi Salim <jhs@mojatatu.com>
cheers,
jamal
> ---
> net/sched/sch_netem.c | 59 ++++++++++++++++++-------------------------
> 1 file changed, 25 insertions(+), 34 deletions(-)
>
> diff --git a/net/sched/sch_netem.c b/net/sched/sch_netem.c
> index e79be1b3e74da3c154f7ee23e16cc9e8da8f7106..b93ec2a3454ebceea559299f90533cbf1c0f7c26 100644
> --- a/net/sched/sch_netem.c
> +++ b/net/sched/sch_netem.c
> @@ -773,12 +773,10 @@ static void dist_free(struct disttable *d)
> * signed 16 bit values.
> */
>
> -static int get_dist_table(struct Qdisc *sch, struct disttable **tbl,
> - const struct nlattr *attr)
> +static int get_dist_table(struct disttable **tbl, const struct nlattr *attr)
> {
> size_t n = nla_len(attr)/sizeof(__s16);
> const __s16 *data = nla_data(attr);
> - spinlock_t *root_lock;
> struct disttable *d;
> int i;
>
> @@ -793,13 +791,7 @@ static int get_dist_table(struct Qdisc *sch, struct disttable **tbl,
> for (i = 0; i < n; i++)
> d->table[i] = data[i];
>
> - root_lock = qdisc_root_sleeping_lock(sch);
> -
> - spin_lock_bh(root_lock);
> - swap(*tbl, d);
> - spin_unlock_bh(root_lock);
> -
> - dist_free(d);
> + *tbl = d;
> return 0;
> }
>
> @@ -956,6 +948,8 @@ static int netem_change(struct Qdisc *sch, struct nlattr *opt,
> {
> struct netem_sched_data *q = qdisc_priv(sch);
> struct nlattr *tb[TCA_NETEM_MAX + 1];
> + struct disttable *delay_dist = NULL;
> + struct disttable *slot_dist = NULL;
> struct tc_netem_qopt *qopt;
> struct clgstate old_clg;
> int old_loss_model = CLG_RANDOM;
> @@ -966,6 +960,18 @@ static int netem_change(struct Qdisc *sch, struct nlattr *opt,
> if (ret < 0)
> return ret;
>
> + if (tb[TCA_NETEM_DELAY_DIST]) {
> + ret = get_dist_table(&delay_dist, tb[TCA_NETEM_DELAY_DIST]);
> + if (ret)
> + goto table_free;
> + }
> +
> + if (tb[TCA_NETEM_SLOT_DIST]) {
> + ret = get_dist_table(&slot_dist, tb[TCA_NETEM_SLOT_DIST]);
> + if (ret)
> + goto table_free;
> + }
> +
> sch_tree_lock(sch);
> /* backup q->clg and q->loss_model */
> old_clg = q->clg;
> @@ -975,26 +981,17 @@ static int netem_change(struct Qdisc *sch, struct nlattr *opt,
> ret = get_loss_clg(q, tb[TCA_NETEM_LOSS]);
> if (ret) {
> q->loss_model = old_loss_model;
> + q->clg = old_clg;
> goto unlock;
> }
> } else {
> q->loss_model = CLG_RANDOM;
> }
>
> - if (tb[TCA_NETEM_DELAY_DIST]) {
> - ret = get_dist_table(sch, &q->delay_dist,
> - tb[TCA_NETEM_DELAY_DIST]);
> - if (ret)
> - goto get_table_failure;
> - }
> -
> - if (tb[TCA_NETEM_SLOT_DIST]) {
> - ret = get_dist_table(sch, &q->slot_dist,
> - tb[TCA_NETEM_SLOT_DIST]);
> - if (ret)
> - goto get_table_failure;
> - }
> -
> + if (delay_dist)
> + swap(q->delay_dist, delay_dist);
> + if (slot_dist)
> + swap(q->slot_dist, slot_dist);
> sch->limit = qopt->limit;
>
> q->latency = PSCHED_TICKS2NS(qopt->latency);
> @@ -1044,17 +1041,11 @@ static int netem_change(struct Qdisc *sch, struct nlattr *opt,
>
> unlock:
> sch_tree_unlock(sch);
> - return ret;
>
> -get_table_failure:
> - /* recover clg and loss_model, in case of
> - * q->clg and q->loss_model were modified
> - * in get_loss_clg()
> - */
> - q->clg = old_clg;
> - q->loss_model = old_loss_model;
> -
> - goto unlock;
> +table_free:
> + dist_free(delay_dist);
> + dist_free(slot_dist);
> + return ret;
> }
>
> static int netem_init(struct Qdisc *sch, struct nlattr *opt,
> --
> 2.41.0.178.g377b9f9a00-goog
>
^ permalink raw reply [flat|nested] 4+ messages in thread
* Re: [PATCH net] sch_netem: fix issues in netem_change() vs get_dist_table()
2023-06-22 18:15 [PATCH net] sch_netem: fix issues in netem_change() vs get_dist_table() Eric Dumazet
2023-06-22 18:56 ` Jamal Hadi Salim
@ 2023-06-23 7:54 ` Simon Horman
2023-06-24 22:20 ` patchwork-bot+netdevbpf
2 siblings, 0 replies; 4+ messages in thread
From: Simon Horman @ 2023-06-23 7:54 UTC (permalink / raw)
To: Eric Dumazet
Cc: David S . Miller, Jakub Kicinski, Paolo Abeni, netdev,
eric.dumazet, syzbot, Stephen Hemminger, Jamal Hadi Salim
On Thu, Jun 22, 2023 at 06:15:03PM +0000, Eric Dumazet wrote:
> In blamed commit, I missed that get_dist_table() was allocating
> memory using GFP_KERNEL, and acquiring qdisc lock to perform
> the swap of newly allocated table with current one.
>
> In this patch, get_dist_table() is allocating memory and
> copy user data before we acquire the qdisc lock.
>
> Then we perform swap operations while being protected by the lock.
>
> Note that after this patch netem_change() no longer can do partial changes.
> If an error is returned, qdisc conf is left unchanged.
>
> Fixes: 2174a08db80d ("sch_netem: acquire qdisc lock in netem_change()")
> Reported-by: syzbot <syzkaller@googlegroups.com>
> Signed-off-by: Eric Dumazet <edumazet@google.com>
> Cc: Stephen Hemminger <stephen@networkplumber.org>
> Cc: Jamal Hadi Salim <jhs@mojatatu.com>
> Cc: Simon Horman <simon.horman@corigine.com>
> ---
> net/sched/sch_netem.c | 59 ++++++++++++++++++-------------------------
> 1 file changed, 25 insertions(+), 34 deletions(-)
Reviewed-by: Simon Horman <simon.horman@corigine.com>
^ permalink raw reply [flat|nested] 4+ messages in thread
* Re: [PATCH net] sch_netem: fix issues in netem_change() vs get_dist_table()
2023-06-22 18:15 [PATCH net] sch_netem: fix issues in netem_change() vs get_dist_table() Eric Dumazet
2023-06-22 18:56 ` Jamal Hadi Salim
2023-06-23 7:54 ` Simon Horman
@ 2023-06-24 22:20 ` patchwork-bot+netdevbpf
2 siblings, 0 replies; 4+ messages in thread
From: patchwork-bot+netdevbpf @ 2023-06-24 22:20 UTC (permalink / raw)
To: Eric Dumazet
Cc: davem, kuba, pabeni, netdev, eric.dumazet, syzkaller, stephen,
jhs, simon.horman
Hello:
This patch was applied to netdev/net.git (main)
by Jakub Kicinski <kuba@kernel.org>:
On Thu, 22 Jun 2023 18:15:03 +0000 you wrote:
> In blamed commit, I missed that get_dist_table() was allocating
> memory using GFP_KERNEL, and acquiring qdisc lock to perform
> the swap of newly allocated table with current one.
>
> In this patch, get_dist_table() is allocating memory and
> copy user data before we acquire the qdisc lock.
>
> [...]
Here is the summary with links:
- [net] sch_netem: fix issues in netem_change() vs get_dist_table()
https://git.kernel.org/netdev/net/c/11b73313c124
You are awesome, thank you!
--
Deet-doot-dot, I am a bot.
https://korg.docs.kernel.org/patchwork/pwbot.html
^ permalink raw reply [flat|nested] 4+ messages in thread
end of thread, other threads:[~2023-06-24 22:20 UTC | newest]
Thread overview: 4+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2023-06-22 18:15 [PATCH net] sch_netem: fix issues in netem_change() vs get_dist_table() Eric Dumazet
2023-06-22 18:56 ` Jamal Hadi Salim
2023-06-23 7:54 ` Simon Horman
2023-06-24 22:20 ` patchwork-bot+netdevbpf
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).