Commit e688a7f8 authored by Pablo Neira Ayuso's avatar Pablo Neira Ayuso

netfilter: nf_tables: safe RCU iteration on list when dumping

The dump operation through netlink is not protected by the nfnl_lock.
Thus, a reader process can be dumping any of the existing object
lists while another process can be updating the list content.

This patch resolves this situation by protecting all the object
lists with RCU in the netlink dump path which is the reader side.
The updater path is already protected via nfnl_lock, so use list
manipulation RCU-safe operations.
Signed-off-by: default avatarPablo Neira Ayuso <pablo@netfilter.org>
parent 63283dd2
...@@ -35,7 +35,7 @@ int nft_register_afinfo(struct net *net, struct nft_af_info *afi) ...@@ -35,7 +35,7 @@ int nft_register_afinfo(struct net *net, struct nft_af_info *afi)
{ {
INIT_LIST_HEAD(&afi->tables); INIT_LIST_HEAD(&afi->tables);
nfnl_lock(NFNL_SUBSYS_NFTABLES); nfnl_lock(NFNL_SUBSYS_NFTABLES);
list_add_tail(&afi->list, &net->nft.af_info); list_add_tail_rcu(&afi->list, &net->nft.af_info);
nfnl_unlock(NFNL_SUBSYS_NFTABLES); nfnl_unlock(NFNL_SUBSYS_NFTABLES);
return 0; return 0;
} }
...@@ -51,7 +51,7 @@ EXPORT_SYMBOL_GPL(nft_register_afinfo); ...@@ -51,7 +51,7 @@ EXPORT_SYMBOL_GPL(nft_register_afinfo);
void nft_unregister_afinfo(struct nft_af_info *afi) void nft_unregister_afinfo(struct nft_af_info *afi)
{ {
nfnl_lock(NFNL_SUBSYS_NFTABLES); nfnl_lock(NFNL_SUBSYS_NFTABLES);
list_del(&afi->list); list_del_rcu(&afi->list);
nfnl_unlock(NFNL_SUBSYS_NFTABLES); nfnl_unlock(NFNL_SUBSYS_NFTABLES);
} }
EXPORT_SYMBOL_GPL(nft_unregister_afinfo); EXPORT_SYMBOL_GPL(nft_unregister_afinfo);
...@@ -277,11 +277,12 @@ static int nf_tables_dump_tables(struct sk_buff *skb, ...@@ -277,11 +277,12 @@ static int nf_tables_dump_tables(struct sk_buff *skb,
struct net *net = sock_net(skb->sk); struct net *net = sock_net(skb->sk);
int family = nfmsg->nfgen_family; int family = nfmsg->nfgen_family;
list_for_each_entry(afi, &net->nft.af_info, list) { rcu_read_lock();
list_for_each_entry_rcu(afi, &net->nft.af_info, list) {
if (family != NFPROTO_UNSPEC && family != afi->family) if (family != NFPROTO_UNSPEC && family != afi->family)
continue; continue;
list_for_each_entry(table, &afi->tables, list) { list_for_each_entry_rcu(table, &afi->tables, list) {
if (idx < s_idx) if (idx < s_idx)
goto cont; goto cont;
if (idx > s_idx) if (idx > s_idx)
...@@ -299,6 +300,7 @@ static int nf_tables_dump_tables(struct sk_buff *skb, ...@@ -299,6 +300,7 @@ static int nf_tables_dump_tables(struct sk_buff *skb,
} }
} }
done: done:
rcu_read_unlock();
cb->args[0] = idx; cb->args[0] = idx;
return skb->len; return skb->len;
} }
...@@ -517,7 +519,7 @@ static int nf_tables_newtable(struct sock *nlsk, struct sk_buff *skb, ...@@ -517,7 +519,7 @@ static int nf_tables_newtable(struct sock *nlsk, struct sk_buff *skb,
module_put(afi->owner); module_put(afi->owner);
return err; return err;
} }
list_add_tail(&table->list, &afi->tables); list_add_tail_rcu(&table->list, &afi->tables);
return 0; return 0;
} }
...@@ -549,7 +551,7 @@ static int nf_tables_deltable(struct sock *nlsk, struct sk_buff *skb, ...@@ -549,7 +551,7 @@ static int nf_tables_deltable(struct sock *nlsk, struct sk_buff *skb,
if (err < 0) if (err < 0)
return err; return err;
list_del(&table->list); list_del_rcu(&table->list);
return 0; return 0;
} }
...@@ -764,12 +766,13 @@ static int nf_tables_dump_chains(struct sk_buff *skb, ...@@ -764,12 +766,13 @@ static int nf_tables_dump_chains(struct sk_buff *skb,
struct net *net = sock_net(skb->sk); struct net *net = sock_net(skb->sk);
int family = nfmsg->nfgen_family; int family = nfmsg->nfgen_family;
list_for_each_entry(afi, &net->nft.af_info, list) { rcu_read_lock();
list_for_each_entry_rcu(afi, &net->nft.af_info, list) {
if (family != NFPROTO_UNSPEC && family != afi->family) if (family != NFPROTO_UNSPEC && family != afi->family)
continue; continue;
list_for_each_entry(table, &afi->tables, list) { list_for_each_entry_rcu(table, &afi->tables, list) {
list_for_each_entry(chain, &table->chains, list) { list_for_each_entry_rcu(chain, &table->chains, list) {
if (idx < s_idx) if (idx < s_idx)
goto cont; goto cont;
if (idx > s_idx) if (idx > s_idx)
...@@ -787,11 +790,11 @@ static int nf_tables_dump_chains(struct sk_buff *skb, ...@@ -787,11 +790,11 @@ static int nf_tables_dump_chains(struct sk_buff *skb,
} }
} }
done: done:
rcu_read_unlock();
cb->args[0] = idx; cb->args[0] = idx;
return skb->len; return skb->len;
} }
static int nf_tables_getchain(struct sock *nlsk, struct sk_buff *skb, static int nf_tables_getchain(struct sock *nlsk, struct sk_buff *skb,
const struct nlmsghdr *nlh, const struct nlmsghdr *nlh,
const struct nlattr * const nla[]) const struct nlattr * const nla[])
...@@ -1133,7 +1136,7 @@ static int nf_tables_newchain(struct sock *nlsk, struct sk_buff *skb, ...@@ -1133,7 +1136,7 @@ static int nf_tables_newchain(struct sock *nlsk, struct sk_buff *skb,
goto err2; goto err2;
table->use++; table->use++;
list_add_tail(&chain->list, &table->chains); list_add_tail_rcu(&chain->list, &table->chains);
return 0; return 0;
err2: err2:
if (!(table->flags & NFT_TABLE_F_DORMANT) && if (!(table->flags & NFT_TABLE_F_DORMANT) &&
...@@ -1183,7 +1186,7 @@ static int nf_tables_delchain(struct sock *nlsk, struct sk_buff *skb, ...@@ -1183,7 +1186,7 @@ static int nf_tables_delchain(struct sock *nlsk, struct sk_buff *skb,
return err; return err;
table->use--; table->use--;
list_del(&chain->list); list_del_rcu(&chain->list);
return 0; return 0;
} }
...@@ -1202,9 +1205,9 @@ int nft_register_expr(struct nft_expr_type *type) ...@@ -1202,9 +1205,9 @@ int nft_register_expr(struct nft_expr_type *type)
{ {
nfnl_lock(NFNL_SUBSYS_NFTABLES); nfnl_lock(NFNL_SUBSYS_NFTABLES);
if (type->family == NFPROTO_UNSPEC) if (type->family == NFPROTO_UNSPEC)
list_add_tail(&type->list, &nf_tables_expressions); list_add_tail_rcu(&type->list, &nf_tables_expressions);
else else
list_add(&type->list, &nf_tables_expressions); list_add_rcu(&type->list, &nf_tables_expressions);
nfnl_unlock(NFNL_SUBSYS_NFTABLES); nfnl_unlock(NFNL_SUBSYS_NFTABLES);
return 0; return 0;
} }
...@@ -1219,7 +1222,7 @@ EXPORT_SYMBOL_GPL(nft_register_expr); ...@@ -1219,7 +1222,7 @@ EXPORT_SYMBOL_GPL(nft_register_expr);
void nft_unregister_expr(struct nft_expr_type *type) void nft_unregister_expr(struct nft_expr_type *type)
{ {
nfnl_lock(NFNL_SUBSYS_NFTABLES); nfnl_lock(NFNL_SUBSYS_NFTABLES);
list_del(&type->list); list_del_rcu(&type->list);
nfnl_unlock(NFNL_SUBSYS_NFTABLES); nfnl_unlock(NFNL_SUBSYS_NFTABLES);
} }
EXPORT_SYMBOL_GPL(nft_unregister_expr); EXPORT_SYMBOL_GPL(nft_unregister_expr);
...@@ -1555,13 +1558,14 @@ static int nf_tables_dump_rules(struct sk_buff *skb, ...@@ -1555,13 +1558,14 @@ static int nf_tables_dump_rules(struct sk_buff *skb,
u8 genctr = ACCESS_ONCE(net->nft.genctr); u8 genctr = ACCESS_ONCE(net->nft.genctr);
u8 gencursor = ACCESS_ONCE(net->nft.gencursor); u8 gencursor = ACCESS_ONCE(net->nft.gencursor);
list_for_each_entry(afi, &net->nft.af_info, list) { rcu_read_lock();
list_for_each_entry_rcu(afi, &net->nft.af_info, list) {
if (family != NFPROTO_UNSPEC && family != afi->family) if (family != NFPROTO_UNSPEC && family != afi->family)
continue; continue;
list_for_each_entry(table, &afi->tables, list) { list_for_each_entry_rcu(table, &afi->tables, list) {
list_for_each_entry(chain, &table->chains, list) { list_for_each_entry_rcu(chain, &table->chains, list) {
list_for_each_entry(rule, &chain->rules, list) { list_for_each_entry_rcu(rule, &chain->rules, list) {
if (!nft_rule_is_active(net, rule)) if (!nft_rule_is_active(net, rule))
goto cont; goto cont;
if (idx < s_idx) if (idx < s_idx)
...@@ -1582,6 +1586,8 @@ static int nf_tables_dump_rules(struct sk_buff *skb, ...@@ -1582,6 +1586,8 @@ static int nf_tables_dump_rules(struct sk_buff *skb,
} }
} }
done: done:
rcu_read_unlock();
/* Invalidate this dump, a transition to the new generation happened */ /* Invalidate this dump, a transition to the new generation happened */
if (gencursor != net->nft.gencursor || genctr != net->nft.genctr) if (gencursor != net->nft.gencursor || genctr != net->nft.genctr)
return -EBUSY; return -EBUSY;
...@@ -1935,7 +1941,7 @@ static LIST_HEAD(nf_tables_set_ops); ...@@ -1935,7 +1941,7 @@ static LIST_HEAD(nf_tables_set_ops);
int nft_register_set(struct nft_set_ops *ops) int nft_register_set(struct nft_set_ops *ops)
{ {
nfnl_lock(NFNL_SUBSYS_NFTABLES); nfnl_lock(NFNL_SUBSYS_NFTABLES);
list_add_tail(&ops->list, &nf_tables_set_ops); list_add_tail_rcu(&ops->list, &nf_tables_set_ops);
nfnl_unlock(NFNL_SUBSYS_NFTABLES); nfnl_unlock(NFNL_SUBSYS_NFTABLES);
return 0; return 0;
} }
...@@ -1944,7 +1950,7 @@ EXPORT_SYMBOL_GPL(nft_register_set); ...@@ -1944,7 +1950,7 @@ EXPORT_SYMBOL_GPL(nft_register_set);
void nft_unregister_set(struct nft_set_ops *ops) void nft_unregister_set(struct nft_set_ops *ops)
{ {
nfnl_lock(NFNL_SUBSYS_NFTABLES); nfnl_lock(NFNL_SUBSYS_NFTABLES);
list_del(&ops->list); list_del_rcu(&ops->list);
nfnl_unlock(NFNL_SUBSYS_NFTABLES); nfnl_unlock(NFNL_SUBSYS_NFTABLES);
} }
EXPORT_SYMBOL_GPL(nft_unregister_set); EXPORT_SYMBOL_GPL(nft_unregister_set);
...@@ -2237,7 +2243,8 @@ static int nf_tables_dump_sets_table(struct nft_ctx *ctx, struct sk_buff *skb, ...@@ -2237,7 +2243,8 @@ static int nf_tables_dump_sets_table(struct nft_ctx *ctx, struct sk_buff *skb,
if (cb->args[1]) if (cb->args[1])
return skb->len; return skb->len;
list_for_each_entry(set, &ctx->table->sets, list) { rcu_read_lock();
list_for_each_entry_rcu(set, &ctx->table->sets, list) {
if (idx < s_idx) if (idx < s_idx)
goto cont; goto cont;
if (nf_tables_fill_set(skb, ctx, set, NFT_MSG_NEWSET, if (nf_tables_fill_set(skb, ctx, set, NFT_MSG_NEWSET,
...@@ -2250,6 +2257,7 @@ static int nf_tables_dump_sets_table(struct nft_ctx *ctx, struct sk_buff *skb, ...@@ -2250,6 +2257,7 @@ static int nf_tables_dump_sets_table(struct nft_ctx *ctx, struct sk_buff *skb,
} }
cb->args[1] = 1; cb->args[1] = 1;
done: done:
rcu_read_unlock();
return skb->len; return skb->len;
} }
...@@ -2263,7 +2271,8 @@ static int nf_tables_dump_sets_family(struct nft_ctx *ctx, struct sk_buff *skb, ...@@ -2263,7 +2271,8 @@ static int nf_tables_dump_sets_family(struct nft_ctx *ctx, struct sk_buff *skb,
if (cb->args[1]) if (cb->args[1])
return skb->len; return skb->len;
list_for_each_entry(table, &ctx->afi->tables, list) { rcu_read_lock();
list_for_each_entry_rcu(table, &ctx->afi->tables, list) {
if (cur_table) { if (cur_table) {
if (cur_table != table) if (cur_table != table)
continue; continue;
...@@ -2272,7 +2281,7 @@ static int nf_tables_dump_sets_family(struct nft_ctx *ctx, struct sk_buff *skb, ...@@ -2272,7 +2281,7 @@ static int nf_tables_dump_sets_family(struct nft_ctx *ctx, struct sk_buff *skb,
} }
ctx->table = table; ctx->table = table;
idx = 0; idx = 0;
list_for_each_entry(set, &ctx->table->sets, list) { list_for_each_entry_rcu(set, &ctx->table->sets, list) {
if (idx < s_idx) if (idx < s_idx)
goto cont; goto cont;
if (nf_tables_fill_set(skb, ctx, set, NFT_MSG_NEWSET, if (nf_tables_fill_set(skb, ctx, set, NFT_MSG_NEWSET,
...@@ -2287,6 +2296,7 @@ static int nf_tables_dump_sets_family(struct nft_ctx *ctx, struct sk_buff *skb, ...@@ -2287,6 +2296,7 @@ static int nf_tables_dump_sets_family(struct nft_ctx *ctx, struct sk_buff *skb,
} }
cb->args[1] = 1; cb->args[1] = 1;
done: done:
rcu_read_unlock();
return skb->len; return skb->len;
} }
...@@ -2303,7 +2313,8 @@ static int nf_tables_dump_sets_all(struct nft_ctx *ctx, struct sk_buff *skb, ...@@ -2303,7 +2313,8 @@ static int nf_tables_dump_sets_all(struct nft_ctx *ctx, struct sk_buff *skb,
if (cb->args[1]) if (cb->args[1])
return skb->len; return skb->len;
list_for_each_entry(afi, &net->nft.af_info, list) { rcu_read_lock();
list_for_each_entry_rcu(afi, &net->nft.af_info, list) {
if (cur_family) { if (cur_family) {
if (afi->family != cur_family) if (afi->family != cur_family)
continue; continue;
...@@ -2311,7 +2322,7 @@ static int nf_tables_dump_sets_all(struct nft_ctx *ctx, struct sk_buff *skb, ...@@ -2311,7 +2322,7 @@ static int nf_tables_dump_sets_all(struct nft_ctx *ctx, struct sk_buff *skb,
cur_family = 0; cur_family = 0;
} }
list_for_each_entry(table, &afi->tables, list) { list_for_each_entry_rcu(table, &afi->tables, list) {
if (cur_table) { if (cur_table) {
if (cur_table != table) if (cur_table != table)
continue; continue;
...@@ -2322,7 +2333,7 @@ static int nf_tables_dump_sets_all(struct nft_ctx *ctx, struct sk_buff *skb, ...@@ -2322,7 +2333,7 @@ static int nf_tables_dump_sets_all(struct nft_ctx *ctx, struct sk_buff *skb,
ctx->table = table; ctx->table = table;
ctx->afi = afi; ctx->afi = afi;
idx = 0; idx = 0;
list_for_each_entry(set, &ctx->table->sets, list) { list_for_each_entry_rcu(set, &ctx->table->sets, list) {
if (idx < s_idx) if (idx < s_idx)
goto cont; goto cont;
if (nf_tables_fill_set(skb, ctx, set, if (nf_tables_fill_set(skb, ctx, set,
...@@ -2342,6 +2353,7 @@ static int nf_tables_dump_sets_all(struct nft_ctx *ctx, struct sk_buff *skb, ...@@ -2342,6 +2353,7 @@ static int nf_tables_dump_sets_all(struct nft_ctx *ctx, struct sk_buff *skb,
} }
cb->args[1] = 1; cb->args[1] = 1;
done: done:
rcu_read_unlock();
return skb->len; return skb->len;
} }
...@@ -2600,7 +2612,7 @@ static int nf_tables_newset(struct sock *nlsk, struct sk_buff *skb, ...@@ -2600,7 +2612,7 @@ static int nf_tables_newset(struct sock *nlsk, struct sk_buff *skb,
if (err < 0) if (err < 0)
goto err2; goto err2;
list_add_tail(&set->list, &table->sets); list_add_tail_rcu(&set->list, &table->sets);
table->use++; table->use++;
return 0; return 0;
...@@ -2620,7 +2632,7 @@ static void nft_set_destroy(struct nft_set *set) ...@@ -2620,7 +2632,7 @@ static void nft_set_destroy(struct nft_set *set)
static void nf_tables_set_destroy(const struct nft_ctx *ctx, struct nft_set *set) static void nf_tables_set_destroy(const struct nft_ctx *ctx, struct nft_set *set)
{ {
list_del(&set->list); list_del_rcu(&set->list);
nf_tables_set_notify(ctx, set, NFT_MSG_DELSET, GFP_ATOMIC); nf_tables_set_notify(ctx, set, NFT_MSG_DELSET, GFP_ATOMIC);
nft_set_destroy(set); nft_set_destroy(set);
} }
...@@ -2655,7 +2667,7 @@ static int nf_tables_delset(struct sock *nlsk, struct sk_buff *skb, ...@@ -2655,7 +2667,7 @@ static int nf_tables_delset(struct sock *nlsk, struct sk_buff *skb,
if (err < 0) if (err < 0)
return err; return err;
list_del(&set->list); list_del_rcu(&set->list);
ctx.table->use--; ctx.table->use--;
return 0; return 0;
} }
...@@ -2707,14 +2719,14 @@ int nf_tables_bind_set(const struct nft_ctx *ctx, struct nft_set *set, ...@@ -2707,14 +2719,14 @@ int nf_tables_bind_set(const struct nft_ctx *ctx, struct nft_set *set,
} }
bind: bind:
binding->chain = ctx->chain; binding->chain = ctx->chain;
list_add_tail(&binding->list, &set->bindings); list_add_tail_rcu(&binding->list, &set->bindings);
return 0; return 0;
} }
void nf_tables_unbind_set(const struct nft_ctx *ctx, struct nft_set *set, void nf_tables_unbind_set(const struct nft_ctx *ctx, struct nft_set *set,
struct nft_set_binding *binding) struct nft_set_binding *binding)
{ {
list_del(&binding->list); list_del_rcu(&binding->list);
if (list_empty(&set->bindings) && set->flags & NFT_SET_ANONYMOUS && if (list_empty(&set->bindings) && set->flags & NFT_SET_ANONYMOUS &&
!(set->flags & NFT_SET_INACTIVE)) !(set->flags & NFT_SET_INACTIVE))
...@@ -3494,12 +3506,12 @@ static int nf_tables_abort(struct sk_buff *skb) ...@@ -3494,12 +3506,12 @@ static int nf_tables_abort(struct sk_buff *skb)
} }
nft_trans_destroy(trans); nft_trans_destroy(trans);
} else { } else {
list_del(&trans->ctx.table->list); list_del_rcu(&trans->ctx.table->list);
} }
break; break;
case NFT_MSG_DELTABLE: case NFT_MSG_DELTABLE:
list_add_tail(&trans->ctx.table->list, list_add_tail_rcu(&trans->ctx.table->list,
&trans->ctx.afi->tables); &trans->ctx.afi->tables);
nft_trans_destroy(trans); nft_trans_destroy(trans);
break; break;
case NFT_MSG_NEWCHAIN: case NFT_MSG_NEWCHAIN:
...@@ -3510,7 +3522,7 @@ static int nf_tables_abort(struct sk_buff *skb) ...@@ -3510,7 +3522,7 @@ static int nf_tables_abort(struct sk_buff *skb)
nft_trans_destroy(trans); nft_trans_destroy(trans);
} else { } else {
trans->ctx.table->use--; trans->ctx.table->use--;
list_del(&trans->ctx.chain->list); list_del_rcu(&trans->ctx.chain->list);
if (!(trans->ctx.table->flags & NFT_TABLE_F_DORMANT) && if (!(trans->ctx.table->flags & NFT_TABLE_F_DORMANT) &&
trans->ctx.chain->flags & NFT_BASE_CHAIN) { trans->ctx.chain->flags & NFT_BASE_CHAIN) {
nf_unregister_hooks(nft_base_chain(trans->ctx.chain)->ops, nf_unregister_hooks(nft_base_chain(trans->ctx.chain)->ops,
...@@ -3520,8 +3532,8 @@ static int nf_tables_abort(struct sk_buff *skb) ...@@ -3520,8 +3532,8 @@ static int nf_tables_abort(struct sk_buff *skb)
break; break;
case NFT_MSG_DELCHAIN: case NFT_MSG_DELCHAIN:
trans->ctx.table->use++; trans->ctx.table->use++;
list_add_tail(&trans->ctx.chain->list, list_add_tail_rcu(&trans->ctx.chain->list,
&trans->ctx.table->chains); &trans->ctx.table->chains);
nft_trans_destroy(trans); nft_trans_destroy(trans);
break; break;
case NFT_MSG_NEWRULE: case NFT_MSG_NEWRULE:
...@@ -3535,12 +3547,12 @@ static int nf_tables_abort(struct sk_buff *skb) ...@@ -3535,12 +3547,12 @@ static int nf_tables_abort(struct sk_buff *skb)
break; break;
case NFT_MSG_NEWSET: case NFT_MSG_NEWSET:
trans->ctx.table->use--; trans->ctx.table->use--;
list_del(&nft_trans_set(trans)->list); list_del_rcu(&nft_trans_set(trans)->list);
break; break;
case NFT_MSG_DELSET: case NFT_MSG_DELSET:
trans->ctx.table->use++; trans->ctx.table->use++;
list_add_tail(&nft_trans_set(trans)->list, list_add_tail_rcu(&nft_trans_set(trans)->list,
&trans->ctx.table->sets); &trans->ctx.table->sets);
nft_trans_destroy(trans); nft_trans_destroy(trans);
break; break;
case NFT_MSG_NEWSETELEM: case NFT_MSG_NEWSETELEM:
......
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