diff options
| author | Eric Dumazet <eric.dumazet@gmail.com> | 2010-05-22 16:37:44 -0400 |
|---|---|---|
| committer | David S. Miller <davem@davemloft.net> | 2010-05-24 02:11:07 -0400 |
| commit | 53b0f08042f04813cd1a7473dacd3edfacb28eb3 (patch) | |
| tree | 025244cdb7c4f0872f372564de2b82130fc7e867 | |
| parent | a6c0f8217c17d46da22fa56923f3cbd03615cb7c (diff) | |
net_sched: Fix qdisc_notify()
Ben Pfaff reported a kernel oops and provided a test program to
reproduce it.
https://kerneltrap.org/mailarchive/linux-netdev/2010/5/21/6277805
tc_fill_qdisc() should not be called for builtin qdisc, or it
dereference a NULL pointer to get device ifindex.
Fix is to always use tc_qdisc_dump_ignore() before calling
tc_fill_qdisc().
Reported-by: Ben Pfaff <blp@nicira.com>
Signed-off-by: Eric Dumazet <eric.dumazet@gmail.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
| -rw-r--r-- | net/sched/sch_api.c | 14 |
1 files changed, 7 insertions, 7 deletions
diff --git a/net/sched/sch_api.c b/net/sched/sch_api.c index fe35c1f338c2..b9e8c3b7d406 100644 --- a/net/sched/sch_api.c +++ b/net/sched/sch_api.c | |||
| @@ -1195,6 +1195,11 @@ nla_put_failure: | |||
| 1195 | return -1; | 1195 | return -1; |
| 1196 | } | 1196 | } |
| 1197 | 1197 | ||
| 1198 | static bool tc_qdisc_dump_ignore(struct Qdisc *q) | ||
| 1199 | { | ||
| 1200 | return (q->flags & TCQ_F_BUILTIN) ? true : false; | ||
| 1201 | } | ||
| 1202 | |||
| 1198 | static int qdisc_notify(struct net *net, struct sk_buff *oskb, | 1203 | static int qdisc_notify(struct net *net, struct sk_buff *oskb, |
| 1199 | struct nlmsghdr *n, u32 clid, | 1204 | struct nlmsghdr *n, u32 clid, |
| 1200 | struct Qdisc *old, struct Qdisc *new) | 1205 | struct Qdisc *old, struct Qdisc *new) |
| @@ -1206,11 +1211,11 @@ static int qdisc_notify(struct net *net, struct sk_buff *oskb, | |||
| 1206 | if (!skb) | 1211 | if (!skb) |
| 1207 | return -ENOBUFS; | 1212 | return -ENOBUFS; |
| 1208 | 1213 | ||
| 1209 | if (old && old->handle) { | 1214 | if (old && !tc_qdisc_dump_ignore(old)) { |
| 1210 | if (tc_fill_qdisc(skb, old, clid, pid, n->nlmsg_seq, 0, RTM_DELQDISC) < 0) | 1215 | if (tc_fill_qdisc(skb, old, clid, pid, n->nlmsg_seq, 0, RTM_DELQDISC) < 0) |
| 1211 | goto err_out; | 1216 | goto err_out; |
| 1212 | } | 1217 | } |
| 1213 | if (new) { | 1218 | if (new && !tc_qdisc_dump_ignore(new)) { |
| 1214 | if (tc_fill_qdisc(skb, new, clid, pid, n->nlmsg_seq, old ? NLM_F_REPLACE : 0, RTM_NEWQDISC) < 0) | 1219 | if (tc_fill_qdisc(skb, new, clid, pid, n->nlmsg_seq, old ? NLM_F_REPLACE : 0, RTM_NEWQDISC) < 0) |
| 1215 | goto err_out; | 1220 | goto err_out; |
| 1216 | } | 1221 | } |
| @@ -1223,11 +1228,6 @@ err_out: | |||
| 1223 | return -EINVAL; | 1228 | return -EINVAL; |
| 1224 | } | 1229 | } |
| 1225 | 1230 | ||
| 1226 | static bool tc_qdisc_dump_ignore(struct Qdisc *q) | ||
| 1227 | { | ||
| 1228 | return (q->flags & TCQ_F_BUILTIN) ? true : false; | ||
| 1229 | } | ||
| 1230 | |||
| 1231 | static int tc_dump_qdisc_root(struct Qdisc *root, struct sk_buff *skb, | 1231 | static int tc_dump_qdisc_root(struct Qdisc *root, struct sk_buff *skb, |
| 1232 | struct netlink_callback *cb, | 1232 | struct netlink_callback *cb, |
| 1233 | int *q_idx_p, int s_q_idx) | 1233 | int *q_idx_p, int s_q_idx) |
