forked from mirrors/linux
		
	net: dccp: Add handling of IPV6_PKTOPTIONS to dccp_v6_do_rcv()
Add handling of IPV6_PKTOPTIONS to dccp_v6_do_rcv() in net/dccp/ipv6.c, similar to the handling in net/ipv6/tcp_ipv6.c Signed-off-by: Andrii Vladyka <tulup@mail.ru> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
		
							parent
							
								
									e3cfddd577
								
							
						
					
					
						commit
						323fbd0edf
					
				
					 1 changed files with 35 additions and 12 deletions
				
			
		| 
						 | 
					@ -16,6 +16,7 @@
 | 
				
			||||||
#include <linux/random.h>
 | 
					#include <linux/random.h>
 | 
				
			||||||
#include <linux/slab.h>
 | 
					#include <linux/slab.h>
 | 
				
			||||||
#include <linux/xfrm.h>
 | 
					#include <linux/xfrm.h>
 | 
				
			||||||
 | 
					#include <linux/string.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#include <net/addrconf.h>
 | 
					#include <net/addrconf.h>
 | 
				
			||||||
#include <net/inet_common.h>
 | 
					#include <net/inet_common.h>
 | 
				
			||||||
| 
						 | 
					@ -30,6 +31,7 @@
 | 
				
			||||||
#include <net/ip6_checksum.h>
 | 
					#include <net/ip6_checksum.h>
 | 
				
			||||||
#include <net/xfrm.h>
 | 
					#include <net/xfrm.h>
 | 
				
			||||||
#include <net/secure_seq.h>
 | 
					#include <net/secure_seq.h>
 | 
				
			||||||
 | 
					#include <net/sock.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#include "dccp.h"
 | 
					#include "dccp.h"
 | 
				
			||||||
#include "ipv6.h"
 | 
					#include "ipv6.h"
 | 
				
			||||||
| 
						 | 
					@ -597,19 +599,13 @@ static int dccp_v6_do_rcv(struct sock *sk, struct sk_buff *skb)
 | 
				
			||||||
					       --ANK (980728)
 | 
										       --ANK (980728)
 | 
				
			||||||
	 */
 | 
						 */
 | 
				
			||||||
	if (np->rxopt.all)
 | 
						if (np->rxopt.all)
 | 
				
			||||||
	/*
 | 
					 | 
				
			||||||
	 * FIXME: Add handling of IPV6_PKTOPTIONS skb. See the comments below
 | 
					 | 
				
			||||||
	 *        (wrt ipv6_pktopions) and net/ipv6/tcp_ipv6.c for an example.
 | 
					 | 
				
			||||||
	 */
 | 
					 | 
				
			||||||
		opt_skb = skb_clone(skb, GFP_ATOMIC);
 | 
							opt_skb = skb_clone(skb, GFP_ATOMIC);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (sk->sk_state == DCCP_OPEN) { /* Fast path */
 | 
						if (sk->sk_state == DCCP_OPEN) { /* Fast path */
 | 
				
			||||||
		if (dccp_rcv_established(sk, skb, dccp_hdr(skb), skb->len))
 | 
							if (dccp_rcv_established(sk, skb, dccp_hdr(skb), skb->len))
 | 
				
			||||||
			goto reset;
 | 
								goto reset;
 | 
				
			||||||
		if (opt_skb) {
 | 
							if (opt_skb)
 | 
				
			||||||
			/* XXX This is where we would goto ipv6_pktoptions. */
 | 
								goto ipv6_pktoptions;
 | 
				
			||||||
			__kfree_skb(opt_skb);
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
		return 0;
 | 
							return 0;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -640,10 +636,8 @@ static int dccp_v6_do_rcv(struct sock *sk, struct sk_buff *skb)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (dccp_rcv_state_process(sk, skb, dccp_hdr(skb), skb->len))
 | 
						if (dccp_rcv_state_process(sk, skb, dccp_hdr(skb), skb->len))
 | 
				
			||||||
		goto reset;
 | 
							goto reset;
 | 
				
			||||||
	if (opt_skb) {
 | 
						if (opt_skb)
 | 
				
			||||||
		/* XXX This is where we would goto ipv6_pktoptions. */
 | 
							goto ipv6_pktoptions;
 | 
				
			||||||
		__kfree_skb(opt_skb);
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
reset:
 | 
					reset:
 | 
				
			||||||
| 
						 | 
					@ -653,6 +647,35 @@ static int dccp_v6_do_rcv(struct sock *sk, struct sk_buff *skb)
 | 
				
			||||||
		__kfree_skb(opt_skb);
 | 
							__kfree_skb(opt_skb);
 | 
				
			||||||
	kfree_skb(skb);
 | 
						kfree_skb(skb);
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Handling IPV6_PKTOPTIONS skb the similar
 | 
				
			||||||
 | 
					 * way it's done for net/ipv6/tcp_ipv6.c
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					ipv6_pktoptions:
 | 
				
			||||||
 | 
						if (!((1 << sk->sk_state) & (DCCPF_CLOSED | DCCPF_LISTEN))) {
 | 
				
			||||||
 | 
							if (np->rxopt.bits.rxinfo || np->rxopt.bits.rxoinfo)
 | 
				
			||||||
 | 
								np->mcast_oif = inet6_iif(opt_skb);
 | 
				
			||||||
 | 
							if (np->rxopt.bits.rxhlim || np->rxopt.bits.rxohlim)
 | 
				
			||||||
 | 
								np->mcast_hops = ipv6_hdr(opt_skb)->hop_limit;
 | 
				
			||||||
 | 
							if (np->rxopt.bits.rxflow || np->rxopt.bits.rxtclass)
 | 
				
			||||||
 | 
								np->rcv_flowinfo = ip6_flowinfo(ipv6_hdr(opt_skb));
 | 
				
			||||||
 | 
							if (np->repflow)
 | 
				
			||||||
 | 
								np->flow_label = ip6_flowlabel(ipv6_hdr(opt_skb));
 | 
				
			||||||
 | 
							if (ipv6_opt_accepted(sk, opt_skb,
 | 
				
			||||||
 | 
									      &DCCP_SKB_CB(opt_skb)->header.h6)) {
 | 
				
			||||||
 | 
								skb_set_owner_r(opt_skb, sk);
 | 
				
			||||||
 | 
								memmove(IP6CB(opt_skb),
 | 
				
			||||||
 | 
									&DCCP_SKB_CB(opt_skb)->header.h6,
 | 
				
			||||||
 | 
									sizeof(struct inet6_skb_parm));
 | 
				
			||||||
 | 
								opt_skb = xchg(&np->pktoptions, opt_skb);
 | 
				
			||||||
 | 
							} else {
 | 
				
			||||||
 | 
								__kfree_skb(opt_skb);
 | 
				
			||||||
 | 
								opt_skb = xchg(&np->pktoptions, NULL);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						kfree_skb(opt_skb);
 | 
				
			||||||
 | 
						return 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static int dccp_v6_rcv(struct sk_buff *skb)
 | 
					static int dccp_v6_rcv(struct sk_buff *skb)
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue