diff options
Diffstat (limited to 'net')
| -rw-r--r-- | net/core/dev.c | 2 | ||||
| -rw-r--r-- | net/core/neighbour.c | 40 | ||||
| -rw-r--r-- | net/dns_resolver/dns_key.c | 13 | ||||
| -rw-r--r-- | net/ipv4/tcp.c | 6 | ||||
| -rw-r--r-- | net/ipv4/tcp_input.c | 7 | ||||
| -rw-r--r-- | net/ipv6/route.c | 2 | ||||
| -rw-r--r-- | net/l2tp/l2tp_ppp.c | 7 | ||||
| -rw-r--r-- | net/llc/af_llc.c | 14 | ||||
| -rw-r--r-- | net/llc/llc_c_ac.c | 9 | ||||
| -rw-r--r-- | net/llc/llc_conn.c | 22 | ||||
| -rw-r--r-- | net/packet/af_packet.c | 88 | ||||
| -rw-r--r-- | net/packet/internal.h | 10 | ||||
| -rw-r--r-- | net/sctp/ipv6.c | 60 | ||||
| -rw-r--r-- | net/tipc/net.c | 3 | 
14 files changed, 181 insertions, 102 deletions
| diff --git a/net/core/dev.c b/net/core/dev.c index dc63c37d5301..3bcbf931a910 100644 --- a/net/core/dev.c +++ b/net/core/dev.c @@ -2706,7 +2706,7 @@ netdev_features_t passthru_features_check(struct sk_buff *skb,  }  EXPORT_SYMBOL(passthru_features_check); -static netdev_features_t dflt_features_check(const struct sk_buff *skb, +static netdev_features_t dflt_features_check(struct sk_buff *skb,  					     struct net_device *dev,  					     netdev_features_t features)  { diff --git a/net/core/neighbour.c b/net/core/neighbour.c index 33432e64804c..f60b93627876 100644 --- a/net/core/neighbour.c +++ b/net/core/neighbour.c @@ -54,7 +54,8 @@ do {						\  static void neigh_timer_handler(unsigned long arg);  static void __neigh_notify(struct neighbour *n, int type, int flags);  static void neigh_update_notify(struct neighbour *neigh); -static int pneigh_ifdown(struct neigh_table *tbl, struct net_device *dev); +static int pneigh_ifdown_and_unlock(struct neigh_table *tbl, +				    struct net_device *dev);  #ifdef CONFIG_PROC_FS  static const struct file_operations neigh_stat_seq_fops; @@ -254,8 +255,7 @@ int neigh_ifdown(struct neigh_table *tbl, struct net_device *dev)  {  	write_lock_bh(&tbl->lock);  	neigh_flush_dev(tbl, dev); -	pneigh_ifdown(tbl, dev); -	write_unlock_bh(&tbl->lock); +	pneigh_ifdown_and_unlock(tbl, dev);  	del_timer_sync(&tbl->proxy_timer);  	pneigh_queue_purge(&tbl->proxy_queue); @@ -645,9 +645,10 @@ int pneigh_delete(struct neigh_table *tbl, struct net *net, const void *pkey,  	return -ENOENT;  } -static int pneigh_ifdown(struct neigh_table *tbl, struct net_device *dev) +static int pneigh_ifdown_and_unlock(struct neigh_table *tbl, +				    struct net_device *dev)  { -	struct pneigh_entry *n, **np; +	struct pneigh_entry *n, **np, *freelist = NULL;  	u32 h;  	for (h = 0; h <= PNEIGH_HASHMASK; h++) { @@ -655,16 +656,23 @@ static int pneigh_ifdown(struct neigh_table *tbl, struct net_device *dev)  		while ((n = *np) != NULL) {  			if (!dev || n->dev == dev) {  				*np = n->next; -				if (tbl->pdestructor) -					tbl->pdestructor(n); -				if (n->dev) -					dev_put(n->dev); -				kfree(n); +				n->next = freelist; +				freelist = n;  				continue;  			}  			np = &n->next;  		}  	} +	write_unlock_bh(&tbl->lock); +	while ((n = freelist)) { +		freelist = n->next; +		n->next = NULL; +		if (tbl->pdestructor) +			tbl->pdestructor(n); +		if (n->dev) +			dev_put(n->dev); +		kfree(n); +	}  	return -ENOENT;  } @@ -2280,12 +2288,16 @@ static int neigh_dump_table(struct neigh_table *tbl, struct sk_buff *skb,  	err = nlmsg_parse(nlh, sizeof(struct ndmsg), tb, NDA_MAX, NULL);  	if (!err) { -		if (tb[NDA_IFINDEX]) +		if (tb[NDA_IFINDEX]) { +			if (nla_len(tb[NDA_IFINDEX]) != sizeof(u32)) +				return -EINVAL;  			filter_idx = nla_get_u32(tb[NDA_IFINDEX]); - -		if (tb[NDA_MASTER]) +		} +		if (tb[NDA_MASTER]) { +			if (nla_len(tb[NDA_MASTER]) != sizeof(u32)) +				return -EINVAL;  			filter_master_idx = nla_get_u32(tb[NDA_MASTER]); - +		}  		if (filter_idx || filter_master_idx)  			flags |= NLM_F_DUMP_FILTERED;  	} diff --git a/net/dns_resolver/dns_key.c b/net/dns_resolver/dns_key.c index 6abc5012200b..e26df2764e83 100644 --- a/net/dns_resolver/dns_key.c +++ b/net/dns_resolver/dns_key.c @@ -25,6 +25,7 @@  #include <linux/moduleparam.h>  #include <linux/slab.h>  #include <linux/string.h> +#include <linux/ratelimit.h>  #include <linux/kernel.h>  #include <linux/keyctl.h>  #include <linux/err.h> @@ -91,9 +92,9 @@ dns_resolver_preparse(struct key_preparsed_payload *prep)  			next_opt = memchr(opt, '#', end - opt) ?: end;  			opt_len = next_opt - opt; -			if (!opt_len) { -				printk(KERN_WARNING -				       "Empty option to dns_resolver key\n"); +			if (opt_len <= 0 || opt_len > 128) { +				pr_warn_ratelimited("Invalid option length (%d) for dns_resolver key\n", +						    opt_len);  				return -EINVAL;  			} @@ -127,10 +128,8 @@ dns_resolver_preparse(struct key_preparsed_payload *prep)  			}  		bad_option_value: -			printk(KERN_WARNING -			       "Option '%*.*s' to dns_resolver key:" -			       " bad/missing value\n", -			       opt_nlen, opt_nlen, opt); +			pr_warn_ratelimited("Option '%*.*s' to dns_resolver key: bad/missing value\n", +					    opt_nlen, opt_nlen, opt);  			return -EINVAL;  		} while (opt = next_opt + 1, opt < end);  	} diff --git a/net/ipv4/tcp.c b/net/ipv4/tcp.c index b7f089b79b42..b74669e23084 100644 --- a/net/ipv4/tcp.c +++ b/net/ipv4/tcp.c @@ -2589,8 +2589,10 @@ static int do_tcp_setsockopt(struct sock *sk, int level,  #ifdef CONFIG_TCP_MD5SIG  	case TCP_MD5SIG: -		/* Read the IP->Key mappings from userspace */ -		err = tp->af_specific->md5_parse(sk, optval, optlen); +		if ((1 << sk->sk_state) & (TCPF_CLOSE | TCPF_LISTEN)) +			err = tp->af_specific->md5_parse(sk, optval, optlen); +		else +			err = -EINVAL;  		break;  #endif  	case TCP_USER_TIMEOUT: diff --git a/net/ipv4/tcp_input.c b/net/ipv4/tcp_input.c index 77ab2670d8fd..60e1349fcfbe 100644 --- a/net/ipv4/tcp_input.c +++ b/net/ipv4/tcp_input.c @@ -3870,11 +3870,8 @@ const u8 *tcp_parse_md5sig_option(const struct tcphdr *th)  	int length = (th->doff << 2) - sizeof(*th);  	const u8 *ptr = (const u8 *)(th + 1); -	/* If the TCP option is too short, we can short cut */ -	if (length < TCPOLEN_MD5SIG) -		return NULL; - -	while (length > 0) { +	/* If not enough data remaining, we can short cut */ +	while (length >= TCPOLEN_MD5SIG) {  		int opcode = *ptr++;  		int opsize; diff --git a/net/ipv6/route.c b/net/ipv6/route.c index b88ca41f9550..0a7403b9d572 100644 --- a/net/ipv6/route.c +++ b/net/ipv6/route.c @@ -2702,6 +2702,7 @@ void rt6_mtu_change(struct net_device *dev, unsigned int mtu)  static const struct nla_policy rtm_ipv6_policy[RTA_MAX+1] = {  	[RTA_GATEWAY]           = { .len = sizeof(struct in6_addr) }, +	[RTA_PREFSRC]		= { .len = sizeof(struct in6_addr) },  	[RTA_OIF]               = { .type = NLA_U32 },  	[RTA_IIF]		= { .type = NLA_U32 },  	[RTA_PRIORITY]          = { .type = NLA_U32 }, @@ -2711,6 +2712,7 @@ static const struct nla_policy rtm_ipv6_policy[RTA_MAX+1] = {  	[RTA_ENCAP_TYPE]	= { .type = NLA_U16 },  	[RTA_ENCAP]		= { .type = NLA_NESTED },  	[RTA_UID]		= { .type = NLA_U32 }, +	[RTA_TABLE]		= { .type = NLA_U32 },  };  static int rtm_to_fib6_config(struct sk_buff *skb, struct nlmsghdr *nlh, diff --git a/net/l2tp/l2tp_ppp.c b/net/l2tp/l2tp_ppp.c index 67f2e72723b2..2764c4bd072c 100644 --- a/net/l2tp/l2tp_ppp.c +++ b/net/l2tp/l2tp_ppp.c @@ -606,6 +606,13 @@ static int pppol2tp_connect(struct socket *sock, struct sockaddr *uservaddr,  	lock_sock(sk);  	error = -EINVAL; + +	if (sockaddr_len != sizeof(struct sockaddr_pppol2tp) && +	    sockaddr_len != sizeof(struct sockaddr_pppol2tpv3) && +	    sockaddr_len != sizeof(struct sockaddr_pppol2tpin6) && +	    sockaddr_len != sizeof(struct sockaddr_pppol2tpv3in6)) +		goto end; +  	if (sp->sa_protocol != PX_PROTO_OL2TP)  		goto end; diff --git a/net/llc/af_llc.c b/net/llc/af_llc.c index 1e698768aca8..09f2f3471ad6 100644 --- a/net/llc/af_llc.c +++ b/net/llc/af_llc.c @@ -197,9 +197,19 @@ static int llc_ui_release(struct socket *sock)  		llc->laddr.lsap, llc->daddr.lsap);  	if (!llc_send_disc(sk))  		llc_ui_wait_for_disc(sk, sk->sk_rcvtimeo); -	if (!sock_flag(sk, SOCK_ZAPPED)) +	if (!sock_flag(sk, SOCK_ZAPPED)) { +		struct llc_sap *sap = llc->sap; + +		/* Hold this for release_sock(), so that llc_backlog_rcv() +		 * could still use it. +		 */ +		llc_sap_hold(sap);  		llc_sap_remove_socket(llc->sap, sk); -	release_sock(sk); +		release_sock(sk); +		llc_sap_put(sap); +	} else { +		release_sock(sk); +	}  	if (llc->dev)  		dev_put(llc->dev);  	sock_put(sk); diff --git a/net/llc/llc_c_ac.c b/net/llc/llc_c_ac.c index ea225bd2672c..f8d4ab8ca1a5 100644 --- a/net/llc/llc_c_ac.c +++ b/net/llc/llc_c_ac.c @@ -1096,14 +1096,7 @@ int llc_conn_ac_inc_tx_win_size(struct sock *sk, struct sk_buff *skb)  int llc_conn_ac_stop_all_timers(struct sock *sk, struct sk_buff *skb)  { -	struct llc_sock *llc = llc_sk(sk); - -	del_timer(&llc->pf_cycle_timer.timer); -	del_timer(&llc->ack_timer.timer); -	del_timer(&llc->rej_sent_timer.timer); -	del_timer(&llc->busy_state_timer.timer); -	llc->ack_must_be_send = 0; -	llc->ack_pf = 0; +	llc_sk_stop_all_timers(sk, false);  	return 0;  } diff --git a/net/llc/llc_conn.c b/net/llc/llc_conn.c index 8bc5a1bd2d45..d861b74ad068 100644 --- a/net/llc/llc_conn.c +++ b/net/llc/llc_conn.c @@ -951,6 +951,26 @@ out:  	return sk;  } +void llc_sk_stop_all_timers(struct sock *sk, bool sync) +{ +	struct llc_sock *llc = llc_sk(sk); + +	if (sync) { +		del_timer_sync(&llc->pf_cycle_timer.timer); +		del_timer_sync(&llc->ack_timer.timer); +		del_timer_sync(&llc->rej_sent_timer.timer); +		del_timer_sync(&llc->busy_state_timer.timer); +	} else { +		del_timer(&llc->pf_cycle_timer.timer); +		del_timer(&llc->ack_timer.timer); +		del_timer(&llc->rej_sent_timer.timer); +		del_timer(&llc->busy_state_timer.timer); +	} + +	llc->ack_must_be_send = 0; +	llc->ack_pf = 0; +} +  /**   *	llc_sk_free - Frees a LLC socket   *	@sk - socket to free @@ -963,7 +983,7 @@ void llc_sk_free(struct sock *sk)  	llc->state = LLC_CONN_OUT_OF_SVC;  	/* Stop all (possibly) running timers */ -	llc_conn_ac_stop_all_timers(sk, NULL); +	llc_sk_stop_all_timers(sk, true);  #ifdef DEBUG_LLC_CONN_ALLOC  	printk(KERN_INFO "%s: unackq=%d, txq=%d\n", __func__,  		skb_queue_len(&llc->pdu_unack_q), diff --git a/net/packet/af_packet.c b/net/packet/af_packet.c index 92ca3e106c2b..f165514a4db5 100644 --- a/net/packet/af_packet.c +++ b/net/packet/af_packet.c @@ -332,11 +332,11 @@ static void packet_pick_tx_queue(struct net_device *dev, struct sk_buff *skb)  	skb_set_queue_mapping(skb, queue_index);  } -/* register_prot_hook must be invoked with the po->bind_lock held, +/* __register_prot_hook must be invoked through register_prot_hook   * or from a context in which asynchronous accesses to the packet   * socket is not possible (packet_create()).   */ -static void register_prot_hook(struct sock *sk) +static void __register_prot_hook(struct sock *sk)  {  	struct packet_sock *po = pkt_sk(sk); @@ -351,8 +351,13 @@ static void register_prot_hook(struct sock *sk)  	}  } -/* {,__}unregister_prot_hook() must be invoked with the po->bind_lock - * held.   If the sync parameter is true, we will temporarily drop +static void register_prot_hook(struct sock *sk) +{ +	lockdep_assert_held_once(&pkt_sk(sk)->bind_lock); +	__register_prot_hook(sk); +} + +/* If the sync parameter is true, we will temporarily drop   * the po->bind_lock and do a synchronize_net to make sure no   * asynchronous packet processing paths still refer to the elements   * of po->prot_hook.  If the sync parameter is false, it is the @@ -362,6 +367,8 @@ static void __unregister_prot_hook(struct sock *sk, bool sync)  {  	struct packet_sock *po = pkt_sk(sk); +	lockdep_assert_held_once(&po->bind_lock); +  	po->running = 0;  	if (po->fanout) @@ -2892,6 +2899,7 @@ static int packet_release(struct socket *sock)  	packet_flush_mclist(sk); +	lock_sock(sk);  	if (po->rx_ring.pg_vec) {  		memset(&req_u, 0, sizeof(req_u));  		packet_set_ring(sk, &req_u, 1, 0); @@ -2901,6 +2909,7 @@ static int packet_release(struct socket *sock)  		memset(&req_u, 0, sizeof(req_u));  		packet_set_ring(sk, &req_u, 1, 1);  	} +	release_sock(sk);  	f = fanout_release(sk); @@ -3134,7 +3143,7 @@ static int packet_create(struct net *net, struct socket *sock, int protocol,  	if (proto) {  		po->prot_hook.type = proto; -		register_prot_hook(sk); +		__register_prot_hook(sk);  	}  	mutex_lock(&net->packet.sklist_lock); @@ -3570,6 +3579,7 @@ packet_setsockopt(struct socket *sock, int level, int optname, char __user *optv  		union tpacket_req_u req_u;  		int len; +		lock_sock(sk);  		switch (po->tp_version) {  		case TPACKET_V1:  		case TPACKET_V2: @@ -3580,14 +3590,21 @@ packet_setsockopt(struct socket *sock, int level, int optname, char __user *optv  			len = sizeof(req_u.req3);  			break;  		} -		if (optlen < len) -			return -EINVAL; -		if (pkt_sk(sk)->has_vnet_hdr) -			return -EINVAL; -		if (copy_from_user(&req_u.req, optval, len)) -			return -EFAULT; -		return packet_set_ring(sk, &req_u, 0, -			optname == PACKET_TX_RING); +		if (optlen < len) { +			ret = -EINVAL; +		} else { +			if (pkt_sk(sk)->has_vnet_hdr) { +				ret = -EINVAL; +			} else { +				if (copy_from_user(&req_u.req, optval, len)) +					ret = -EFAULT; +				else +					ret = packet_set_ring(sk, &req_u, 0, +							      optname == PACKET_TX_RING); +			} +		} +		release_sock(sk); +		return ret;  	}  	case PACKET_COPY_THRESH:  	{ @@ -3653,12 +3670,18 @@ packet_setsockopt(struct socket *sock, int level, int optname, char __user *optv  		if (optlen != sizeof(val))  			return -EINVAL; -		if (po->rx_ring.pg_vec || po->tx_ring.pg_vec) -			return -EBUSY;  		if (copy_from_user(&val, optval, sizeof(val)))  			return -EFAULT; -		po->tp_loss = !!val; -		return 0; + +		lock_sock(sk); +		if (po->rx_ring.pg_vec || po->tx_ring.pg_vec) { +			ret = -EBUSY; +		} else { +			po->tp_loss = !!val; +			ret = 0; +		} +		release_sock(sk); +		return ret;  	}  	case PACKET_AUXDATA:  	{ @@ -3669,7 +3692,9 @@ packet_setsockopt(struct socket *sock, int level, int optname, char __user *optv  		if (copy_from_user(&val, optval, sizeof(val)))  			return -EFAULT; +		lock_sock(sk);  		po->auxdata = !!val; +		release_sock(sk);  		return 0;  	}  	case PACKET_ORIGDEV: @@ -3681,7 +3706,9 @@ packet_setsockopt(struct socket *sock, int level, int optname, char __user *optv  		if (copy_from_user(&val, optval, sizeof(val)))  			return -EFAULT; +		lock_sock(sk);  		po->origdev = !!val; +		release_sock(sk);  		return 0;  	}  	case PACKET_VNET_HDR: @@ -3690,15 +3717,20 @@ packet_setsockopt(struct socket *sock, int level, int optname, char __user *optv  		if (sock->type != SOCK_RAW)  			return -EINVAL; -		if (po->rx_ring.pg_vec || po->tx_ring.pg_vec) -			return -EBUSY;  		if (optlen < sizeof(val))  			return -EINVAL;  		if (copy_from_user(&val, optval, sizeof(val)))  			return -EFAULT; -		po->has_vnet_hdr = !!val; -		return 0; +		lock_sock(sk); +		if (po->rx_ring.pg_vec || po->tx_ring.pg_vec) { +			ret = -EBUSY; +		} else { +			po->has_vnet_hdr = !!val; +			ret = 0; +		} +		release_sock(sk); +		return ret;  	}  	case PACKET_TIMESTAMP:  	{ @@ -3736,11 +3768,17 @@ packet_setsockopt(struct socket *sock, int level, int optname, char __user *optv  		if (optlen != sizeof(val))  			return -EINVAL; -		if (po->rx_ring.pg_vec || po->tx_ring.pg_vec) -			return -EBUSY;  		if (copy_from_user(&val, optval, sizeof(val)))  			return -EFAULT; -		po->tp_tx_has_off = !!val; + +		lock_sock(sk); +		if (po->rx_ring.pg_vec || po->tx_ring.pg_vec) { +			ret = -EBUSY; +		} else { +			po->tp_tx_has_off = !!val; +			ret = 0; +		} +		release_sock(sk);  		return 0;  	}  	case PACKET_QDISC_BYPASS: @@ -4116,7 +4154,6 @@ static int packet_set_ring(struct sock *sk, union tpacket_req_u *req_u,  	/* Added to avoid minimal code churn */  	struct tpacket_req *req = &req_u->req; -	lock_sock(sk);  	/* Opening a Tx-ring is NOT supported in TPACKET_V3 */  	if (!closing && tx_ring && (po->tp_version > TPACKET_V2)) {  		WARN(1, "Tx-ring is not supported.\n"); @@ -4252,7 +4289,6 @@ static int packet_set_ring(struct sock *sk, union tpacket_req_u *req_u,  	if (pg_vec)  		free_pg_vec(pg_vec, order, req->tp_block_nr);  out: -	release_sock(sk);  	return err;  } diff --git a/net/packet/internal.h b/net/packet/internal.h index d55bfc34d6b3..1309e2a7baad 100644 --- a/net/packet/internal.h +++ b/net/packet/internal.h @@ -109,10 +109,12 @@ struct packet_sock {  	int			copy_thresh;  	spinlock_t		bind_lock;  	struct mutex		pg_vec_lock; -	unsigned int		running:1,	/* prot_hook is attached*/ -				auxdata:1, +	unsigned int		running;	/* bind_lock must be held */ +	unsigned int		auxdata:1,	/* writer must hold sock lock */  				origdev:1, -				has_vnet_hdr:1; +				has_vnet_hdr:1, +				tp_loss:1, +				tp_tx_has_off:1;  	int			pressure;  	int			ifindex;	/* bound device		*/  	__be16			num; @@ -122,8 +124,6 @@ struct packet_sock {  	enum tpacket_versions	tp_version;  	unsigned int		tp_hdrlen;  	unsigned int		tp_reserve; -	unsigned int		tp_loss:1; -	unsigned int		tp_tx_has_off:1;  	unsigned int		tp_tstamp;  	struct net_device __rcu	*cached_dev;  	int			(*xmit)(struct sk_buff *skb); diff --git a/net/sctp/ipv6.c b/net/sctp/ipv6.c index edb8514b4e00..1cd7b7e33fa3 100644 --- a/net/sctp/ipv6.c +++ b/net/sctp/ipv6.c @@ -519,46 +519,49 @@ static void sctp_v6_to_addr(union sctp_addr *addr, struct in6_addr *saddr,  	addr->v6.sin6_scope_id = 0;  } -/* Compare addresses exactly. - * v4-mapped-v6 is also in consideration. - */ -static int sctp_v6_cmp_addr(const union sctp_addr *addr1, -			    const union sctp_addr *addr2) +static int __sctp_v6_cmp_addr(const union sctp_addr *addr1, +			      const union sctp_addr *addr2)  {  	if (addr1->sa.sa_family != addr2->sa.sa_family) {  		if (addr1->sa.sa_family == AF_INET &&  		    addr2->sa.sa_family == AF_INET6 && -		    ipv6_addr_v4mapped(&addr2->v6.sin6_addr)) { -			if (addr2->v6.sin6_port == addr1->v4.sin_port && -			    addr2->v6.sin6_addr.s6_addr32[3] == -			    addr1->v4.sin_addr.s_addr) -				return 1; -		} +		    ipv6_addr_v4mapped(&addr2->v6.sin6_addr) && +		    addr2->v6.sin6_addr.s6_addr32[3] == +		    addr1->v4.sin_addr.s_addr) +			return 1; +  		if (addr2->sa.sa_family == AF_INET &&  		    addr1->sa.sa_family == AF_INET6 && -		    ipv6_addr_v4mapped(&addr1->v6.sin6_addr)) { -			if (addr1->v6.sin6_port == addr2->v4.sin_port && -			    addr1->v6.sin6_addr.s6_addr32[3] == -			    addr2->v4.sin_addr.s_addr) -				return 1; -		} +		    ipv6_addr_v4mapped(&addr1->v6.sin6_addr) && +		    addr1->v6.sin6_addr.s6_addr32[3] == +		    addr2->v4.sin_addr.s_addr) +			return 1; +  		return 0;  	} -	if (addr1->v6.sin6_port != addr2->v6.sin6_port) -		return 0; +  	if (!ipv6_addr_equal(&addr1->v6.sin6_addr, &addr2->v6.sin6_addr))  		return 0; +  	/* If this is a linklocal address, compare the scope_id. */ -	if (ipv6_addr_type(&addr1->v6.sin6_addr) & IPV6_ADDR_LINKLOCAL) { -		if (addr1->v6.sin6_scope_id && addr2->v6.sin6_scope_id && -		    (addr1->v6.sin6_scope_id != addr2->v6.sin6_scope_id)) { -			return 0; -		} -	} +	if ((ipv6_addr_type(&addr1->v6.sin6_addr) & IPV6_ADDR_LINKLOCAL) && +	    addr1->v6.sin6_scope_id && addr2->v6.sin6_scope_id && +	    addr1->v6.sin6_scope_id != addr2->v6.sin6_scope_id) +		return 0;  	return 1;  } +/* Compare addresses exactly. + * v4-mapped-v6 is also in consideration. + */ +static int sctp_v6_cmp_addr(const union sctp_addr *addr1, +			    const union sctp_addr *addr2) +{ +	return __sctp_v6_cmp_addr(addr1, addr2) && +	       addr1->v6.sin6_port == addr2->v6.sin6_port; +} +  /* Initialize addr struct to INADDR_ANY. */  static void sctp_v6_inaddr_any(union sctp_addr *addr, __be16 port)  { @@ -843,8 +846,8 @@ static int sctp_inet6_cmp_addr(const union sctp_addr *addr1,  			       const union sctp_addr *addr2,  			       struct sctp_sock *opt)  { -	struct sctp_af *af1, *af2;  	struct sock *sk = sctp_opt2sk(opt); +	struct sctp_af *af1, *af2;  	af1 = sctp_get_af_specific(addr1->sa.sa_family);  	af2 = sctp_get_af_specific(addr2->sa.sa_family); @@ -860,10 +863,7 @@ static int sctp_inet6_cmp_addr(const union sctp_addr *addr1,  	if (sctp_is_any(sk, addr1) || sctp_is_any(sk, addr2))  		return 1; -	if (addr1->sa.sa_family != addr2->sa.sa_family) -		return 0; - -	return af1->cmp_addr(addr1, addr2); +	return __sctp_v6_cmp_addr(addr1, addr2);  }  /* Verify that the provided sockaddr looks bindable.   Common verification, diff --git a/net/tipc/net.c b/net/tipc/net.c index 77bf9113c7a7..2763bd369b79 100644 --- a/net/tipc/net.c +++ b/net/tipc/net.c @@ -44,7 +44,8 @@  static const struct nla_policy tipc_nl_net_policy[TIPC_NLA_NET_MAX + 1] = {  	[TIPC_NLA_NET_UNSPEC]	= { .type = NLA_UNSPEC }, -	[TIPC_NLA_NET_ID]	= { .type = NLA_U32 } +	[TIPC_NLA_NET_ID]	= { .type = NLA_U32 }, +	[TIPC_NLA_NET_ADDR]	= { .type = NLA_U32 },  };  /* | 
