mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-18 22:14:16 +00:00 
			
		
		
		
	Merge git://git.kernel.org/pub/scm/linux/kernel/git/pablo/nf-next
Pablo Neira Ayuso says:
====================
Netfilter/IPVS updates for net-next
The following patchset contains Netfilter/IPVS updates for your net-next
tree. This batch comes with more input sanitization for xtables to
address bug reports from fuzzers, preparation works to the flowtable
infrastructure and assorted updates. In no particular order, they are:
1) Make sure userspace provides a valid standard target verdict, from
   Florian Westphal.
2) Sanitize error target size, also from Florian.
3) Validate that last rule in basechain matches underflow/policy since
   userspace assumes this when decoding the ruleset blob that comes
   from the kernel, from Florian.
4) Consolidate hook entry checks through xt_check_table_hooks(),
   patch from Florian.
5) Cap ruleset allocations at 512 mbytes, 134217728 rules and reject
   very large compat offset arrays, so we have a reasonable upper limit
   and fuzzers don't exercise the oom-killer. Patches from Florian.
6) Several WARN_ON checks on xtables mutex helper, from Florian.
7) xt_rateest now has a hashtable per net, from Cong Wang.
8) Consolidate counter allocation in xt_counters_alloc(), from Florian.
9) Earlier xt_table_unlock() call in {ip,ip6,arp,eb}tables, patch
   from Xin Long.
10) Set FLOW_OFFLOAD_DIR_* to IP_CT_DIR_* definitions, patch from
    Felix Fietkau.
11) Consolidate code through flow_offload_fill_dir(), also from Felix.
12) Inline ip6_dst_mtu_forward() just like ip_dst_mtu_maybe_forward()
    to remove a dependency with flowtable and ipv6.ko, from Felix.
13) Cache mtu size in flow_offload_tuple object, this is safe for
    forwarding as f87c10a8aa describes, from Felix.
14) Rename nf_flow_table.c to nf_flow_table_core.o, to simplify too
    modular infrastructure, from Felix.
15) Add rt0, rt2 and rt4 IPv6 routing extension support, patch from
    Ahmed Abdelsalam.
16) Remove unused parameter in nf_conncount_count(), from Yi-Hung Wei.
17) Support for counting only to nf_conncount infrastructure, patch
    from Yi-Hung Wei.
18) Add strict NFT_CT_{SRC_IP,DST_IP,SRC_IP6,DST_IP6} key datatypes
    to nft_ct.
19) Use boolean as return value from ipt_ah and from IPVS too, patch
    from Gustavo A. R. Silva.
20) Remove useless parameters in nfnl_acct_overquota() and
    nf_conntrack_broadcast_help(), from Taehee Yoo.
21) Use ipv6_addr_is_multicast() from xt_cluster, also from Taehee Yoo.
22) Statify nf_tables_obj_lookup_byhandle, patch from Fengguang Wu.
23) Fix typo in xt_limit, from Geert Uytterhoeven.
24) Do no use VLAs in Netfilter code, again from Gustavo.
25) Use ADD_COUNTER from ebtables, from Taehee Yoo.
26) Bitshift support for CONNMARK and MARK targets, from Jack Ma.
27) Use pr_*() and add pr_fmt(), from Arushi Singhal.
28) Add synproxy support to ctnetlink.
29) ICMP type and IGMP matching support for ebtables, patches from
    Matthias Schiffer.
30) Support for the revision infrastructure to ebtables, from
    Bernie Harris.
31) String match support for ebtables, also from Bernie.
32) Documentation for the new flowtable infrastructure.
33) Use generic comparison functions in ebt_stp, from Joe Perches.
34) Demodularize filter chains in nftables.
35) Register conntrack hooks in case nftables NAT chain is added.
36) Merge assignments with return in a couple of spots in the
    Netfilter codebase, also from Arushi.
37) Document that xtables percpu counters are stored in the same
    memory area, from Ben Hutchings.
38) Revert mark_source_chains() sanity checks that break existing
    rulesets, from Florian Westphal.
39) Use is_zero_ether_addr() in the ipset codebase, from Joe Perches.
====================
Signed-off-by: David S. Miller <davem@davemloft.net>
			
			
This commit is contained in:
		
						commit
						d162190bde
					
				
					 75 changed files with 1383 additions and 858 deletions
				
			
		
							
								
								
									
										112
									
								
								Documentation/networking/nf_flowtable.txt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										112
									
								
								Documentation/networking/nf_flowtable.txt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,112 @@ | |||
| Netfilter's flowtable infrastructure | ||||
| ==================================== | ||||
| 
 | ||||
| This documentation describes the software flowtable infrastructure available in | ||||
| Netfilter since Linux kernel 4.16. | ||||
| 
 | ||||
| Overview | ||||
| -------- | ||||
| 
 | ||||
| Initial packets follow the classic forwarding path, once the flow enters the | ||||
| established state according to the conntrack semantics (ie. we have seen traffic | ||||
| in both directions), then you can decide to offload the flow to the flowtable | ||||
| from the forward chain via the 'flow offload' action available in nftables. | ||||
| 
 | ||||
| Packets that find an entry in the flowtable (ie. flowtable hit) are sent to the | ||||
| output netdevice via neigh_xmit(), hence, they bypass the classic forwarding | ||||
| path (the visible effect is that you do not see these packets from any of the | ||||
| netfilter hooks coming after the ingress). In case of flowtable miss, the packet | ||||
| follows the classic forward path. | ||||
| 
 | ||||
| The flowtable uses a resizable hashtable, lookups are based on the following | ||||
| 7-tuple selectors: source, destination, layer 3 and layer 4 protocols, source | ||||
| and destination ports and the input interface (useful in case there are several | ||||
| conntrack zones in place). | ||||
| 
 | ||||
| Flowtables are populated via the 'flow offload' nftables action, so the user can | ||||
| selectively specify what flows are placed into the flow table. Hence, packets | ||||
| follow the classic forwarding path unless the user explicitly instruct packets | ||||
| to use this new alternative forwarding path via nftables policy. | ||||
| 
 | ||||
| This is represented in Fig.1, which describes the classic forwarding path | ||||
| including the Netfilter hooks and the flowtable fastpath bypass. | ||||
| 
 | ||||
|                                          userspace process | ||||
|                                           ^              | | ||||
|                                           |              | | ||||
|                                      _____|____     ____\/___ | ||||
|                                     /          \   /         \ | ||||
|                                     |   input   |  |  output  | | ||||
|                                     \__________/   \_________/ | ||||
|                                          ^               | | ||||
|                                          |               | | ||||
|       _________      __________      ---------     _____\/_____ | ||||
|      /         \    /          \     |Routing |   /            \ | ||||
|   -->  ingress  ---> prerouting ---> |decision|   | postrouting |--> neigh_xmit | ||||
|      \_________/    \__________/     ----------   \____________/          ^ | ||||
|        |      ^          |               |               ^                | | ||||
|    flowtable  |          |          ____\/___            |                | | ||||
|        |      |          |         /         \           |                | | ||||
|     __\/___   |          --------->| forward |------------                | | ||||
|     |-----|   |                    \_________/                            | | ||||
|     |-----|   |                 'flow offload' rule                       | | ||||
|     |-----|   |                   adds entry to                           | | ||||
|     |_____|   |                     flowtable                             | | ||||
|        |      |                                                           | | ||||
|       / \     |                                                           | | ||||
|      /hit\_no_|                                                           | | ||||
|      \ ? /                                                                | | ||||
|       \ /                                                                 | | ||||
|        |__yes_________________fastpath bypass ____________________________| | ||||
| 
 | ||||
|                Fig.1 Netfilter hooks and flowtable interactions | ||||
| 
 | ||||
| The flowtable entry also stores the NAT configuration, so all packets are | ||||
| mangled according to the NAT policy that matches the initial packets that went | ||||
| through the classic forwarding path. The TTL is decremented before calling | ||||
| neigh_xmit(). Fragmented traffic is passed up to follow the classic forwarding | ||||
| path given that the transport selectors are missing, therefore flowtable lookup | ||||
| is not possible. | ||||
| 
 | ||||
| Example configuration | ||||
| --------------------- | ||||
| 
 | ||||
| Enabling the flowtable bypass is relatively easy, you only need to create a | ||||
| flowtable and add one rule to your forward chain. | ||||
| 
 | ||||
|         table inet x { | ||||
| 		flowtable f { | ||||
| 			hook ingress priority 0 devices = { eth0, eth1 }; | ||||
| 		} | ||||
|                 chain y { | ||||
|                         type filter hook forward priority 0; policy accept; | ||||
|                         ip protocol tcp flow offload @f | ||||
|                         counter packets 0 bytes 0 | ||||
|                 } | ||||
|         } | ||||
| 
 | ||||
| This example adds the flowtable 'f' to the ingress hook of the eth0 and eth1 | ||||
| netdevices. You can create as many flowtables as you want in case you need to | ||||
| perform resource partitioning. The flowtable priority defines the order in which | ||||
| hooks are run in the pipeline, this is convenient in case you already have a | ||||
| nftables ingress chain (make sure the flowtable priority is smaller than the | ||||
| nftables ingress chain hence the flowtable runs before in the pipeline). | ||||
| 
 | ||||
| The 'flow offload' action from the forward chain 'y' adds an entry to the | ||||
| flowtable for the TCP syn-ack packet coming in the reply direction. Once the | ||||
| flow is offloaded, you will observe that the counter rule in the example above | ||||
| does not get updated for the packets that are being forwarded through the | ||||
| forwarding bypass. | ||||
| 
 | ||||
| More reading | ||||
| ------------ | ||||
| 
 | ||||
| This documentation is based on the LWN.net articles [1][2]. Rafal Milecki also | ||||
| made a very complete and comprehensive summary called "A state of network | ||||
| acceleration" that describes how things were before this infrastructure was | ||||
| mailined [3] and it also makes a rough summary of this work [4]. | ||||
| 
 | ||||
| [1] https://lwn.net/Articles/738214/ | ||||
| [2] https://lwn.net/Articles/742164/ | ||||
| [3] http://lists.infradead.org/pipermail/lede-dev/2018-January/010830.html | ||||
| [4] http://lists.infradead.org/pipermail/lede-dev/2018-January/010829.html | ||||
|  | @ -16,6 +16,5 @@ struct nf_acct; | |||
| struct nf_acct *nfnl_acct_find_get(struct net *net, const char *filter_name); | ||||
| void nfnl_acct_put(struct nf_acct *acct); | ||||
| void nfnl_acct_update(const struct sk_buff *skb, struct nf_acct *nfacct); | ||||
| int nfnl_acct_overquota(struct net *net, const struct sk_buff *skb, | ||||
| 			struct nf_acct *nfacct); | ||||
| int nfnl_acct_overquota(struct net *net, struct nf_acct *nfacct); | ||||
| #endif /* _NFNL_ACCT_H */ | ||||
|  |  | |||
|  | @ -281,6 +281,8 @@ int xt_check_entry_offsets(const void *base, const char *elems, | |||
| 			   unsigned int target_offset, | ||||
| 			   unsigned int next_offset); | ||||
| 
 | ||||
| int xt_check_table_hooks(const struct xt_table_info *info, unsigned int valid_hooks); | ||||
| 
 | ||||
