mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-04 10:40:15 +02:00 
			
		
		
		
	socket: move compat timeout handling into sock.c
This is a cleanup to prepare for the addition of 64-bit time_t in O_SNDTIMEO/O_RCVTIMEO. The existing compat handler seems unnecessarily complex and error-prone, moving it all into the main setsockopt()/getsockopt() implementation requires half as much code and is easier to extend. 32-bit user space can now use old_timeval32 on both 32-bit and 64-bit machines, while 64-bit code can use __old_kernel_timeval. Signed-off-by: Arnd Bergmann <arnd@arndb.de> Signed-off-by: Deepa Dinamani <deepa.kernel@gmail.com> Acked-by: Willem de Bruijn <willemb@google.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
		
							parent
							
								
									a9bcfd1d17
								
							
						
					
					
						commit
						fe0c72f3db
					
				
					 3 changed files with 46 additions and 89 deletions
				
			
		
							
								
								
									
										66
									
								
								net/compat.c
									
									
									
									
									
								
							
							
						
						
									
										66
									
								
								net/compat.c
									
									
									
									
									
								
							| 
						 | 
					@ -348,28 +348,6 @@ static int do_set_attach_filter(struct socket *sock, int level, int optname,
 | 
				
			||||||
			      sizeof(struct sock_fprog));
 | 
								      sizeof(struct sock_fprog));
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static int do_set_sock_timeout(struct socket *sock, int level,
 | 
					 | 
				
			||||||
		int optname, char __user *optval, unsigned int optlen)
 | 
					 | 
				
			||||||
{
 | 
					 | 
				
			||||||
	struct compat_timeval __user *up = (struct compat_timeval __user *)optval;
 | 
					 | 
				
			||||||
	struct timeval ktime;
 | 
					 | 
				
			||||||
	mm_segment_t old_fs;
 | 
					 | 
				
			||||||
	int err;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	if (optlen < sizeof(*up))
 | 
					 | 
				
			||||||
		return -EINVAL;
 | 
					 | 
				
			||||||
	if (!access_ok(up, sizeof(*up)) ||
 | 
					 | 
				
			||||||
	    __get_user(ktime.tv_sec, &up->tv_sec) ||
 | 
					 | 
				
			||||||
	    __get_user(ktime.tv_usec, &up->tv_usec))
 | 
					 | 
				
			||||||
		return -EFAULT;
 | 
					 | 
				
			||||||
	old_fs = get_fs();
 | 
					 | 
				
			||||||
	set_fs(KERNEL_DS);
 | 
					 | 
				
			||||||
	err = sock_setsockopt(sock, level, optname, (char *)&ktime, sizeof(ktime));
 | 
					 | 
				
			||||||
	set_fs(old_fs);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	return err;
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
static int compat_sock_setsockopt(struct socket *sock, int level, int optname,
 | 
					static int compat_sock_setsockopt(struct socket *sock, int level, int optname,
 | 
				
			||||||
				char __user *optval, unsigned int optlen)
 | 
									char __user *optval, unsigned int optlen)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
| 
						 | 
					@ -377,10 +355,6 @@ static int compat_sock_setsockopt(struct socket *sock, int level, int optname,
 | 
				
			||||||
	    optname == SO_ATTACH_REUSEPORT_CBPF)
 | 
						    optname == SO_ATTACH_REUSEPORT_CBPF)
 | 
				
			||||||
		return do_set_attach_filter(sock, level, optname,
 | 
							return do_set_attach_filter(sock, level, optname,
 | 
				
			||||||
					    optval, optlen);
 | 
										    optval, optlen);
 | 
				
			||||||
	if (!COMPAT_USE_64BIT_TIME &&
 | 
					 | 
				
			||||||
	    (optname == SO_RCVTIMEO || optname == SO_SNDTIMEO))
 | 
					 | 
				
			||||||
		return do_set_sock_timeout(sock, level, optname, optval, optlen);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	return sock_setsockopt(sock, level, optname, optval, optlen);
 | 
						return sock_setsockopt(sock, level, optname, optval, optlen);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -417,44 +391,6 @@ COMPAT_SYSCALL_DEFINE5(setsockopt, int, fd, int, level, int, optname,
 | 
				
			||||||
	return __compat_sys_setsockopt(fd, level, optname, optval, optlen);
 | 
						return __compat_sys_setsockopt(fd, level, optname, optval, optlen);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static int do_get_sock_timeout(struct socket *sock, int level, int optname,
 | 
					 | 
				
			||||||
		char __user *optval, int __user *optlen)
 | 
					 | 
				
			||||||
{
 | 
					 | 
				
			||||||
	struct compat_timeval __user *up;
 | 
					 | 
				
			||||||
	struct timeval ktime;
 | 
					 | 
				
			||||||
	mm_segment_t old_fs;
 | 
					 | 
				
			||||||
	int len, err;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	up = (struct compat_timeval __user *) optval;
 | 
					 | 
				
			||||||
	if (get_user(len, optlen))
 | 
					 | 
				
			||||||
		return -EFAULT;
 | 
					 | 
				
			||||||
	if (len < sizeof(*up))
 | 
					 | 
				
			||||||
		return -EINVAL;
 | 
					 | 
				
			||||||
	len = sizeof(ktime);
 | 
					 | 
				
			||||||
	old_fs = get_fs();
 | 
					 | 
				
			||||||
	set_fs(KERNEL_DS);
 | 
					 | 
				
			||||||
	err = sock_getsockopt(sock, level, optname, (char *) &ktime, &len);
 | 
					 | 
				
			||||||
	set_fs(old_fs);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	if (!err) {
 | 
					 | 
				
			||||||
		if (put_user(sizeof(*up), optlen) ||
 | 
					 | 
				
			||||||
		    !access_ok(up, sizeof(*up)) ||
 | 
					 | 
				
			||||||
		    __put_user(ktime.tv_sec, &up->tv_sec) ||
 | 
					 | 
				
			||||||
		    __put_user(ktime.tv_usec, &up->tv_usec))
 | 
					 | 
				
			||||||
			err = -EFAULT;
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
	return err;
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
static int compat_sock_getsockopt(struct socket *sock, int level, int optname,
 | 
					 | 
				
			||||||
				char __user *optval, int __user *optlen)
 | 
					 | 
				
			||||||
{
 | 
					 | 
				
			||||||
	if (!COMPAT_USE_64BIT_TIME &&
 | 
					 | 
				
			||||||
	    (optname == SO_RCVTIMEO || optname == SO_SNDTIMEO))
 | 
					 | 
				
			||||||
		return do_get_sock_timeout(sock, level, optname, optval, optlen);
 | 
					 | 
				
			||||||
	return sock_getsockopt(sock, level, optname, optval, optlen);
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
int compat_sock_get_timestamp(struct sock *sk, struct timeval __user *userstamp)
 | 
					int compat_sock_get_timestamp(struct sock *sk, struct timeval __user *userstamp)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct compat_timeval __user *ctv;
 | 
						struct compat_timeval __user *ctv;
 | 
				
			||||||
| 
						 | 
					@ -527,7 +463,7 @@ static int __compat_sys_getsockopt(int fd, int level, int optname,
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		if (level == SOL_SOCKET)
 | 
							if (level == SOL_SOCKET)
 | 
				
			||||||
			err = compat_sock_getsockopt(sock, level,
 | 
								err = sock_getsockopt(sock, level,
 | 
				
			||||||
					optname, optval, optlen);
 | 
										optname, optval, optlen);
 | 
				
			||||||
		else if (sock->ops->compat_getsockopt)
 | 
							else if (sock->ops->compat_getsockopt)
 | 
				
			||||||
			err = sock->ops->compat_getsockopt(sock, level,
 | 
								err = sock->ops->compat_getsockopt(sock, level,
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -335,14 +335,48 @@ int __sk_backlog_rcv(struct sock *sk, struct sk_buff *skb)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
EXPORT_SYMBOL(__sk_backlog_rcv);
 | 
					EXPORT_SYMBOL(__sk_backlog_rcv);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static int sock_get_timeout(long timeo, void *optval)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						struct __kernel_old_timeval tv;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (timeo == MAX_SCHEDULE_TIMEOUT) {
 | 
				
			||||||
 | 
							tv.tv_sec = 0;
 | 
				
			||||||
 | 
							tv.tv_usec = 0;
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							tv.tv_sec = timeo / HZ;
 | 
				
			||||||
 | 
							tv.tv_usec = ((timeo % HZ) * USEC_PER_SEC) / HZ;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (in_compat_syscall() && !COMPAT_USE_64BIT_TIME) {
 | 
				
			||||||
 | 
							struct old_timeval32 tv32 = { tv.tv_sec, tv.tv_usec };
 | 
				
			||||||
 | 
							*(struct old_timeval32 *)optval = tv32;
 | 
				
			||||||
 | 
							return sizeof(tv32);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						*(struct __kernel_old_timeval *)optval = tv;
 | 
				
			||||||
 | 
						return sizeof(tv);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static int sock_set_timeout(long *timeo_p, char __user *optval, int optlen)
 | 
					static int sock_set_timeout(long *timeo_p, char __user *optval, int optlen)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct timeval tv;
 | 
						struct __kernel_old_timeval tv;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (in_compat_syscall() && !COMPAT_USE_64BIT_TIME) {
 | 
				
			||||||
 | 
							struct old_timeval32 tv32;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if (optlen < sizeof(tv32))
 | 
				
			||||||
 | 
								return -EINVAL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if (copy_from_user(&tv32, optval, sizeof(tv32)))
 | 
				
			||||||
 | 
								return -EFAULT;
 | 
				
			||||||
 | 
							tv.tv_sec = tv32.tv_sec;
 | 
				
			||||||
 | 
							tv.tv_usec = tv32.tv_usec;
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
		if (optlen < sizeof(tv))
 | 
							if (optlen < sizeof(tv))
 | 
				
			||||||
			return -EINVAL;
 | 
								return -EINVAL;
 | 
				
			||||||
		if (copy_from_user(&tv, optval, sizeof(tv)))
 | 
							if (copy_from_user(&tv, optval, sizeof(tv)))
 | 
				
			||||||
			return -EFAULT;
 | 
								return -EFAULT;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	if (tv.tv_usec < 0 || tv.tv_usec >= USEC_PER_SEC)
 | 
						if (tv.tv_usec < 0 || tv.tv_usec >= USEC_PER_SEC)
 | 
				
			||||||
		return -EDOM;
 | 
							return -EDOM;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1121,7 +1155,8 @@ int sock_getsockopt(struct socket *sock, int level, int optname,
 | 
				
			||||||
		int val;
 | 
							int val;
 | 
				
			||||||
		u64 val64;
 | 
							u64 val64;
 | 
				
			||||||
		struct linger ling;
 | 
							struct linger ling;
 | 
				
			||||||
		struct timeval tm;
 | 
							struct old_timeval32 tm32;
 | 
				
			||||||
 | 
							struct __kernel_old_timeval tm;
 | 
				
			||||||
		struct sock_txtime txtime;
 | 
							struct sock_txtime txtime;
 | 
				
			||||||
	} v;
 | 
						} v;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1222,25 +1257,11 @@ int sock_getsockopt(struct socket *sock, int level, int optname,
 | 
				
			||||||
		break;
 | 
							break;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	case SO_RCVTIMEO:
 | 
						case SO_RCVTIMEO:
 | 
				
			||||||
		lv = sizeof(struct timeval);
 | 
							lv = sock_get_timeout(sk->sk_rcvtimeo, &v);
 | 
				
			||||||
		if (sk->sk_rcvtimeo == MAX_SCHEDULE_TIMEOUT) {
 | 
					 | 
				
			||||||
			v.tm.tv_sec = 0;
 | 
					 | 
				
			||||||
			v.tm.tv_usec = 0;
 | 
					 | 
				
			||||||
		} else {
 | 
					 | 
				
			||||||
			v.tm.tv_sec = sk->sk_rcvtimeo / HZ;
 | 
					 | 
				
			||||||
			v.tm.tv_usec = ((sk->sk_rcvtimeo % HZ) * USEC_PER_SEC) / HZ;
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
		break;
 | 
							break;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	case SO_SNDTIMEO:
 | 
						case SO_SNDTIMEO:
 | 
				
			||||||
		lv = sizeof(struct timeval);
 | 
							lv = sock_get_timeout(sk->sk_sndtimeo, &v);
 | 
				
			||||||
		if (sk->sk_sndtimeo == MAX_SCHEDULE_TIMEOUT) {
 | 
					 | 
				
			||||||
			v.tm.tv_sec = 0;
 | 
					 | 
				
			||||||
			v.tm.tv_usec = 0;
 | 
					 | 
				
			||||||
		} else {
 | 
					 | 
				
			||||||
			v.tm.tv_sec = sk->sk_sndtimeo / HZ;
 | 
					 | 
				
			||||||
			v.tm.tv_usec = ((sk->sk_sndtimeo % HZ) * USEC_PER_SEC) / HZ;
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
		break;
 | 
							break;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	case SO_RCVLOWAT:
 | 
						case SO_RCVLOWAT:
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1439,7 +1439,7 @@ static int vsock_stream_setsockopt(struct socket *sock,
 | 
				
			||||||
		break;
 | 
							break;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	case SO_VM_SOCKETS_CONNECT_TIMEOUT: {
 | 
						case SO_VM_SOCKETS_CONNECT_TIMEOUT: {
 | 
				
			||||||
		struct timeval tv;
 | 
							struct __kernel_old_timeval tv;
 | 
				
			||||||
		COPY_IN(tv);
 | 
							COPY_IN(tv);
 | 
				
			||||||
		if (tv.tv_sec >= 0 && tv.tv_usec < USEC_PER_SEC &&
 | 
							if (tv.tv_sec >= 0 && tv.tv_usec < USEC_PER_SEC &&
 | 
				
			||||||
		    tv.tv_sec < (MAX_SCHEDULE_TIMEOUT / HZ - 1)) {
 | 
							    tv.tv_sec < (MAX_SCHEDULE_TIMEOUT / HZ - 1)) {
 | 
				
			||||||
| 
						 | 
					@ -1517,7 +1517,7 @@ static int vsock_stream_getsockopt(struct socket *sock,
 | 
				
			||||||
		break;
 | 
							break;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	case SO_VM_SOCKETS_CONNECT_TIMEOUT: {
 | 
						case SO_VM_SOCKETS_CONNECT_TIMEOUT: {
 | 
				
			||||||
		struct timeval tv;
 | 
							struct __kernel_old_timeval tv;
 | 
				
			||||||
		tv.tv_sec = vsk->connect_timeout / HZ;
 | 
							tv.tv_sec = vsk->connect_timeout / HZ;
 | 
				
			||||||
		tv.tv_usec =
 | 
							tv.tv_usec =
 | 
				
			||||||
		    (vsk->connect_timeout -
 | 
							    (vsk->connect_timeout -
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue