1 // SPDX-License-Identifier: GPL-2.0-only
3 * Copyright (c) 2015 Patrick McHardy <kaber@trash.net>
6 #include <linux/kernel.h>
7 #include <linux/module.h>
8 #include <linux/init.h>
9 #include <linux/netlink.h>
10 #include <linux/netfilter.h>
11 #include <linux/netfilter/nf_tables.h>
12 #include <net/netfilter/nf_tables.h>
13 #include <net/netfilter/nf_tables_core.h>
17 struct nft_set_ext_tmpl tmpl
;
18 enum nft_dynset_ops op
:8;
19 enum nft_registers sreg_key
:8;
20 enum nft_registers sreg_data
:8;
23 struct nft_expr
*expr
;
24 struct nft_set_binding binding
;
27 static int nft_expr_clone(struct nft_expr
*dst
, struct nft_expr
*src
)
31 if (src
->ops
->clone
) {
33 err
= src
->ops
->clone(dst
, src
);
37 memcpy(dst
, src
, src
->ops
->size
);
40 __module_get(src
->ops
->type
->owner
);
44 static void *nft_dynset_new(struct nft_set
*set
, const struct nft_expr
*expr
,
45 struct nft_regs
*regs
)
47 const struct nft_dynset
*priv
= nft_expr_priv(expr
);
48 struct nft_set_ext
*ext
;
52 if (!atomic_add_unless(&set
->nelems
, 1, set
->size
))
55 timeout
= priv
->timeout
? : set
->timeout
;
56 elem
= nft_set_elem_init(set
, &priv
->tmpl
,
57 ®s
->data
[priv
->sreg_key
],
58 ®s
->data
[priv
->sreg_data
],
63 ext
= nft_set_elem_ext(set
, elem
);
64 if (priv
->expr
!= NULL
&&
65 nft_expr_clone(nft_set_ext_expr(ext
), priv
->expr
) < 0)
71 nft_set_elem_destroy(set
, elem
, false);
74 atomic_dec(&set
->nelems
);
78 void nft_dynset_eval(const struct nft_expr
*expr
,
79 struct nft_regs
*regs
, const struct nft_pktinfo
*pkt
)
81 const struct nft_dynset
*priv
= nft_expr_priv(expr
);
82 struct nft_set
*set
= priv
->set
;
83 const struct nft_set_ext
*ext
;
84 const struct nft_expr
*sexpr
;
87 if (set
->ops
->update(set
, ®s
->data
[priv
->sreg_key
], nft_dynset_new
,
90 if (nft_set_ext_exists(ext
, NFT_SET_EXT_EXPR
))
91 sexpr
= nft_set_ext_expr(ext
);
93 if (priv
->op
== NFT_DYNSET_OP_UPDATE
&&
94 nft_set_ext_exists(ext
, NFT_SET_EXT_EXPIRATION
)) {
95 timeout
= priv
->timeout
? : set
->timeout
;
96 *nft_set_ext_expiration(ext
) = get_jiffies_64() + timeout
;
100 sexpr
->ops
->eval(sexpr
, regs
, pkt
);
103 regs
->verdict
.code
= NFT_BREAK
;
108 regs
->verdict
.code
= NFT_BREAK
;
111 static const struct nla_policy nft_dynset_policy
[NFTA_DYNSET_MAX
+ 1] = {
112 [NFTA_DYNSET_SET_NAME
] = { .type
= NLA_STRING
,
113 .len
= NFT_SET_MAXNAMELEN
- 1 },
114 [NFTA_DYNSET_SET_ID
] = { .type
= NLA_U32
},
115 [NFTA_DYNSET_OP
] = { .type
= NLA_U32
},
116 [NFTA_DYNSET_SREG_KEY
] = { .type
= NLA_U32
},
117 [NFTA_DYNSET_SREG_DATA
] = { .type
= NLA_U32
},
118 [NFTA_DYNSET_TIMEOUT
] = { .type
= NLA_U64
},
119 [NFTA_DYNSET_EXPR
] = { .type
= NLA_NESTED
},
120 [NFTA_DYNSET_FLAGS
] = { .type
= NLA_U32
},
123 static int nft_dynset_init(const struct nft_ctx
*ctx
,
124 const struct nft_expr
*expr
,
125 const struct nlattr
* const tb
[])
127 struct nft_dynset
*priv
= nft_expr_priv(expr
);
128 u8 genmask
= nft_genmask_next(ctx
->net
);
133 lockdep_assert_held(&ctx
->net
->nft
.commit_mutex
);
135 if (tb
[NFTA_DYNSET_SET_NAME
] == NULL
||
136 tb
[NFTA_DYNSET_OP
] == NULL
||
137 tb
[NFTA_DYNSET_SREG_KEY
] == NULL
)
140 if (tb
[NFTA_DYNSET_FLAGS
]) {
141 u32 flags
= ntohl(nla_get_be32(tb
[NFTA_DYNSET_FLAGS
]));
143 if (flags
& ~NFT_DYNSET_F_INV
)
145 if (flags
& NFT_DYNSET_F_INV
)
149 set
= nft_set_lookup_global(ctx
->net
, ctx
->table
,
150 tb
[NFTA_DYNSET_SET_NAME
],
151 tb
[NFTA_DYNSET_SET_ID
], genmask
);
155 if (set
->ops
->update
== NULL
)
158 if (set
->flags
& NFT_SET_CONSTANT
)
161 priv
->op
= ntohl(nla_get_be32(tb
[NFTA_DYNSET_OP
]));
163 case NFT_DYNSET_OP_ADD
:
165 case NFT_DYNSET_OP_UPDATE
:
166 if (!(set
->flags
& NFT_SET_TIMEOUT
))
174 if (tb
[NFTA_DYNSET_TIMEOUT
] != NULL
) {
175 if (!(set
->flags
& NFT_SET_TIMEOUT
))
177 timeout
= msecs_to_jiffies(be64_to_cpu(nla_get_be64(
178 tb
[NFTA_DYNSET_TIMEOUT
])));
181 priv
->sreg_key
= nft_parse_register(tb
[NFTA_DYNSET_SREG_KEY
]);
182 err
= nft_validate_register_load(priv
->sreg_key
, set
->klen
);
186 if (tb
[NFTA_DYNSET_SREG_DATA
] != NULL
) {
187 if (!(set
->flags
& NFT_SET_MAP
))
189 if (set
->dtype
== NFT_DATA_VERDICT
)
192 priv
->sreg_data
= nft_parse_register(tb
[NFTA_DYNSET_SREG_DATA
]);
193 err
= nft_validate_register_load(priv
->sreg_data
, set
->dlen
);
196 } else if (set
->flags
& NFT_SET_MAP
)
199 if (tb
[NFTA_DYNSET_EXPR
] != NULL
) {
200 if (!(set
->flags
& NFT_SET_EVAL
))
203 priv
->expr
= nft_expr_init(ctx
, tb
[NFTA_DYNSET_EXPR
]);
204 if (IS_ERR(priv
->expr
))
205 return PTR_ERR(priv
->expr
);
208 if (!(priv
->expr
->ops
->type
->flags
& NFT_EXPR_STATEFUL
))
211 if (priv
->expr
->ops
->type
->flags
& NFT_EXPR_GC
) {
212 if (set
->flags
& NFT_SET_TIMEOUT
)
214 if (!set
->ops
->gc_init
)
216 set
->ops
->gc_init(set
);
220 nft_set_ext_prepare(&priv
->tmpl
);
221 nft_set_ext_add_length(&priv
->tmpl
, NFT_SET_EXT_KEY
, set
->klen
);
222 if (set
->flags
& NFT_SET_MAP
)
223 nft_set_ext_add_length(&priv
->tmpl
, NFT_SET_EXT_DATA
, set
->dlen
);
224 if (priv
->expr
!= NULL
)
225 nft_set_ext_add_length(&priv
->tmpl
, NFT_SET_EXT_EXPR
,
226 priv
->expr
->ops
->size
);
227 if (set
->flags
& NFT_SET_TIMEOUT
) {
228 if (timeout
|| set
->timeout
)
229 nft_set_ext_add(&priv
->tmpl
, NFT_SET_EXT_EXPIRATION
);
232 priv
->timeout
= timeout
;
234 err
= nf_tables_bind_set(ctx
, set
, &priv
->binding
);
245 if (priv
->expr
!= NULL
)
246 nft_expr_destroy(ctx
, priv
->expr
);
250 static void nft_dynset_deactivate(const struct nft_ctx
*ctx
,
251 const struct nft_expr
*expr
,
252 enum nft_trans_phase phase
)
254 struct nft_dynset
*priv
= nft_expr_priv(expr
);
256 nf_tables_deactivate_set(ctx
, priv
->set
, &priv
->binding
, phase
);
259 static void nft_dynset_activate(const struct nft_ctx
*ctx
,
260 const struct nft_expr
*expr
)
262 struct nft_dynset
*priv
= nft_expr_priv(expr
);
267 static void nft_dynset_destroy(const struct nft_ctx
*ctx
,
268 const struct nft_expr
*expr
)
270 struct nft_dynset
*priv
= nft_expr_priv(expr
);
272 if (priv
->expr
!= NULL
)
273 nft_expr_destroy(ctx
, priv
->expr
);
275 nf_tables_destroy_set(ctx
, priv
->set
);
278 static int nft_dynset_dump(struct sk_buff
*skb
, const struct nft_expr
*expr
)
280 const struct nft_dynset
*priv
= nft_expr_priv(expr
);
281 u32 flags
= priv
->invert
? NFT_DYNSET_F_INV
: 0;
283 if (nft_dump_register(skb
, NFTA_DYNSET_SREG_KEY
, priv
->sreg_key
))
284 goto nla_put_failure
;
285 if (priv
->set
->flags
& NFT_SET_MAP
&&
286 nft_dump_register(skb
, NFTA_DYNSET_SREG_DATA
, priv
->sreg_data
))
287 goto nla_put_failure
;
288 if (nla_put_be32(skb
, NFTA_DYNSET_OP
, htonl(priv
->op
)))
289 goto nla_put_failure
;
290 if (nla_put_string(skb
, NFTA_DYNSET_SET_NAME
, priv
->set
->name
))
291 goto nla_put_failure
;
292 if (nla_put_be64(skb
, NFTA_DYNSET_TIMEOUT
,
293 cpu_to_be64(jiffies_to_msecs(priv
->timeout
)),
295 goto nla_put_failure
;
296 if (priv
->expr
&& nft_expr_dump(skb
, NFTA_DYNSET_EXPR
, priv
->expr
))
297 goto nla_put_failure
;
298 if (nla_put_be32(skb
, NFTA_DYNSET_FLAGS
, htonl(flags
)))
299 goto nla_put_failure
;
306 static const struct nft_expr_ops nft_dynset_ops
= {
307 .type
= &nft_dynset_type
,
308 .size
= NFT_EXPR_SIZE(sizeof(struct nft_dynset
)),
309 .eval
= nft_dynset_eval
,
310 .init
= nft_dynset_init
,
311 .destroy
= nft_dynset_destroy
,
312 .activate
= nft_dynset_activate
,
313 .deactivate
= nft_dynset_deactivate
,
314 .dump
= nft_dynset_dump
,
317 struct nft_expr_type nft_dynset_type __read_mostly
= {
319 .ops
= &nft_dynset_ops
,
320 .policy
= nft_dynset_policy
,
321 .maxattr
= NFTA_DYNSET_MAX
,
322 .owner
= THIS_MODULE
,