forked from mirrors/linux
		
	xfrm: remove decode_session indirection from afinfo_policy
No external dependencies, might as well handle this directly. xfrm_afinfo_policy is now 40 bytes on x86_64. Signed-off-by: Florian Westphal <fw@strlen.de> Signed-off-by: Steffen Klassert <steffen.klassert@secunet.com>
This commit is contained in:
		
							parent
							
								
									2e8b4aa816
								
							
						
					
					
						commit
						c53ac41e37
					
				
					 4 changed files with 222 additions and 232 deletions
				
			
		|  | @ -326,9 +326,6 @@ struct xfrm_policy_afinfo { | ||||||
| 					     xfrm_address_t *saddr, | 					     xfrm_address_t *saddr, | ||||||
| 					     xfrm_address_t *daddr, | 					     xfrm_address_t *daddr, | ||||||
| 					     u32 mark); | 					     u32 mark); | ||||||
| 	void			(*decode_session)(struct sk_buff *skb, |  | ||||||
| 						  struct flowi *fl, |  | ||||||
| 						  int reverse); |  | ||||||
| 	int			(*fill_dst)(struct xfrm_dst *xdst, | 	int			(*fill_dst)(struct xfrm_dst *xdst, | ||||||
| 					    struct net_device *dev, | 					    struct net_device *dev, | ||||||
| 					    const struct flowi *fl); | 					    const struct flowi *fl); | ||||||
|  |  | ||||||
|  | @ -12,7 +12,6 @@ | ||||||
| #include <linux/err.h> | #include <linux/err.h> | ||||||
| #include <linux/kernel.h> | #include <linux/kernel.h> | ||||||
| #include <linux/inetdevice.h> | #include <linux/inetdevice.h> | ||||||
| #include <linux/if_tunnel.h> |  | ||||||
| #include <net/dst.h> | #include <net/dst.h> | ||||||
| #include <net/xfrm.h> | #include <net/xfrm.h> | ||||||
| #include <net/ip.h> | #include <net/ip.h> | ||||||
|  | @ -96,118 +95,6 @@ static int xfrm4_fill_dst(struct xfrm_dst *xdst, struct net_device *dev, | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static void |  | ||||||
| _decode_session4(struct sk_buff *skb, struct flowi *fl, int reverse) |  | ||||||
| { |  | ||||||
| 	const struct iphdr *iph = ip_hdr(skb); |  | ||||||
| 	u8 *xprth = skb_network_header(skb) + iph->ihl * 4; |  | ||||||
| 	struct flowi4 *fl4 = &fl->u.ip4; |  | ||||||
| 	int oif = 0; |  | ||||||
| 
 |  | ||||||
| 	if (skb_dst(skb)) |  | ||||||
| 		oif = skb_dst(skb)->dev->ifindex; |  | ||||||
| 
 |  | ||||||
| 	memset(fl4, 0, sizeof(struct flowi4)); |  | ||||||
| 	fl4->flowi4_mark = skb->mark; |  | ||||||
| 	fl4->flowi4_oif = reverse ? skb->skb_iif : oif; |  | ||||||
| 
 |  | ||||||
| 	if (!ip_is_fragment(iph)) { |  | ||||||
| 		switch (iph->protocol) { |  | ||||||
| 		case IPPROTO_UDP: |  | ||||||
| 		case IPPROTO_UDPLITE: |  | ||||||
| 		case IPPROTO_TCP: |  | ||||||
| 		case IPPROTO_SCTP: |  | ||||||
| 		case IPPROTO_DCCP: |  | ||||||
| 			if (xprth + 4 < skb->data || |  | ||||||
| 			    pskb_may_pull(skb, xprth + 4 - skb->data)) { |  | ||||||
| 				__be16 *ports; |  | ||||||
| 
 |  | ||||||
| 				xprth = skb_network_header(skb) + iph->ihl * 4; |  | ||||||
| 				ports = (__be16 *)xprth; |  | ||||||
| 
 |  | ||||||
| 				fl4->fl4_sport = ports[!!reverse]; |  | ||||||
| 				fl4->fl4_dport = ports[!reverse]; |  | ||||||
| 			} |  | ||||||
| 			break; |  | ||||||
| 
 |  | ||||||
| 		case IPPROTO_ICMP: |  | ||||||
| 			if (xprth + 2 < skb->data || |  | ||||||
| 			    pskb_may_pull(skb, xprth + 2 - skb->data)) { |  | ||||||
| 				u8 *icmp; |  | ||||||
| 
 |  | ||||||
| 				xprth = skb_network_header(skb) + iph->ihl * 4; |  | ||||||
| 				icmp = xprth; |  | ||||||
| 
 |  | ||||||
| 				fl4->fl4_icmp_type = icmp[0]; |  | ||||||
| 				fl4->fl4_icmp_code = icmp[1]; |  | ||||||
| 			} |  | ||||||
| 			break; |  | ||||||
| 
 |  | ||||||
| 		case IPPROTO_ESP: |  | ||||||
| 			if (xprth + 4 < skb->data || |  | ||||||
| 			    pskb_may_pull(skb, xprth + 4 - skb->data)) { |  | ||||||
| 				__be32 *ehdr; |  | ||||||
| 
 |  | ||||||
| 				xprth = skb_network_header(skb) + iph->ihl * 4; |  | ||||||
| 				ehdr = (__be32 *)xprth; |  | ||||||
| 
 |  | ||||||
| 				fl4->fl4_ipsec_spi = ehdr[0]; |  | ||||||
| 			} |  | ||||||
| 			break; |  | ||||||
| 
 |  | ||||||
| 		case IPPROTO_AH: |  | ||||||
| 			if (xprth + 8 < skb->data || |  | ||||||
| 			    pskb_may_pull(skb, xprth + 8 - skb->data)) { |  | ||||||
| 				__be32 *ah_hdr; |  | ||||||
| 
 |  | ||||||
| 				xprth = skb_network_header(skb) + iph->ihl * 4; |  | ||||||
| 				ah_hdr = (__be32 *)xprth; |  | ||||||
| 
 |  | ||||||
| 				fl4->fl4_ipsec_spi = ah_hdr[1]; |  | ||||||
| 			} |  | ||||||
| 			break; |  | ||||||
| 
 |  | ||||||
| 		case IPPROTO_COMP: |  | ||||||
| 			if (xprth + 4 < skb->data || |  | ||||||
| 			    pskb_may_pull(skb, xprth + 4 - skb->data)) { |  | ||||||
| 				__be16 *ipcomp_hdr; |  | ||||||
| 
 |  | ||||||
| 				xprth = skb_network_header(skb) + iph->ihl * 4; |  | ||||||
| 				ipcomp_hdr = (__be16 *)xprth; |  | ||||||
| 
 |  | ||||||
| 				fl4->fl4_ipsec_spi = htonl(ntohs(ipcomp_hdr[1])); |  | ||||||
| 			} |  | ||||||
| 			break; |  | ||||||
| 
 |  | ||||||
| 		case IPPROTO_GRE: |  | ||||||
| 			if (xprth + 12 < skb->data || |  | ||||||
| 			    pskb_may_pull(skb, xprth + 12 - skb->data)) { |  | ||||||
| 				__be16 *greflags; |  | ||||||
| 				__be32 *gre_hdr; |  | ||||||
| 
 |  | ||||||
| 				xprth = skb_network_header(skb) + iph->ihl * 4; |  | ||||||
| 				greflags = (__be16 *)xprth; |  | ||||||
| 				gre_hdr = (__be32 *)xprth; |  | ||||||
| 
 |  | ||||||
| 				if (greflags[0] & GRE_KEY) { |  | ||||||
| 					if (greflags[0] & GRE_CSUM) |  | ||||||
| 						gre_hdr++; |  | ||||||
| 					fl4->fl4_gre_key = gre_hdr[1]; |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 			break; |  | ||||||
| 
 |  | ||||||
| 		default: |  | ||||||
| 			fl4->fl4_ipsec_spi = 0; |  | ||||||
| 			break; |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	fl4->flowi4_proto = iph->protocol; |  | ||||||
| 	fl4->daddr = reverse ? iph->saddr : iph->daddr; |  | ||||||
| 	fl4->saddr = reverse ? iph->daddr : iph->saddr; |  | ||||||
| 	fl4->flowi4_tos = iph->tos; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| static void xfrm4_update_pmtu(struct dst_entry *dst, struct sock *sk, | static void xfrm4_update_pmtu(struct dst_entry *dst, struct sock *sk, | ||||||
| 			      struct sk_buff *skb, u32 mtu) | 			      struct sk_buff *skb, u32 mtu) | ||||||
| { | { | ||||||
|  | @ -260,7 +147,6 @@ static const struct xfrm_policy_afinfo xfrm4_policy_afinfo = { | ||||||
| 	.dst_ops =		&xfrm4_dst_ops_template, | 	.dst_ops =		&xfrm4_dst_ops_template, | ||||||
| 	.dst_lookup =		xfrm4_dst_lookup, | 	.dst_lookup =		xfrm4_dst_lookup, | ||||||
| 	.get_saddr =		xfrm4_get_saddr, | 	.get_saddr =		xfrm4_get_saddr, | ||||||
| 	.decode_session =	_decode_session4, |  | ||||||
| 	.fill_dst =		xfrm4_fill_dst, | 	.fill_dst =		xfrm4_fill_dst, | ||||||
| 	.blackhole_route =	ipv4_blackhole_route, | 	.blackhole_route =	ipv4_blackhole_route, | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | @ -22,9 +22,6 @@ | ||||||
| #include <net/ipv6.h> | #include <net/ipv6.h> | ||||||
| #include <net/ip6_route.h> | #include <net/ip6_route.h> | ||||||
| #include <net/l3mdev.h> | #include <net/l3mdev.h> | ||||||
| #if IS_ENABLED(CONFIG_IPV6_MIP6) |  | ||||||
| #include <net/mip6.h> |  | ||||||
| #endif |  | ||||||
| 
 | 
 | ||||||
| static struct dst_entry *xfrm6_dst_lookup(struct net *net, int tos, int oif, | static struct dst_entry *xfrm6_dst_lookup(struct net *net, int tos, int oif, | ||||||
| 					  const xfrm_address_t *saddr, | 					  const xfrm_address_t *saddr, | ||||||
|  | @ -100,108 +97,6 @@ static int xfrm6_fill_dst(struct xfrm_dst *xdst, struct net_device *dev, | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static inline void |  | ||||||
| _decode_session6(struct sk_buff *skb, struct flowi *fl, int reverse) |  | ||||||
| { |  | ||||||
| 	struct flowi6 *fl6 = &fl->u.ip6; |  | ||||||
| 	int onlyproto = 0; |  | ||||||
| 	const struct ipv6hdr *hdr = ipv6_hdr(skb); |  | ||||||
| 	u32 offset = sizeof(*hdr); |  | ||||||
| 	struct ipv6_opt_hdr *exthdr; |  | ||||||
| 	const unsigned char *nh = skb_network_header(skb); |  | ||||||
| 	u16 nhoff = IP6CB(skb)->nhoff; |  | ||||||
| 	int oif = 0; |  | ||||||
| 	u8 nexthdr; |  | ||||||
| 
 |  | ||||||
| 	if (!nhoff) |  | ||||||
| 		nhoff = offsetof(struct ipv6hdr, nexthdr); |  | ||||||
| 
 |  | ||||||
| 	nexthdr = nh[nhoff]; |  | ||||||
| 
 |  | ||||||
| 	if (skb_dst(skb)) |  | ||||||
| 		oif = skb_dst(skb)->dev->ifindex; |  | ||||||
| 
 |  | ||||||
| 	memset(fl6, 0, sizeof(struct flowi6)); |  | ||||||
| 	fl6->flowi6_mark = skb->mark; |  | ||||||
| 	fl6->flowi6_oif = reverse ? skb->skb_iif : oif; |  | ||||||
| 
 |  | ||||||
| 	fl6->daddr = reverse ? hdr->saddr : hdr->daddr; |  | ||||||
| 	fl6->saddr = reverse ? hdr->daddr : hdr->saddr; |  | ||||||
| 
 |  | ||||||
| 	while (nh + offset + sizeof(*exthdr) < skb->data || |  | ||||||
| 	       pskb_may_pull(skb, nh + offset + sizeof(*exthdr) - skb->data)) { |  | ||||||
| 		nh = skb_network_header(skb); |  | ||||||
| 		exthdr = (struct ipv6_opt_hdr *)(nh + offset); |  | ||||||
| 
 |  | ||||||
| 		switch (nexthdr) { |  | ||||||
| 		case NEXTHDR_FRAGMENT: |  | ||||||
| 			onlyproto = 1; |  | ||||||
| 			/* fall through */ |  | ||||||
| 		case NEXTHDR_ROUTING: |  | ||||||
| 		case NEXTHDR_HOP: |  | ||||||
| 		case NEXTHDR_DEST: |  | ||||||
| 			offset += ipv6_optlen(exthdr); |  | ||||||
| 			nexthdr = exthdr->nexthdr; |  | ||||||
| 			exthdr = (struct ipv6_opt_hdr *)(nh + offset); |  | ||||||
| 			break; |  | ||||||
| 
 |  | ||||||
| 		case IPPROTO_UDP: |  | ||||||
| 		case IPPROTO_UDPLITE: |  | ||||||
| 		case IPPROTO_TCP: |  | ||||||
| 		case IPPROTO_SCTP: |  | ||||||
| 		case IPPROTO_DCCP: |  | ||||||
| 			if (!onlyproto && (nh + offset + 4 < skb->data || |  | ||||||
| 			     pskb_may_pull(skb, nh + offset + 4 - skb->data))) { |  | ||||||
| 				__be16 *ports; |  | ||||||
| 
 |  | ||||||
| 				nh = skb_network_header(skb); |  | ||||||
| 				ports = (__be16 *)(nh + offset); |  | ||||||
| 				fl6->fl6_sport = ports[!!reverse]; |  | ||||||
| 				fl6->fl6_dport = ports[!reverse]; |  | ||||||
| 			} |  | ||||||
| 			fl6->flowi6_proto = nexthdr; |  | ||||||
| 			return; |  | ||||||
| 
 |  | ||||||
| 		case IPPROTO_ICMPV6: |  | ||||||
| 			if (!onlyproto && (nh + offset + 2 < skb->data || |  | ||||||
| 			    pskb_may_pull(skb, nh + offset + 2 - skb->data))) { |  | ||||||
| 				u8 *icmp; |  | ||||||
| 
 |  | ||||||
| 				nh = skb_network_header(skb); |  | ||||||
| 				icmp = (u8 *)(nh + offset); |  | ||||||
| 				fl6->fl6_icmp_type = icmp[0]; |  | ||||||
| 				fl6->fl6_icmp_code = icmp[1]; |  | ||||||
| 			} |  | ||||||
| 			fl6->flowi6_proto = nexthdr; |  | ||||||
| 			return; |  | ||||||
| 
 |  | ||||||
| #if IS_ENABLED(CONFIG_IPV6_MIP6) |  | ||||||
| 		case IPPROTO_MH: |  | ||||||
| 			offset += ipv6_optlen(exthdr); |  | ||||||
| 			if (!onlyproto && (nh + offset + 3 < skb->data || |  | ||||||
| 			    pskb_may_pull(skb, nh + offset + 3 - skb->data))) { |  | ||||||
| 				struct ip6_mh *mh; |  | ||||||
| 
 |  | ||||||
| 				nh = skb_network_header(skb); |  | ||||||
| 				mh = (struct ip6_mh *)(nh + offset); |  | ||||||
| 				fl6->fl6_mh_type = mh->ip6mh_type; |  | ||||||
| 			} |  | ||||||
| 			fl6->flowi6_proto = nexthdr; |  | ||||||
| 			return; |  | ||||||
| #endif |  | ||||||
| 
 |  | ||||||
| 		/* XXX Why are there these headers? */ |  | ||||||
| 		case IPPROTO_AH: |  | ||||||
| 		case IPPROTO_ESP: |  | ||||||
| 		case IPPROTO_COMP: |  | ||||||
| 		default: |  | ||||||
| 			fl6->fl6_ipsec_spi = 0; |  | ||||||
| 			fl6->flowi6_proto = nexthdr; |  | ||||||
| 			return; |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| static void xfrm6_update_pmtu(struct dst_entry *dst, struct sock *sk, | static void xfrm6_update_pmtu(struct dst_entry *dst, struct sock *sk, | ||||||
| 			      struct sk_buff *skb, u32 mtu) | 			      struct sk_buff *skb, u32 mtu) | ||||||
| { | { | ||||||
|  | @ -273,7 +168,6 @@ static const struct xfrm_policy_afinfo xfrm6_policy_afinfo = { | ||||||
| 	.dst_ops =		&xfrm6_dst_ops_template, | 	.dst_ops =		&xfrm6_dst_ops_template, | ||||||
| 	.dst_lookup =		xfrm6_dst_lookup, | 	.dst_lookup =		xfrm6_dst_lookup, | ||||||
| 	.get_saddr =		xfrm6_get_saddr, | 	.get_saddr =		xfrm6_get_saddr, | ||||||
| 	.decode_session =	_decode_session6, |  | ||||||
| 	.fill_dst =		xfrm6_fill_dst, | 	.fill_dst =		xfrm6_fill_dst, | ||||||
| 	.blackhole_route =	ip6_blackhole_route, | 	.blackhole_route =	ip6_blackhole_route, | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | @ -27,10 +27,14 @@ | ||||||
| #include <linux/cpu.h> | #include <linux/cpu.h> | ||||||
| #include <linux/audit.h> | #include <linux/audit.h> | ||||||
| #include <linux/rhashtable.h> | #include <linux/rhashtable.h> | ||||||
|  | #include <linux/if_tunnel.h> | ||||||
| #include <net/dst.h> | #include <net/dst.h> | ||||||
| #include <net/flow.h> | #include <net/flow.h> | ||||||
| #include <net/xfrm.h> | #include <net/xfrm.h> | ||||||
| #include <net/ip.h> | #include <net/ip.h> | ||||||
|  | #if IS_ENABLED(CONFIG_IPV6_MIP6) | ||||||
|  | #include <net/mip6.h> | ||||||
|  | #endif | ||||||
| #ifdef CONFIG_XFRM_STATISTICS | #ifdef CONFIG_XFRM_STATISTICS | ||||||
| #include <net/snmp.h> | #include <net/snmp.h> | ||||||
| #endif | #endif | ||||||
|  | @ -3256,20 +3260,229 @@ xfrm_policy_ok(const struct xfrm_tmpl *tmpl, const struct sec_path *sp, int star | ||||||
| 	return start; | 	return start; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static void | ||||||
|  | decode_session4(struct sk_buff *skb, struct flowi *fl, bool reverse) | ||||||
|  | { | ||||||
|  | 	const struct iphdr *iph = ip_hdr(skb); | ||||||
|  | 	u8 *xprth = skb_network_header(skb) + iph->ihl * 4; | ||||||
|  | 	struct flowi4 *fl4 = &fl->u.ip4; | ||||||
|  | 	int oif = 0; | ||||||
|  | 
 | ||||||
|  | 	if (skb_dst(skb)) | ||||||
|  | 		oif = skb_dst(skb)->dev->ifindex; | ||||||
|  | 
 | ||||||
|  | 	memset(fl4, 0, sizeof(struct flowi4)); | ||||||
|  | 	fl4->flowi4_mark = skb->mark; | ||||||
|  | 	fl4->flowi4_oif = reverse ? skb->skb_iif : oif; | ||||||
|  | 
 | ||||||
|  | 	if (!ip_is_fragment(iph)) { | ||||||
|  | 		switch (iph->protocol) { | ||||||
|  | 		case IPPROTO_UDP: | ||||||
|  | 		case IPPROTO_UDPLITE: | ||||||
|  | 		case IPPROTO_TCP: | ||||||
|  | 		case IPPROTO_SCTP: | ||||||
|  | 		case IPPROTO_DCCP: | ||||||
|  | 			if (xprth + 4 < skb->data || | ||||||
|  | 			    pskb_may_pull(skb, xprth + 4 - skb->data)) { | ||||||
|  | 				__be16 *ports; | ||||||
|  | 
 | ||||||
|  | 				xprth = skb_network_header(skb) + iph->ihl * 4; | ||||||
|  | 				ports = (__be16 *)xprth; | ||||||
|  | 
 | ||||||
|  | 				fl4->fl4_sport = ports[!!reverse]; | ||||||
|  | 				fl4->fl4_dport = ports[!reverse]; | ||||||
|  | 			} | ||||||
|  | 			break; | ||||||
|  | 		case IPPROTO_ICMP: | ||||||
|  | 			if (xprth + 2 < skb->data || | ||||||
|  | 			    pskb_may_pull(skb, xprth + 2 - skb->data)) { | ||||||
|  | 				u8 *icmp; | ||||||
|  | 
 | ||||||
|  | 				xprth = skb_network_header(skb) + iph->ihl * 4; | ||||||
|  | 				icmp = xprth; | ||||||
|  | 
 | ||||||
|  | 				fl4->fl4_icmp_type = icmp[0]; | ||||||
|  | 				fl4->fl4_icmp_code = icmp[1]; | ||||||
|  | 			} | ||||||
|  | 			break; | ||||||
|  | 		case IPPROTO_ESP: | ||||||
|  | 			if (xprth + 4 < skb->data || | ||||||
|  | 			    pskb_may_pull(skb, xprth + 4 - skb->data)) { | ||||||
|  | 				__be32 *ehdr; | ||||||
|  | 
 | ||||||
|  | 				xprth = skb_network_header(skb) + iph->ihl * 4; | ||||||
|  | 				ehdr = (__be32 *)xprth; | ||||||
|  | 
 | ||||||
|  | 				fl4->fl4_ipsec_spi = ehdr[0]; | ||||||
|  | 			} | ||||||
|  | 			break; | ||||||
|  | 		case IPPROTO_AH: | ||||||
|  | 			if (xprth + 8 < skb->data || | ||||||
|  | 			    pskb_may_pull(skb, xprth + 8 - skb->data)) { | ||||||
|  | 				__be32 *ah_hdr; | ||||||
|  | 
 | ||||||
|  | 				xprth = skb_network_header(skb) + iph->ihl * 4; | ||||||
|  | 				ah_hdr = (__be32 *)xprth; | ||||||
|  | 
 | ||||||
|  | 				fl4->fl4_ipsec_spi = ah_hdr[1]; | ||||||
|  | 			} | ||||||
|  | 			break; | ||||||
|  | 		case IPPROTO_COMP: | ||||||
|  | 			if (xprth + 4 < skb->data || | ||||||
|  | 			    pskb_may_pull(skb, xprth + 4 - skb->data)) { | ||||||
|  | 				__be16 *ipcomp_hdr; | ||||||
|  | 
 | ||||||
|  | 				xprth = skb_network_header(skb) + iph->ihl * 4; | ||||||
|  | 				ipcomp_hdr = (__be16 *)xprth; | ||||||
|  | 
 | ||||||
|  | 				fl4->fl4_ipsec_spi = htonl(ntohs(ipcomp_hdr[1])); | ||||||
|  | 			} | ||||||
|  | 			break; | ||||||
|  | 		case IPPROTO_GRE: | ||||||
|  | 			if (xprth + 12 < skb->data || | ||||||
|  | 			    pskb_may_pull(skb, xprth + 12 - skb->data)) { | ||||||
|  | 				__be16 *greflags; | ||||||
|  | 				__be32 *gre_hdr; | ||||||
|  | 
 | ||||||
|  | 				xprth = skb_network_header(skb) + iph->ihl * 4; | ||||||
|  | 				greflags = (__be16 *)xprth; | ||||||
|  | 				gre_hdr = (__be32 *)xprth; | ||||||
|  | 
 | ||||||
|  | 				if (greflags[0] & GRE_KEY) { | ||||||
|  | 					if (greflags[0] & GRE_CSUM) | ||||||
|  | 						gre_hdr++; | ||||||
|  | 					fl4->fl4_gre_key = gre_hdr[1]; | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			break; | ||||||
|  | 		default: | ||||||
|  | 			fl4->fl4_ipsec_spi = 0; | ||||||
|  | 			break; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	fl4->flowi4_proto = iph->protocol; | ||||||
|  | 	fl4->daddr = reverse ? iph->saddr : iph->daddr; | ||||||
|  | 	fl4->saddr = reverse ? iph->daddr : iph->saddr; | ||||||
|  | 	fl4->flowi4_tos = iph->tos; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #if IS_ENABLED(CONFIG_IPV6) | ||||||
|  | static void | ||||||
|  | decode_session6(struct sk_buff *skb, struct flowi *fl, bool reverse) | ||||||
|  | { | ||||||
|  | 	struct flowi6 *fl6 = &fl->u.ip6; | ||||||
|  | 	int onlyproto = 0; | ||||||
|  | 	const struct ipv6hdr *hdr = ipv6_hdr(skb); | ||||||
|  | 	u32 offset = sizeof(*hdr); | ||||||
|  | 	struct ipv6_opt_hdr *exthdr; | ||||||
|  | 	const unsigned char *nh = skb_network_header(skb); | ||||||
|  | 	u16 nhoff = IP6CB(skb)->nhoff; | ||||||
|  | 	int oif = 0; | ||||||
|  | 	u8 nexthdr; | ||||||
|  | 
 | ||||||
|  | 	if (!nhoff) | ||||||
|  | 		nhoff = offsetof(struct ipv6hdr, nexthdr); | ||||||
|  | 
 | ||||||
|  | 	nexthdr = nh[nhoff]; | ||||||
|  | 
 | ||||||
|  | 	if (skb_dst(skb)) | ||||||
|  | 		oif = skb_dst(skb)->dev->ifindex; | ||||||
|  | 
 | ||||||
|  | 	memset(fl6, 0, sizeof(struct flowi6)); | ||||||
|  | 	fl6->flowi6_mark = skb->mark; | ||||||
|  | 	fl6->flowi6_oif = reverse ? skb->skb_iif : oif; | ||||||
|  | 
 | ||||||
|  | 	fl6->daddr = reverse ? hdr->saddr : hdr->daddr; | ||||||
|  | 	fl6->saddr = reverse ? hdr->daddr : hdr->saddr; | ||||||
|  | 
 | ||||||
|  | 	while (nh + offset + sizeof(*exthdr) < skb->data || | ||||||
|  | 	       pskb_may_pull(skb, nh + offset + sizeof(*exthdr) - skb->data)) { | ||||||
|  | 		nh = skb_network_header(skb); | ||||||
|  | 		exthdr = (struct ipv6_opt_hdr *)(nh + offset); | ||||||
|  | 
 | ||||||
|  | 		switch (nexthdr) { | ||||||
|  | 		case NEXTHDR_FRAGMENT: | ||||||
|  | 			onlyproto = 1; | ||||||
|  | 			/* fall through */ | ||||||
|  | 		case NEXTHDR_ROUTING: | ||||||
|  | 		case NEXTHDR_HOP: | ||||||
|  | 		case NEXTHDR_DEST: | ||||||
|  | 			offset += ipv6_optlen(exthdr); | ||||||
|  | 			nexthdr = exthdr->nexthdr; | ||||||
|  | 			exthdr = (struct ipv6_opt_hdr *)(nh + offset); | ||||||
|  | 			break; | ||||||
|  | 		case IPPROTO_UDP: | ||||||
|  | 		case IPPROTO_UDPLITE: | ||||||
|  | 		case IPPROTO_TCP: | ||||||
|  | 		case IPPROTO_SCTP: | ||||||
|  | 		case IPPROTO_DCCP: | ||||||
|  | 			if (!onlyproto && (nh + offset + 4 < skb->data || | ||||||
|  | 			     pskb_may_pull(skb, nh + offset + 4 - skb->data))) { | ||||||
|  | 				__be16 *ports; | ||||||
|  | 
 | ||||||
|  | 				nh = skb_network_header(skb); | ||||||
|  | 				ports = (__be16 *)(nh + offset); | ||||||
|  | 				fl6->fl6_sport = ports[!!reverse]; | ||||||
|  | 				fl6->fl6_dport = ports[!reverse]; | ||||||
|  | 			} | ||||||
|  | 			fl6->flowi6_proto = nexthdr; | ||||||
|  | 			return; | ||||||
|  | 		case IPPROTO_ICMPV6: | ||||||
|  | 			if (!onlyproto && (nh + offset + 2 < skb->data || | ||||||
|  | 			    pskb_may_pull(skb, nh + offset + 2 - skb->data))) { | ||||||
|  | 				u8 *icmp; | ||||||
|  | 
 | ||||||
|  | 				nh = skb_network_header(skb); | ||||||
|  | 				icmp = (u8 *)(nh + offset); | ||||||
|  | 				fl6->fl6_icmp_type = icmp[0]; | ||||||
|  | 				fl6->fl6_icmp_code = icmp[1]; | ||||||
|  | 			} | ||||||
|  | 			fl6->flowi6_proto = nexthdr; | ||||||
|  | 			return; | ||||||
|  | #if IS_ENABLED(CONFIG_IPV6_MIP6) | ||||||
|  | 		case IPPROTO_MH: | ||||||
|  | 			offset += ipv6_optlen(exthdr); | ||||||
|  | 			if (!onlyproto && (nh + offset + 3 < skb->data || | ||||||
|  | 			    pskb_may_pull(skb, nh + offset + 3 - skb->data))) { | ||||||
|  | 				struct ip6_mh *mh; | ||||||
|  | 
 | ||||||
|  | 				nh = skb_network_header(skb); | ||||||
|  | 				mh = (struct ip6_mh *)(nh + offset); | ||||||
|  | 				fl6->fl6_mh_type = mh->ip6mh_type; | ||||||
|  | 			} | ||||||
|  | 			fl6->flowi6_proto = nexthdr; | ||||||
|  | 			return; | ||||||
|  | #endif | ||||||
|  | 		/* XXX Why are there these headers? */ | ||||||
|  | 		case IPPROTO_AH: | ||||||
|  | 		case IPPROTO_ESP: | ||||||
|  | 		case IPPROTO_COMP: | ||||||
|  | 		default: | ||||||
|  | 			fl6->fl6_ipsec_spi = 0; | ||||||
|  | 			fl6->flowi6_proto = nexthdr; | ||||||
|  | 			return; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
| int __xfrm_decode_session(struct sk_buff *skb, struct flowi *fl, | int __xfrm_decode_session(struct sk_buff *skb, struct flowi *fl, | ||||||
| 			  unsigned int family, int reverse) | 			  unsigned int family, int reverse) | ||||||
| { | { | ||||||
| 	const struct xfrm_policy_afinfo *afinfo = xfrm_policy_get_afinfo(family); | 	switch (family) { | ||||||
| 	int err; | 	case AF_INET: | ||||||
| 
 | 		decode_session4(skb, fl, reverse); | ||||||
| 	if (unlikely(afinfo == NULL)) | 		break; | ||||||
|  | #if IS_ENABLED(CONFIG_IPV6) | ||||||
|  | 	case AF_INET6: | ||||||
|  | 		decode_session6(skb, fl, reverse); | ||||||
|  | 		break; | ||||||
|  | #endif | ||||||
|  | 	default: | ||||||
| 		return -EAFNOSUPPORT; | 		return -EAFNOSUPPORT; | ||||||
|  | 	} | ||||||
| 
 | 
 | ||||||
| 	afinfo->decode_session(skb, fl, reverse); | 	return security_xfrm_decode_session(skb, &fl->flowi_secid); | ||||||
| 
 |  | ||||||
| 	err = security_xfrm_decode_session(skb, &fl->flowi_secid); |  | ||||||
| 	rcu_read_unlock(); |  | ||||||
| 	return err; |  | ||||||
| } | } | ||||||
| EXPORT_SYMBOL(__xfrm_decode_session); | EXPORT_SYMBOL(__xfrm_decode_session); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue
	
	 Florian Westphal
						Florian Westphal