Commit e33d2b74 authored by Cong Wang's avatar Cong Wang Committed by David S. Miller

idr: fix overflow case for idr_for_each_entry_ul()

idr_for_each_entry_ul() is buggy as it can't handle overflow
case correctly. When we have an ID == UINT_MAX, it becomes an
infinite loop. This happens when running on 32-bit CPU where
unsigned long has the same size with unsigned int.

There is no better way to fix this than casting it to a larger
integer, but we can't just 64 bit integer on 32 bit CPU. Instead
we could just use an additional integer to help us to detect this
overflow case, that is, adding a new parameter to this macro.
Fortunately tc action is its only user right now.

Fixes: 65a206c0 ("net/sched: Change act_api and act_xxx modules to use IDR")
Reported-by: default avatarLi Shuang <shuali@redhat.com>
Tested-by: default avatarDavide Caratti <dcaratti@redhat.com>
Cc: Matthew Wilcox <willy@infradead.org>
Cc: Chris Mi <chrism@mellanox.com>
Signed-off-by: default avatarCong Wang <xiyou.wangcong@gmail.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent eb1f5c02
...@@ -191,14 +191,17 @@ static inline void idr_preload_end(void) ...@@ -191,14 +191,17 @@ static inline void idr_preload_end(void)
* idr_for_each_entry_ul() - Iterate over an IDR's elements of a given type. * idr_for_each_entry_ul() - Iterate over an IDR's elements of a given type.
* @idr: IDR handle. * @idr: IDR handle.
* @entry: The type * to use as cursor. * @entry: The type * to use as cursor.
* @tmp: A temporary placeholder for ID.
* @id: Entry ID. * @id: Entry ID.
* *
* @entry and @id do not need to be initialized before the loop, and * @entry and @id do not need to be initialized before the loop, and
* after normal termination @entry is left with the value NULL. This * after normal termination @entry is left with the value NULL. This
* is convenient for a "not found" value. * is convenient for a "not found" value.
*/ */
#define idr_for_each_entry_ul(idr, entry, id) \ #define idr_for_each_entry_ul(idr, entry, tmp, id) \
for (id = 0; ((entry) = idr_get_next_ul(idr, &(id))) != NULL; ++id) for (tmp = 0, id = 0; \
tmp <= id && ((entry) = idr_get_next_ul(idr, &(id))) != NULL; \
tmp = id, ++id)
/** /**
* idr_for_each_entry_continue() - Continue iteration over an IDR's elements of a given type * idr_for_each_entry_continue() - Continue iteration over an IDR's elements of a given type
......
...@@ -221,12 +221,13 @@ static int tcf_dump_walker(struct tcf_idrinfo *idrinfo, struct sk_buff *skb, ...@@ -221,12 +221,13 @@ static int tcf_dump_walker(struct tcf_idrinfo *idrinfo, struct sk_buff *skb,
struct idr *idr = &idrinfo->action_idr; struct idr *idr = &idrinfo->action_idr;
struct tc_action *p; struct tc_action *p;
unsigned long id = 1; unsigned long id = 1;
unsigned long tmp;
mutex_lock(&idrinfo->lock); mutex_lock(&idrinfo->lock);
s_i = cb->args[0]; s_i = cb->args[0];
idr_for_each_entry_ul(idr, p, id) { idr_for_each_entry_ul(idr, p, tmp, id) {
index++; index++;
if (index < s_i) if (index < s_i)
continue; continue;
...@@ -292,6 +293,7 @@ static int tcf_del_walker(struct tcf_idrinfo *idrinfo, struct sk_buff *skb, ...@@ -292,6 +293,7 @@ static int tcf_del_walker(struct tcf_idrinfo *idrinfo, struct sk_buff *skb,
struct idr *idr = &idrinfo->action_idr; struct idr *idr = &idrinfo->action_idr;
struct tc_action *p; struct tc_action *p;
unsigned long id = 1; unsigned long id = 1;
unsigned long tmp;
nest = nla_nest_start_noflag(skb, 0); nest = nla_nest_start_noflag(skb, 0);
if (nest == NULL) if (nest == NULL)
...@@ -300,7 +302,7 @@ static int tcf_del_walker(struct tcf_idrinfo *idrinfo, struct sk_buff *skb, ...@@ -300,7 +302,7 @@ static int tcf_del_walker(struct tcf_idrinfo *idrinfo, struct sk_buff *skb,
goto nla_put_failure; goto nla_put_failure;
mutex_lock(&idrinfo->lock); mutex_lock(&idrinfo->lock);
idr_for_each_entry_ul(idr, p, id) { idr_for_each_entry_ul(idr, p, tmp, id) {
ret = tcf_idr_release_unsafe(p); ret = tcf_idr_release_unsafe(p);
if (ret == ACT_P_DELETED) { if (ret == ACT_P_DELETED) {
module_put(ops->owner); module_put(ops->owner);
...@@ -533,8 +535,9 @@ void tcf_idrinfo_destroy(const struct tc_action_ops *ops, ...@@ -533,8 +535,9 @@ void tcf_idrinfo_destroy(const struct tc_action_ops *ops,
struct tc_action *p; struct tc_action *p;
int ret; int ret;
unsigned long id = 1; unsigned long id = 1;
unsigned long tmp;
idr_for_each_entry_ul(idr, p, id) { idr_for_each_entry_ul(idr, p, tmp, id) {
ret = __tcf_idr_release(p, false, true); ret = __tcf_idr_release(p, false, true);
if (ret == ACT_P_DELETED) if (ret == ACT_P_DELETED)
module_put(ops->owner); module_put(ops->owner);
......
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