mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-04 10:40:15 +02:00 
			
		
		
		
	udp: Simplify __udp*_lib_mcast_deliver.
Switch to using sk_nulls_for_each which shortens the code and makes it easier to update. Signed-off-by: David Held <drheld@google.com> Acked-by: Eric Dumazet <edumazet@google.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
		
							parent
							
								
									3e1c0f0b06
								
							
						
					
					
						commit
						5cf3d46192
					
				
					 2 changed files with 48 additions and 86 deletions
				
			
		| 
						 | 
				
			
			@ -594,26 +594,6 @@ static inline bool __udp_is_mcast_sock(struct net *net, struct sock *sk,
 | 
			
		|||
	return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline struct sock *udp_v4_mcast_next(struct net *net, struct sock *sk,
 | 
			
		||||
					     __be16 loc_port, __be32 loc_addr,
 | 
			
		||||
					     __be16 rmt_port, __be32 rmt_addr,
 | 
			
		||||
					     int dif)
 | 
			
		||||
{
 | 
			
		||||
	struct hlist_nulls_node *node;
 | 
			
		||||
	unsigned short hnum = ntohs(loc_port);
 | 
			
		||||
 | 
			
		||||
	sk_nulls_for_each_from(sk, node) {
 | 
			
		||||
		if (__udp_is_mcast_sock(net, sk,
 | 
			
		||||
					loc_port, loc_addr,
 | 
			
		||||
					rmt_port, rmt_addr,
 | 
			
		||||
					dif, hnum))
 | 
			
		||||
			goto found;
 | 
			
		||||
	}
 | 
			
		||||
	sk = NULL;
 | 
			
		||||
found:
 | 
			
		||||
	return sk;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * This routine is called by the ICMP module when it gets some
 | 
			
		||||
 * sort of error condition.  If err < 0 then the socket should
 | 
			
		||||
| 
						 | 
				
			
			@ -1667,23 +1647,23 @@ static int __udp4_lib_mcast_deliver(struct net *net, struct sk_buff *skb,
 | 
			
		|||
				    struct udp_table *udptable)
 | 
			
		||||
{
 | 
			
		||||
	struct sock *sk, *stack[256 / sizeof(struct sock *)];
 | 
			
		||||
	struct udp_hslot *hslot = udp_hashslot(udptable, net, ntohs(uh->dest));
 | 
			
		||||
	int dif;
 | 
			
		||||
	struct hlist_nulls_node *node;
 | 
			
		||||
	unsigned short hnum = ntohs(uh->dest);
 | 
			
		||||
	struct udp_hslot *hslot = udp_hashslot(udptable, net, hnum);
 | 
			
		||||
	int dif = skb->dev->ifindex;
 | 
			
		||||
	unsigned int i, count = 0;
 | 
			
		||||
 | 
			
		||||
	spin_lock(&hslot->lock);
 | 
			
		||||
	sk = sk_nulls_head(&hslot->head);
 | 
			
		||||
	dif = skb->dev->ifindex;
 | 
			
		||||
	sk = udp_v4_mcast_next(net, sk, uh->dest, daddr, uh->source, saddr, dif);
 | 
			
		||||
	while (sk) {
 | 
			
		||||
		stack[count++] = sk;
 | 
			
		||||
		sk = udp_v4_mcast_next(net, sk_nulls_next(sk), uh->dest,
 | 
			
		||||
				       daddr, uh->source, saddr, dif);
 | 
			
		||||
		if (unlikely(count == ARRAY_SIZE(stack))) {
 | 
			
		||||
			if (!sk)
 | 
			
		||||
				break;
 | 
			
		||||
			flush_stack(stack, count, skb, ~0);
 | 
			
		||||
			count = 0;
 | 
			
		||||
	sk_nulls_for_each(sk, node, &hslot->head) {
 | 
			
		||||
		if (__udp_is_mcast_sock(net, sk,
 | 
			
		||||
					uh->dest, daddr,
 | 
			
		||||
					uh->source, saddr,
 | 
			
		||||
					dif, hnum)) {
 | 
			
		||||
			if (unlikely(count == ARRAY_SIZE(stack))) {
 | 
			
		||||
				flush_stack(stack, count, skb, ~0);
 | 
			
		||||
				count = 0;
 | 
			
		||||
			}
 | 
			
		||||
			stack[count++] = sk;
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
	/*
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -702,43 +702,26 @@ int udpv6_queue_rcv_skb(struct sock *sk, struct sk_buff *skb)
 | 
			
		|||
	return -1;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static struct sock *udp_v6_mcast_next(struct net *net, struct sock *sk,
 | 
			
		||||
				      __be16 loc_port, const struct in6_addr *loc_addr,
 | 
			
		||||
				      __be16 rmt_port, const struct in6_addr *rmt_addr,
 | 
			
		||||
				      int dif)
 | 
			
		||||
static bool __udp_v6_is_mcast_sock(struct net *net, struct sock *sk,
 | 
			
		||||
				   __be16 loc_port, const struct in6_addr *loc_addr,
 | 
			
		||||
				   __be16 rmt_port, const struct in6_addr *rmt_addr,
 | 
			
		||||
				   int dif, unsigned short hnum)
 | 
			
		||||
{
 | 
			
		||||
	struct hlist_nulls_node *node;
 | 
			
		||||
	unsigned short num = ntohs(loc_port);
 | 
			
		||||
	struct inet_sock *inet = inet_sk(sk);
 | 
			
		||||
 | 
			
		||||
	sk_nulls_for_each_from(sk, node) {
 | 
			
		||||
		struct inet_sock *inet = inet_sk(sk);
 | 
			
		||||
	if (!net_eq(sock_net(sk), net))
 | 
			
		||||
		return false;
 | 
			
		||||
 | 
			
		||||
		if (!net_eq(sock_net(sk), net))
 | 
			
		||||
			continue;
 | 
			
		||||
 | 
			
		||||
		if (udp_sk(sk)->udp_port_hash == num &&
 | 
			
		||||
		    sk->sk_family == PF_INET6) {
 | 
			
		||||
			if (inet->inet_dport) {
 | 
			
		||||
				if (inet->inet_dport != rmt_port)
 | 
			
		||||
					continue;
 | 
			
		||||
			}
 | 
			
		||||
			if (!ipv6_addr_any(&sk->sk_v6_daddr) &&
 | 
			
		||||
			    !ipv6_addr_equal(&sk->sk_v6_daddr, rmt_addr))
 | 
			
		||||
				continue;
 | 
			
		||||
 | 
			
		||||
			if (sk->sk_bound_dev_if && sk->sk_bound_dev_if != dif)
 | 
			
		||||
				continue;
 | 
			
		||||
 | 
			
		||||
			if (!ipv6_addr_any(&sk->sk_v6_rcv_saddr)) {
 | 
			
		||||
				if (!ipv6_addr_equal(&sk->sk_v6_rcv_saddr, loc_addr))
 | 
			
		||||
					continue;
 | 
			
		||||
			}
 | 
			
		||||
			if (!inet6_mc_check(sk, loc_addr, rmt_addr))
 | 
			
		||||
				continue;
 | 
			
		||||
			return sk;
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
	return NULL;
 | 
			
		||||
	if (udp_sk(sk)->udp_port_hash != hnum ||
 | 
			
		||||
	    sk->sk_family != PF_INET6 ||
 | 
			
		||||
	    (inet->inet_dport && inet->inet_dport != rmt_port) ||
 | 
			
		||||
	    (!ipv6_addr_any(&sk->sk_v6_daddr) &&
 | 
			
		||||
		    !ipv6_addr_equal(&sk->sk_v6_daddr, rmt_addr)) ||
 | 
			
		||||
	    (sk->sk_bound_dev_if && sk->sk_bound_dev_if != dif))
 | 
			
		||||
		return false;
 | 
			
		||||
	if (!inet6_mc_check(sk, loc_addr, rmt_addr))
 | 
			
		||||
		return false;
 | 
			
		||||
	return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void flush_stack(struct sock **stack, unsigned int count,
 | 
			
		||||
| 
						 | 
				
			
			@ -787,28 +770,27 @@ static int __udp6_lib_mcast_deliver(struct net *net, struct sk_buff *skb,
 | 
			
		|||
{
 | 
			
		||||
	struct sock *sk, *stack[256 / sizeof(struct sock *)];
 | 
			
		||||
	const struct udphdr *uh = udp_hdr(skb);
 | 
			
		||||
	struct udp_hslot *hslot = udp_hashslot(udptable, net, ntohs(uh->dest));
 | 
			
		||||
	int dif;
 | 
			
		||||
	struct hlist_nulls_node *node;
 | 
			
		||||
	unsigned short hnum = ntohs(uh->dest);
 | 
			
		||||
	struct udp_hslot *hslot = udp_hashslot(udptable, net, hnum);
 | 
			
		||||
	int dif = inet6_iif(skb);
 | 
			
		||||
	unsigned int i, count = 0;
 | 
			
		||||
 | 
			
		||||
	spin_lock(&hslot->lock);
 | 
			
		||||
	sk = sk_nulls_head(&hslot->head);
 | 
			
		||||
	dif = inet6_iif(skb);
 | 
			
		||||
	sk = udp_v6_mcast_next(net, sk, uh->dest, daddr, uh->source, saddr, dif);
 | 
			
		||||
	while (sk) {
 | 
			
		||||
		/* If zero checksum and no_check is not on for
 | 
			
		||||
		 * the socket then skip it.
 | 
			
		||||
		 */
 | 
			
		||||
		if (uh->check || udp_sk(sk)->no_check6_rx)
 | 
			
		||||
	sk_nulls_for_each(sk, node, &hslot->head) {
 | 
			
		||||
		if (__udp_v6_is_mcast_sock(net, sk,
 | 
			
		||||
					   uh->dest, daddr,
 | 
			
		||||
					   uh->source, saddr,
 | 
			
		||||
					   dif, hnum) &&
 | 
			
		||||
		    /* If zero checksum and no_check is not on for
 | 
			
		||||
		     * the socket then skip it.
 | 
			
		||||
		     */
 | 
			
		||||
		    (uh->check || udp_sk(sk)->no_check6_rx)) {
 | 
			
		||||
			if (unlikely(count == ARRAY_SIZE(stack))) {
 | 
			
		||||
				flush_stack(stack, count, skb, ~0);
 | 
			
		||||
				count = 0;
 | 
			
		||||
			}
 | 
			
		||||
			stack[count++] = sk;
 | 
			
		||||
 | 
			
		||||
		sk = udp_v6_mcast_next(net, sk_nulls_next(sk), uh->dest, daddr,
 | 
			
		||||
				       uh->source, saddr, dif);
 | 
			
		||||
		if (unlikely(count == ARRAY_SIZE(stack))) {
 | 
			
		||||
			if (!sk)
 | 
			
		||||
				break;
 | 
			
		||||
			flush_stack(stack, count, skb, ~0);
 | 
			
		||||
			count = 0;
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
	/*
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
		Loading…
	
		Reference in a new issue