diff options
| author | Dmitry Shmidt <dimitrysh@google.com> | 2016-10-10 12:43:03 -0700 | 
|---|---|---|
| committer | Dmitry Shmidt <dimitrysh@google.com> | 2016-10-10 12:43:03 -0700 | 
| commit | 09f6247a9ccaffcc9af1e4a9038e14ecdb76a79d (patch) | |
| tree | f28f9f114c1aab70185f424ec2be689d1e6cbd3b /net | |
| parent | 8760f8e3d9806a940d759c9e12fff7eb7130b9ab (diff) | |
| parent | d19e48fe5da7b83d02ed4aec3567f08ae02a168c (diff) | |
Merge tag 'v4.4.23' into android-4.4.y
This is the 4.4.23 stable release
Diffstat (limited to 'net')
| -rw-r--r-- | net/bridge/br_multicast.c | 2 | ||||
| -rw-r--r-- | net/caif/cfpkt_skbuff.c | 2 | ||||
| -rw-r--r-- | net/core/dev.c | 16 | ||||
| -rw-r--r-- | net/ipv4/fib_trie.c | 8 | ||||
| -rw-r--r-- | net/ipv4/ip_vti.c | 31 | ||||
| -rw-r--r-- | net/ipv4/tcp_ipv4.c | 8 | ||||
| -rw-r--r-- | net/ipv4/tcp_yeah.c | 2 | ||||
| -rw-r--r-- | net/ipv6/ping.c | 9 | ||||
| -rw-r--r-- | net/ipv6/tcp_ipv6.c | 8 | ||||
| -rw-r--r-- | net/irda/iriap.c | 8 | ||||
| -rw-r--r-- | net/tipc/socket.c | 3 | ||||
| -rw-r--r-- | net/unix/af_unix.c | 107 | ||||
| -rw-r--r-- | net/wireless/nl80211.c | 2 | 
13 files changed, 129 insertions, 77 deletions
| diff --git a/net/bridge/br_multicast.c b/net/bridge/br_multicast.c index 7173a685309a..9542e84a9455 100644 --- a/net/bridge/br_multicast.c +++ b/net/bridge/br_multicast.c @@ -1113,7 +1113,7 @@ static int br_ip6_multicast_mld2_report(struct net_bridge *br,  		} else {  			err = br_ip6_multicast_add_group(br, port,  							 &grec->grec_mca, vid); -			if (!err) +			if (err)  				break;  		}  	} diff --git a/net/caif/cfpkt_skbuff.c b/net/caif/cfpkt_skbuff.c index f6c3b2137eea..59ce1fcc220c 100644 --- a/net/caif/cfpkt_skbuff.c +++ b/net/caif/cfpkt_skbuff.c @@ -286,7 +286,7 @@ int cfpkt_setlen(struct cfpkt *pkt, u16 len)  		else  			skb_trim(skb, len); -			return cfpkt_getlen(pkt); +		return cfpkt_getlen(pkt);  	}  	/* Need to expand SKB */ diff --git a/net/core/dev.c b/net/core/dev.c index 9efbdb3ff78a..de4ed2b5a221 100644 --- a/net/core/dev.c +++ b/net/core/dev.c @@ -3722,6 +3722,22 @@ static inline struct sk_buff *handle_ing(struct sk_buff *skb,  }  /** + *	netdev_is_rx_handler_busy - check if receive handler is registered + *	@dev: device to check + * + *	Check if a receive handler is already registered for a given device. + *	Return true if there one. + * + *	The caller must hold the rtnl_mutex. + */ +bool netdev_is_rx_handler_busy(struct net_device *dev) +{ +	ASSERT_RTNL(); +	return dev && rtnl_dereference(dev->rx_handler); +} +EXPORT_SYMBOL_GPL(netdev_is_rx_handler_busy); + +/**   *	netdev_rx_handler_register - register receive handler   *	@dev: device to register a handler for   *	@rx_handler: receive handler to register diff --git a/net/ipv4/fib_trie.c b/net/ipv4/fib_trie.c index 744e5936c10d..e5a3ff210fec 100644 --- a/net/ipv4/fib_trie.c +++ b/net/ipv4/fib_trie.c @@ -2453,9 +2453,7 @@ struct fib_route_iter {  static struct key_vector *fib_route_get_idx(struct fib_route_iter *iter,  					    loff_t pos)  { -	struct fib_table *tb = iter->main_tb;  	struct key_vector *l, **tp = &iter->tnode; -	struct trie *t;  	t_key key;  	/* use cache location of next-to-find key */ @@ -2463,8 +2461,6 @@ static struct key_vector *fib_route_get_idx(struct fib_route_iter *iter,  		pos -= iter->pos;  		key = iter->key;  	} else { -		t = (struct trie *)tb->tb_data; -		iter->tnode = t->kv;  		iter->pos = 0;  		key = 0;  	} @@ -2505,12 +2501,12 @@ static void *fib_route_seq_start(struct seq_file *seq, loff_t *pos)  		return NULL;  	iter->main_tb = tb; +	t = (struct trie *)tb->tb_data; +	iter->tnode = t->kv;  	if (*pos != 0)  		return fib_route_get_idx(iter, *pos); -	t = (struct trie *)tb->tb_data; -	iter->tnode = t->kv;  	iter->pos = 0;  	iter->key = 0; diff --git a/net/ipv4/ip_vti.c b/net/ipv4/ip_vti.c index 4d8f0b698777..65036891e080 100644 --- a/net/ipv4/ip_vti.c +++ b/net/ipv4/ip_vti.c @@ -540,6 +540,33 @@ static struct rtnl_link_ops vti_link_ops __read_mostly = {  	.get_link_net	= ip_tunnel_get_link_net,  }; +static bool is_vti_tunnel(const struct net_device *dev) +{ +	return dev->netdev_ops == &vti_netdev_ops; +} + +static int vti_device_event(struct notifier_block *unused, +			    unsigned long event, void *ptr) +{ +	struct net_device *dev = netdev_notifier_info_to_dev(ptr); +	struct ip_tunnel *tunnel = netdev_priv(dev); + +	if (!is_vti_tunnel(dev)) +		return NOTIFY_DONE; + +	switch (event) { +	case NETDEV_DOWN: +		if (!net_eq(tunnel->net, dev_net(dev))) +			xfrm_garbage_collect(tunnel->net); +		break; +	} +	return NOTIFY_DONE; +} + +static struct notifier_block vti_notifier_block __read_mostly = { +	.notifier_call = vti_device_event, +}; +  static int __init vti_init(void)  {  	const char *msg; @@ -547,6 +574,8 @@ static int __init vti_init(void)  	pr_info("IPv4 over IPsec tunneling driver\n"); +	register_netdevice_notifier(&vti_notifier_block); +  	msg = "tunnel device";  	err = register_pernet_device(&vti_net_ops);  	if (err < 0) @@ -579,6 +608,7 @@ xfrm_proto_ah_failed:  xfrm_proto_esp_failed:  	unregister_pernet_device(&vti_net_ops);  pernet_dev_failed: +	unregister_netdevice_notifier(&vti_notifier_block);  	pr_err("vti init: failed to register %s\n", msg);  	return err;  } @@ -590,6 +620,7 @@ static void __exit vti_fini(void)  	xfrm4_protocol_deregister(&vti_ah4_protocol, IPPROTO_AH);  	xfrm4_protocol_deregister(&vti_esp4_protocol, IPPROTO_ESP);  	unregister_pernet_device(&vti_net_ops); +	unregister_netdevice_notifier(&vti_notifier_block);  }  module_init(vti_init); diff --git a/net/ipv4/tcp_ipv4.c b/net/ipv4/tcp_ipv4.c index 7decaa439360..364ba22ef2ea 100644 --- a/net/ipv4/tcp_ipv4.c +++ b/net/ipv4/tcp_ipv4.c @@ -808,8 +808,14 @@ static void tcp_v4_reqsk_send_ack(const struct sock *sk, struct sk_buff *skb,  	u32 seq = (sk->sk_state == TCP_LISTEN) ? tcp_rsk(req)->snt_isn + 1 :  					     tcp_sk(sk)->snd_nxt; +	/* RFC 7323 2.3 +	 * The window field (SEG.WND) of every outgoing segment, with the +	 * exception of <SYN> segments, MUST be right-shifted by +	 * Rcv.Wind.Shift bits: +	 */  	tcp_v4_send_ack(sock_net(sk), skb, seq, -			tcp_rsk(req)->rcv_nxt, req->rsk_rcv_wnd, +			tcp_rsk(req)->rcv_nxt, +			req->rsk_rcv_wnd >> inet_rsk(req)->rcv_wscale,  			tcp_time_stamp,  			req->ts_recent,  			0, diff --git a/net/ipv4/tcp_yeah.c b/net/ipv4/tcp_yeah.c index 3e6a472e6b88..92ab5bc91592 100644 --- a/net/ipv4/tcp_yeah.c +++ b/net/ipv4/tcp_yeah.c @@ -75,7 +75,7 @@ static void tcp_yeah_cong_avoid(struct sock *sk, u32 ack, u32 acked)  	if (!tcp_is_cwnd_limited(sk))  		return; -	if (tp->snd_cwnd <= tp->snd_ssthresh) +	if (tcp_in_slow_start(tp))  		tcp_slow_start(tp, acked);  	else if (!yeah->doing_reno_now) { diff --git a/net/ipv6/ping.c b/net/ipv6/ping.c index 9411c8d770a5..fa65e92e9510 100644 --- a/net/ipv6/ping.c +++ b/net/ipv6/ping.c @@ -152,8 +152,10 @@ int ping_v6_sendmsg(struct sock *sk, struct msghdr *msg, size_t len)  	rt = (struct rt6_info *) dst;  	np = inet6_sk(sk); -	if (!np) -		return -EBADF; +	if (!np) { +		err = -EBADF; +		goto dst_err_out; +	}  	pfh.icmph.type = user_icmph.icmp6_type;  	pfh.icmph.code = user_icmph.icmp6_code; @@ -183,6 +185,9 @@ int ping_v6_sendmsg(struct sock *sk, struct msghdr *msg, size_t len)  	}  	release_sock(sk); +dst_err_out: +	dst_release(dst); +  	if (err)  		return err; diff --git a/net/ipv6/tcp_ipv6.c b/net/ipv6/tcp_ipv6.c index 3c6acb67d8e5..f58632cc45dc 100644 --- a/net/ipv6/tcp_ipv6.c +++ b/net/ipv6/tcp_ipv6.c @@ -933,9 +933,15 @@ static void tcp_v6_reqsk_send_ack(const struct sock *sk, struct sk_buff *skb,  	/* sk->sk_state == TCP_LISTEN -> for regular TCP_SYN_RECV  	 * sk->sk_state == TCP_SYN_RECV -> for Fast Open.  	 */ +	/* RFC 7323 2.3 +	 * The window field (SEG.WND) of every outgoing segment, with the +	 * exception of <SYN> segments, MUST be right-shifted by +	 * Rcv.Wind.Shift bits: +	 */  	tcp_v6_send_ack(sk, skb, (sk->sk_state == TCP_LISTEN) ?  			tcp_rsk(req)->snt_isn + 1 : tcp_sk(sk)->snd_nxt, -			tcp_rsk(req)->rcv_nxt, req->rsk_rcv_wnd, +			tcp_rsk(req)->rcv_nxt, +			req->rsk_rcv_wnd >> inet_rsk(req)->rcv_wscale,  			tcp_time_stamp, req->ts_recent, sk->sk_bound_dev_if,  			tcp_v6_md5_do_lookup(sk, &ipv6_hdr(skb)->daddr),  			0, 0); diff --git a/net/irda/iriap.c b/net/irda/iriap.c index 4a7ae32afa09..1138eaf5c682 100644 --- a/net/irda/iriap.c +++ b/net/irda/iriap.c @@ -185,8 +185,12 @@ struct iriap_cb *iriap_open(__u8 slsap_sel, int mode, void *priv,  	self->magic = IAS_MAGIC;  	self->mode = mode; -	if (mode == IAS_CLIENT) -		iriap_register_lsap(self, slsap_sel, mode); +	if (mode == IAS_CLIENT) { +		if (iriap_register_lsap(self, slsap_sel, mode)) { +			kfree(self); +			return NULL; +		} +	}  	self->confirm = callback;  	self->priv = priv; diff --git a/net/tipc/socket.c b/net/tipc/socket.c index 9b713e0ce00d..b26b7a127773 100644 --- a/net/tipc/socket.c +++ b/net/tipc/socket.c @@ -2111,7 +2111,8 @@ restart:  					      TIPC_CONN_MSG, SHORT_H_SIZE,  					      0, dnode, onode, dport, oport,  					      TIPC_CONN_SHUTDOWN); -			tipc_node_xmit_skb(net, skb, dnode, tsk->portid); +			if (skb) +				tipc_node_xmit_skb(net, skb, dnode, tsk->portid);  		}  		tsk->connected = 0;  		sock->state = SS_DISCONNECTING; diff --git a/net/unix/af_unix.c b/net/unix/af_unix.c index 6579fd6e7459..824cc1e160bc 100644 --- a/net/unix/af_unix.c +++ b/net/unix/af_unix.c @@ -661,11 +661,11 @@ static int unix_set_peek_off(struct sock *sk, int val)  {  	struct unix_sock *u = unix_sk(sk); -	if (mutex_lock_interruptible(&u->readlock)) +	if (mutex_lock_interruptible(&u->iolock))  		return -EINTR;  	sk->sk_peek_off = val; -	mutex_unlock(&u->readlock); +	mutex_unlock(&u->iolock);  	return 0;  } @@ -778,7 +778,8 @@ static struct sock *unix_create1(struct net *net, struct socket *sock, int kern)  	spin_lock_init(&u->lock);  	atomic_long_set(&u->inflight, 0);  	INIT_LIST_HEAD(&u->link); -	mutex_init(&u->readlock); /* single task reading lock */ +	mutex_init(&u->iolock); /* single task reading lock */ +	mutex_init(&u->bindlock); /* single task binding lock */  	init_waitqueue_head(&u->peer_wait);  	init_waitqueue_func_entry(&u->peer_wake, unix_dgram_peer_wake_relay);  	unix_insert_socket(unix_sockets_unbound(sk), sk); @@ -847,7 +848,7 @@ static int unix_autobind(struct socket *sock)  	int err;  	unsigned int retries = 0; -	err = mutex_lock_interruptible(&u->readlock); +	err = mutex_lock_interruptible(&u->bindlock);  	if (err)  		return err; @@ -894,7 +895,7 @@ retry:  	spin_unlock(&unix_table_lock);  	err = 0; -out:	mutex_unlock(&u->readlock); +out:	mutex_unlock(&u->bindlock);  	return err;  } @@ -953,20 +954,32 @@ fail:  	return NULL;  } -static int unix_mknod(struct dentry *dentry, struct path *path, umode_t mode, -		      struct path *res) +static int unix_mknod(const char *sun_path, umode_t mode, struct path *res)  { -	int err; +	struct dentry *dentry; +	struct path path; +	int err = 0; +	/* +	 * Get the parent directory, calculate the hash for last +	 * component. +	 */ +	dentry = kern_path_create(AT_FDCWD, sun_path, &path, 0); +	err = PTR_ERR(dentry); +	if (IS_ERR(dentry)) +		return err; -	err = security_path_mknod(path, dentry, mode, 0); +	/* +	 * All right, let's create it. +	 */ +	err = security_path_mknod(&path, dentry, mode, 0);  	if (!err) { -		err = vfs_mknod(d_inode(path->dentry), dentry, mode, 0); +		err = vfs_mknod(d_inode(path.dentry), dentry, mode, 0);  		if (!err) { -			res->mnt = mntget(path->mnt); +			res->mnt = mntget(path.mnt);  			res->dentry = dget(dentry);  		}  	} - +	done_path_create(&path, dentry);  	return err;  } @@ -977,12 +990,10 @@ static int unix_bind(struct socket *sock, struct sockaddr *uaddr, int addr_len)  	struct unix_sock *u = unix_sk(sk);  	struct sockaddr_un *sunaddr = (struct sockaddr_un *)uaddr;  	char *sun_path = sunaddr->sun_path; -	int err, name_err; +	int err;  	unsigned int hash;  	struct unix_address *addr;  	struct hlist_head *list; -	struct path path; -	struct dentry *dentry;  	err = -EINVAL;  	if (sunaddr->sun_family != AF_UNIX) @@ -998,34 +1009,14 @@ static int unix_bind(struct socket *sock, struct sockaddr *uaddr, int addr_len)  		goto out;  	addr_len = err; -	name_err = 0; -	dentry = NULL; -	if (sun_path[0]) { -		/* Get the parent directory, calculate the hash for last -		 * component. -		 */ -		dentry = kern_path_create(AT_FDCWD, sun_path, &path, 0); - -		if (IS_ERR(dentry)) { -			/* delay report until after 'already bound' check */ -			name_err = PTR_ERR(dentry); -			dentry = NULL; -		} -	} - -	err = mutex_lock_interruptible(&u->readlock); +	err = mutex_lock_interruptible(&u->bindlock);  	if (err) -		goto out_path; +		goto out;  	err = -EINVAL;  	if (u->addr)  		goto out_up; -	if (name_err) { -		err = name_err == -EEXIST ? -EADDRINUSE : name_err; -		goto out_up; -	} -  	err = -ENOMEM;  	addr = kmalloc(sizeof(*addr)+addr_len, GFP_KERNEL);  	if (!addr) @@ -1036,11 +1027,11 @@ static int unix_bind(struct socket *sock, struct sockaddr *uaddr, int addr_len)  	addr->hash = hash ^ sk->sk_type;  	atomic_set(&addr->refcnt, 1); -	if (dentry) { -		struct path u_path; +	if (sun_path[0]) { +		struct path path;  		umode_t mode = S_IFSOCK |  		       (SOCK_INODE(sock)->i_mode & ~current_umask()); -		err = unix_mknod(dentry, &path, mode, &u_path); +		err = unix_mknod(sun_path, mode, &path);  		if (err) {  			if (err == -EEXIST)  				err = -EADDRINUSE; @@ -1048,9 +1039,9 @@ static int unix_bind(struct socket *sock, struct sockaddr *uaddr, int addr_len)  			goto out_up;  		}  		addr->hash = UNIX_HASH_SIZE; -		hash = d_real_inode(dentry)->i_ino & (UNIX_HASH_SIZE - 1); +		hash = d_real_inode(path.dentry)->i_ino & (UNIX_HASH_SIZE - 1);  		spin_lock(&unix_table_lock); -		u->path = u_path; +		u->path = path;  		list = &unix_socket_table[hash];  	} else {  		spin_lock(&unix_table_lock); @@ -1072,11 +1063,7 @@ static int unix_bind(struct socket *sock, struct sockaddr *uaddr, int addr_len)  out_unlock:  	spin_unlock(&unix_table_lock);  out_up: -	mutex_unlock(&u->readlock); -out_path: -	if (dentry) -		done_path_create(&path, dentry); - +	mutex_unlock(&u->bindlock);  out:  	return err;  } @@ -1971,17 +1958,17 @@ static ssize_t unix_stream_sendpage(struct socket *socket, struct page *page,  	if (false) {  alloc_skb:  		unix_state_unlock(other); -		mutex_unlock(&unix_sk(other)->readlock); +		mutex_unlock(&unix_sk(other)->iolock);  		newskb = sock_alloc_send_pskb(sk, 0, 0, flags & MSG_DONTWAIT,  					      &err, 0);  		if (!newskb)  			goto err;  	} -	/* we must acquire readlock as we modify already present +	/* we must acquire iolock as we modify already present  	 * skbs in the sk_receive_queue and mess with skb->len  	 */ -	err = mutex_lock_interruptible(&unix_sk(other)->readlock); +	err = mutex_lock_interruptible(&unix_sk(other)->iolock);  	if (err) {  		err = flags & MSG_DONTWAIT ? -EAGAIN : -ERESTARTSYS;  		goto err; @@ -2048,7 +2035,7 @@ alloc_skb:  	}  	unix_state_unlock(other); -	mutex_unlock(&unix_sk(other)->readlock); +	mutex_unlock(&unix_sk(other)->iolock);  	other->sk_data_ready(other);  	scm_destroy(&scm); @@ -2057,7 +2044,7 @@ alloc_skb:  err_state_unlock:  	unix_state_unlock(other);  err_unlock: -	mutex_unlock(&unix_sk(other)->readlock); +	mutex_unlock(&unix_sk(other)->iolock);  err:  	kfree_skb(newskb);  	if (send_sigpipe && !(flags & MSG_NOSIGNAL)) @@ -2122,7 +2109,7 @@ static int unix_dgram_recvmsg(struct socket *sock, struct msghdr *msg,  	if (flags&MSG_OOB)  		goto out; -	err = mutex_lock_interruptible(&u->readlock); +	err = mutex_lock_interruptible(&u->iolock);  	if (unlikely(err)) {  		/* recvmsg() in non blocking mode is supposed to return -EAGAIN  		 * sk_rcvtimeo is not honored by mutex_lock_interruptible() @@ -2198,7 +2185,7 @@ static int unix_dgram_recvmsg(struct socket *sock, struct msghdr *msg,  out_free:  	skb_free_datagram(sk, skb);  out_unlock: -	mutex_unlock(&u->readlock); +	mutex_unlock(&u->iolock);  out:  	return err;  } @@ -2293,7 +2280,7 @@ static int unix_stream_read_generic(struct unix_stream_read_state *state)  	/* Lock the socket to prevent queue disordering  	 * while sleeps in memcpy_tomsg  	 */ -	mutex_lock(&u->readlock); +	mutex_lock(&u->iolock);  	if (flags & MSG_PEEK)  		skip = sk_peek_offset(sk, flags); @@ -2334,7 +2321,7 @@ again:  				break;  			} -			mutex_unlock(&u->readlock); +			mutex_unlock(&u->iolock);  			timeo = unix_stream_data_wait(sk, timeo, last,  						      last_len); @@ -2345,7 +2332,7 @@ again:  				goto out;  			} -			mutex_lock(&u->readlock); +			mutex_lock(&u->iolock);  			continue;  unlock:  			unix_state_unlock(sk); @@ -2448,7 +2435,7 @@ unlock:  		}  	} while (size); -	mutex_unlock(&u->readlock); +	mutex_unlock(&u->iolock);  	if (state->msg)  		scm_recv(sock, state->msg, &scm, flags);  	else @@ -2489,9 +2476,9 @@ static ssize_t skb_unix_socket_splice(struct sock *sk,  	int ret;  	struct unix_sock *u = unix_sk(sk); -	mutex_unlock(&u->readlock); +	mutex_unlock(&u->iolock);  	ret = splice_to_pipe(pipe, spd); -	mutex_lock(&u->readlock); +	mutex_lock(&u->iolock);  	return ret;  } diff --git a/net/wireless/nl80211.c b/net/wireless/nl80211.c index 5d89f13a98db..bf65f31bd55e 100644 --- a/net/wireless/nl80211.c +++ b/net/wireless/nl80211.c @@ -6628,7 +6628,7 @@ static int nl80211_channel_switch(struct sk_buff *skb, struct genl_info *info)  		params.n_counter_offsets_presp = len / sizeof(u16);  		if (rdev->wiphy.max_num_csa_counters && -		    (params.n_counter_offsets_beacon > +		    (params.n_counter_offsets_presp >  		     rdev->wiphy.max_num_csa_counters))  			return -EINVAL; | 
