mirror of https://github.com/torvalds/linux.git
ipv6: Protect fib6_link_table() with spinlock.
We will get rid of RTNL from RTM_NEWROUTE and SIOCADDRT. If the request specifies a new table ID, fib6_new_table() is called to create a new routing table. Two concurrent requests could specify the same table ID, so we need a lock to protect net->ipv6.fib_table_hash[h]. Let's add a spinlock to protect the hash bucket linkage. Signed-off-by: Kuniyuki Iwashima <kuniyu@amazon.com> Acked-by: Paolo Abeni <pabeni@redhat.com> Link: https://patch.msgid.link/20250418000443.43734-13-kuniyu@amazon.com Signed-off-by: Paolo Abeni <pabeni@redhat.com>
This commit is contained in:
parent
71c0efb6d1
commit
834d97843e
|
|
@ -72,6 +72,7 @@ struct netns_ipv6 {
|
||||||
struct rt6_statistics *rt6_stats;
|
struct rt6_statistics *rt6_stats;
|
||||||
struct timer_list ip6_fib_timer;
|
struct timer_list ip6_fib_timer;
|
||||||
struct hlist_head *fib_table_hash;
|
struct hlist_head *fib_table_hash;
|
||||||
|
spinlock_t fib_table_hash_lock;
|
||||||
struct fib6_table *fib6_main_tbl;
|
struct fib6_table *fib6_main_tbl;
|
||||||
struct list_head fib6_walkers;
|
struct list_head fib6_walkers;
|
||||||
rwlock_t fib6_walker_lock;
|
rwlock_t fib6_walker_lock;
|
||||||
|
|
|
||||||
|
|
@ -249,19 +249,33 @@ static struct fib6_table *fib6_alloc_table(struct net *net, u32 id)
|
||||||
|
|
||||||
struct fib6_table *fib6_new_table(struct net *net, u32 id)
|
struct fib6_table *fib6_new_table(struct net *net, u32 id)
|
||||||
{
|
{
|
||||||
struct fib6_table *tb;
|
struct fib6_table *tb, *new_tb;
|
||||||
|
|
||||||
if (id == 0)
|
if (id == 0)
|
||||||
id = RT6_TABLE_MAIN;
|
id = RT6_TABLE_MAIN;
|
||||||
|
|
||||||
tb = fib6_get_table(net, id);
|
tb = fib6_get_table(net, id);
|
||||||
if (tb)
|
if (tb)
|
||||||
return tb;
|
return tb;
|
||||||
|
|
||||||
tb = fib6_alloc_table(net, id);
|
new_tb = fib6_alloc_table(net, id);
|
||||||
if (tb)
|
if (!new_tb)
|
||||||
fib6_link_table(net, tb);
|
return NULL;
|
||||||
|
|
||||||
return tb;
|
spin_lock_bh(&net->ipv6.fib_table_hash_lock);
|
||||||
|
|
||||||
|
tb = fib6_get_table(net, id);
|
||||||
|
if (unlikely(tb)) {
|
||||||
|
spin_unlock_bh(&net->ipv6.fib_table_hash_lock);
|
||||||
|
kfree(new_tb);
|
||||||
|
return tb;
|
||||||
|
}
|
||||||
|
|
||||||
|
fib6_link_table(net, new_tb);
|
||||||
|
|
||||||
|
spin_unlock_bh(&net->ipv6.fib_table_hash_lock);
|
||||||
|
|
||||||
|
return new_tb;
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(fib6_new_table);
|
EXPORT_SYMBOL_GPL(fib6_new_table);
|
||||||
|
|
||||||
|
|
@ -2423,6 +2437,8 @@ static int __net_init fib6_net_init(struct net *net)
|
||||||
if (!net->ipv6.fib_table_hash)
|
if (!net->ipv6.fib_table_hash)
|
||||||
goto out_rt6_stats;
|
goto out_rt6_stats;
|
||||||
|
|
||||||
|
spin_lock_init(&net->ipv6.fib_table_hash_lock);
|
||||||
|
|
||||||
net->ipv6.fib6_main_tbl = kzalloc(sizeof(*net->ipv6.fib6_main_tbl),
|
net->ipv6.fib6_main_tbl = kzalloc(sizeof(*net->ipv6.fib6_main_tbl),
|
||||||
GFP_KERNEL);
|
GFP_KERNEL);
|
||||||
if (!net->ipv6.fib6_main_tbl)
|
if (!net->ipv6.fib6_main_tbl)
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue