forked from mirrors/linux
		
	udp: remove headers from UDP packets before queueing
Remove UDP transport headers before queueing packets for reception. This change simplifies a follow-up patch to add MSG_PEEK support. Signed-off-by: Sam Kumar <samanthakumar@google.com> Signed-off-by: Willem de Bruijn <willemb@google.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
		
							parent
							
								
									b9bb53f383
								
							
						
					
					
						commit
						e6afc8ace6
					
				
					 5 changed files with 41 additions and 20 deletions
				
			
		|  | @ -1864,6 +1864,7 @@ void sk_reset_timer(struct sock *sk, struct timer_list *timer, | ||||||
| 
 | 
 | ||||||
| void sk_stop_timer(struct sock *sk, struct timer_list *timer); | void sk_stop_timer(struct sock *sk, struct timer_list *timer); | ||||||
| 
 | 
 | ||||||
|  | int __sock_queue_rcv_skb(struct sock *sk, struct sk_buff *skb); | ||||||
| int sock_queue_rcv_skb(struct sock *sk, struct sk_buff *skb); | int sock_queue_rcv_skb(struct sock *sk, struct sk_buff *skb); | ||||||
| 
 | 
 | ||||||
| int sock_queue_err_skb(struct sock *sk, struct sk_buff *skb); | int sock_queue_err_skb(struct sock *sk, struct sk_buff *skb); | ||||||
|  |  | ||||||
|  | @ -158,6 +158,15 @@ static inline __sum16 udp_v4_check(int len, __be32 saddr, | ||||||
| void udp_set_csum(bool nocheck, struct sk_buff *skb, | void udp_set_csum(bool nocheck, struct sk_buff *skb, | ||||||
| 		  __be32 saddr, __be32 daddr, int len); | 		  __be32 saddr, __be32 daddr, int len); | ||||||
| 
 | 
 | ||||||
|  | static inline void udp_csum_pull_header(struct sk_buff *skb) | ||||||
|  | { | ||||||
|  | 	if (skb->ip_summed == CHECKSUM_NONE) | ||||||
|  | 		skb->csum = csum_partial(udp_hdr(skb), sizeof(struct udphdr), | ||||||
|  | 					 skb->csum); | ||||||
|  | 	skb_pull_rcsum(skb, sizeof(struct udphdr)); | ||||||
|  | 	UDP_SKB_CB(skb)->cscov -= sizeof(struct udphdr); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| struct sk_buff **udp_gro_receive(struct sk_buff **head, struct sk_buff *skb, | struct sk_buff **udp_gro_receive(struct sk_buff **head, struct sk_buff *skb, | ||||||
| 				 struct udphdr *uh); | 				 struct udphdr *uh); | ||||||
| int udp_gro_complete(struct sk_buff *skb, int nhoff); | int udp_gro_complete(struct sk_buff *skb, int nhoff); | ||||||
|  |  | ||||||
|  | @ -402,9 +402,8 @@ static void sock_disable_timestamp(struct sock *sk, unsigned long flags) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| int sock_queue_rcv_skb(struct sock *sk, struct sk_buff *skb) | int __sock_queue_rcv_skb(struct sock *sk, struct sk_buff *skb) | ||||||
| { | { | ||||||
| 	int err; |  | ||||||
| 	unsigned long flags; | 	unsigned long flags; | ||||||
| 	struct sk_buff_head *list = &sk->sk_receive_queue; | 	struct sk_buff_head *list = &sk->sk_receive_queue; | ||||||
| 
 | 
 | ||||||
|  | @ -414,10 +413,6 @@ int sock_queue_rcv_skb(struct sock *sk, struct sk_buff *skb) | ||||||
| 		return -ENOMEM; | 		return -ENOMEM; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	err = sk_filter(sk, skb); |  | ||||||
| 	if (err) |  | ||||||
| 		return err; |  | ||||||
| 
 |  | ||||||
| 	if (!sk_rmem_schedule(sk, skb, skb->truesize)) { | 	if (!sk_rmem_schedule(sk, skb, skb->truesize)) { | ||||||
| 		atomic_inc(&sk->sk_drops); | 		atomic_inc(&sk->sk_drops); | ||||||
| 		return -ENOBUFS; | 		return -ENOBUFS; | ||||||
|  | @ -440,6 +435,18 @@ int sock_queue_rcv_skb(struct sock *sk, struct sk_buff *skb) | ||||||
| 		sk->sk_data_ready(sk); | 		sk->sk_data_ready(sk); | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
|  | EXPORT_SYMBOL(__sock_queue_rcv_skb); | ||||||
|  | 
 | ||||||
|  | int sock_queue_rcv_skb(struct sock *sk, struct sk_buff *skb) | ||||||
|  | { | ||||||
|  | 	int err; | ||||||
|  | 
 | ||||||
|  | 	err = sk_filter(sk, skb); | ||||||
|  | 	if (err) | ||||||
|  | 		return err; | ||||||
|  | 
 | ||||||
|  | 	return __sock_queue_rcv_skb(sk, skb); | ||||||
|  | } | ||||||
| EXPORT_SYMBOL(sock_queue_rcv_skb); | EXPORT_SYMBOL(sock_queue_rcv_skb); | ||||||
| 
 | 
 | ||||||
| int sk_receive_skb(struct sock *sk, struct sk_buff *skb, const int nested) | int sk_receive_skb(struct sock *sk, struct sk_buff *skb, const int nested) | ||||||
|  |  | ||||||
|  | @ -1309,7 +1309,7 @@ int udp_recvmsg(struct sock *sk, struct msghdr *msg, size_t len, int noblock, | ||||||
| 	if (!skb) | 	if (!skb) | ||||||
| 		goto out; | 		goto out; | ||||||
| 
 | 
 | ||||||
| 	ulen = skb->len - sizeof(struct udphdr); | 	ulen = skb->len; | ||||||
| 	copied = len; | 	copied = len; | ||||||
| 	if (copied > ulen) | 	if (copied > ulen) | ||||||
| 		copied = ulen; | 		copied = ulen; | ||||||
|  | @ -1329,11 +1329,9 @@ int udp_recvmsg(struct sock *sk, struct msghdr *msg, size_t len, int noblock, | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (checksum_valid || skb_csum_unnecessary(skb)) | 	if (checksum_valid || skb_csum_unnecessary(skb)) | ||||||
| 		err = skb_copy_datagram_msg(skb, sizeof(struct udphdr), | 		err = skb_copy_datagram_msg(skb, 0, msg, copied); | ||||||
| 					    msg, copied); |  | ||||||
| 	else { | 	else { | ||||||
| 		err = skb_copy_and_csum_datagram_msg(skb, sizeof(struct udphdr), | 		err = skb_copy_and_csum_datagram_msg(skb, 0, msg); | ||||||
| 						     msg); |  | ||||||
| 
 | 
 | ||||||
| 		if (err == -EINVAL) | 		if (err == -EINVAL) | ||||||
| 			goto csum_copy_err; | 			goto csum_copy_err; | ||||||
|  | @ -1500,7 +1498,7 @@ static int __udp_queue_rcv_skb(struct sock *sk, struct sk_buff *skb) | ||||||
| 		sk_incoming_cpu_update(sk); | 		sk_incoming_cpu_update(sk); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	rc = sock_queue_rcv_skb(sk, skb); | 	rc = __sock_queue_rcv_skb(sk, skb); | ||||||
| 	if (rc < 0) { | 	if (rc < 0) { | ||||||
| 		int is_udplite = IS_UDPLITE(sk); | 		int is_udplite = IS_UDPLITE(sk); | ||||||
| 
 | 
 | ||||||
|  | @ -1616,10 +1614,14 @@ int udp_queue_rcv_skb(struct sock *sk, struct sk_buff *skb) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (rcu_access_pointer(sk->sk_filter) && | 	if (rcu_access_pointer(sk->sk_filter)) { | ||||||
| 	    udp_lib_checksum_complete(skb)) | 		if (udp_lib_checksum_complete(skb)) | ||||||
| 		goto csum_error; | 			goto csum_error; | ||||||
|  | 		if (sk_filter(sk, skb)) | ||||||
|  | 			goto drop; | ||||||
|  | 	} | ||||||
| 
 | 
 | ||||||
|  | 	udp_csum_pull_header(skb); | ||||||
| 	if (sk_rcvqueues_full(sk, sk->sk_rcvbuf)) { | 	if (sk_rcvqueues_full(sk, sk->sk_rcvbuf)) { | ||||||
| 		UDP_INC_STATS_BH(sock_net(sk), UDP_MIB_RCVBUFERRORS, | 		UDP_INC_STATS_BH(sock_net(sk), UDP_MIB_RCVBUFERRORS, | ||||||
| 				 is_udplite); | 				 is_udplite); | ||||||
|  |  | ||||||
|  | @ -376,7 +376,7 @@ int udpv6_recvmsg(struct sock *sk, struct msghdr *msg, size_t len, | ||||||
| 	if (!skb) | 	if (!skb) | ||||||
| 		goto out; | 		goto out; | ||||||
| 
 | 
 | ||||||
| 	ulen = skb->len - sizeof(struct udphdr); | 	ulen = skb->len; | ||||||
| 	copied = len; | 	copied = len; | ||||||
| 	if (copied > ulen) | 	if (copied > ulen) | ||||||
| 		copied = ulen; | 		copied = ulen; | ||||||
|  | @ -398,10 +398,9 @@ int udpv6_recvmsg(struct sock *sk, struct msghdr *msg, size_t len, | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (checksum_valid || skb_csum_unnecessary(skb)) | 	if (checksum_valid || skb_csum_unnecessary(skb)) | ||||||
| 		err = skb_copy_datagram_msg(skb, sizeof(struct udphdr), | 		err = skb_copy_datagram_msg(skb, 0, msg, copied); | ||||||
| 					    msg, copied); |  | ||||||
| 	else { | 	else { | ||||||
| 		err = skb_copy_and_csum_datagram_msg(skb, sizeof(struct udphdr), msg); | 		err = skb_copy_and_csum_datagram_msg(skb, 0, msg); | ||||||
| 		if (err == -EINVAL) | 		if (err == -EINVAL) | ||||||
| 			goto csum_copy_err; | 			goto csum_copy_err; | ||||||
| 	} | 	} | ||||||
|  | @ -554,7 +553,7 @@ static int __udpv6_queue_rcv_skb(struct sock *sk, struct sk_buff *skb) | ||||||
| 		sk_incoming_cpu_update(sk); | 		sk_incoming_cpu_update(sk); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	rc = sock_queue_rcv_skb(sk, skb); | 	rc = __sock_queue_rcv_skb(sk, skb); | ||||||
| 	if (rc < 0) { | 	if (rc < 0) { | ||||||
| 		int is_udplite = IS_UDPLITE(sk); | 		int is_udplite = IS_UDPLITE(sk); | ||||||
| 
 | 
 | ||||||
|  | @ -648,8 +647,11 @@ int udpv6_queue_rcv_skb(struct sock *sk, struct sk_buff *skb) | ||||||
| 	if (rcu_access_pointer(sk->sk_filter)) { | 	if (rcu_access_pointer(sk->sk_filter)) { | ||||||
| 		if (udp_lib_checksum_complete(skb)) | 		if (udp_lib_checksum_complete(skb)) | ||||||
| 			goto csum_error; | 			goto csum_error; | ||||||
|  | 		if (sk_filter(sk, skb)) | ||||||
|  | 			goto drop; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	udp_csum_pull_header(skb); | ||||||
| 	if (sk_rcvqueues_full(sk, sk->sk_rcvbuf)) { | 	if (sk_rcvqueues_full(sk, sk->sk_rcvbuf)) { | ||||||
| 		UDP6_INC_STATS_BH(sock_net(sk), | 		UDP6_INC_STATS_BH(sock_net(sk), | ||||||
| 				  UDP_MIB_RCVBUFERRORS, is_udplite); | 				  UDP_MIB_RCVBUFERRORS, is_udplite); | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue
	
	 samanthakumar
						samanthakumar