2 * (C) 2012-2013 by Pablo Neira Ayuso <pablo@netfilter.org>
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License version 2 as
6 * published by the Free Software Foundation.
8 * This software has been sponsored by Sophos Astaro <http://www.sophos.com>
11 #include <linux/kernel.h>
12 #include <linux/init.h>
13 #include <linux/module.h>
14 #include <linux/netlink.h>
15 #include <linux/netfilter.h>
16 #include <linux/netfilter/nfnetlink.h>
17 #include <linux/netfilter/nf_tables.h>
18 #include <linux/netfilter/nf_tables_compat.h>
19 #include <linux/netfilter/x_tables.h>
20 #include <linux/netfilter_ipv4/ip_tables.h>
21 #include <linux/netfilter_ipv6/ip6_tables.h>
22 #include <linux/netfilter_bridge/ebtables.h>
23 #include <linux/netfilter_arp/arp_tables.h>
24 #include <net/netfilter/nf_tables.h>
26 static int nft_compat_chain_validate_dependency(const char *tablename
,
27 const struct nft_chain
*chain
)
29 const struct nft_base_chain
*basechain
;
31 if (!tablename
|| !(chain
->flags
& NFT_BASE_CHAIN
))
34 basechain
= nft_base_chain(chain
);
35 if (strcmp(tablename
, "nat") == 0 &&
36 basechain
->type
->type
!= NFT_CHAIN_T_NAT
)
46 struct arpt_entry arp
;
50 nft_compat_set_par(struct xt_action_param
*par
, void *xt
, const void *xt_info
)
53 par
->targinfo
= xt_info
;
57 static void nft_target_eval_xt(const struct nft_expr
*expr
,
58 struct nft_regs
*regs
,
59 const struct nft_pktinfo
*pkt
)
61 void *info
= nft_expr_priv(expr
);
62 struct xt_target
*target
= expr
->ops
->data
;
63 struct sk_buff
*skb
= pkt
->skb
;
66 nft_compat_set_par((struct xt_action_param
*)&pkt
->xt
, target
, info
);
68 ret
= target
->target(skb
, &pkt
->xt
);
75 regs
->verdict
.code
= NFT_CONTINUE
;
78 regs
->verdict
.code
= ret
;
83 static void nft_target_eval_bridge(const struct nft_expr
*expr
,
84 struct nft_regs
*regs
,
85 const struct nft_pktinfo
*pkt
)
87 void *info
= nft_expr_priv(expr
);
88 struct xt_target
*target
= expr
->ops
->data
;
89 struct sk_buff
*skb
= pkt
->skb
;
92 nft_compat_set_par((struct xt_action_param
*)&pkt
->xt
, target
, info
);
94 ret
= target
->target(skb
, &pkt
->xt
);
101 regs
->verdict
.code
= NF_ACCEPT
;
104 regs
->verdict
.code
= NF_DROP
;
107 regs
->verdict
.code
= NFT_CONTINUE
;
110 regs
->verdict
.code
= NFT_RETURN
;
113 regs
->verdict
.code
= ret
;
118 static const struct nla_policy nft_target_policy
[NFTA_TARGET_MAX
+ 1] = {
119 [NFTA_TARGET_NAME
] = { .type
= NLA_NUL_STRING
},
120 [NFTA_TARGET_REV
] = { .type
= NLA_U32
},
121 [NFTA_TARGET_INFO
] = { .type
= NLA_BINARY
},
125 nft_target_set_tgchk_param(struct xt_tgchk_param
*par
,
126 const struct nft_ctx
*ctx
,
127 struct xt_target
*target
, void *info
,
128 union nft_entry
*entry
, u16 proto
, bool inv
)
131 par
->table
= ctx
->table
->name
;
132 switch (ctx
->afi
->family
) {
134 entry
->e4
.ip
.proto
= proto
;
135 entry
->e4
.ip
.invflags
= inv
? IPT_INV_PROTO
: 0;
139 entry
->e6
.ipv6
.flags
|= IP6T_F_PROTO
;
141 entry
->e6
.ipv6
.proto
= proto
;
142 entry
->e6
.ipv6
.invflags
= inv
? IP6T_INV_PROTO
: 0;
145 entry
->ebt
.ethproto
= (__force __be16
)proto
;
146 entry
->ebt
.invflags
= inv
? EBT_IPROTO
: 0;
151 par
->entryinfo
= entry
;
152 par
->target
= target
;
153 par
->targinfo
= info
;
154 if (ctx
->chain
->flags
& NFT_BASE_CHAIN
) {
155 const struct nft_base_chain
*basechain
=
156 nft_base_chain(ctx
->chain
);
157 const struct nf_hook_ops
*ops
= &basechain
->ops
[0];
159 par
->hook_mask
= 1 << ops
->hooknum
;
163 par
->family
= ctx
->afi
->family
;
164 par
->nft_compat
= true;
167 static void target_compat_from_user(struct xt_target
*t
, void *in
, void *out
)
171 memcpy(out
, in
, t
->targetsize
);
172 pad
= XT_ALIGN(t
->targetsize
) - t
->targetsize
;
174 memset(out
+ t
->targetsize
, 0, pad
);
177 static const struct nla_policy nft_rule_compat_policy
[NFTA_RULE_COMPAT_MAX
+ 1] = {
178 [NFTA_RULE_COMPAT_PROTO
] = { .type
= NLA_U32
},
179 [NFTA_RULE_COMPAT_FLAGS
] = { .type
= NLA_U32
},
182 static int nft_parse_compat(const struct nlattr
*attr
, u16
*proto
, bool *inv
)
184 struct nlattr
*tb
[NFTA_RULE_COMPAT_MAX
+1];
188 err
= nla_parse_nested(tb
, NFTA_RULE_COMPAT_MAX
, attr
,
189 nft_rule_compat_policy
);
193 if (!tb
[NFTA_RULE_COMPAT_PROTO
] || !tb
[NFTA_RULE_COMPAT_FLAGS
])
196 flags
= ntohl(nla_get_be32(tb
[NFTA_RULE_COMPAT_FLAGS
]));
197 if (flags
& ~NFT_RULE_COMPAT_F_MASK
)
199 if (flags
& NFT_RULE_COMPAT_F_INV
)
202 *proto
= ntohl(nla_get_be32(tb
[NFTA_RULE_COMPAT_PROTO
]));
207 nft_target_init(const struct nft_ctx
*ctx
, const struct nft_expr
*expr
,
208 const struct nlattr
* const tb
[])
210 void *info
= nft_expr_priv(expr
);
211 struct xt_target
*target
= expr
->ops
->data
;
212 struct xt_tgchk_param par
;
213 size_t size
= XT_ALIGN(nla_len(tb
[NFTA_TARGET_INFO
]));
216 union nft_entry e
= {};
219 ret
= nft_compat_chain_validate_dependency(target
->table
, ctx
->chain
);
223 target_compat_from_user(target
, nla_data(tb
[NFTA_TARGET_INFO
]), info
);
225 if (ctx
->nla
[NFTA_RULE_COMPAT
]) {
226 ret
= nft_parse_compat(ctx
->nla
[NFTA_RULE_COMPAT
], &proto
, &inv
);
231 nft_target_set_tgchk_param(&par
, ctx
, target
, info
, &e
, proto
, inv
);
233 ret
= xt_check_target(&par
, size
, proto
, inv
);
237 /* The standard target cannot be used */
238 if (target
->target
== NULL
) {
245 module_put(target
->me
);
250 nft_target_destroy(const struct nft_ctx
*ctx
, const struct nft_expr
*expr
)
252 struct xt_target
*target
= expr
->ops
->data
;
253 void *info
= nft_expr_priv(expr
);
254 struct xt_tgdtor_param par
;
259 par
.family
= ctx
->afi
->family
;
260 if (par
.target
->destroy
!= NULL
)
261 par
.target
->destroy(&par
);
263 module_put(target
->me
);
266 static int nft_target_dump(struct sk_buff
*skb
, const struct nft_expr
*expr
)
268 const struct xt_target
*target
= expr
->ops
->data
;
269 void *info
= nft_expr_priv(expr
);
271 if (nla_put_string(skb
, NFTA_TARGET_NAME
, target
->name
) ||
272 nla_put_be32(skb
, NFTA_TARGET_REV
, htonl(target
->revision
)) ||
273 nla_put(skb
, NFTA_TARGET_INFO
, XT_ALIGN(target
->targetsize
), info
))
274 goto nla_put_failure
;
282 static int nft_target_validate(const struct nft_ctx
*ctx
,
283 const struct nft_expr
*expr
,
284 const struct nft_data
**data
)
286 struct xt_target
*target
= expr
->ops
->data
;
287 unsigned int hook_mask
= 0;
290 if (ctx
->chain
->flags
& NFT_BASE_CHAIN
) {
291 const struct nft_base_chain
*basechain
=
292 nft_base_chain(ctx
->chain
);
293 const struct nf_hook_ops
*ops
= &basechain
->ops
[0];
295 hook_mask
= 1 << ops
->hooknum
;
296 if (!(hook_mask
& target
->hooks
))
299 ret
= nft_compat_chain_validate_dependency(target
->table
,
307 static void nft_match_eval(const struct nft_expr
*expr
,
308 struct nft_regs
*regs
,
309 const struct nft_pktinfo
*pkt
)
311 void *info
= nft_expr_priv(expr
);
312 struct xt_match
*match
= expr
->ops
->data
;
313 struct sk_buff
*skb
= pkt
->skb
;
316 nft_compat_set_par((struct xt_action_param
*)&pkt
->xt
, match
, info
);
318 ret
= match
->match(skb
, (struct xt_action_param
*)&pkt
->xt
);
320 if (pkt
->xt
.hotdrop
) {
321 regs
->verdict
.code
= NF_DROP
;
325 switch (ret
? 1 : 0) {
327 regs
->verdict
.code
= NFT_CONTINUE
;
330 regs
->verdict
.code
= NFT_BREAK
;
335 static const struct nla_policy nft_match_policy
[NFTA_MATCH_MAX
+ 1] = {
336 [NFTA_MATCH_NAME
] = { .type
= NLA_NUL_STRING
},
337 [NFTA_MATCH_REV
] = { .type
= NLA_U32
},
338 [NFTA_MATCH_INFO
] = { .type
= NLA_BINARY
},
341 /* struct xt_mtchk_param and xt_tgchk_param look very similar */
343 nft_match_set_mtchk_param(struct xt_mtchk_param
*par
, const struct nft_ctx
*ctx
,
344 struct xt_match
*match
, void *info
,
345 union nft_entry
*entry
, u16 proto
, bool inv
)
348 par
->table
= ctx
->table
->name
;
349 switch (ctx
->afi
->family
) {
351 entry
->e4
.ip
.proto
= proto
;
352 entry
->e4
.ip
.invflags
= inv
? IPT_INV_PROTO
: 0;
356 entry
->e6
.ipv6
.flags
|= IP6T_F_PROTO
;
358 entry
->e6
.ipv6
.proto
= proto
;
359 entry
->e6
.ipv6
.invflags
= inv
? IP6T_INV_PROTO
: 0;
362 entry
->ebt
.ethproto
= (__force __be16
)proto
;
363 entry
->ebt
.invflags
= inv
? EBT_IPROTO
: 0;
368 par
->entryinfo
= entry
;
370 par
->matchinfo
= info
;
371 if (ctx
->chain
->flags
& NFT_BASE_CHAIN
) {
372 const struct nft_base_chain
*basechain
=
373 nft_base_chain(ctx
->chain
);
374 const struct nf_hook_ops
*ops
= &basechain
->ops
[0];
376 par
->hook_mask
= 1 << ops
->hooknum
;
380 par
->family
= ctx
->afi
->family
;
381 par
->nft_compat
= true;
384 static void match_compat_from_user(struct xt_match
*m
, void *in
, void *out
)
388 memcpy(out
, in
, m
->matchsize
);
389 pad
= XT_ALIGN(m
->matchsize
) - m
->matchsize
;
391 memset(out
+ m
->matchsize
, 0, pad
);
395 nft_match_init(const struct nft_ctx
*ctx
, const struct nft_expr
*expr
,
396 const struct nlattr
* const tb
[])
398 void *info
= nft_expr_priv(expr
);
399 struct xt_match
*match
= expr
->ops
->data
;
400 struct xt_mtchk_param par
;
401 size_t size
= XT_ALIGN(nla_len(tb
[NFTA_MATCH_INFO
]));
404 union nft_entry e
= {};
407 ret
= nft_compat_chain_validate_dependency(match
->table
, ctx
->chain
);
411 match_compat_from_user(match
, nla_data(tb
[NFTA_MATCH_INFO
]), info
);
413 if (ctx
->nla
[NFTA_RULE_COMPAT
]) {
414 ret
= nft_parse_compat(ctx
->nla
[NFTA_RULE_COMPAT
], &proto
, &inv
);
419 nft_match_set_mtchk_param(&par
, ctx
, match
, info
, &e
, proto
, inv
);
421 ret
= xt_check_match(&par
, size
, proto
, inv
);
427 module_put(match
->me
);
432 nft_match_destroy(const struct nft_ctx
*ctx
, const struct nft_expr
*expr
)
434 struct xt_match
*match
= expr
->ops
->data
;
435 void *info
= nft_expr_priv(expr
);
436 struct xt_mtdtor_param par
;
440 par
.matchinfo
= info
;
441 par
.family
= ctx
->afi
->family
;
442 if (par
.match
->destroy
!= NULL
)
443 par
.match
->destroy(&par
);
445 module_put(match
->me
);
448 static int nft_match_dump(struct sk_buff
*skb
, const struct nft_expr
*expr
)
450 void *info
= nft_expr_priv(expr
);
451 struct xt_match
*match
= expr
->ops
->data
;
453 if (nla_put_string(skb
, NFTA_MATCH_NAME
, match
->name
) ||
454 nla_put_be32(skb
, NFTA_MATCH_REV
, htonl(match
->revision
)) ||
455 nla_put(skb
, NFTA_MATCH_INFO
, XT_ALIGN(match
->matchsize
), info
))
456 goto nla_put_failure
;
464 static int nft_match_validate(const struct nft_ctx
*ctx
,
465 const struct nft_expr
*expr
,
466 const struct nft_data
**data
)
468 struct xt_match
*match
= expr
->ops
->data
;
469 unsigned int hook_mask
= 0;
472 if (ctx
->chain
->flags
& NFT_BASE_CHAIN
) {
473 const struct nft_base_chain
*basechain
=
474 nft_base_chain(ctx
->chain
);
475 const struct nf_hook_ops
*ops
= &basechain
->ops
[0];
477 hook_mask
= 1 << ops
->hooknum
;
478 if (!(hook_mask
& match
->hooks
))
481 ret
= nft_compat_chain_validate_dependency(match
->table
,
490 nfnl_compat_fill_info(struct sk_buff
*skb
, u32 portid
, u32 seq
, u32 type
,
491 int event
, u16 family
, const char *name
,
494 struct nlmsghdr
*nlh
;
495 struct nfgenmsg
*nfmsg
;
496 unsigned int flags
= portid
? NLM_F_MULTI
: 0;
498 event
|= NFNL_SUBSYS_NFT_COMPAT
<< 8;
499 nlh
= nlmsg_put(skb
, portid
, seq
, event
, sizeof(*nfmsg
), flags
);
503 nfmsg
= nlmsg_data(nlh
);
504 nfmsg
->nfgen_family
= family
;
505 nfmsg
->version
= NFNETLINK_V0
;
508 if (nla_put_string(skb
, NFTA_COMPAT_NAME
, name
) ||
509 nla_put_be32(skb
, NFTA_COMPAT_REV
, htonl(rev
)) ||
510 nla_put_be32(skb
, NFTA_COMPAT_TYPE
, htonl(target
)))
511 goto nla_put_failure
;
518 nlmsg_cancel(skb
, nlh
);
523 nfnl_compat_get(struct sock
*nfnl
, struct sk_buff
*skb
,
524 const struct nlmsghdr
*nlh
, const struct nlattr
* const tb
[])
527 struct nfgenmsg
*nfmsg
;
531 struct sk_buff
*skb2
;
533 if (tb
[NFTA_COMPAT_NAME
] == NULL
||
534 tb
[NFTA_COMPAT_REV
] == NULL
||
535 tb
[NFTA_COMPAT_TYPE
] == NULL
)
538 name
= nla_data(tb
[NFTA_COMPAT_NAME
]);
539 rev
= ntohl(nla_get_be32(tb
[NFTA_COMPAT_REV
]));
540 target
= ntohl(nla_get_be32(tb
[NFTA_COMPAT_TYPE
]));
542 nfmsg
= nlmsg_data(nlh
);
544 switch(nfmsg
->nfgen_family
) {
558 pr_err("nft_compat: unsupported protocol %d\n",
559 nfmsg
->nfgen_family
);
563 try_then_request_module(xt_find_revision(nfmsg
->nfgen_family
, name
,
570 skb2
= nlmsg_new(NLMSG_DEFAULT_SIZE
, GFP_KERNEL
);
574 /* include the best revision for this extension in the message */
575 if (nfnl_compat_fill_info(skb2
, NETLINK_CB(skb
).portid
,
577 NFNL_MSG_TYPE(nlh
->nlmsg_type
),
580 name
, ret
, target
) <= 0) {
585 ret
= netlink_unicast(nfnl
, skb2
, NETLINK_CB(skb
).portid
,
590 return ret
== -EAGAIN
? -ENOBUFS
: ret
;
593 static const struct nla_policy nfnl_compat_policy_get
[NFTA_COMPAT_MAX
+1] = {
594 [NFTA_COMPAT_NAME
] = { .type
= NLA_NUL_STRING
,
595 .len
= NFT_COMPAT_NAME_MAX
-1 },
596 [NFTA_COMPAT_REV
] = { .type
= NLA_U32
},
597 [NFTA_COMPAT_TYPE
] = { .type
= NLA_U32
},
600 static const struct nfnl_callback nfnl_nft_compat_cb
[NFNL_MSG_COMPAT_MAX
] = {
601 [NFNL_MSG_COMPAT_GET
] = { .call
= nfnl_compat_get
,
602 .attr_count
= NFTA_COMPAT_MAX
,
603 .policy
= nfnl_compat_policy_get
},
606 static const struct nfnetlink_subsystem nfnl_compat_subsys
= {
607 .name
= "nft-compat",
608 .subsys_id
= NFNL_SUBSYS_NFT_COMPAT
,
609 .cb_count
= NFNL_MSG_COMPAT_MAX
,
610 .cb
= nfnl_nft_compat_cb
,
613 static LIST_HEAD(nft_match_list
);
616 struct list_head head
;
617 struct nft_expr_ops ops
;
620 static struct nft_expr_type nft_match_type
;
622 static const struct nft_expr_ops
*
623 nft_match_select_ops(const struct nft_ctx
*ctx
,
624 const struct nlattr
* const tb
[])
626 struct nft_xt
*nft_match
;
627 struct xt_match
*match
;
631 if (tb
[NFTA_MATCH_NAME
] == NULL
||
632 tb
[NFTA_MATCH_REV
] == NULL
||
633 tb
[NFTA_MATCH_INFO
] == NULL
)
634 return ERR_PTR(-EINVAL
);
636 mt_name
= nla_data(tb
[NFTA_MATCH_NAME
]);
637 rev
= ntohl(nla_get_be32(tb
[NFTA_MATCH_REV
]));
638 family
= ctx
->afi
->family
;
640 /* Re-use the existing match if it's already loaded. */
641 list_for_each_entry(nft_match
, &nft_match_list
, head
) {
642 struct xt_match
*match
= nft_match
->ops
.data
;
644 if (strcmp(match
->name
, mt_name
) == 0 &&
645 match
->revision
== rev
&& match
->family
== family
) {
646 if (!try_module_get(match
->me
))
647 return ERR_PTR(-ENOENT
);
649 return &nft_match
->ops
;
653 match
= xt_request_find_match(family
, mt_name
, rev
);
655 return ERR_PTR(-ENOENT
);
657 /* This is the first time we use this match, allocate operations */
658 nft_match
= kzalloc(sizeof(struct nft_xt
), GFP_KERNEL
);
659 if (nft_match
== NULL
)
660 return ERR_PTR(-ENOMEM
);
662 nft_match
->ops
.type
= &nft_match_type
;
663 nft_match
->ops
.size
= NFT_EXPR_SIZE(XT_ALIGN(match
->matchsize
));
664 nft_match
->ops
.eval
= nft_match_eval
;
665 nft_match
->ops
.init
= nft_match_init
;
666 nft_match
->ops
.destroy
= nft_match_destroy
;
667 nft_match
->ops
.dump
= nft_match_dump
;
668 nft_match
->ops
.validate
= nft_match_validate
;
669 nft_match
->ops
.data
= match
;
671 list_add(&nft_match
->head
, &nft_match_list
);
673 return &nft_match
->ops
;
676 static void nft_match_release(void)
678 struct nft_xt
*nft_match
, *tmp
;
680 list_for_each_entry_safe(nft_match
, tmp
, &nft_match_list
, head
)
684 static struct nft_expr_type nft_match_type __read_mostly
= {
686 .select_ops
= nft_match_select_ops
,
687 .policy
= nft_match_policy
,
688 .maxattr
= NFTA_MATCH_MAX
,
689 .owner
= THIS_MODULE
,
692 static LIST_HEAD(nft_target_list
);
694 static struct nft_expr_type nft_target_type
;
696 static const struct nft_expr_ops
*
697 nft_target_select_ops(const struct nft_ctx
*ctx
,
698 const struct nlattr
* const tb
[])
700 struct nft_xt
*nft_target
;
701 struct xt_target
*target
;
705 if (tb
[NFTA_TARGET_NAME
] == NULL
||
706 tb
[NFTA_TARGET_REV
] == NULL
||
707 tb
[NFTA_TARGET_INFO
] == NULL
)
708 return ERR_PTR(-EINVAL
);
710 tg_name
= nla_data(tb
[NFTA_TARGET_NAME
]);
711 rev
= ntohl(nla_get_be32(tb
[NFTA_TARGET_REV
]));
712 family
= ctx
->afi
->family
;
714 /* Re-use the existing target if it's already loaded. */
715 list_for_each_entry(nft_target
, &nft_target_list
, head
) {
716 struct xt_target
*target
= nft_target
->ops
.data
;
718 if (strcmp(target
->name
, tg_name
) == 0 &&
719 target
->revision
== rev
&& target
->family
== family
) {
720 if (!try_module_get(target
->me
))
721 return ERR_PTR(-ENOENT
);
723 return &nft_target
->ops
;
727 target
= xt_request_find_target(family
, tg_name
, rev
);
729 return ERR_PTR(-ENOENT
);
731 /* This is the first time we use this target, allocate operations */
732 nft_target
= kzalloc(sizeof(struct nft_xt
), GFP_KERNEL
);
733 if (nft_target
== NULL
)
734 return ERR_PTR(-ENOMEM
);
736 nft_target
->ops
.type
= &nft_target_type
;
737 nft_target
->ops
.size
= NFT_EXPR_SIZE(XT_ALIGN(target
->targetsize
));
738 nft_target
->ops
.init
= nft_target_init
;
739 nft_target
->ops
.destroy
= nft_target_destroy
;
740 nft_target
->ops
.dump
= nft_target_dump
;
741 nft_target
->ops
.validate
= nft_target_validate
;
742 nft_target
->ops
.data
= target
;
744 if (family
== NFPROTO_BRIDGE
)
745 nft_target
->ops
.eval
= nft_target_eval_bridge
;
747 nft_target
->ops
.eval
= nft_target_eval_xt
;
749 list_add(&nft_target
->head
, &nft_target_list
);
751 return &nft_target
->ops
;
754 static void nft_target_release(void)
756 struct nft_xt
*nft_target
, *tmp
;
758 list_for_each_entry_safe(nft_target
, tmp
, &nft_target_list
, head
)
762 static struct nft_expr_type nft_target_type __read_mostly
= {
764 .select_ops
= nft_target_select_ops
,
765 .policy
= nft_target_policy
,
766 .maxattr
= NFTA_TARGET_MAX
,
767 .owner
= THIS_MODULE
,
770 static int __init
nft_compat_module_init(void)
774 ret
= nft_register_expr(&nft_match_type
);
778 ret
= nft_register_expr(&nft_target_type
);
782 ret
= nfnetlink_subsys_register(&nfnl_compat_subsys
);
784 pr_err("nft_compat: cannot register with nfnetlink.\n");
788 pr_info("nf_tables_compat: (c) 2012 Pablo Neira Ayuso <pablo@netfilter.org>\n");
793 nft_unregister_expr(&nft_target_type
);
795 nft_unregister_expr(&nft_match_type
);
799 static void __exit
nft_compat_module_exit(void)
801 nfnetlink_subsys_unregister(&nfnl_compat_subsys
);
802 nft_unregister_expr(&nft_target_type
);
803 nft_unregister_expr(&nft_match_type
);
805 nft_target_release();
808 MODULE_ALIAS_NFNL_SUBSYS(NFNL_SUBSYS_NFT_COMPAT
);
810 module_init(nft_compat_module_init
);
811 module_exit(nft_compat_module_exit
);
813 MODULE_LICENSE("GPL");
814 MODULE_AUTHOR("Pablo Neira Ayuso <pablo@netfilter.org>");
815 MODULE_ALIAS_NFT_EXPR("match");
816 MODULE_ALIAS_NFT_EXPR("target");