ipv6: Convert to use flowi6 where applicable.
Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
		
					parent
					
						
							
								9cce96df5b
							
						
					
				
			
			
				commit
				
					
						4c9483b2fb
					
				
			
		
					 35 changed files with 622 additions and 620 deletions
				
			
		|  | @ -231,28 +231,28 @@ static int addr6_resolve(struct sockaddr_in6 *src_in, | ||||||
| 			 struct sockaddr_in6 *dst_in, | 			 struct sockaddr_in6 *dst_in, | ||||||
| 			 struct rdma_dev_addr *addr) | 			 struct rdma_dev_addr *addr) | ||||||
| { | { | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 	struct neighbour *neigh; | 	struct neighbour *neigh; | ||||||
| 	struct dst_entry *dst; | 	struct dst_entry *dst; | ||||||
| 	int ret; | 	int ret; | ||||||
| 
 | 
 | ||||||
| 	memset(&fl, 0, sizeof fl); | 	memset(&fl6, 0, sizeof fl6); | ||||||
| 	ipv6_addr_copy(&fl.fl6_dst, &dst_in->sin6_addr); | 	ipv6_addr_copy(&fl6.daddr, &dst_in->sin6_addr); | ||||||
| 	ipv6_addr_copy(&fl.fl6_src, &src_in->sin6_addr); | 	ipv6_addr_copy(&fl6.saddr, &src_in->sin6_addr); | ||||||
| 	fl.flowi_oif = addr->bound_dev_if; | 	fl6.flowi6_oif = addr->bound_dev_if; | ||||||
| 
 | 
 | ||||||
| 	dst = ip6_route_output(&init_net, NULL, &fl); | 	dst = ip6_route_output(&init_net, NULL, &fl6); | ||||||
| 	if ((ret = dst->error)) | 	if ((ret = dst->error)) | ||||||
| 		goto put; | 		goto put; | ||||||
| 
 | 
 | ||||||
| 	if (ipv6_addr_any(&fl.fl6_src)) { | 	if (ipv6_addr_any(&fl6.saddr)) { | ||||||
| 		ret = ipv6_dev_get_saddr(&init_net, ip6_dst_idev(dst)->dev, | 		ret = ipv6_dev_get_saddr(&init_net, ip6_dst_idev(dst)->dev, | ||||||
| 					 &fl.fl6_dst, 0, &fl.fl6_src); | 					 &fl6.daddr, 0, &fl6.saddr); | ||||||
| 		if (ret) | 		if (ret) | ||||||
| 			goto put; | 			goto put; | ||||||
| 
 | 
 | ||||||
| 		src_in->sin6_family = AF_INET6; | 		src_in->sin6_family = AF_INET6; | ||||||
| 		ipv6_addr_copy(&src_in->sin6_addr, &fl.fl6_src); | 		ipv6_addr_copy(&src_in->sin6_addr, &fl6.saddr); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (dst->dev->flags & IFF_LOOPBACK) { | 	if (dst->dev->flags & IFF_LOOPBACK) { | ||||||
|  |  | ||||||
|  | @ -3424,14 +3424,14 @@ static int cnic_get_v6_route(struct sockaddr_in6 *dst_addr, | ||||||
| 			     struct dst_entry **dst) | 			     struct dst_entry **dst) | ||||||
| { | { | ||||||
| #if defined(CONFIG_IPV6) || (defined(CONFIG_IPV6_MODULE) && defined(MODULE)) | #if defined(CONFIG_IPV6) || (defined(CONFIG_IPV6_MODULE) && defined(MODULE)) | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 
 | 
 | ||||||
| 	memset(&fl, 0, sizeof(fl)); | 	memset(&fl6, 0, sizeof(fl6)); | ||||||
| 	ipv6_addr_copy(&fl.fl6_dst, &dst_addr->sin6_addr); | 	ipv6_addr_copy(&fl6.daddr, &dst_addr->sin6_addr); | ||||||
| 	if (ipv6_addr_type(&fl.fl6_dst) & IPV6_ADDR_LINKLOCAL) | 	if (ipv6_addr_type(&fl6.daddr) & IPV6_ADDR_LINKLOCAL) | ||||||
| 		fl.flowi_oif = dst_addr->sin6_scope_id; | 		fl6.flowi6_oif = dst_addr->sin6_scope_id; | ||||||
| 
 | 
 | ||||||
| 	*dst = ip6_route_output(&init_net, NULL, &fl); | 	*dst = ip6_route_output(&init_net, NULL, &fl6); | ||||||
| 	if (*dst) | 	if (*dst) | ||||||
| 		return 0; | 		return 0; | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|  | @ -183,10 +183,10 @@ extern void				icmpv6_cleanup(void); | ||||||
| extern void				icmpv6_param_prob(struct sk_buff *skb, | extern void				icmpv6_param_prob(struct sk_buff *skb, | ||||||
| 							  u8 code, int pos); | 							  u8 code, int pos); | ||||||
| 
 | 
 | ||||||
| struct flowi; | struct flowi6; | ||||||
| struct in6_addr; | struct in6_addr; | ||||||
| extern void				icmpv6_flow_init(struct sock *sk, | extern void				icmpv6_flow_init(struct sock *sk, | ||||||
| 							 struct flowi *fl, | 							 struct flowi6 *fl6, | ||||||
| 							 u8 type, | 							 u8 type, | ||||||
| 							 const struct in6_addr *saddr, | 							 const struct in6_addr *saddr, | ||||||
| 							 const struct in6_addr *daddr, | 							 const struct in6_addr *daddr, | ||||||
|  |  | ||||||
|  | @ -183,7 +183,7 @@ struct fib6_table { | ||||||
| 
 | 
 | ||||||
| typedef struct rt6_info *(*pol_lookup_t)(struct net *, | typedef struct rt6_info *(*pol_lookup_t)(struct net *, | ||||||
| 					 struct fib6_table *, | 					 struct fib6_table *, | ||||||
| 					 struct flowi *, int); | 					 struct flowi6 *, int); | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
|  *	exported functions |  *	exported functions | ||||||
|  | @ -192,7 +192,7 @@ typedef struct rt6_info *(*pol_lookup_t)(struct net *, | ||||||
| extern struct fib6_table        *fib6_get_table(struct net *net, u32 id); | extern struct fib6_table        *fib6_get_table(struct net *net, u32 id); | ||||||
| extern struct fib6_table        *fib6_new_table(struct net *net, u32 id); | extern struct fib6_table        *fib6_new_table(struct net *net, u32 id); | ||||||
| extern struct dst_entry         *fib6_rule_lookup(struct net *net, | extern struct dst_entry         *fib6_rule_lookup(struct net *net, | ||||||
| 						  struct flowi *fl, int flags, | 						  struct flowi6 *fl6, int flags, | ||||||
| 						  pol_lookup_t lookup); | 						  pol_lookup_t lookup); | ||||||
| 
 | 
 | ||||||
| extern struct fib6_node		*fib6_lookup(struct fib6_node *root, | extern struct fib6_node		*fib6_lookup(struct fib6_node *root, | ||||||
|  |  | ||||||
|  | @ -71,7 +71,7 @@ extern void			ip6_route_input(struct sk_buff *skb); | ||||||
| 
 | 
 | ||||||
| extern struct dst_entry *	ip6_route_output(struct net *net, | extern struct dst_entry *	ip6_route_output(struct net *net, | ||||||
| 						 struct sock *sk, | 						 struct sock *sk, | ||||||
| 						 struct flowi *fl); | 						 struct flowi6 *fl6); | ||||||
| 
 | 
 | ||||||
| extern int			ip6_route_init(void); | extern int			ip6_route_init(void); | ||||||
| extern void			ip6_route_cleanup(void); | extern void			ip6_route_cleanup(void); | ||||||
|  |  | ||||||
|  | @ -492,7 +492,7 @@ extern int			ip6_rcv_finish(struct sk_buff *skb); | ||||||
|  */ |  */ | ||||||
| extern int			ip6_xmit(struct sock *sk, | extern int			ip6_xmit(struct sock *sk, | ||||||
| 					 struct sk_buff *skb, | 					 struct sk_buff *skb, | ||||||
| 					 struct flowi *fl, | 					 struct flowi6 *fl6, | ||||||
| 					 struct ipv6_txoptions *opt); | 					 struct ipv6_txoptions *opt); | ||||||
| 
 | 
 | ||||||
| extern int			ip6_nd_hdr(struct sock *sk, | extern int			ip6_nd_hdr(struct sock *sk, | ||||||
|  | @ -512,7 +512,7 @@ extern int			ip6_append_data(struct sock *sk, | ||||||
| 		      				int hlimit, | 		      				int hlimit, | ||||||
| 		      				int tclass, | 		      				int tclass, | ||||||
| 						struct ipv6_txoptions *opt, | 						struct ipv6_txoptions *opt, | ||||||
| 						struct flowi *fl, | 						struct flowi6 *fl6, | ||||||
| 						struct rt6_info *rt, | 						struct rt6_info *rt, | ||||||
| 						unsigned int flags, | 						unsigned int flags, | ||||||
| 						int dontfrag); | 						int dontfrag); | ||||||
|  | @ -523,13 +523,13 @@ extern void			ip6_flush_pending_frames(struct sock *sk); | ||||||
| 
 | 
 | ||||||
| extern int			ip6_dst_lookup(struct sock *sk, | extern int			ip6_dst_lookup(struct sock *sk, | ||||||
| 					       struct dst_entry **dst, | 					       struct dst_entry **dst, | ||||||
| 					       struct flowi *fl); | 					       struct flowi6 *fl6); | ||||||
| extern struct dst_entry *	ip6_dst_lookup_flow(struct sock *sk, | extern struct dst_entry *	ip6_dst_lookup_flow(struct sock *sk, | ||||||
| 						    struct flowi *fl, | 						    struct flowi6 *fl6, | ||||||
| 						    const struct in6_addr *final_dst, | 						    const struct in6_addr *final_dst, | ||||||
| 						    bool can_sleep); | 						    bool can_sleep); | ||||||
| extern struct dst_entry *	ip6_sk_dst_lookup_flow(struct sock *sk, | extern struct dst_entry *	ip6_sk_dst_lookup_flow(struct sock *sk, | ||||||
| 						       struct flowi *fl, | 						       struct flowi6 *fl6, | ||||||
| 						       const struct in6_addr *final_dst, | 						       const struct in6_addr *final_dst, | ||||||
| 						       bool can_sleep); | 						       bool can_sleep); | ||||||
| extern struct dst_entry *	ip6_blackhole_route(struct net *net, | extern struct dst_entry *	ip6_blackhole_route(struct net *net, | ||||||
|  | @ -566,7 +566,7 @@ extern int 			ipv6_ext_hdr(u8 nexthdr); | ||||||
| 
 | 
 | ||||||
| extern int ipv6_find_tlv(struct sk_buff *skb, int offset, int type); | extern int ipv6_find_tlv(struct sk_buff *skb, int offset, int type); | ||||||
| 
 | 
 | ||||||
| extern struct in6_addr *fl6_update_dst(struct flowi *fl, | extern struct in6_addr *fl6_update_dst(struct flowi6 *fl6, | ||||||
| 				       const struct ipv6_txoptions *opt, | 				       const struct ipv6_txoptions *opt, | ||||||
| 				       struct in6_addr *orig); | 				       struct in6_addr *orig); | ||||||
| 
 | 
 | ||||||
|  | @ -600,8 +600,8 @@ extern int 			ipv6_recv_error(struct sock *sk, struct msghdr *msg, int len); | ||||||
| extern int 			ipv6_recv_rxpmtu(struct sock *sk, struct msghdr *msg, int len); | extern int 			ipv6_recv_rxpmtu(struct sock *sk, struct msghdr *msg, int len); | ||||||
| extern void			ipv6_icmp_error(struct sock *sk, struct sk_buff *skb, int err, __be16 port, | extern void			ipv6_icmp_error(struct sock *sk, struct sk_buff *skb, int err, __be16 port, | ||||||
| 						u32 info, u8 *payload); | 						u32 info, u8 *payload); | ||||||
| extern void			ipv6_local_error(struct sock *sk, int err, struct flowi *fl, u32 info); | extern void			ipv6_local_error(struct sock *sk, int err, struct flowi6 *fl6, u32 info); | ||||||
| extern void			ipv6_local_rxpmtu(struct sock *sk, struct flowi *fl, u32 mtu); | extern void			ipv6_local_rxpmtu(struct sock *sk, struct flowi6 *fl6, u32 mtu); | ||||||
| 
 | 
 | ||||||
| extern int inet6_release(struct socket *sock); | extern int inet6_release(struct socket *sock); | ||||||
| extern int inet6_bind(struct socket *sock, struct sockaddr *uaddr,  | extern int inet6_bind(struct socket *sock, struct sockaddr *uaddr,  | ||||||
|  |  | ||||||
|  | @ -14,7 +14,7 @@ extern struct proto udpv6_prot; | ||||||
| extern struct proto udplitev6_prot; | extern struct proto udplitev6_prot; | ||||||
| extern struct proto tcpv6_prot; | extern struct proto tcpv6_prot; | ||||||
| 
 | 
 | ||||||
| struct flowi; | struct flowi6; | ||||||
| 
 | 
 | ||||||
| /* extention headers */ | /* extention headers */ | ||||||
| extern int				ipv6_exthdrs_init(void); | extern int				ipv6_exthdrs_init(void); | ||||||
|  | @ -42,7 +42,7 @@ extern int			datagram_recv_ctl(struct sock *sk, | ||||||
| 
 | 
 | ||||||
| extern int			datagram_send_ctl(struct net *net, | extern int			datagram_send_ctl(struct net *net, | ||||||
| 						  struct msghdr *msg, | 						  struct msghdr *msg, | ||||||
| 						  struct flowi *fl, | 						  struct flowi6 *fl6, | ||||||
| 						  struct ipv6_txoptions *opt, | 						  struct ipv6_txoptions *opt, | ||||||
| 						  int *hlimit, int *tclass, | 						  int *hlimit, int *tclass, | ||||||
| 						  int *dontfrag); | 						  int *dontfrag); | ||||||
|  |  | ||||||
							
								
								
									
										126
									
								
								net/dccp/ipv6.c
									
										
									
									
									
								
							
							
						
						
									
										126
									
								
								net/dccp/ipv6.c
									
										
									
									
									
								
							|  | @ -147,22 +147,22 @@ static void dccp_v6_err(struct sk_buff *skb, struct inet6_skb_parm *opt, | ||||||
| 		dst = __sk_dst_check(sk, np->dst_cookie); | 		dst = __sk_dst_check(sk, np->dst_cookie); | ||||||
| 		if (dst == NULL) { | 		if (dst == NULL) { | ||||||
| 			struct inet_sock *inet = inet_sk(sk); | 			struct inet_sock *inet = inet_sk(sk); | ||||||
| 			struct flowi fl; | 			struct flowi6 fl6; | ||||||
| 
 | 
 | ||||||
| 			/* BUGGG_FUTURE: Again, it is not clear how
 | 			/* BUGGG_FUTURE: Again, it is not clear how
 | ||||||
| 			   to handle rthdr case. Ignore this complexity | 			   to handle rthdr case. Ignore this complexity | ||||||
| 			   for now. | 			   for now. | ||||||
| 			 */ | 			 */ | ||||||
| 			memset(&fl, 0, sizeof(fl)); | 			memset(&fl6, 0, sizeof(fl6)); | ||||||
| 			fl.flowi_proto = IPPROTO_DCCP; | 			fl6.flowi6_proto = IPPROTO_DCCP; | ||||||
| 			ipv6_addr_copy(&fl.fl6_dst, &np->daddr); | 			ipv6_addr_copy(&fl6.daddr, &np->daddr); | ||||||
| 			ipv6_addr_copy(&fl.fl6_src, &np->saddr); | 			ipv6_addr_copy(&fl6.saddr, &np->saddr); | ||||||
| 			fl.flowi_oif = sk->sk_bound_dev_if; | 			fl6.flowi6_oif = sk->sk_bound_dev_if; | ||||||
| 			fl.fl6_dport = inet->inet_dport; | 			fl6.uli.ports.dport = inet->inet_dport; | ||||||
| 			fl.fl6_sport = inet->inet_sport; | 			fl6.uli.ports.sport = inet->inet_sport; | ||||||
| 			security_sk_classify_flow(sk, &fl); | 			security_sk_classify_flow(sk, flowi6_to_flowi(&fl6)); | ||||||
| 
 | 
 | ||||||
| 			dst = ip6_dst_lookup_flow(sk, &fl, NULL, false); | 			dst = ip6_dst_lookup_flow(sk, &fl6, NULL, false); | ||||||
| 			if (IS_ERR(dst)) { | 			if (IS_ERR(dst)) { | ||||||
| 				sk->sk_err_soft = -PTR_ERR(dst); | 				sk->sk_err_soft = -PTR_ERR(dst); | ||||||
| 				goto out; | 				goto out; | ||||||
|  | @ -243,25 +243,25 @@ static int dccp_v6_send_response(struct sock *sk, struct request_sock *req, | ||||||
| 	struct sk_buff *skb; | 	struct sk_buff *skb; | ||||||
| 	struct ipv6_txoptions *opt = NULL; | 	struct ipv6_txoptions *opt = NULL; | ||||||
| 	struct in6_addr *final_p, final; | 	struct in6_addr *final_p, final; | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 	int err = -1; | 	int err = -1; | ||||||
| 	struct dst_entry *dst; | 	struct dst_entry *dst; | ||||||
| 
 | 
 | ||||||
| 	memset(&fl, 0, sizeof(fl)); | 	memset(&fl6, 0, sizeof(fl6)); | ||||||
| 	fl.flowi_proto = IPPROTO_DCCP; | 	fl6.flowi6_proto = IPPROTO_DCCP; | ||||||
| 	ipv6_addr_copy(&fl.fl6_dst, &ireq6->rmt_addr); | 	ipv6_addr_copy(&fl6.daddr, &ireq6->rmt_addr); | ||||||
| 	ipv6_addr_copy(&fl.fl6_src, &ireq6->loc_addr); | 	ipv6_addr_copy(&fl6.saddr, &ireq6->loc_addr); | ||||||
| 	fl.fl6_flowlabel = 0; | 	fl6.flowlabel = 0; | ||||||
| 	fl.flowi_oif = ireq6->iif; | 	fl6.flowi6_oif = ireq6->iif; | ||||||
| 	fl.fl6_dport = inet_rsk(req)->rmt_port; | 	fl6.uli.ports.dport = inet_rsk(req)->rmt_port; | ||||||
| 	fl.fl6_sport = inet_rsk(req)->loc_port; | 	fl6.uli.ports.sport = inet_rsk(req)->loc_port; | ||||||
| 	security_req_classify_flow(req, &fl); | 	security_req_classify_flow(req, flowi6_to_flowi(&fl6)); | ||||||
| 
 | 
 | ||||||
| 	opt = np->opt; | 	opt = np->opt; | ||||||
| 
 | 
 | ||||||
| 	final_p = fl6_update_dst(&fl, opt, &final); | 	final_p = fl6_update_dst(&fl6, opt, &final); | ||||||
| 
 | 
 | ||||||
| 	dst = ip6_dst_lookup_flow(sk, &fl, final_p, false); | 	dst = ip6_dst_lookup_flow(sk, &fl6, final_p, false); | ||||||
| 	if (IS_ERR(dst)) { | 	if (IS_ERR(dst)) { | ||||||
| 		err = PTR_ERR(dst); | 		err = PTR_ERR(dst); | ||||||
| 		dst = NULL; | 		dst = NULL; | ||||||
|  | @ -275,8 +275,8 @@ static int dccp_v6_send_response(struct sock *sk, struct request_sock *req, | ||||||
| 		dh->dccph_checksum = dccp_v6_csum_finish(skb, | 		dh->dccph_checksum = dccp_v6_csum_finish(skb, | ||||||
| 							 &ireq6->loc_addr, | 							 &ireq6->loc_addr, | ||||||
| 							 &ireq6->rmt_addr); | 							 &ireq6->rmt_addr); | ||||||
| 		ipv6_addr_copy(&fl.fl6_dst, &ireq6->rmt_addr); | 		ipv6_addr_copy(&fl6.daddr, &ireq6->rmt_addr); | ||||||
| 		err = ip6_xmit(sk, skb, &fl, opt); | 		err = ip6_xmit(sk, skb, &fl6, opt); | ||||||
| 		err = net_xmit_eval(err); | 		err = net_xmit_eval(err); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | @ -298,7 +298,7 @@ static void dccp_v6_ctl_send_reset(struct sock *sk, struct sk_buff *rxskb) | ||||||
| { | { | ||||||
| 	struct ipv6hdr *rxip6h; | 	struct ipv6hdr *rxip6h; | ||||||
| 	struct sk_buff *skb; | 	struct sk_buff *skb; | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 	struct net *net = dev_net(skb_dst(rxskb)->dev); | 	struct net *net = dev_net(skb_dst(rxskb)->dev); | ||||||
| 	struct sock *ctl_sk = net->dccp.v6_ctl_sk; | 	struct sock *ctl_sk = net->dccp.v6_ctl_sk; | ||||||
| 	struct dst_entry *dst; | 	struct dst_entry *dst; | ||||||
|  | @ -317,21 +317,21 @@ static void dccp_v6_ctl_send_reset(struct sock *sk, struct sk_buff *rxskb) | ||||||
| 	dccp_hdr(skb)->dccph_checksum = dccp_v6_csum_finish(skb, &rxip6h->saddr, | 	dccp_hdr(skb)->dccph_checksum = dccp_v6_csum_finish(skb, &rxip6h->saddr, | ||||||
| 							    &rxip6h->daddr); | 							    &rxip6h->daddr); | ||||||
| 
 | 
 | ||||||
| 	memset(&fl, 0, sizeof(fl)); | 	memset(&fl6, 0, sizeof(fl6)); | ||||||
| 	ipv6_addr_copy(&fl.fl6_dst, &rxip6h->saddr); | 	ipv6_addr_copy(&fl6.daddr, &rxip6h->saddr); | ||||||
| 	ipv6_addr_copy(&fl.fl6_src, &rxip6h->daddr); | 	ipv6_addr_copy(&fl6.saddr, &rxip6h->daddr); | ||||||
| 
 | 
 | ||||||
| 	fl.flowi_proto = IPPROTO_DCCP; | 	fl6.flowi6_proto = IPPROTO_DCCP; | ||||||
| 	fl.flowi_oif = inet6_iif(rxskb); | 	fl6.flowi6_oif = inet6_iif(rxskb); | ||||||
| 	fl.fl6_dport = dccp_hdr(skb)->dccph_dport; | 	fl6.uli.ports.dport = dccp_hdr(skb)->dccph_dport; | ||||||
| 	fl.fl6_sport = dccp_hdr(skb)->dccph_sport; | 	fl6.uli.ports.sport = dccp_hdr(skb)->dccph_sport; | ||||||
| 	security_skb_classify_flow(rxskb, &fl); | 	security_skb_classify_flow(rxskb, flowi6_to_flowi(&fl6)); | ||||||
| 
 | 
 | ||||||
| 	/* sk = NULL, but it is safe for now. RST socket required. */ | 	/* sk = NULL, but it is safe for now. RST socket required. */ | ||||||
| 	dst = ip6_dst_lookup_flow(ctl_sk, &fl, NULL, false); | 	dst = ip6_dst_lookup_flow(ctl_sk, &fl6, NULL, false); | ||||||
| 	if (!IS_ERR(dst)) { | 	if (!IS_ERR(dst)) { | ||||||
| 		skb_dst_set(skb, dst); | 		skb_dst_set(skb, dst); | ||||||
| 		ip6_xmit(ctl_sk, skb, &fl, NULL); | 		ip6_xmit(ctl_sk, skb, &fl6, NULL); | ||||||
| 		DCCP_INC_STATS_BH(DCCP_MIB_OUTSEGS); | 		DCCP_INC_STATS_BH(DCCP_MIB_OUTSEGS); | ||||||
| 		DCCP_INC_STATS_BH(DCCP_MIB_OUTRSTS); | 		DCCP_INC_STATS_BH(DCCP_MIB_OUTRSTS); | ||||||
| 		return; | 		return; | ||||||
|  | @ -527,19 +527,19 @@ static struct sock *dccp_v6_request_recv_sock(struct sock *sk, | ||||||
| 
 | 
 | ||||||
| 	if (dst == NULL) { | 	if (dst == NULL) { | ||||||
| 		struct in6_addr *final_p, final; | 		struct in6_addr *final_p, final; | ||||||
| 		struct flowi fl; | 		struct flowi6 fl6; | ||||||
| 
 | 
 | ||||||
| 		memset(&fl, 0, sizeof(fl)); | 		memset(&fl6, 0, sizeof(fl6)); | ||||||
| 		fl.flowi_proto = IPPROTO_DCCP; | 		fl6.flowi6_proto = IPPROTO_DCCP; | ||||||
| 		ipv6_addr_copy(&fl.fl6_dst, &ireq6->rmt_addr); | 		ipv6_addr_copy(&fl6.daddr, &ireq6->rmt_addr); | ||||||
| 		final_p = fl6_update_dst(&fl, opt, &final); | 		final_p = fl6_update_dst(&fl6, opt, &final); | ||||||
| 		ipv6_addr_copy(&fl.fl6_src, &ireq6->loc_addr); | 		ipv6_addr_copy(&fl6.saddr, &ireq6->loc_addr); | ||||||
| 		fl.flowi_oif = sk->sk_bound_dev_if; | 		fl6.flowi6_oif = sk->sk_bound_dev_if; | ||||||
| 		fl.fl6_dport = inet_rsk(req)->rmt_port; | 		fl6.uli.ports.dport = inet_rsk(req)->rmt_port; | ||||||
| 		fl.fl6_sport = inet_rsk(req)->loc_port; | 		fl6.uli.ports.sport = inet_rsk(req)->loc_port; | ||||||
| 		security_sk_classify_flow(sk, &fl); | 		security_sk_classify_flow(sk, flowi6_to_flowi(&fl6)); | ||||||
| 
 | 
 | ||||||
| 		dst = ip6_dst_lookup_flow(sk, &fl, final_p, false); | 		dst = ip6_dst_lookup_flow(sk, &fl6, final_p, false); | ||||||
| 		if (IS_ERR(dst)) | 		if (IS_ERR(dst)) | ||||||
| 			goto out; | 			goto out; | ||||||
| 	} | 	} | ||||||
|  | @ -859,7 +859,7 @@ static int dccp_v6_connect(struct sock *sk, struct sockaddr *uaddr, | ||||||
| 	struct ipv6_pinfo *np = inet6_sk(sk); | 	struct ipv6_pinfo *np = inet6_sk(sk); | ||||||
| 	struct dccp_sock *dp = dccp_sk(sk); | 	struct dccp_sock *dp = dccp_sk(sk); | ||||||
| 	struct in6_addr *saddr = NULL, *final_p, final; | 	struct in6_addr *saddr = NULL, *final_p, final; | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 	struct dst_entry *dst; | 	struct dst_entry *dst; | ||||||
| 	int addr_type; | 	int addr_type; | ||||||
| 	int err; | 	int err; | ||||||
|  | @ -872,14 +872,14 @@ static int dccp_v6_connect(struct sock *sk, struct sockaddr *uaddr, | ||||||
| 	if (usin->sin6_family != AF_INET6) | 	if (usin->sin6_family != AF_INET6) | ||||||
| 		return -EAFNOSUPPORT; | 		return -EAFNOSUPPORT; | ||||||
| 
 | 
 | ||||||
| 	memset(&fl, 0, sizeof(fl)); | 	memset(&fl6, 0, sizeof(fl6)); | ||||||
| 
 | 
 | ||||||
| 	if (np->sndflow) { | 	if (np->sndflow) { | ||||||
| 		fl.fl6_flowlabel = usin->sin6_flowinfo & IPV6_FLOWINFO_MASK; | 		fl6.flowlabel = usin->sin6_flowinfo & IPV6_FLOWINFO_MASK; | ||||||
| 		IP6_ECN_flow_init(fl.fl6_flowlabel); | 		IP6_ECN_flow_init(fl6.flowlabel); | ||||||
| 		if (fl.fl6_flowlabel & IPV6_FLOWLABEL_MASK) { | 		if (fl6.flowlabel & IPV6_FLOWLABEL_MASK) { | ||||||
| 			struct ip6_flowlabel *flowlabel; | 			struct ip6_flowlabel *flowlabel; | ||||||
| 			flowlabel = fl6_sock_lookup(sk, fl.fl6_flowlabel); | 			flowlabel = fl6_sock_lookup(sk, fl6.flowlabel); | ||||||
| 			if (flowlabel == NULL) | 			if (flowlabel == NULL) | ||||||
| 				return -EINVAL; | 				return -EINVAL; | ||||||
| 			ipv6_addr_copy(&usin->sin6_addr, &flowlabel->dst); | 			ipv6_addr_copy(&usin->sin6_addr, &flowlabel->dst); | ||||||
|  | @ -916,7 +916,7 @@ static int dccp_v6_connect(struct sock *sk, struct sockaddr *uaddr, | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	ipv6_addr_copy(&np->daddr, &usin->sin6_addr); | 	ipv6_addr_copy(&np->daddr, &usin->sin6_addr); | ||||||
| 	np->flow_label = fl.fl6_flowlabel; | 	np->flow_label = fl6.flowlabel; | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * DCCP over IPv4 | 	 * DCCP over IPv4 | ||||||
|  | @ -953,24 +953,24 @@ static int dccp_v6_connect(struct sock *sk, struct sockaddr *uaddr, | ||||||
| 	if (!ipv6_addr_any(&np->rcv_saddr)) | 	if (!ipv6_addr_any(&np->rcv_saddr)) | ||||||
| 		saddr = &np->rcv_saddr; | 		saddr = &np->rcv_saddr; | ||||||
| 
 | 
 | ||||||
| 	fl.flowi_proto = IPPROTO_DCCP; | 	fl6.flowi6_proto = IPPROTO_DCCP; | ||||||
| 	ipv6_addr_copy(&fl.fl6_dst, &np->daddr); | 	ipv6_addr_copy(&fl6.daddr, &np->daddr); | ||||||
| 	ipv6_addr_copy(&fl.fl6_src, saddr ? saddr : &np->saddr); | 	ipv6_addr_copy(&fl6.saddr, saddr ? saddr : &np->saddr); | ||||||
| 	fl.flowi_oif = sk->sk_bound_dev_if; | 	fl6.flowi6_oif = sk->sk_bound_dev_if; | ||||||
| 	fl.fl6_dport = usin->sin6_port; | 	fl6.uli.ports.dport = usin->sin6_port; | ||||||
| 	fl.fl6_sport = inet->inet_sport; | 	fl6.uli.ports.sport = inet->inet_sport; | ||||||
| 	security_sk_classify_flow(sk, &fl); | 	security_sk_classify_flow(sk, flowi6_to_flowi(&fl6)); | ||||||
| 
 | 
 | ||||||
| 	final_p = fl6_update_dst(&fl, np->opt, &final); | 	final_p = fl6_update_dst(&fl6, np->opt, &final); | ||||||
| 
 | 
 | ||||||
| 	dst = ip6_dst_lookup_flow(sk, &fl, final_p, true); | 	dst = ip6_dst_lookup_flow(sk, &fl6, final_p, true); | ||||||
| 	if (IS_ERR(dst)) { | 	if (IS_ERR(dst)) { | ||||||
| 		err = PTR_ERR(dst); | 		err = PTR_ERR(dst); | ||||||
| 		goto failure; | 		goto failure; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (saddr == NULL) { | 	if (saddr == NULL) { | ||||||
| 		saddr = &fl.fl6_src; | 		saddr = &fl6.saddr; | ||||||
| 		ipv6_addr_copy(&np->rcv_saddr, saddr); | 		ipv6_addr_copy(&np->rcv_saddr, saddr); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -652,22 +652,22 @@ int inet6_sk_rebuild_header(struct sock *sk) | ||||||
| 	if (dst == NULL) { | 	if (dst == NULL) { | ||||||
| 		struct inet_sock *inet = inet_sk(sk); | 		struct inet_sock *inet = inet_sk(sk); | ||||||
| 		struct in6_addr *final_p, final; | 		struct in6_addr *final_p, final; | ||||||
| 		struct flowi fl; | 		struct flowi6 fl6; | ||||||
| 
 | 
 | ||||||
| 		memset(&fl, 0, sizeof(fl)); | 		memset(&fl6, 0, sizeof(fl6)); | ||||||
| 		fl.flowi_proto = sk->sk_protocol; | 		fl6.flowi6_proto = sk->sk_protocol; | ||||||
| 		ipv6_addr_copy(&fl.fl6_dst, &np->daddr); | 		ipv6_addr_copy(&fl6.daddr, &np->daddr); | ||||||
| 		ipv6_addr_copy(&fl.fl6_src, &np->saddr); | 		ipv6_addr_copy(&fl6.saddr, &np->saddr); | ||||||
| 		fl.fl6_flowlabel = np->flow_label; | 		fl6.flowlabel = np->flow_label; | ||||||
| 		fl.flowi_oif = sk->sk_bound_dev_if; | 		fl6.flowi6_oif = sk->sk_bound_dev_if; | ||||||
| 		fl.flowi_mark = sk->sk_mark; | 		fl6.flowi6_mark = sk->sk_mark; | ||||||
| 		fl.fl6_dport = inet->inet_dport; | 		fl6.uli.ports.dport = inet->inet_dport; | ||||||
| 		fl.fl6_sport = inet->inet_sport; | 		fl6.uli.ports.sport = inet->inet_sport; | ||||||
| 		security_sk_classify_flow(sk, &fl); | 		security_sk_classify_flow(sk, flowi6_to_flowi(&fl6)); | ||||||
| 
 | 
 | ||||||
| 		final_p = fl6_update_dst(&fl, np->opt, &final); | 		final_p = fl6_update_dst(&fl6, np->opt, &final); | ||||||
| 
 | 
 | ||||||
| 		dst = ip6_dst_lookup_flow(sk, &fl, final_p, false); | 		dst = ip6_dst_lookup_flow(sk, &fl6, final_p, false); | ||||||
| 		if (IS_ERR(dst)) { | 		if (IS_ERR(dst)) { | ||||||
| 			sk->sk_route_caps = 0; | 			sk->sk_route_caps = 0; | ||||||
| 			sk->sk_err_soft = -PTR_ERR(dst); | 			sk->sk_err_soft = -PTR_ERR(dst); | ||||||
|  |  | ||||||
|  | @ -40,7 +40,7 @@ int ip6_datagram_connect(struct sock *sk, struct sockaddr *uaddr, int addr_len) | ||||||
| 	struct ipv6_pinfo      	*np = inet6_sk(sk); | 	struct ipv6_pinfo      	*np = inet6_sk(sk); | ||||||
| 	struct in6_addr		*daddr, *final_p, final; | 	struct in6_addr		*daddr, *final_p, final; | ||||||
| 	struct dst_entry	*dst; | 	struct dst_entry	*dst; | ||||||
| 	struct flowi		fl; | 	struct flowi6		fl6; | ||||||
| 	struct ip6_flowlabel	*flowlabel = NULL; | 	struct ip6_flowlabel	*flowlabel = NULL; | ||||||
| 	struct ipv6_txoptions   *opt; | 	struct ipv6_txoptions   *opt; | ||||||
| 	int			addr_type; | 	int			addr_type; | ||||||
|  | @ -59,11 +59,11 @@ int ip6_datagram_connect(struct sock *sk, struct sockaddr *uaddr, int addr_len) | ||||||
| 	if (usin->sin6_family != AF_INET6) | 	if (usin->sin6_family != AF_INET6) | ||||||
| 		return -EAFNOSUPPORT; | 		return -EAFNOSUPPORT; | ||||||
| 
 | 
 | ||||||
| 	memset(&fl, 0, sizeof(fl)); | 	memset(&fl6, 0, sizeof(fl6)); | ||||||
| 	if (np->sndflow) { | 	if (np->sndflow) { | ||||||
| 		fl.fl6_flowlabel = usin->sin6_flowinfo&IPV6_FLOWINFO_MASK; | 		fl6.flowlabel = usin->sin6_flowinfo&IPV6_FLOWINFO_MASK; | ||||||
| 		if (fl.fl6_flowlabel&IPV6_FLOWLABEL_MASK) { | 		if (fl6.flowlabel&IPV6_FLOWLABEL_MASK) { | ||||||
| 			flowlabel = fl6_sock_lookup(sk, fl.fl6_flowlabel); | 			flowlabel = fl6_sock_lookup(sk, fl6.flowlabel); | ||||||
| 			if (flowlabel == NULL) | 			if (flowlabel == NULL) | ||||||
| 				return -EINVAL; | 				return -EINVAL; | ||||||
| 			ipv6_addr_copy(&usin->sin6_addr, &flowlabel->dst); | 			ipv6_addr_copy(&usin->sin6_addr, &flowlabel->dst); | ||||||
|  | @ -137,7 +137,7 @@ ipv4_connected: | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	ipv6_addr_copy(&np->daddr, daddr); | 	ipv6_addr_copy(&np->daddr, daddr); | ||||||
| 	np->flow_label = fl.fl6_flowlabel; | 	np->flow_label = fl6.flowlabel; | ||||||
| 
 | 
 | ||||||
| 	inet->inet_dport = usin->sin6_port; | 	inet->inet_dport = usin->sin6_port; | ||||||
| 
 | 
 | ||||||
|  | @ -146,23 +146,23 @@ ipv4_connected: | ||||||
| 	 *	destination cache for it. | 	 *	destination cache for it. | ||||||
| 	 */ | 	 */ | ||||||
| 
 | 
 | ||||||
| 	fl.flowi_proto = sk->sk_protocol; | 	fl6.flowi6_proto = sk->sk_protocol; | ||||||
| 	ipv6_addr_copy(&fl.fl6_dst, &np->daddr); | 	ipv6_addr_copy(&fl6.daddr, &np->daddr); | ||||||
| 	ipv6_addr_copy(&fl.fl6_src, &np->saddr); | 	ipv6_addr_copy(&fl6.saddr, &np->saddr); | ||||||
| 	fl.flowi_oif = sk->sk_bound_dev_if; | 	fl6.flowi6_oif = sk->sk_bound_dev_if; | ||||||
| 	fl.flowi_mark = sk->sk_mark; | 	fl6.flowi6_mark = sk->sk_mark; | ||||||
| 	fl.fl6_dport = inet->inet_dport; | 	fl6.uli.ports.dport = inet->inet_dport; | ||||||
| 	fl.fl6_sport = inet->inet_sport; | 	fl6.uli.ports.sport = inet->inet_sport; | ||||||
| 
 | 
 | ||||||
| 	if (!fl.flowi_oif && (addr_type&IPV6_ADDR_MULTICAST)) | 	if (!fl6.flowi6_oif && (addr_type&IPV6_ADDR_MULTICAST)) | ||||||
| 		fl.flowi_oif = np->mcast_oif; | 		fl6.flowi6_oif = np->mcast_oif; | ||||||
| 
 | 
 | ||||||
| 	security_sk_classify_flow(sk, &fl); | 	security_sk_classify_flow(sk, flowi6_to_flowi(&fl6)); | ||||||
| 
 | 
 | ||||||
| 	opt = flowlabel ? flowlabel->opt : np->opt; | 	opt = flowlabel ? flowlabel->opt : np->opt; | ||||||
| 	final_p = fl6_update_dst(&fl, opt, &final); | 	final_p = fl6_update_dst(&fl6, opt, &final); | ||||||
| 
 | 
 | ||||||
| 	dst = ip6_dst_lookup_flow(sk, &fl, final_p, true); | 	dst = ip6_dst_lookup_flow(sk, &fl6, final_p, true); | ||||||
| 	err = 0; | 	err = 0; | ||||||
| 	if (IS_ERR(dst)) { | 	if (IS_ERR(dst)) { | ||||||
| 		err = PTR_ERR(dst); | 		err = PTR_ERR(dst); | ||||||
|  | @ -172,20 +172,20 @@ ipv4_connected: | ||||||
| 	/* source address lookup done in ip6_dst_lookup */ | 	/* source address lookup done in ip6_dst_lookup */ | ||||||
| 
 | 
 | ||||||
| 	if (ipv6_addr_any(&np->saddr)) | 	if (ipv6_addr_any(&np->saddr)) | ||||||
| 		ipv6_addr_copy(&np->saddr, &fl.fl6_src); | 		ipv6_addr_copy(&np->saddr, &fl6.saddr); | ||||||
| 
 | 
 | ||||||
| 	if (ipv6_addr_any(&np->rcv_saddr)) { | 	if (ipv6_addr_any(&np->rcv_saddr)) { | ||||||
| 		ipv6_addr_copy(&np->rcv_saddr, &fl.fl6_src); | 		ipv6_addr_copy(&np->rcv_saddr, &fl6.saddr); | ||||||
| 		inet->inet_rcv_saddr = LOOPBACK4_IPV6; | 		inet->inet_rcv_saddr = LOOPBACK4_IPV6; | ||||||
| 		if (sk->sk_prot->rehash) | 		if (sk->sk_prot->rehash) | ||||||
| 			sk->sk_prot->rehash(sk); | 			sk->sk_prot->rehash(sk); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	ip6_dst_store(sk, dst, | 	ip6_dst_store(sk, dst, | ||||||
| 		      ipv6_addr_equal(&fl.fl6_dst, &np->daddr) ? | 		      ipv6_addr_equal(&fl6.daddr, &np->daddr) ? | ||||||
| 		      &np->daddr : NULL, | 		      &np->daddr : NULL, | ||||||
| #ifdef CONFIG_IPV6_SUBTREES | #ifdef CONFIG_IPV6_SUBTREES | ||||||
| 		      ipv6_addr_equal(&fl.fl6_src, &np->saddr) ? | 		      ipv6_addr_equal(&fl6.saddr, &np->saddr) ? | ||||||
| 		      &np->saddr : | 		      &np->saddr : | ||||||
| #endif | #endif | ||||||
| 		      NULL); | 		      NULL); | ||||||
|  | @ -231,7 +231,7 @@ void ipv6_icmp_error(struct sock *sk, struct sk_buff *skb, int err, | ||||||
| 		kfree_skb(skb); | 		kfree_skb(skb); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void ipv6_local_error(struct sock *sk, int err, struct flowi *fl, u32 info) | void ipv6_local_error(struct sock *sk, int err, struct flowi6 *fl6, u32 info) | ||||||
| { | { | ||||||
| 	struct ipv6_pinfo *np = inet6_sk(sk); | 	struct ipv6_pinfo *np = inet6_sk(sk); | ||||||
| 	struct sock_exterr_skb *serr; | 	struct sock_exterr_skb *serr; | ||||||
|  | @ -250,7 +250,7 @@ void ipv6_local_error(struct sock *sk, int err, struct flowi *fl, u32 info) | ||||||
| 	skb_put(skb, sizeof(struct ipv6hdr)); | 	skb_put(skb, sizeof(struct ipv6hdr)); | ||||||
| 	skb_reset_network_header(skb); | 	skb_reset_network_header(skb); | ||||||
| 	iph = ipv6_hdr(skb); | 	iph = ipv6_hdr(skb); | ||||||
| 	ipv6_addr_copy(&iph->daddr, &fl->fl6_dst); | 	ipv6_addr_copy(&iph->daddr, &fl6->daddr); | ||||||
| 
 | 
 | ||||||
| 	serr = SKB_EXT_ERR(skb); | 	serr = SKB_EXT_ERR(skb); | ||||||
| 	serr->ee.ee_errno = err; | 	serr->ee.ee_errno = err; | ||||||
|  | @ -261,7 +261,7 @@ void ipv6_local_error(struct sock *sk, int err, struct flowi *fl, u32 info) | ||||||
| 	serr->ee.ee_info = info; | 	serr->ee.ee_info = info; | ||||||
| 	serr->ee.ee_data = 0; | 	serr->ee.ee_data = 0; | ||||||
| 	serr->addr_offset = (u8 *)&iph->daddr - skb_network_header(skb); | 	serr->addr_offset = (u8 *)&iph->daddr - skb_network_header(skb); | ||||||
| 	serr->port = fl->fl6_dport; | 	serr->port = fl6->uli.ports.dport; | ||||||
| 
 | 
 | ||||||
| 	__skb_pull(skb, skb_tail_pointer(skb) - skb->data); | 	__skb_pull(skb, skb_tail_pointer(skb) - skb->data); | ||||||
| 	skb_reset_transport_header(skb); | 	skb_reset_transport_header(skb); | ||||||
|  | @ -270,7 +270,7 @@ void ipv6_local_error(struct sock *sk, int err, struct flowi *fl, u32 info) | ||||||
| 		kfree_skb(skb); | 		kfree_skb(skb); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void ipv6_local_rxpmtu(struct sock *sk, struct flowi *fl, u32 mtu) | void ipv6_local_rxpmtu(struct sock *sk, struct flowi6 *fl6, u32 mtu) | ||||||
| { | { | ||||||
| 	struct ipv6_pinfo *np = inet6_sk(sk); | 	struct ipv6_pinfo *np = inet6_sk(sk); | ||||||
| 	struct ipv6hdr *iph; | 	struct ipv6hdr *iph; | ||||||
|  | @ -287,7 +287,7 @@ void ipv6_local_rxpmtu(struct sock *sk, struct flowi *fl, u32 mtu) | ||||||
| 	skb_put(skb, sizeof(struct ipv6hdr)); | 	skb_put(skb, sizeof(struct ipv6hdr)); | ||||||
| 	skb_reset_network_header(skb); | 	skb_reset_network_header(skb); | ||||||
| 	iph = ipv6_hdr(skb); | 	iph = ipv6_hdr(skb); | ||||||
| 	ipv6_addr_copy(&iph->daddr, &fl->fl6_dst); | 	ipv6_addr_copy(&iph->daddr, &fl6->daddr); | ||||||
| 
 | 
 | ||||||
| 	mtu_info = IP6CBMTU(skb); | 	mtu_info = IP6CBMTU(skb); | ||||||
| 	if (!mtu_info) { | 	if (!mtu_info) { | ||||||
|  | @ -299,7 +299,7 @@ void ipv6_local_rxpmtu(struct sock *sk, struct flowi *fl, u32 mtu) | ||||||
| 	mtu_info->ip6m_addr.sin6_family = AF_INET6; | 	mtu_info->ip6m_addr.sin6_family = AF_INET6; | ||||||
| 	mtu_info->ip6m_addr.sin6_port = 0; | 	mtu_info->ip6m_addr.sin6_port = 0; | ||||||
| 	mtu_info->ip6m_addr.sin6_flowinfo = 0; | 	mtu_info->ip6m_addr.sin6_flowinfo = 0; | ||||||
| 	mtu_info->ip6m_addr.sin6_scope_id = fl->flowi_oif; | 	mtu_info->ip6m_addr.sin6_scope_id = fl6->flowi6_oif; | ||||||
| 	ipv6_addr_copy(&mtu_info->ip6m_addr.sin6_addr, &ipv6_hdr(skb)->daddr); | 	ipv6_addr_copy(&mtu_info->ip6m_addr.sin6_addr, &ipv6_hdr(skb)->daddr); | ||||||
| 
 | 
 | ||||||
| 	__skb_pull(skb, skb_tail_pointer(skb) - skb->data); | 	__skb_pull(skb, skb_tail_pointer(skb) - skb->data); | ||||||
|  | @ -593,7 +593,7 @@ int datagram_recv_ctl(struct sock *sk, struct msghdr *msg, struct sk_buff *skb) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| int datagram_send_ctl(struct net *net, | int datagram_send_ctl(struct net *net, | ||||||
| 		      struct msghdr *msg, struct flowi *fl, | 		      struct msghdr *msg, struct flowi6 *fl6, | ||||||
| 		      struct ipv6_txoptions *opt, | 		      struct ipv6_txoptions *opt, | ||||||
| 		      int *hlimit, int *tclass, int *dontfrag) | 		      int *hlimit, int *tclass, int *dontfrag) | ||||||
| { | { | ||||||
|  | @ -629,16 +629,17 @@ int datagram_send_ctl(struct net *net, | ||||||
| 			src_info = (struct in6_pktinfo *)CMSG_DATA(cmsg); | 			src_info = (struct in6_pktinfo *)CMSG_DATA(cmsg); | ||||||
| 
 | 
 | ||||||
| 			if (src_info->ipi6_ifindex) { | 			if (src_info->ipi6_ifindex) { | ||||||
| 				if (fl->flowi_oif && src_info->ipi6_ifindex != fl->flowi_oif) | 				if (fl6->flowi6_oif && | ||||||
|  | 				    src_info->ipi6_ifindex != fl6->flowi6_oif) | ||||||
| 					return -EINVAL; | 					return -EINVAL; | ||||||
| 				fl->flowi_oif = src_info->ipi6_ifindex; | 				fl6->flowi6_oif = src_info->ipi6_ifindex; | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			addr_type = __ipv6_addr_type(&src_info->ipi6_addr); | 			addr_type = __ipv6_addr_type(&src_info->ipi6_addr); | ||||||
| 
 | 
 | ||||||
| 			rcu_read_lock(); | 			rcu_read_lock(); | ||||||
| 			if (fl->flowi_oif) { | 			if (fl6->flowi6_oif) { | ||||||
| 				dev = dev_get_by_index_rcu(net, fl->flowi_oif); | 				dev = dev_get_by_index_rcu(net, fl6->flowi6_oif); | ||||||
| 				if (!dev) { | 				if (!dev) { | ||||||
| 					rcu_read_unlock(); | 					rcu_read_unlock(); | ||||||
| 					return -ENODEV; | 					return -ENODEV; | ||||||
|  | @ -654,7 +655,7 @@ int datagram_send_ctl(struct net *net, | ||||||
| 						   strict ? dev : NULL, 0)) | 						   strict ? dev : NULL, 0)) | ||||||
| 					err = -EINVAL; | 					err = -EINVAL; | ||||||
| 				else | 				else | ||||||
| 					ipv6_addr_copy(&fl->fl6_src, &src_info->ipi6_addr); | 					ipv6_addr_copy(&fl6->saddr, &src_info->ipi6_addr); | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			rcu_read_unlock(); | 			rcu_read_unlock(); | ||||||
|  | @ -671,13 +672,13 @@ int datagram_send_ctl(struct net *net, | ||||||
| 				goto exit_f; | 				goto exit_f; | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			if (fl->fl6_flowlabel&IPV6_FLOWINFO_MASK) { | 			if (fl6->flowlabel&IPV6_FLOWINFO_MASK) { | ||||||
| 				if ((fl->fl6_flowlabel^*(__be32 *)CMSG_DATA(cmsg))&~IPV6_FLOWINFO_MASK) { | 				if ((fl6->flowlabel^*(__be32 *)CMSG_DATA(cmsg))&~IPV6_FLOWINFO_MASK) { | ||||||
| 					err = -EINVAL; | 					err = -EINVAL; | ||||||
| 					goto exit_f; | 					goto exit_f; | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 			fl->fl6_flowlabel = IPV6_FLOWINFO_MASK & *(__be32 *)CMSG_DATA(cmsg); | 			fl6->flowlabel = IPV6_FLOWINFO_MASK & *(__be32 *)CMSG_DATA(cmsg); | ||||||
| 			break; | 			break; | ||||||
| 
 | 
 | ||||||
| 		case IPV6_2292HOPOPTS: | 		case IPV6_2292HOPOPTS: | ||||||
|  |  | ||||||
|  | @ -876,22 +876,22 @@ struct ipv6_txoptions *ipv6_fixup_options(struct ipv6_txoptions *opt_space, | ||||||
|  * fl6_update_dst - update flowi destination address with info given |  * fl6_update_dst - update flowi destination address with info given | ||||||
|  *                  by srcrt option, if any. |  *                  by srcrt option, if any. | ||||||
|  * |  * | ||||||
|  * @fl: flowi for which fl6_dst is to be updated |  * @fl6: flowi6 for which daddr is to be updated | ||||||
|  * @opt: struct ipv6_txoptions in which to look for srcrt opt |  * @opt: struct ipv6_txoptions in which to look for srcrt opt | ||||||
|  * @orig: copy of original fl6_dst address if modified |  * @orig: copy of original daddr address if modified | ||||||
|  * |  * | ||||||
|  * Returns NULL if no txoptions or no srcrt, otherwise returns orig |  * Returns NULL if no txoptions or no srcrt, otherwise returns orig | ||||||
|  * and initial value of fl->fl6_dst set in orig |  * and initial value of fl6->daddr set in orig | ||||||
|  */ |  */ | ||||||
| struct in6_addr *fl6_update_dst(struct flowi *fl, | struct in6_addr *fl6_update_dst(struct flowi6 *fl6, | ||||||
| 				const struct ipv6_txoptions *opt, | 				const struct ipv6_txoptions *opt, | ||||||
| 				struct in6_addr *orig) | 				struct in6_addr *orig) | ||||||
| { | { | ||||||
| 	if (!opt || !opt->srcrt) | 	if (!opt || !opt->srcrt) | ||||||
| 		return NULL; | 		return NULL; | ||||||
| 
 | 
 | ||||||
| 	ipv6_addr_copy(orig, &fl->fl6_dst); | 	ipv6_addr_copy(orig, &fl6->daddr); | ||||||
| 	ipv6_addr_copy(&fl->fl6_dst, ((struct rt0_hdr *)opt->srcrt)->addr); | 	ipv6_addr_copy(&fl6->daddr, ((struct rt0_hdr *)opt->srcrt)->addr); | ||||||
| 	return orig; | 	return orig; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -29,7 +29,7 @@ struct fib6_rule | ||||||
| 	u8			tclass; | 	u8			tclass; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| struct dst_entry *fib6_rule_lookup(struct net *net, struct flowi *fl, | struct dst_entry *fib6_rule_lookup(struct net *net, struct flowi6 *fl6, | ||||||
| 				   int flags, pol_lookup_t lookup) | 				   int flags, pol_lookup_t lookup) | ||||||
| { | { | ||||||
| 	struct fib_lookup_arg arg = { | 	struct fib_lookup_arg arg = { | ||||||
|  | @ -37,7 +37,8 @@ struct dst_entry *fib6_rule_lookup(struct net *net, struct flowi *fl, | ||||||
| 		.flags = FIB_LOOKUP_NOREF, | 		.flags = FIB_LOOKUP_NOREF, | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| 	fib_rules_lookup(net->ipv6.fib6_rules_ops, fl, flags, &arg); | 	fib_rules_lookup(net->ipv6.fib6_rules_ops, | ||||||
|  | 			 flowi6_to_flowi(fl6), flags, &arg); | ||||||
| 
 | 
 | ||||||
| 	if (arg.result) | 	if (arg.result) | ||||||
| 		return arg.result; | 		return arg.result; | ||||||
|  | @ -49,6 +50,7 @@ struct dst_entry *fib6_rule_lookup(struct net *net, struct flowi *fl, | ||||||
| static int fib6_rule_action(struct fib_rule *rule, struct flowi *flp, | static int fib6_rule_action(struct fib_rule *rule, struct flowi *flp, | ||||||
| 			    int flags, struct fib_lookup_arg *arg) | 			    int flags, struct fib_lookup_arg *arg) | ||||||
| { | { | ||||||
|  | 	struct flowi6 *flp6 = &flp->u.ip6; | ||||||
| 	struct rt6_info *rt = NULL; | 	struct rt6_info *rt = NULL; | ||||||
| 	struct fib6_table *table; | 	struct fib6_table *table; | ||||||
| 	struct net *net = rule->fr_net; | 	struct net *net = rule->fr_net; | ||||||
|  | @ -71,7 +73,7 @@ static int fib6_rule_action(struct fib_rule *rule, struct flowi *flp, | ||||||
| 
 | 
 | ||||||
| 	table = fib6_get_table(net, rule->table); | 	table = fib6_get_table(net, rule->table); | ||||||
| 	if (table) | 	if (table) | ||||||
| 		rt = lookup(net, table, flp, flags); | 		rt = lookup(net, table, flp6, flags); | ||||||
| 
 | 
 | ||||||
| 	if (rt != net->ipv6.ip6_null_entry) { | 	if (rt != net->ipv6.ip6_null_entry) { | ||||||
| 		struct fib6_rule *r = (struct fib6_rule *)rule; | 		struct fib6_rule *r = (struct fib6_rule *)rule; | ||||||
|  | @ -86,14 +88,14 @@ static int fib6_rule_action(struct fib_rule *rule, struct flowi *flp, | ||||||
| 
 | 
 | ||||||
| 			if (ipv6_dev_get_saddr(net, | 			if (ipv6_dev_get_saddr(net, | ||||||
| 					       ip6_dst_idev(&rt->dst)->dev, | 					       ip6_dst_idev(&rt->dst)->dev, | ||||||
| 					       &flp->fl6_dst, | 					       &flp6->daddr, | ||||||
| 					       rt6_flags2srcprefs(flags), | 					       rt6_flags2srcprefs(flags), | ||||||
| 					       &saddr)) | 					       &saddr)) | ||||||
| 				goto again; | 				goto again; | ||||||
| 			if (!ipv6_prefix_equal(&saddr, &r->src.addr, | 			if (!ipv6_prefix_equal(&saddr, &r->src.addr, | ||||||
| 					       r->src.plen)) | 					       r->src.plen)) | ||||||
| 				goto again; | 				goto again; | ||||||
| 			ipv6_addr_copy(&flp->fl6_src, &saddr); | 			ipv6_addr_copy(&flp6->saddr, &saddr); | ||||||
| 		} | 		} | ||||||
| 		goto out; | 		goto out; | ||||||
| 	} | 	} | ||||||
|  | @ -113,9 +115,10 @@ out: | ||||||
| static int fib6_rule_match(struct fib_rule *rule, struct flowi *fl, int flags) | static int fib6_rule_match(struct fib_rule *rule, struct flowi *fl, int flags) | ||||||
| { | { | ||||||
| 	struct fib6_rule *r = (struct fib6_rule *) rule; | 	struct fib6_rule *r = (struct fib6_rule *) rule; | ||||||
|  | 	struct flowi6 *fl6 = &fl->u.ip6; | ||||||
| 
 | 
 | ||||||
| 	if (r->dst.plen && | 	if (r->dst.plen && | ||||||
| 	    !ipv6_prefix_equal(&fl->fl6_dst, &r->dst.addr, r->dst.plen)) | 	    !ipv6_prefix_equal(&fl6->daddr, &r->dst.addr, r->dst.plen)) | ||||||
| 		return 0; | 		return 0; | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
|  | @ -125,14 +128,14 @@ static int fib6_rule_match(struct fib_rule *rule, struct flowi *fl, int flags) | ||||||
| 	 */ | 	 */ | ||||||
| 	if (r->src.plen) { | 	if (r->src.plen) { | ||||||
| 		if (flags & RT6_LOOKUP_F_HAS_SADDR) { | 		if (flags & RT6_LOOKUP_F_HAS_SADDR) { | ||||||
| 			if (!ipv6_prefix_equal(&fl->fl6_src, &r->src.addr, | 			if (!ipv6_prefix_equal(&fl6->saddr, &r->src.addr, | ||||||
| 					       r->src.plen)) | 					       r->src.plen)) | ||||||
| 				return 0; | 				return 0; | ||||||
| 		} else if (!(r->common.flags & FIB_RULE_FIND_SADDR)) | 		} else if (!(r->common.flags & FIB_RULE_FIND_SADDR)) | ||||||
| 			return 0; | 			return 0; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (r->tclass && r->tclass != ((ntohl(fl->fl6_flowlabel) >> 20) & 0xff)) | 	if (r->tclass && r->tclass != ((ntohl(fl6->flowlabel) >> 20) & 0xff)) | ||||||
| 		return 0; | 		return 0; | ||||||
| 
 | 
 | ||||||
| 	return 1; | 	return 1; | ||||||
|  |  | ||||||
							
								
								
									
										110
									
								
								net/ipv6/icmp.c
									
										
									
									
									
								
							
							
						
						
									
										110
									
								
								net/ipv6/icmp.c
									
										
									
									
									
								
							|  | @ -158,7 +158,7 @@ static int is_ineligible(struct sk_buff *skb) | ||||||
|  * Check the ICMP output rate limit |  * Check the ICMP output rate limit | ||||||
|  */ |  */ | ||||||
| static inline bool icmpv6_xrlim_allow(struct sock *sk, u8 type, | static inline bool icmpv6_xrlim_allow(struct sock *sk, u8 type, | ||||||
| 				      struct flowi *fl) | 				      struct flowi6 *fl6) | ||||||
| { | { | ||||||
| 	struct dst_entry *dst; | 	struct dst_entry *dst; | ||||||
| 	struct net *net = sock_net(sk); | 	struct net *net = sock_net(sk); | ||||||
|  | @ -177,7 +177,7 @@ static inline bool icmpv6_xrlim_allow(struct sock *sk, u8 type, | ||||||
| 	 * XXX: perhaps the expire for routing entries cloned by | 	 * XXX: perhaps the expire for routing entries cloned by | ||||||
| 	 * this lookup should be more aggressive (not longer than timeout). | 	 * this lookup should be more aggressive (not longer than timeout). | ||||||
| 	 */ | 	 */ | ||||||
| 	dst = ip6_route_output(net, sk, fl); | 	dst = ip6_route_output(net, sk, fl6); | ||||||
| 	if (dst->error) { | 	if (dst->error) { | ||||||
| 		IP6_INC_STATS(net, ip6_dst_idev(dst), | 		IP6_INC_STATS(net, ip6_dst_idev(dst), | ||||||
| 			      IPSTATS_MIB_OUTNOROUTES); | 			      IPSTATS_MIB_OUTNOROUTES); | ||||||
|  | @ -217,7 +217,7 @@ static __inline__ int opt_unrec(struct sk_buff *skb, __u32 offset) | ||||||
| 	return (*op & 0xC0) == 0x80; | 	return (*op & 0xC0) == 0x80; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int icmpv6_push_pending_frames(struct sock *sk, struct flowi *fl, struct icmp6hdr *thdr, int len) | static int icmpv6_push_pending_frames(struct sock *sk, struct flowi6 *fl6, struct icmp6hdr *thdr, int len) | ||||||
| { | { | ||||||
| 	struct sk_buff *skb; | 	struct sk_buff *skb; | ||||||
| 	struct icmp6hdr *icmp6h; | 	struct icmp6hdr *icmp6h; | ||||||
|  | @ -233,9 +233,9 @@ static int icmpv6_push_pending_frames(struct sock *sk, struct flowi *fl, struct | ||||||
| 	if (skb_queue_len(&sk->sk_write_queue) == 1) { | 	if (skb_queue_len(&sk->sk_write_queue) == 1) { | ||||||
| 		skb->csum = csum_partial(icmp6h, | 		skb->csum = csum_partial(icmp6h, | ||||||
| 					sizeof(struct icmp6hdr), skb->csum); | 					sizeof(struct icmp6hdr), skb->csum); | ||||||
| 		icmp6h->icmp6_cksum = csum_ipv6_magic(&fl->fl6_src, | 		icmp6h->icmp6_cksum = csum_ipv6_magic(&fl6->saddr, | ||||||
| 						      &fl->fl6_dst, | 						      &fl6->daddr, | ||||||
| 						      len, fl->flowi_proto, | 						      len, fl6->flowi6_proto, | ||||||
| 						      skb->csum); | 						      skb->csum); | ||||||
| 	} else { | 	} else { | ||||||
| 		__wsum tmp_csum = 0; | 		__wsum tmp_csum = 0; | ||||||
|  | @ -246,9 +246,9 @@ static int icmpv6_push_pending_frames(struct sock *sk, struct flowi *fl, struct | ||||||
| 
 | 
 | ||||||
| 		tmp_csum = csum_partial(icmp6h, | 		tmp_csum = csum_partial(icmp6h, | ||||||
| 					sizeof(struct icmp6hdr), tmp_csum); | 					sizeof(struct icmp6hdr), tmp_csum); | ||||||
| 		icmp6h->icmp6_cksum = csum_ipv6_magic(&fl->fl6_src, | 		icmp6h->icmp6_cksum = csum_ipv6_magic(&fl6->saddr, | ||||||
| 						      &fl->fl6_dst, | 						      &fl6->daddr, | ||||||
| 						      len, fl->flowi_proto, | 						      len, fl6->flowi6_proto, | ||||||
| 						      tmp_csum); | 						      tmp_csum); | ||||||
| 	} | 	} | ||||||
| 	ip6_push_pending_frames(sk); | 	ip6_push_pending_frames(sk); | ||||||
|  | @ -301,13 +301,13 @@ static inline void mip6_addr_swap(struct sk_buff *skb) {} | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
| static struct dst_entry *icmpv6_route_lookup(struct net *net, struct sk_buff *skb, | static struct dst_entry *icmpv6_route_lookup(struct net *net, struct sk_buff *skb, | ||||||
| 					     struct sock *sk, struct flowi *fl) | 					     struct sock *sk, struct flowi6 *fl6) | ||||||
| { | { | ||||||
| 	struct dst_entry *dst, *dst2; | 	struct dst_entry *dst, *dst2; | ||||||
| 	struct flowi fl2; | 	struct flowi6 fl2; | ||||||
| 	int err; | 	int err; | ||||||
| 
 | 
 | ||||||
| 	err = ip6_dst_lookup(sk, &dst, fl); | 	err = ip6_dst_lookup(sk, &dst, fl6); | ||||||
| 	if (err) | 	if (err) | ||||||
| 		return ERR_PTR(err); | 		return ERR_PTR(err); | ||||||
| 
 | 
 | ||||||
|  | @ -324,7 +324,7 @@ static struct dst_entry *icmpv6_route_lookup(struct net *net, struct sk_buff *sk | ||||||
| 	/* No need to clone since we're just using its address. */ | 	/* No need to clone since we're just using its address. */ | ||||||
| 	dst2 = dst; | 	dst2 = dst; | ||||||
| 
 | 
 | ||||||
| 	dst = xfrm_lookup(net, dst, fl, sk, 0); | 	dst = xfrm_lookup(net, dst, flowi6_to_flowi(fl6), sk, 0); | ||||||
| 	if (!IS_ERR(dst)) { | 	if (!IS_ERR(dst)) { | ||||||
| 		if (dst != dst2) | 		if (dst != dst2) | ||||||
| 			return dst; | 			return dst; | ||||||
|  | @ -335,7 +335,7 @@ static struct dst_entry *icmpv6_route_lookup(struct net *net, struct sk_buff *sk | ||||||
| 			return dst; | 			return dst; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	err = xfrm_decode_session_reverse(skb, &fl2, AF_INET6); | 	err = xfrm_decode_session_reverse(skb, flowi6_to_flowi(&fl2), AF_INET6); | ||||||
| 	if (err) | 	if (err) | ||||||
| 		goto relookup_failed; | 		goto relookup_failed; | ||||||
| 
 | 
 | ||||||
|  | @ -343,7 +343,7 @@ static struct dst_entry *icmpv6_route_lookup(struct net *net, struct sk_buff *sk | ||||||
| 	if (err) | 	if (err) | ||||||
| 		goto relookup_failed; | 		goto relookup_failed; | ||||||
| 
 | 
 | ||||||
| 	dst2 = xfrm_lookup(net, dst2, &fl2, sk, XFRM_LOOKUP_ICMP); | 	dst2 = xfrm_lookup(net, dst2, flowi6_to_flowi(&fl2), sk, XFRM_LOOKUP_ICMP); | ||||||
| 	if (!IS_ERR(dst2)) { | 	if (!IS_ERR(dst2)) { | ||||||
| 		dst_release(dst); | 		dst_release(dst); | ||||||
| 		dst = dst2; | 		dst = dst2; | ||||||
|  | @ -375,7 +375,7 @@ void icmpv6_send(struct sk_buff *skb, u8 type, u8 code, __u32 info) | ||||||
| 	struct in6_addr *saddr = NULL; | 	struct in6_addr *saddr = NULL; | ||||||
| 	struct dst_entry *dst; | 	struct dst_entry *dst; | ||||||
| 	struct icmp6hdr tmp_hdr; | 	struct icmp6hdr tmp_hdr; | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 	struct icmpv6_msg msg; | 	struct icmpv6_msg msg; | ||||||
| 	int iif = 0; | 	int iif = 0; | ||||||
| 	int addr_type = 0; | 	int addr_type = 0; | ||||||
|  | @ -442,22 +442,22 @@ void icmpv6_send(struct sk_buff *skb, u8 type, u8 code, __u32 info) | ||||||
| 
 | 
 | ||||||
| 	mip6_addr_swap(skb); | 	mip6_addr_swap(skb); | ||||||
| 
 | 
 | ||||||
| 	memset(&fl, 0, sizeof(fl)); | 	memset(&fl6, 0, sizeof(fl6)); | ||||||
| 	fl.flowi_proto = IPPROTO_ICMPV6; | 	fl6.flowi6_proto = IPPROTO_ICMPV6; | ||||||
| 	ipv6_addr_copy(&fl.fl6_dst, &hdr->saddr); | 	ipv6_addr_copy(&fl6.daddr, &hdr->saddr); | ||||||
| 	if (saddr) | 	if (saddr) | ||||||
| 		ipv6_addr_copy(&fl.fl6_src, saddr); | 		ipv6_addr_copy(&fl6.saddr, saddr); | ||||||
| 	fl.flowi_oif = iif; | 	fl6.flowi6_oif = iif; | ||||||
| 	fl.fl6_icmp_type = type; | 	fl6.uli.icmpt.type = type; | ||||||
| 	fl.fl6_icmp_code = code; | 	fl6.uli.icmpt.code = code; | ||||||
| 	security_skb_classify_flow(skb, &fl); | 	security_skb_classify_flow(skb, flowi6_to_flowi(&fl6)); | ||||||
| 
 | 
 | ||||||
| 	sk = icmpv6_xmit_lock(net); | 	sk = icmpv6_xmit_lock(net); | ||||||
| 	if (sk == NULL) | 	if (sk == NULL) | ||||||
| 		return; | 		return; | ||||||
| 	np = inet6_sk(sk); | 	np = inet6_sk(sk); | ||||||
| 
 | 
 | ||||||
| 	if (!icmpv6_xrlim_allow(sk, type, &fl)) | 	if (!icmpv6_xrlim_allow(sk, type, &fl6)) | ||||||
| 		goto out; | 		goto out; | ||||||
| 
 | 
 | ||||||
| 	tmp_hdr.icmp6_type = type; | 	tmp_hdr.icmp6_type = type; | ||||||
|  | @ -465,14 +465,14 @@ void icmpv6_send(struct sk_buff *skb, u8 type, u8 code, __u32 info) | ||||||
| 	tmp_hdr.icmp6_cksum = 0; | 	tmp_hdr.icmp6_cksum = 0; | ||||||
| 	tmp_hdr.icmp6_pointer = htonl(info); | 	tmp_hdr.icmp6_pointer = htonl(info); | ||||||
| 
 | 
 | ||||||
| 	if (!fl.flowi_oif && ipv6_addr_is_multicast(&fl.fl6_dst)) | 	if (!fl6.flowi6_oif && ipv6_addr_is_multicast(&fl6.daddr)) | ||||||
| 		fl.flowi_oif = np->mcast_oif; | 		fl6.flowi6_oif = np->mcast_oif; | ||||||
| 
 | 
 | ||||||
| 	dst = icmpv6_route_lookup(net, skb, sk, &fl); | 	dst = icmpv6_route_lookup(net, skb, sk, &fl6); | ||||||
| 	if (IS_ERR(dst)) | 	if (IS_ERR(dst)) | ||||||
| 		goto out; | 		goto out; | ||||||
| 
 | 
 | ||||||
| 	if (ipv6_addr_is_multicast(&fl.fl6_dst)) | 	if (ipv6_addr_is_multicast(&fl6.daddr)) | ||||||
| 		hlimit = np->mcast_hops; | 		hlimit = np->mcast_hops; | ||||||
| 	else | 	else | ||||||
| 		hlimit = np->hop_limit; | 		hlimit = np->hop_limit; | ||||||
|  | @ -495,14 +495,14 @@ void icmpv6_send(struct sk_buff *skb, u8 type, u8 code, __u32 info) | ||||||
| 	err = ip6_append_data(sk, icmpv6_getfrag, &msg, | 	err = ip6_append_data(sk, icmpv6_getfrag, &msg, | ||||||
| 			      len + sizeof(struct icmp6hdr), | 			      len + sizeof(struct icmp6hdr), | ||||||
| 			      sizeof(struct icmp6hdr), hlimit, | 			      sizeof(struct icmp6hdr), hlimit, | ||||||
| 			      np->tclass, NULL, &fl, (struct rt6_info*)dst, | 			      np->tclass, NULL, &fl6, (struct rt6_info*)dst, | ||||||
| 			      MSG_DONTWAIT, np->dontfrag); | 			      MSG_DONTWAIT, np->dontfrag); | ||||||
| 	if (err) { | 	if (err) { | ||||||
| 		ICMP6_INC_STATS_BH(net, idev, ICMP6_MIB_OUTERRORS); | 		ICMP6_INC_STATS_BH(net, idev, ICMP6_MIB_OUTERRORS); | ||||||
| 		ip6_flush_pending_frames(sk); | 		ip6_flush_pending_frames(sk); | ||||||
| 		goto out_put; | 		goto out_put; | ||||||
| 	} | 	} | ||||||
| 	err = icmpv6_push_pending_frames(sk, &fl, &tmp_hdr, len + sizeof(struct icmp6hdr)); | 	err = icmpv6_push_pending_frames(sk, &fl6, &tmp_hdr, len + sizeof(struct icmp6hdr)); | ||||||
| 
 | 
 | ||||||
| out_put: | out_put: | ||||||
| 	if (likely(idev != NULL)) | 	if (likely(idev != NULL)) | ||||||
|  | @ -524,7 +524,7 @@ static void icmpv6_echo_reply(struct sk_buff *skb) | ||||||
| 	struct in6_addr *saddr = NULL; | 	struct in6_addr *saddr = NULL; | ||||||
| 	struct icmp6hdr *icmph = icmp6_hdr(skb); | 	struct icmp6hdr *icmph = icmp6_hdr(skb); | ||||||
| 	struct icmp6hdr tmp_hdr; | 	struct icmp6hdr tmp_hdr; | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 	struct icmpv6_msg msg; | 	struct icmpv6_msg msg; | ||||||
| 	struct dst_entry *dst; | 	struct dst_entry *dst; | ||||||
| 	int err = 0; | 	int err = 0; | ||||||
|  | @ -538,31 +538,31 @@ static void icmpv6_echo_reply(struct sk_buff *skb) | ||||||
| 	memcpy(&tmp_hdr, icmph, sizeof(tmp_hdr)); | 	memcpy(&tmp_hdr, icmph, sizeof(tmp_hdr)); | ||||||
| 	tmp_hdr.icmp6_type = ICMPV6_ECHO_REPLY; | 	tmp_hdr.icmp6_type = ICMPV6_ECHO_REPLY; | ||||||
| 
 | 
 | ||||||
| 	memset(&fl, 0, sizeof(fl)); | 	memset(&fl6, 0, sizeof(fl6)); | ||||||
| 	fl.flowi_proto = IPPROTO_ICMPV6; | 	fl6.flowi6_proto = IPPROTO_ICMPV6; | ||||||
| 	ipv6_addr_copy(&fl.fl6_dst, &ipv6_hdr(skb)->saddr); | 	ipv6_addr_copy(&fl6.daddr, &ipv6_hdr(skb)->saddr); | ||||||
| 	if (saddr) | 	if (saddr) | ||||||
| 		ipv6_addr_copy(&fl.fl6_src, saddr); | 		ipv6_addr_copy(&fl6.saddr, saddr); | ||||||
| 	fl.flowi_oif = skb->dev->ifindex; | 	fl6.flowi6_oif = skb->dev->ifindex; | ||||||
| 	fl.fl6_icmp_type = ICMPV6_ECHO_REPLY; | 	fl6.uli.icmpt.type = ICMPV6_ECHO_REPLY; | ||||||
| 	security_skb_classify_flow(skb, &fl); | 	security_skb_classify_flow(skb, flowi6_to_flowi(&fl6)); | ||||||
| 
 | 
 | ||||||
| 	sk = icmpv6_xmit_lock(net); | 	sk = icmpv6_xmit_lock(net); | ||||||
| 	if (sk == NULL) | 	if (sk == NULL) | ||||||
| 		return; | 		return; | ||||||
| 	np = inet6_sk(sk); | 	np = inet6_sk(sk); | ||||||
| 
 | 
 | ||||||
| 	if (!fl.flowi_oif && ipv6_addr_is_multicast(&fl.fl6_dst)) | 	if (!fl6.flowi6_oif && ipv6_addr_is_multicast(&fl6.daddr)) | ||||||
| 		fl.flowi_oif = np->mcast_oif; | 		fl6.flowi6_oif = np->mcast_oif; | ||||||
| 
 | 
 | ||||||
| 	err = ip6_dst_lookup(sk, &dst, &fl); | 	err = ip6_dst_lookup(sk, &dst, &fl6); | ||||||
| 	if (err) | 	if (err) | ||||||
| 		goto out; | 		goto out; | ||||||
| 	dst = xfrm_lookup(net, dst, &fl, sk, 0); | 	dst = xfrm_lookup(net, dst, flowi6_to_flowi(&fl6), sk, 0); | ||||||
| 	if (IS_ERR(dst)) | 	if (IS_ERR(dst)) | ||||||
| 		goto out; | 		goto out; | ||||||
| 
 | 
 | ||||||
| 	if (ipv6_addr_is_multicast(&fl.fl6_dst)) | 	if (ipv6_addr_is_multicast(&fl6.daddr)) | ||||||
| 		hlimit = np->mcast_hops; | 		hlimit = np->mcast_hops; | ||||||
| 	else | 	else | ||||||
| 		hlimit = np->hop_limit; | 		hlimit = np->hop_limit; | ||||||
|  | @ -576,7 +576,7 @@ static void icmpv6_echo_reply(struct sk_buff *skb) | ||||||
| 	msg.type = ICMPV6_ECHO_REPLY; | 	msg.type = ICMPV6_ECHO_REPLY; | ||||||
| 
 | 
 | ||||||
| 	err = ip6_append_data(sk, icmpv6_getfrag, &msg, skb->len + sizeof(struct icmp6hdr), | 	err = ip6_append_data(sk, icmpv6_getfrag, &msg, skb->len + sizeof(struct icmp6hdr), | ||||||
| 				sizeof(struct icmp6hdr), hlimit, np->tclass, NULL, &fl, | 				sizeof(struct icmp6hdr), hlimit, np->tclass, NULL, &fl6, | ||||||
| 				(struct rt6_info*)dst, MSG_DONTWAIT, | 				(struct rt6_info*)dst, MSG_DONTWAIT, | ||||||
| 				np->dontfrag); | 				np->dontfrag); | ||||||
| 
 | 
 | ||||||
|  | @ -585,7 +585,7 @@ static void icmpv6_echo_reply(struct sk_buff *skb) | ||||||
| 		ip6_flush_pending_frames(sk); | 		ip6_flush_pending_frames(sk); | ||||||
| 		goto out_put; | 		goto out_put; | ||||||
| 	} | 	} | ||||||
| 	err = icmpv6_push_pending_frames(sk, &fl, &tmp_hdr, skb->len + sizeof(struct icmp6hdr)); | 	err = icmpv6_push_pending_frames(sk, &fl6, &tmp_hdr, skb->len + sizeof(struct icmp6hdr)); | ||||||
| 
 | 
 | ||||||
| out_put: | out_put: | ||||||
| 	if (likely(idev != NULL)) | 	if (likely(idev != NULL)) | ||||||
|  | @ -784,20 +784,20 @@ drop_no_count: | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void icmpv6_flow_init(struct sock *sk, struct flowi *fl, | void icmpv6_flow_init(struct sock *sk, struct flowi6 *fl6, | ||||||
| 		      u8 type, | 		      u8 type, | ||||||
| 		      const struct in6_addr *saddr, | 		      const struct in6_addr *saddr, | ||||||
| 		      const struct in6_addr *daddr, | 		      const struct in6_addr *daddr, | ||||||
| 		      int oif) | 		      int oif) | ||||||
| { | { | ||||||
| 	memset(fl, 0, sizeof(*fl)); | 	memset(fl6, 0, sizeof(*fl6)); | ||||||
| 	ipv6_addr_copy(&fl->fl6_src, saddr); | 	ipv6_addr_copy(&fl6->saddr, saddr); | ||||||
| 	ipv6_addr_copy(&fl->fl6_dst, daddr); | 	ipv6_addr_copy(&fl6->daddr, daddr); | ||||||
| 	fl->flowi_proto	 	= IPPROTO_ICMPV6; | 	fl6->flowi6_proto 	= IPPROTO_ICMPV6; | ||||||
| 	fl->fl6_icmp_type	= type; | 	fl6->uli.icmpt.type	= type; | ||||||
| 	fl->fl6_icmp_code	= 0; | 	fl6->uli.icmpt.code	= 0; | ||||||
| 	fl->flowi_oif		= oif; | 	fl6->flowi6_oif		= oif; | ||||||
| 	security_sk_classify_flow(sk, fl); | 	security_sk_classify_flow(sk, flowi6_to_flowi(fl6)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
|  |  | ||||||
|  | @ -61,20 +61,20 @@ struct dst_entry *inet6_csk_route_req(struct sock *sk, | ||||||
| 	struct ipv6_pinfo *np = inet6_sk(sk); | 	struct ipv6_pinfo *np = inet6_sk(sk); | ||||||
| 	struct in6_addr *final_p, final; | 	struct in6_addr *final_p, final; | ||||||
| 	struct dst_entry *dst; | 	struct dst_entry *dst; | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 
 | 
 | ||||||
| 	memset(&fl, 0, sizeof(fl)); | 	memset(&fl6, 0, sizeof(fl6)); | ||||||
| 	fl.flowi_proto = IPPROTO_TCP; | 	fl6.flowi6_proto = IPPROTO_TCP; | ||||||
| 	ipv6_addr_copy(&fl.fl6_dst, &treq->rmt_addr); | 	ipv6_addr_copy(&fl6.daddr, &treq->rmt_addr); | ||||||
| 	final_p = fl6_update_dst(&fl, np->opt, &final); | 	final_p = fl6_update_dst(&fl6, np->opt, &final); | ||||||
| 	ipv6_addr_copy(&fl.fl6_src, &treq->loc_addr); | 	ipv6_addr_copy(&fl6.saddr, &treq->loc_addr); | ||||||
| 	fl.flowi_oif = sk->sk_bound_dev_if; | 	fl6.flowi6_oif = sk->sk_bound_dev_if; | ||||||
| 	fl.flowi_mark = sk->sk_mark; | 	fl6.flowi6_mark = sk->sk_mark; | ||||||
| 	fl.fl6_dport = inet_rsk(req)->rmt_port; | 	fl6.uli.ports.dport = inet_rsk(req)->rmt_port; | ||||||
| 	fl.fl6_sport = inet_rsk(req)->loc_port; | 	fl6.uli.ports.sport = inet_rsk(req)->loc_port; | ||||||
| 	security_req_classify_flow(req, &fl); | 	security_req_classify_flow(req, flowi6_to_flowi(&fl6)); | ||||||
| 
 | 
 | ||||||
| 	dst = ip6_dst_lookup_flow(sk, &fl, final_p, false); | 	dst = ip6_dst_lookup_flow(sk, &fl6, final_p, false); | ||||||
| 	if (IS_ERR(dst)) | 	if (IS_ERR(dst)) | ||||||
| 		return NULL; | 		return NULL; | ||||||
| 
 | 
 | ||||||
|  | @ -208,28 +208,28 @@ int inet6_csk_xmit(struct sk_buff *skb) | ||||||
| 	struct sock *sk = skb->sk; | 	struct sock *sk = skb->sk; | ||||||
| 	struct inet_sock *inet = inet_sk(sk); | 	struct inet_sock *inet = inet_sk(sk); | ||||||
| 	struct ipv6_pinfo *np = inet6_sk(sk); | 	struct ipv6_pinfo *np = inet6_sk(sk); | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 	struct dst_entry *dst; | 	struct dst_entry *dst; | ||||||
| 	struct in6_addr *final_p, final; | 	struct in6_addr *final_p, final; | ||||||
| 
 | 
 | ||||||
| 	memset(&fl, 0, sizeof(fl)); | 	memset(&fl6, 0, sizeof(fl6)); | ||||||
| 	fl.flowi_proto = sk->sk_protocol; | 	fl6.flowi6_proto = sk->sk_protocol; | ||||||
| 	ipv6_addr_copy(&fl.fl6_dst, &np->daddr); | 	ipv6_addr_copy(&fl6.daddr, &np->daddr); | ||||||
| 	ipv6_addr_copy(&fl.fl6_src, &np->saddr); | 	ipv6_addr_copy(&fl6.saddr, &np->saddr); | ||||||
| 	fl.fl6_flowlabel = np->flow_label; | 	fl6.flowlabel = np->flow_label; | ||||||
| 	IP6_ECN_flow_xmit(sk, fl.fl6_flowlabel); | 	IP6_ECN_flow_xmit(sk, fl6.flowlabel); | ||||||
| 	fl.flowi_oif = sk->sk_bound_dev_if; | 	fl6.flowi6_oif = sk->sk_bound_dev_if; | ||||||
| 	fl.flowi_mark = sk->sk_mark; | 	fl6.flowi6_mark = sk->sk_mark; | ||||||
| 	fl.fl6_sport = inet->inet_sport; | 	fl6.uli.ports.sport = inet->inet_sport; | ||||||
| 	fl.fl6_dport = inet->inet_dport; | 	fl6.uli.ports.dport = inet->inet_dport; | ||||||
| 	security_sk_classify_flow(sk, &fl); | 	security_sk_classify_flow(sk, flowi6_to_flowi(&fl6)); | ||||||
| 
 | 
 | ||||||
| 	final_p = fl6_update_dst(&fl, np->opt, &final); | 	final_p = fl6_update_dst(&fl6, np->opt, &final); | ||||||
| 
 | 
 | ||||||
| 	dst = __inet6_csk_dst_check(sk, np->dst_cookie); | 	dst = __inet6_csk_dst_check(sk, np->dst_cookie); | ||||||
| 
 | 
 | ||||||
| 	if (dst == NULL) { | 	if (dst == NULL) { | ||||||
| 		dst = ip6_dst_lookup_flow(sk, &fl, final_p, false); | 		dst = ip6_dst_lookup_flow(sk, &fl6, final_p, false); | ||||||
| 
 | 
 | ||||||
| 		if (IS_ERR(dst)) { | 		if (IS_ERR(dst)) { | ||||||
| 			sk->sk_err_soft = -PTR_ERR(dst); | 			sk->sk_err_soft = -PTR_ERR(dst); | ||||||
|  | @ -244,9 +244,9 @@ int inet6_csk_xmit(struct sk_buff *skb) | ||||||
| 	skb_dst_set(skb, dst_clone(dst)); | 	skb_dst_set(skb, dst_clone(dst)); | ||||||
| 
 | 
 | ||||||
| 	/* Restore final destination back after routing done */ | 	/* Restore final destination back after routing done */ | ||||||
| 	ipv6_addr_copy(&fl.fl6_dst, &np->daddr); | 	ipv6_addr_copy(&fl6.daddr, &np->daddr); | ||||||
| 
 | 
 | ||||||
| 	return ip6_xmit(sk, skb, &fl, np->opt); | 	return ip6_xmit(sk, skb, &fl6, np->opt); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| EXPORT_SYMBOL_GPL(inet6_csk_xmit); | EXPORT_SYMBOL_GPL(inet6_csk_xmit); | ||||||
|  |  | ||||||
|  | @ -260,10 +260,10 @@ struct fib6_table *fib6_get_table(struct net *net, u32 id) | ||||||
| 	  return net->ipv6.fib6_main_tbl; | 	  return net->ipv6.fib6_main_tbl; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| struct dst_entry *fib6_rule_lookup(struct net *net, struct flowi *fl, | struct dst_entry *fib6_rule_lookup(struct net *net, struct flowi6 *fl6, | ||||||
| 				   int flags, pol_lookup_t lookup) | 				   int flags, pol_lookup_t lookup) | ||||||
| { | { | ||||||
| 	return (struct dst_entry *) lookup(net, net->ipv6.fib6_main_tbl, fl, flags); | 	return (struct dst_entry *) lookup(net, net->ipv6.fib6_main_tbl, fl6, flags); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static void __net_init fib6_tables_init(struct net *net) | static void __net_init fib6_tables_init(struct net *net) | ||||||
|  |  | ||||||
|  | @ -342,7 +342,7 @@ fl_create(struct net *net, struct in6_flowlabel_req *freq, char __user *optval, | ||||||
| 
 | 
 | ||||||
| 	if (olen > 0) { | 	if (olen > 0) { | ||||||
| 		struct msghdr msg; | 		struct msghdr msg; | ||||||
| 		struct flowi flowi; | 		struct flowi6 flowi6; | ||||||
| 		int junk; | 		int junk; | ||||||
| 
 | 
 | ||||||
| 		err = -ENOMEM; | 		err = -ENOMEM; | ||||||
|  | @ -358,9 +358,9 @@ fl_create(struct net *net, struct in6_flowlabel_req *freq, char __user *optval, | ||||||
| 
 | 
 | ||||||
| 		msg.msg_controllen = olen; | 		msg.msg_controllen = olen; | ||||||
| 		msg.msg_control = (void*)(fl->opt+1); | 		msg.msg_control = (void*)(fl->opt+1); | ||||||
| 		flowi.flowi_oif = 0; | 		memset(&flowi6, 0, sizeof(flowi6)); | ||||||
| 
 | 
 | ||||||
| 		err = datagram_send_ctl(net, &msg, &flowi, fl->opt, &junk, | 		err = datagram_send_ctl(net, &msg, &flowi6, fl->opt, &junk, | ||||||
| 					&junk, &junk); | 					&junk, &junk); | ||||||
| 		if (err) | 		if (err) | ||||||
| 			goto done; | 			goto done; | ||||||
|  |  | ||||||
|  | @ -174,15 +174,15 @@ int ip6_output(struct sk_buff *skb) | ||||||
|  *	xmit an sk_buff (used by TCP, SCTP and DCCP) |  *	xmit an sk_buff (used by TCP, SCTP and DCCP) | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| int ip6_xmit(struct sock *sk, struct sk_buff *skb, struct flowi *fl, | int ip6_xmit(struct sock *sk, struct sk_buff *skb, struct flowi6 *fl6, | ||||||
| 	     struct ipv6_txoptions *opt) | 	     struct ipv6_txoptions *opt) | ||||||
| { | { | ||||||
| 	struct net *net = sock_net(sk); | 	struct net *net = sock_net(sk); | ||||||
| 	struct ipv6_pinfo *np = inet6_sk(sk); | 	struct ipv6_pinfo *np = inet6_sk(sk); | ||||||
| 	struct in6_addr *first_hop = &fl->fl6_dst; | 	struct in6_addr *first_hop = &fl6->daddr; | ||||||
| 	struct dst_entry *dst = skb_dst(skb); | 	struct dst_entry *dst = skb_dst(skb); | ||||||
| 	struct ipv6hdr *hdr; | 	struct ipv6hdr *hdr; | ||||||
| 	u8  proto = fl->flowi_proto; | 	u8  proto = fl6->flowi6_proto; | ||||||
| 	int seg_len = skb->len; | 	int seg_len = skb->len; | ||||||
| 	int hlimit = -1; | 	int hlimit = -1; | ||||||
| 	int tclass = 0; | 	int tclass = 0; | ||||||
|  | @ -230,13 +230,13 @@ int ip6_xmit(struct sock *sk, struct sk_buff *skb, struct flowi *fl, | ||||||
| 	if (hlimit < 0) | 	if (hlimit < 0) | ||||||
| 		hlimit = ip6_dst_hoplimit(dst); | 		hlimit = ip6_dst_hoplimit(dst); | ||||||
| 
 | 
 | ||||||
| 	*(__be32 *)hdr = htonl(0x60000000 | (tclass << 20)) | fl->fl6_flowlabel; | 	*(__be32 *)hdr = htonl(0x60000000 | (tclass << 20)) | fl6->flowlabel; | ||||||
| 
 | 
 | ||||||
| 	hdr->payload_len = htons(seg_len); | 	hdr->payload_len = htons(seg_len); | ||||||
| 	hdr->nexthdr = proto; | 	hdr->nexthdr = proto; | ||||||
| 	hdr->hop_limit = hlimit; | 	hdr->hop_limit = hlimit; | ||||||
| 
 | 
 | ||||||
| 	ipv6_addr_copy(&hdr->saddr, &fl->fl6_src); | 	ipv6_addr_copy(&hdr->saddr, &fl6->saddr); | ||||||
| 	ipv6_addr_copy(&hdr->daddr, first_hop); | 	ipv6_addr_copy(&hdr->daddr, first_hop); | ||||||
| 
 | 
 | ||||||
| 	skb->priority = sk->sk_priority; | 	skb->priority = sk->sk_priority; | ||||||
|  | @ -879,7 +879,7 @@ static inline int ip6_rt_check(struct rt6key *rt_key, | ||||||
| 
 | 
 | ||||||
| static struct dst_entry *ip6_sk_dst_check(struct sock *sk, | static struct dst_entry *ip6_sk_dst_check(struct sock *sk, | ||||||
| 					  struct dst_entry *dst, | 					  struct dst_entry *dst, | ||||||
| 					  struct flowi *fl) | 					  struct flowi6 *fl6) | ||||||
| { | { | ||||||
| 	struct ipv6_pinfo *np = inet6_sk(sk); | 	struct ipv6_pinfo *np = inet6_sk(sk); | ||||||
| 	struct rt6_info *rt = (struct rt6_info *)dst; | 	struct rt6_info *rt = (struct rt6_info *)dst; | ||||||
|  | @ -904,11 +904,11 @@ static struct dst_entry *ip6_sk_dst_check(struct sock *sk, | ||||||
| 	 *    sockets. | 	 *    sockets. | ||||||
| 	 * 2. oif also should be the same. | 	 * 2. oif also should be the same. | ||||||
| 	 */ | 	 */ | ||||||
| 	if (ip6_rt_check(&rt->rt6i_dst, &fl->fl6_dst, np->daddr_cache) || | 	if (ip6_rt_check(&rt->rt6i_dst, &fl6->daddr, np->daddr_cache) || | ||||||
| #ifdef CONFIG_IPV6_SUBTREES | #ifdef CONFIG_IPV6_SUBTREES | ||||||
| 	    ip6_rt_check(&rt->rt6i_src, &fl->fl6_src, np->saddr_cache) || | 	    ip6_rt_check(&rt->rt6i_src, &fl6->saddr, np->saddr_cache) || | ||||||
| #endif | #endif | ||||||
| 	    (fl->flowi_oif && fl->flowi_oif != dst->dev->ifindex)) { | 	    (fl6->flowi6_oif && fl6->flowi6_oif != dst->dev->ifindex)) { | ||||||
| 		dst_release(dst); | 		dst_release(dst); | ||||||
| 		dst = NULL; | 		dst = NULL; | ||||||
| 	} | 	} | ||||||
|  | @ -918,22 +918,22 @@ out: | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int ip6_dst_lookup_tail(struct sock *sk, | static int ip6_dst_lookup_tail(struct sock *sk, | ||||||
| 			       struct dst_entry **dst, struct flowi *fl) | 			       struct dst_entry **dst, struct flowi6 *fl6) | ||||||
| { | { | ||||||
| 	int err; | 	int err; | ||||||
| 	struct net *net = sock_net(sk); | 	struct net *net = sock_net(sk); | ||||||
| 
 | 
 | ||||||
| 	if (*dst == NULL) | 	if (*dst == NULL) | ||||||
| 		*dst = ip6_route_output(net, sk, fl); | 		*dst = ip6_route_output(net, sk, fl6); | ||||||
| 
 | 
 | ||||||
| 	if ((err = (*dst)->error)) | 	if ((err = (*dst)->error)) | ||||||
| 		goto out_err_release; | 		goto out_err_release; | ||||||
| 
 | 
 | ||||||
| 	if (ipv6_addr_any(&fl->fl6_src)) { | 	if (ipv6_addr_any(&fl6->saddr)) { | ||||||
| 		err = ipv6_dev_get_saddr(net, ip6_dst_idev(*dst)->dev, | 		err = ipv6_dev_get_saddr(net, ip6_dst_idev(*dst)->dev, | ||||||
| 					 &fl->fl6_dst, | 					 &fl6->daddr, | ||||||
| 					 sk ? inet6_sk(sk)->srcprefs : 0, | 					 sk ? inet6_sk(sk)->srcprefs : 0, | ||||||
| 					 &fl->fl6_src); | 					 &fl6->saddr); | ||||||
| 		if (err) | 		if (err) | ||||||
| 			goto out_err_release; | 			goto out_err_release; | ||||||
| 	} | 	} | ||||||
|  | @ -949,10 +949,10 @@ static int ip6_dst_lookup_tail(struct sock *sk, | ||||||
| 	 */ | 	 */ | ||||||
| 	if ((*dst)->neighbour && !((*dst)->neighbour->nud_state & NUD_VALID)) { | 	if ((*dst)->neighbour && !((*dst)->neighbour->nud_state & NUD_VALID)) { | ||||||
| 		struct inet6_ifaddr *ifp; | 		struct inet6_ifaddr *ifp; | ||||||
| 		struct flowi fl_gw; | 		struct flowi6 fl_gw6; | ||||||
| 		int redirect; | 		int redirect; | ||||||
| 
 | 
 | ||||||
| 		ifp = ipv6_get_ifaddr(net, &fl->fl6_src, | 		ifp = ipv6_get_ifaddr(net, &fl6->saddr, | ||||||
| 				      (*dst)->dev, 1); | 				      (*dst)->dev, 1); | ||||||
| 
 | 
 | ||||||
| 		redirect = (ifp && ifp->flags & IFA_F_OPTIMISTIC); | 		redirect = (ifp && ifp->flags & IFA_F_OPTIMISTIC); | ||||||
|  | @ -965,9 +965,9 @@ static int ip6_dst_lookup_tail(struct sock *sk, | ||||||
| 			 * default router instead | 			 * default router instead | ||||||
| 			 */ | 			 */ | ||||||
| 			dst_release(*dst); | 			dst_release(*dst); | ||||||
| 			memcpy(&fl_gw, fl, sizeof(struct flowi)); | 			memcpy(&fl_gw6, fl6, sizeof(struct flowi6)); | ||||||
| 			memset(&fl_gw.fl6_dst, 0, sizeof(struct in6_addr)); | 			memset(&fl_gw6.daddr, 0, sizeof(struct in6_addr)); | ||||||
| 			*dst = ip6_route_output(net, sk, &fl_gw); | 			*dst = ip6_route_output(net, sk, &fl_gw6); | ||||||
| 			if ((err = (*dst)->error)) | 			if ((err = (*dst)->error)) | ||||||
| 				goto out_err_release; | 				goto out_err_release; | ||||||
| 		} | 		} | ||||||
|  | @ -988,23 +988,23 @@ out_err_release: | ||||||
|  *	ip6_dst_lookup - perform route lookup on flow |  *	ip6_dst_lookup - perform route lookup on flow | ||||||
|  *	@sk: socket which provides route info |  *	@sk: socket which provides route info | ||||||
|  *	@dst: pointer to dst_entry * for result |  *	@dst: pointer to dst_entry * for result | ||||||
|  *	@fl: flow to lookup |  *	@fl6: flow to lookup | ||||||
|  * |  * | ||||||
|  *	This function performs a route lookup on the given flow. |  *	This function performs a route lookup on the given flow. | ||||||
|  * |  * | ||||||
|  *	It returns zero on success, or a standard errno code on error. |  *	It returns zero on success, or a standard errno code on error. | ||||||
|  */ |  */ | ||||||
| int ip6_dst_lookup(struct sock *sk, struct dst_entry **dst, struct flowi *fl) | int ip6_dst_lookup(struct sock *sk, struct dst_entry **dst, struct flowi6 *fl6) | ||||||
| { | { | ||||||
| 	*dst = NULL; | 	*dst = NULL; | ||||||
| 	return ip6_dst_lookup_tail(sk, dst, fl); | 	return ip6_dst_lookup_tail(sk, dst, fl6); | ||||||
| } | } | ||||||
| EXPORT_SYMBOL_GPL(ip6_dst_lookup); | EXPORT_SYMBOL_GPL(ip6_dst_lookup); | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  *	ip6_dst_lookup_flow - perform route lookup on flow with ipsec |  *	ip6_dst_lookup_flow - perform route lookup on flow with ipsec | ||||||
|  *	@sk: socket which provides route info |  *	@sk: socket which provides route info | ||||||
|  *	@fl: flow to lookup |  *	@fl6: flow to lookup | ||||||
|  *	@final_dst: final destination address for ipsec lookup |  *	@final_dst: final destination address for ipsec lookup | ||||||
|  *	@can_sleep: we are in a sleepable context |  *	@can_sleep: we are in a sleepable context | ||||||
|  * |  * | ||||||
|  | @ -1013,29 +1013,29 @@ EXPORT_SYMBOL_GPL(ip6_dst_lookup); | ||||||
|  *	It returns a valid dst pointer on success, or a pointer encoded |  *	It returns a valid dst pointer on success, or a pointer encoded | ||||||
|  *	error code. |  *	error code. | ||||||
|  */ |  */ | ||||||
| struct dst_entry *ip6_dst_lookup_flow(struct sock *sk, struct flowi *fl, | struct dst_entry *ip6_dst_lookup_flow(struct sock *sk, struct flowi6 *fl6, | ||||||
| 				      const struct in6_addr *final_dst, | 				      const struct in6_addr *final_dst, | ||||||
| 				      bool can_sleep) | 				      bool can_sleep) | ||||||
| { | { | ||||||
| 	struct dst_entry *dst = NULL; | 	struct dst_entry *dst = NULL; | ||||||
| 	int err; | 	int err; | ||||||
| 
 | 
 | ||||||
| 	err = ip6_dst_lookup_tail(sk, &dst, fl); | 	err = ip6_dst_lookup_tail(sk, &dst, fl6); | ||||||
| 	if (err) | 	if (err) | ||||||
| 		return ERR_PTR(err); | 		return ERR_PTR(err); | ||||||
| 	if (final_dst) | 	if (final_dst) | ||||||
| 		ipv6_addr_copy(&fl->fl6_dst, final_dst); | 		ipv6_addr_copy(&fl6->daddr, final_dst); | ||||||
| 	if (can_sleep) | 	if (can_sleep) | ||||||
| 		fl->flowi_flags |= FLOWI_FLAG_CAN_SLEEP; | 		fl6->flowi6_flags |= FLOWI_FLAG_CAN_SLEEP; | ||||||
| 
 | 
 | ||||||
| 	return xfrm_lookup(sock_net(sk), dst, fl, sk, 0); | 	return xfrm_lookup(sock_net(sk), dst, flowi6_to_flowi(fl6), sk, 0); | ||||||
| } | } | ||||||
| EXPORT_SYMBOL_GPL(ip6_dst_lookup_flow); | EXPORT_SYMBOL_GPL(ip6_dst_lookup_flow); | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  *	ip6_sk_dst_lookup_flow - perform socket cached route lookup on flow |  *	ip6_sk_dst_lookup_flow - perform socket cached route lookup on flow | ||||||
|  *	@sk: socket which provides the dst cache and route info |  *	@sk: socket which provides the dst cache and route info | ||||||
|  *	@fl: flow to lookup |  *	@fl6: flow to lookup | ||||||
|  *	@final_dst: final destination address for ipsec lookup |  *	@final_dst: final destination address for ipsec lookup | ||||||
|  *	@can_sleep: we are in a sleepable context |  *	@can_sleep: we are in a sleepable context | ||||||
|  * |  * | ||||||
|  | @ -1047,24 +1047,24 @@ EXPORT_SYMBOL_GPL(ip6_dst_lookup_flow); | ||||||
|  *	It returns a valid dst pointer on success, or a pointer encoded |  *	It returns a valid dst pointer on success, or a pointer encoded | ||||||
|  *	error code. |  *	error code. | ||||||
|  */ |  */ | ||||||
| struct dst_entry *ip6_sk_dst_lookup_flow(struct sock *sk, struct flowi *fl, | struct dst_entry *ip6_sk_dst_lookup_flow(struct sock *sk, struct flowi6 *fl6, | ||||||
| 					 const struct in6_addr *final_dst, | 					 const struct in6_addr *final_dst, | ||||||
| 					 bool can_sleep) | 					 bool can_sleep) | ||||||
| { | { | ||||||
| 	struct dst_entry *dst = sk_dst_check(sk, inet6_sk(sk)->dst_cookie); | 	struct dst_entry *dst = sk_dst_check(sk, inet6_sk(sk)->dst_cookie); | ||||||
| 	int err; | 	int err; | ||||||
| 
 | 
 | ||||||
| 	dst = ip6_sk_dst_check(sk, dst, fl); | 	dst = ip6_sk_dst_check(sk, dst, fl6); | ||||||
| 
 | 
 | ||||||
| 	err = ip6_dst_lookup_tail(sk, &dst, fl); | 	err = ip6_dst_lookup_tail(sk, &dst, fl6); | ||||||
| 	if (err) | 	if (err) | ||||||
| 		return ERR_PTR(err); | 		return ERR_PTR(err); | ||||||
| 	if (final_dst) | 	if (final_dst) | ||||||
| 		ipv6_addr_copy(&fl->fl6_dst, final_dst); | 		ipv6_addr_copy(&fl6->daddr, final_dst); | ||||||
| 	if (can_sleep) | 	if (can_sleep) | ||||||
| 		fl->flowi_flags |= FLOWI_FLAG_CAN_SLEEP; | 		fl6->flowi6_flags |= FLOWI_FLAG_CAN_SLEEP; | ||||||
| 
 | 
 | ||||||
| 	return xfrm_lookup(sock_net(sk), dst, fl, sk, 0); | 	return xfrm_lookup(sock_net(sk), dst, flowi6_to_flowi(fl6), sk, 0); | ||||||
| } | } | ||||||
| EXPORT_SYMBOL_GPL(ip6_sk_dst_lookup_flow); | EXPORT_SYMBOL_GPL(ip6_sk_dst_lookup_flow); | ||||||
| 
 | 
 | ||||||
|  | @ -1145,7 +1145,7 @@ static inline struct ipv6_rt_hdr *ip6_rthdr_dup(struct ipv6_rt_hdr *src, | ||||||
| int ip6_append_data(struct sock *sk, int getfrag(void *from, char *to, | int ip6_append_data(struct sock *sk, int getfrag(void *from, char *to, | ||||||
| 	int offset, int len, int odd, struct sk_buff *skb), | 	int offset, int len, int odd, struct sk_buff *skb), | ||||||
| 	void *from, int length, int transhdrlen, | 	void *from, int length, int transhdrlen, | ||||||
| 	int hlimit, int tclass, struct ipv6_txoptions *opt, struct flowi *fl, | 	int hlimit, int tclass, struct ipv6_txoptions *opt, struct flowi6 *fl6, | ||||||
| 	struct rt6_info *rt, unsigned int flags, int dontfrag) | 	struct rt6_info *rt, unsigned int flags, int dontfrag) | ||||||
| { | { | ||||||
| 	struct inet_sock *inet = inet_sk(sk); | 	struct inet_sock *inet = inet_sk(sk); | ||||||
|  | @ -1203,7 +1203,7 @@ int ip6_append_data(struct sock *sk, int getfrag(void *from, char *to, | ||||||
| 		} | 		} | ||||||
| 		dst_hold(&rt->dst); | 		dst_hold(&rt->dst); | ||||||
| 		inet->cork.dst = &rt->dst; | 		inet->cork.dst = &rt->dst; | ||||||
| 		inet->cork.fl = *fl; | 		inet->cork.fl.u.ip6 = *fl6; | ||||||
| 		np->cork.hop_limit = hlimit; | 		np->cork.hop_limit = hlimit; | ||||||
| 		np->cork.tclass = tclass; | 		np->cork.tclass = tclass; | ||||||
| 		mtu = np->pmtudisc == IPV6_PMTUDISC_PROBE ? | 		mtu = np->pmtudisc == IPV6_PMTUDISC_PROBE ? | ||||||
|  | @ -1224,7 +1224,7 @@ int ip6_append_data(struct sock *sk, int getfrag(void *from, char *to, | ||||||
| 		transhdrlen += exthdrlen; | 		transhdrlen += exthdrlen; | ||||||
| 	} else { | 	} else { | ||||||
| 		rt = (struct rt6_info *)inet->cork.dst; | 		rt = (struct rt6_info *)inet->cork.dst; | ||||||
| 		fl = &inet->cork.fl; | 		fl6 = &inet->cork.fl.u.ip6; | ||||||
| 		opt = np->cork.opt; | 		opt = np->cork.opt; | ||||||
| 		transhdrlen = 0; | 		transhdrlen = 0; | ||||||
| 		exthdrlen = 0; | 		exthdrlen = 0; | ||||||
|  | @ -1239,7 +1239,7 @@ int ip6_append_data(struct sock *sk, int getfrag(void *from, char *to, | ||||||
| 
 | 
 | ||||||
| 	if (mtu <= sizeof(struct ipv6hdr) + IPV6_MAXPLEN) { | 	if (mtu <= sizeof(struct ipv6hdr) + IPV6_MAXPLEN) { | ||||||
| 		if (inet->cork.length + length > sizeof(struct ipv6hdr) + IPV6_MAXPLEN - fragheaderlen) { | 		if (inet->cork.length + length > sizeof(struct ipv6hdr) + IPV6_MAXPLEN - fragheaderlen) { | ||||||
| 			ipv6_local_error(sk, EMSGSIZE, fl, mtu-exthdrlen); | 			ipv6_local_error(sk, EMSGSIZE, fl6, mtu-exthdrlen); | ||||||
| 			return -EMSGSIZE; | 			return -EMSGSIZE; | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | @ -1271,7 +1271,7 @@ int ip6_append_data(struct sock *sk, int getfrag(void *from, char *to, | ||||||
| 	if (length > mtu) { | 	if (length > mtu) { | ||||||
| 		int proto = sk->sk_protocol; | 		int proto = sk->sk_protocol; | ||||||
| 		if (dontfrag && (proto == IPPROTO_UDP || proto == IPPROTO_RAW)){ | 		if (dontfrag && (proto == IPPROTO_UDP || proto == IPPROTO_RAW)){ | ||||||
| 			ipv6_local_rxpmtu(sk, fl, mtu-exthdrlen); | 			ipv6_local_rxpmtu(sk, fl6, mtu-exthdrlen); | ||||||
| 			return -EMSGSIZE; | 			return -EMSGSIZE; | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
|  | @ -1516,8 +1516,8 @@ int ip6_push_pending_frames(struct sock *sk) | ||||||
| 	struct ipv6hdr *hdr; | 	struct ipv6hdr *hdr; | ||||||
| 	struct ipv6_txoptions *opt = np->cork.opt; | 	struct ipv6_txoptions *opt = np->cork.opt; | ||||||
| 	struct rt6_info *rt = (struct rt6_info *)inet->cork.dst; | 	struct rt6_info *rt = (struct rt6_info *)inet->cork.dst; | ||||||
| 	struct flowi *fl = &inet->cork.fl; | 	struct flowi6 *fl6 = &inet->cork.fl.u.ip6; | ||||||
| 	unsigned char proto = fl->flowi_proto; | 	unsigned char proto = fl6->flowi6_proto; | ||||||
| 	int err = 0; | 	int err = 0; | ||||||
| 
 | 
 | ||||||
| 	if ((skb = __skb_dequeue(&sk->sk_write_queue)) == NULL) | 	if ((skb = __skb_dequeue(&sk->sk_write_queue)) == NULL) | ||||||
|  | @ -1542,7 +1542,7 @@ int ip6_push_pending_frames(struct sock *sk) | ||||||
| 	if (np->pmtudisc < IPV6_PMTUDISC_DO) | 	if (np->pmtudisc < IPV6_PMTUDISC_DO) | ||||||
| 		skb->local_df = 1; | 		skb->local_df = 1; | ||||||
| 
 | 
 | ||||||
| 	ipv6_addr_copy(final_dst, &fl->fl6_dst); | 	ipv6_addr_copy(final_dst, &fl6->daddr); | ||||||
| 	__skb_pull(skb, skb_network_header_len(skb)); | 	__skb_pull(skb, skb_network_header_len(skb)); | ||||||
| 	if (opt && opt->opt_flen) | 	if (opt && opt->opt_flen) | ||||||
| 		ipv6_push_frag_opts(skb, opt, &proto); | 		ipv6_push_frag_opts(skb, opt, &proto); | ||||||
|  | @ -1553,12 +1553,12 @@ int ip6_push_pending_frames(struct sock *sk) | ||||||
| 	skb_reset_network_header(skb); | 	skb_reset_network_header(skb); | ||||||
| 	hdr = ipv6_hdr(skb); | 	hdr = ipv6_hdr(skb); | ||||||
| 
 | 
 | ||||||
| 	*(__be32*)hdr = fl->fl6_flowlabel | | 	*(__be32*)hdr = fl6->flowlabel | | ||||||
| 		     htonl(0x60000000 | ((int)np->cork.tclass << 20)); | 		     htonl(0x60000000 | ((int)np->cork.tclass << 20)); | ||||||
| 
 | 
 | ||||||
| 	hdr->hop_limit = np->cork.hop_limit; | 	hdr->hop_limit = np->cork.hop_limit; | ||||||
| 	hdr->nexthdr = proto; | 	hdr->nexthdr = proto; | ||||||
| 	ipv6_addr_copy(&hdr->saddr, &fl->fl6_src); | 	ipv6_addr_copy(&hdr->saddr, &fl6->saddr); | ||||||
| 	ipv6_addr_copy(&hdr->daddr, final_dst); | 	ipv6_addr_copy(&hdr->daddr, final_dst); | ||||||
| 
 | 
 | ||||||
| 	skb->priority = sk->sk_priority; | 	skb->priority = sk->sk_priority; | ||||||
|  |  | ||||||
|  | @ -884,7 +884,7 @@ static inline int ip6_tnl_xmit_ctl(struct ip6_tnl *t) | ||||||
| static int ip6_tnl_xmit2(struct sk_buff *skb, | static int ip6_tnl_xmit2(struct sk_buff *skb, | ||||||
| 			 struct net_device *dev, | 			 struct net_device *dev, | ||||||
| 			 __u8 dsfield, | 			 __u8 dsfield, | ||||||
| 			 struct flowi *fl, | 			 struct flowi6 *fl6, | ||||||
| 			 int encap_limit, | 			 int encap_limit, | ||||||
| 			 __u32 *pmtu) | 			 __u32 *pmtu) | ||||||
| { | { | ||||||
|  | @ -904,11 +904,11 @@ static int ip6_tnl_xmit2(struct sk_buff *skb, | ||||||
| 	if ((dst = ip6_tnl_dst_check(t)) != NULL) | 	if ((dst = ip6_tnl_dst_check(t)) != NULL) | ||||||
| 		dst_hold(dst); | 		dst_hold(dst); | ||||||
| 	else { | 	else { | ||||||
| 		dst = ip6_route_output(net, NULL, fl); | 		dst = ip6_route_output(net, NULL, fl6); | ||||||
| 
 | 
 | ||||||
| 		if (dst->error) | 		if (dst->error) | ||||||
| 			goto tx_err_link_failure; | 			goto tx_err_link_failure; | ||||||
| 		dst = xfrm_lookup(net, dst, fl, NULL, 0); | 		dst = xfrm_lookup(net, dst, flowi6_to_flowi(fl6), NULL, 0); | ||||||
| 		if (IS_ERR(dst)) { | 		if (IS_ERR(dst)) { | ||||||
| 			err = PTR_ERR(dst); | 			err = PTR_ERR(dst); | ||||||
| 			dst = NULL; | 			dst = NULL; | ||||||
|  | @ -963,7 +963,7 @@ static int ip6_tnl_xmit2(struct sk_buff *skb, | ||||||
| 
 | 
 | ||||||
| 	skb->transport_header = skb->network_header; | 	skb->transport_header = skb->network_header; | ||||||
| 
 | 
 | ||||||
| 	proto = fl->flowi_proto; | 	proto = fl6->flowi6_proto; | ||||||
| 	if (encap_limit >= 0) { | 	if (encap_limit >= 0) { | ||||||
| 		init_tel_txopt(&opt, encap_limit); | 		init_tel_txopt(&opt, encap_limit); | ||||||
| 		ipv6_push_nfrag_opts(skb, &opt.ops, &proto, NULL); | 		ipv6_push_nfrag_opts(skb, &opt.ops, &proto, NULL); | ||||||
|  | @ -971,13 +971,13 @@ static int ip6_tnl_xmit2(struct sk_buff *skb, | ||||||
| 	skb_push(skb, sizeof(struct ipv6hdr)); | 	skb_push(skb, sizeof(struct ipv6hdr)); | ||||||
| 	skb_reset_network_header(skb); | 	skb_reset_network_header(skb); | ||||||
| 	ipv6h = ipv6_hdr(skb); | 	ipv6h = ipv6_hdr(skb); | ||||||
| 	*(__be32*)ipv6h = fl->fl6_flowlabel | htonl(0x60000000); | 	*(__be32*)ipv6h = fl6->flowlabel | htonl(0x60000000); | ||||||
| 	dsfield = INET_ECN_encapsulate(0, dsfield); | 	dsfield = INET_ECN_encapsulate(0, dsfield); | ||||||
| 	ipv6_change_dsfield(ipv6h, ~INET_ECN_MASK, dsfield); | 	ipv6_change_dsfield(ipv6h, ~INET_ECN_MASK, dsfield); | ||||||
| 	ipv6h->hop_limit = t->parms.hop_limit; | 	ipv6h->hop_limit = t->parms.hop_limit; | ||||||
| 	ipv6h->nexthdr = proto; | 	ipv6h->nexthdr = proto; | ||||||
| 	ipv6_addr_copy(&ipv6h->saddr, &fl->fl6_src); | 	ipv6_addr_copy(&ipv6h->saddr, &fl6->saddr); | ||||||
| 	ipv6_addr_copy(&ipv6h->daddr, &fl->fl6_dst); | 	ipv6_addr_copy(&ipv6h->daddr, &fl6->daddr); | ||||||
| 	nf_reset(skb); | 	nf_reset(skb); | ||||||
| 	pkt_len = skb->len; | 	pkt_len = skb->len; | ||||||
| 	err = ip6_local_out(skb); | 	err = ip6_local_out(skb); | ||||||
|  | @ -1007,7 +1007,7 @@ ip4ip6_tnl_xmit(struct sk_buff *skb, struct net_device *dev) | ||||||
| 	struct ip6_tnl *t = netdev_priv(dev); | 	struct ip6_tnl *t = netdev_priv(dev); | ||||||
| 	struct iphdr  *iph = ip_hdr(skb); | 	struct iphdr  *iph = ip_hdr(skb); | ||||||
| 	int encap_limit = -1; | 	int encap_limit = -1; | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 	__u8 dsfield; | 	__u8 dsfield; | ||||||
| 	__u32 mtu; | 	__u32 mtu; | ||||||
| 	int err; | 	int err; | ||||||
|  | @ -1019,16 +1019,16 @@ ip4ip6_tnl_xmit(struct sk_buff *skb, struct net_device *dev) | ||||||
| 	if (!(t->parms.flags & IP6_TNL_F_IGN_ENCAP_LIMIT)) | 	if (!(t->parms.flags & IP6_TNL_F_IGN_ENCAP_LIMIT)) | ||||||
| 		encap_limit = t->parms.encap_limit; | 		encap_limit = t->parms.encap_limit; | ||||||
| 
 | 
 | ||||||
| 	memcpy(&fl, &t->fl, sizeof (fl)); | 	memcpy(&fl6, &t->fl.u.ip6, sizeof (fl6)); | ||||||
| 	fl.flowi_proto = IPPROTO_IPIP; | 	fl6.flowi6_proto = IPPROTO_IPIP; | ||||||
| 
 | 
 | ||||||
| 	dsfield = ipv4_get_dsfield(iph); | 	dsfield = ipv4_get_dsfield(iph); | ||||||
| 
 | 
 | ||||||
| 	if ((t->parms.flags & IP6_TNL_F_USE_ORIG_TCLASS)) | 	if ((t->parms.flags & IP6_TNL_F_USE_ORIG_TCLASS)) | ||||||
| 		fl.fl6_flowlabel |= htonl((__u32)iph->tos << IPV6_TCLASS_SHIFT) | 		fl6.flowlabel |= htonl((__u32)iph->tos << IPV6_TCLASS_SHIFT) | ||||||
| 					  & IPV6_TCLASS_MASK; | 					  & IPV6_TCLASS_MASK; | ||||||
| 
 | 
 | ||||||
| 	err = ip6_tnl_xmit2(skb, dev, dsfield, &fl, encap_limit, &mtu); | 	err = ip6_tnl_xmit2(skb, dev, dsfield, &fl6, encap_limit, &mtu); | ||||||
| 	if (err != 0) { | 	if (err != 0) { | ||||||
| 		/* XXX: send ICMP error even if DF is not set. */ | 		/* XXX: send ICMP error even if DF is not set. */ | ||||||
| 		if (err == -EMSGSIZE) | 		if (err == -EMSGSIZE) | ||||||
|  | @ -1047,7 +1047,7 @@ ip6ip6_tnl_xmit(struct sk_buff *skb, struct net_device *dev) | ||||||
| 	struct ipv6hdr *ipv6h = ipv6_hdr(skb); | 	struct ipv6hdr *ipv6h = ipv6_hdr(skb); | ||||||
| 	int encap_limit = -1; | 	int encap_limit = -1; | ||||||
| 	__u16 offset; | 	__u16 offset; | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 	__u8 dsfield; | 	__u8 dsfield; | ||||||
| 	__u32 mtu; | 	__u32 mtu; | ||||||
| 	int err; | 	int err; | ||||||
|  | @ -1069,16 +1069,16 @@ ip6ip6_tnl_xmit(struct sk_buff *skb, struct net_device *dev) | ||||||
| 	} else if (!(t->parms.flags & IP6_TNL_F_IGN_ENCAP_LIMIT)) | 	} else if (!(t->parms.flags & IP6_TNL_F_IGN_ENCAP_LIMIT)) | ||||||
| 		encap_limit = t->parms.encap_limit; | 		encap_limit = t->parms.encap_limit; | ||||||
| 
 | 
 | ||||||
| 	memcpy(&fl, &t->fl, sizeof (fl)); | 	memcpy(&fl6, &t->fl.u.ip6, sizeof (fl6)); | ||||||
| 	fl.flowi_proto = IPPROTO_IPV6; | 	fl6.flowi6_proto = IPPROTO_IPV6; | ||||||
| 
 | 
 | ||||||
| 	dsfield = ipv6_get_dsfield(ipv6h); | 	dsfield = ipv6_get_dsfield(ipv6h); | ||||||
| 	if ((t->parms.flags & IP6_TNL_F_USE_ORIG_TCLASS)) | 	if ((t->parms.flags & IP6_TNL_F_USE_ORIG_TCLASS)) | ||||||
| 		fl.fl6_flowlabel |= (*(__be32 *) ipv6h & IPV6_TCLASS_MASK); | 		fl6.flowlabel |= (*(__be32 *) ipv6h & IPV6_TCLASS_MASK); | ||||||
| 	if ((t->parms.flags & IP6_TNL_F_USE_ORIG_FLOWLABEL)) | 	if ((t->parms.flags & IP6_TNL_F_USE_ORIG_FLOWLABEL)) | ||||||
| 		fl.fl6_flowlabel |= (*(__be32 *) ipv6h & IPV6_FLOWLABEL_MASK); | 		fl6.flowlabel |= (*(__be32 *) ipv6h & IPV6_FLOWLABEL_MASK); | ||||||
| 
 | 
 | ||||||
| 	err = ip6_tnl_xmit2(skb, dev, dsfield, &fl, encap_limit, &mtu); | 	err = ip6_tnl_xmit2(skb, dev, dsfield, &fl6, encap_limit, &mtu); | ||||||
| 	if (err != 0) { | 	if (err != 0) { | ||||||
| 		if (err == -EMSGSIZE) | 		if (err == -EMSGSIZE) | ||||||
| 			icmpv6_send(skb, ICMPV6_PKT_TOOBIG, 0, mtu); | 			icmpv6_send(skb, ICMPV6_PKT_TOOBIG, 0, mtu); | ||||||
|  | @ -1141,21 +1141,21 @@ static void ip6_tnl_link_config(struct ip6_tnl *t) | ||||||
| { | { | ||||||
| 	struct net_device *dev = t->dev; | 	struct net_device *dev = t->dev; | ||||||
| 	struct ip6_tnl_parm *p = &t->parms; | 	struct ip6_tnl_parm *p = &t->parms; | ||||||
| 	struct flowi *fl = &t->fl; | 	struct flowi6 *fl6 = &t->fl.u.ip6; | ||||||
| 
 | 
 | ||||||
| 	memcpy(dev->dev_addr, &p->laddr, sizeof(struct in6_addr)); | 	memcpy(dev->dev_addr, &p->laddr, sizeof(struct in6_addr)); | ||||||
| 	memcpy(dev->broadcast, &p->raddr, sizeof(struct in6_addr)); | 	memcpy(dev->broadcast, &p->raddr, sizeof(struct in6_addr)); | ||||||
| 
 | 
 | ||||||
| 	/* Set up flowi template */ | 	/* Set up flowi template */ | ||||||
| 	ipv6_addr_copy(&fl->fl6_src, &p->laddr); | 	ipv6_addr_copy(&fl6->saddr, &p->laddr); | ||||||
| 	ipv6_addr_copy(&fl->fl6_dst, &p->raddr); | 	ipv6_addr_copy(&fl6->daddr, &p->raddr); | ||||||
| 	fl->flowi_oif = p->link; | 	fl6->flowi6_oif = p->link; | ||||||
| 	fl->fl6_flowlabel = 0; | 	fl6->flowlabel = 0; | ||||||
| 
 | 
 | ||||||
| 	if (!(p->flags&IP6_TNL_F_USE_ORIG_TCLASS)) | 	if (!(p->flags&IP6_TNL_F_USE_ORIG_TCLASS)) | ||||||
| 		fl->fl6_flowlabel |= IPV6_TCLASS_MASK & p->flowinfo; | 		fl6->flowlabel |= IPV6_TCLASS_MASK & p->flowinfo; | ||||||
| 	if (!(p->flags&IP6_TNL_F_USE_ORIG_FLOWLABEL)) | 	if (!(p->flags&IP6_TNL_F_USE_ORIG_FLOWLABEL)) | ||||||
| 		fl->fl6_flowlabel |= IPV6_FLOWLABEL_MASK & p->flowinfo; | 		fl6->flowlabel |= IPV6_FLOWLABEL_MASK & p->flowinfo; | ||||||
| 
 | 
 | ||||||
| 	ip6_tnl_set_cap(t); | 	ip6_tnl_set_cap(t); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -135,14 +135,15 @@ static struct mr6_table *ip6mr_get_table(struct net *net, u32 id) | ||||||
| 	return NULL; | 	return NULL; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int ip6mr_fib_lookup(struct net *net, struct flowi *flp, | static int ip6mr_fib_lookup(struct net *net, struct flowi6 *flp6, | ||||||
| 			    struct mr6_table **mrt) | 			    struct mr6_table **mrt) | ||||||
| { | { | ||||||
| 	struct ip6mr_result res; | 	struct ip6mr_result res; | ||||||
| 	struct fib_lookup_arg arg = { .result = &res, }; | 	struct fib_lookup_arg arg = { .result = &res, }; | ||||||
| 	int err; | 	int err; | ||||||
| 
 | 
 | ||||||
| 	err = fib_rules_lookup(net->ipv6.mr6_rules_ops, flp, 0, &arg); | 	err = fib_rules_lookup(net->ipv6.mr6_rules_ops, | ||||||
|  | 			       flowi6_to_flowi(flp6), 0, &arg); | ||||||
| 	if (err < 0) | 	if (err < 0) | ||||||
| 		return err; | 		return err; | ||||||
| 	*mrt = res.mrt; | 	*mrt = res.mrt; | ||||||
|  | @ -270,7 +271,7 @@ static struct mr6_table *ip6mr_get_table(struct net *net, u32 id) | ||||||
| 	return net->ipv6.mrt6; | 	return net->ipv6.mrt6; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int ip6mr_fib_lookup(struct net *net, struct flowi *flp, | static int ip6mr_fib_lookup(struct net *net, struct flowi6 *flp6, | ||||||
| 			    struct mr6_table **mrt) | 			    struct mr6_table **mrt) | ||||||
| { | { | ||||||
| 	*mrt = net->ipv6.mrt6; | 	*mrt = net->ipv6.mrt6; | ||||||
|  | @ -617,9 +618,9 @@ static int pim6_rcv(struct sk_buff *skb) | ||||||
| 	struct net_device  *reg_dev = NULL; | 	struct net_device  *reg_dev = NULL; | ||||||
| 	struct net *net = dev_net(skb->dev); | 	struct net *net = dev_net(skb->dev); | ||||||
| 	struct mr6_table *mrt; | 	struct mr6_table *mrt; | ||||||
| 	struct flowi fl = { | 	struct flowi6 fl6 = { | ||||||
| 		.flowi_iif	= skb->dev->ifindex, | 		.flowi6_iif	= skb->dev->ifindex, | ||||||
| 		.flowi_mark = skb->mark, | 		.flowi6_mark	= skb->mark, | ||||||
| 	}; | 	}; | ||||||
| 	int reg_vif_num; | 	int reg_vif_num; | ||||||
| 
 | 
 | ||||||
|  | @ -644,7 +645,7 @@ static int pim6_rcv(struct sk_buff *skb) | ||||||
| 	    ntohs(encap->payload_len) + sizeof(*pim) > skb->len) | 	    ntohs(encap->payload_len) + sizeof(*pim) > skb->len) | ||||||
| 		goto drop; | 		goto drop; | ||||||
| 
 | 
 | ||||||
| 	if (ip6mr_fib_lookup(net, &fl, &mrt) < 0) | 	if (ip6mr_fib_lookup(net, &fl6, &mrt) < 0) | ||||||
| 		goto drop; | 		goto drop; | ||||||
| 	reg_vif_num = mrt->mroute_reg_vif_num; | 	reg_vif_num = mrt->mroute_reg_vif_num; | ||||||
| 
 | 
 | ||||||
|  | @ -687,14 +688,14 @@ static netdev_tx_t reg_vif_xmit(struct sk_buff *skb, | ||||||
| { | { | ||||||
| 	struct net *net = dev_net(dev); | 	struct net *net = dev_net(dev); | ||||||
| 	struct mr6_table *mrt; | 	struct mr6_table *mrt; | ||||||
| 	struct flowi fl = { | 	struct flowi6 fl6 = { | ||||||
| 		.flowi_oif		= dev->ifindex, | 		.flowi6_oif	= dev->ifindex, | ||||||
| 		.flowi_iif		= skb->skb_iif, | 		.flowi6_iif	= skb->skb_iif, | ||||||
| 		.flowi_mark	= skb->mark, | 		.flowi6_mark	= skb->mark, | ||||||
| 	}; | 	}; | ||||||
| 	int err; | 	int err; | ||||||
| 
 | 
 | ||||||
| 	err = ip6mr_fib_lookup(net, &fl, &mrt); | 	err = ip6mr_fib_lookup(net, &fl6, &mrt); | ||||||
| 	if (err < 0) | 	if (err < 0) | ||||||
| 		return err; | 		return err; | ||||||
| 
 | 
 | ||||||
|  | @ -1547,13 +1548,13 @@ int ip6mr_sk_done(struct sock *sk) | ||||||
| struct sock *mroute6_socket(struct net *net, struct sk_buff *skb) | struct sock *mroute6_socket(struct net *net, struct sk_buff *skb) | ||||||
| { | { | ||||||
| 	struct mr6_table *mrt; | 	struct mr6_table *mrt; | ||||||
| 	struct flowi fl = { | 	struct flowi6 fl6 = { | ||||||
| 		.flowi_iif	= skb->skb_iif, | 		.flowi6_iif	= skb->skb_iif, | ||||||
| 		.flowi_oif	= skb->dev->ifindex, | 		.flowi6_oif	= skb->dev->ifindex, | ||||||
| 		.flowi_mark= skb->mark, | 		.flowi6_mark	= skb->mark, | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| 	if (ip6mr_fib_lookup(net, &fl, &mrt) < 0) | 	if (ip6mr_fib_lookup(net, &fl6, &mrt) < 0) | ||||||
| 		return NULL; | 		return NULL; | ||||||
| 
 | 
 | ||||||
| 	return mrt->mroute6_sk; | 	return mrt->mroute6_sk; | ||||||
|  | @ -1897,7 +1898,7 @@ static int ip6mr_forward2(struct net *net, struct mr6_table *mrt, | ||||||
| 	struct mif_device *vif = &mrt->vif6_table[vifi]; | 	struct mif_device *vif = &mrt->vif6_table[vifi]; | ||||||
| 	struct net_device *dev; | 	struct net_device *dev; | ||||||
| 	struct dst_entry *dst; | 	struct dst_entry *dst; | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 
 | 
 | ||||||
| 	if (vif->dev == NULL) | 	if (vif->dev == NULL) | ||||||
| 		goto out_free; | 		goto out_free; | ||||||
|  | @ -1915,12 +1916,12 @@ static int ip6mr_forward2(struct net *net, struct mr6_table *mrt, | ||||||
| 
 | 
 | ||||||
| 	ipv6h = ipv6_hdr(skb); | 	ipv6h = ipv6_hdr(skb); | ||||||
| 
 | 
 | ||||||
| 	fl = (struct flowi) { | 	fl6 = (struct flowi6) { | ||||||
| 		.flowi_oif = vif->link, | 		.flowi6_oif = vif->link, | ||||||
| 		.fl6_dst = ipv6h->daddr, | 		.daddr = ipv6h->daddr, | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| 	dst = ip6_route_output(net, NULL, &fl); | 	dst = ip6_route_output(net, NULL, &fl6); | ||||||
| 	if (!dst) | 	if (!dst) | ||||||
| 		goto out_free; | 		goto out_free; | ||||||
| 
 | 
 | ||||||
|  | @ -2043,13 +2044,13 @@ int ip6_mr_input(struct sk_buff *skb) | ||||||
| 	struct mfc6_cache *cache; | 	struct mfc6_cache *cache; | ||||||
| 	struct net *net = dev_net(skb->dev); | 	struct net *net = dev_net(skb->dev); | ||||||
| 	struct mr6_table *mrt; | 	struct mr6_table *mrt; | ||||||
| 	struct flowi fl = { | 	struct flowi6 fl6 = { | ||||||
| 		.flowi_iif	= skb->dev->ifindex, | 		.flowi6_iif	= skb->dev->ifindex, | ||||||
| 		.flowi_mark= skb->mark, | 		.flowi6_mark	= skb->mark, | ||||||
| 	}; | 	}; | ||||||
| 	int err; | 	int err; | ||||||
| 
 | 
 | ||||||
| 	err = ip6mr_fib_lookup(net, &fl, &mrt); | 	err = ip6mr_fib_lookup(net, &fl6, &mrt); | ||||||
| 	if (err < 0) | 	if (err < 0) | ||||||
| 		return err; | 		return err; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -444,12 +444,12 @@ sticky_done: | ||||||
| 	{ | 	{ | ||||||
| 		struct ipv6_txoptions *opt = NULL; | 		struct ipv6_txoptions *opt = NULL; | ||||||
| 		struct msghdr msg; | 		struct msghdr msg; | ||||||
| 		struct flowi fl; | 		struct flowi6 fl6; | ||||||
| 		int junk; | 		int junk; | ||||||
| 
 | 
 | ||||||
| 		fl.fl6_flowlabel = 0; | 		memset(&fl6, 0, sizeof(fl6)); | ||||||
| 		fl.flowi_oif = sk->sk_bound_dev_if; | 		fl6.flowi6_oif = sk->sk_bound_dev_if; | ||||||
| 		fl.flowi_mark = sk->sk_mark; | 		fl6.flowi6_mark = sk->sk_mark; | ||||||
| 
 | 
 | ||||||
| 		if (optlen == 0) | 		if (optlen == 0) | ||||||
| 			goto update; | 			goto update; | ||||||
|  | @ -475,7 +475,7 @@ sticky_done: | ||||||
| 		msg.msg_controllen = optlen; | 		msg.msg_controllen = optlen; | ||||||
| 		msg.msg_control = (void*)(opt+1); | 		msg.msg_control = (void*)(opt+1); | ||||||
| 
 | 
 | ||||||
| 		retv = datagram_send_ctl(net, &msg, &fl, opt, &junk, &junk, | 		retv = datagram_send_ctl(net, &msg, &fl6, opt, &junk, &junk, | ||||||
| 					 &junk); | 					 &junk); | ||||||
| 		if (retv) | 		if (retv) | ||||||
| 			goto done; | 			goto done; | ||||||
|  |  | ||||||
|  | @ -1396,7 +1396,7 @@ static void mld_sendpack(struct sk_buff *skb) | ||||||
| 	struct inet6_dev *idev; | 	struct inet6_dev *idev; | ||||||
| 	struct net *net = dev_net(skb->dev); | 	struct net *net = dev_net(skb->dev); | ||||||
| 	int err; | 	int err; | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 	struct dst_entry *dst; | 	struct dst_entry *dst; | ||||||
| 
 | 
 | ||||||
| 	rcu_read_lock(); | 	rcu_read_lock(); | ||||||
|  | @ -1419,11 +1419,11 @@ static void mld_sendpack(struct sk_buff *skb) | ||||||
| 		goto err_out; | 		goto err_out; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	icmpv6_flow_init(net->ipv6.igmp_sk, &fl, ICMPV6_MLD2_REPORT, | 	icmpv6_flow_init(net->ipv6.igmp_sk, &fl6, ICMPV6_MLD2_REPORT, | ||||||
| 			 &ipv6_hdr(skb)->saddr, &ipv6_hdr(skb)->daddr, | 			 &ipv6_hdr(skb)->saddr, &ipv6_hdr(skb)->daddr, | ||||||
| 			 skb->dev->ifindex); | 			 skb->dev->ifindex); | ||||||
| 
 | 
 | ||||||
| 	dst = xfrm_lookup(net, dst, &fl, NULL, 0); | 	dst = xfrm_lookup(net, dst, flowi6_to_flowi(&fl6), NULL, 0); | ||||||
| 	err = 0; | 	err = 0; | ||||||
| 	if (IS_ERR(dst)) { | 	if (IS_ERR(dst)) { | ||||||
| 		err = PTR_ERR(dst); | 		err = PTR_ERR(dst); | ||||||
|  | @ -1731,7 +1731,7 @@ static void igmp6_send(struct in6_addr *addr, struct net_device *dev, int type) | ||||||
| 	u8 ra[8] = { IPPROTO_ICMPV6, 0, | 	u8 ra[8] = { IPPROTO_ICMPV6, 0, | ||||||
| 		     IPV6_TLV_ROUTERALERT, 2, 0, 0, | 		     IPV6_TLV_ROUTERALERT, 2, 0, 0, | ||||||
| 		     IPV6_TLV_PADN, 0 }; | 		     IPV6_TLV_PADN, 0 }; | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 	struct dst_entry *dst; | 	struct dst_entry *dst; | ||||||
| 
 | 
 | ||||||
| 	if (type == ICMPV6_MGM_REDUCTION) | 	if (type == ICMPV6_MGM_REDUCTION) | ||||||
|  | @ -1791,11 +1791,11 @@ static void igmp6_send(struct in6_addr *addr, struct net_device *dev, int type) | ||||||
| 		goto err_out; | 		goto err_out; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	icmpv6_flow_init(sk, &fl, type, | 	icmpv6_flow_init(sk, &fl6, type, | ||||||
| 			 &ipv6_hdr(skb)->saddr, &ipv6_hdr(skb)->daddr, | 			 &ipv6_hdr(skb)->saddr, &ipv6_hdr(skb)->daddr, | ||||||
| 			 skb->dev->ifindex); | 			 skb->dev->ifindex); | ||||||
| 
 | 
 | ||||||
| 	dst = xfrm_lookup(net, dst, &fl, NULL, 0); | 	dst = xfrm_lookup(net, dst, flowi6_to_flowi(&fl6), NULL, 0); | ||||||
| 	if (IS_ERR(dst)) { | 	if (IS_ERR(dst)) { | ||||||
| 		err = PTR_ERR(dst); | 		err = PTR_ERR(dst); | ||||||
| 		goto err_out; | 		goto err_out; | ||||||
|  |  | ||||||
|  | @ -208,14 +208,15 @@ static int mip6_destopt_reject(struct xfrm_state *x, struct sk_buff *skb, | ||||||
| { | { | ||||||
| 	struct net *net = xs_net(x); | 	struct net *net = xs_net(x); | ||||||
| 	struct inet6_skb_parm *opt = (struct inet6_skb_parm *)skb->cb; | 	struct inet6_skb_parm *opt = (struct inet6_skb_parm *)skb->cb; | ||||||
|  | 	const struct flowi6 *fl6 = &fl->u.ip6; | ||||||
| 	struct ipv6_destopt_hao *hao = NULL; | 	struct ipv6_destopt_hao *hao = NULL; | ||||||
| 	struct xfrm_selector sel; | 	struct xfrm_selector sel; | ||||||
| 	int offset; | 	int offset; | ||||||
| 	struct timeval stamp; | 	struct timeval stamp; | ||||||
| 	int err = 0; | 	int err = 0; | ||||||
| 
 | 
 | ||||||
| 	if (unlikely(fl->flowi_proto == IPPROTO_MH && | 	if (unlikely(fl6->flowi6_proto == IPPROTO_MH && | ||||||
| 		     fl->fl6_mh_type <= IP6_MH_TYPE_MAX)) | 		     fl6->uli.mht.type <= IP6_MH_TYPE_MAX)) | ||||||
| 		goto out; | 		goto out; | ||||||
| 
 | 
 | ||||||
| 	if (likely(opt->dsthao)) { | 	if (likely(opt->dsthao)) { | ||||||
|  | @ -240,14 +241,14 @@ static int mip6_destopt_reject(struct xfrm_state *x, struct sk_buff *skb, | ||||||
| 	       sizeof(sel.saddr)); | 	       sizeof(sel.saddr)); | ||||||
| 	sel.prefixlen_s = 128; | 	sel.prefixlen_s = 128; | ||||||
| 	sel.family = AF_INET6; | 	sel.family = AF_INET6; | ||||||
| 	sel.proto = fl->flowi_proto; | 	sel.proto = fl6->flowi6_proto; | ||||||
| 	sel.dport = xfrm_flowi_dport(fl, &fl->u.ip6.uli); | 	sel.dport = xfrm_flowi_dport(fl, &fl6->uli); | ||||||
| 	if (sel.dport) | 	if (sel.dport) | ||||||
| 		sel.dport_mask = htons(~0); | 		sel.dport_mask = htons(~0); | ||||||
| 	sel.sport = xfrm_flowi_sport(fl, &fl->u.ip6.uli); | 	sel.sport = xfrm_flowi_sport(fl, &fl6->uli); | ||||||
| 	if (sel.sport) | 	if (sel.sport) | ||||||
| 		sel.sport_mask = htons(~0); | 		sel.sport_mask = htons(~0); | ||||||
| 	sel.ifindex = fl->flowi_oif; | 	sel.ifindex = fl6->flowi6_oif; | ||||||
| 
 | 
 | ||||||
| 	err = km_report(net, IPPROTO_DSTOPTS, &sel, | 	err = km_report(net, IPPROTO_DSTOPTS, &sel, | ||||||
| 			(hao ? (xfrm_address_t *)&hao->addr : NULL)); | 			(hao ? (xfrm_address_t *)&hao->addr : NULL)); | ||||||
|  |  | ||||||
|  | @ -511,7 +511,7 @@ void ndisc_send_skb(struct sk_buff *skb, | ||||||
| 		    const struct in6_addr *saddr, | 		    const struct in6_addr *saddr, | ||||||
| 		    struct icmp6hdr *icmp6h) | 		    struct icmp6hdr *icmp6h) | ||||||
| { | { | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 	struct dst_entry *dst; | 	struct dst_entry *dst; | ||||||
| 	struct net *net = dev_net(dev); | 	struct net *net = dev_net(dev); | ||||||
| 	struct sock *sk = net->ipv6.ndisc_sk; | 	struct sock *sk = net->ipv6.ndisc_sk; | ||||||
|  | @ -521,7 +521,7 @@ void ndisc_send_skb(struct sk_buff *skb, | ||||||
| 
 | 
 | ||||||
| 	type = icmp6h->icmp6_type; | 	type = icmp6h->icmp6_type; | ||||||
| 
 | 
 | ||||||
| 	icmpv6_flow_init(sk, &fl, type, saddr, daddr, dev->ifindex); | 	icmpv6_flow_init(sk, &fl6, type, saddr, daddr, dev->ifindex); | ||||||
| 
 | 
 | ||||||
| 	dst = icmp6_dst_alloc(dev, neigh, daddr); | 	dst = icmp6_dst_alloc(dev, neigh, daddr); | ||||||
| 	if (!dst) { | 	if (!dst) { | ||||||
|  | @ -529,7 +529,7 @@ void ndisc_send_skb(struct sk_buff *skb, | ||||||
| 		return; | 		return; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	dst = xfrm_lookup(net, dst, &fl, NULL, 0); | 	dst = xfrm_lookup(net, dst, flowi6_to_flowi(&fl6), NULL, 0); | ||||||
| 	if (IS_ERR(dst)) { | 	if (IS_ERR(dst)) { | ||||||
| 		kfree_skb(skb); | 		kfree_skb(skb); | ||||||
| 		return; | 		return; | ||||||
|  | @ -1515,7 +1515,7 @@ void ndisc_send_redirect(struct sk_buff *skb, struct neighbour *neigh, | ||||||
| 	struct rt6_info *rt; | 	struct rt6_info *rt; | ||||||
| 	struct dst_entry *dst; | 	struct dst_entry *dst; | ||||||
| 	struct inet6_dev *idev; | 	struct inet6_dev *idev; | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 	u8 *opt; | 	u8 *opt; | ||||||
| 	int rd_len; | 	int rd_len; | ||||||
| 	int err; | 	int err; | ||||||
|  | @ -1535,14 +1535,14 @@ void ndisc_send_redirect(struct sk_buff *skb, struct neighbour *neigh, | ||||||
| 		return; | 		return; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	icmpv6_flow_init(sk, &fl, NDISC_REDIRECT, | 	icmpv6_flow_init(sk, &fl6, NDISC_REDIRECT, | ||||||
| 			 &saddr_buf, &ipv6_hdr(skb)->saddr, dev->ifindex); | 			 &saddr_buf, &ipv6_hdr(skb)->saddr, dev->ifindex); | ||||||
| 
 | 
 | ||||||
| 	dst = ip6_route_output(net, NULL, &fl); | 	dst = ip6_route_output(net, NULL, &fl6); | ||||||
| 	if (dst == NULL) | 	if (dst == NULL) | ||||||
| 		return; | 		return; | ||||||
| 
 | 
 | ||||||
| 	dst = xfrm_lookup(net, dst, &fl, NULL, 0); | 	dst = xfrm_lookup(net, dst, flowi6_to_flowi(&fl6), NULL, 0); | ||||||
| 	if (IS_ERR(dst)) | 	if (IS_ERR(dst)) | ||||||
| 		return; | 		return; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -15,14 +15,14 @@ int ip6_route_me_harder(struct sk_buff *skb) | ||||||
| 	struct net *net = dev_net(skb_dst(skb)->dev); | 	struct net *net = dev_net(skb_dst(skb)->dev); | ||||||
| 	struct ipv6hdr *iph = ipv6_hdr(skb); | 	struct ipv6hdr *iph = ipv6_hdr(skb); | ||||||
| 	struct dst_entry *dst; | 	struct dst_entry *dst; | ||||||
| 	struct flowi fl = { | 	struct flowi6 fl6 = { | ||||||
| 		.flowi_oif = skb->sk ? skb->sk->sk_bound_dev_if : 0, | 		.flowi6_oif = skb->sk ? skb->sk->sk_bound_dev_if : 0, | ||||||
| 		.flowi_mark = skb->mark, | 		.flowi6_mark = skb->mark, | ||||||
| 		.fl6_dst = iph->daddr, | 		.daddr = iph->daddr, | ||||||
| 		.fl6_src = iph->saddr, | 		.saddr = iph->saddr, | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| 	dst = ip6_route_output(net, skb->sk, &fl); | 	dst = ip6_route_output(net, skb->sk, &fl6); | ||||||
| 	if (dst->error) { | 	if (dst->error) { | ||||||
| 		IP6_INC_STATS(net, ip6_dst_idev(dst), IPSTATS_MIB_OUTNOROUTES); | 		IP6_INC_STATS(net, ip6_dst_idev(dst), IPSTATS_MIB_OUTNOROUTES); | ||||||
| 		LIMIT_NETDEBUG(KERN_DEBUG "ip6_route_me_harder: No more route.\n"); | 		LIMIT_NETDEBUG(KERN_DEBUG "ip6_route_me_harder: No more route.\n"); | ||||||
|  | @ -37,9 +37,9 @@ int ip6_route_me_harder(struct sk_buff *skb) | ||||||
| 
 | 
 | ||||||
| #ifdef CONFIG_XFRM | #ifdef CONFIG_XFRM | ||||||
| 	if (!(IP6CB(skb)->flags & IP6SKB_XFRM_TRANSFORMED) && | 	if (!(IP6CB(skb)->flags & IP6SKB_XFRM_TRANSFORMED) && | ||||||
| 	    xfrm_decode_session(skb, &fl, AF_INET6) == 0) { | 	    xfrm_decode_session(skb, flowi6_to_flowi(&fl6), AF_INET6) == 0) { | ||||||
| 		skb_dst_set(skb, NULL); | 		skb_dst_set(skb, NULL); | ||||||
| 		dst = xfrm_lookup(net, dst, &fl, skb->sk, 0); | 		dst = xfrm_lookup(net, dst, flowi6_to_flowi(&fl6), skb->sk, 0); | ||||||
| 		if (IS_ERR(dst)) | 		if (IS_ERR(dst)) | ||||||
| 			return -1; | 			return -1; | ||||||
| 		skb_dst_set(skb, dst); | 		skb_dst_set(skb, dst); | ||||||
|  | @ -92,7 +92,7 @@ static int nf_ip6_reroute(struct sk_buff *skb, | ||||||
| 
 | 
 | ||||||
| static int nf_ip6_route(struct dst_entry **dst, struct flowi *fl) | static int nf_ip6_route(struct dst_entry **dst, struct flowi *fl) | ||||||
| { | { | ||||||
| 	*dst = ip6_route_output(&init_net, NULL, fl); | 	*dst = ip6_route_output(&init_net, NULL, &fl->u.ip6); | ||||||
| 	return (*dst)->error; | 	return (*dst)->error; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -47,7 +47,7 @@ static void send_reset(struct net *net, struct sk_buff *oldskb) | ||||||
| 	struct ipv6hdr *ip6h; | 	struct ipv6hdr *ip6h; | ||||||
| 	struct dst_entry *dst = NULL; | 	struct dst_entry *dst = NULL; | ||||||
| 	u8 proto; | 	u8 proto; | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 
 | 
 | ||||||
| 	if ((!(ipv6_addr_type(&oip6h->saddr) & IPV6_ADDR_UNICAST)) || | 	if ((!(ipv6_addr_type(&oip6h->saddr) & IPV6_ADDR_UNICAST)) || | ||||||
| 	    (!(ipv6_addr_type(&oip6h->daddr) & IPV6_ADDR_UNICAST))) { | 	    (!(ipv6_addr_type(&oip6h->daddr) & IPV6_ADDR_UNICAST))) { | ||||||
|  | @ -89,19 +89,19 @@ static void send_reset(struct net *net, struct sk_buff *oldskb) | ||||||
| 		return; | 		return; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	memset(&fl, 0, sizeof(fl)); | 	memset(&fl6, 0, sizeof(fl6)); | ||||||
| 	fl.flowi_proto = IPPROTO_TCP; | 	fl6.flowi6_proto = IPPROTO_TCP; | ||||||
| 	ipv6_addr_copy(&fl.fl6_src, &oip6h->daddr); | 	ipv6_addr_copy(&fl6.saddr, &oip6h->daddr); | ||||||
| 	ipv6_addr_copy(&fl.fl6_dst, &oip6h->saddr); | 	ipv6_addr_copy(&fl6.daddr, &oip6h->saddr); | ||||||
| 	fl.fl6_sport = otcph.dest; | 	fl6.uli.ports.sport = otcph.dest; | ||||||
| 	fl.fl6_dport = otcph.source; | 	fl6.uli.ports.dport = otcph.source; | ||||||
| 	security_skb_classify_flow(oldskb, &fl); | 	security_skb_classify_flow(oldskb, flowi6_to_flowi(&fl6)); | ||||||
| 	dst = ip6_route_output(net, NULL, &fl); | 	dst = ip6_route_output(net, NULL, &fl6); | ||||||
| 	if (dst == NULL || dst->error) { | 	if (dst == NULL || dst->error) { | ||||||
| 		dst_release(dst); | 		dst_release(dst); | ||||||
| 		return; | 		return; | ||||||
| 	} | 	} | ||||||
| 	dst = xfrm_lookup(net, dst, &fl, NULL, 0); | 	dst = xfrm_lookup(net, dst, flowi6_to_flowi(&fl6), NULL, 0); | ||||||
| 	if (IS_ERR(dst)) | 	if (IS_ERR(dst)) | ||||||
| 		return; | 		return; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -524,7 +524,7 @@ csum_copy_err: | ||||||
| 	goto out; | 	goto out; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int rawv6_push_pending_frames(struct sock *sk, struct flowi *fl, | static int rawv6_push_pending_frames(struct sock *sk, struct flowi6 *fl6, | ||||||
| 				     struct raw6_sock *rp) | 				     struct raw6_sock *rp) | ||||||
| { | { | ||||||
| 	struct sk_buff *skb; | 	struct sk_buff *skb; | ||||||
|  | @ -586,11 +586,10 @@ static int rawv6_push_pending_frames(struct sock *sk, struct flowi *fl, | ||||||
| 	if (unlikely(csum)) | 	if (unlikely(csum)) | ||||||
| 		tmp_csum = csum_sub(tmp_csum, csum_unfold(csum)); | 		tmp_csum = csum_sub(tmp_csum, csum_unfold(csum)); | ||||||
| 
 | 
 | ||||||
| 	csum = csum_ipv6_magic(&fl->fl6_src, | 	csum = csum_ipv6_magic(&fl6->saddr, &fl6->daddr, | ||||||
| 				   &fl->fl6_dst, | 			       total_len, fl6->flowi6_proto, tmp_csum); | ||||||
| 				   total_len, fl->flowi_proto, tmp_csum); |  | ||||||
| 
 | 
 | ||||||
| 	if (csum == 0 && fl->flowi_proto == IPPROTO_UDP) | 	if (csum == 0 && fl6->flowi6_proto == IPPROTO_UDP) | ||||||
| 		csum = CSUM_MANGLED_0; | 		csum = CSUM_MANGLED_0; | ||||||
| 
 | 
 | ||||||
| 	if (skb_store_bits(skb, offset, &csum, 2)) | 	if (skb_store_bits(skb, offset, &csum, 2)) | ||||||
|  | @ -603,7 +602,7 @@ out: | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int rawv6_send_hdrinc(struct sock *sk, void *from, int length, | static int rawv6_send_hdrinc(struct sock *sk, void *from, int length, | ||||||
| 			struct flowi *fl, struct dst_entry **dstp, | 			struct flowi6 *fl6, struct dst_entry **dstp, | ||||||
| 			unsigned int flags) | 			unsigned int flags) | ||||||
| { | { | ||||||
| 	struct ipv6_pinfo *np = inet6_sk(sk); | 	struct ipv6_pinfo *np = inet6_sk(sk); | ||||||
|  | @ -613,7 +612,7 @@ static int rawv6_send_hdrinc(struct sock *sk, void *from, int length, | ||||||
| 	struct rt6_info *rt = (struct rt6_info *)*dstp; | 	struct rt6_info *rt = (struct rt6_info *)*dstp; | ||||||
| 
 | 
 | ||||||
| 	if (length > rt->dst.dev->mtu) { | 	if (length > rt->dst.dev->mtu) { | ||||||
| 		ipv6_local_error(sk, EMSGSIZE, fl, rt->dst.dev->mtu); | 		ipv6_local_error(sk, EMSGSIZE, fl6, rt->dst.dev->mtu); | ||||||
| 		return -EMSGSIZE; | 		return -EMSGSIZE; | ||||||
| 	} | 	} | ||||||
| 	if (flags&MSG_PROBE) | 	if (flags&MSG_PROBE) | ||||||
|  | @ -662,7 +661,7 @@ error: | ||||||
| 	return err; | 	return err; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int rawv6_probe_proto_opt(struct flowi *fl, struct msghdr *msg) | static int rawv6_probe_proto_opt(struct flowi6 *fl6, struct msghdr *msg) | ||||||
| { | { | ||||||
| 	struct iovec *iov; | 	struct iovec *iov; | ||||||
| 	u8 __user *type = NULL; | 	u8 __user *type = NULL; | ||||||
|  | @ -679,7 +678,7 @@ static int rawv6_probe_proto_opt(struct flowi *fl, struct msghdr *msg) | ||||||
| 		if (!iov) | 		if (!iov) | ||||||
| 			continue; | 			continue; | ||||||
| 
 | 
 | ||||||
| 		switch (fl->flowi_proto) { | 		switch (fl6->flowi6_proto) { | ||||||
| 		case IPPROTO_ICMPV6: | 		case IPPROTO_ICMPV6: | ||||||
| 			/* check if one-byte field is readable or not. */ | 			/* check if one-byte field is readable or not. */ | ||||||
| 			if (iov->iov_base && iov->iov_len < 1) | 			if (iov->iov_base && iov->iov_len < 1) | ||||||
|  | @ -694,8 +693,8 @@ static int rawv6_probe_proto_opt(struct flowi *fl, struct msghdr *msg) | ||||||
| 				code = iov->iov_base; | 				code = iov->iov_base; | ||||||
| 
 | 
 | ||||||
| 			if (type && code) { | 			if (type && code) { | ||||||
| 				if (get_user(fl->fl6_icmp_type, type) || | 				if (get_user(fl6->uli.icmpt.type, type) || | ||||||
| 				    get_user(fl->fl6_icmp_code, code)) | 				    get_user(fl6->uli.icmpt.code, code)) | ||||||
| 					return -EFAULT; | 					return -EFAULT; | ||||||
| 				probed = 1; | 				probed = 1; | ||||||
| 			} | 			} | ||||||
|  | @ -706,7 +705,7 @@ static int rawv6_probe_proto_opt(struct flowi *fl, struct msghdr *msg) | ||||||
| 			/* check if type field is readable or not. */ | 			/* check if type field is readable or not. */ | ||||||
| 			if (iov->iov_len > 2 - len) { | 			if (iov->iov_len > 2 - len) { | ||||||
| 				u8 __user *p = iov->iov_base; | 				u8 __user *p = iov->iov_base; | ||||||
| 				if (get_user(fl->fl6_mh_type, &p[2 - len])) | 				if (get_user(fl6->uli.mht.type, &p[2 - len])) | ||||||
| 					return -EFAULT; | 					return -EFAULT; | ||||||
| 				probed = 1; | 				probed = 1; | ||||||
| 			} else | 			} else | ||||||
|  | @ -735,7 +734,7 @@ static int rawv6_sendmsg(struct kiocb *iocb, struct sock *sk, | ||||||
| 	struct ipv6_txoptions *opt = NULL; | 	struct ipv6_txoptions *opt = NULL; | ||||||
| 	struct ip6_flowlabel *flowlabel = NULL; | 	struct ip6_flowlabel *flowlabel = NULL; | ||||||
| 	struct dst_entry *dst = NULL; | 	struct dst_entry *dst = NULL; | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 	int addr_len = msg->msg_namelen; | 	int addr_len = msg->msg_namelen; | ||||||
| 	int hlimit = -1; | 	int hlimit = -1; | ||||||
| 	int tclass = -1; | 	int tclass = -1; | ||||||
|  | @ -756,9 +755,9 @@ static int rawv6_sendmsg(struct kiocb *iocb, struct sock *sk, | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 *	Get and verify the address. | 	 *	Get and verify the address. | ||||||
| 	 */ | 	 */ | ||||||
| 	memset(&fl, 0, sizeof(fl)); | 	memset(&fl6, 0, sizeof(fl6)); | ||||||
| 
 | 
 | ||||||
| 	fl.flowi_mark = sk->sk_mark; | 	fl6.flowi6_mark = sk->sk_mark; | ||||||
| 
 | 
 | ||||||
| 	if (sin6) { | 	if (sin6) { | ||||||
| 		if (addr_len < SIN6_LEN_RFC2133) | 		if (addr_len < SIN6_LEN_RFC2133) | ||||||
|  | @ -780,9 +779,9 @@ static int rawv6_sendmsg(struct kiocb *iocb, struct sock *sk, | ||||||
| 
 | 
 | ||||||
| 		daddr = &sin6->sin6_addr; | 		daddr = &sin6->sin6_addr; | ||||||
| 		if (np->sndflow) { | 		if (np->sndflow) { | ||||||
| 			fl.fl6_flowlabel = sin6->sin6_flowinfo&IPV6_FLOWINFO_MASK; | 			fl6.flowlabel = sin6->sin6_flowinfo&IPV6_FLOWINFO_MASK; | ||||||
| 			if (fl.fl6_flowlabel&IPV6_FLOWLABEL_MASK) { | 			if (fl6.flowlabel&IPV6_FLOWLABEL_MASK) { | ||||||
| 				flowlabel = fl6_sock_lookup(sk, fl.fl6_flowlabel); | 				flowlabel = fl6_sock_lookup(sk, fl6.flowlabel); | ||||||
| 				if (flowlabel == NULL) | 				if (flowlabel == NULL) | ||||||
| 					return -EINVAL; | 					return -EINVAL; | ||||||
| 				daddr = &flowlabel->dst; | 				daddr = &flowlabel->dst; | ||||||
|  | @ -800,32 +799,32 @@ static int rawv6_sendmsg(struct kiocb *iocb, struct sock *sk, | ||||||
| 		if (addr_len >= sizeof(struct sockaddr_in6) && | 		if (addr_len >= sizeof(struct sockaddr_in6) && | ||||||
| 		    sin6->sin6_scope_id && | 		    sin6->sin6_scope_id && | ||||||
| 		    ipv6_addr_type(daddr)&IPV6_ADDR_LINKLOCAL) | 		    ipv6_addr_type(daddr)&IPV6_ADDR_LINKLOCAL) | ||||||
| 			fl.flowi_oif = sin6->sin6_scope_id; | 			fl6.flowi6_oif = sin6->sin6_scope_id; | ||||||
| 	} else { | 	} else { | ||||||
| 		if (sk->sk_state != TCP_ESTABLISHED) | 		if (sk->sk_state != TCP_ESTABLISHED) | ||||||
| 			return -EDESTADDRREQ; | 			return -EDESTADDRREQ; | ||||||
| 
 | 
 | ||||||
| 		proto = inet->inet_num; | 		proto = inet->inet_num; | ||||||
| 		daddr = &np->daddr; | 		daddr = &np->daddr; | ||||||
| 		fl.fl6_flowlabel = np->flow_label; | 		fl6.flowlabel = np->flow_label; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (fl.flowi_oif == 0) | 	if (fl6.flowi6_oif == 0) | ||||||
| 		fl.flowi_oif = sk->sk_bound_dev_if; | 		fl6.flowi6_oif = sk->sk_bound_dev_if; | ||||||
| 
 | 
 | ||||||
| 	if (msg->msg_controllen) { | 	if (msg->msg_controllen) { | ||||||
| 		opt = &opt_space; | 		opt = &opt_space; | ||||||
| 		memset(opt, 0, sizeof(struct ipv6_txoptions)); | 		memset(opt, 0, sizeof(struct ipv6_txoptions)); | ||||||
| 		opt->tot_len = sizeof(struct ipv6_txoptions); | 		opt->tot_len = sizeof(struct ipv6_txoptions); | ||||||
| 
 | 
 | ||||||
| 		err = datagram_send_ctl(sock_net(sk), msg, &fl, opt, &hlimit, | 		err = datagram_send_ctl(sock_net(sk), msg, &fl6, opt, &hlimit, | ||||||
| 					&tclass, &dontfrag); | 					&tclass, &dontfrag); | ||||||
| 		if (err < 0) { | 		if (err < 0) { | ||||||
| 			fl6_sock_release(flowlabel); | 			fl6_sock_release(flowlabel); | ||||||
| 			return err; | 			return err; | ||||||
| 		} | 		} | ||||||
| 		if ((fl.fl6_flowlabel&IPV6_FLOWLABEL_MASK) && !flowlabel) { | 		if ((fl6.flowlabel&IPV6_FLOWLABEL_MASK) && !flowlabel) { | ||||||
| 			flowlabel = fl6_sock_lookup(sk, fl.fl6_flowlabel); | 			flowlabel = fl6_sock_lookup(sk, fl6.flowlabel); | ||||||
| 			if (flowlabel == NULL) | 			if (flowlabel == NULL) | ||||||
| 				return -EINVAL; | 				return -EINVAL; | ||||||
| 		} | 		} | ||||||
|  | @ -838,31 +837,31 @@ static int rawv6_sendmsg(struct kiocb *iocb, struct sock *sk, | ||||||
| 		opt = fl6_merge_options(&opt_space, flowlabel, opt); | 		opt = fl6_merge_options(&opt_space, flowlabel, opt); | ||||||
| 	opt = ipv6_fixup_options(&opt_space, opt); | 	opt = ipv6_fixup_options(&opt_space, opt); | ||||||
| 
 | 
 | ||||||
| 	fl.flowi_proto = proto; | 	fl6.flowi6_proto = proto; | ||||||
| 	err = rawv6_probe_proto_opt(&fl, msg); | 	err = rawv6_probe_proto_opt(&fl6, msg); | ||||||
| 	if (err) | 	if (err) | ||||||
| 		goto out; | 		goto out; | ||||||
| 
 | 
 | ||||||
| 	if (!ipv6_addr_any(daddr)) | 	if (!ipv6_addr_any(daddr)) | ||||||
| 		ipv6_addr_copy(&fl.fl6_dst, daddr); | 		ipv6_addr_copy(&fl6.daddr, daddr); | ||||||
| 	else | 	else | ||||||
| 		fl.fl6_dst.s6_addr[15] = 0x1; /* :: means loopback (BSD'ism) */ | 		fl6.daddr.s6_addr[15] = 0x1; /* :: means loopback (BSD'ism) */ | ||||||
| 	if (ipv6_addr_any(&fl.fl6_src) && !ipv6_addr_any(&np->saddr)) | 	if (ipv6_addr_any(&fl6.saddr) && !ipv6_addr_any(&np->saddr)) | ||||||
| 		ipv6_addr_copy(&fl.fl6_src, &np->saddr); | 		ipv6_addr_copy(&fl6.saddr, &np->saddr); | ||||||
| 
 | 
 | ||||||
| 	final_p = fl6_update_dst(&fl, opt, &final); | 	final_p = fl6_update_dst(&fl6, opt, &final); | ||||||
| 
 | 
 | ||||||
| 	if (!fl.flowi_oif && ipv6_addr_is_multicast(&fl.fl6_dst)) | 	if (!fl6.flowi6_oif && ipv6_addr_is_multicast(&fl6.daddr)) | ||||||
| 		fl.flowi_oif = np->mcast_oif; | 		fl6.flowi6_oif = np->mcast_oif; | ||||||
| 	security_sk_classify_flow(sk, &fl); | 	security_sk_classify_flow(sk, flowi6_to_flowi(&fl6)); | ||||||
| 
 | 
 | ||||||
| 	dst = ip6_dst_lookup_flow(sk, &fl, final_p, true); | 	dst = ip6_dst_lookup_flow(sk, &fl6, final_p, true); | ||||||
| 	if (IS_ERR(dst)) { | 	if (IS_ERR(dst)) { | ||||||
| 		err = PTR_ERR(dst); | 		err = PTR_ERR(dst); | ||||||
| 		goto out; | 		goto out; | ||||||
| 	} | 	} | ||||||
| 	if (hlimit < 0) { | 	if (hlimit < 0) { | ||||||
| 		if (ipv6_addr_is_multicast(&fl.fl6_dst)) | 		if (ipv6_addr_is_multicast(&fl6.daddr)) | ||||||
| 			hlimit = np->mcast_hops; | 			hlimit = np->mcast_hops; | ||||||
| 		else | 		else | ||||||
| 			hlimit = np->hop_limit; | 			hlimit = np->hop_limit; | ||||||
|  | @ -881,17 +880,17 @@ static int rawv6_sendmsg(struct kiocb *iocb, struct sock *sk, | ||||||
| 
 | 
 | ||||||
| back_from_confirm: | back_from_confirm: | ||||||
| 	if (inet->hdrincl) | 	if (inet->hdrincl) | ||||||
| 		err = rawv6_send_hdrinc(sk, msg->msg_iov, len, &fl, &dst, msg->msg_flags); | 		err = rawv6_send_hdrinc(sk, msg->msg_iov, len, &fl6, &dst, msg->msg_flags); | ||||||
| 	else { | 	else { | ||||||
| 		lock_sock(sk); | 		lock_sock(sk); | ||||||
| 		err = ip6_append_data(sk, ip_generic_getfrag, msg->msg_iov, | 		err = ip6_append_data(sk, ip_generic_getfrag, msg->msg_iov, | ||||||
| 			len, 0, hlimit, tclass, opt, &fl, (struct rt6_info*)dst, | 			len, 0, hlimit, tclass, opt, &fl6, (struct rt6_info*)dst, | ||||||
| 			msg->msg_flags, dontfrag); | 			msg->msg_flags, dontfrag); | ||||||
| 
 | 
 | ||||||
| 		if (err) | 		if (err) | ||||||
| 			ip6_flush_pending_frames(sk); | 			ip6_flush_pending_frames(sk); | ||||||
| 		else if (!(msg->msg_flags & MSG_MORE)) | 		else if (!(msg->msg_flags & MSG_MORE)) | ||||||
| 			err = rawv6_push_pending_frames(sk, &fl, rp); | 			err = rawv6_push_pending_frames(sk, &fl6, rp); | ||||||
| 		release_sock(sk); | 		release_sock(sk); | ||||||
| 	} | 	} | ||||||
| done: | done: | ||||||
|  |  | ||||||
|  | @ -599,17 +599,17 @@ do { \ | ||||||
| 
 | 
 | ||||||
| static struct rt6_info *ip6_pol_route_lookup(struct net *net, | static struct rt6_info *ip6_pol_route_lookup(struct net *net, | ||||||
| 					     struct fib6_table *table, | 					     struct fib6_table *table, | ||||||
| 					     struct flowi *fl, int flags) | 					     struct flowi6 *fl6, int flags) | ||||||
| { | { | ||||||
| 	struct fib6_node *fn; | 	struct fib6_node *fn; | ||||||
| 	struct rt6_info *rt; | 	struct rt6_info *rt; | ||||||
| 
 | 
 | ||||||
| 	read_lock_bh(&table->tb6_lock); | 	read_lock_bh(&table->tb6_lock); | ||||||
| 	fn = fib6_lookup(&table->tb6_root, &fl->fl6_dst, &fl->fl6_src); | 	fn = fib6_lookup(&table->tb6_root, &fl6->daddr, &fl6->saddr); | ||||||
| restart: | restart: | ||||||
| 	rt = fn->leaf; | 	rt = fn->leaf; | ||||||
| 	rt = rt6_device_match(net, rt, &fl->fl6_src, fl->flowi_oif, flags); | 	rt = rt6_device_match(net, rt, &fl6->saddr, fl6->flowi6_oif, flags); | ||||||
| 	BACKTRACK(net, &fl->fl6_src); | 	BACKTRACK(net, &fl6->saddr); | ||||||
| out: | out: | ||||||
| 	dst_use(&rt->dst, jiffies); | 	dst_use(&rt->dst, jiffies); | ||||||
| 	read_unlock_bh(&table->tb6_lock); | 	read_unlock_bh(&table->tb6_lock); | ||||||
|  | @ -620,19 +620,19 @@ out: | ||||||
| struct rt6_info *rt6_lookup(struct net *net, const struct in6_addr *daddr, | struct rt6_info *rt6_lookup(struct net *net, const struct in6_addr *daddr, | ||||||
| 			    const struct in6_addr *saddr, int oif, int strict) | 			    const struct in6_addr *saddr, int oif, int strict) | ||||||
| { | { | ||||||
| 	struct flowi fl = { | 	struct flowi6 fl6 = { | ||||||
| 		.flowi_oif = oif, | 		.flowi6_oif = oif, | ||||||
| 		.fl6_dst = *daddr, | 		.daddr = *daddr, | ||||||
| 	}; | 	}; | ||||||
| 	struct dst_entry *dst; | 	struct dst_entry *dst; | ||||||
| 	int flags = strict ? RT6_LOOKUP_F_IFACE : 0; | 	int flags = strict ? RT6_LOOKUP_F_IFACE : 0; | ||||||
| 
 | 
 | ||||||
| 	if (saddr) { | 	if (saddr) { | ||||||
| 		memcpy(&fl.fl6_src, saddr, sizeof(*saddr)); | 		memcpy(&fl6.saddr, saddr, sizeof(*saddr)); | ||||||
| 		flags |= RT6_LOOKUP_F_HAS_SADDR; | 		flags |= RT6_LOOKUP_F_HAS_SADDR; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	dst = fib6_rule_lookup(net, &fl, flags, ip6_pol_route_lookup); | 	dst = fib6_rule_lookup(net, &fl6, flags, ip6_pol_route_lookup); | ||||||
| 	if (dst->error == 0) | 	if (dst->error == 0) | ||||||
| 		return (struct rt6_info *) dst; | 		return (struct rt6_info *) dst; | ||||||
| 
 | 
 | ||||||
|  | @ -753,7 +753,7 @@ static struct rt6_info *rt6_alloc_clone(struct rt6_info *ort, struct in6_addr *d | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static struct rt6_info *ip6_pol_route(struct net *net, struct fib6_table *table, int oif, | static struct rt6_info *ip6_pol_route(struct net *net, struct fib6_table *table, int oif, | ||||||
| 				      struct flowi *fl, int flags) | 				      struct flowi6 *fl6, int flags) | ||||||
| { | { | ||||||
| 	struct fib6_node *fn; | 	struct fib6_node *fn; | ||||||
| 	struct rt6_info *rt, *nrt; | 	struct rt6_info *rt, *nrt; | ||||||
|  | @ -768,12 +768,12 @@ relookup: | ||||||
| 	read_lock_bh(&table->tb6_lock); | 	read_lock_bh(&table->tb6_lock); | ||||||
| 
 | 
 | ||||||
| restart_2: | restart_2: | ||||||
| 	fn = fib6_lookup(&table->tb6_root, &fl->fl6_dst, &fl->fl6_src); | 	fn = fib6_lookup(&table->tb6_root, &fl6->daddr, &fl6->saddr); | ||||||
| 
 | 
 | ||||||
| restart: | restart: | ||||||
| 	rt = rt6_select(fn, oif, strict | reachable); | 	rt = rt6_select(fn, oif, strict | reachable); | ||||||
| 
 | 
 | ||||||
| 	BACKTRACK(net, &fl->fl6_src); | 	BACKTRACK(net, &fl6->saddr); | ||||||
| 	if (rt == net->ipv6.ip6_null_entry || | 	if (rt == net->ipv6.ip6_null_entry || | ||||||
| 	    rt->rt6i_flags & RTF_CACHE) | 	    rt->rt6i_flags & RTF_CACHE) | ||||||
| 		goto out; | 		goto out; | ||||||
|  | @ -782,9 +782,9 @@ restart: | ||||||
| 	read_unlock_bh(&table->tb6_lock); | 	read_unlock_bh(&table->tb6_lock); | ||||||
| 
 | 
 | ||||||
| 	if (!rt->rt6i_nexthop && !(rt->rt6i_flags & RTF_NONEXTHOP)) | 	if (!rt->rt6i_nexthop && !(rt->rt6i_flags & RTF_NONEXTHOP)) | ||||||
| 		nrt = rt6_alloc_cow(rt, &fl->fl6_dst, &fl->fl6_src); | 		nrt = rt6_alloc_cow(rt, &fl6->daddr, &fl6->saddr); | ||||||
| 	else if (!(rt->dst.flags & DST_HOST)) | 	else if (!(rt->dst.flags & DST_HOST)) | ||||||
| 		nrt = rt6_alloc_clone(rt, &fl->fl6_dst); | 		nrt = rt6_alloc_clone(rt, &fl6->daddr); | ||||||
| 	else | 	else | ||||||
| 		goto out2; | 		goto out2; | ||||||
| 
 | 
 | ||||||
|  | @ -823,9 +823,9 @@ out2: | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static struct rt6_info *ip6_pol_route_input(struct net *net, struct fib6_table *table, | static struct rt6_info *ip6_pol_route_input(struct net *net, struct fib6_table *table, | ||||||
| 					    struct flowi *fl, int flags) | 					    struct flowi6 *fl6, int flags) | ||||||
| { | { | ||||||
| 	return ip6_pol_route(net, table, fl->flowi_iif, fl, flags); | 	return ip6_pol_route(net, table, fl6->flowi6_iif, fl6, flags); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void ip6_route_input(struct sk_buff *skb) | void ip6_route_input(struct sk_buff *skb) | ||||||
|  | @ -833,41 +833,41 @@ void ip6_route_input(struct sk_buff *skb) | ||||||
| 	struct ipv6hdr *iph = ipv6_hdr(skb); | 	struct ipv6hdr *iph = ipv6_hdr(skb); | ||||||
| 	struct net *net = dev_net(skb->dev); | 	struct net *net = dev_net(skb->dev); | ||||||
| 	int flags = RT6_LOOKUP_F_HAS_SADDR; | 	int flags = RT6_LOOKUP_F_HAS_SADDR; | ||||||
| 	struct flowi fl = { | 	struct flowi6 fl6 = { | ||||||
| 		.flowi_iif = skb->dev->ifindex, | 		.flowi6_iif = skb->dev->ifindex, | ||||||
| 		.fl6_dst = iph->daddr, | 		.daddr = iph->daddr, | ||||||
| 		.fl6_src = iph->saddr, | 		.saddr = iph->saddr, | ||||||
| 		.fl6_flowlabel = (* (__be32 *) iph)&IPV6_FLOWINFO_MASK, | 		.flowlabel = (* (__be32 *) iph)&IPV6_FLOWINFO_MASK, | ||||||
| 		.flowi_mark = skb->mark, | 		.flowi6_mark = skb->mark, | ||||||
| 		.flowi_proto = iph->nexthdr, | 		.flowi6_proto = iph->nexthdr, | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| 	if (rt6_need_strict(&iph->daddr) && skb->dev->type != ARPHRD_PIMREG) | 	if (rt6_need_strict(&iph->daddr) && skb->dev->type != ARPHRD_PIMREG) | ||||||
| 		flags |= RT6_LOOKUP_F_IFACE; | 		flags |= RT6_LOOKUP_F_IFACE; | ||||||
| 
 | 
 | ||||||
| 	skb_dst_set(skb, fib6_rule_lookup(net, &fl, flags, ip6_pol_route_input)); | 	skb_dst_set(skb, fib6_rule_lookup(net, &fl6, flags, ip6_pol_route_input)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static struct rt6_info *ip6_pol_route_output(struct net *net, struct fib6_table *table, | static struct rt6_info *ip6_pol_route_output(struct net *net, struct fib6_table *table, | ||||||
| 					     struct flowi *fl, int flags) | 					     struct flowi6 *fl6, int flags) | ||||||
| { | { | ||||||
| 	return ip6_pol_route(net, table, fl->flowi_oif, fl, flags); | 	return ip6_pol_route(net, table, fl6->flowi6_oif, fl6, flags); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| struct dst_entry * ip6_route_output(struct net *net, struct sock *sk, | struct dst_entry * ip6_route_output(struct net *net, struct sock *sk, | ||||||
| 				    struct flowi *fl) | 				    struct flowi6 *fl6) | ||||||
| { | { | ||||||
| 	int flags = 0; | 	int flags = 0; | ||||||
| 
 | 
 | ||||||
| 	if ((sk && sk->sk_bound_dev_if) || rt6_need_strict(&fl->fl6_dst)) | 	if ((sk && sk->sk_bound_dev_if) || rt6_need_strict(&fl6->daddr)) | ||||||
| 		flags |= RT6_LOOKUP_F_IFACE; | 		flags |= RT6_LOOKUP_F_IFACE; | ||||||
| 
 | 
 | ||||||
| 	if (!ipv6_addr_any(&fl->fl6_src)) | 	if (!ipv6_addr_any(&fl6->saddr)) | ||||||
| 		flags |= RT6_LOOKUP_F_HAS_SADDR; | 		flags |= RT6_LOOKUP_F_HAS_SADDR; | ||||||
| 	else if (sk) | 	else if (sk) | ||||||
| 		flags |= rt6_srcprefs2flags(inet6_sk(sk)->srcprefs); | 		flags |= rt6_srcprefs2flags(inet6_sk(sk)->srcprefs); | ||||||
| 
 | 
 | ||||||
| 	return fib6_rule_lookup(net, fl, flags, ip6_pol_route_output); | 	return fib6_rule_lookup(net, fl6, flags, ip6_pol_route_output); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| EXPORT_SYMBOL(ip6_route_output); | EXPORT_SYMBOL(ip6_route_output); | ||||||
|  | @ -1444,16 +1444,16 @@ static int ip6_route_del(struct fib6_config *cfg) | ||||||
|  *	Handle redirects |  *	Handle redirects | ||||||
|  */ |  */ | ||||||
| struct ip6rd_flowi { | struct ip6rd_flowi { | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 	struct in6_addr gateway; | 	struct in6_addr gateway; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| static struct rt6_info *__ip6_route_redirect(struct net *net, | static struct rt6_info *__ip6_route_redirect(struct net *net, | ||||||
| 					     struct fib6_table *table, | 					     struct fib6_table *table, | ||||||
| 					     struct flowi *fl, | 					     struct flowi6 *fl6, | ||||||
| 					     int flags) | 					     int flags) | ||||||
| { | { | ||||||
| 	struct ip6rd_flowi *rdfl = (struct ip6rd_flowi *)fl; | 	struct ip6rd_flowi *rdfl = (struct ip6rd_flowi *)fl6; | ||||||
| 	struct rt6_info *rt; | 	struct rt6_info *rt; | ||||||
| 	struct fib6_node *fn; | 	struct fib6_node *fn; | ||||||
| 
 | 
 | ||||||
|  | @ -1469,7 +1469,7 @@ static struct rt6_info *__ip6_route_redirect(struct net *net, | ||||||
| 	 */ | 	 */ | ||||||
| 
 | 
 | ||||||
| 	read_lock_bh(&table->tb6_lock); | 	read_lock_bh(&table->tb6_lock); | ||||||
| 	fn = fib6_lookup(&table->tb6_root, &fl->fl6_dst, &fl->fl6_src); | 	fn = fib6_lookup(&table->tb6_root, &fl6->daddr, &fl6->saddr); | ||||||
| restart: | restart: | ||||||
| 	for (rt = fn->leaf; rt; rt = rt->dst.rt6_next) { | 	for (rt = fn->leaf; rt; rt = rt->dst.rt6_next) { | ||||||
| 		/*
 | 		/*
 | ||||||
|  | @ -1484,7 +1484,7 @@ restart: | ||||||
| 			continue; | 			continue; | ||||||
| 		if (!(rt->rt6i_flags & RTF_GATEWAY)) | 		if (!(rt->rt6i_flags & RTF_GATEWAY)) | ||||||
| 			continue; | 			continue; | ||||||
| 		if (fl->flowi_oif != rt->rt6i_dev->ifindex) | 		if (fl6->flowi6_oif != rt->rt6i_dev->ifindex) | ||||||
| 			continue; | 			continue; | ||||||
| 		if (!ipv6_addr_equal(&rdfl->gateway, &rt->rt6i_gateway)) | 		if (!ipv6_addr_equal(&rdfl->gateway, &rt->rt6i_gateway)) | ||||||
| 			continue; | 			continue; | ||||||
|  | @ -1493,7 +1493,7 @@ restart: | ||||||
| 
 | 
 | ||||||
| 	if (!rt) | 	if (!rt) | ||||||
| 		rt = net->ipv6.ip6_null_entry; | 		rt = net->ipv6.ip6_null_entry; | ||||||
| 	BACKTRACK(net, &fl->fl6_src); | 	BACKTRACK(net, &fl6->saddr); | ||||||
| out: | out: | ||||||
| 	dst_hold(&rt->dst); | 	dst_hold(&rt->dst); | ||||||
| 
 | 
 | ||||||
|  | @ -1510,10 +1510,10 @@ static struct rt6_info *ip6_route_redirect(struct in6_addr *dest, | ||||||
| 	int flags = RT6_LOOKUP_F_HAS_SADDR; | 	int flags = RT6_LOOKUP_F_HAS_SADDR; | ||||||
| 	struct net *net = dev_net(dev); | 	struct net *net = dev_net(dev); | ||||||
| 	struct ip6rd_flowi rdfl = { | 	struct ip6rd_flowi rdfl = { | ||||||
| 		.fl = { | 		.fl6 = { | ||||||
| 			.flowi_oif = dev->ifindex, | 			.flowi6_oif = dev->ifindex, | ||||||
| 			.fl6_dst = *dest, | 			.daddr = *dest, | ||||||
| 			.fl6_src = *src, | 			.saddr = *src, | ||||||
| 		}, | 		}, | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
|  | @ -1522,7 +1522,7 @@ static struct rt6_info *ip6_route_redirect(struct in6_addr *dest, | ||||||
| 	if (rt6_need_strict(dest)) | 	if (rt6_need_strict(dest)) | ||||||
| 		flags |= RT6_LOOKUP_F_IFACE; | 		flags |= RT6_LOOKUP_F_IFACE; | ||||||
| 
 | 
 | ||||||
| 	return (struct rt6_info *)fib6_rule_lookup(net, (struct flowi *)&rdfl, | 	return (struct rt6_info *)fib6_rule_lookup(net, &rdfl.fl6, | ||||||
| 						   flags, __ip6_route_redirect); | 						   flags, __ip6_route_redirect); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -2385,7 +2385,7 @@ static int inet6_rtm_getroute(struct sk_buff *in_skb, struct nlmsghdr* nlh, void | ||||||
| 	struct rt6_info *rt; | 	struct rt6_info *rt; | ||||||
| 	struct sk_buff *skb; | 	struct sk_buff *skb; | ||||||
| 	struct rtmsg *rtm; | 	struct rtmsg *rtm; | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 	int err, iif = 0; | 	int err, iif = 0; | ||||||
| 
 | 
 | ||||||
| 	err = nlmsg_parse(nlh, sizeof(*rtm), tb, RTA_MAX, rtm_ipv6_policy); | 	err = nlmsg_parse(nlh, sizeof(*rtm), tb, RTA_MAX, rtm_ipv6_policy); | ||||||
|  | @ -2393,27 +2393,27 @@ static int inet6_rtm_getroute(struct sk_buff *in_skb, struct nlmsghdr* nlh, void | ||||||
| 		goto errout; | 		goto errout; | ||||||
| 
 | 
 | ||||||
| 	err = -EINVAL; | 	err = -EINVAL; | ||||||
| 	memset(&fl, 0, sizeof(fl)); | 	memset(&fl6, 0, sizeof(fl6)); | ||||||
| 
 | 
 | ||||||
| 	if (tb[RTA_SRC]) { | 	if (tb[RTA_SRC]) { | ||||||
| 		if (nla_len(tb[RTA_SRC]) < sizeof(struct in6_addr)) | 		if (nla_len(tb[RTA_SRC]) < sizeof(struct in6_addr)) | ||||||
| 			goto errout; | 			goto errout; | ||||||
| 
 | 
 | ||||||
| 		ipv6_addr_copy(&fl.fl6_src, nla_data(tb[RTA_SRC])); | 		ipv6_addr_copy(&fl6.saddr, nla_data(tb[RTA_SRC])); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (tb[RTA_DST]) { | 	if (tb[RTA_DST]) { | ||||||
| 		if (nla_len(tb[RTA_DST]) < sizeof(struct in6_addr)) | 		if (nla_len(tb[RTA_DST]) < sizeof(struct in6_addr)) | ||||||
| 			goto errout; | 			goto errout; | ||||||
| 
 | 
 | ||||||
| 		ipv6_addr_copy(&fl.fl6_dst, nla_data(tb[RTA_DST])); | 		ipv6_addr_copy(&fl6.daddr, nla_data(tb[RTA_DST])); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (tb[RTA_IIF]) | 	if (tb[RTA_IIF]) | ||||||
| 		iif = nla_get_u32(tb[RTA_IIF]); | 		iif = nla_get_u32(tb[RTA_IIF]); | ||||||
| 
 | 
 | ||||||
| 	if (tb[RTA_OIF]) | 	if (tb[RTA_OIF]) | ||||||
| 		fl.flowi_oif = nla_get_u32(tb[RTA_OIF]); | 		fl6.flowi6_oif = nla_get_u32(tb[RTA_OIF]); | ||||||
| 
 | 
 | ||||||
| 	if (iif) { | 	if (iif) { | ||||||
| 		struct net_device *dev; | 		struct net_device *dev; | ||||||
|  | @ -2436,10 +2436,10 @@ static int inet6_rtm_getroute(struct sk_buff *in_skb, struct nlmsghdr* nlh, void | ||||||
| 	skb_reset_mac_header(skb); | 	skb_reset_mac_header(skb); | ||||||
| 	skb_reserve(skb, MAX_HEADER + sizeof(struct ipv6hdr)); | 	skb_reserve(skb, MAX_HEADER + sizeof(struct ipv6hdr)); | ||||||
| 
 | 
 | ||||||
| 	rt = (struct rt6_info*) ip6_route_output(net, NULL, &fl); | 	rt = (struct rt6_info*) ip6_route_output(net, NULL, &fl6); | ||||||
| 	skb_dst_set(skb, &rt->dst); | 	skb_dst_set(skb, &rt->dst); | ||||||
| 
 | 
 | ||||||
| 	err = rt6_fill_node(net, skb, rt, &fl.fl6_dst, &fl.fl6_src, iif, | 	err = rt6_fill_node(net, skb, rt, &fl6.daddr, &fl6.saddr, iif, | ||||||
| 			    RTM_NEWROUTE, NETLINK_CB(in_skb).pid, | 			    RTM_NEWROUTE, NETLINK_CB(in_skb).pid, | ||||||
| 			    nlh->nlmsg_seq, 0, 0, 0); | 			    nlh->nlmsg_seq, 0, 0, 0); | ||||||
| 	if (err < 0) { | 	if (err < 0) { | ||||||
|  |  | ||||||
|  | @ -232,19 +232,19 @@ struct sock *cookie_v6_check(struct sock *sk, struct sk_buff *skb) | ||||||
| 	 */ | 	 */ | ||||||
| 	{ | 	{ | ||||||
| 		struct in6_addr *final_p, final; | 		struct in6_addr *final_p, final; | ||||||
| 		struct flowi fl; | 		struct flowi6 fl6; | ||||||
| 		memset(&fl, 0, sizeof(fl)); | 		memset(&fl6, 0, sizeof(fl6)); | ||||||
| 		fl.flowi_proto = IPPROTO_TCP; | 		fl6.flowi6_proto = IPPROTO_TCP; | ||||||
| 		ipv6_addr_copy(&fl.fl6_dst, &ireq6->rmt_addr); | 		ipv6_addr_copy(&fl6.daddr, &ireq6->rmt_addr); | ||||||
| 		final_p = fl6_update_dst(&fl, np->opt, &final); | 		final_p = fl6_update_dst(&fl6, np->opt, &final); | ||||||
| 		ipv6_addr_copy(&fl.fl6_src, &ireq6->loc_addr); | 		ipv6_addr_copy(&fl6.saddr, &ireq6->loc_addr); | ||||||
| 		fl.flowi_oif = sk->sk_bound_dev_if; | 		fl6.flowi6_oif = sk->sk_bound_dev_if; | ||||||
| 		fl.flowi_mark = sk->sk_mark; | 		fl6.flowi6_mark = sk->sk_mark; | ||||||
| 		fl.fl6_dport = inet_rsk(req)->rmt_port; | 		fl6.uli.ports.dport = inet_rsk(req)->rmt_port; | ||||||
| 		fl.fl6_sport = inet_sk(sk)->inet_sport; | 		fl6.uli.ports.sport = inet_sk(sk)->inet_sport; | ||||||
| 		security_req_classify_flow(req, &fl); | 		security_req_classify_flow(req, flowi6_to_flowi(&fl6)); | ||||||
| 
 | 
 | ||||||
| 		dst = ip6_dst_lookup_flow(sk, &fl, final_p, false); | 		dst = ip6_dst_lookup_flow(sk, &fl6, final_p, false); | ||||||
| 		if (IS_ERR(dst)) | 		if (IS_ERR(dst)) | ||||||
| 			goto out_free; | 			goto out_free; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | @ -131,7 +131,7 @@ static int tcp_v6_connect(struct sock *sk, struct sockaddr *uaddr, | ||||||
| 	struct tcp_sock *tp = tcp_sk(sk); | 	struct tcp_sock *tp = tcp_sk(sk); | ||||||
| 	struct in6_addr *saddr = NULL, *final_p, final; | 	struct in6_addr *saddr = NULL, *final_p, final; | ||||||
| 	struct rt6_info *rt; | 	struct rt6_info *rt; | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 	struct dst_entry *dst; | 	struct dst_entry *dst; | ||||||
| 	int addr_type; | 	int addr_type; | ||||||
| 	int err; | 	int err; | ||||||
|  | @ -142,14 +142,14 @@ static int tcp_v6_connect(struct sock *sk, struct sockaddr *uaddr, | ||||||
| 	if (usin->sin6_family != AF_INET6) | 	if (usin->sin6_family != AF_INET6) | ||||||
| 		return -EAFNOSUPPORT; | 		return -EAFNOSUPPORT; | ||||||
| 
 | 
 | ||||||
| 	memset(&fl, 0, sizeof(fl)); | 	memset(&fl6, 0, sizeof(fl6)); | ||||||
| 
 | 
 | ||||||
| 	if (np->sndflow) { | 	if (np->sndflow) { | ||||||
| 		fl.fl6_flowlabel = usin->sin6_flowinfo&IPV6_FLOWINFO_MASK; | 		fl6.flowlabel = usin->sin6_flowinfo&IPV6_FLOWINFO_MASK; | ||||||
| 		IP6_ECN_flow_init(fl.fl6_flowlabel); | 		IP6_ECN_flow_init(fl6.flowlabel); | ||||||
| 		if (fl.fl6_flowlabel&IPV6_FLOWLABEL_MASK) { | 		if (fl6.flowlabel&IPV6_FLOWLABEL_MASK) { | ||||||
| 			struct ip6_flowlabel *flowlabel; | 			struct ip6_flowlabel *flowlabel; | ||||||
| 			flowlabel = fl6_sock_lookup(sk, fl.fl6_flowlabel); | 			flowlabel = fl6_sock_lookup(sk, fl6.flowlabel); | ||||||
| 			if (flowlabel == NULL) | 			if (flowlabel == NULL) | ||||||
| 				return -EINVAL; | 				return -EINVAL; | ||||||
| 			ipv6_addr_copy(&usin->sin6_addr, &flowlabel->dst); | 			ipv6_addr_copy(&usin->sin6_addr, &flowlabel->dst); | ||||||
|  | @ -195,7 +195,7 @@ static int tcp_v6_connect(struct sock *sk, struct sockaddr *uaddr, | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	ipv6_addr_copy(&np->daddr, &usin->sin6_addr); | 	ipv6_addr_copy(&np->daddr, &usin->sin6_addr); | ||||||
| 	np->flow_label = fl.fl6_flowlabel; | 	np->flow_label = fl6.flowlabel; | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 *	TCP over IPv4 | 	 *	TCP over IPv4 | ||||||
|  | @ -242,27 +242,27 @@ static int tcp_v6_connect(struct sock *sk, struct sockaddr *uaddr, | ||||||
| 	if (!ipv6_addr_any(&np->rcv_saddr)) | 	if (!ipv6_addr_any(&np->rcv_saddr)) | ||||||
| 		saddr = &np->rcv_saddr; | 		saddr = &np->rcv_saddr; | ||||||
| 
 | 
 | ||||||
| 	fl.flowi_proto = IPPROTO_TCP; | 	fl6.flowi6_proto = IPPROTO_TCP; | ||||||
| 	ipv6_addr_copy(&fl.fl6_dst, &np->daddr); | 	ipv6_addr_copy(&fl6.daddr, &np->daddr); | ||||||
| 	ipv6_addr_copy(&fl.fl6_src, | 	ipv6_addr_copy(&fl6.saddr, | ||||||
| 		       (saddr ? saddr : &np->saddr)); | 		       (saddr ? saddr : &np->saddr)); | ||||||
| 	fl.flowi_oif = sk->sk_bound_dev_if; | 	fl6.flowi6_oif = sk->sk_bound_dev_if; | ||||||
| 	fl.flowi_mark = sk->sk_mark; | 	fl6.flowi6_mark = sk->sk_mark; | ||||||
| 	fl.fl6_dport = usin->sin6_port; | 	fl6.uli.ports.dport = usin->sin6_port; | ||||||
| 	fl.fl6_sport = inet->inet_sport; | 	fl6.uli.ports.sport = inet->inet_sport; | ||||||
| 
 | 
 | ||||||
| 	final_p = fl6_update_dst(&fl, np->opt, &final); | 	final_p = fl6_update_dst(&fl6, np->opt, &final); | ||||||
| 
 | 
 | ||||||
| 	security_sk_classify_flow(sk, &fl); | 	security_sk_classify_flow(sk, flowi6_to_flowi(&fl6)); | ||||||
| 
 | 
 | ||||||
| 	dst = ip6_dst_lookup_flow(sk, &fl, final_p, true); | 	dst = ip6_dst_lookup_flow(sk, &fl6, final_p, true); | ||||||
| 	if (IS_ERR(dst)) { | 	if (IS_ERR(dst)) { | ||||||
| 		err = PTR_ERR(dst); | 		err = PTR_ERR(dst); | ||||||
| 		goto failure; | 		goto failure; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (saddr == NULL) { | 	if (saddr == NULL) { | ||||||
| 		saddr = &fl.fl6_src; | 		saddr = &fl6.saddr; | ||||||
| 		ipv6_addr_copy(&np->rcv_saddr, saddr); | 		ipv6_addr_copy(&np->rcv_saddr, saddr); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | @ -389,23 +389,23 @@ static void tcp_v6_err(struct sk_buff *skb, struct inet6_skb_parm *opt, | ||||||
| 
 | 
 | ||||||
| 		if (dst == NULL) { | 		if (dst == NULL) { | ||||||
| 			struct inet_sock *inet = inet_sk(sk); | 			struct inet_sock *inet = inet_sk(sk); | ||||||
| 			struct flowi fl; | 			struct flowi6 fl6; | ||||||
| 
 | 
 | ||||||
| 			/* BUGGG_FUTURE: Again, it is not clear how
 | 			/* BUGGG_FUTURE: Again, it is not clear how
 | ||||||
| 			   to handle rthdr case. Ignore this complexity | 			   to handle rthdr case. Ignore this complexity | ||||||
| 			   for now. | 			   for now. | ||||||
| 			 */ | 			 */ | ||||||
| 			memset(&fl, 0, sizeof(fl)); | 			memset(&fl6, 0, sizeof(fl6)); | ||||||
| 			fl.flowi_proto = IPPROTO_TCP; | 			fl6.flowi6_proto = IPPROTO_TCP; | ||||||
| 			ipv6_addr_copy(&fl.fl6_dst, &np->daddr); | 			ipv6_addr_copy(&fl6.daddr, &np->daddr); | ||||||
| 			ipv6_addr_copy(&fl.fl6_src, &np->saddr); | 			ipv6_addr_copy(&fl6.saddr, &np->saddr); | ||||||
| 			fl.flowi_oif = sk->sk_bound_dev_if; | 			fl6.flowi6_oif = sk->sk_bound_dev_if; | ||||||
| 			fl.flowi_mark = sk->sk_mark; | 			fl6.flowi6_mark = sk->sk_mark; | ||||||
| 			fl.fl6_dport = inet->inet_dport; | 			fl6.uli.ports.dport = inet->inet_dport; | ||||||
| 			fl.fl6_sport = inet->inet_sport; | 			fl6.uli.ports.sport = inet->inet_sport; | ||||||
| 			security_skb_classify_flow(skb, &fl); | 			security_skb_classify_flow(skb, flowi6_to_flowi(&fl6)); | ||||||
| 
 | 
 | ||||||
| 			dst = ip6_dst_lookup_flow(sk, &fl, NULL, false); | 			dst = ip6_dst_lookup_flow(sk, &fl6, NULL, false); | ||||||
| 			if (IS_ERR(dst)) { | 			if (IS_ERR(dst)) { | ||||||
| 				sk->sk_err_soft = -PTR_ERR(dst); | 				sk->sk_err_soft = -PTR_ERR(dst); | ||||||
| 				goto out; | 				goto out; | ||||||
|  | @ -482,25 +482,25 @@ static int tcp_v6_send_synack(struct sock *sk, struct request_sock *req, | ||||||
| 	struct sk_buff * skb; | 	struct sk_buff * skb; | ||||||
| 	struct ipv6_txoptions *opt = NULL; | 	struct ipv6_txoptions *opt = NULL; | ||||||
| 	struct in6_addr * final_p, final; | 	struct in6_addr * final_p, final; | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 	struct dst_entry *dst; | 	struct dst_entry *dst; | ||||||
| 	int err; | 	int err; | ||||||
| 
 | 
 | ||||||
| 	memset(&fl, 0, sizeof(fl)); | 	memset(&fl6, 0, sizeof(fl6)); | ||||||
| 	fl.flowi_proto = IPPROTO_TCP; | 	fl6.flowi6_proto = IPPROTO_TCP; | ||||||
| 	ipv6_addr_copy(&fl.fl6_dst, &treq->rmt_addr); | 	ipv6_addr_copy(&fl6.daddr, &treq->rmt_addr); | ||||||
| 	ipv6_addr_copy(&fl.fl6_src, &treq->loc_addr); | 	ipv6_addr_copy(&fl6.saddr, &treq->loc_addr); | ||||||
| 	fl.fl6_flowlabel = 0; | 	fl6.flowlabel = 0; | ||||||
| 	fl.flowi_oif = treq->iif; | 	fl6.flowi6_oif = treq->iif; | ||||||
| 	fl.flowi_mark = sk->sk_mark; | 	fl6.flowi6_mark = sk->sk_mark; | ||||||
| 	fl.fl6_dport = inet_rsk(req)->rmt_port; | 	fl6.uli.ports.dport = inet_rsk(req)->rmt_port; | ||||||
| 	fl.fl6_sport = inet_rsk(req)->loc_port; | 	fl6.uli.ports.sport = inet_rsk(req)->loc_port; | ||||||
| 	security_req_classify_flow(req, &fl); | 	security_req_classify_flow(req, flowi6_to_flowi(&fl6)); | ||||||
| 
 | 
 | ||||||
| 	opt = np->opt; | 	opt = np->opt; | ||||||
| 	final_p = fl6_update_dst(&fl, opt, &final); | 	final_p = fl6_update_dst(&fl6, opt, &final); | ||||||
| 
 | 
 | ||||||
| 	dst = ip6_dst_lookup_flow(sk, &fl, final_p, false); | 	dst = ip6_dst_lookup_flow(sk, &fl6, final_p, false); | ||||||
| 	if (IS_ERR(dst)) { | 	if (IS_ERR(dst)) { | ||||||
| 		err = PTR_ERR(dst); | 		err = PTR_ERR(dst); | ||||||
| 		goto done; | 		goto done; | ||||||
|  | @ -510,8 +510,8 @@ static int tcp_v6_send_synack(struct sock *sk, struct request_sock *req, | ||||||
| 	if (skb) { | 	if (skb) { | ||||||
| 		__tcp_v6_send_check(skb, &treq->loc_addr, &treq->rmt_addr); | 		__tcp_v6_send_check(skb, &treq->loc_addr, &treq->rmt_addr); | ||||||
| 
 | 
 | ||||||
| 		ipv6_addr_copy(&fl.fl6_dst, &treq->rmt_addr); | 		ipv6_addr_copy(&fl6.daddr, &treq->rmt_addr); | ||||||
| 		err = ip6_xmit(sk, skb, &fl, opt); | 		err = ip6_xmit(sk, skb, &fl6, opt); | ||||||
| 		err = net_xmit_eval(err); | 		err = net_xmit_eval(err); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | @ -992,7 +992,7 @@ static void tcp_v6_send_response(struct sk_buff *skb, u32 seq, u32 ack, u32 win, | ||||||
| { | { | ||||||
| 	struct tcphdr *th = tcp_hdr(skb), *t1; | 	struct tcphdr *th = tcp_hdr(skb), *t1; | ||||||
| 	struct sk_buff *buff; | 	struct sk_buff *buff; | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 	struct net *net = dev_net(skb_dst(skb)->dev); | 	struct net *net = dev_net(skb_dst(skb)->dev); | ||||||
| 	struct sock *ctl_sk = net->ipv6.tcp_sk; | 	struct sock *ctl_sk = net->ipv6.tcp_sk; | ||||||
| 	unsigned int tot_len = sizeof(struct tcphdr); | 	unsigned int tot_len = sizeof(struct tcphdr); | ||||||
|  | @ -1046,29 +1046,29 @@ static void tcp_v6_send_response(struct sk_buff *skb, u32 seq, u32 ack, u32 win, | ||||||
| 	} | 	} | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
| 	memset(&fl, 0, sizeof(fl)); | 	memset(&fl6, 0, sizeof(fl6)); | ||||||
| 	ipv6_addr_copy(&fl.fl6_dst, &ipv6_hdr(skb)->saddr); | 	ipv6_addr_copy(&fl6.daddr, &ipv6_hdr(skb)->saddr); | ||||||
| 	ipv6_addr_copy(&fl.fl6_src, &ipv6_hdr(skb)->daddr); | 	ipv6_addr_copy(&fl6.saddr, &ipv6_hdr(skb)->daddr); | ||||||
| 
 | 
 | ||||||
| 	buff->ip_summed = CHECKSUM_PARTIAL; | 	buff->ip_summed = CHECKSUM_PARTIAL; | ||||||
| 	buff->csum = 0; | 	buff->csum = 0; | ||||||
| 
 | 
 | ||||||
| 	__tcp_v6_send_check(buff, &fl.fl6_src, &fl.fl6_dst); | 	__tcp_v6_send_check(buff, &fl6.saddr, &fl6.daddr); | ||||||
| 
 | 
 | ||||||
| 	fl.flowi_proto = IPPROTO_TCP; | 	fl6.flowi6_proto = IPPROTO_TCP; | ||||||
| 	fl.flowi_oif = inet6_iif(skb); | 	fl6.flowi6_oif = inet6_iif(skb); | ||||||
| 	fl.fl6_dport = t1->dest; | 	fl6.uli.ports.dport = t1->dest; | ||||||
| 	fl.fl6_sport = t1->source; | 	fl6.uli.ports.sport = t1->source; | ||||||
| 	security_skb_classify_flow(skb, &fl); | 	security_skb_classify_flow(skb, flowi6_to_flowi(&fl6)); | ||||||
| 
 | 
 | ||||||
| 	/* Pass a socket to ip6_dst_lookup either it is for RST
 | 	/* Pass a socket to ip6_dst_lookup either it is for RST
 | ||||||
| 	 * Underlying function will use this to retrieve the network | 	 * Underlying function will use this to retrieve the network | ||||||
| 	 * namespace | 	 * namespace | ||||||
| 	 */ | 	 */ | ||||||
| 	dst = ip6_dst_lookup_flow(ctl_sk, &fl, NULL, false); | 	dst = ip6_dst_lookup_flow(ctl_sk, &fl6, NULL, false); | ||||||
| 	if (!IS_ERR(dst)) { | 	if (!IS_ERR(dst)) { | ||||||
| 		skb_dst_set(buff, dst); | 		skb_dst_set(buff, dst); | ||||||
| 		ip6_xmit(ctl_sk, buff, &fl, NULL); | 		ip6_xmit(ctl_sk, buff, &fl6, NULL); | ||||||
| 		TCP_INC_STATS_BH(net, TCP_MIB_OUTSEGS); | 		TCP_INC_STATS_BH(net, TCP_MIB_OUTSEGS); | ||||||
| 		if (rst) | 		if (rst) | ||||||
| 			TCP_INC_STATS_BH(net, TCP_MIB_OUTRSTS); | 			TCP_INC_STATS_BH(net, TCP_MIB_OUTRSTS); | ||||||
|  |  | ||||||
|  | @ -886,7 +886,7 @@ static int udp_v6_push_pending_frames(struct sock *sk) | ||||||
| 	struct udphdr *uh; | 	struct udphdr *uh; | ||||||
| 	struct udp_sock  *up = udp_sk(sk); | 	struct udp_sock  *up = udp_sk(sk); | ||||||
| 	struct inet_sock *inet = inet_sk(sk); | 	struct inet_sock *inet = inet_sk(sk); | ||||||
| 	struct flowi *fl = &inet->cork.fl; | 	struct flowi6 *fl6 = &inet->cork.fl.u.ip6; | ||||||
| 	int err = 0; | 	int err = 0; | ||||||
| 	int is_udplite = IS_UDPLITE(sk); | 	int is_udplite = IS_UDPLITE(sk); | ||||||
| 	__wsum csum = 0; | 	__wsum csum = 0; | ||||||
|  | @ -899,23 +899,23 @@ static int udp_v6_push_pending_frames(struct sock *sk) | ||||||
| 	 * Create a UDP header | 	 * Create a UDP header | ||||||
| 	 */ | 	 */ | ||||||
| 	uh = udp_hdr(skb); | 	uh = udp_hdr(skb); | ||||||
| 	uh->source = fl->fl6_sport; | 	uh->source = fl6->uli.ports.sport; | ||||||
| 	uh->dest = fl->fl6_dport; | 	uh->dest = fl6->uli.ports.dport; | ||||||
| 	uh->len = htons(up->len); | 	uh->len = htons(up->len); | ||||||
| 	uh->check = 0; | 	uh->check = 0; | ||||||
| 
 | 
 | ||||||
| 	if (is_udplite) | 	if (is_udplite) | ||||||
| 		csum = udplite_csum_outgoing(sk, skb); | 		csum = udplite_csum_outgoing(sk, skb); | ||||||
| 	else if (skb->ip_summed == CHECKSUM_PARTIAL) { /* UDP hardware csum */ | 	else if (skb->ip_summed == CHECKSUM_PARTIAL) { /* UDP hardware csum */ | ||||||
| 		udp6_hwcsum_outgoing(sk, skb, &fl->fl6_src, &fl->fl6_dst, | 		udp6_hwcsum_outgoing(sk, skb, &fl6->saddr, &fl6->daddr, | ||||||
| 				     up->len); | 				     up->len); | ||||||
| 		goto send; | 		goto send; | ||||||
| 	} else | 	} else | ||||||
| 		csum = udp_csum_outgoing(sk, skb); | 		csum = udp_csum_outgoing(sk, skb); | ||||||
| 
 | 
 | ||||||
| 	/* add protocol-dependent pseudo-header */ | 	/* add protocol-dependent pseudo-header */ | ||||||
| 	uh->check = csum_ipv6_magic(&fl->fl6_src, &fl->fl6_dst, | 	uh->check = csum_ipv6_magic(&fl6->saddr, &fl6->daddr, | ||||||
| 				    up->len, fl->flowi_proto, csum); | 				    up->len, fl6->flowi6_proto, csum); | ||||||
| 	if (uh->check == 0) | 	if (uh->check == 0) | ||||||
| 		uh->check = CSUM_MANGLED_0; | 		uh->check = CSUM_MANGLED_0; | ||||||
| 
 | 
 | ||||||
|  | @ -947,7 +947,7 @@ int udpv6_sendmsg(struct kiocb *iocb, struct sock *sk, | ||||||
| 	struct in6_addr *daddr, *final_p, final; | 	struct in6_addr *daddr, *final_p, final; | ||||||
| 	struct ipv6_txoptions *opt = NULL; | 	struct ipv6_txoptions *opt = NULL; | ||||||
| 	struct ip6_flowlabel *flowlabel = NULL; | 	struct ip6_flowlabel *flowlabel = NULL; | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 	struct dst_entry *dst; | 	struct dst_entry *dst; | ||||||
| 	int addr_len = msg->msg_namelen; | 	int addr_len = msg->msg_namelen; | ||||||
| 	int ulen = len; | 	int ulen = len; | ||||||
|  | @ -1030,19 +1030,19 @@ do_udp_sendmsg: | ||||||
| 	} | 	} | ||||||
| 	ulen += sizeof(struct udphdr); | 	ulen += sizeof(struct udphdr); | ||||||
| 
 | 
 | ||||||
| 	memset(&fl, 0, sizeof(fl)); | 	memset(&fl6, 0, sizeof(fl6)); | ||||||
| 
 | 
 | ||||||
| 	if (sin6) { | 	if (sin6) { | ||||||
| 		if (sin6->sin6_port == 0) | 		if (sin6->sin6_port == 0) | ||||||
| 			return -EINVAL; | 			return -EINVAL; | ||||||
| 
 | 
 | ||||||
| 		fl.fl6_dport = sin6->sin6_port; | 		fl6.uli.ports.dport = sin6->sin6_port; | ||||||
| 		daddr = &sin6->sin6_addr; | 		daddr = &sin6->sin6_addr; | ||||||
| 
 | 
 | ||||||
| 		if (np->sndflow) { | 		if (np->sndflow) { | ||||||
| 			fl.fl6_flowlabel = sin6->sin6_flowinfo&IPV6_FLOWINFO_MASK; | 			fl6.flowlabel = sin6->sin6_flowinfo&IPV6_FLOWINFO_MASK; | ||||||
| 			if (fl.fl6_flowlabel&IPV6_FLOWLABEL_MASK) { | 			if (fl6.flowlabel&IPV6_FLOWLABEL_MASK) { | ||||||
| 				flowlabel = fl6_sock_lookup(sk, fl.fl6_flowlabel); | 				flowlabel = fl6_sock_lookup(sk, fl6.flowlabel); | ||||||
| 				if (flowlabel == NULL) | 				if (flowlabel == NULL) | ||||||
| 					return -EINVAL; | 					return -EINVAL; | ||||||
| 				daddr = &flowlabel->dst; | 				daddr = &flowlabel->dst; | ||||||
|  | @ -1060,38 +1060,38 @@ do_udp_sendmsg: | ||||||
| 		if (addr_len >= sizeof(struct sockaddr_in6) && | 		if (addr_len >= sizeof(struct sockaddr_in6) && | ||||||
| 		    sin6->sin6_scope_id && | 		    sin6->sin6_scope_id && | ||||||
| 		    ipv6_addr_type(daddr)&IPV6_ADDR_LINKLOCAL) | 		    ipv6_addr_type(daddr)&IPV6_ADDR_LINKLOCAL) | ||||||
| 			fl.flowi_oif = sin6->sin6_scope_id; | 			fl6.flowi6_oif = sin6->sin6_scope_id; | ||||||
| 	} else { | 	} else { | ||||||
| 		if (sk->sk_state != TCP_ESTABLISHED) | 		if (sk->sk_state != TCP_ESTABLISHED) | ||||||
| 			return -EDESTADDRREQ; | 			return -EDESTADDRREQ; | ||||||
| 
 | 
 | ||||||
| 		fl.fl6_dport = inet->inet_dport; | 		fl6.uli.ports.dport = inet->inet_dport; | ||||||
| 		daddr = &np->daddr; | 		daddr = &np->daddr; | ||||||
| 		fl.fl6_flowlabel = np->flow_label; | 		fl6.flowlabel = np->flow_label; | ||||||
| 		connected = 1; | 		connected = 1; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (!fl.flowi_oif) | 	if (!fl6.flowi6_oif) | ||||||
| 		fl.flowi_oif = sk->sk_bound_dev_if; | 		fl6.flowi6_oif = sk->sk_bound_dev_if; | ||||||
| 
 | 
 | ||||||
| 	if (!fl.flowi_oif) | 	if (!fl6.flowi6_oif) | ||||||
| 		fl.flowi_oif = np->sticky_pktinfo.ipi6_ifindex; | 		fl6.flowi6_oif = np->sticky_pktinfo.ipi6_ifindex; | ||||||
| 
 | 
 | ||||||
| 	fl.flowi_mark = sk->sk_mark; | 	fl6.flowi6_mark = sk->sk_mark; | ||||||
| 
 | 
 | ||||||
| 	if (msg->msg_controllen) { | 	if (msg->msg_controllen) { | ||||||
| 		opt = &opt_space; | 		opt = &opt_space; | ||||||
| 		memset(opt, 0, sizeof(struct ipv6_txoptions)); | 		memset(opt, 0, sizeof(struct ipv6_txoptions)); | ||||||
| 		opt->tot_len = sizeof(*opt); | 		opt->tot_len = sizeof(*opt); | ||||||
| 
 | 
 | ||||||
| 		err = datagram_send_ctl(sock_net(sk), msg, &fl, opt, &hlimit, | 		err = datagram_send_ctl(sock_net(sk), msg, &fl6, opt, &hlimit, | ||||||
| 					&tclass, &dontfrag); | 					&tclass, &dontfrag); | ||||||
| 		if (err < 0) { | 		if (err < 0) { | ||||||
| 			fl6_sock_release(flowlabel); | 			fl6_sock_release(flowlabel); | ||||||
| 			return err; | 			return err; | ||||||
| 		} | 		} | ||||||
| 		if ((fl.fl6_flowlabel&IPV6_FLOWLABEL_MASK) && !flowlabel) { | 		if ((fl6.flowlabel&IPV6_FLOWLABEL_MASK) && !flowlabel) { | ||||||
| 			flowlabel = fl6_sock_lookup(sk, fl.fl6_flowlabel); | 			flowlabel = fl6_sock_lookup(sk, fl6.flowlabel); | ||||||
| 			if (flowlabel == NULL) | 			if (flowlabel == NULL) | ||||||
| 				return -EINVAL; | 				return -EINVAL; | ||||||
| 		} | 		} | ||||||
|  | @ -1105,27 +1105,27 @@ do_udp_sendmsg: | ||||||
| 		opt = fl6_merge_options(&opt_space, flowlabel, opt); | 		opt = fl6_merge_options(&opt_space, flowlabel, opt); | ||||||
| 	opt = ipv6_fixup_options(&opt_space, opt); | 	opt = ipv6_fixup_options(&opt_space, opt); | ||||||
| 
 | 
 | ||||||
| 	fl.flowi_proto = sk->sk_protocol; | 	fl6.flowi6_proto = sk->sk_protocol; | ||||||
| 	if (!ipv6_addr_any(daddr)) | 	if (!ipv6_addr_any(daddr)) | ||||||
| 		ipv6_addr_copy(&fl.fl6_dst, daddr); | 		ipv6_addr_copy(&fl6.daddr, daddr); | ||||||
| 	else | 	else | ||||||
| 		fl.fl6_dst.s6_addr[15] = 0x1; /* :: means loopback (BSD'ism) */ | 		fl6.daddr.s6_addr[15] = 0x1; /* :: means loopback (BSD'ism) */ | ||||||
| 	if (ipv6_addr_any(&fl.fl6_src) && !ipv6_addr_any(&np->saddr)) | 	if (ipv6_addr_any(&fl6.saddr) && !ipv6_addr_any(&np->saddr)) | ||||||
| 		ipv6_addr_copy(&fl.fl6_src, &np->saddr); | 		ipv6_addr_copy(&fl6.saddr, &np->saddr); | ||||||
| 	fl.fl6_sport = inet->inet_sport; | 	fl6.uli.ports.sport = inet->inet_sport; | ||||||
| 
 | 
 | ||||||
| 	final_p = fl6_update_dst(&fl, opt, &final); | 	final_p = fl6_update_dst(&fl6, opt, &final); | ||||||
| 	if (final_p) | 	if (final_p) | ||||||
| 		connected = 0; | 		connected = 0; | ||||||
| 
 | 
 | ||||||
| 	if (!fl.flowi_oif && ipv6_addr_is_multicast(&fl.fl6_dst)) { | 	if (!fl6.flowi6_oif && ipv6_addr_is_multicast(&fl6.daddr)) { | ||||||
| 		fl.flowi_oif = np->mcast_oif; | 		fl6.flowi6_oif = np->mcast_oif; | ||||||
| 		connected = 0; | 		connected = 0; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	security_sk_classify_flow(sk, &fl); | 	security_sk_classify_flow(sk, flowi6_to_flowi(&fl6)); | ||||||
| 
 | 
 | ||||||
| 	dst = ip6_sk_dst_lookup_flow(sk, &fl, final_p, true); | 	dst = ip6_sk_dst_lookup_flow(sk, &fl6, final_p, true); | ||||||
| 	if (IS_ERR(dst)) { | 	if (IS_ERR(dst)) { | ||||||
| 		err = PTR_ERR(dst); | 		err = PTR_ERR(dst); | ||||||
| 		dst = NULL; | 		dst = NULL; | ||||||
|  | @ -1133,7 +1133,7 @@ do_udp_sendmsg: | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (hlimit < 0) { | 	if (hlimit < 0) { | ||||||
| 		if (ipv6_addr_is_multicast(&fl.fl6_dst)) | 		if (ipv6_addr_is_multicast(&fl6.daddr)) | ||||||
| 			hlimit = np->mcast_hops; | 			hlimit = np->mcast_hops; | ||||||
| 		else | 		else | ||||||
| 			hlimit = np->hop_limit; | 			hlimit = np->hop_limit; | ||||||
|  | @ -1168,7 +1168,7 @@ do_append_data: | ||||||
| 	up->len += ulen; | 	up->len += ulen; | ||||||
| 	getfrag  =  is_udplite ?  udplite_getfrag : ip_generic_getfrag; | 	getfrag  =  is_udplite ?  udplite_getfrag : ip_generic_getfrag; | ||||||
| 	err = ip6_append_data(sk, getfrag, msg->msg_iov, ulen, | 	err = ip6_append_data(sk, getfrag, msg->msg_iov, ulen, | ||||||
| 		sizeof(struct udphdr), hlimit, tclass, opt, &fl, | 		sizeof(struct udphdr), hlimit, tclass, opt, &fl6, | ||||||
| 		(struct rt6_info*)dst, | 		(struct rt6_info*)dst, | ||||||
| 		corkreq ? msg->msg_flags|MSG_MORE : msg->msg_flags, dontfrag); | 		corkreq ? msg->msg_flags|MSG_MORE : msg->msg_flags, dontfrag); | ||||||
| 	if (err) | 	if (err) | ||||||
|  | @ -1181,10 +1181,10 @@ do_append_data: | ||||||
| 	if (dst) { | 	if (dst) { | ||||||
| 		if (connected) { | 		if (connected) { | ||||||
| 			ip6_dst_store(sk, dst, | 			ip6_dst_store(sk, dst, | ||||||
| 				      ipv6_addr_equal(&fl.fl6_dst, &np->daddr) ? | 				      ipv6_addr_equal(&fl6.daddr, &np->daddr) ? | ||||||
| 				      &np->daddr : NULL, | 				      &np->daddr : NULL, | ||||||
| #ifdef CONFIG_IPV6_SUBTREES | #ifdef CONFIG_IPV6_SUBTREES | ||||||
| 				      ipv6_addr_equal(&fl.fl6_src, &np->saddr) ? | 				      ipv6_addr_equal(&fl6.saddr, &np->saddr) ? | ||||||
| 				      &np->saddr : | 				      &np->saddr : | ||||||
| #endif | #endif | ||||||
| 				      NULL); | 				      NULL); | ||||||
|  |  | ||||||
|  | @ -39,8 +39,7 @@ static struct dst_entry *xfrm6_dst_lookup(struct net *net, int tos, | ||||||
| 	if (saddr) | 	if (saddr) | ||||||
| 		memcpy(&fl6.saddr, saddr, sizeof(fl6.saddr)); | 		memcpy(&fl6.saddr, saddr, sizeof(fl6.saddr)); | ||||||
| 
 | 
 | ||||||
| 	dst = ip6_route_output(net, NULL, | 	dst = ip6_route_output(net, NULL, &fl6); | ||||||
| 			       flowi6_to_flowi(&fl6)); |  | ||||||
| 
 | 
 | ||||||
| 	err = dst->error; | 	err = dst->error; | ||||||
| 	if (dst->error) { | 	if (dst->error) { | ||||||
|  |  | ||||||
|  | @ -75,13 +75,11 @@ static int __ip_vs_addr_is_local_v6(struct net *net, | ||||||
| 				    const struct in6_addr *addr) | 				    const struct in6_addr *addr) | ||||||
| { | { | ||||||
| 	struct rt6_info *rt; | 	struct rt6_info *rt; | ||||||
| 	struct flowi fl = { | 	struct flowi6 fl6 = { | ||||||
| 		.flowi_oif = 0, | 		.daddr = *addr, | ||||||
| 		.fl6_dst = *addr, |  | ||||||
| 		.fl6_src = { .s6_addr32 = {0, 0, 0, 0} }, |  | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| 	rt = (struct rt6_info *)ip6_route_output(net, NULL, &fl); | 	rt = (struct rt6_info *)ip6_route_output(net, NULL, &fl6); | ||||||
| 	if (rt && rt->rt6i_dev && (rt->rt6i_dev->flags & IFF_LOOPBACK)) | 	if (rt && rt->rt6i_dev && (rt->rt6i_dev->flags & IFF_LOOPBACK)) | ||||||
| 		return 1; | 		return 1; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -198,27 +198,27 @@ __ip_vs_route_output_v6(struct net *net, struct in6_addr *daddr, | ||||||
| 			struct in6_addr *ret_saddr, int do_xfrm) | 			struct in6_addr *ret_saddr, int do_xfrm) | ||||||
| { | { | ||||||
| 	struct dst_entry *dst; | 	struct dst_entry *dst; | ||||||
| 	struct flowi fl = { | 	struct flowi6 fl6 = { | ||||||
| 		.fl6_dst = *daddr, | 		.daddr = *daddr, | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| 	dst = ip6_route_output(net, NULL, &fl); | 	dst = ip6_route_output(net, NULL, &fl6); | ||||||
| 	if (dst->error) | 	if (dst->error) | ||||||
| 		goto out_err; | 		goto out_err; | ||||||
| 	if (!ret_saddr) | 	if (!ret_saddr) | ||||||
| 		return dst; | 		return dst; | ||||||
| 	if (ipv6_addr_any(&fl.fl6_src) && | 	if (ipv6_addr_any(&fl6.saddr) && | ||||||
| 	    ipv6_dev_get_saddr(net, ip6_dst_idev(dst)->dev, | 	    ipv6_dev_get_saddr(net, ip6_dst_idev(dst)->dev, | ||||||
| 			       &fl.fl6_dst, 0, &fl.fl6_src) < 0) | 			       &fl6.daddr, 0, &fl6.saddr) < 0) | ||||||
| 		goto out_err; | 		goto out_err; | ||||||
| 	if (do_xfrm) { | 	if (do_xfrm) { | ||||||
| 		dst = xfrm_lookup(net, dst, &fl, NULL, 0); | 		dst = xfrm_lookup(net, dst, flowi6_to_flowi(&fl6), NULL, 0); | ||||||
| 		if (IS_ERR(dst)) { | 		if (IS_ERR(dst)) { | ||||||
| 			dst = NULL; | 			dst = NULL; | ||||||
| 			goto out_err; | 			goto out_err; | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	ipv6_addr_copy(ret_saddr, &fl.fl6_src); | 	ipv6_addr_copy(ret_saddr, &fl6.saddr); | ||||||
| 	return dst; | 	return dst; | ||||||
| 
 | 
 | ||||||
| out_err: | out_err: | ||||||
|  |  | ||||||
|  | @ -143,18 +143,18 @@ tee_tg_route6(struct sk_buff *skb, const struct xt_tee_tginfo *info) | ||||||
| 	const struct ipv6hdr *iph = ipv6_hdr(skb); | 	const struct ipv6hdr *iph = ipv6_hdr(skb); | ||||||
| 	struct net *net = pick_net(skb); | 	struct net *net = pick_net(skb); | ||||||
| 	struct dst_entry *dst; | 	struct dst_entry *dst; | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 
 | 
 | ||||||
| 	memset(&fl, 0, sizeof(fl)); | 	memset(&fl6, 0, sizeof(fl6)); | ||||||
| 	if (info->priv) { | 	if (info->priv) { | ||||||
| 		if (info->priv->oif == -1) | 		if (info->priv->oif == -1) | ||||||
| 			return false; | 			return false; | ||||||
| 		fl.flowi_oif = info->priv->oif; | 		fl6.flowi6_oif = info->priv->oif; | ||||||
| 	} | 	} | ||||||
| 	fl.fl6_dst = info->gw.in6; | 	fl6.daddr = info->gw.in6; | ||||||
| 	fl.fl6_flowlabel = ((iph->flow_lbl[0] & 0xF) << 16) | | 	fl6.flowlabel = ((iph->flow_lbl[0] & 0xF) << 16) | | ||||||
| 			   (iph->flow_lbl[1] << 8) | iph->flow_lbl[2]; | 			   (iph->flow_lbl[1] << 8) | iph->flow_lbl[2]; | ||||||
| 	dst = ip6_route_output(net, NULL, &fl); | 	dst = ip6_route_output(net, NULL, &fl6); | ||||||
| 	if (dst == NULL) | 	if (dst == NULL) | ||||||
| 		return false; | 		return false; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -201,40 +201,40 @@ static int sctp_v6_xmit(struct sk_buff *skb, struct sctp_transport *transport) | ||||||
| { | { | ||||||
| 	struct sock *sk = skb->sk; | 	struct sock *sk = skb->sk; | ||||||
| 	struct ipv6_pinfo *np = inet6_sk(sk); | 	struct ipv6_pinfo *np = inet6_sk(sk); | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 
 | 
 | ||||||
| 	memset(&fl, 0, sizeof(fl)); | 	memset(&fl6, 0, sizeof(fl6)); | ||||||
| 
 | 
 | ||||||
| 	fl.flowi_proto = sk->sk_protocol; | 	fl6.flowi6_proto = sk->sk_protocol; | ||||||
| 
 | 
 | ||||||
| 	/* Fill in the dest address from the route entry passed with the skb
 | 	/* Fill in the dest address from the route entry passed with the skb
 | ||||||
| 	 * and the source address from the transport. | 	 * and the source address from the transport. | ||||||
| 	 */ | 	 */ | ||||||
| 	ipv6_addr_copy(&fl.fl6_dst, &transport->ipaddr.v6.sin6_addr); | 	ipv6_addr_copy(&fl6.daddr, &transport->ipaddr.v6.sin6_addr); | ||||||
| 	ipv6_addr_copy(&fl.fl6_src, &transport->saddr.v6.sin6_addr); | 	ipv6_addr_copy(&fl6.saddr, &transport->saddr.v6.sin6_addr); | ||||||
| 
 | 
 | ||||||
| 	fl.fl6_flowlabel = np->flow_label; | 	fl6.flowlabel = np->flow_label; | ||||||
| 	IP6_ECN_flow_xmit(sk, fl.fl6_flowlabel); | 	IP6_ECN_flow_xmit(sk, fl6.flowlabel); | ||||||
| 	if (ipv6_addr_type(&fl.fl6_src) & IPV6_ADDR_LINKLOCAL) | 	if (ipv6_addr_type(&fl6.saddr) & IPV6_ADDR_LINKLOCAL) | ||||||
| 		fl.flowi_oif = transport->saddr.v6.sin6_scope_id; | 		fl6.flowi6_oif = transport->saddr.v6.sin6_scope_id; | ||||||
| 	else | 	else | ||||||
| 		fl.flowi_oif = sk->sk_bound_dev_if; | 		fl6.flowi6_oif = sk->sk_bound_dev_if; | ||||||
| 
 | 
 | ||||||
| 	if (np->opt && np->opt->srcrt) { | 	if (np->opt && np->opt->srcrt) { | ||||||
| 		struct rt0_hdr *rt0 = (struct rt0_hdr *) np->opt->srcrt; | 		struct rt0_hdr *rt0 = (struct rt0_hdr *) np->opt->srcrt; | ||||||
| 		ipv6_addr_copy(&fl.fl6_dst, rt0->addr); | 		ipv6_addr_copy(&fl6.daddr, rt0->addr); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	SCTP_DEBUG_PRINTK("%s: skb:%p, len:%d, src:%pI6 dst:%pI6\n", | 	SCTP_DEBUG_PRINTK("%s: skb:%p, len:%d, src:%pI6 dst:%pI6\n", | ||||||
| 			  __func__, skb, skb->len, | 			  __func__, skb, skb->len, | ||||||
| 			  &fl.fl6_src, &fl.fl6_dst); | 			  &fl6.saddr, &fl6.daddr); | ||||||
| 
 | 
 | ||||||
| 	SCTP_INC_STATS(SCTP_MIB_OUTSCTPPACKS); | 	SCTP_INC_STATS(SCTP_MIB_OUTSCTPPACKS); | ||||||
| 
 | 
 | ||||||
| 	if (!(transport->param_flags & SPP_PMTUD_ENABLE)) | 	if (!(transport->param_flags & SPP_PMTUD_ENABLE)) | ||||||
| 		skb->local_df = 1; | 		skb->local_df = 1; | ||||||
| 
 | 
 | ||||||
| 	return ip6_xmit(sk, skb, &fl, np->opt); | 	return ip6_xmit(sk, skb, &fl6, np->opt); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /* Returns the dst cache entry for the given source and destination ip
 | /* Returns the dst cache entry for the given source and destination ip
 | ||||||
|  | @ -245,22 +245,22 @@ static struct dst_entry *sctp_v6_get_dst(struct sctp_association *asoc, | ||||||
| 					 union sctp_addr *saddr) | 					 union sctp_addr *saddr) | ||||||
| { | { | ||||||
| 	struct dst_entry *dst; | 	struct dst_entry *dst; | ||||||
| 	struct flowi fl; | 	struct flowi6 fl6; | ||||||
| 
 | 
 | ||||||
| 	memset(&fl, 0, sizeof(fl)); | 	memset(&fl6, 0, sizeof(fl6)); | ||||||
| 	ipv6_addr_copy(&fl.fl6_dst, &daddr->v6.sin6_addr); | 	ipv6_addr_copy(&fl6.daddr, &daddr->v6.sin6_addr); | ||||||
| 	if (ipv6_addr_type(&daddr->v6.sin6_addr) & IPV6_ADDR_LINKLOCAL) | 	if (ipv6_addr_type(&daddr->v6.sin6_addr) & IPV6_ADDR_LINKLOCAL) | ||||||
| 		fl.flowi_oif = daddr->v6.sin6_scope_id; | 		fl6.flowi6_oif = daddr->v6.sin6_scope_id; | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 	SCTP_DEBUG_PRINTK("%s: DST=%pI6 ", __func__, &fl.fl6_dst); | 	SCTP_DEBUG_PRINTK("%s: DST=%pI6 ", __func__, &fl6.daddr); | ||||||
| 
 | 
 | ||||||
| 	if (saddr) { | 	if (saddr) { | ||||||
| 		ipv6_addr_copy(&fl.fl6_src, &saddr->v6.sin6_addr); | 		ipv6_addr_copy(&fl6.saddr, &saddr->v6.sin6_addr); | ||||||
| 		SCTP_DEBUG_PRINTK("SRC=%pI6 - ", &fl.fl6_src); | 		SCTP_DEBUG_PRINTK("SRC=%pI6 - ", &fl6.saddr); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	dst = ip6_route_output(&init_net, NULL, &fl); | 	dst = ip6_route_output(&init_net, NULL, &fl6); | ||||||
| 	if (!dst->error) { | 	if (!dst->error) { | ||||||
| 		struct rt6_info *rt; | 		struct rt6_info *rt; | ||||||
| 		rt = (struct rt6_info *)dst; | 		rt = (struct rt6_info *)dst; | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 David S. Miller
				David S. Miller