| unsigned int *xt_alloc_entry_offsets(unsigned int size); | ||||
| bool xt_find_jump_offset(const unsigned int *offsets, | ||||
| 			 unsigned int target, unsigned int size); | ||||
|  | @ -301,6 +303,7 @@ int xt_data_to_user(void __user *dst, const void *src, | |||
| 
 | ||||
| void *xt_copy_counters_from_user(const void __user *user, unsigned int len, | ||||
| 				 struct xt_counters_info *info, bool compat); | ||||
| struct xt_counters *xt_counters_alloc(unsigned int counters); | ||||
| 
 | ||||
| struct xt_table *xt_register_table(struct net *net, | ||||
| 				   const struct xt_table *table, | ||||
|  | @ -509,7 +512,7 @@ void xt_compat_unlock(u_int8_t af); | |||
| 
 | ||||
| int xt_compat_add_offset(u_int8_t af, unsigned int offset, int delta); | ||||
| void xt_compat_flush_offsets(u_int8_t af); | ||||
| void xt_compat_init_offsets(u_int8_t af, unsigned int number); | ||||
| int xt_compat_init_offsets(u8 af, unsigned int number); | ||||
| int xt_compat_calc_jump(u_int8_t af, unsigned int offset); | ||||
| 
 | ||||
| int xt_compat_match_offset(const struct xt_match *match); | ||||
|  |  | |||
|  | @ -11,7 +11,6 @@ void nf_conncount_destroy(struct net *net, unsigned int family, | |||
| unsigned int nf_conncount_count(struct net *net, | ||||
| 				struct nf_conncount_data *data, | ||||
| 				const u32 *key, | ||||
| 				unsigned int family, | ||||
| 				const struct nf_conntrack_tuple *tuple, | ||||
| 				const struct nf_conntrack_zone *zone); | ||||
| #endif | ||||
|  |  | |||
|  | @ -132,8 +132,7 @@ void nf_conntrack_helper_pernet_fini(struct net *net); | |||
| int nf_conntrack_helper_init(void); | ||||
| void nf_conntrack_helper_fini(void); | ||||
| 
 | ||||
| int nf_conntrack_broadcast_help(struct sk_buff *skb, unsigned int protoff, | ||||
| 				struct nf_conn *ct, | ||||
| int nf_conntrack_broadcast_help(struct sk_buff *skb, struct nf_conn *ct, | ||||
| 				enum ip_conntrack_info ctinfo, | ||||
| 				unsigned int timeout); | ||||
| 
 | ||||
|  |  | |||
|  | @ -434,11 +434,11 @@ static inline struct nft_set *nft_set_container_of(const void *priv) | |||
| 	return (void *)priv - offsetof(struct nft_set, data); | ||||
| } | ||||
| 
 | ||||
| struct nft_set *nft_set_lookup(const struct net *net, | ||||
| 			       const struct nft_table *table, | ||||
| 			       const struct nlattr *nla_set_name, | ||||
| 			       const struct nlattr *nla_set_id, | ||||
| 			       u8 genmask); | ||||
| struct nft_set *nft_set_lookup_global(const struct net *net, | ||||
| 				      const struct nft_table *table, | ||||
| 				      const struct nlattr *nla_set_name, | ||||
| 				      const struct nlattr *nla_set_id, | ||||
| 				      u8 genmask); | ||||
| 
 | ||||
| static inline unsigned long nft_set_gc_interval(const struct nft_set *set) | ||||
| { | ||||
|  | @ -868,7 +868,7 @@ struct nft_chain { | |||
| 	char				*name; | ||||
| }; | ||||
| 
 | ||||
| enum nft_chain_type { | ||||
| enum nft_chain_types { | ||||
| 	NFT_CHAIN_T_DEFAULT = 0, | ||||
| 	NFT_CHAIN_T_ROUTE, | ||||
| 	NFT_CHAIN_T_NAT, | ||||
|  | @ -876,7 +876,7 @@ enum nft_chain_type { | |||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * 	struct nf_chain_type - nf_tables chain type info | ||||
|  * 	struct nft_chain_type - nf_tables chain type info | ||||
|  * | ||||
|  * 	@name: name of the type | ||||
|  * 	@type: numeric identifier | ||||
|  | @ -884,18 +884,22 @@ enum nft_chain_type { | |||
|  * 	@owner: module owner | ||||
|  * 	@hook_mask: mask of valid hooks | ||||
|  * 	@hooks: array of hook functions | ||||
|  *	@init: chain initialization function | ||||
|  *	@free: chain release function | ||||
|  */ | ||||
| struct nf_chain_type { | ||||
| struct nft_chain_type { | ||||
| 	const char			*name; | ||||
| 	enum nft_chain_type		type; | ||||
| 	enum nft_chain_types		type; | ||||
| 	int				family; | ||||
| 	struct module			*owner; | ||||
| 	unsigned int			hook_mask; | ||||
| 	nf_hookfn			*hooks[NF_MAX_HOOKS]; | ||||
| 	int				(*init)(struct nft_ctx *ctx); | ||||
| 	void				(*free)(struct nft_ctx *ctx); | ||||
| }; | ||||
| 
 | ||||
| int nft_chain_validate_dependency(const struct nft_chain *chain, | ||||
| 				  enum nft_chain_type type); | ||||
| 				  enum nft_chain_types type); | ||||
| int nft_chain_validate_hooks(const struct nft_chain *chain, | ||||
|                              unsigned int hook_flags); | ||||
| 
 | ||||
|  | @ -917,7 +921,7 @@ struct nft_stats { | |||
|  */ | ||||
| struct nft_base_chain { | ||||
| 	struct nf_hook_ops		ops; | ||||
| 	const struct nf_chain_type	*type; | ||||
| 	const struct nft_chain_type	*type; | ||||
| 	u8				policy; | ||||
| 	u8				flags; | ||||
| 	struct nft_stats __percpu	*stats; | ||||
|  | @ -970,8 +974,8 @@ struct nft_table { | |||
| 	char				*name; | ||||
| }; | ||||
| 
 | ||||
| int nft_register_chain_type(const struct nf_chain_type *); | ||||
| void nft_unregister_chain_type(const struct nf_chain_type *); | ||||
| void nft_register_chain_type(const struct nft_chain_type *); | ||||
| void nft_unregister_chain_type(const struct nft_chain_type *); | ||||
| 
 | ||||
| int nft_register_expr(struct nft_expr_type *); | ||||
| void nft_unregister_expr(struct nft_expr_type *); | ||||
|  | @ -1345,4 +1349,7 @@ struct nft_trans_flowtable { | |||
| #define nft_trans_flowtable(trans)	\ | ||||
| 	(((struct nft_trans_flowtable *)trans->data)->flowtable) | ||||
| 
 | ||||
| int __init nft_chain_filter_init(void); | ||||
| void __exit nft_chain_filter_fini(void); | ||||
| 
 | ||||
| #endif /* _NET_NF_TABLES_H */ | ||||
|  |  | |||
|  | @ -21,7 +21,7 @@ struct xt_rateest { | |||
| 	struct net_rate_estimator __rcu *rate_est; | ||||
| }; | ||||
| 
 | ||||
| struct xt_rateest *xt_rateest_lookup(const char *name); | ||||
| void xt_rateest_put(struct xt_rateest *est); | ||||
| struct xt_rateest *xt_rateest_lookup(struct net *net, const char *name); | ||||
| void xt_rateest_put(struct net *net, struct xt_rateest *est); | ||||
| 
 | ||||
| #endif /* _XT_RATEEST_H */ | ||||
|  |  | |||
|  | @ -129,6 +129,7 @@ enum ip_conntrack_events { | |||
| 	IPCT_NATSEQADJ = IPCT_SEQADJ, | ||||
| 	IPCT_SECMARK,		/* new security mark has been set */ | ||||
| 	IPCT_LABEL,		/* new connlabel has been set */ | ||||
| 	IPCT_SYNPROXY,		/* synproxy has been set */ | ||||
| #ifdef __KERNEL__ | ||||
| 	__IPCT_MAX | ||||
| #endif | ||||
|  |  | |||
|  | @ -909,8 +909,8 @@ enum nft_rt_attributes { | |||
|  * @NFT_CT_EXPIRATION: relative conntrack expiration time in ms | ||||
|  * @NFT_CT_HELPER: connection tracking helper assigned to conntrack | ||||
|  * @NFT_CT_L3PROTOCOL: conntrack layer 3 protocol | ||||
|  * @NFT_CT_SRC: conntrack layer 3 protocol source (IPv4/IPv6 address) | ||||
|  * @NFT_CT_DST: conntrack layer 3 protocol destination (IPv4/IPv6 address) | ||||
|  * @NFT_CT_SRC: conntrack layer 3 protocol source (IPv4/IPv6 address, deprecated) | ||||
|  * @NFT_CT_DST: conntrack layer 3 protocol destination (IPv4/IPv6 address, deprecated) | ||||
|  * @NFT_CT_PROTOCOL: conntrack layer 4 protocol | ||||
|  * @NFT_CT_PROTO_SRC: conntrack layer 4 protocol source | ||||
|  * @NFT_CT_PROTO_DST: conntrack layer 4 protocol destination | ||||
|  | @ -920,6 +920,10 @@ enum nft_rt_attributes { | |||
|  * @NFT_CT_AVGPKT: conntrack average bytes per packet | ||||
|  * @NFT_CT_ZONE: conntrack zone | ||||
|  * @NFT_CT_EVENTMASK: ctnetlink events to be generated for this conntrack | ||||
|  * @NFT_CT_SRC_IP: conntrack layer 3 protocol source (IPv4 address) | ||||
|  * @NFT_CT_DST_IP: conntrack layer 3 protocol destination (IPv4 address) | ||||
|  * @NFT_CT_SRC_IP6: conntrack layer 3 protocol source (IPv6 address) | ||||
|  * @NFT_CT_DST_IP6: conntrack layer 3 protocol destination (IPv6 address) | ||||
|  */ | ||||
| enum nft_ct_keys { | ||||
| 	NFT_CT_STATE, | ||||
|  | @ -941,6 +945,10 @@ enum nft_ct_keys { | |||
| 	NFT_CT_AVGPKT, | ||||
| 	NFT_CT_ZONE, | ||||
| 	NFT_CT_EVENTMASK, | ||||
| 	NFT_CT_SRC_IP, | ||||
| 	NFT_CT_DST_IP, | ||||
| 	NFT_CT_SRC_IP6, | ||||
| 	NFT_CT_DST_IP6, | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  |  | |||
|  | @ -54,6 +54,7 @@ enum ctattr_type { | |||
| 	CTA_MARK_MASK, | ||||
| 	CTA_LABELS, | ||||
| 	CTA_LABELS_MASK, | ||||
| 	CTA_SYNPROXY, | ||||
| 	__CTA_MAX | ||||
| }; | ||||
| #define CTA_MAX (__CTA_MAX - 1) | ||||
|  | @ -190,6 +191,15 @@ enum ctattr_natseq { | |||
| }; | ||||
| #define CTA_NAT_SEQ_MAX (__CTA_NAT_SEQ_MAX - 1) | ||||
| 
 | ||||
| enum ctattr_synproxy { | ||||
| 	CTA_SYNPROXY_UNSPEC, | ||||
| 	CTA_SYNPROXY_ISN, | ||||
| 	CTA_SYNPROXY_ITS, | ||||
| 	CTA_SYNPROXY_TSOFF, | ||||
| 	__CTA_SYNPROXY_MAX, | ||||
| }; | ||||
| #define CTA_SYNPROXY_MAX (__CTA_SYNPROXY_MAX - 1) | ||||
| 
 | ||||
| enum ctattr_expect { | ||||
| 	CTA_EXPECT_UNSPEC, | ||||
| 	CTA_EXPECT_MASTER, | ||||
|  |  | |||
|  | @ -19,11 +19,21 @@ enum { | |||
| 	XT_CONNMARK_RESTORE | ||||
| }; | ||||
| 
 | ||||
| enum { | ||||
| 	D_SHIFT_LEFT = 0, | ||||
| 	D_SHIFT_RIGHT, | ||||
| }; | ||||
| 
 | ||||
| struct xt_connmark_tginfo1 { | ||||
| 	__u32 ctmark, ctmask, nfmask; | ||||
| 	__u8 mode; | ||||
| }; | ||||
| 
 | ||||
| struct xt_connmark_tginfo2 { | ||||
| 	__u32 ctmark, ctmask, nfmask; | ||||
| 	__u8 shift_dir, shift_bits, mode; | ||||
| }; | ||||
| 
 | ||||
| struct xt_connmark_mtinfo1 { | ||||
| 	__u32 mark, mask; | ||||
| 	__u8 invert; | ||||
|  |  | |||
|  | @ -24,8 +24,10 @@ | |||
| #define EBT_IP_PROTO 0x08 | ||||
| #define EBT_IP_SPORT 0x10 | ||||
| #define EBT_IP_DPORT 0x20 | ||||
| #define EBT_IP_ICMP 0x40 | ||||
| #define EBT_IP_IGMP 0x80 | ||||
| #define EBT_IP_MASK (EBT_IP_SOURCE | EBT_IP_DEST | EBT_IP_TOS | EBT_IP_PROTO |\ | ||||
|  EBT_IP_SPORT | EBT_IP_DPORT ) | ||||
| 		     EBT_IP_SPORT | EBT_IP_DPORT | EBT_IP_ICMP | EBT_IP_IGMP) | ||||
| #define EBT_IP_MATCH "ip" | ||||
| 
 | ||||
| /* the same values are used for the invflags */ | ||||
|  | @ -38,8 +40,15 @@ struct ebt_ip_info { | |||
| 	__u8  protocol; | ||||
| 	__u8  bitmask; | ||||
| 	__u8  invflags; | ||||
| 	__u16 sport[2]; | ||||
| 	__u16 dport[2]; | ||||
| 	union { | ||||
| 		__u16 sport[2]; | ||||
| 		__u8 icmp_type[2]; | ||||
| 		__u8 igmp_type[2]; | ||||
| 	}; | ||||
| 	union { | ||||
| 		__u16 dport[2]; | ||||
| 		__u8 icmp_code[2]; | ||||
| 	}; | ||||
| }; | ||||
| 
 | ||||
| #endif | ||||
|  |  | |||
|  | @ -20,6 +20,7 @@ | |||
| #define EBT_TABLE_MAXNAMELEN 32 | ||||
| #define EBT_CHAIN_MAXNAMELEN EBT_TABLE_MAXNAMELEN | ||||
| #define EBT_FUNCTION_MAXNAMELEN EBT_TABLE_MAXNAMELEN | ||||
| #define EBT_EXTENSION_MAXNAMELEN 31 | ||||
| 
 | ||||
| /* verdicts >0 are "branches" */ | ||||
| #define EBT_ACCEPT   -1 | ||||
|  | @ -120,7 +121,10 @@ struct ebt_entries { | |||
| 
 | ||||
| struct ebt_entry_match { | ||||
| 	union { | ||||
| 		char name[EBT_FUNCTION_MAXNAMELEN]; | ||||
| 		struct { | ||||
| 			char name[EBT_EXTENSION_MAXNAMELEN]; | ||||
| 			uint8_t revision; | ||||
| 		}; | ||||
| 		struct xt_match *match; | ||||
| 	} u; | ||||
| 	/* size of data */ | ||||
|  | @ -130,7 +134,10 @@ struct ebt_entry_match { | |||
| 
 | ||||
| struct ebt_entry_watcher { | ||||
| 	union { | ||||
| 		char name[EBT_FUNCTION_MAXNAMELEN]; | ||||
| 		struct { | ||||
| 			char name[EBT_EXTENSION_MAXNAMELEN]; | ||||
| 			uint8_t revision; | ||||
| 		}; | ||||
| 		struct xt_target *watcher; | ||||
| 	} u; | ||||
| 	/* size of data */ | ||||
|  | @ -140,7 +147,10 @@ struct ebt_entry_watcher { | |||
| 
 | ||||
| struct ebt_entry_target { | ||||
| 	union { | ||||
| 		char name[EBT_FUNCTION_MAXNAMELEN]; | ||||
| 		struct { | ||||
| 			char name[EBT_EXTENSION_MAXNAMELEN]; | ||||
| 			uint8_t revision; | ||||
| 		}; | ||||
| 		struct xt_target *target; | ||||
| 	} u; | ||||
| 	/* size of data */ | ||||
|  |  | |||
|  | @ -5,7 +5,7 @@ | |||
| menuconfig NF_TABLES_BRIDGE | ||||
| 	depends on BRIDGE && NETFILTER && NF_TABLES | ||||
| 	select NETFILTER_FAMILY_BRIDGE | ||||
| 	tristate "Ethernet Bridge nf_tables support" | ||||
| 	bool "Ethernet Bridge nf_tables support" | ||||
| 
 | ||||
| if NF_TABLES_BRIDGE | ||||
| 
 | ||||
|  |  | |||
|  | @ -3,7 +3,6 @@ | |||
| # Makefile for the netfilter modules for Link Layer filtering on a bridge.
 | ||||
| #
 | ||||
| 
 | ||||
| obj-$(CONFIG_NF_TABLES_BRIDGE) += nf_tables_bridge.o | ||||
| obj-$(CONFIG_NFT_BRIDGE_META)  += nft_meta_bridge.o | ||||
| obj-$(CONFIG_NFT_BRIDGE_REJECT)  += nft_reject_bridge.o | ||||
| 
 | ||||
|  |  | |||
|  | @ -19,9 +19,18 @@ | |||
| #include <linux/netfilter_bridge/ebtables.h> | ||||
| #include <linux/netfilter_bridge/ebt_ip.h> | ||||
| 
 | ||||
| struct tcpudphdr { | ||||
| 	__be16 src; | ||||
| 	__be16 dst; | ||||
| union pkthdr { | ||||
| 	struct { | ||||
| 		__be16 src; | ||||
| 		__be16 dst; | ||||
| 	} tcpudphdr; | ||||
| 	struct { | ||||
| 		u8 type; | ||||
| 		u8 code; | ||||
| 	} icmphdr; | ||||
| 	struct { | ||||
| 		u8 type; | ||||
| 	} igmphdr; | ||||
| }; | ||||
| 
 | ||||
| static bool | ||||
|  | @ -30,8 +39,8 @@ ebt_ip_mt(const struct sk_buff *skb, struct xt_action_param *par) | |||
| 	const struct ebt_ip_info *info = par->matchinfo; | ||||
| 	const struct iphdr *ih; | ||||
| 	struct iphdr _iph; | ||||
| 	const struct tcpudphdr *pptr; | ||||
| 	struct tcpudphdr _ports; | ||||
| 	const union pkthdr *pptr; | ||||
| 	union pkthdr _pkthdr; | ||||
| 
 | ||||
| 	ih = skb_header_pointer(skb, 0, sizeof(_iph), &_iph); | ||||
| 	if (ih == NULL) | ||||
|  | @ -50,29 +59,43 @@ ebt_ip_mt(const struct sk_buff *skb, struct xt_action_param *par) | |||
| 	if (info->bitmask & EBT_IP_PROTO) { | ||||
| 		if (NF_INVF(info, EBT_IP_PROTO, info->protocol != ih->protocol)) | ||||
| 			return false; | ||||
| 		if (!(info->bitmask & EBT_IP_DPORT) && | ||||
| 		    !(info->bitmask & EBT_IP_SPORT)) | ||||
| 		if (!(info->bitmask & (EBT_IP_DPORT | EBT_IP_SPORT | | ||||
| 				       EBT_IP_ICMP | EBT_IP_IGMP))) | ||||
| 			return true; | ||||
| 		if (ntohs(ih->frag_off) & IP_OFFSET) | ||||
| 			return false; | ||||
| 
 | ||||
| 		/* min icmp/igmp headersize is 4, so sizeof(_pkthdr) is ok. */ | ||||
| 		pptr = skb_header_pointer(skb, ih->ihl*4, | ||||
| 					  sizeof(_ports), &_ports); | ||||
| 					  sizeof(_pkthdr), &_pkthdr); | ||||
| 		if (pptr == NULL) | ||||
| 			return false; | ||||
| 		if (info->bitmask & EBT_IP_DPORT) { | ||||
| 			u32 dst = ntohs(pptr->dst); | ||||
| 			u32 dst = ntohs(pptr->tcpudphdr.dst); | ||||
| 			if (NF_INVF(info, EBT_IP_DPORT, | ||||
| 				    dst < info->dport[0] || | ||||
| 				    dst > info->dport[1])) | ||||
| 				return false; | ||||
| 		} | ||||
| 		if (info->bitmask & EBT_IP_SPORT) { | ||||
| 			u32 src = ntohs(pptr->src); | ||||
| 			u32 src = ntohs(pptr->tcpudphdr.src); | ||||
| 			if (NF_INVF(info, EBT_IP_SPORT, | ||||
| 				    src < info->sport[0] || | ||||
| 				    src > info->sport[1])) | ||||
| 				return false; | ||||
| 		} | ||||
| 		if ((info->bitmask & EBT_IP_ICMP) && | ||||
| 		    NF_INVF(info, EBT_IP_ICMP, | ||||
| 			    pptr->icmphdr.type < info->icmp_type[0] || | ||||
| 			    pptr->icmphdr.type > info->icmp_type[1] || | ||||
| 			    pptr->icmphdr.code < info->icmp_code[0] || | ||||
| 			    pptr->icmphdr.code > info->icmp_code[1])) | ||||
| 			return false; | ||||
| 		if ((info->bitmask & EBT_IP_IGMP) && | ||||
| 		    NF_INVF(info, EBT_IP_IGMP, | ||||
| 			    pptr->igmphdr.type < info->igmp_type[0] || | ||||
| 			    pptr->igmphdr.type > info->igmp_type[1])) | ||||
| 			return false; | ||||
| 	} | ||||
| 	return true; | ||||
| } | ||||
|  | @ -101,6 +124,21 @@ static int ebt_ip_mt_check(const struct xt_mtchk_param *par) | |||
| 		return -EINVAL; | ||||
| 	if (info->bitmask & EBT_IP_SPORT && info->sport[0] > info->sport[1]) | ||||
| 		return -EINVAL; | ||||
| 	if (info->bitmask & EBT_IP_ICMP) { | ||||
| 		if ((info->invflags & EBT_IP_PROTO) || | ||||
| 		    info->protocol != IPPROTO_ICMP) | ||||
| 			return -EINVAL; | ||||
| 		if (info->icmp_type[0] > info->icmp_type[1] || | ||||
| 		    info->icmp_code[0] > info->icmp_code[1]) | ||||
| 			return -EINVAL; | ||||
| 	} | ||||
| 	if (info->bitmask & EBT_IP_IGMP) { | ||||
| 		if ((info->invflags & EBT_IP_PROTO) || | ||||
| 		    info->protocol != IPPROTO_IGMP) | ||||
| 			return -EINVAL; | ||||
| 		if (info->igmp_type[0] > info->igmp_type[1]) | ||||
| 			return -EINVAL; | ||||
| 	} | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
|  |  | |||
|  | @ -153,8 +153,6 @@ ebt_stp_mt(const struct sk_buff *skb, struct xt_action_param *par) | |||
| static int ebt_stp_mt_check(const struct xt_mtchk_param *par) | ||||
| { | ||||
| 	const struct ebt_stp_info *info = par->matchinfo; | ||||
| 	const u8 bridge_ula[6] = {0x01, 0x80, 0xc2, 0x00, 0x00, 0x00}; | ||||
| 	const u8 msk[6] = {0xff, 0xff, 0xff, 0xff, 0xff, 0xff}; | ||||
| 	const struct ebt_entry *e = par->entryinfo; | ||||
| 
 | ||||
| 	if (info->bitmask & ~EBT_STP_MASK || info->invflags & ~EBT_STP_MASK || | ||||
|  | @ -162,8 +160,8 @@ static int ebt_stp_mt_check(const struct xt_mtchk_param *par) | |||
| 		return -EINVAL; | ||||
| 	/* Make sure the match only receives stp frames */ | ||||
| 	if (!par->nft_compat && | ||||
| 	    (!ether_addr_equal(e->destmac, bridge_ula) || | ||||
| 	     !ether_addr_equal(e->destmsk, msk) || | ||||
| 	    (!ether_addr_equal(e->destmac, eth_stp_addr) || | ||||
| 	     !is_broadcast_ether_addr(e->destmsk) || | ||||
| 	     !(e->bitmask & EBT_DESTMAC))) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
|  |  | |||
|  | @ -223,9 +223,7 @@ unsigned int ebt_do_table(struct sk_buff *skb, | |||
| 			return NF_DROP; | ||||
| 		} | ||||
| 
 | ||||
| 		/* increase counter */ | ||||
| 		(*(counter_base + i)).pcnt++; | ||||
| 		(*(counter_base + i)).bcnt += skb->len; | ||||
| 		ADD_COUNTER(*(counter_base + i), 1, skb->len); | ||||
| 
 | ||||
| 		/* these should only watch: not modify, nor tell us
 | ||||
| 		 * what to do with the packet | ||||
|  | @ -358,12 +356,12 @@ ebt_check_match(struct ebt_entry_match *m, struct xt_mtchk_param *par, | |||
| 	    left - sizeof(struct ebt_entry_match) < m->match_size) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	match = xt_find_match(NFPROTO_BRIDGE, m->u.name, 0); | ||||
| 	match = xt_find_match(NFPROTO_BRIDGE, m->u.name, m->u.revision); | ||||
| 	if (IS_ERR(match) || match->family != NFPROTO_BRIDGE) { | ||||
| 		if (!IS_ERR(match)) | ||||
| 			module_put(match->me); | ||||
| 		request_module("ebt_%s", m->u.name); | ||||
| 		match = xt_find_match(NFPROTO_BRIDGE, m->u.name, 0); | ||||
| 		match = xt_find_match(NFPROTO_BRIDGE, m->u.name, m->u.revision); | ||||
| 	} | ||||
| 	if (IS_ERR(match)) | ||||
| 		return PTR_ERR(match); | ||||
|  | @ -968,10 +966,9 @@ static void get_counters(const struct ebt_counter *oldcounters, | |||
| 		if (cpu == 0) | ||||
| 			continue; | ||||
| 		counter_base = COUNTER_BASE(oldcounters, nentries, cpu); | ||||
| 		for (i = 0; i < nentries; i++) { | ||||
| 			counters[i].pcnt += counter_base[i].pcnt; | ||||
| 			counters[i].bcnt += counter_base[i].bcnt; | ||||
| 		} | ||||
| 		for (i = 0; i < nentries; i++) | ||||
| 			ADD_COUNTER(counters[i], counter_base[i].pcnt, | ||||
| 				    counter_base[i].bcnt); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
|  | @ -1324,10 +1321,8 @@ static int do_update_counters(struct net *net, const char *name, | |||
| 	write_lock_bh(&t->lock); | ||||
| 
 | ||||
| 	/* we add to the counters of the first cpu */ | ||||
| 	for (i = 0; i < num_counters; i++) { | ||||
| 		t->private->counters[i].pcnt += tmp[i].pcnt; | ||||
| 		t->private->counters[i].bcnt += tmp[i].bcnt; | ||||
| 	} | ||||
| 	for (i = 0; i < num_counters; i++) | ||||
| 		ADD_COUNTER(t->private->counters[i], tmp[i].pcnt, tmp[i].bcnt); | ||||
| 
 | ||||
| 	write_unlock_bh(&t->lock); | ||||
| 	ret = 0; | ||||
|  | @ -1355,16 +1350,17 @@ static int update_counters(struct net *net, const void __user *user, | |||
| 
 | ||||
| static inline int ebt_obj_to_user(char __user *um, const char *_name, | ||||
| 				  const char *data, int entrysize, | ||||
| 				  int usersize, int datasize) | ||||
| 				  int usersize, int datasize, u8 revision) | ||||
| { | ||||
| 	char name[EBT_FUNCTION_MAXNAMELEN] = {0}; | ||||
| 	char name[EBT_EXTENSION_MAXNAMELEN] = {0}; | ||||
| 
 | ||||
| 	/* ebtables expects 32 bytes long names but xt_match names are 29 bytes
 | ||||
| 	/* ebtables expects 31 bytes long names but xt_match names are 29 bytes
 | ||||
| 	 * long. Copy 29 bytes and fill remaining bytes with zeroes. | ||||
| 	 */ | ||||
| 	strlcpy(name, _name, sizeof(name)); | ||||
| 	if (copy_to_user(um, name, EBT_FUNCTION_MAXNAMELEN) || | ||||
| 	    put_user(datasize, (int __user *)(um + EBT_FUNCTION_MAXNAMELEN)) || | ||||
| 	if (copy_to_user(um, name, EBT_EXTENSION_MAXNAMELEN) || | ||||
| 	    put_user(revision, (u8 __user *)(um + EBT_EXTENSION_MAXNAMELEN)) || | ||||
| 	    put_user(datasize, (int __user *)(um + EBT_EXTENSION_MAXNAMELEN + 1)) || | ||||
| 	    xt_data_to_user(um + entrysize, data, usersize, datasize, | ||||
| 			    XT_ALIGN(datasize))) | ||||
| 		return -EFAULT; | ||||
|  | @ -1377,7 +1373,8 @@ static inline int ebt_match_to_user(const struct ebt_entry_match *m, | |||
| { | ||||
| 	return ebt_obj_to_user(ubase + ((char *)m - base), | ||||
| 			       m->u.match->name, m->data, sizeof(*m), | ||||
| 			       m->u.match->usersize, m->match_size); | ||||
| 			       m->u.match->usersize, m->match_size, | ||||
| 			       m->u.match->revision); | ||||
| } | ||||
| 
 | ||||
| static inline int ebt_watcher_to_user(const struct ebt_entry_watcher *w, | ||||
|  | @ -1385,7 +1382,8 @@ static inline int ebt_watcher_to_user(const struct ebt_entry_watcher *w, | |||
| { | ||||
| 	return ebt_obj_to_user(ubase + ((char *)w - base), | ||||
| 			       w->u.watcher->name, w->data, sizeof(*w), | ||||
| 			       w->u.watcher->usersize, w->watcher_size); | ||||
| 			       w->u.watcher->usersize, w->watcher_size, | ||||
| 			       w->u.watcher->revision); | ||||
| } | ||||
| 
 | ||||
| static inline int ebt_entry_to_user(struct ebt_entry *e, const char *base, | ||||
|  | @ -1416,7 +1414,8 @@ static inline int ebt_entry_to_user(struct ebt_entry *e, const char *base, | |||
| 	if (ret != 0) | ||||
| 		return ret; | ||||
| 	ret = ebt_obj_to_user(hlp, t->u.target->name, t->data, sizeof(*t), | ||||
| 			      t->u.target->usersize, t->target_size); | ||||
| 			      t->u.target->usersize, t->target_size, | ||||
| 			      t->u.target->revision); | ||||
| 	if (ret != 0) | ||||
| 		return ret; | ||||
| 
 | ||||
|  | @ -1604,7 +1603,10 @@ struct compat_ebt_replace { | |||
| /* struct ebt_entry_match, _target and _watcher have same layout */ | ||||
| struct compat_ebt_entry_mwt { | ||||
| 	union { | ||||
| 		char name[EBT_FUNCTION_MAXNAMELEN]; | ||||
| 		struct { | ||||
| 			char name[EBT_EXTENSION_MAXNAMELEN]; | ||||
| 			u8 revision; | ||||
| 		}; | ||||
| 		compat_uptr_t ptr; | ||||
| 	} u; | ||||
| 	compat_uint_t match_size; | ||||
|  | @ -1644,8 +1646,9 @@ static int compat_match_to_user(struct ebt_entry_match *m, void __user **dstptr, | |||
| 	if (WARN_ON(off >= m->match_size)) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	if (copy_to_user(cm->u.name, match->name, | ||||
| 	    strlen(match->name) + 1) || put_user(msize, &cm->match_size)) | ||||
| 	if (copy_to_user(cm->u.name, match->name, strlen(match->name) + 1) || | ||||
| 	    put_user(match->revision, &cm->u.revision) || | ||||
| 	    put_user(msize, &cm->match_size)) | ||||
| 		return -EFAULT; | ||||
| 
 | ||||
| 	if (match->compat_to_user) { | ||||
|  | @ -1675,8 +1678,9 @@ static int compat_target_to_user(struct ebt_entry_target *t, | |||
| 	if (WARN_ON(off >= t->target_size)) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	if (copy_to_user(cm->u.name, target->name, | ||||
| 	    strlen(target->name) + 1) || put_user(tsize, &cm->match_size)) | ||||
| 	if (copy_to_user(cm->u.name, target->name, strlen(target->name) + 1) || | ||||
| 	    put_user(target->revision, &cm->u.revision) || | ||||
| 	    put_user(tsize, &cm->match_size)) | ||||
| 		return -EFAULT; | ||||
| 
 | ||||
| 	if (target->compat_to_user) { | ||||
|  | @ -1821,10 +1825,14 @@ static int compat_table_info(const struct ebt_table_info *info, | |||
| { | ||||
| 	unsigned int size = info->entries_size; | ||||
| 	const void *entries = info->entries; | ||||
| 	int ret; | ||||
| 
 | ||||
| 	newinfo->entries_size = size; | ||||
| 
 | ||||
| 	xt_compat_init_offsets(NFPROTO_BRIDGE, info->nentries); | ||||
| 	ret = xt_compat_init_offsets(NFPROTO_BRIDGE, info->nentries); | ||||
| 	if (ret) | ||||
| 		return ret; | ||||
| 
 | ||||
| 	return EBT_ENTRY_ITERATE(entries, size, compat_calc_entry, info, | ||||
| 							entries, newinfo); | ||||
| } | ||||
|  | @ -1938,7 +1946,7 @@ static int compat_mtw_from_user(struct compat_ebt_entry_mwt *mwt, | |||
| 				struct ebt_entries_buf_state *state, | ||||
| 				const unsigned char *base) | ||||
| { | ||||
| 	char name[EBT_FUNCTION_MAXNAMELEN]; | ||||
| 	char name[EBT_EXTENSION_MAXNAMELEN]; | ||||
| 	struct xt_match *match; | ||||
| 	struct xt_target *wt; | ||||
| 	void *dst = NULL; | ||||
|  | @ -1952,7 +1960,8 @@ static int compat_mtw_from_user(struct compat_ebt_entry_mwt *mwt, | |||
| 
 | ||||
| 	switch (compat_mwt) { | ||||
| 	case EBT_COMPAT_MATCH: | ||||
| 		match = xt_request_find_match(NFPROTO_BRIDGE, name, 0); | ||||
| 		match = xt_request_find_match(NFPROTO_BRIDGE, name, | ||||
| 					      mwt->u.revision); | ||||
| 		if (IS_ERR(match)) | ||||
| 			return PTR_ERR(match); | ||||
| 
 | ||||
|  | @ -1971,7 +1980,8 @@ static int compat_mtw_from_user(struct compat_ebt_entry_mwt *mwt, | |||
| 		break; | ||||
| 	case EBT_COMPAT_WATCHER: /* fallthrough */ | ||||
| 	case EBT_COMPAT_TARGET: | ||||
| 		wt = xt_request_find_target(NFPROTO_BRIDGE, name, 0); | ||||
| 		wt = xt_request_find_target(NFPROTO_BRIDGE, name, | ||||
| 					    mwt->u.revision); | ||||
| 		if (IS_ERR(wt)) | ||||
| 			return PTR_ERR(wt); | ||||
| 		off = xt_compat_target_offset(wt); | ||||
|  | @ -2268,7 +2278,9 @@ static int compat_do_replace(struct net *net, void __user *user, | |||
| 
 | ||||
| 	xt_compat_lock(NFPROTO_BRIDGE); | ||||
| 
 | ||||
| 	xt_compat_init_offsets(NFPROTO_BRIDGE, tmp.nentries); | ||||
| 	ret = xt_compat_init_offsets(NFPROTO_BRIDGE, tmp.nentries); | ||||
| 	if (ret < 0) | ||||
| 		goto out_unlock; | ||||
| 	ret = compat_copy_entries(entries_tmp, tmp.entries_size, &state); | ||||
| 	if (ret < 0) | ||||
| 		goto out_unlock; | ||||
|  |  | |||
|  | @ -1,79 +0,0 @@ | |||
| /*
 | ||||
|  * Copyright (c) 2008 Patrick McHardy <kaber@trash.net> | ||||
|  * Copyright (c) 2013 Pablo Neira Ayuso <pablo@netfilter.org> | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 as | ||||
|  * published by the Free Software Foundation. | ||||
|  * | ||||
|  * Development of this code funded by Astaro AG (http://www.astaro.com/)
 | ||||
|  */ | ||||
| 
 | ||||
| #include <linux/init.h> | ||||
| #include <linux/module.h> | ||||
| #include <linux/netfilter_bridge.h> | ||||
| #include <net/netfilter/nf_tables.h> | ||||
| #include <linux/ip.h> | ||||
| #include <linux/ipv6.h> | ||||
| #include <net/netfilter/nf_tables_ipv4.h> | ||||
| #include <net/netfilter/nf_tables_ipv6.h> | ||||
| 
 | ||||
| static unsigned int | ||||
| nft_do_chain_bridge(void *priv, | ||||
| 		    struct sk_buff *skb, | ||||
| 		    const struct nf_hook_state *state) | ||||
| { | ||||
| 	struct nft_pktinfo pkt; | ||||
| 
 | ||||
| 	nft_set_pktinfo(&pkt, skb, state); | ||||
| 
 | ||||
| 	switch (eth_hdr(skb)->h_proto) { | ||||
| 	case htons(ETH_P_IP): | ||||
| 		nft_set_pktinfo_ipv4_validate(&pkt, skb); | ||||
| 		break; | ||||
| 	case htons(ETH_P_IPV6): | ||||
| 		nft_set_pktinfo_ipv6_validate(&pkt, skb); | ||||
| 		break; | ||||
| 	default: | ||||
| 		nft_set_pktinfo_unspec(&pkt, skb); | ||||
| 		break; | ||||
| 	} | ||||
| 
 | ||||
| 	return nft_do_chain(&pkt, priv); | ||||
| } | ||||
| 
 | ||||
| static const struct nf_chain_type filter_bridge = { | ||||
| 	.name		= "filter", | ||||
| 	.type		= NFT_CHAIN_T_DEFAULT, | ||||
| 	.family		= NFPROTO_BRIDGE, | ||||
| 	.owner		= THIS_MODULE, | ||||
| 	.hook_mask	= (1 << NF_BR_PRE_ROUTING) | | ||||
| 			  (1 << NF_BR_LOCAL_IN) | | ||||
| 			  (1 << NF_BR_FORWARD) | | ||||
| 			  (1 << NF_BR_LOCAL_OUT) | | ||||
| 			  (1 << NF_BR_POST_ROUTING), | ||||
| 	.hooks		= { | ||||
| 		[NF_BR_PRE_ROUTING]	= nft_do_chain_bridge, | ||||
| 		[NF_BR_LOCAL_IN]	= nft_do_chain_bridge, | ||||
| 		[NF_BR_FORWARD]		= nft_do_chain_bridge, | ||||
| 		[NF_BR_LOCAL_OUT]	= nft_do_chain_bridge, | ||||
| 		[NF_BR_POST_ROUTING]	= nft_do_chain_bridge, | ||||
| 	}, | ||||
| }; | ||||
| 
 | ||||
| static int __init nf_tables_bridge_init(void) | ||||
| { | ||||
| 	return nft_register_chain_type(&filter_bridge); | ||||
| } | ||||
| 
 | ||||
| static void __exit nf_tables_bridge_exit(void) | ||||
| { | ||||
| 	nft_unregister_chain_type(&filter_bridge); | ||||
| } | ||||
| 
 | ||||
| module_init(nf_tables_bridge_init); | ||||
| module_exit(nf_tables_bridge_exit); | ||||
| 
 | ||||
| MODULE_LICENSE("GPL"); | ||||
| MODULE_AUTHOR("Patrick McHardy <kaber@trash.net>"); | ||||
| MODULE_ALIAS_NFT_CHAIN(AF_BRIDGE, "filter"); | ||||
|  | @ -34,7 +34,7 @@ config NF_SOCKET_IPV4 | |||
| if NF_TABLES | ||||
| 
 | ||||
| config NF_TABLES_IPV4 | ||||
| 	tristate "IPv4 nf_tables support" | ||||
| 	bool "IPv4 nf_tables support" | ||||
| 	help | ||||
| 	  This option enables the IPv4 support for nf_tables. | ||||
| 
 | ||||
|  | @ -71,7 +71,7 @@ config NFT_FIB_IPV4 | |||
| endif # NF_TABLES_IPV4 | ||||
| 
 | ||||
| config NF_TABLES_ARP | ||||
| 	tristate "ARP nf_tables support" | ||||
| 	bool "ARP nf_tables support" | ||||
| 	select NETFILTER_FAMILY_ARP | ||||
| 	help | ||||
| 	  This option enables the ARP support for nf_tables. | ||||
|  |  | |||
|  | @ -39,7 +39,6 @@ obj-$(CONFIG_NF_NAT_MASQUERADE_IPV4) += nf_nat_masquerade_ipv4.o | |||
| # NAT protocols (nf_nat)
 | ||||
| obj-$(CONFIG_NF_NAT_PROTO_GRE) += nf_nat_proto_gre.o | ||||
| 
 | ||||
| obj-$(CONFIG_NF_TABLES_IPV4) += nf_tables_ipv4.o | ||||
| obj-$(CONFIG_NFT_CHAIN_ROUTE_IPV4) += nft_chain_route_ipv4.o | ||||
| obj-$(CONFIG_NFT_CHAIN_NAT_IPV4) += nft_chain_nat_ipv4.o | ||||
| obj-$(CONFIG_NFT_REJECT_IPV4) += nft_reject_ipv4.o | ||||
|  | @ -47,7 +46,6 @@ obj-$(CONFIG_NFT_FIB_IPV4) += nft_fib_ipv4.o | |||
| obj-$(CONFIG_NFT_MASQ_IPV4) += nft_masq_ipv4.o | ||||
| obj-$(CONFIG_NFT_REDIR_IPV4) += nft_redir_ipv4.o | ||||
| obj-$(CONFIG_NFT_DUP_IPV4) += nft_dup_ipv4.o | ||||
| obj-$(CONFIG_NF_TABLES_ARP) += nf_tables_arp.o | ||||
| 
 | ||||
| # flow table support
 | ||||
| obj-$(CONFIG_NF_FLOW_TABLE_IPV4) += nf_flow_table_ipv4.o | ||||
|  |  | |||
|  | @ -334,11 +334,6 @@ static int mark_source_chains(const struct xt_table_info *newinfo, | |||
| 			     t->verdict < 0) || visited) { | ||||
| 				unsigned int oldpos, size; | ||||
| 
 | ||||
| 				if ((strcmp(t->target.u.user.name, | ||||
| 					    XT_STANDARD_TARGET) == 0) && | ||||
| 				    t->verdict < -NF_MAX_VERDICT - 1) | ||||
| 					return 0; | ||||
| 
 | ||||
| 				/* Return: backtrack through the last
 | ||||
| 				 * big jump. | ||||
| 				 */ | ||||
|  | @ -560,16 +555,9 @@ static int translate_table(struct xt_table_info *newinfo, void *entry0, | |||
| 	if (i != repl->num_entries) | ||||
| 		goto out_free; | ||||
| 
 | ||||
| 	/* Check hooks all assigned */ | ||||
| 	for (i = 0; i < NF_ARP_NUMHOOKS; i++) { | ||||
| 		/* Only hooks which are valid */ | ||||
| 		if (!(repl->valid_hooks & (1 << i))) | ||||
| 			continue; | ||||
| 		if (newinfo->hook_entry[i] == 0xFFFFFFFF) | ||||
| 			goto out_free; | ||||
| 		if (newinfo->underflow[i] == 0xFFFFFFFF) | ||||
| 			goto out_free; | ||||
| 	} | ||||
| 	ret = xt_check_table_hooks(newinfo, repl->valid_hooks); | ||||
| 	if (ret) | ||||
| 		goto out_free; | ||||
| 
 | ||||
| 	if (!mark_source_chains(newinfo, repl->valid_hooks, entry0, offsets)) { | ||||
| 		ret = -ELOOP; | ||||
|  | @ -781,7 +769,9 @@ static int compat_table_info(const struct xt_table_info *info, | |||
| 	memcpy(newinfo, info, offsetof(struct xt_table_info, entries)); | ||||
| 	newinfo->initial_entries = 0; | ||||
| 	loc_cpu_entry = info->entries; | ||||
| 	xt_compat_init_offsets(NFPROTO_ARP, info->number); | ||||
| 	ret = xt_compat_init_offsets(NFPROTO_ARP, info->number); | ||||
| 	if (ret) | ||||
| 		return ret; | ||||
| 	xt_entry_foreach(iter, loc_cpu_entry, info->size) { | ||||
| 		ret = compat_calc_entry(iter, info, loc_cpu_entry, newinfo); | ||||
| 		if (ret != 0) | ||||
|  | @ -895,7 +885,7 @@ static int __do_replace(struct net *net, const char *name, | |||
| 	struct arpt_entry *iter; | ||||
| 
 | ||||
| 	ret = 0; | ||||
| 	counters = vzalloc(num_counters * sizeof(struct xt_counters)); | ||||
| 	counters = xt_counters_alloc(num_counters); | ||||
| 	if (!counters) { | ||||
| 		ret = -ENOMEM; | ||||
| 		goto out; | ||||
|  | @ -925,6 +915,8 @@ static int __do_replace(struct net *net, const char *name, | |||
| 	    (newinfo->number <= oldinfo->initial_entries)) | ||||
| 		module_put(t->me); | ||||
| 
 | ||||
| 	xt_table_unlock(t); | ||||
| 
 | ||||
| 	get_old_counters(oldinfo, counters); | ||||
| 
 | ||||
| 	/* Decrease module usage counts and free resource */ | ||||
|  | @ -939,7 +931,6 @@ static int __do_replace(struct net *net, const char *name, | |||
| 		net_warn_ratelimited("arptables: counters copy to user failed while replacing table\n"); | ||||
| 	} | ||||
| 	vfree(counters); | ||||
| 	xt_table_unlock(t); | ||||
| 	return ret; | ||||
| 
 | ||||
|  put_module: | ||||
|  | @ -1167,7 +1158,7 @@ static int translate_compat_table(struct xt_table_info **pinfo, | |||
| 	struct compat_arpt_entry *iter0; | ||||
| 	struct arpt_replace repl; | ||||
| 	unsigned int size; | ||||
| 	int ret = 0; | ||||
| 	int ret; | ||||
| 
 | ||||
| 	info = *pinfo; | ||||
| 	entry0 = *pentry0; | ||||
|  | @ -1176,7 +1167,9 @@ static int translate_compat_table(struct xt_table_info **pinfo, | |||
| 
 | ||||
| 	j = 0; | ||||
| 	xt_compat_lock(NFPROTO_ARP); | ||||
| 	xt_compat_init_offsets(NFPROTO_ARP, compatr->num_entries); | ||||
| 	ret = xt_compat_init_offsets(NFPROTO_ARP, compatr->num_entries); | ||||
| 	if (ret) | ||||
| 		goto out_unlock; | ||||
| 	/* Walk through entries, checking offsets. */ | ||||
| 	xt_entry_foreach(iter0, entry0, compatr->size) { | ||||
| 		ret = check_compat_entry_size_and_hooks(iter0, info, &size, | ||||
|  |  | |||
|  | @ -402,11 +402,6 @@ mark_source_chains(const struct xt_table_info *newinfo, | |||
| 			     t->verdict < 0) || visited) { | ||||
| 				unsigned int oldpos, size; | ||||
| 
 | ||||
| 				if ((strcmp(t->target.u.user.name, | ||||
| 					    XT_STANDARD_TARGET) == 0) && | ||||
| 				    t->verdict < -NF_MAX_VERDICT - 1) | ||||
| 					return 0; | ||||
| 
 | ||||
| 				/* Return: backtrack through the last
 | ||||
| 				   big jump. */ | ||||
| 				do { | ||||
|  | @ -707,16 +702,9 @@ translate_table(struct net *net, struct xt_table_info *newinfo, void *entry0, | |||
| 	if (i != repl->num_entries) | ||||
| 		goto out_free; | ||||
| 
 | ||||
| 	/* Check hooks all assigned */ | ||||
| 	for (i = 0; i < NF_INET_NUMHOOKS; i++) { | ||||
| 		/* Only hooks which are valid */ | ||||
| 		if (!(repl->valid_hooks & (1 << i))) | ||||
| 			continue; | ||||
| 		if (newinfo->hook_entry[i] == 0xFFFFFFFF) | ||||
| 			goto out_free; | ||||
| 		if (newinfo->underflow[i] == 0xFFFFFFFF) | ||||
| 			goto out_free; | ||||
| 	} | ||||
| 	ret = xt_check_table_hooks(newinfo, repl->valid_hooks); | ||||
| 	if (ret) | ||||
| 		goto out_free; | ||||
| 
 | ||||
| 	if (!mark_source_chains(newinfo, repl->valid_hooks, entry0, offsets)) { | ||||
| 		ret = -ELOOP; | ||||
|  | @ -945,7 +933,9 @@ static int compat_table_info(const struct xt_table_info *info, | |||
| 	memcpy(newinfo, info, offsetof(struct xt_table_info, entries)); | ||||
| 	newinfo->initial_entries = 0; | ||||
| 	loc_cpu_entry = info->entries; | ||||
| 	xt_compat_init_offsets(AF_INET, info->number); | ||||
| 	ret = xt_compat_init_offsets(AF_INET, info->number); | ||||
| 	if (ret) | ||||
| 		return ret; | ||||
| 	xt_entry_foreach(iter, loc_cpu_entry, info->size) { | ||||
| 		ret = compat_calc_entry(iter, info, loc_cpu_entry, newinfo); | ||||
| 		if (ret != 0) | ||||
|  | @ -1057,7 +1047,7 @@ __do_replace(struct net *net, const char *name, unsigned int valid_hooks, | |||
| 	struct ipt_entry *iter; | ||||
| 
 | ||||
| 	ret = 0; | ||||
| 	counters = vzalloc(num_counters * sizeof(struct xt_counters)); | ||||
| 	counters = xt_counters_alloc(num_counters); | ||||
| 	if (!counters) { | ||||
| 		ret = -ENOMEM; | ||||
| 		goto out; | ||||
|  | @ -1087,6 +1077,8 @@ __do_replace(struct net *net, const char *name, unsigned int valid_hooks, | |||
| 	    (newinfo->number <= oldinfo->initial_entries)) | ||||
| 		module_put(t->me); | ||||
| 
 | ||||
| 	xt_table_unlock(t); | ||||
| 
 | ||||
| 	get_old_counters(oldinfo, counters); | ||||
| 
 | ||||
| 	/* Decrease module usage counts and free resource */ | ||||
|  | @ -1100,7 +1092,6 @@ __do_replace(struct net *net, const char *name, unsigned int valid_hooks, | |||
| 		net_warn_ratelimited("iptables: counters copy to user failed while replacing table\n"); | ||||
| 	} | ||||
| 	vfree(counters); | ||||
| 	xt_table_unlock(t); | ||||
| 	return ret; | ||||
| 
 | ||||
|  put_module: | ||||
|  | @ -1418,7 +1409,9 @@ translate_compat_table(struct net *net, | |||
| 
 | ||||
| 	j = 0; | ||||
| 	xt_compat_lock(AF_INET); | ||||
| 	xt_compat_init_offsets(AF_INET, compatr->num_entries); | ||||
| 	ret = xt_compat_init_offsets(AF_INET, compatr->num_entries); | ||||
| 	if (ret) | ||||
| 		goto out_unlock; | ||||
| 	/* Walk through entries, checking offsets. */ | ||||
| 	xt_entry_foreach(iter0, entry0, compatr->size) { | ||||
| 		ret = check_compat_entry_size_and_hooks(iter0, info, &size, | ||||
|  |  | |||
|  | @ -16,6 +16,7 @@ | |||
| #include <net/netfilter/nf_conntrack.h> | ||||
| #include <net/netfilter/nf_conntrack_seqadj.h> | ||||
| #include <net/netfilter/nf_conntrack_synproxy.h> | ||||
| #include <net/netfilter/nf_conntrack_ecache.h> | ||||
| 
 | ||||
| static struct iphdr * | ||||
| synproxy_build_ip(struct net *net, struct sk_buff *skb, __be32 saddr, | ||||
|  | @ -384,6 +385,8 @@ static unsigned int ipv4_synproxy_hook(void *priv, | |||
| 		synproxy->isn = ntohl(th->ack_seq); | ||||
| 		if (opts.options & XT_SYNPROXY_OPT_TIMESTAMP) | ||||
| 			synproxy->its = opts.tsecr; | ||||
| 
 | ||||
| 		nf_conntrack_event_cache(IPCT_SYNPROXY, ct); | ||||
| 		break; | ||||
| 	case TCP_CONNTRACK_SYN_RECV: | ||||
| 		if (!th->syn || !th->ack) | ||||
|  | @ -392,8 +395,10 @@ static unsigned int ipv4_synproxy_hook(void *priv, | |||
| 		if (!synproxy_parse_options(skb, thoff, th, &opts)) | ||||
| 			return NF_DROP; | ||||
| 
 | ||||
| 		if (opts.options & XT_SYNPROXY_OPT_TIMESTAMP) | ||||
| 		if (opts.options & XT_SYNPROXY_OPT_TIMESTAMP) { | ||||
| 			synproxy->tsoff = opts.tsval - synproxy->its; | ||||
| 			nf_conntrack_event_cache(IPCT_SYNPROXY, ct); | ||||
| 		} | ||||
| 
 | ||||
| 		opts.options &= ~(XT_SYNPROXY_OPT_MSS | | ||||
| 				  XT_SYNPROXY_OPT_WSCALE | | ||||
|  | @ -403,6 +408,7 @@ static unsigned int ipv4_synproxy_hook(void *priv, | |||
| 		synproxy_send_server_ack(net, state, skb, th, &opts); | ||||
| 
 | ||||
| 		nf_ct_seqadj_init(ct, ctinfo, synproxy->isn - ntohl(th->seq)); | ||||
| 		nf_conntrack_event_cache(IPCT_SEQADJ, ct); | ||||
| 
 | ||||
| 		swap(opts.tsval, opts.tsecr); | ||||
| 		synproxy_send_client_ack(net, skb, th, &opts); | ||||
|  |  | |||
|  | @ -47,7 +47,7 @@ static bool ah_mt(const struct sk_buff *skb, struct xt_action_param *par) | |||
| 		 */ | ||||
| 		pr_debug("Dropping evil AH tinygram.\n"); | ||||
| 		par->hotdrop = true; | ||||
| 		return 0; | ||||
| 		return false; | ||||
| 	} | ||||
| 
 | ||||
| 	return spi_match(ahinfo->spis[0], ahinfo->spis[1], | ||||
|  |  | |||
|  | @ -1,58 +0,0 @@ | |||
| /*
 | ||||
|  * Copyright (c) 2008-2010 Patrick McHardy <kaber@trash.net> | ||||
|  * Copyright (c) 2013 Pablo Neira Ayuso <pablo@netfilter.org> | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 as | ||||
|  * published by the Free Software Foundation. | ||||
|  * | ||||
|  * Development of this code funded by Astaro AG (http://www.astaro.com/)
 | ||||
|  */ | ||||
| 
 | ||||
| #include <linux/module.h> | ||||
| #include <linux/init.h> | ||||
| #include <linux/netfilter_arp.h> | ||||
| #include <net/netfilter/nf_tables.h> | ||||
| 
 | ||||
| static unsigned int | ||||
| nft_do_chain_arp(void *priv, | ||||
| 		  struct sk_buff *skb, | ||||
| 		  const struct nf_hook_state *state) | ||||
| { | ||||
| 	struct nft_pktinfo pkt; | ||||
| 
 | ||||
| 	nft_set_pktinfo(&pkt, skb, state); | ||||
| 	nft_set_pktinfo_unspec(&pkt, skb); | ||||
| 
 | ||||
| 	return nft_do_chain(&pkt, priv); | ||||
| } | ||||
| 
 | ||||
| static const struct nf_chain_type filter_arp = { | ||||
| 	.name		= "filter", | ||||
| 	.type		= NFT_CHAIN_T_DEFAULT, | ||||
| 	.family		= NFPROTO_ARP, | ||||
| 	.owner		= THIS_MODULE, | ||||
| 	.hook_mask	= (1 << NF_ARP_IN) | | ||||
| 			  (1 << NF_ARP_OUT), | ||||
| 	.hooks		= { | ||||
| 		[NF_ARP_IN]		= nft_do_chain_arp, | ||||
| 		[NF_ARP_OUT]		= nft_do_chain_arp, | ||||
| 	}, | ||||
| }; | ||||
| 
 | ||||
| static int __init nf_tables_arp_init(void) | ||||
| { | ||||
| 	return nft_register_chain_type(&filter_arp); | ||||
| } | ||||
| 
 | ||||
| static void __exit nf_tables_arp_exit(void) | ||||
| { | ||||
| 	nft_unregister_chain_type(&filter_arp); | ||||
| } | ||||
| 
 | ||||
| module_init(nf_tables_arp_init); | ||||
| module_exit(nf_tables_arp_exit); | ||||
| 
 | ||||
| MODULE_LICENSE("GPL"); | ||||
| MODULE_AUTHOR("Patrick McHardy <kaber@trash.net>"); | ||||
| MODULE_ALIAS_NFT_CHAIN(3, "filter"); /* NFPROTO_ARP */ | ||||
|  | @ -1,67 +0,0 @@ | |||
| /*
 | ||||
|  * Copyright (c) 2008 Patrick McHardy <kaber@trash.net> | ||||
|  * Copyright (c) 2012-2013 Pablo Neira Ayuso <pablo@netfilter.org> | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 as | ||||
|  * published by the Free Software Foundation. | ||||
|  * | ||||
|  * Development of this code funded by Astaro AG (http://www.astaro.com/)
 | ||||
|  */ | ||||
| 
 | ||||
| #include <linux/init.h> | ||||
| #include <linux/module.h> | ||||
| #include <linux/ip.h> | ||||
| #include <linux/netfilter_ipv4.h> | ||||
| #include <net/netfilter/nf_tables.h> | ||||
| #include <net/net_namespace.h> | ||||
| #include <net/ip.h> | ||||
| #include <net/netfilter/nf_tables_ipv4.h> | ||||
| 
 | ||||
| static unsigned int nft_do_chain_ipv4(void *priv, | ||||
| 				      struct sk_buff *skb, | ||||
| 				      const struct nf_hook_state *state) | ||||
| { | ||||
| 	struct nft_pktinfo pkt; | ||||
| 
 | ||||
| 	nft_set_pktinfo(&pkt, skb, state); | ||||
| 	nft_set_pktinfo_ipv4(&pkt, skb); | ||||
| 
 | ||||
| 	return nft_do_chain(&pkt, priv); | ||||
| } | ||||
| 
 | ||||
| static const struct nf_chain_type filter_ipv4 = { | ||||
| 	.name		= "filter", | ||||
| 	.type		= NFT_CHAIN_T_DEFAULT, | ||||
| 	.family		= NFPROTO_IPV4, | ||||
| 	.owner		= THIS_MODULE, | ||||
| 	.hook_mask	= (1 << NF_INET_LOCAL_IN) | | ||||
| 			  (1 << NF_INET_LOCAL_OUT) | | ||||
| 			  (1 << NF_INET_FORWARD) | | ||||
| 			  (1 << NF_INET_PRE_ROUTING) | | ||||
| 			  (1 << NF_INET_POST_ROUTING), | ||||
| 	.hooks		= { | ||||
| 		[NF_INET_LOCAL_IN]	= nft_do_chain_ipv4, | ||||
| 		[NF_INET_LOCAL_OUT]	= nft_do_chain_ipv4, | ||||
| 		[NF_INET_FORWARD]	= nft_do_chain_ipv4, | ||||
| 		[NF_INET_PRE_ROUTING]	= nft_do_chain_ipv4, | ||||
| 		[NF_INET_POST_ROUTING]	= nft_do_chain_ipv4, | ||||
| 	}, | ||||
| }; | ||||
| 
 | ||||
| static int __init nf_tables_ipv4_init(void) | ||||
| { | ||||
| 	return nft_register_chain_type(&filter_ipv4); | ||||
| } | ||||
| 
 | ||||
| static void __exit nf_tables_ipv4_exit(void) | ||||
| { | ||||
| 	nft_unregister_chain_type(&filter_ipv4); | ||||
| } | ||||
| 
 | ||||
| module_init(nf_tables_ipv4_init); | ||||
| module_exit(nf_tables_ipv4_exit); | ||||
| 
 | ||||
| MODULE_LICENSE("GPL"); | ||||
| MODULE_AUTHOR("Patrick McHardy <kaber@trash.net>"); | ||||
| MODULE_ALIAS_NFT_CHAIN(AF_INET, "filter"); | ||||
|  | @ -67,7 +67,17 @@ static unsigned int nft_nat_ipv4_local_fn(void *priv, | |||
| 	return nf_nat_ipv4_local_fn(priv, skb, state, nft_nat_do_chain); | ||||
| } | ||||
| 
 | ||||
| static const struct nf_chain_type nft_chain_nat_ipv4 = { | ||||
| static int nft_nat_ipv4_init(struct nft_ctx *ctx) | ||||
| { | ||||
| 	return nf_ct_netns_get(ctx->net, ctx->family); | ||||
| } | ||||
| 
 | ||||
| static void nft_nat_ipv4_free(struct nft_ctx *ctx) | ||||
| { | ||||
| 	nf_ct_netns_put(ctx->net, ctx->family); | ||||
| } | ||||
| 
 | ||||
| static const struct nft_chain_type nft_chain_nat_ipv4 = { | ||||
| 	.name		= "nat", | ||||
| 	.type		= NFT_CHAIN_T_NAT, | ||||
| 	.family		= NFPROTO_IPV4, | ||||
|  | @ -82,15 +92,13 @@ static const struct nf_chain_type nft_chain_nat_ipv4 = { | |||
| 		[NF_INET_LOCAL_OUT]	= nft_nat_ipv4_local_fn, | ||||
| 		[NF_INET_LOCAL_IN]	= nft_nat_ipv4_fn, | ||||
| 	}, | ||||
| 	.init		= nft_nat_ipv4_init, | ||||
| 	.free		= nft_nat_ipv4_free, | ||||
| }; | ||||
| 
 | ||||
| static int __init nft_chain_nat_init(void) | ||||
| { | ||||
| 	int err; | ||||
| 
 | ||||
| 	err = nft_register_chain_type(&nft_chain_nat_ipv4); | ||||
| 	if (err < 0) | ||||
| 		return err; | ||||
| 	nft_register_chain_type(&nft_chain_nat_ipv4); | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
|  |  | |||
|  | @ -58,7 +58,7 @@ static unsigned int nf_route_table_hook(void *priv, | |||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
| static const struct nf_chain_type nft_chain_route_ipv4 = { | ||||
| static const struct nft_chain_type nft_chain_route_ipv4 = { | ||||
| 	.name		= "route", | ||||
| 	.type		= NFT_CHAIN_T_ROUTE, | ||||
| 	.family		= NFPROTO_IPV4, | ||||
|  | @ -71,7 +71,9 @@ static const struct nf_chain_type nft_chain_route_ipv4 = { | |||
| 
 | ||||
| static int __init nft_chain_route_init(void) | ||||
| { | ||||
| 	return nft_register_chain_type(&nft_chain_route_ipv4); | ||||
| 	nft_register_chain_type(&nft_chain_route_ipv4); | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static void __exit nft_chain_route_exit(void) | ||||
|  |  | |||
|  | @ -34,7 +34,7 @@ config NF_SOCKET_IPV6 | |||
| if NF_TABLES | ||||
| 
 | ||||
| config NF_TABLES_IPV6 | ||||
| 	tristate "IPv6 nf_tables support" | ||||
| 	bool "IPv6 nf_tables support" | ||||
| 	help | ||||
| 	  This option enables the IPv6 support for nf_tables. | ||||
| 
 | ||||
|  |  | |||
|  | @ -36,7 +36,6 @@ obj-$(CONFIG_NF_REJECT_IPV6) += nf_reject_ipv6.o | |||
| obj-$(CONFIG_NF_DUP_IPV6) += nf_dup_ipv6.o | ||||
| 
 | ||||
| # nf_tables
 | ||||
| obj-$(CONFIG_NF_TABLES_IPV6) += nf_tables_ipv6.o | ||||
| obj-$(CONFIG_NFT_CHAIN_ROUTE_IPV6) += nft_chain_route_ipv6.o | ||||
| obj-$(CONFIG_NFT_CHAIN_NAT_IPV6) += nft_chain_nat_ipv6.o | ||||
| obj-$(CONFIG_NFT_REJECT_IPV6) += nft_reject_ipv6.o | ||||
|  |  | |||
|  | @ -420,11 +420,6 @@ mark_source_chains(const struct xt_table_info *newinfo, | |||
| 			     t->verdict < 0) || visited) { | ||||
| 				unsigned int oldpos, size; | ||||
| 
 | ||||
| 				if ((strcmp(t->target.u.user.name, | ||||
| 					    XT_STANDARD_TARGET) == 0) && | ||||
| 				    t->verdict < -NF_MAX_VERDICT - 1) | ||||
| 					return 0; | ||||
| 
 | ||||
| 				/* Return: backtrack through the last
 | ||||
| 				   big jump. */ | ||||
| 				do { | ||||
|  | @ -725,16 +720,9 @@ translate_table(struct net *net, struct xt_table_info *newinfo, void *entry0, | |||
| 	if (i != repl->num_entries) | ||||
| 		goto out_free; | ||||
| 
 | ||||
| 	/* Check hooks all assigned */ | ||||
| 	for (i = 0; i < NF_INET_NUMHOOKS; i++) { | ||||
| 		/* Only hooks which are valid */ | ||||
| 		if (!(repl->valid_hooks & (1 << i))) | ||||
| 			continue; | ||||
| 		if (newinfo->hook_entry[i] == 0xFFFFFFFF) | ||||
| 			goto out_free; | ||||
| 		if (newinfo->underflow[i] == 0xFFFFFFFF) | ||||
| 			goto out_free; | ||||
| 	} | ||||
| 	ret = xt_check_table_hooks(newinfo, repl->valid_hooks); | ||||
| 	if (ret) | ||||
| 		goto out_free; | ||||
| 
 | ||||
| 	if (!mark_source_chains(newinfo, repl->valid_hooks, entry0, offsets)) { | ||||
| 		ret = -ELOOP; | ||||
|  | @ -962,7 +950,9 @@ static int compat_table_info(const struct xt_table_info *info, | |||
| 	memcpy(newinfo, info, offsetof(struct xt_table_info, entries)); | ||||
| 	newinfo->initial_entries = 0; | ||||
| 	loc_cpu_entry = info->entries; | ||||
| 	xt_compat_init_offsets(AF_INET6, info->number); | ||||
| 	ret = xt_compat_init_offsets(AF_INET6, info->number); | ||||
| 	if (ret) | ||||
| 		return ret; | ||||
| 	xt_entry_foreach(iter, loc_cpu_entry, info->size) { | ||||
| 		ret = compat_calc_entry(iter, info, loc_cpu_entry, newinfo); | ||||
| 		if (ret != 0) | ||||
|  | @ -1075,7 +1065,7 @@ __do_replace(struct net *net, const char *name, unsigned int valid_hooks, | |||
| 	struct ip6t_entry *iter; | ||||
| 
 | ||||
| 	ret = 0; | ||||
| 	counters = vzalloc(num_counters * sizeof(struct xt_counters)); | ||||
| 	counters = xt_counters_alloc(num_counters); | ||||
| 	if (!counters) { | ||||
| 		ret = -ENOMEM; | ||||
| 		goto out; | ||||
|  | @ -1105,6 +1095,8 @@ __do_replace(struct net *net, const char *name, unsigned int valid_hooks, | |||
| 	    (newinfo->number <= oldinfo->initial_entries)) | ||||
| 		module_put(t->me); | ||||
| 
 | ||||
| 	xt_table_unlock(t); | ||||
| 
 | ||||
| 	get_old_counters(oldinfo, counters); | ||||
| 
 | ||||
| 	/* Decrease module usage counts and free resource */ | ||||
|  | @ -1118,7 +1110,6 @@ __do_replace(struct net *net, const char *name, unsigned int valid_hooks, | |||
| 		net_warn_ratelimited("ip6tables: counters copy to user failed while replacing table\n"); | ||||
| 	} | ||||
| 	vfree(counters); | ||||
| 	xt_table_unlock(t); | ||||
| 	return ret; | ||||
| 
 | ||||
|  put_module: | ||||
|  | @ -1425,7 +1416,7 @@ translate_compat_table(struct net *net, | |||
| 	struct compat_ip6t_entry *iter0; | ||||
| 	struct ip6t_replace repl; | ||||
| 	unsigned int size; | ||||
| 	int ret = 0; | ||||
| 	int ret; | ||||
| 
 | ||||
| 	info = *pinfo; | ||||
| 	entry0 = *pentry0; | ||||
|  | @ -1434,7 +1425,9 @@ translate_compat_table(struct net *net, | |||
| 
 | ||||
| 	j = 0; | ||||
| 	xt_compat_lock(AF_INET6); | ||||
| 	xt_compat_init_offsets(AF_INET6, compatr->num_entries); | ||||
| 	ret = xt_compat_init_offsets(AF_INET6, compatr->num_entries); | ||||
| 	if (ret) | ||||
| 		goto out_unlock; | ||||
| 	/* Walk through entries, checking offsets. */ | ||||
| 	xt_entry_foreach(iter0, entry0, compatr->size) { | ||||
| 		ret = check_compat_entry_size_and_hooks(iter0, info, &size, | ||||
|  |  | |||
|  | @ -18,6 +18,7 @@ | |||
| #include <net/netfilter/nf_conntrack.h> | ||||
| #include <net/netfilter/nf_conntrack_seqadj.h> | ||||
| #include <net/netfilter/nf_conntrack_synproxy.h> | ||||
| #include <net/netfilter/nf_conntrack_ecache.h> | ||||
| 
 | ||||
| static struct ipv6hdr * | ||||
| synproxy_build_ip(struct net *net, struct sk_buff *skb, | ||||
|  | @ -405,6 +406,8 @@ static unsigned int ipv6_synproxy_hook(void *priv, | |||
| 		synproxy->isn = ntohl(th->ack_seq); | ||||
| 		if (opts.options & XT_SYNPROXY_OPT_TIMESTAMP) | ||||
| 			synproxy->its = opts.tsecr; | ||||
| 
 | ||||
| 		nf_conntrack_event_cache(IPCT_SYNPROXY, ct); | ||||
| 		break; | ||||
| 	case TCP_CONNTRACK_SYN_RECV: | ||||
| 		if (!th->syn || !th->ack) | ||||
|  | @ -413,8 +416,10 @@ static unsigned int ipv6_synproxy_hook(void *priv, | |||
| 		if (!synproxy_parse_options(skb, thoff, th, &opts)) | ||||
| 			return NF_DROP; | ||||
| 
 | ||||
| 		if (opts.options & XT_SYNPROXY_OPT_TIMESTAMP) | ||||
| 		if (opts.options & XT_SYNPROXY_OPT_TIMESTAMP) { | ||||
| 			synproxy->tsoff = opts.tsval - synproxy->its; | ||||
| 			nf_conntrack_event_cache(IPCT_SYNPROXY, ct); | ||||
| 		} | ||||
| 
 | ||||
| 		opts.options &= ~(XT_SYNPROXY_OPT_MSS | | ||||
| 				  XT_SYNPROXY_OPT_WSCALE | | ||||
|  | @ -424,6 +429,7 @@ static unsigned int ipv6_synproxy_hook(void *priv, | |||
| 		synproxy_send_server_ack(net, state, skb, th, &opts); | ||||
| 
 | ||||
| 		nf_ct_seqadj_init(ct, ctinfo, synproxy->isn - ntohl(th->seq)); | ||||
| 		nf_conntrack_event_cache(IPCT_SEQADJ, ct); | ||||
| 
 | ||||
| 		swap(opts.tsval, opts.tsecr); | ||||
| 		synproxy_send_client_ack(net, skb, th, &opts); | ||||
|  |  | |||
|  | @ -1,65 +0,0 @@ | |||
| /*
 | ||||
|  * Copyright (c) 2008 Patrick McHardy <kaber@trash.net> | ||||
|  * Copyright (c) 2012-2013 Pablo Neira Ayuso <pablo@netfilter.org> | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 as | ||||
|  * published by the Free Software Foundation. | ||||
|  * | ||||
|  * Development of this code funded by Astaro AG (http://www.astaro.com/)
 | ||||
|  */ | ||||
| 
 | ||||
| #include <linux/init.h> | ||||
| #include <linux/module.h> | ||||
| #include <linux/ipv6.h> | ||||
| #include <linux/netfilter_ipv6.h> | ||||
| #include <net/netfilter/nf_tables.h> | ||||
| #include <net/netfilter/nf_tables_ipv6.h> | ||||
| 
 | ||||
| static unsigned int nft_do_chain_ipv6(void *priv, | ||||
| 				      struct sk_buff *skb, | ||||
| 				      const struct nf_hook_state *state) | ||||
| { | ||||
| 	struct nft_pktinfo pkt; | ||||
| 
 | ||||
| 	nft_set_pktinfo(&pkt, skb, state); | ||||
| 	nft_set_pktinfo_ipv6(&pkt, skb); | ||||
| 
 | ||||
| 	return nft_do_chain(&pkt, priv); | ||||
| } | ||||
| 
 | ||||
| static const struct nf_chain_type filter_ipv6 = { | ||||
| 	.name		= "filter", | ||||
| 	.type		= NFT_CHAIN_T_DEFAULT, | ||||
| 	.family		= NFPROTO_IPV6, | ||||
| 	.owner		= THIS_MODULE, | ||||
| 	.hook_mask	= (1 << NF_INET_LOCAL_IN) | | ||||
| 			  (1 << NF_INET_LOCAL_OUT) | | ||||
| 			  (1 << NF_INET_FORWARD) | | ||||
| 			  (1 << NF_INET_PRE_ROUTING) | | ||||
| 			  (1 << NF_INET_POST_ROUTING), | ||||
| 	.hooks		= { | ||||
| 		[NF_INET_LOCAL_IN]	= nft_do_chain_ipv6, | ||||
| 		[NF_INET_LOCAL_OUT]	= nft_do_chain_ipv6, | ||||
| 		[NF_INET_FORWARD]	= nft_do_chain_ipv6, | ||||
| 		[NF_INET_PRE_ROUTING]	= nft_do_chain_ipv6, | ||||
| 		[NF_INET_POST_ROUTING]	= nft_do_chain_ipv6, | ||||
| 	}, | ||||
| }; | ||||
| 
 | ||||
| static int __init nf_tables_ipv6_init(void) | ||||
| { | ||||
| 	return nft_register_chain_type(&filter_ipv6); | ||||
| } | ||||
| 
 | ||||
| static void __exit nf_tables_ipv6_exit(void) | ||||
| { | ||||
| 	nft_unregister_chain_type(&filter_ipv6); | ||||
| } | ||||
| 
 | ||||
| module_init(nf_tables_ipv6_init); | ||||
| module_exit(nf_tables_ipv6_exit); | ||||
| 
 | ||||
| MODULE_LICENSE("GPL"); | ||||
| MODULE_AUTHOR("Patrick McHardy <kaber@trash.net>"); | ||||
| MODULE_ALIAS_NFT_CHAIN(AF_INET6, "filter"); | ||||
|  | @ -65,7 +65,17 @@ static unsigned int nft_nat_ipv6_local_fn(void *priv, | |||
| 	return nf_nat_ipv6_local_fn(priv, skb, state, nft_nat_do_chain); | ||||
| } | ||||
| 
 | ||||
| static const struct nf_chain_type nft_chain_nat_ipv6 = { | ||||
| static int nft_nat_ipv6_init(struct nft_ctx *ctx) | ||||
| { | ||||
| 	return nf_ct_netns_get(ctx->net, ctx->family); | ||||
| } | ||||
| 
 | ||||
| static void nft_nat_ipv6_free(struct nft_ctx *ctx) | ||||
| { | ||||
| 	nf_ct_netns_put(ctx->net, ctx->family); | ||||
| } | ||||
| 
 | ||||
| static const struct nft_chain_type nft_chain_nat_ipv6 = { | ||||
| 	.name		= "nat", | ||||
| 	.type		= NFT_CHAIN_T_NAT, | ||||
| 	.family		= NFPROTO_IPV6, | ||||
|  | @ -80,15 +90,13 @@ static const struct nf_chain_type nft_chain_nat_ipv6 = { | |||
| 		[NF_INET_LOCAL_OUT]	= nft_nat_ipv6_local_fn, | ||||
| 		[NF_INET_LOCAL_IN]	= nft_nat_ipv6_fn, | ||||
| 	}, | ||||
| 	.init		= nft_nat_ipv6_init, | ||||
| 	.free		= nft_nat_ipv6_free, | ||||
| }; | ||||
| 
 | ||||
| static int __init nft_chain_nat_ipv6_init(void) | ||||
| { | ||||
| 	int err; | ||||
| 
 | ||||
| 	err = nft_register_chain_type(&nft_chain_nat_ipv6); | ||||
| 	if (err < 0) | ||||
| 		return err; | ||||
| 	nft_register_chain_type(&nft_chain_nat_ipv6); | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
|  |  | |||
|  | @ -60,7 +60,7 @@ static unsigned int nf_route_table_hook(void *priv, | |||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
| static const struct nf_chain_type nft_chain_route_ipv6 = { | ||||
| static const struct nft_chain_type nft_chain_route_ipv6 = { | ||||
| 	.name		= "route", | ||||
| 	.type		= NFT_CHAIN_T_ROUTE, | ||||
| 	.family		= NFPROTO_IPV6, | ||||
|  | @ -73,7 +73,9 @@ static const struct nf_chain_type nft_chain_route_ipv6 = { | |||
| 
 | ||||
| static int __init nft_chain_route_init(void) | ||||
| { | ||||
| 	return nft_register_chain_type(&nft_chain_route_ipv6); | ||||
| 	nft_register_chain_type(&nft_chain_route_ipv6); | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static void __exit nft_chain_route_exit(void) | ||||
|  |  | |||
|  | @ -465,12 +465,12 @@ config NF_TABLES_INET | |||
| 	depends on IPV6 | ||||
| 	select NF_TABLES_IPV4 | ||||
| 	select NF_TABLES_IPV6 | ||||
| 	tristate "Netfilter nf_tables mixed IPv4/IPv6 tables support" | ||||
| 	bool "Netfilter nf_tables mixed IPv4/IPv6 tables support" | ||||
| 	help | ||||
| 	  This option enables support for a mixed IPv4/IPv6 "inet" table. | ||||
| 
 | ||||
| config NF_TABLES_NETDEV | ||||
| 	tristate "Netfilter nf_tables netdev tables support" | ||||
| 	bool "Netfilter nf_tables netdev tables support" | ||||
| 	help | ||||
| 	  This option enables support for the "netdev" table. | ||||
| 
 | ||||
|  |  | |||
|  | @ -73,13 +73,12 @@ obj-$(CONFIG_NETFILTER_CONNCOUNT) += nf_conncount.o | |||
| obj-$(CONFIG_NF_DUP_NETDEV)	+= nf_dup_netdev.o | ||||
| 
 | ||||
| # nf_tables
 | ||||
| nf_tables-objs := nf_tables_core.o nf_tables_api.o nf_tables_trace.o \
 | ||||
| 		  nft_immediate.o nft_cmp.o nft_range.o nft_bitwise.o \
 | ||||
| 		  nft_byteorder.o nft_payload.o nft_lookup.o nft_dynset.o | ||||
| nf_tables-objs := nf_tables_core.o nf_tables_api.o nft_chain_filter.o \
 | ||||
| 		  nf_tables_trace.o nft_immediate.o nft_cmp.o nft_range.o \
 | ||||
| 		  nft_bitwise.o nft_byteorder.o nft_payload.o nft_lookup.o \
 | ||||
| 		  nft_dynset.o | ||||
| 
 | ||||
| obj-$(CONFIG_NF_TABLES)		+= nf_tables.o | ||||
| obj-$(CONFIG_NF_TABLES_INET)	+= nf_tables_inet.o | ||||
| obj-$(CONFIG_NF_TABLES_NETDEV)	+= nf_tables_netdev.o | ||||
| obj-$(CONFIG_NFT_COMPAT)	+= nft_compat.o | ||||
| obj-$(CONFIG_NFT_EXTHDR)	+= nft_exthdr.o | ||||
| obj-$(CONFIG_NFT_META)		+= nft_meta.o | ||||
|  |  | |||
|  | @ -72,9 +72,6 @@ hash_mac4_data_next(struct hash_mac4_elem *next, | |||
| #define IP_SET_PROTO_UNDEF | ||||
| #include "ip_set_hash_gen.h" | ||||
| 
 | ||||
| /* Zero valued element is not supported */ | ||||
| static const unsigned char invalid_ether[ETH_ALEN] = { 0 }; | ||||
| 
 | ||||
| static int | ||||
| hash_mac4_kadt(struct ip_set *set, const struct sk_buff *skb, | ||||
| 	       const struct xt_action_param *par, | ||||
|  | @ -93,7 +90,7 @@ hash_mac4_kadt(struct ip_set *set, const struct sk_buff *skb, | |||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	ether_addr_copy(e.ether, eth_hdr(skb)->h_source); | ||||
| 	if (memcmp(e.ether, invalid_ether, ETH_ALEN) == 0) | ||||
| 	if (is_zero_ether_addr(e.ether)) | ||||
| 		return -EINVAL; | ||||
| 	return adtfn(set, &e, &ext, &opt->ext, opt->cmdflags); | ||||
| } | ||||
|  | @ -118,7 +115,7 @@ hash_mac4_uadt(struct ip_set *set, struct nlattr *tb[], | |||
| 	if (ret) | ||||
| 		return ret; | ||||
| 	ether_addr_copy(e.ether, nla_data(tb[IPSET_ATTR_ETHER])); | ||||
| 	if (memcmp(e.ether, invalid_ether, ETH_ALEN) == 0) | ||||
| 	if (is_zero_ether_addr(e.ether)) | ||||
| 		return -IPSET_ERR_HASH_ELEM; | ||||
| 
 | ||||
| 	return adtfn(set, &e, &ext, &ext, flags); | ||||
|  |  | |||
|  | @ -238,7 +238,7 @@ static void ip_vs_lblc_flush(struct ip_vs_service *svc) | |||
| 	int i; | ||||
| 
 | ||||
| 	spin_lock_bh(&svc->sched_lock); | ||||
| 	tbl->dead = 1; | ||||
| 	tbl->dead = true; | ||||
| 	for (i = 0; i < IP_VS_LBLC_TAB_SIZE; i++) { | ||||
| 		hlist_for_each_entry_safe(en, next, &tbl->bucket[i], list) { | ||||
| 			ip_vs_lblc_del(en); | ||||
|  | @ -369,7 +369,7 @@ static int ip_vs_lblc_init_svc(struct ip_vs_service *svc) | |||
| 	tbl->max_size = IP_VS_LBLC_TAB_SIZE*16; | ||||
| 	tbl->rover = 0; | ||||
| 	tbl->counter = 1; | ||||
| 	tbl->dead = 0; | ||||
| 	tbl->dead = false; | ||||
| 	tbl->svc = svc; | ||||
| 
 | ||||
| 	/*
 | ||||
|  |  | |||
|  | @ -404,7 +404,7 @@ static void ip_vs_lblcr_flush(struct ip_vs_service *svc) | |||
| 	struct hlist_node *next; | ||||
| 
 | ||||
| 	spin_lock_bh(&svc->sched_lock); | ||||
| 	tbl->dead = 1; | ||||
| 	tbl->dead = true; | ||||
| 	for (i = 0; i < IP_VS_LBLCR_TAB_SIZE; i++) { | ||||
| 		hlist_for_each_entry_safe(en, next, &tbl->bucket[i], list) { | ||||
| 			ip_vs_lblcr_free(en); | ||||
|  | @ -532,7 +532,7 @@ static int ip_vs_lblcr_init_svc(struct ip_vs_service *svc) | |||
| 	tbl->max_size = IP_VS_LBLCR_TAB_SIZE*16; | ||||
| 	tbl->rover = 0; | ||||
| 	tbl->counter = 1; | ||||
| 	tbl->dead = 0; | ||||
| 	tbl->dead = false; | ||||
| 	tbl->svc = svc; | ||||
| 
 | ||||
| 	/*
 | ||||
|  |  | |||
|  | @ -104,7 +104,7 @@ static unsigned int check_hlist(struct net *net, | |||
| 	struct nf_conn *found_ct; | ||||
| 	unsigned int length = 0; | ||||
| 
 | ||||
| 	*addit = true; | ||||
| 	*addit = tuple ? true : false; | ||||
| 
 | ||||
| 	/* check the saved connections */ | ||||
| 	hlist_for_each_entry_safe(conn, n, head, node) { | ||||
|  | @ -117,7 +117,7 @@ static unsigned int check_hlist(struct net *net, | |||
| 
 | ||||
| 		found_ct = nf_ct_tuplehash_to_ctrack(found); | ||||
| 
 | ||||
| 		if (nf_ct_tuple_equal(&conn->tuple, tuple)) { | ||||
| 		if (tuple && nf_ct_tuple_equal(&conn->tuple, tuple)) { | ||||
| 			/*
 | ||||
| 			 * Just to be sure we have it only once in the list. | ||||
| 			 * We should not see tuples twice unless someone hooks | ||||
|  | @ -158,7 +158,6 @@ static void tree_nodes_free(struct rb_root *root, | |||
| static unsigned int | ||||
| count_tree(struct net *net, struct rb_root *root, | ||||
| 	   const u32 *key, u8 keylen, | ||||
| 	   u8 family, | ||||
| 	   const struct nf_conntrack_tuple *tuple, | ||||
| 	   const struct nf_conntrack_zone *zone) | ||||
| { | ||||
|  | @ -221,6 +220,9 @@ count_tree(struct net *net, struct rb_root *root, | |||
| 		goto restart; | ||||
| 	} | ||||
| 
 | ||||
| 	if (!tuple) | ||||
| 		return 0; | ||||
| 
 | ||||
| 	/* no match, need to insert new node */ | ||||
| 	rbconn = kmem_cache_alloc(conncount_rb_cachep, GFP_ATOMIC); | ||||
| 	if (rbconn == NULL) | ||||
|  | @ -243,10 +245,12 @@ count_tree(struct net *net, struct rb_root *root, | |||
| 	return 1; | ||||
| } | ||||
| 
 | ||||
| /* Count and return number of conntrack entries in 'net' with particular 'key'.
 | ||||
|  * If 'tuple' is not null, insert it into the accounting data structure. | ||||
|  */ | ||||
| unsigned int nf_conncount_count(struct net *net, | ||||
| 				struct nf_conncount_data *data, | ||||
| 				const u32 *key, | ||||
| 				unsigned int family, | ||||
| 				const struct nf_conntrack_tuple *tuple, | ||||
| 				const struct nf_conntrack_zone *zone) | ||||
| { | ||||
|  | @ -259,7 +263,7 @@ unsigned int nf_conncount_count(struct net *net, | |||
| 
 | ||||
| 	spin_lock_bh(&nf_conncount_locks[hash % CONNCOUNT_LOCK_SLOTS]); | ||||
| 
 | ||||
| 	count = count_tree(net, root, key, data->keylen, family, tuple, zone); | ||||
| 	count = count_tree(net, root, key, data->keylen, tuple, zone); | ||||
| 
 | ||||
| 	spin_unlock_bh(&nf_conncount_locks[hash % CONNCOUNT_LOCK_SLOTS]); | ||||
| 
 | ||||
|  |  | |||
|  | @ -8,6 +8,8 @@ | |||
|  * published by the Free Software Foundation. | ||||
|  */ | ||||
| 
 | ||||
| #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt | ||||
| 
 | ||||
| #include <linux/netfilter.h> | ||||
| #include <linux/slab.h> | ||||
| #include <linux/kernel.h> | ||||
|  | @ -80,7 +82,7 @@ static int nf_conntrack_acct_init_sysctl(struct net *net) | |||
| 	net->ct.acct_sysctl_header = register_net_sysctl(net, "net/netfilter", | ||||
| 							 table); | ||||
| 	if (!net->ct.acct_sysctl_header) { | ||||
| 		printk(KERN_ERR "nf_conntrack_acct: can't register to sysctl.\n"); | ||||
| 		pr_err("can't register to sysctl\n"); | ||||
| 		goto out_register; | ||||
| 	} | ||||
| 	return 0; | ||||
|  | @ -125,7 +127,7 @@ int nf_conntrack_acct_init(void) | |||
| { | ||||
| 	int ret = nf_ct_extend_register(&acct_extend); | ||||
| 	if (ret < 0) | ||||
| 		pr_err("nf_conntrack_acct: Unable to register extension\n"); | ||||
| 		pr_err("Unable to register extension\n"); | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
|  |  | |||
|  | @ -20,7 +20,6 @@ | |||
| #include <net/netfilter/nf_conntrack_expect.h> | ||||
| 
 | ||||
| int nf_conntrack_broadcast_help(struct sk_buff *skb, | ||||
| 				unsigned int protoff, | ||||
| 				struct nf_conn *ct, | ||||
| 				enum ip_conntrack_info ctinfo, | ||||
| 				unsigned int timeout) | ||||
|  |  | |||
|  | @ -11,6 +11,8 @@ | |||
|  * published by the Free Software Foundation. | ||||
|  */ | ||||
| 
 | ||||
| #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt | ||||
| 
 | ||||
| #include <linux/types.h> | ||||
| #include <linux/netfilter.h> | ||||
| #include <linux/skbuff.h> | ||||
|  | @ -372,7 +374,7 @@ static int nf_conntrack_event_init_sysctl(struct net *net) | |||
| 	net->ct.event_sysctl_header = | ||||
| 		register_net_sysctl(net, "net/netfilter", table); | ||||
| 	if (!net->ct.event_sysctl_header) { | ||||
| 		printk(KERN_ERR "nf_ct_event: can't register to sysctl.\n"); | ||||
| 		pr_err("can't register to sysctl\n"); | ||||
| 		goto out_register; | ||||
| 	} | ||||
| 	return 0; | ||||
|  | @ -419,7 +421,7 @@ int nf_conntrack_ecache_init(void) | |||
| { | ||||
| 	int ret = nf_ct_extend_register(&event_extend); | ||||
| 	if (ret < 0) | ||||
| 		pr_err("nf_ct_event: Unable to register event extension.\n"); | ||||
| 		pr_err("Unable to register event extension\n"); | ||||
| 
 | ||||
| 	BUILD_BUG_ON(__IPCT_MAX >= 16);	/* ctmask, missed use u16 */ | ||||
| 
 | ||||
|  |  | |||
|  | @ -41,9 +41,10 @@ static struct nf_conntrack_expect_policy exp_policy = { | |||
| }; | ||||
| 
 | ||||
| static int netbios_ns_help(struct sk_buff *skb, unsigned int protoff, | ||||
| 		   struct nf_conn *ct, enum ip_conntrack_info ctinfo) | ||||
| 			   struct nf_conn *ct, | ||||
| 			   enum ip_conntrack_info ctinfo) | ||||
| { | ||||
| 	return nf_conntrack_broadcast_help(skb, protoff, ct, ctinfo, timeout); | ||||
| 	return nf_conntrack_broadcast_help(skb, ct, ctinfo, timeout); | ||||
| } | ||||
| 
 | ||||
| static struct nf_conntrack_helper helper __read_mostly = { | ||||
|  |  | |||
|  | @ -440,6 +440,31 @@ err: | |||
| 	return -1; | ||||
| } | ||||
| 
 | ||||
| static int ctnetlink_dump_ct_synproxy(struct sk_buff *skb, struct nf_conn *ct) | ||||
| { | ||||
| 	struct nf_conn_synproxy *synproxy = nfct_synproxy(ct); | ||||
| 	struct nlattr *nest_parms; | ||||
| 
 | ||||
| 	if (!synproxy) | ||||
| 		return 0; | ||||
| 
 | ||||
| 	nest_parms = nla_nest_start(skb, CTA_SYNPROXY | NLA_F_NESTED); | ||||
| 	if (!nest_parms) | ||||
| 		goto nla_put_failure; | ||||
| 
 | ||||
| 	if (nla_put_be32(skb, CTA_SYNPROXY_ISN, htonl(synproxy->isn)) || | ||||
| 	    nla_put_be32(skb, CTA_SYNPROXY_ITS, htonl(synproxy->its)) || | ||||
| 	    nla_put_be32(skb, CTA_SYNPROXY_TSOFF, htonl(synproxy->tsoff))) | ||||
| 		goto nla_put_failure; | ||||
| 
 | ||||
| 	nla_nest_end(skb, nest_parms); | ||||
| 
 | ||||
| 	return 0; | ||||
| 
 | ||||
| nla_put_failure: | ||||
| 	return -1; | ||||
| } | ||||
| 
 | ||||
| static int ctnetlink_dump_id(struct sk_buff *skb, const struct nf_conn *ct) | ||||
| { | ||||
| 	if (nla_put_be32(skb, CTA_ID, htonl((unsigned long)ct))) | ||||
|  | @ -518,7 +543,8 @@ ctnetlink_fill_info(struct sk_buff *skb, u32 portid, u32 seq, u32 type, | |||
| 	    ctnetlink_dump_id(skb, ct) < 0 || | ||||
| 	    ctnetlink_dump_use(skb, ct) < 0 || | ||||
| 	    ctnetlink_dump_master(skb, ct) < 0 || | ||||
| 	    ctnetlink_dump_ct_seq_adj(skb, ct) < 0) | ||||
| 	    ctnetlink_dump_ct_seq_adj(skb, ct) < 0 || | ||||
| 	    ctnetlink_dump_ct_synproxy(skb, ct) < 0) | ||||
| 		goto nla_put_failure; | ||||
| 
 | ||||
| 	nlmsg_end(skb, nlh); | ||||
|  | @ -730,6 +756,10 @@ ctnetlink_conntrack_event(unsigned int events, struct nf_ct_event *item) | |||
| 		if (events & (1 << IPCT_SEQADJ) && | ||||
| 		    ctnetlink_dump_ct_seq_adj(skb, ct) < 0) | ||||
| 			goto nla_put_failure; | ||||
| 
 | ||||
| 		if (events & (1 << IPCT_SYNPROXY) && | ||||
| 		    ctnetlink_dump_ct_synproxy(skb, ct) < 0) | ||||
| 			goto nla_put_failure; | ||||
| 	} | ||||
| 
 | ||||
| #ifdef CONFIG_NF_CONNTRACK_MARK | ||||
|  | @ -1497,9 +1527,8 @@ ctnetlink_setup_nat(struct nf_conn *ct, const struct nlattr * const cda[]) | |||
| 	if (ret < 0) | ||||
| 		return ret; | ||||
| 
 | ||||
| 	ret = ctnetlink_parse_nat_setup(ct, NF_NAT_MANIP_SRC, | ||||
| 					cda[CTA_NAT_SRC]); | ||||
| 	return ret; | ||||
| 	return ctnetlink_parse_nat_setup(ct, NF_NAT_MANIP_SRC, | ||||
| 					 cda[CTA_NAT_SRC]); | ||||
| #else | ||||
| 	if (!cda[CTA_NAT_DST] && !cda[CTA_NAT_SRC]) | ||||
| 		return 0; | ||||
|  | @ -1689,6 +1718,39 @@ err: | |||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
| static const struct nla_policy synproxy_policy[CTA_SYNPROXY_MAX + 1] = { | ||||
| 	[CTA_SYNPROXY_ISN]	= { .type = NLA_U32 }, | ||||
| 	[CTA_SYNPROXY_ITS]	= { .type = NLA_U32 }, | ||||
| 	[CTA_SYNPROXY_TSOFF]	= { .type = NLA_U32 }, | ||||
| }; | ||||
| 
 | ||||
| static int ctnetlink_change_synproxy(struct nf_conn *ct, | ||||
| 				     const struct nlattr * const cda[]) | ||||
| { | ||||
| 	struct nf_conn_synproxy *synproxy = nfct_synproxy(ct); | ||||
| 	struct nlattr *tb[CTA_SYNPROXY_MAX + 1]; | ||||
| 	int err; | ||||
| 
 | ||||
| 	if (!synproxy) | ||||
| 		return 0; | ||||
| 
 | ||||
| 	err = nla_parse_nested(tb, CTA_SYNPROXY_MAX, cda[CTA_SYNPROXY], | ||||
| 			       synproxy_policy, NULL); | ||||
| 	if (err < 0) | ||||
| 		return err; | ||||
| 
 | ||||
| 	if (!tb[CTA_SYNPROXY_ISN] || | ||||
| 	    !tb[CTA_SYNPROXY_ITS] || | ||||
| 	    !tb[CTA_SYNPROXY_TSOFF]) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	synproxy->isn = ntohl(nla_get_be32(tb[CTA_SYNPROXY_ISN])); | ||||
| 	synproxy->its = ntohl(nla_get_be32(tb[CTA_SYNPROXY_ITS])); | ||||
| 	synproxy->tsoff = ntohl(nla_get_be32(tb[CTA_SYNPROXY_TSOFF])); | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static int | ||||
| ctnetlink_attach_labels(struct nf_conn *ct, const struct nlattr * const cda[]) | ||||
| { | ||||
|  | @ -1759,6 +1821,12 @@ ctnetlink_change_conntrack(struct nf_conn *ct, | |||
| 			return err; | ||||
| 	} | ||||
| 
 | ||||
| 	if (cda[CTA_SYNPROXY]) { | ||||
| 		err = ctnetlink_change_synproxy(ct, cda); | ||||
| 		if (err < 0) | ||||
| 			return err; | ||||
| 	} | ||||
| 
 | ||||
| 	if (cda[CTA_LABELS]) { | ||||
| 		err = ctnetlink_attach_labels(ct, cda); | ||||
| 		if (err < 0) | ||||
|  | @ -1880,6 +1948,12 @@ ctnetlink_create_conntrack(struct net *net, | |||
| 			goto err2; | ||||
| 	} | ||||
| 
 | ||||
| 	if (cda[CTA_SYNPROXY]) { | ||||
| 		err = ctnetlink_change_synproxy(ct, cda); | ||||
| 		if (err < 0) | ||||
| 			goto err2; | ||||
| 	} | ||||
| 
 | ||||
| #if defined(CONFIG_NF_CONNTRACK_MARK) | ||||
| 	if (cda[CTA_MARK]) | ||||
| 		ct->mark = ntohl(nla_get_be32(cda[CTA_MARK])); | ||||
|  | @ -1991,7 +2065,9 @@ static int ctnetlink_new_conntrack(struct net *net, struct sock *ctnl, | |||
| 						      (1 << IPCT_HELPER) | | ||||
| 						      (1 << IPCT_PROTOINFO) | | ||||
| 						      (1 << IPCT_SEQADJ) | | ||||
| 						      (1 << IPCT_MARK) | events, | ||||
| 						      (1 << IPCT_MARK) | | ||||
| 						      (1 << IPCT_SYNPROXY) | | ||||
| 						      events, | ||||
| 						      ct, NETLINK_CB(skb).portid, | ||||
| 						      nlmsg_report(nlh)); | ||||
| 			nf_ct_put(ct); | ||||
|  | @ -2012,7 +2088,8 @@ static int ctnetlink_new_conntrack(struct net *net, struct sock *ctnl, | |||
| 						      (1 << IPCT_LABEL) | | ||||
| 						      (1 << IPCT_PROTOINFO) | | ||||
| 						      (1 << IPCT_SEQADJ) | | ||||
| 						      (1 << IPCT_MARK), | ||||
| 						      (1 << IPCT_MARK) | | ||||
| 						      (1 << IPCT_SYNPROXY), | ||||
| 						      ct, NETLINK_CB(skb).portid, | ||||
| 						      nlmsg_report(nlh)); | ||||
| 		} | ||||
|  | @ -2282,6 +2359,9 @@ static int __ctnetlink_glue_build(struct sk_buff *skb, struct nf_conn *ct) | |||
| 	    ctnetlink_dump_ct_seq_adj(skb, ct) < 0) | ||||
| 		goto nla_put_failure; | ||||
| 
 | ||||
| 	if (ctnetlink_dump_ct_synproxy(skb, ct) < 0) | ||||
| 		goto nla_put_failure; | ||||
| 
 | ||||
| #ifdef CONFIG_NF_CONNTRACK_MARK | ||||
| 	if (ct->mark && ctnetlink_dump_mark(skb, ct) < 0) | ||||
| 		goto nla_put_failure; | ||||
|  |  | |||
|  | @ -36,11 +36,12 @@ int (*nf_nat_snmp_hook)(struct sk_buff *skb, | |||
| EXPORT_SYMBOL_GPL(nf_nat_snmp_hook); | ||||
| 
 | ||||
| static int snmp_conntrack_help(struct sk_buff *skb, unsigned int protoff, | ||||
| 		struct nf_conn *ct, enum ip_conntrack_info ctinfo) | ||||
| 			       struct nf_conn *ct, | ||||
| 			       enum ip_conntrack_info ctinfo) | ||||
| { | ||||
| 	typeof(nf_nat_snmp_hook) nf_nat_snmp; | ||||
| 
 | ||||
| 	nf_conntrack_broadcast_help(skb, protoff, ct, ctinfo, timeout); | ||||
| 	nf_conntrack_broadcast_help(skb, ct, ctinfo, timeout); | ||||
| 
 | ||||
| 	nf_nat_snmp = rcu_dereference(nf_nat_snmp_hook); | ||||
| 	if (nf_nat_snmp && ct->status & IPS_NAT_MASK) | ||||
|  |  | |||
|  | @ -6,6 +6,8 @@ | |||
|  * published by the Free Software Foundation (or any later at your option). | ||||
|  */ | ||||
| 
 | ||||
| #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt | ||||
| 
 | ||||
| #include <linux/netfilter.h> | ||||
| #include <linux/slab.h> | ||||
| #include <linux/kernel.h> | ||||
|  | @ -58,7 +60,7 @@ static int nf_conntrack_tstamp_init_sysctl(struct net *net) | |||
| 	net->ct.tstamp_sysctl_header = register_net_sysctl(net,	"net/netfilter", | ||||
| 							   table); | ||||
| 	if (!net->ct.tstamp_sysctl_header) { | ||||
| 		printk(KERN_ERR "nf_ct_tstamp: can't register to sysctl.\n"); | ||||
| 		pr_err("can't register to sysctl\n"); | ||||
| 		goto out_register; | ||||
| 	} | ||||
| 	return 0; | ||||
|  | @ -104,7 +106,7 @@ int nf_conntrack_tstamp_init(void) | |||
| 	int ret; | ||||
| 	ret = nf_ct_extend_register(&tstamp_extend); | ||||
| 	if (ret < 0) | ||||
| 		pr_err("nf_ct_tstamp: Unable to register extension\n"); | ||||
| 		pr_err("Unable to register extension\n"); | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
|  |  | |||
|  | @ -8,6 +8,8 @@ | |||
|  * published by the Free Software Foundation. | ||||
|  */ | ||||
| 
 | ||||
| #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt | ||||
| 
 | ||||
| #include <linux/module.h> | ||||
| #include <linux/types.h> | ||||
| #include <linux/timer.h> | ||||
|  | @ -814,7 +816,7 @@ static int __init nf_nat_init(void) | |||
| 	ret = nf_ct_extend_register(&nat_extend); | ||||
| 	if (ret < 0) { | ||||
| 		nf_ct_free_hashtable(nf_nat_bysource, nf_nat_htable_size); | ||||
| 		printk(KERN_ERR "nf_nat_core: Unable to register extension\n"); | ||||
| 		pr_err("Unable to register extension\n"); | ||||
| 		return ret; | ||||
| 	} | ||||
| 
 | ||||
|  |  | |||
|  | @ -8,6 +8,8 @@ | |||
|  * published by the Free Software Foundation. | ||||
|  */ | ||||
| 
 | ||||
| #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt | ||||
| 
 | ||||
| #include <linux/module.h> | ||||
| #include <linux/moduleparam.h> | ||||
| #include <linux/inet.h> | ||||
|  | @ -71,7 +73,7 @@ static unsigned int nf_nat_ftp(struct sk_buff *skb, | |||
| 	char buffer[sizeof("|1||65535|") + INET6_ADDRSTRLEN]; | ||||
| 	unsigned int buflen; | ||||
| 
 | ||||
| 	pr_debug("FTP_NAT: type %i, off %u len %u\n", type, matchoff, matchlen); | ||||
| 	pr_debug("type %i, off %u len %u\n", type, matchoff, matchlen); | ||||
| 
 | ||||
| 	/* Connection will come from wherever this packet goes, hence !dir */ | ||||
| 	newaddr = ct->tuplehash[!dir].tuple.dst.u3; | ||||
|  | @ -136,8 +138,7 @@ static int __init nf_nat_ftp_init(void) | |||
| /* Prior to 2.6.11, we had a ports param.  No longer, but don't break users. */ | ||||
| static int warn_set(const char *val, const struct kernel_param *kp) | ||||
| { | ||||
| 	printk(KERN_INFO KBUILD_MODNAME | ||||
| 	       ": kernel >= 2.6.10 only uses 'ports' for conntrack modules\n"); | ||||
| 	pr_info("kernel >= 2.6.10 only uses 'ports' for conntrack modules\n"); | ||||
| 	return 0; | ||||
| } | ||||
| module_param_call(ports, warn_set, NULL, NULL, 0); | ||||
|  |  | |||
|  | @ -10,6 +10,8 @@ | |||
|  * 2 of the License, or (at your option) any later version. | ||||
|  */ | ||||
| 
 | ||||
| #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt | ||||
| 
 | ||||
| #include <linux/module.h> | ||||
| #include <linux/moduleparam.h> | ||||
| #include <linux/tcp.h> | ||||
|  | @ -79,7 +81,7 @@ static unsigned int help(struct sk_buff *skb, | |||
| 	 */ | ||||
| 	/* AAA = "us", ie. where server normally talks to. */ | ||||
| 	snprintf(buffer, sizeof(buffer), "%u %u", ntohl(newaddr.ip), port); | ||||
| 	pr_debug("nf_nat_irc: inserting '%s' == %pI4, port %u\n", | ||||
| 	pr_debug("inserting '%s' == %pI4, port %u\n", | ||||
| 		 buffer, &newaddr.ip, port); | ||||
| 
 | ||||
| 	if (!nf_nat_mangle_tcp_packet(skb, ct, ctinfo, protoff, matchoff, | ||||
|  | @ -108,8 +110,7 @@ static int __init nf_nat_irc_init(void) | |||
| /* Prior to 2.6.11, we had a ports param.  No longer, but don't break users. */ | ||||
| static int warn_set(const char *val, const struct kernel_param *kp) | ||||
| { | ||||
| 	printk(KERN_INFO KBUILD_MODNAME | ||||
| 	       ": kernel >= 2.6.10 only uses 'ports' for conntrack modules\n"); | ||||
| 	pr_info("kernel >= 2.6.10 only uses 'ports' for conntrack modules\n"); | ||||
| 	return 0; | ||||
| } | ||||
| module_param_call(ports, warn_set, NULL, NULL, 0); | ||||
|  |  | |||
|  | @ -384,9 +384,9 @@ static inline u64 nf_tables_alloc_handle(struct nft_table *table) | |||
| 	return ++table->hgenerator; | ||||
| } | ||||
| 
 | ||||
| static const struct nf_chain_type *chain_type[NFPROTO_NUMPROTO][NFT_CHAIN_T_MAX]; | ||||
| static const struct nft_chain_type *chain_type[NFPROTO_NUMPROTO][NFT_CHAIN_T_MAX]; | ||||
| 
 | ||||
| static const struct nf_chain_type * | ||||
| static const struct nft_chain_type * | ||||
| __nf_tables_chain_type_lookup(const struct nlattr *nla, u8 family) | ||||
| { | ||||
| 	int i; | ||||
|  | @ -399,10 +399,10 @@ __nf_tables_chain_type_lookup(const struct nlattr *nla, u8 family) | |||
| 	return NULL; | ||||
| } | ||||
| 
 | ||||
| static const struct nf_chain_type * | ||||
| static const struct nft_chain_type * | ||||
| nf_tables_chain_type_lookup(const struct nlattr *nla, u8 family, bool autoload) | ||||
| { | ||||
| 	const struct nf_chain_type *type; | ||||
| 	const struct nft_chain_type *type; | ||||
| 
 | ||||
| 	type = __nf_tables_chain_type_lookup(nla, family); | ||||
| 	if (type != NULL) | ||||
|  | @ -859,26 +859,22 @@ static void nf_tables_table_destroy(struct nft_ctx *ctx) | |||
| 	kfree(ctx->table); | ||||
| } | ||||
| 
 | ||||
| int nft_register_chain_type(const struct nf_chain_type *ctype) | ||||
| void nft_register_chain_type(const struct nft_chain_type *ctype) | ||||
| { | ||||
| 	int err = 0; | ||||
| 
 | ||||
| 	if (WARN_ON(ctype->family >= NFPROTO_NUMPROTO)) | ||||
| 		return -EINVAL; | ||||
| 		return; | ||||
| 
 | ||||
| 	nfnl_lock(NFNL_SUBSYS_NFTABLES); | ||||
| 	if (chain_type[ctype->family][ctype->type] != NULL) { | ||||
| 		err = -EBUSY; | ||||
| 		goto out; | ||||
| 	if (WARN_ON(chain_type[ctype->family][ctype->type] != NULL)) { | ||||
| 		nfnl_unlock(NFNL_SUBSYS_NFTABLES); | ||||
| 		return; | ||||
| 	} | ||||
| 	chain_type[ctype->family][ctype->type] = ctype; | ||||
| out: | ||||
| 	nfnl_unlock(NFNL_SUBSYS_NFTABLES); | ||||
| 	return err; | ||||
| } | ||||
| EXPORT_SYMBOL_GPL(nft_register_chain_type); | ||||
| 
 | ||||
| void nft_unregister_chain_type(const struct nf_chain_type *ctype) | ||||
| void nft_unregister_chain_type(const struct nft_chain_type *ctype) | ||||
| { | ||||
| 	nfnl_lock(NFNL_SUBSYS_NFTABLES); | ||||
| 	chain_type[ctype->family][ctype->type] = NULL; | ||||
|  | @ -1215,13 +1211,17 @@ static void nft_chain_stats_replace(struct nft_base_chain *chain, | |||
| 		rcu_assign_pointer(chain->stats, newstats); | ||||
| } | ||||
| 
 | ||||
| static void nf_tables_chain_destroy(struct nft_chain *chain) | ||||
| static void nf_tables_chain_destroy(struct nft_ctx *ctx) | ||||
| { | ||||
| 	struct nft_chain *chain = ctx->chain; | ||||
| 
 | ||||
| 	BUG_ON(chain->use > 0); | ||||
| 
 | ||||
| 	if (nft_is_base_chain(chain)) { | ||||
| 		struct nft_base_chain *basechain = nft_base_chain(chain); | ||||
| 
 | ||||
| 		if (basechain->type->free) | ||||
| 			basechain->type->free(ctx); | ||||
| 		module_put(basechain->type->owner); | ||||
| 		free_percpu(basechain->stats); | ||||
| 		if (basechain->stats) | ||||
|  | @ -1239,7 +1239,7 @@ static void nf_tables_chain_destroy(struct nft_chain *chain) | |||
| struct nft_chain_hook { | ||||
| 	u32				num; | ||||
| 	s32				priority; | ||||
| 	const struct nf_chain_type	*type; | ||||
| 	const struct nft_chain_type	*type; | ||||
| 	struct net_device		*dev; | ||||
| }; | ||||
| 
 | ||||
|  | @ -1249,7 +1249,7 @@ static int nft_chain_parse_hook(struct net *net, | |||
| 				bool create) | ||||
| { | ||||
| 	struct nlattr *ha[NFTA_HOOK_MAX + 1]; | ||||
| 	const struct nf_chain_type *type; | ||||
| 	const struct nft_chain_type *type; | ||||
| 	struct net_device *dev; | ||||
| 	int err; | ||||
| 
 | ||||
|  | @ -1358,6 +1358,9 @@ static int nf_tables_addchain(struct nft_ctx *ctx, u8 family, u8 genmask, | |||
| 		} | ||||
| 
 | ||||
| 		basechain->type = hook.type; | ||||
| 		if (basechain->type->init) | ||||
| 			basechain->type->init(ctx); | ||||
| 
 | ||||
| 		chain = &basechain->chain; | ||||
| 
 | ||||
| 		ops		= &basechain->ops; | ||||
|  | @ -1378,6 +1381,8 @@ static int nf_tables_addchain(struct nft_ctx *ctx, u8 family, u8 genmask, | |||
| 		if (chain == NULL) | ||||
| 			return -ENOMEM; | ||||
| 	} | ||||
| 	ctx->chain = chain; | ||||
| 
 | ||||
| 	INIT_LIST_HEAD(&chain->rules); | ||||
| 	chain->handle = nf_tables_alloc_handle(table); | ||||
| 	chain->table = table; | ||||
|  | @ -1391,7 +1396,6 @@ static int nf_tables_addchain(struct nft_ctx *ctx, u8 family, u8 genmask, | |||
| 	if (err < 0) | ||||
| 		goto err1; | ||||
| 
 | ||||
| 	ctx->chain = chain; | ||||
| 	err = nft_trans_chain_add(ctx, NFT_MSG_NEWCHAIN); | ||||
| 	if (err < 0) | ||||
| 		goto err2; | ||||
|  | @ -1403,7 +1407,7 @@ static int nf_tables_addchain(struct nft_ctx *ctx, u8 family, u8 genmask, | |||
| err2: | ||||
| 	nf_tables_unregister_hook(net, table, chain); | ||||
| err1: | ||||
| 	nf_tables_chain_destroy(chain); | ||||
| 	nf_tables_chain_destroy(ctx); | ||||
| 
 | ||||
| 	return err; | ||||
| } | ||||
|  | @ -2629,11 +2633,11 @@ static struct nft_set *nf_tables_set_lookup_byid(const struct net *net, | |||
| 	return ERR_PTR(-ENOENT); | ||||
| } | ||||
| 
 | ||||
| struct nft_set *nft_set_lookup(const struct net *net, | ||||
| 			       const struct nft_table *table, | ||||
| 			       const struct nlattr *nla_set_name, | ||||
| 			       const struct nlattr *nla_set_id, | ||||
| 			       u8 genmask) | ||||
| struct nft_set *nft_set_lookup_global(const struct net *net, | ||||
| 				      const struct nft_table *table, | ||||
| 				      const struct nlattr *nla_set_name, | ||||
| 				      const struct nlattr *nla_set_id, | ||||
| 				      u8 genmask) | ||||
| { | ||||
| 	struct nft_set *set; | ||||
| 
 | ||||
|  | @ -2646,7 +2650,7 @@ struct nft_set *nft_set_lookup(const struct net *net, | |||
| 	} | ||||
| 	return set; | ||||
| } | ||||
| EXPORT_SYMBOL_GPL(nft_set_lookup); | ||||
| EXPORT_SYMBOL_GPL(nft_set_lookup_global); | ||||
| 
 | ||||
| static int nf_tables_set_alloc_name(struct nft_ctx *ctx, struct nft_set *set, | ||||
| 				    const char *name) | ||||
|  | @ -4028,17 +4032,10 @@ static int nf_tables_newsetelem(struct net *net, struct sock *nlsk, | |||
| 	if (err < 0) | ||||
| 		return err; | ||||
| 
 | ||||
| 	set = nf_tables_set_lookup(ctx.table, nla[NFTA_SET_ELEM_LIST_SET], | ||||
| 				   genmask); | ||||
| 	if (IS_ERR(set)) { | ||||
| 		if (nla[NFTA_SET_ELEM_LIST_SET_ID]) { | ||||
| 			set = nf_tables_set_lookup_byid(net, | ||||
| 					nla[NFTA_SET_ELEM_LIST_SET_ID], | ||||
| 					genmask); | ||||
| 		} | ||||
| 		if (IS_ERR(set)) | ||||
| 			return PTR_ERR(set); | ||||
| 	} | ||||
| 	set = nft_set_lookup_global(net, ctx.table, nla[NFTA_SET_ELEM_LIST_SET], | ||||
| 				    nla[NFTA_SET_ELEM_LIST_SET_ID], genmask); | ||||
| 	if (IS_ERR(set)) | ||||
| 		return PTR_ERR(set); | ||||
| 
 | ||||
| 	if (!list_empty(&set->bindings) && set->flags & NFT_SET_CONSTANT) | ||||
| 		return -EBUSY; | ||||
|  | @ -4328,9 +4325,9 @@ struct nft_object *nf_tables_obj_lookup(const struct nft_table *table, | |||
| } | ||||
| EXPORT_SYMBOL_GPL(nf_tables_obj_lookup); | ||||
| 
 | ||||
| struct nft_object *nf_tables_obj_lookup_byhandle(const struct nft_table *table, | ||||
| 						 const struct nlattr *nla, | ||||
| 						 u32 objtype, u8 genmask) | ||||
| static struct nft_object *nf_tables_obj_lookup_byhandle(const struct nft_table *table, | ||||
| 							const struct nlattr *nla, | ||||
| 							u32 objtype, u8 genmask) | ||||
| { | ||||
| 	struct nft_object *obj; | ||||
| 
 | ||||
|  | @ -4357,16 +4354,20 @@ static struct nft_object *nft_obj_init(const struct nft_ctx *ctx, | |||
| 				       const struct nft_object_type *type, | ||||
| 				       const struct nlattr *attr) | ||||
| { | ||||
| 	struct nlattr *tb[type->maxattr + 1]; | ||||
| 	struct nlattr **tb; | ||||
| 	const struct nft_object_ops *ops; | ||||
| 	struct nft_object *obj; | ||||
| 	int err; | ||||
| 	int err = -ENOMEM; | ||||
| 
 | ||||
| 	tb = kmalloc_array(type->maxattr + 1, sizeof(*tb), GFP_KERNEL); | ||||
| 	if (!tb) | ||||
| 		goto err1; | ||||
| 
 | ||||
| 	if (attr) { | ||||
| 		err = nla_parse_nested(tb, type->maxattr, attr, type->policy, | ||||
| 				       NULL); | ||||
| 		if (err < 0) | ||||
| 			goto err1; | ||||
| 			goto err2; | ||||
| 	} else { | ||||
| 		memset(tb, 0, sizeof(tb[0]) * (type->maxattr + 1)); | ||||
| 	} | ||||
|  | @ -4375,7 +4376,7 @@ static struct nft_object *nft_obj_init(const struct nft_ctx *ctx, | |||
| 		ops = type->select_ops(ctx, (const struct nlattr * const *)tb); | ||||
| 		if (IS_ERR(ops)) { | ||||
| 			err = PTR_ERR(ops); | ||||
| 			goto err1; | ||||
| 			goto err2; | ||||
| 		} | ||||
| 	} else { | ||||
| 		ops = type->ops; | ||||
|  | @ -4383,18 +4384,21 @@ static struct nft_object *nft_obj_init(const struct nft_ctx *ctx, | |||
| 
 | ||||
| 	err = -ENOMEM; | ||||
| 	obj = kzalloc(sizeof(*obj) + ops->size, GFP_KERNEL); | ||||
| 	if (obj == NULL) | ||||
| 		goto err1; | ||||
| 	if (!obj) | ||||
| 		goto err2; | ||||
| 
 | ||||
| 	err = ops->init(ctx, (const struct nlattr * const *)tb, obj); | ||||
| 	if (err < 0) | ||||
| 		goto err2; | ||||
| 		goto err3; | ||||
| 
 | ||||
| 	obj->ops = ops; | ||||
| 
 | ||||
| 	kfree(tb); | ||||
| 	return obj; | ||||
| err2: | ||||
| err3: | ||||
| 	kfree(obj); | ||||
| err2: | ||||
| 	kfree(tb); | ||||
| err1: | ||||
| 	return ERR_PTR(err); | ||||
| } | ||||
|  | @ -4850,7 +4854,7 @@ struct nft_flowtable *nf_tables_flowtable_lookup(const struct nft_table *table, | |||
| } | ||||
| EXPORT_SYMBOL_GPL(nf_tables_flowtable_lookup); | ||||
| 
 | ||||
| struct nft_flowtable * | ||||
| static struct nft_flowtable * | ||||
| nf_tables_flowtable_lookup_byhandle(const struct nft_table *table, | ||||
| 				    const struct nlattr *nla, u8 genmask) | ||||
| { | ||||
|  | @ -5697,7 +5701,7 @@ static void nf_tables_commit_release(struct nft_trans *trans) | |||
| 		nf_tables_table_destroy(&trans->ctx); | ||||
| 		break; | ||||
| 	case NFT_MSG_DELCHAIN: | ||||
| 		nf_tables_chain_destroy(trans->ctx.chain); | ||||
| 		nf_tables_chain_destroy(&trans->ctx); | ||||
| 		break; | ||||
| 	case NFT_MSG_DELRULE: | ||||
| 		nf_tables_rule_destroy(&trans->ctx, nft_trans_rule(trans)); | ||||
|  | @ -5868,7 +5872,7 @@ static void nf_tables_abort_release(struct nft_trans *trans) | |||
| 		nf_tables_table_destroy(&trans->ctx); | ||||
| 		break; | ||||
| 	case NFT_MSG_NEWCHAIN: | ||||
| 		nf_tables_chain_destroy(trans->ctx.chain); | ||||
| 		nf_tables_chain_destroy(&trans->ctx); | ||||
| 		break; | ||||
| 	case NFT_MSG_NEWRULE: | ||||
| 		nf_tables_rule_destroy(&trans->ctx, nft_trans_rule(trans)); | ||||
|  | @ -6015,7 +6019,7 @@ static const struct nfnetlink_subsystem nf_tables_subsys = { | |||
| }; | ||||
| 
 | ||||
| int nft_chain_validate_dependency(const struct nft_chain *chain, | ||||
| 				  enum nft_chain_type type) | ||||
| 				  enum nft_chain_types type) | ||||
| { | ||||
| 	const struct nft_base_chain *basechain; | ||||
| 
 | ||||
|  | @ -6518,7 +6522,7 @@ int __nft_release_basechain(struct nft_ctx *ctx) | |||
| 	} | ||||
| 	list_del(&ctx->chain->list); | ||||
| 	ctx->table->use--; | ||||
| 	nf_tables_chain_destroy(ctx->chain); | ||||
| 	nf_tables_chain_destroy(ctx); | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
|  | @ -6534,6 +6538,7 @@ static void __nft_release_tables(struct net *net) | |||
| 	struct nft_set *set, *ns; | ||||
| 	struct nft_ctx ctx = { | ||||
| 		.net	= net, | ||||
| 		.family	= NFPROTO_NETDEV, | ||||
| 	}; | ||||
| 
 | ||||
| 	list_for_each_entry_safe(table, nt, &net->nft.tables, list) { | ||||
|  | @ -6570,9 +6575,10 @@ static void __nft_release_tables(struct net *net) | |||
| 			nft_obj_destroy(obj); | ||||
| 		} | ||||
| 		list_for_each_entry_safe(chain, nc, &table->chains, list) { | ||||
| 			ctx.chain = chain; | ||||
| 			list_del(&chain->list); | ||||
| 			table->use--; | ||||
| 			nf_tables_chain_destroy(chain); | ||||
| 			nf_tables_chain_destroy(&ctx); | ||||
| 		} | ||||
| 		list_del(&table->list); | ||||
| 		nf_tables_table_destroy(&ctx); | ||||
|  | @ -6603,6 +6609,8 @@ static int __init nf_tables_module_init(void) | |||
| { | ||||
| 	int err; | ||||
| 
 | ||||
| 	nft_chain_filter_init(); | ||||
| 
 | ||||
| 	info = kmalloc(sizeof(struct nft_expr_info) * NFT_RULE_MAXEXPRS, | ||||
| 		       GFP_KERNEL); | ||||
| 	if (info == NULL) { | ||||
|  | @ -6637,6 +6645,7 @@ static void __exit nf_tables_module_exit(void) | |||
| 	rcu_barrier(); | ||||
| 	nf_tables_core_module_exit(); | ||||
| 	kfree(info); | ||||
| 	nft_chain_filter_fini(); | ||||
| } | ||||
| 
 | ||||
| module_init(nf_tables_module_init); | ||||
|  |  | |||
|  | @ -1,75 +0,0 @@ | |||
| /*
 | ||||
|  * Copyright (c) 2012-2014 Patrick McHardy <kaber@trash.net> | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 as | ||||
|  * published by the Free Software Foundation. | ||||
|  */ | ||||
| 
 | ||||
| #include <linux/init.h> | ||||
| #include <linux/module.h> | ||||
| #include <linux/ip.h> | ||||
| #include <linux/ipv6.h> | ||||
| #include <linux/netfilter_ipv4.h> | ||||
| #include <linux/netfilter_ipv6.h> | ||||
| #include <net/netfilter/nf_tables.h> | ||||
| #include <net/netfilter/nf_tables_ipv4.h> | ||||
| #include <net/netfilter/nf_tables_ipv6.h> | ||||
| #include <net/ip.h> | ||||
| 
 | ||||
| static unsigned int nft_do_chain_inet(void *priv, struct sk_buff *skb, | ||||
| 				      const struct nf_hook_state *state) | ||||
| { | ||||
| 	struct nft_pktinfo pkt; | ||||
| 
 | ||||
| 	nft_set_pktinfo(&pkt, skb, state); | ||||
| 
 | ||||
| 	switch (state->pf) { | ||||
| 	case NFPROTO_IPV4: | ||||
| 		nft_set_pktinfo_ipv4(&pkt, skb); | ||||
| 		break; | ||||
| 	case NFPROTO_IPV6: | ||||
| 		nft_set_pktinfo_ipv6(&pkt, skb); | ||||
| 		break; | ||||
| 	default: | ||||
| 		break; | ||||
| 	} | ||||
| 
 | ||||
| 	return nft_do_chain(&pkt, priv); | ||||
| } | ||||
| 
 | ||||
| static const struct nf_chain_type filter_inet = { | ||||
| 	.name		= "filter", | ||||
| 	.type		= NFT_CHAIN_T_DEFAULT, | ||||
| 	.family		= NFPROTO_INET, | ||||
| 	.owner		= THIS_MODULE, | ||||
| 	.hook_mask	= (1 << NF_INET_LOCAL_IN) | | ||||
| 			  (1 << NF_INET_LOCAL_OUT) | | ||||
| 			  (1 << NF_INET_FORWARD) | | ||||
| 			  (1 << NF_INET_PRE_ROUTING) | | ||||
| 			  (1 << NF_INET_POST_ROUTING), | ||||
| 	.hooks		= { | ||||
| 		[NF_INET_LOCAL_IN]	= nft_do_chain_inet, | ||||
| 		[NF_INET_LOCAL_OUT]	= nft_do_chain_inet, | ||||
| 		[NF_INET_FORWARD]	= nft_do_chain_inet, | ||||
| 		[NF_INET_PRE_ROUTING]	= nft_do_chain_inet, | ||||
| 		[NF_INET_POST_ROUTING]	= nft_do_chain_inet, | ||||
|         }, | ||||
| }; | ||||
| 
 | ||||
| static int __init nf_tables_inet_init(void) | ||||
| { | ||||
| 	return nft_register_chain_type(&filter_inet); | ||||
| } | ||||
| 
 | ||||
| static void __exit nf_tables_inet_exit(void) | ||||
| { | ||||
| 	nft_unregister_chain_type(&filter_inet); | ||||
| } | ||||
| 
 | ||||
| module_init(nf_tables_inet_init); | ||||
| module_exit(nf_tables_inet_exit); | ||||
| 
 | ||||
| MODULE_LICENSE("GPL"); | ||||
| MODULE_AUTHOR("Patrick McHardy <kaber@trash.net>"); | ||||
| MODULE_ALIAS_NFT_CHAIN(1, "filter"); | ||||
|  | @ -1,142 +0,0 @@ | |||
| /*
 | ||||
|  * Copyright (c) 2015 Pablo Neira Ayuso <pablo@netfilter.org> | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 as | ||||
|  * published by the Free Software Foundation. | ||||
|  */ | ||||
| 
 | ||||
| #include <linux/init.h> | ||||
| #include <linux/module.h> | ||||
| #include <linux/netdevice.h> | ||||
| #include <net/netfilter/nf_tables.h> | ||||
| #include <linux/ip.h> | ||||
| #include <linux/ipv6.h> | ||||
| #include <net/netfilter/nf_tables_ipv4.h> | ||||
| #include <net/netfilter/nf_tables_ipv6.h> | ||||
| 
 | ||||
| static unsigned int | ||||
| nft_do_chain_netdev(void *priv, struct sk_buff *skb, | ||||
| 		    const struct nf_hook_state *state) | ||||
| { | ||||
| 	struct nft_pktinfo pkt; | ||||
| 
 | ||||
| 	nft_set_pktinfo(&pkt, skb, state); | ||||
| 
 | ||||
| 	switch (skb->protocol) { | ||||
| 	case htons(ETH_P_IP): | ||||
| 		nft_set_pktinfo_ipv4_validate(&pkt, skb); | ||||
| 		break; | ||||
| 	case htons(ETH_P_IPV6): | ||||
| 		nft_set_pktinfo_ipv6_validate(&pkt, skb); | ||||
| 		break; | ||||
| 	default: | ||||
| 		nft_set_pktinfo_unspec(&pkt, skb); | ||||
| 		break; | ||||
| 	} | ||||
| 
 | ||||
| 	return nft_do_chain(&pkt, priv); | ||||
| } | ||||
| 
 | ||||
| static const struct nf_chain_type nft_filter_chain_netdev = { | ||||
| 	.name		= "filter", | ||||
| 	.type		= NFT_CHAIN_T_DEFAULT, | ||||
| 	.family		= NFPROTO_NETDEV, | ||||
| 	.owner		= THIS_MODULE, | ||||
| 	.hook_mask	= (1 << NF_NETDEV_INGRESS), | ||||
| 	.hooks		= { | ||||
| 		[NF_NETDEV_INGRESS]	= nft_do_chain_netdev, | ||||
| 	}, | ||||
| }; | ||||
| 
 | ||||
| static void nft_netdev_event(unsigned long event, struct net_device *dev, | ||||
| 			     struct nft_ctx *ctx) | ||||
| { | ||||
| 	struct nft_base_chain *basechain = nft_base_chain(ctx->chain); | ||||
| 
 | ||||
| 	switch (event) { | ||||
| 	case NETDEV_UNREGISTER: | ||||
| 		if (strcmp(basechain->dev_name, dev->name) != 0) | ||||
| 			return; | ||||
| 
 | ||||
| 		__nft_release_basechain(ctx); | ||||
| 		break; | ||||
| 	case NETDEV_CHANGENAME: | ||||
| 		if (dev->ifindex != basechain->ops.dev->ifindex) | ||||
| 			return; | ||||
| 
 | ||||
| 		strncpy(basechain->dev_name, dev->name, IFNAMSIZ); | ||||
| 		break; | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| static int nf_tables_netdev_event(struct notifier_block *this, | ||||
| 				  unsigned long event, void *ptr) | ||||
| { | ||||
| 	struct net_device *dev = netdev_notifier_info_to_dev(ptr); | ||||
| 	struct nft_table *table; | ||||
| 	struct nft_chain *chain, *nr; | ||||
| 	struct nft_ctx ctx = { | ||||
| 		.net	= dev_net(dev), | ||||
| 	}; | ||||
| 
 | ||||
| 	if (event != NETDEV_UNREGISTER && | ||||
| 	    event != NETDEV_CHANGENAME) | ||||
| 		return NOTIFY_DONE; | ||||
| 
 | ||||
| 	nfnl_lock(NFNL_SUBSYS_NFTABLES); | ||||
| 	list_for_each_entry(table, &ctx.net->nft.tables, list) { | ||||
| 		if (table->family != NFPROTO_NETDEV) | ||||
| 			continue; | ||||
| 
 | ||||
| 		ctx.family = table->family; | ||||
| 		ctx.table = table; | ||||
| 		list_for_each_entry_safe(chain, nr, &table->chains, list) { | ||||
| 			if (!nft_is_base_chain(chain)) | ||||
| 				continue; | ||||
| 
 | ||||
| 			ctx.chain = chain; | ||||
| 			nft_netdev_event(event, dev, &ctx); | ||||
| 		} | ||||
| 	} | ||||
| 	nfnl_unlock(NFNL_SUBSYS_NFTABLES); | ||||
| 
 | ||||
| 	return NOTIFY_DONE; | ||||
| } | ||||
| 
 | ||||
| static struct notifier_block nf_tables_netdev_notifier = { | ||||
| 	.notifier_call	= nf_tables_netdev_event, | ||||
| }; | ||||
| 
 | ||||
| static int __init nf_tables_netdev_init(void) | ||||
| { | ||||
| 	int ret; | ||||
| 
 | ||||
| 	ret = nft_register_chain_type(&nft_filter_chain_netdev); | ||||
| 	if (ret) | ||||
| 		return ret; | ||||
| 
 | ||||
| 	ret = register_netdevice_notifier(&nf_tables_netdev_notifier); | ||||
| 	if (ret) | ||||
| 		goto err_register_netdevice_notifier; | ||||
| 
 | ||||
| 	return 0; | ||||
| 
 | ||||
| err_register_netdevice_notifier: | ||||
| 	nft_unregister_chain_type(&nft_filter_chain_netdev); | ||||
| 
 | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
| static void __exit nf_tables_netdev_exit(void) | ||||
| { | ||||
| 	unregister_netdevice_notifier(&nf_tables_netdev_notifier); | ||||
| 	nft_unregister_chain_type(&nft_filter_chain_netdev); | ||||
| } | ||||
| 
 | ||||
| module_init(nf_tables_netdev_init); | ||||
| module_exit(nf_tables_netdev_exit); | ||||
| 
 | ||||
| MODULE_LICENSE("GPL"); | ||||
| MODULE_AUTHOR("Pablo Neira Ayuso <pablo@netfilter.org>"); | ||||
| MODULE_ALIAS_NFT_CHAIN(5, "filter"); /* NFPROTO_NETDEV */ | ||||
|  | @ -467,8 +467,7 @@ static void nfnl_overquota_report(struct net *net, struct nf_acct *nfacct) | |||
| 			  GFP_ATOMIC); | ||||
| } | ||||
| 
 | ||||
| int nfnl_acct_overquota(struct net *net, const struct sk_buff *skb, | ||||
| 			struct nf_acct *nfacct) | ||||
| int nfnl_acct_overquota(struct net *net, struct nf_acct *nfacct) | ||||
| { | ||||
| 	u64 now; | ||||
| 	u64 *quota; | ||||
|  |  | |||
|  | @ -314,23 +314,30 @@ nfnl_cthelper_update_policy_one(const struct nf_conntrack_expect_policy *policy, | |||
| static int nfnl_cthelper_update_policy_all(struct nlattr *tb[], | ||||
| 					   struct nf_conntrack_helper *helper) | ||||
| { | ||||
| 	struct nf_conntrack_expect_policy new_policy[helper->expect_class_max + 1]; | ||||
| 	struct nf_conntrack_expect_policy *new_policy; | ||||
| 	struct nf_conntrack_expect_policy *policy; | ||||
| 	int i, err; | ||||
| 	int i, ret = 0; | ||||
| 
 | ||||
| 	new_policy = kmalloc_array(helper->expect_class_max + 1, | ||||
| 				   sizeof(*new_policy), GFP_KERNEL); | ||||
| 	if (!new_policy) | ||||
| 		return -ENOMEM; | ||||
| 
 | ||||
| 	/* Check first that all policy attributes are well-formed, so we don't
 | ||||
| 	 * leave things in inconsistent state on errors. | ||||
| 	 */ | ||||
| 	for (i = 0; i < helper->expect_class_max + 1; i++) { | ||||
| 
 | ||||
| 		if (!tb[NFCTH_POLICY_SET + i]) | ||||
| 			return -EINVAL; | ||||
| 		if (!tb[NFCTH_POLICY_SET + i]) { | ||||
| 			ret = -EINVAL; | ||||
| 			goto err; | ||||
| 		} | ||||
| 
 | ||||
| 		err = nfnl_cthelper_update_policy_one(&helper->expect_policy[i], | ||||
| 		ret = nfnl_cthelper_update_policy_one(&helper->expect_policy[i], | ||||
| 						      &new_policy[i], | ||||
| 						      tb[NFCTH_POLICY_SET + i]); | ||||
| 		if (err < 0) | ||||
| 			return err; | ||||
| 		if (ret < 0) | ||||
| 			goto err; | ||||
| 	} | ||||
| 	/* Now we can safely update them. */ | ||||
| 	for (i = 0; i < helper->expect_class_max + 1; i++) { | ||||
|  | @ -340,7 +347,9 @@ static int nfnl_cthelper_update_policy_all(struct nlattr *tb[], | |||
| 		policy->timeout	= new_policy->timeout; | ||||
| 	} | ||||
| 
 | ||||
| 	return 0; | ||||
| err: | ||||
| 	kfree(new_policy); | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
| static int nfnl_cthelper_update_policy(struct nf_conntrack_helper *helper, | ||||
|  |  | |||
|  | @ -51,19 +51,27 @@ ctnl_timeout_parse_policy(void *timeouts, | |||
| 			  const struct nf_conntrack_l4proto *l4proto, | ||||
| 			  struct net *net, const struct nlattr *attr) | ||||
| { | ||||
| 	struct nlattr **tb; | ||||
| 	int ret = 0; | ||||
| 
 | ||||
| 	if (likely(l4proto->ctnl_timeout.nlattr_to_obj)) { | ||||
| 		struct nlattr *tb[l4proto->ctnl_timeout.nlattr_max+1]; | ||||
| 	if (!l4proto->ctnl_timeout.nlattr_to_obj) | ||||
| 		return 0; | ||||
| 
 | ||||
| 		ret = nla_parse_nested(tb, l4proto->ctnl_timeout.nlattr_max, | ||||
| 				       attr, l4proto->ctnl_timeout.nla_policy, | ||||
| 				       NULL); | ||||
| 		if (ret < 0) | ||||
| 			return ret; | ||||
| 	tb = kcalloc(l4proto->ctnl_timeout.nlattr_max + 1, sizeof(*tb), | ||||
| 		     GFP_KERNEL); | ||||
| 
 | ||||
| 		ret = l4proto->ctnl_timeout.nlattr_to_obj(tb, net, timeouts); | ||||
| 	} | ||||
| 	if (!tb) | ||||
| 		return -ENOMEM; | ||||
| 
 | ||||
| 	ret = nla_parse_nested(tb, l4proto->ctnl_timeout.nlattr_max, attr, | ||||
| 			       l4proto->ctnl_timeout.nla_policy, NULL); | ||||
| 	if (ret < 0) | ||||
| 		goto err; | ||||
| 
 | ||||
| 	ret = l4proto->ctnl_timeout.nlattr_to_obj(tb, net, timeouts); | ||||
| 
 | ||||
| err: | ||||
| 	kfree(tb); | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
|  |  | |||
|  | @ -14,6 +14,9 @@ | |||
|  * published by the Free Software Foundation. | ||||
|  * | ||||
|  */ | ||||
| 
 | ||||
| #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt | ||||
| 
 | ||||
| #include <linux/module.h> | ||||
| #include <linux/skbuff.h> | ||||
| #include <linux/init.h> | ||||
|  | @ -1533,20 +1536,20 @@ static int __init nfnetlink_queue_init(void) | |||
| 
 | ||||
| 	status = register_pernet_subsys(&nfnl_queue_net_ops); | ||||
| 	if (status < 0) { | ||||
| 		pr_err("nf_queue: failed to register pernet ops\n"); | ||||
| 		pr_err("failed to register pernet ops\n"); | ||||
| 		goto out; | ||||
| 	} | ||||
| 
 | ||||
| 	netlink_register_notifier(&nfqnl_rtnl_notifier); | ||||
| 	status = nfnetlink_subsys_register(&nfqnl_subsys); | ||||
| 	if (status < 0) { | ||||
| 		pr_err("nf_queue: failed to create netlink socket\n"); | ||||
| 		pr_err("failed to create netlink socket\n"); | ||||
| 		goto cleanup_netlink_notifier; | ||||
| 	} | ||||
| 
 | ||||
| 	status = register_netdevice_notifier(&nfqnl_dev_notifier); | ||||
| 	if (status < 0) { | ||||
| 		pr_err("nf_queue: failed to register netdevice notifier\n"); | ||||
| 		pr_err("failed to register netdevice notifier\n"); | ||||
| 		goto cleanup_netlink_subsys; | ||||
| 	} | ||||
| 
 | ||||
|  |  | |||
							
								
								
									
										398
									
								
								net/netfilter/nft_chain_filter.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										398
									
								
								net/netfilter/nft_chain_filter.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,398 @@ | |||
| #include <linux/init.h> | ||||
| #include <linux/kernel.h> | ||||
| #include <linux/netdevice.h> | ||||
| #include <net/net_namespace.h> | ||||
| #include <net/netfilter/nf_tables.h> | ||||
| #include <linux/netfilter_ipv4.h> | ||||
| #include <linux/netfilter_ipv6.h> | ||||
| #include <linux/netfilter_bridge.h> | ||||
| #include <linux/netfilter_arp.h> | ||||
| #include <net/netfilter/nf_tables_ipv4.h> | ||||
| #include <net/netfilter/nf_tables_ipv6.h> | ||||
| 
 | ||||
| #ifdef CONFIG_NF_TABLES_IPV4 | ||||
| static unsigned int nft_do_chain_ipv4(void *priv, | ||||
| 				      struct sk_buff *skb, | ||||
| 				      const struct nf_hook_state *state) | ||||
| { | ||||
| 	struct nft_pktinfo pkt; | ||||
| 
 | ||||
| 	nft_set_pktinfo(&pkt, skb, state); | ||||
| 	nft_set_pktinfo_ipv4(&pkt, skb); | ||||
| 
 | ||||
| 	return nft_do_chain(&pkt, priv); | ||||
| } | ||||
| 
 | ||||
| static const struct nft_chain_type nft_chain_filter_ipv4 = { | ||||
| 	.name		= "filter", | ||||
| 	.type		= NFT_CHAIN_T_DEFAULT, | ||||
| 	.family		= NFPROTO_IPV4, | ||||
| 	.hook_mask	= (1 << NF_INET_LOCAL_IN) | | ||||
| 			  (1 << NF_INET_LOCAL_OUT) | | ||||
| 			  (1 << NF_INET_FORWARD) | | ||||
| 			  (1 << NF_INET_PRE_ROUTING) | | ||||
| 			  (1 << NF_INET_POST_ROUTING), | ||||
| 	.hooks		= { | ||||
| 		[NF_INET_LOCAL_IN]	= nft_do_chain_ipv4, | ||||
| 		[NF_INET_LOCAL_OUT]	= nft_do_chain_ipv4, | ||||
| 		[NF_INET_FORWARD]	= nft_do_chain_ipv4, | ||||
| 		[NF_INET_PRE_ROUTING]	= nft_do_chain_ipv4, | ||||
| 		[NF_INET_POST_ROUTING]	= nft_do_chain_ipv4, | ||||
| 	}, | ||||
| }; | ||||
| 
 | ||||
| static void nft_chain_filter_ipv4_init(void) | ||||
| { | ||||
| 	nft_register_chain_type(&nft_chain_filter_ipv4); | ||||
| } | ||||
| static void nft_chain_filter_ipv4_fini(void) | ||||
| { | ||||
| 	nft_unregister_chain_type(&nft_chain_filter_ipv4); | ||||
| } | ||||
| 
 | ||||
| #else | ||||
| static inline void nft_chain_filter_ipv4_init(void) {} | ||||
| static inline void nft_chain_filter_ipv4_fini(void) {} | ||||
| #endif /* CONFIG_NF_TABLES_IPV4 */ | ||||
| 
 | ||||
| #ifdef CONFIG_NF_TABLES_ARP | ||||
| static unsigned int nft_do_chain_arp(void *priv, struct sk_buff *skb, | ||||
| 				     const struct nf_hook_state *state) | ||||
| { | ||||
| 	struct nft_pktinfo pkt; | ||||
| 
 | ||||
| 	nft_set_pktinfo(&pkt, skb, state); | ||||
| 	nft_set_pktinfo_unspec(&pkt, skb); | ||||
| 
 | ||||
| 	return nft_do_chain(&pkt, priv); | ||||
| } | ||||
| 
 | ||||
| static const struct nft_chain_type nft_chain_filter_arp = { | ||||
| 	.name		= "filter", | ||||
| 	.type		= NFT_CHAIN_T_DEFAULT, | ||||
| 	.family		= NFPROTO_ARP, | ||||
| 	.owner		= THIS_MODULE, | ||||
| 	.hook_mask	= (1 << NF_ARP_IN) | | ||||
| 			  (1 << NF_ARP_OUT), | ||||
| 	.hooks		= { | ||||
| 		[NF_ARP_IN]		= nft_do_chain_arp, | ||||
| 		[NF_ARP_OUT]		= nft_do_chain_arp, | ||||
| 	}, | ||||
| }; | ||||
| 
 | ||||
| static void nft_chain_filter_arp_init(void) | ||||
| { | ||||
| 	nft_register_chain_type(&nft_chain_filter_arp); | ||||
| } | ||||
| 
 | ||||
| static void nft_chain_filter_arp_fini(void) | ||||
| { | ||||
| 	nft_unregister_chain_type(&nft_chain_filter_arp); | ||||
| } | ||||
| #else | ||||
| static inline void nft_chain_filter_arp_init(void) {} | ||||
| static inline void nft_chain_filter_arp_fini(void) {} | ||||
| #endif /* CONFIG_NF_TABLES_ARP */ | ||||
| 
 | ||||
| #ifdef CONFIG_NF_TABLES_IPV6 | ||||
| static unsigned int nft_do_chain_ipv6(void *priv, | ||||
| 				      struct sk_buff *skb, | ||||
| 				      const struct nf_hook_state *state) | ||||
| { | ||||
| 	struct nft_pktinfo pkt; | ||||
| 
 | ||||
| 	nft_set_pktinfo(&pkt, skb, state); | ||||
| 	nft_set_pktinfo_ipv6(&pkt, skb); | ||||
| 
 | ||||
| 	return nft_do_chain(&pkt, priv); | ||||
| } | ||||
| 
 | ||||
| static const struct nft_chain_type nft_chain_filter_ipv6 = { | ||||
| 	.name		= "filter", | ||||
| 	.type		= NFT_CHAIN_T_DEFAULT, | ||||
| 	.family		= NFPROTO_IPV6, | ||||
| 	.hook_mask	= (1 << NF_INET_LOCAL_IN) | | ||||
| 			  (1 << NF_INET_LOCAL_OUT) | | ||||
| 			  (1 << NF_INET_FORWARD) | | ||||
| 			  (1 << NF_INET_PRE_ROUTING) | | ||||
| 			  (1 << NF_INET_POST_ROUTING), | ||||
| 	.hooks		= { | ||||
| 		[NF_INET_LOCAL_IN]	= nft_do_chain_ipv6, | ||||
| 		[NF_INET_LOCAL_OUT]	= nft_do_chain_ipv6, | ||||
| 		[NF_INET_FORWARD]	= nft_do_chain_ipv6, | ||||
| 		[NF_INET_PRE_ROUTING]	= nft_do_chain_ipv6, | ||||
| 		[NF_INET_POST_ROUTING]	= nft_do_chain_ipv6, | ||||
| 	}, | ||||
| }; | ||||
| 
 | ||||
| static void nft_chain_filter_ipv6_init(void) | ||||
| { | ||||
| 	nft_register_chain_type(&nft_chain_filter_ipv6); | ||||
| } | ||||
| 
 | ||||
| static void nft_chain_filter_ipv6_fini(void) | ||||
| { | ||||
| 	nft_unregister_chain_type(&nft_chain_filter_ipv6); | ||||
| } | ||||
| #else | ||||
| static inline void nft_chain_filter_ipv6_init(void) {} | ||||
| static inline void nft_chain_filter_ipv6_fini(void) {} | ||||
| #endif /* CONFIG_NF_TABLES_IPV6 */ | ||||
| 
 | ||||
| #ifdef CONFIG_NF_TABLES_INET | ||||
| static unsigned int nft_do_chain_inet(void *priv, struct sk_buff *skb, | ||||
| 				      const struct nf_hook_state *state) | ||||
| { | ||||
| 	struct nft_pktinfo pkt; | ||||
| 
 | ||||
| 	nft_set_pktinfo(&pkt, skb, state); | ||||
| 
 | ||||
| 	switch (state->pf) { | ||||
| 	case NFPROTO_IPV4: | ||||
| 		nft_set_pktinfo_ipv4(&pkt, skb); | ||||
| 		break; | ||||
| 	case NFPROTO_IPV6: | ||||
| 		nft_set_pktinfo_ipv6(&pkt, skb); | ||||
| 		break; | ||||
| 	default: | ||||
| 		break; | ||||
| 	} | ||||
| 
 | ||||
| 	return nft_do_chain(&pkt, priv); | ||||
| } | ||||
| 
 | ||||
| static const struct nft_chain_type nft_chain_filter_inet = { | ||||
| 	.name		= "filter", | ||||
| 	.type		= NFT_CHAIN_T_DEFAULT, | ||||
| 	.family		= NFPROTO_INET, | ||||
| 	.hook_mask	= (1 << NF_INET_LOCAL_IN) | | ||||
| 			  (1 << NF_INET_LOCAL_OUT) | | ||||
| 			  (1 << NF_INET_FORWARD) | | ||||
| 			  (1 << NF_INET_PRE_ROUTING) | | ||||
| 			  (1 << NF_INET_POST_ROUTING), | ||||
| 	.hooks		= { | ||||
| 		[NF_INET_LOCAL_IN]	= nft_do_chain_inet, | ||||
| 		[NF_INET_LOCAL_OUT]	= nft_do_chain_inet, | ||||
| 		[NF_INET_FORWARD]	= nft_do_chain_inet, | ||||
| 		[NF_INET_PRE_ROUTING]	= nft_do_chain_inet, | ||||
| 		[NF_INET_POST_ROUTING]	= nft_do_chain_inet, | ||||
|         }, | ||||
| }; | ||||
| 
 | ||||
| static void nft_chain_filter_inet_init(void) | ||||
| { | ||||
| 	nft_register_chain_type(&nft_chain_filter_inet); | ||||
| } | ||||
| 
 | ||||
| static void nft_chain_filter_inet_fini(void) | ||||
| { | ||||
| 	nft_unregister_chain_type(&nft_chain_filter_inet); | ||||
| } | ||||
| #else | ||||
| static inline void nft_chain_filter_inet_init(void) {} | ||||
| static inline void nft_chain_filter_inet_fini(void) {} | ||||
| #endif /* CONFIG_NF_TABLES_IPV6 */ | ||||
| 
 | ||||
| #ifdef CONFIG_NF_TABLES_BRIDGE | ||||
| static unsigned int | ||||
| nft_do_chain_bridge(void *priv, | ||||
| 		    struct sk_buff *skb, | ||||
| 		    const struct nf_hook_state *state) | ||||
| { | ||||
| 	struct nft_pktinfo pkt; | ||||
| 
 | ||||
| 	nft_set_pktinfo(&pkt, skb, state); | ||||
| 
 | ||||
| 	switch (eth_hdr(skb)->h_proto) { | ||||
| 	case htons(ETH_P_IP): | ||||
| 		nft_set_pktinfo_ipv4_validate(&pkt, skb); | ||||
| 		break; | ||||
| 	case htons(ETH_P_IPV6): | ||||
| 		nft_set_pktinfo_ipv6_validate(&pkt, skb); | ||||
| 		break; | ||||
| 	default: | ||||
| 		nft_set_pktinfo_unspec(&pkt, skb); | ||||
| 		break; | ||||
| 	} | ||||
| 
 | ||||
| 	return nft_do_chain(&pkt, priv); | ||||
| } | ||||
| 
 | ||||
| static const struct nft_chain_type nft_chain_filter_bridge = { | ||||
| 	.name		= "filter", | ||||
| 	.type		= NFT_CHAIN_T_DEFAULT, | ||||
| 	.family		= NFPROTO_BRIDGE, | ||||
| 	.hook_mask	= (1 << NF_BR_PRE_ROUTING) | | ||||
| 			  (1 << NF_BR_LOCAL_IN) | | ||||
| 			  (1 << NF_BR_FORWARD) | | ||||
| 			  (1 << NF_BR_LOCAL_OUT) | | ||||
| 			  (1 << NF_BR_POST_ROUTING), | ||||
| 	.hooks		= { | ||||
| 		[NF_BR_PRE_ROUTING]	= nft_do_chain_bridge, | ||||
| 		[NF_BR_LOCAL_IN]	= nft_do_chain_bridge, | ||||
| 		[NF_BR_FORWARD]		= nft_do_chain_bridge, | ||||
| 		[NF_BR_LOCAL_OUT]	= nft_do_chain_bridge, | ||||
| 		[NF_BR_POST_ROUTING]	= nft_do_chain_bridge, | ||||
| 	}, | ||||
| }; | ||||
| 
 | ||||
| static void nft_chain_filter_bridge_init(void) | ||||
| { | ||||
| 	nft_register_chain_type(&nft_chain_filter_bridge); | ||||
| } | ||||
| 
 | ||||
| static void nft_chain_filter_bridge_fini(void) | ||||
| { | ||||
| 	nft_unregister_chain_type(&nft_chain_filter_bridge); | ||||
| } | ||||
| #else | ||||
| static inline void nft_chain_filter_bridge_init(void) {} | ||||
| static inline void nft_chain_filter_bridge_fini(void) {} | ||||
| #endif /* CONFIG_NF_TABLES_BRIDGE */ | ||||
| 
 | ||||
| #ifdef CONFIG_NF_TABLES_NETDEV | ||||
| static unsigned int nft_do_chain_netdev(void *priv, struct sk_buff *skb, | ||||
| 					const struct nf_hook_state *state) | ||||
| { | ||||
| 	struct nft_pktinfo pkt; | ||||
| 
 | ||||
| 	nft_set_pktinfo(&pkt, skb, state); | ||||
| 
 | ||||
| 	switch (skb->protocol) { | ||||
| 	case htons(ETH_P_IP): | ||||
| 		nft_set_pktinfo_ipv4_validate(&pkt, skb); | ||||
| 		break; | ||||
| 	case htons(ETH_P_IPV6): | ||||
| 		nft_set_pktinfo_ipv6_validate(&pkt, skb); | ||||
| 		break; | ||||
| 	default: | ||||
| 		nft_set_pktinfo_unspec(&pkt, skb); | ||||
| 		break; | ||||
| 	} | ||||
| 
 | ||||
| 	return nft_do_chain(&pkt, priv); | ||||
| } | ||||
| 
 | ||||
| static const struct nft_chain_type nft_chain_filter_netdev = { | ||||
| 	.name		= "filter", | ||||
| 	.type		= NFT_CHAIN_T_DEFAULT, | ||||
| 	.family		= NFPROTO_NETDEV, | ||||
| 	.hook_mask	= (1 << NF_NETDEV_INGRESS), | ||||
| 	.hooks		= { | ||||
| 		[NF_NETDEV_INGRESS]	= nft_do_chain_netdev, | ||||
| 	}, | ||||
| }; | ||||
| 
 | ||||
| static void nft_netdev_event(unsigned long event, struct net_device *dev, | ||||
| 			     struct nft_ctx *ctx) | ||||
| { | ||||
| 	struct nft_base_chain *basechain = nft_base_chain(ctx->chain); | ||||
| 
 | ||||
| 	switch (event) { | ||||
| 	case NETDEV_UNREGISTER: | ||||
| 		if (strcmp(basechain->dev_name, dev->name) != 0) | ||||
| 			return; | ||||
| 
 | ||||
| 		__nft_release_basechain(ctx); | ||||
| 		break; | ||||
| 	case NETDEV_CHANGENAME: | ||||
| 		if (dev->ifindex != basechain->ops.dev->ifindex) | ||||
| 			return; | ||||
| 
 | ||||
| 		strncpy(basechain->dev_name, dev->name, IFNAMSIZ); | ||||
| 		break; | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| static int nf_tables_netdev_event(struct notifier_block *this, | ||||
| 				  unsigned long event, void *ptr) | ||||
| { | ||||
| 	struct net_device *dev = netdev_notifier_info_to_dev(ptr); | ||||
| 	struct nft_table *table; | ||||
| 	struct nft_chain *chain, *nr; | ||||
| 	struct nft_ctx ctx = { | ||||
| 		.net	= dev_net(dev), | ||||
| 	}; | ||||
| 
 | ||||
| 	if (event != NETDEV_UNREGISTER && | ||||
| 	    event != NETDEV_CHANGENAME) | ||||
| 		return NOTIFY_DONE; | ||||
| 
 | ||||
| 	nfnl_lock(NFNL_SUBSYS_NFTABLES); | ||||
| 	list_for_each_entry(table, &ctx.net->nft.tables, list) { | ||||
| 		if (table->family != NFPROTO_NETDEV) | ||||
| 			continue; | ||||
| 
 | ||||
| 		ctx.family = table->family; | ||||
| 		ctx.table = table; | ||||
| 		list_for_each_entry_safe(chain, nr, &table->chains, list) { | ||||
| 			if (!nft_is_base_chain(chain)) | ||||
| 				continue; | ||||
| 
 | ||||
| 			ctx.chain = chain; | ||||
| 			nft_netdev_event(event, dev, &ctx); | ||||
| 		} | ||||
| 	} | ||||
| 	nfnl_unlock(NFNL_SUBSYS_NFTABLES); | ||||
| 
 | ||||
| 	return NOTIFY_DONE; | ||||
| } | ||||
| 
 | ||||
| static struct notifier_block nf_tables_netdev_notifier = { | ||||
| 	.notifier_call	= nf_tables_netdev_event, | ||||
| }; | ||||
| 
 | ||||
| static int nft_chain_filter_netdev_init(void) | ||||
| { | ||||
| 	int err; | ||||
| 
 | ||||
| 	nft_register_chain_type(&nft_chain_filter_netdev); | ||||
| 
 | ||||
| 	err = register_netdevice_notifier(&nf_tables_netdev_notifier); | ||||
| 	if (err) | ||||
| 		goto err_register_netdevice_notifier; | ||||
| 
 | ||||
| 	return 0; | ||||
| 
 | ||||
| err_register_netdevice_notifier: | ||||
| 	nft_unregister_chain_type(&nft_chain_filter_netdev); | ||||
| 
 | ||||
| 	return err; | ||||
| } | ||||
| 
 | ||||
| static void nft_chain_filter_netdev_fini(void) | ||||
| { | ||||
| 	nft_unregister_chain_type(&nft_chain_filter_netdev); | ||||
| 	unregister_netdevice_notifier(&nf_tables_netdev_notifier); | ||||
| } | ||||
| #else | ||||
| static inline int nft_chain_filter_netdev_init(void) { return 0; } | ||||
| static inline void nft_chain_filter_netdev_fini(void) {} | ||||
| #endif /* CONFIG_NF_TABLES_NETDEV */ | ||||
| 
 | ||||
| int __init nft_chain_filter_init(void) | ||||
| { | ||||
| 	int err; | ||||
| 
 | ||||
| 	err = nft_chain_filter_netdev_init(); | ||||
| 	if (err < 0) | ||||
| 		return err; | ||||
| 
 | ||||
| 	nft_chain_filter_ipv4_init(); | ||||
| 	nft_chain_filter_ipv6_init(); | ||||
| 	nft_chain_filter_arp_init(); | ||||
| 	nft_chain_filter_inet_init(); | ||||
| 	nft_chain_filter_bridge_init(); | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| void __exit nft_chain_filter_fini(void) | ||||
| { | ||||
| 	nft_chain_filter_bridge_fini(); | ||||
| 	nft_chain_filter_inet_fini(); | ||||
| 	nft_chain_filter_arp_fini(); | ||||
| 	nft_chain_filter_ipv6_fini(); | ||||
| 	nft_chain_filter_ipv4_fini(); | ||||
| 	nft_chain_filter_netdev_fini(); | ||||
| } | ||||
|  | @ -196,6 +196,26 @@ static void nft_ct_get_eval(const struct nft_expr *expr, | |||
| 	case NFT_CT_PROTO_DST: | ||||
| 		nft_reg_store16(dest, (__force u16)tuple->dst.u.all); | ||||
| 		return; | ||||
| 	case NFT_CT_SRC_IP: | ||||
| 		if (nf_ct_l3num(ct) != NFPROTO_IPV4) | ||||
| 			goto err; | ||||
| 		*dest = tuple->src.u3.ip; | ||||
| 		return; | ||||
| 	case NFT_CT_DST_IP: | ||||
| 		if (nf_ct_l3num(ct) != NFPROTO_IPV4) | ||||
| 			goto err; | ||||
| 		*dest = tuple->dst.u3.ip; | ||||
| 		return; | ||||
| 	case NFT_CT_SRC_IP6: | ||||
| 		if (nf_ct_l3num(ct) != NFPROTO_IPV6) | ||||
| 			goto err; | ||||
| 		memcpy(dest, tuple->src.u3.ip6, sizeof(struct in6_addr)); | ||||
| 		return; | ||||
| 	case NFT_CT_DST_IP6: | ||||
| 		if (nf_ct_l3num(ct) != NFPROTO_IPV6) | ||||
| 			goto err; | ||||
| 		memcpy(dest, tuple->dst.u3.ip6, sizeof(struct in6_addr)); | ||||
| 		return; | ||||
| 	default: | ||||
| 		break; | ||||
| 	} | ||||
|  | @ -419,6 +439,20 @@ static int nft_ct_get_init(const struct nft_ctx *ctx, | |||
| 			return -EAFNOSUPPORT; | ||||
| 		} | ||||
| 		break; | ||||
| 	case NFT_CT_SRC_IP: | ||||
| 	case NFT_CT_DST_IP: | ||||
| 		if (tb[NFTA_CT_DIRECTION] == NULL) | ||||
| 			return -EINVAL; | ||||
| 
 | ||||
| 		len = FIELD_SIZEOF(struct nf_conntrack_tuple, src.u3.ip); | ||||
| 		break; | ||||
| 	case NFT_CT_SRC_IP6: | ||||
| 	case NFT_CT_DST_IP6: | ||||
| 		if (tb[NFTA_CT_DIRECTION] == NULL) | ||||
| 			return -EINVAL; | ||||
| 
 | ||||
| 		len = FIELD_SIZEOF(struct nf_conntrack_tuple, src.u3.ip6); | ||||
| 		break; | ||||
| 	case NFT_CT_PROTO_SRC: | ||||
| 	case NFT_CT_PROTO_DST: | ||||
| 		if (tb[NFTA_CT_DIRECTION] == NULL) | ||||
|  | @ -588,6 +622,10 @@ static int nft_ct_get_dump(struct sk_buff *skb, const struct nft_expr *expr) | |||
| 	switch (priv->key) { | ||||
| 	case NFT_CT_SRC: | ||||
| 	case NFT_CT_DST: | ||||
| 	case NFT_CT_SRC_IP: | ||||
| 	case NFT_CT_DST_IP: | ||||
| 	case NFT_CT_SRC_IP6: | ||||
| 	case NFT_CT_DST_IP6: | ||||
| 	case NFT_CT_PROTO_SRC: | ||||
| 	case NFT_CT_PROTO_DST: | ||||
| 		if (nla_put_u8(skb, NFTA_CT_DIRECTION, priv->dir)) | ||||
|  |  | |||
|  | @ -132,8 +132,9 @@ static int nft_dynset_init(const struct nft_ctx *ctx, | |||
| 			priv->invert = true; | ||||
| 	} | ||||
| 
 | ||||
| 	set = nft_set_lookup(ctx->net, ctx->table, tb[NFTA_DYNSET_SET_NAME], | ||||
| 			     tb[NFTA_DYNSET_SET_ID], genmask); | ||||
| 	set = nft_set_lookup_global(ctx->net, ctx->table, | ||||
| 				    tb[NFTA_DYNSET_SET_NAME], | ||||
| 				    tb[NFTA_DYNSET_SET_ID], genmask); | ||||
| 	if (IS_ERR(set)) | ||||
| 		return PTR_ERR(set); | ||||
| 
 | ||||
|  |  | |||
|  | @ -71,8 +71,8 @@ static int nft_lookup_init(const struct nft_ctx *ctx, | |||
| 	    tb[NFTA_LOOKUP_SREG] == NULL) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	set = nft_set_lookup(ctx->net, ctx->table, tb[NFTA_LOOKUP_SET], | ||||
| 			     tb[NFTA_LOOKUP_SET_ID], genmask); | ||||
| 	set = nft_set_lookup_global(ctx->net, ctx->table, tb[NFTA_LOOKUP_SET], | ||||
| 				    tb[NFTA_LOOKUP_SET_ID], genmask); | ||||
| 	if (IS_ERR(set)) | ||||
| 		return PTR_ERR(set); | ||||
| 
 | ||||
|  |  | |||
|  | @ -117,8 +117,9 @@ static int nft_objref_map_init(const struct nft_ctx *ctx, | |||
| 	struct nft_set *set; | ||||
| 	int err; | ||||
| 
 | ||||
| 	set = nft_set_lookup(ctx->net, ctx->table, tb[NFTA_OBJREF_SET_NAME], | ||||
| 			     tb[NFTA_OBJREF_SET_ID], genmask); | ||||
| 	set = nft_set_lookup_global(ctx->net, ctx->table, | ||||
| 				    tb[NFTA_OBJREF_SET_NAME], | ||||
| 				    tb[NFTA_OBJREF_SET_ID], genmask); | ||||
| 	if (IS_ERR(set)) | ||||
| 		return PTR_ERR(set); | ||||
| 
 | ||||
|  |  | |||
|  | @ -40,6 +40,7 @@ MODULE_AUTHOR("Harald Welte <laforge@netfilter.org>"); | |||
| MODULE_DESCRIPTION("{ip,ip6,arp,eb}_tables backend module"); | ||||
| 
 | ||||
| #define XT_PCPU_BLOCK_SIZE 4096 | ||||
| #define XT_MAX_TABLE_SIZE	(512 * 1024 * 1024) | ||||
| 
 | ||||
| struct compat_delta { | ||||
| 	unsigned int offset; /* offset in kernel */ | ||||
|  | @ -548,19 +549,104 @@ static int xt_check_entry_match(const char *match, const char *target, | |||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| /** xt_check_table_hooks - check hook entry points are sane
 | ||||
|  * | ||||
|  * @info xt_table_info to check | ||||
|  * @valid_hooks - hook entry points that we can enter from | ||||
|  * | ||||
|  * Validates that the hook entry and underflows points are set up. | ||||
|  * | ||||
|  * Return: 0 on success, negative errno on failure. | ||||
|  */ | ||||
| int xt_check_table_hooks(const struct xt_table_info *info, unsigned int valid_hooks) | ||||
| { | ||||
| 	const char *err = "unsorted underflow"; | ||||
| 	unsigned int i, max_uflow, max_entry; | ||||
| 	bool check_hooks = false; | ||||
| 
 | ||||
| 	BUILD_BUG_ON(ARRAY_SIZE(info->hook_entry) != ARRAY_SIZE(info->underflow)); | ||||
| 
 | ||||
| 	max_entry = 0; | ||||
| 	max_uflow = 0; | ||||
| 
 | ||||
| 	for (i = 0; i < ARRAY_SIZE(info->hook_entry); i++) { | ||||
| 		if (!(valid_hooks & (1 << i))) | ||||
| 			continue; | ||||
| 
 | ||||
| 		if (info->hook_entry[i] == 0xFFFFFFFF) | ||||
| 			return -EINVAL; | ||||
| 		if (info->underflow[i] == 0xFFFFFFFF) | ||||
| 			return -EINVAL; | ||||
| 
 | ||||
| 		if (check_hooks) { | ||||
| 			if (max_uflow > info->underflow[i]) | ||||
| 				goto error; | ||||
| 
 | ||||
| 			if (max_uflow == info->underflow[i]) { | ||||
| 				err = "duplicate underflow"; | ||||
| 				goto error; | ||||
| 			} | ||||
| 			if (max_entry > info->hook_entry[i]) { | ||||
| 				err = "unsorted entry"; | ||||
| 				goto error; | ||||
| 			} | ||||
| 			if (max_entry == info->hook_entry[i]) { | ||||
| 				err = "duplicate entry"; | ||||
| 				goto error; | ||||
| 			} | ||||
| 		} | ||||
| 		max_entry = info->hook_entry[i]; | ||||
| 		max_uflow = info->underflow[i]; | ||||
| 		check_hooks = true; | ||||
| 	} | ||||
| 
 | ||||
| 	return 0; | ||||
| error: | ||||
| 	pr_err_ratelimited("%s at hook %d\n", err, i); | ||||
| 	return -EINVAL; | ||||
| } | ||||
| EXPORT_SYMBOL(xt_check_table_hooks); | ||||
| 
 | ||||
| static bool verdict_ok(int verdict) | ||||
| { | ||||
| 	if (verdict > 0) | ||||
| 		return true; | ||||
| 
 | ||||
| 	if (verdict < 0) { | ||||
| 		int v = -verdict - 1; | ||||
| 
 | ||||
| 		if (verdict == XT_RETURN) | ||||
| 			return true; | ||||
| 
 | ||||
| 		switch (v) { | ||||
| 		case NF_ACCEPT: return true; | ||||
| 		case NF_DROP: return true; | ||||
| 		case NF_QUEUE: return true; | ||||
| 		default: | ||||
| 			break; | ||||
| 		} | ||||
| 
 | ||||
| 		return false; | ||||
| 	} | ||||
| 
 | ||||
| 	return false; | ||||
| } | ||||
| 
 | ||||
| static bool error_tg_ok(unsigned int usersize, unsigned int kernsize, | ||||
| 			const char *msg, unsigned int msglen) | ||||
| { | ||||
| 	return usersize == kernsize && strnlen(msg, msglen) < msglen; | ||||
| } | ||||
| 
 | ||||
| #ifdef CONFIG_COMPAT | ||||
| int xt_compat_add_offset(u_int8_t af, unsigned int offset, int delta) | ||||
| { | ||||
| 	struct xt_af *xp = &xt[af]; | ||||
| 
 | ||||
| 	if (!xp->compat_tab) { | ||||
| 		if (!xp->number) | ||||
| 			return -EINVAL; | ||||
| 		xp->compat_tab = vmalloc(sizeof(struct compat_delta) * xp->number); | ||||
| 		if (!xp->compat_tab) | ||||
| 			return -ENOMEM; | ||||
| 		xp->cur = 0; | ||||
| 	} | ||||
| 	WARN_ON(!mutex_is_locked(&xt[af].compat_mutex)); | ||||
| 
 | ||||
| 	if (WARN_ON(!xp->compat_tab)) | ||||
| 		return -ENOMEM; | ||||
| 
 | ||||
| 	if (xp->cur >= xp->number) | ||||
| 		return -EINVAL; | ||||
|  | @ -576,6 +662,8 @@ EXPORT_SYMBOL_GPL(xt_compat_add_offset); | |||
| 
 | ||||
| void xt_compat_flush_offsets(u_int8_t af) | ||||
| { | ||||
| 	WARN_ON(!mutex_is_locked(&xt[af].compat_mutex)); | ||||
| 
 | ||||
| 	if (xt[af].compat_tab) { | ||||
| 		vfree(xt[af].compat_tab); | ||||
| 		xt[af].compat_tab = NULL; | ||||
|  | @ -603,10 +691,30 @@ int xt_compat_calc_jump(u_int8_t af, unsigned int offset) | |||
| } | ||||
| EXPORT_SYMBOL_GPL(xt_compat_calc_jump); | ||||
| 
 | ||||
| void xt_compat_init_offsets(u_int8_t af, unsigned int number) | ||||
| int xt_compat_init_offsets(u8 af, unsigned int number) | ||||
| { | ||||
| 	size_t mem; | ||||
| 
 | ||||
| 	WARN_ON(!mutex_is_locked(&xt[af].compat_mutex)); | ||||
| 
 | ||||
| 	if (!number || number > (INT_MAX / sizeof(struct compat_delta))) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	if (WARN_ON(xt[af].compat_tab)) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	mem = sizeof(struct compat_delta) * number; | ||||
| 	if (mem > XT_MAX_TABLE_SIZE) | ||||
| 		return -ENOMEM; | ||||
| 
 | ||||
| 	xt[af].compat_tab = vmalloc(mem); | ||||
| 	if (!xt[af].compat_tab) | ||||
| 		return -ENOMEM; | ||||
| 
 | ||||
| 	xt[af].number = number; | ||||
| 	xt[af].cur = 0; | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| EXPORT_SYMBOL(xt_compat_init_offsets); | ||||
| 
 | ||||
|  | @ -684,6 +792,11 @@ struct compat_xt_standard_target { | |||
| 	compat_uint_t verdict; | ||||
| }; | ||||
| 
 | ||||
| struct compat_xt_error_target { | ||||
| 	struct compat_xt_entry_target t; | ||||
| 	char errorname[XT_FUNCTION_MAXNAMELEN]; | ||||
| }; | ||||
| 
 | ||||
| int xt_compat_check_entry_offsets(const void *base, const char *elems, | ||||
| 				  unsigned int target_offset, | ||||
| 				  unsigned int next_offset) | ||||
|  | @ -705,9 +818,21 @@ int xt_compat_check_entry_offsets(const void *base, const char *elems, | |||
| 	if (target_offset + t->u.target_size > next_offset) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	if (strcmp(t->u.user.name, XT_STANDARD_TARGET) == 0 && | ||||
| 	    COMPAT_XT_ALIGN(target_offset + sizeof(struct compat_xt_standard_target)) != next_offset) | ||||
| 		return -EINVAL; | ||||
| 	if (strcmp(t->u.user.name, XT_STANDARD_TARGET) == 0) { | ||||
| 		const struct compat_xt_standard_target *st = (const void *)t; | ||||
| 
 | ||||
| 		if (COMPAT_XT_ALIGN(target_offset + sizeof(*st)) != next_offset) | ||||
| 			return -EINVAL; | ||||
| 
 | ||||
| 		if (!verdict_ok(st->verdict)) | ||||
| 			return -EINVAL; | ||||
| 	} else if (strcmp(t->u.user.name, XT_ERROR_TARGET) == 0) { | ||||
| 		const struct compat_xt_error_target *et = (const void *)t; | ||||
| 
 | ||||
| 		if (!error_tg_ok(t->u.target_size, sizeof(*et), | ||||
| 				 et->errorname, sizeof(et->errorname))) | ||||
| 			return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	/* compat_xt_entry match has less strict alignment requirements,
 | ||||
| 	 * otherwise they are identical.  In case of padding differences | ||||
|  | @ -787,9 +912,21 @@ int xt_check_entry_offsets(const void *base, | |||
| 	if (target_offset + t->u.target_size > next_offset) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	if (strcmp(t->u.user.name, XT_STANDARD_TARGET) == 0 && | ||||
| 	    XT_ALIGN(target_offset + sizeof(struct xt_standard_target)) != next_offset) | ||||
| 		return -EINVAL; | ||||
| 	if (strcmp(t->u.user.name, XT_STANDARD_TARGET) == 0) { | ||||
| 		const struct xt_standard_target *st = (const void *)t; | ||||
| 
 | ||||
| 		if (XT_ALIGN(target_offset + sizeof(*st)) != next_offset) | ||||
| 			return -EINVAL; | ||||
| 
 | ||||
| 		if (!verdict_ok(st->verdict)) | ||||
| 			return -EINVAL; | ||||
| 	} else if (strcmp(t->u.user.name, XT_ERROR_TARGET) == 0) { | ||||
| 		const struct xt_error_target *et = (const void *)t; | ||||
| 
 | ||||
| 		if (!error_tg_ok(t->u.target_size, sizeof(*et), | ||||
| 				 et->errorname, sizeof(et->errorname))) | ||||
| 			return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	return xt_check_entry_match(elems, base + target_offset, | ||||
| 				    __alignof__(struct xt_entry_match)); | ||||
|  | @ -805,6 +942,9 @@ EXPORT_SYMBOL(xt_check_entry_offsets); | |||
|  */ | ||||
| unsigned int *xt_alloc_entry_offsets(unsigned int size) | ||||
| { | ||||
| 	if (size > XT_MAX_TABLE_SIZE / sizeof(unsigned int)) | ||||
| 		return NULL; | ||||
| 
 | ||||
| 	return kvmalloc_array(size, sizeof(unsigned int), GFP_KERNEL | __GFP_ZERO); | ||||
| 
 | ||||
| } | ||||
|  | @ -1029,7 +1169,7 @@ struct xt_table_info *xt_alloc_table_info(unsigned int size) | |||
| 	struct xt_table_info *info = NULL; | ||||
| 	size_t sz = sizeof(*info) + size; | ||||
| 
 | ||||
| 	if (sz < sizeof(*info)) | ||||
| 	if (sz < sizeof(*info) || sz >= XT_MAX_TABLE_SIZE) | ||||
| 		return NULL; | ||||
| 
 | ||||
| 	/* __GFP_NORETRY is not fully supported by kvmalloc but it should
 | ||||
|  | @ -1198,6 +1338,21 @@ static int xt_jumpstack_alloc(struct xt_table_info *i) | |||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| struct xt_counters *xt_counters_alloc(unsigned int counters) | ||||
| { | ||||
| 	struct xt_counters *mem; | ||||
| 
 | ||||
| 	if (counters == 0 || counters > INT_MAX / sizeof(*mem)) | ||||
| 		return NULL; | ||||
| 
 | ||||
| 	counters *= sizeof(*mem); | ||||
| 	if (counters > XT_MAX_TABLE_SIZE) | ||||
| 		return NULL; | ||||
| 
 | ||||
| 	return vzalloc(counters); | ||||
| } | ||||
| EXPORT_SYMBOL(xt_counters_alloc); | ||||
| 
 | ||||
| struct xt_table_info * | ||||
| xt_replace_table(struct xt_table *table, | ||||
| 	      unsigned int num_counters, | ||||
|  | @ -1729,7 +1884,9 @@ EXPORT_SYMBOL_GPL(xt_proto_fini); | |||
|  * to fetch the real percpu counter. | ||||
|  * | ||||
|  * To speed up allocation and improve data locality, a 4kb block is | ||||
|  * allocated. | ||||
|  * allocated.  Freeing any counter may free an entire block, so all | ||||
|  * counters allocated using the same state must be freed at the same | ||||
|  * time. | ||||
|  * | ||||
|  * xt_percpu_counter_alloc_state contains the base address of the | ||||
|  * allocated page and the current sub-offset. | ||||
|  |  | |||
|  | @ -14,15 +14,21 @@ | |||
| #include <linux/slab.h> | ||||
| #include <net/gen_stats.h> | ||||
| #include <net/netlink.h> | ||||
| #include <net/netns/generic.h> | ||||
| 
 | ||||
| #include <linux/netfilter/x_tables.h> | ||||
| #include <linux/netfilter/xt_RATEEST.h> | ||||
| #include <net/netfilter/xt_rateest.h> | ||||
| 
 | ||||
| static DEFINE_MUTEX(xt_rateest_mutex); | ||||
| 
 | ||||
| #define RATEEST_HSIZE	16 | ||||
| static struct hlist_head rateest_hash[RATEEST_HSIZE] __read_mostly; | ||||
| 
 | ||||
| struct xt_rateest_net { | ||||
| 	struct mutex hash_lock; | ||||
| 	struct hlist_head hash[RATEEST_HSIZE]; | ||||
| }; | ||||
| 
 | ||||
| static unsigned int xt_rateest_id; | ||||
| 
 | ||||
| static unsigned int jhash_rnd __read_mostly; | ||||
| 
 | ||||
| static unsigned int xt_rateest_hash(const char *name) | ||||
|  | @ -31,21 +37,23 @@ static unsigned int xt_rateest_hash(const char *name) | |||
| 	       (RATEEST_HSIZE - 1); | ||||
| } | ||||
| 
 | ||||
| static void xt_rateest_hash_insert(struct xt_rateest *est) | ||||
| static void xt_rateest_hash_insert(struct xt_rateest_net *xn, | ||||
| 				   struct xt_rateest *est) | ||||
| { | ||||
| 	unsigned int h; | ||||
| 
 | ||||
| 	h = xt_rateest_hash(est->name); | ||||
| 	hlist_add_head(&est->list, &rateest_hash[h]); | ||||
| 	hlist_add_head(&est->list, &xn->hash[h]); | ||||
| } | ||||
| 
 | ||||
| static struct xt_rateest *__xt_rateest_lookup(const char *name) | ||||
| static struct xt_rateest *__xt_rateest_lookup(struct xt_rateest_net *xn, | ||||
| 					      const char *name) | ||||
| { | ||||
| 	struct xt_rateest *est; | ||||
| 	unsigned int h; | ||||
| 
 | ||||
| 	h = xt_rateest_hash(name); | ||||
| 	hlist_for_each_entry(est, &rateest_hash[h], list) { | ||||
| 	hlist_for_each_entry(est, &xn->hash[h], list) { | ||||
| 		if (strcmp(est->name, name) == 0) { | ||||
| 			est->refcnt++; | ||||
| 			return est; | ||||
|  | @ -55,20 +63,23 @@ static struct xt_rateest *__xt_rateest_lookup(const char *name) | |||
| 	return NULL; | ||||
| } | ||||
| 
 | ||||
| struct xt_rateest *xt_rateest_lookup(const char *name) | ||||
| struct xt_rateest *xt_rateest_lookup(struct net *net, const char *name) | ||||
| { | ||||
| 	struct xt_rateest_net *xn = net_generic(net, xt_rateest_id); | ||||
| 	struct xt_rateest *est; | ||||
| 
 | ||||
| 	mutex_lock(&xt_rateest_mutex); | ||||
| 	est = __xt_rateest_lookup(name); | ||||
| 	mutex_unlock(&xt_rateest_mutex); | ||||
| 	mutex_lock(&xn->hash_lock); | ||||
| 	est = __xt_rateest_lookup(xn, name); | ||||
| 	mutex_unlock(&xn->hash_lock); | ||||
| 	return est; | ||||
| } | ||||
| EXPORT_SYMBOL_GPL(xt_rateest_lookup); | ||||
| 
 | ||||
| void xt_rateest_put(struct xt_rateest *est) | ||||
| void xt_rateest_put(struct net *net, struct xt_rateest *est) | ||||
| { | ||||
| 	mutex_lock(&xt_rateest_mutex); | ||||
| 	struct xt_rateest_net *xn = net_generic(net, xt_rateest_id); | ||||
| 
 | ||||
| 	mutex_lock(&xn->hash_lock); | ||||
| 	if (--est->refcnt == 0) { | ||||
| 		hlist_del(&est->list); | ||||
| 		gen_kill_estimator(&est->rate_est); | ||||
|  | @ -78,7 +89,7 @@ void xt_rateest_put(struct xt_rateest *est) | |||
| 		 */ | ||||
| 		kfree_rcu(est, rcu); | ||||
| 	} | ||||
| 	mutex_unlock(&xt_rateest_mutex); | ||||
| 	mutex_unlock(&xn->hash_lock); | ||||
| } | ||||
| EXPORT_SYMBOL_GPL(xt_rateest_put); | ||||
| 
 | ||||
|  | @ -98,6 +109,7 @@ xt_rateest_tg(struct sk_buff *skb, const struct xt_action_param *par) | |||
| 
 | ||||
| static int xt_rateest_tg_checkentry(const struct xt_tgchk_param *par) | ||||
| { | ||||
| 	struct xt_rateest_net *xn = net_generic(par->net, xt_rateest_id); | ||||
| 	struct xt_rateest_target_info *info = par->targinfo; | ||||
| 	struct xt_rateest *est; | ||||
| 	struct { | ||||
|  | @ -108,10 +120,10 @@ static int xt_rateest_tg_checkentry(const struct xt_tgchk_param *par) | |||
| 
 | ||||
| 	net_get_random_once(&jhash_rnd, sizeof(jhash_rnd)); | ||||
| 
 | ||||
| 	mutex_lock(&xt_rateest_mutex); | ||||
| 	est = __xt_rateest_lookup(info->name); | ||||
| 	mutex_lock(&xn->hash_lock); | ||||
| 	est = __xt_rateest_lookup(xn, info->name); | ||||
| 	if (est) { | ||||
| 		mutex_unlock(&xt_rateest_mutex); | ||||
| 		mutex_unlock(&xn->hash_lock); | ||||
| 		/*
 | ||||
| 		 * If estimator parameters are specified, they must match the | ||||
| 		 * existing estimator. | ||||
|  | @ -119,7 +131,7 @@ static int xt_rateest_tg_checkentry(const struct xt_tgchk_param *par) | |||
| 		if ((!info->interval && !info->ewma_log) || | ||||
| 		    (info->interval != est->params.interval || | ||||
| 		     info->ewma_log != est->params.ewma_log)) { | ||||
| 			xt_rateest_put(est); | ||||
| 			xt_rateest_put(par->net, est); | ||||
| 			return -EINVAL; | ||||
| 		} | ||||
| 		info->est = est; | ||||
|  | @ -148,14 +160,14 @@ static int xt_rateest_tg_checkentry(const struct xt_tgchk_param *par) | |||
| 		goto err2; | ||||
| 
 | ||||
| 	info->est = est; | ||||
| 	xt_rateest_hash_insert(est); | ||||
| 	mutex_unlock(&xt_rateest_mutex); | ||||
| 	xt_rateest_hash_insert(xn, est); | ||||
| 	mutex_unlock(&xn->hash_lock); | ||||
| 	return 0; | ||||
| 
 | ||||
| err2: | ||||
| 	kfree(est); | ||||
| err1: | ||||
| 	mutex_unlock(&xt_rateest_mutex); | ||||
| 	mutex_unlock(&xn->hash_lock); | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
|  | @ -163,7 +175,7 @@ static void xt_rateest_tg_destroy(const struct xt_tgdtor_param *par) | |||
| { | ||||
| 	struct xt_rateest_target_info *info = par->targinfo; | ||||
| 
 | ||||
| 	xt_rateest_put(info->est); | ||||
| 	xt_rateest_put(par->net, info->est); | ||||
| } | ||||
| 
 | ||||
| static struct xt_target xt_rateest_tg_reg __read_mostly = { | ||||
|  | @ -178,19 +190,46 @@ static struct xt_target xt_rateest_tg_reg __read_mostly = { | |||
| 	.me         = THIS_MODULE, | ||||
| }; | ||||
| 
 | ||||
| static __net_init int xt_rateest_net_init(struct net *net) | ||||
| { | ||||
| 	struct xt_rateest_net *xn = net_generic(net, xt_rateest_id); | ||||
| 	int i; | ||||
| 
 | ||||
| 	mutex_init(&xn->hash_lock); | ||||
| 	for (i = 0; i < ARRAY_SIZE(xn->hash); i++) | ||||
| 		INIT_HLIST_HEAD(&xn->hash[i]); | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static void __net_exit xt_rateest_net_exit(struct net *net) | ||||
| { | ||||
| 	struct xt_rateest_net *xn = net_generic(net, xt_rateest_id); | ||||
| 	int i; | ||||
| 
 | ||||
| 	for (i = 0; i < ARRAY_SIZE(xn->hash); i++) | ||||
| 		WARN_ON_ONCE(!hlist_empty(&xn->hash[i])); | ||||
| } | ||||
| 
 | ||||
| static struct pernet_operations xt_rateest_net_ops = { | ||||
| 	.init = xt_rateest_net_init, | ||||
| 	.exit = xt_rateest_net_exit, | ||||
| 	.id   = &xt_rateest_id, | ||||
| 	.size = sizeof(struct xt_rateest_net), | ||||
| }; | ||||
| 
 | ||||
| static int __init xt_rateest_tg_init(void) | ||||
| { | ||||
| 	unsigned int i; | ||||
| 
 | ||||
| 	for (i = 0; i < ARRAY_SIZE(rateest_hash); i++) | ||||
| 		INIT_HLIST_HEAD(&rateest_hash[i]); | ||||
| 	int err = register_pernet_subsys(&xt_rateest_net_ops); | ||||
| 
 | ||||
| 	if (err) | ||||
| 		return err; | ||||
| 	return xt_register_target(&xt_rateest_tg_reg); | ||||
| } | ||||
| 
 | ||||
| static void __exit xt_rateest_tg_fini(void) | ||||
| { | ||||
| 	xt_unregister_target(&xt_rateest_tg_reg); | ||||
| 	unregister_pernet_subsys(&xt_rateest_net_ops); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
|  |  | |||
|  | @ -59,13 +59,6 @@ xt_cluster_hash(const struct nf_conn *ct, | |||
| 	return reciprocal_scale(hash, info->total_nodes); | ||||
| } | ||||
| 
 | ||||
| static inline bool | ||||
| xt_cluster_ipv6_is_multicast(const struct in6_addr *addr) | ||||
| { | ||||
| 	__be32 st = addr->s6_addr32[0]; | ||||
| 	return ((st & htonl(0xFF000000)) == htonl(0xFF000000)); | ||||
| } | ||||
| 
 | ||||
| static inline bool | ||||
| xt_cluster_is_multicast_addr(const struct sk_buff *skb, u_int8_t family) | ||||
| { | ||||
|  | @ -76,8 +69,7 @@ xt_cluster_is_multicast_addr(const struct sk_buff *skb, u_int8_t family) | |||
| 		is_multicast = ipv4_is_multicast(ip_hdr(skb)->daddr); | ||||
| 		break; | ||||
| 	case NFPROTO_IPV6: | ||||
| 		is_multicast = | ||||
| 			xt_cluster_ipv6_is_multicast(&ipv6_hdr(skb)->daddr); | ||||
| 		is_multicast = ipv6_addr_is_multicast(&ipv6_hdr(skb)->daddr); | ||||
| 		break; | ||||
| 	default: | ||||
| 		WARN_ON(1); | ||||
|  |  | |||
|  | @ -67,8 +67,8 @@ connlimit_mt(const struct sk_buff *skb, struct xt_action_param *par) | |||
| 		key[1] = zone->id; | ||||
| 	} | ||||
| 
 | ||||
| 	connections = nf_conncount_count(net, info->data, key, | ||||
| 					 xt_family(par), tuple_ptr, zone); | ||||
| 	connections = nf_conncount_count(net, info->data, key, tuple_ptr, | ||||
| 					 zone); | ||||
| 	if (connections == 0) | ||||
| 		/* kmalloc failed, drop it entirely */ | ||||
| 		goto hotdrop; | ||||
|  |  | |||
|  | @ -36,9 +36,10 @@ MODULE_ALIAS("ipt_connmark"); | |||
| MODULE_ALIAS("ip6t_connmark"); | ||||
| 
 | ||||
| static unsigned int | ||||
| connmark_tg(struct sk_buff *skb, const struct xt_action_param *par) | ||||
| connmark_tg_shift(struct sk_buff *skb, | ||||
| 		const struct xt_connmark_tginfo1 *info, | ||||
| 		u8 shift_bits, u8 shift_dir) | ||||
| { | ||||
| 	const struct xt_connmark_tginfo1 *info = par->targinfo; | ||||
| 	enum ip_conntrack_info ctinfo; | ||||
| 	struct nf_conn *ct; | ||||
| 	u_int32_t newmark; | ||||
|  | @ -50,6 +51,10 @@ connmark_tg(struct sk_buff *skb, const struct xt_action_param *par) | |||
| 	switch (info->mode) { | ||||
| 	case XT_CONNMARK_SET: | ||||
| 		newmark = (ct->mark & ~info->ctmask) ^ info->ctmark; | ||||
| 		if (shift_dir == D_SHIFT_RIGHT) | ||||
| 			newmark >>= shift_bits; | ||||
| 		else | ||||
| 			newmark <<= shift_bits; | ||||
| 		if (ct->mark != newmark) { | ||||
| 			ct->mark = newmark; | ||||
| 			nf_conntrack_event_cache(IPCT_MARK, ct); | ||||
|  | @ -57,7 +62,11 @@ connmark_tg(struct sk_buff *skb, const struct xt_action_param *par) | |||
| 		break; | ||||
| 	case XT_CONNMARK_SAVE: | ||||
| 		newmark = (ct->mark & ~info->ctmask) ^ | ||||
| 		          (skb->mark & info->nfmask); | ||||
| 			  (skb->mark & info->nfmask); | ||||
| 		if (shift_dir == D_SHIFT_RIGHT) | ||||
| 			newmark >>= shift_bits; | ||||
| 		else | ||||
| 			newmark <<= shift_bits; | ||||
| 		if (ct->mark != newmark) { | ||||
| 			ct->mark = newmark; | ||||
| 			nf_conntrack_event_cache(IPCT_MARK, ct); | ||||
|  | @ -65,14 +74,34 @@ connmark_tg(struct sk_buff *skb, const struct xt_action_param *par) | |||
| 		break; | ||||
| 	case XT_CONNMARK_RESTORE: | ||||
| 		newmark = (skb->mark & ~info->nfmask) ^ | ||||
| 		          (ct->mark & info->ctmask); | ||||
| 			  (ct->mark & info->ctmask); | ||||
| 		if (shift_dir == D_SHIFT_RIGHT) | ||||
| 			newmark >>= shift_bits; | ||||
| 		else | ||||
| 			newmark <<= shift_bits; | ||||
| 		skb->mark = newmark; | ||||
| 		break; | ||||
| 	} | ||||
| 
 | ||||
| 	return XT_CONTINUE; | ||||
| } | ||||
| 
 | ||||
| static unsigned int | ||||
| connmark_tg(struct sk_buff *skb, const struct xt_action_param *par) | ||||
| { | ||||
| 	const struct xt_connmark_tginfo1 *info = par->targinfo; | ||||
| 
 | ||||
| 	return connmark_tg_shift(skb, info, 0, 0); | ||||
| } | ||||
| 
 | ||||
| static unsigned int | ||||
| connmark_tg_v2(struct sk_buff *skb, const struct xt_action_param *par) | ||||
| { | ||||
| 	const struct xt_connmark_tginfo2 *info = par->targinfo; | ||||
| 
 | ||||
| 	return connmark_tg_shift(skb, (const struct xt_connmark_tginfo1 *)info, | ||||
| 				 info->shift_bits, info->shift_dir); | ||||
| } | ||||
| 
 | ||||
| static int connmark_tg_check(const struct xt_tgchk_param *par) | ||||
| { | ||||
| 	int ret; | ||||
|  | @ -119,15 +148,27 @@ static void connmark_mt_destroy(const struct xt_mtdtor_param *par) | |||
| 	nf_ct_netns_put(par->net, par->family); | ||||
| } | ||||
| 
 | ||||
| static struct xt_target connmark_tg_reg __read_mostly = { | ||||
| 	.name           = "CONNMARK", | ||||
| 	.revision       = 1, | ||||
| 	.family         = NFPROTO_UNSPEC, | ||||
| 	.checkentry     = connmark_tg_check, | ||||
| 	.target         = connmark_tg, | ||||
| 	.targetsize     = sizeof(struct xt_connmark_tginfo1), | ||||
| 	.destroy        = connmark_tg_destroy, | ||||
| 	.me             = THIS_MODULE, | ||||
| static struct xt_target connmark_tg_reg[] __read_mostly = { | ||||
| 	{ | ||||
| 		.name           = "CONNMARK", | ||||
| 		.revision       = 1, | ||||
| 		.family         = NFPROTO_UNSPEC, | ||||
| 		.checkentry     = connmark_tg_check, | ||||
| 		.target         = connmark_tg, | ||||
| 		.targetsize     = sizeof(struct xt_connmark_tginfo1), | ||||
| 		.destroy        = connmark_tg_destroy, | ||||
| 		.me             = THIS_MODULE, | ||||
| 	}, | ||||
| 	{ | ||||
| 		.name           = "CONNMARK", | ||||
| 		.revision       = 2, | ||||
| 		.family         = NFPROTO_UNSPEC, | ||||
| 		.checkentry     = connmark_tg_check, | ||||
| 		.target         = connmark_tg_v2, | ||||
| 		.targetsize     = sizeof(struct xt_connmark_tginfo2), | ||||
| 		.destroy        = connmark_tg_destroy, | ||||
| 		.me             = THIS_MODULE, | ||||
| 	} | ||||
| }; | ||||
| 
 | ||||
| static struct xt_match connmark_mt_reg __read_mostly = { | ||||
|  | @ -145,12 +186,14 @@ static int __init connmark_mt_init(void) | |||
| { | ||||
| 	int ret; | ||||
| 
 | ||||
| 	ret = xt_register_target(&connmark_tg_reg); | ||||
| 	ret = xt_register_targets(connmark_tg_reg, | ||||
| 				  ARRAY_SIZE(connmark_tg_reg)); | ||||
| 	if (ret < 0) | ||||
| 		return ret; | ||||
| 	ret = xt_register_match(&connmark_mt_reg); | ||||
| 	if (ret < 0) { | ||||
| 		xt_unregister_target(&connmark_tg_reg); | ||||
| 		xt_unregister_targets(connmark_tg_reg, | ||||
| 				      ARRAY_SIZE(connmark_tg_reg)); | ||||
| 		return ret; | ||||
| 	} | ||||
| 	return 0; | ||||
|  | @ -159,7 +202,7 @@ static int __init connmark_mt_init(void) | |||
| static void __exit connmark_mt_exit(void) | ||||
| { | ||||
| 	xt_unregister_match(&connmark_mt_reg); | ||||
| 	xt_unregister_target(&connmark_tg_reg); | ||||
| 	xt_unregister_target(connmark_tg_reg); | ||||
| } | ||||
| 
 | ||||
| module_init(connmark_mt_init); | ||||
|  |  | |||
|  | @ -534,8 +534,7 @@ static u64 user2rate_bytes(u32 user) | |||
| 	u64 r; | ||||
| 
 | ||||
| 	r = user ? U32_MAX / user : U32_MAX; | ||||
| 	r = (r - 1) << XT_HASHLIMIT_BYTE_SHIFT; | ||||
| 	return r; | ||||
| 	return (r - 1) << XT_HASHLIMIT_BYTE_SHIFT; | ||||
| } | ||||
| 
 | ||||
| static void rateinfo_recalc(struct dsthash_ent *dh, unsigned long now, | ||||
|  |  | |||
|  | @ -46,7 +46,7 @@ MODULE_ALIAS("ip6t_limit"); | |||
| 
 | ||||
|    See Alexey's formal explanation in net/sched/sch_tbf.c. | ||||
| 
 | ||||
|    To get the maxmum range, we multiply by this factor (ie. you get N | ||||
|    To get the maximum range, we multiply by this factor (ie. you get N | ||||
|    credits per jiffy).  We want to allow a rate as low as 1 per day | ||||
|    (slowest userspace tool allows), which means | ||||
|    CREDITS_PER_JIFFY*HZ*60*60*24 < 2^32. ie. */ | ||||
|  |  | |||
|  | @ -28,7 +28,7 @@ static bool nfacct_mt(const struct sk_buff *skb, struct xt_action_param *par) | |||
| 
 | ||||
| 	nfnl_acct_update(skb, info->nfacct); | ||||
| 
 | ||||
| 	overquota = nfnl_acct_overquota(xt_net(par), skb, info->nfacct); | ||||
| 	overquota = nfnl_acct_overquota(xt_net(par), info->nfacct); | ||||
| 
 | ||||
| 	return overquota == NFACCT_UNDERQUOTA ? false : true; | ||||
| } | ||||
|  |  | |||
|  | @ -95,13 +95,13 @@ static int xt_rateest_mt_checkentry(const struct xt_mtchk_param *par) | |||
| 	} | ||||
| 
 | ||||
| 	ret  = -ENOENT; | ||||
| 	est1 = xt_rateest_lookup(info->name1); | ||||
| 	est1 = xt_rateest_lookup(par->net, info->name1); | ||||
| 	if (!est1) | ||||
| 		goto err1; | ||||
| 
 | ||||
| 	est2 = NULL; | ||||
| 	if (info->flags & XT_RATEEST_MATCH_REL) { | ||||
| 		est2 = xt_rateest_lookup(info->name2); | ||||
| 		est2 = xt_rateest_lookup(par->net, info->name2); | ||||
| 		if (!est2) | ||||
| 			goto err2; | ||||
| 	} | ||||
|  | @ -111,7 +111,7 @@ static int xt_rateest_mt_checkentry(const struct xt_mtchk_param *par) | |||
| 	return 0; | ||||
| 
 | ||||
| err2: | ||||
| 	xt_rateest_put(est1); | ||||
| 	xt_rateest_put(par->net, est1); | ||||
| err1: | ||||
| 	return ret; | ||||
| } | ||||
|  | @ -120,9 +120,9 @@ static void xt_rateest_mt_destroy(const struct xt_mtdtor_param *par) | |||
| { | ||||
| 	struct xt_rateest_match_info *info = par->matchinfo; | ||||
| 
 | ||||
| 	xt_rateest_put(info->est1); | ||||
| 	xt_rateest_put(par->net, info->est1); | ||||
| 	if (info->est2) | ||||
| 		xt_rateest_put(info->est2); | ||||
| 		xt_rateest_put(par->net, info->est2); | ||||
| } | ||||
| 
 | ||||
| static struct xt_match xt_rateest_mt_reg __read_mostly = { | ||||
|  |  | |||
|  | @ -21,6 +21,7 @@ MODULE_DESCRIPTION("Xtables: string-based matching"); | |||
| MODULE_LICENSE("GPL"); | ||||
| MODULE_ALIAS("ipt_string"); | ||||
| MODULE_ALIAS("ip6t_string"); | ||||
| MODULE_ALIAS("ebt_string"); | ||||
| 
 | ||||
| static bool | ||||
| string_mt(const struct sk_buff *skb, struct xt_action_param *par) | ||||
|  |  | |||
|  | @ -9,6 +9,9 @@ | |||
|  *	This file is distributed under the terms of the GNU General Public | ||||
|  *	License (GPL). Copies of the GPL can be obtained from gnu.org/gpl. | ||||
|  */ | ||||
| 
 | ||||
| #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt | ||||
| 
 | ||||
| #include <linux/ktime.h> | ||||
| #include <linux/module.h> | ||||
| #include <linux/skbuff.h> | ||||
|  | @ -266,13 +269,11 @@ static int __init time_mt_init(void) | |||
| 	int minutes = sys_tz.tz_minuteswest; | ||||
| 
 | ||||
| 	if (minutes < 0) /* east of Greenwich */ | ||||
| 		printk(KERN_INFO KBUILD_MODNAME | ||||
| 		       ": kernel timezone is +%02d%02d\n", | ||||
| 		       -minutes / 60, -minutes % 60); | ||||
| 		pr_info("kernel timezone is +%02d%02d\n", | ||||
| 			-minutes / 60, -minutes % 60); | ||||
| 	else /* west of Greenwich */ | ||||
| 		printk(KERN_INFO KBUILD_MODNAME | ||||
| 		       ": kernel timezone is -%02d%02d\n", | ||||
| 		       minutes / 60, minutes % 60); | ||||
| 		pr_info("kernel timezone is -%02d%02d\n", | ||||
| 			minutes / 60, minutes % 60); | ||||
| 
 | ||||
| 	return xt_register_match(&xt_time_mt_reg); | ||||
| } | ||||
|  |  | |||
		Loading…
	
	Add table
		
		Reference in a new issue
	
	 David S. Miller
						David S. Miller