1 // SPDX-License-Identifier: GPL-2.0-only
3 * Copyright (c) 2008-2009 Patrick McHardy <kaber@trash.net>
5 * Development of this code funded by Astaro AG (http://www.astaro.com/)
8 #include <linux/kernel.h>
9 #include <linux/init.h>
10 #include <linux/module.h>
11 #include <linux/netlink.h>
12 #include <linux/netfilter.h>
13 #include <linux/if_arp.h>
14 #include <linux/netfilter/nf_tables.h>
15 #include <net/netfilter/nf_tables_core.h>
16 #include <net/netfilter/nf_tables_offload.h>
17 #include <net/netfilter/nf_tables.h>
21 enum nft_registers sreg
:8;
23 enum nft_cmp_ops op
:8;
26 void nft_cmp_eval(const struct nft_expr
*expr
,
27 struct nft_regs
*regs
,
28 const struct nft_pktinfo
*pkt
)
30 const struct nft_cmp_expr
*priv
= nft_expr_priv(expr
);
33 d
= memcmp(®s
->data
[priv
->sreg
], &priv
->data
, priv
->len
);
63 regs
->verdict
.code
= NFT_BREAK
;
66 static const struct nla_policy nft_cmp_policy
[NFTA_CMP_MAX
+ 1] = {
67 [NFTA_CMP_SREG
] = { .type
= NLA_U32
},
68 [NFTA_CMP_OP
] = { .type
= NLA_U32
},
69 [NFTA_CMP_DATA
] = { .type
= NLA_NESTED
},
72 static int nft_cmp_init(const struct nft_ctx
*ctx
, const struct nft_expr
*expr
,
73 const struct nlattr
* const tb
[])
75 struct nft_cmp_expr
*priv
= nft_expr_priv(expr
);
76 struct nft_data_desc desc
;
79 err
= nft_data_init(NULL
, &priv
->data
, sizeof(priv
->data
), &desc
,
84 if (desc
.type
!= NFT_DATA_VALUE
) {
86 nft_data_release(&priv
->data
, desc
.type
);
90 priv
->sreg
= nft_parse_register(tb
[NFTA_CMP_SREG
]);
91 err
= nft_validate_register_load(priv
->sreg
, desc
.len
);
95 priv
->op
= ntohl(nla_get_be32(tb
[NFTA_CMP_OP
]));
100 static int nft_cmp_dump(struct sk_buff
*skb
, const struct nft_expr
*expr
)
102 const struct nft_cmp_expr
*priv
= nft_expr_priv(expr
);
104 if (nft_dump_register(skb
, NFTA_CMP_SREG
, priv
->sreg
))
105 goto nla_put_failure
;
106 if (nla_put_be32(skb
, NFTA_CMP_OP
, htonl(priv
->op
)))
107 goto nla_put_failure
;
109 if (nft_data_dump(skb
, NFTA_CMP_DATA
, &priv
->data
,
110 NFT_DATA_VALUE
, priv
->len
) < 0)
111 goto nla_put_failure
;
118 static int __nft_cmp_offload(struct nft_offload_ctx
*ctx
,
119 struct nft_flow_rule
*flow
,
120 const struct nft_cmp_expr
*priv
)
122 struct nft_offload_reg
*reg
= &ctx
->regs
[priv
->sreg
];
123 u8
*mask
= (u8
*)&flow
->match
.mask
;
124 u8
*key
= (u8
*)&flow
->match
.key
;
126 if (priv
->op
!= NFT_CMP_EQ
|| reg
->len
!= priv
->len
)
129 memcpy(key
+ reg
->offset
, &priv
->data
, priv
->len
);
130 memcpy(mask
+ reg
->offset
, ®
->mask
, priv
->len
);
132 flow
->match
.dissector
.used_keys
|= BIT(reg
->key
);
133 flow
->match
.dissector
.offset
[reg
->key
] = reg
->base_offset
;
135 if (reg
->key
== FLOW_DISSECTOR_KEY_META
&&
136 reg
->offset
== offsetof(struct nft_flow_key
, meta
.ingress_iftype
) &&
137 nft_reg_load16(priv
->data
.data
) != ARPHRD_ETHER
)
140 nft_offload_update_dependency(ctx
, &priv
->data
, priv
->len
);
145 static int nft_cmp_offload(struct nft_offload_ctx
*ctx
,
146 struct nft_flow_rule
*flow
,
147 const struct nft_expr
*expr
)
149 const struct nft_cmp_expr
*priv
= nft_expr_priv(expr
);
151 return __nft_cmp_offload(ctx
, flow
, priv
);
154 static const struct nft_expr_ops nft_cmp_ops
= {
155 .type
= &nft_cmp_type
,
156 .size
= NFT_EXPR_SIZE(sizeof(struct nft_cmp_expr
)),
157 .eval
= nft_cmp_eval
,
158 .init
= nft_cmp_init
,
159 .dump
= nft_cmp_dump
,
160 .offload
= nft_cmp_offload
,
163 static int nft_cmp_fast_init(const struct nft_ctx
*ctx
,
164 const struct nft_expr
*expr
,
165 const struct nlattr
* const tb
[])
167 struct nft_cmp_fast_expr
*priv
= nft_expr_priv(expr
);
168 struct nft_data_desc desc
;
169 struct nft_data data
;
173 err
= nft_data_init(NULL
, &data
, sizeof(data
), &desc
,
178 priv
->sreg
= nft_parse_register(tb
[NFTA_CMP_SREG
]);
179 err
= nft_validate_register_load(priv
->sreg
, desc
.len
);
183 desc
.len
*= BITS_PER_BYTE
;
184 mask
= nft_cmp_fast_mask(desc
.len
);
186 priv
->data
= data
.data
[0] & mask
;
187 priv
->len
= desc
.len
;
191 static int nft_cmp_fast_offload(struct nft_offload_ctx
*ctx
,
192 struct nft_flow_rule
*flow
,
193 const struct nft_expr
*expr
)
195 const struct nft_cmp_fast_expr
*priv
= nft_expr_priv(expr
);
196 struct nft_cmp_expr cmp
= {
203 .len
= priv
->len
/ BITS_PER_BYTE
,
207 return __nft_cmp_offload(ctx
, flow
, &cmp
);
210 static int nft_cmp_fast_dump(struct sk_buff
*skb
, const struct nft_expr
*expr
)
212 const struct nft_cmp_fast_expr
*priv
= nft_expr_priv(expr
);
213 struct nft_data data
;
215 if (nft_dump_register(skb
, NFTA_CMP_SREG
, priv
->sreg
))
216 goto nla_put_failure
;
217 if (nla_put_be32(skb
, NFTA_CMP_OP
, htonl(NFT_CMP_EQ
)))
218 goto nla_put_failure
;
220 data
.data
[0] = priv
->data
;
221 if (nft_data_dump(skb
, NFTA_CMP_DATA
, &data
,
222 NFT_DATA_VALUE
, priv
->len
/ BITS_PER_BYTE
) < 0)
223 goto nla_put_failure
;
230 const struct nft_expr_ops nft_cmp_fast_ops
= {
231 .type
= &nft_cmp_type
,
232 .size
= NFT_EXPR_SIZE(sizeof(struct nft_cmp_fast_expr
)),
233 .eval
= NULL
, /* inlined */
234 .init
= nft_cmp_fast_init
,
235 .dump
= nft_cmp_fast_dump
,
236 .offload
= nft_cmp_fast_offload
,
239 static const struct nft_expr_ops
*
240 nft_cmp_select_ops(const struct nft_ctx
*ctx
, const struct nlattr
* const tb
[])
242 struct nft_data_desc desc
;
243 struct nft_data data
;
247 if (tb
[NFTA_CMP_SREG
] == NULL
||
248 tb
[NFTA_CMP_OP
] == NULL
||
249 tb
[NFTA_CMP_DATA
] == NULL
)
250 return ERR_PTR(-EINVAL
);
252 op
= ntohl(nla_get_be32(tb
[NFTA_CMP_OP
]));
262 return ERR_PTR(-EINVAL
);
265 err
= nft_data_init(NULL
, &data
, sizeof(data
), &desc
,
270 if (desc
.type
!= NFT_DATA_VALUE
) {
275 if (desc
.len
<= sizeof(u32
) && op
== NFT_CMP_EQ
)
276 return &nft_cmp_fast_ops
;
280 nft_data_release(&data
, desc
.type
);
281 return ERR_PTR(-EINVAL
);
284 struct nft_expr_type nft_cmp_type __read_mostly
= {
286 .select_ops
= nft_cmp_select_ops
,
287 .policy
= nft_cmp_policy
,
288 .maxattr
= NFTA_CMP_MAX
,
289 .owner
= THIS_MODULE
,