Commit 4c1e34c0 authored by Richard Palethorpe's avatar Richard Palethorpe Committed by David S. Miller

vsock: Enable y2038 safe timeval for timeout

Reuse the timeval compat code from core/sock to handle 32-bit and
64-bit timeval structures. Also introduce a new socket option define
to allow using y2038 safe timeval under 32-bit.

The existing behavior of sock_set_timeout and vsock's timeout setter
differ when the time value is out of bounds. vsocks current behavior
is retained at the expense of not being able to share the full
implementation.

This allows the LTP test vsock01 to pass under 32-bit compat mode.

Fixes: fe0c72f3 ("socket: move compat timeout handling into sock.c")
Signed-off-by: default avatarRichard Palethorpe <rpalethorpe@suse.com>
Cc: Richard Palethorpe <rpalethorpe@richiejp.com>
Reviewed-by: default avatarArnd Bergmann <arnd@arndb.de>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 685c3f2f
...@@ -2836,4 +2836,8 @@ void sock_set_sndtimeo(struct sock *sk, s64 secs); ...@@ -2836,4 +2836,8 @@ void sock_set_sndtimeo(struct sock *sk, s64 secs);
int sock_bind_add(struct sock *sk, struct sockaddr *addr, int addr_len); int sock_bind_add(struct sock *sk, struct sockaddr *addr, int addr_len);
int sock_get_timeout(long timeo, void *optval, bool old_timeval);
int sock_copy_user_timeval(struct __kernel_sock_timeval *tv,
sockptr_t optval, int optlen, bool old_timeval);
#endif /* _SOCK_H */ #endif /* _SOCK_H */
...@@ -64,7 +64,7 @@ ...@@ -64,7 +64,7 @@
* timeout for a STREAM socket. * timeout for a STREAM socket.
*/ */
#define SO_VM_SOCKETS_CONNECT_TIMEOUT 6 #define SO_VM_SOCKETS_CONNECT_TIMEOUT_OLD 6
/* Option name for using non-blocking send/receive. Use as the option name /* Option name for using non-blocking send/receive. Use as the option name
* for setsockopt(3) or getsockopt(3) to set or get the non-blocking * for setsockopt(3) or getsockopt(3) to set or get the non-blocking
...@@ -81,6 +81,17 @@ ...@@ -81,6 +81,17 @@
#define SO_VM_SOCKETS_NONBLOCK_TXRX 7 #define SO_VM_SOCKETS_NONBLOCK_TXRX 7
#define SO_VM_SOCKETS_CONNECT_TIMEOUT_NEW 8
#if !defined(__KERNEL__)
#if __BITS_PER_LONG == 64 || (defined(__x86_64__) && defined(__ILP32__))
#define SO_VM_SOCKETS_CONNECT_TIMEOUT SO_VM_SOCKETS_CONNECT_TIMEOUT_OLD
#else
#define SO_VM_SOCKETS_CONNECT_TIMEOUT \
(sizeof(time_t) == sizeof(__kernel_long_t) ? SO_VM_SOCKETS_CONNECT_TIMEOUT_OLD : SO_VM_SOCKETS_CONNECT_TIMEOUT_NEW)
#endif
#endif
/* The vSocket equivalent of INADDR_ANY. This works for the svm_cid field of /* The vSocket equivalent of INADDR_ANY. This works for the svm_cid field of
* sockaddr_vm and indicates the context ID of the current endpoint. * sockaddr_vm and indicates the context ID of the current endpoint.
*/ */
......
...@@ -350,7 +350,7 @@ void sk_error_report(struct sock *sk) ...@@ -350,7 +350,7 @@ void sk_error_report(struct sock *sk)
} }
EXPORT_SYMBOL(sk_error_report); EXPORT_SYMBOL(sk_error_report);
static int sock_get_timeout(long timeo, void *optval, bool old_timeval) int sock_get_timeout(long timeo, void *optval, bool old_timeval)
{ {
struct __kernel_sock_timeval tv; struct __kernel_sock_timeval tv;
...@@ -379,12 +379,11 @@ static int sock_get_timeout(long timeo, void *optval, bool old_timeval) ...@@ -379,12 +379,11 @@ static int sock_get_timeout(long timeo, void *optval, bool old_timeval)
*(struct __kernel_sock_timeval *)optval = tv; *(struct __kernel_sock_timeval *)optval = tv;
return sizeof(tv); return sizeof(tv);
} }
EXPORT_SYMBOL(sock_get_timeout);
static int sock_set_timeout(long *timeo_p, sockptr_t optval, int optlen, int sock_copy_user_timeval(struct __kernel_sock_timeval *tv,
bool old_timeval) sockptr_t optval, int optlen, bool old_timeval)
{ {
struct __kernel_sock_timeval tv;
if (old_timeval && in_compat_syscall() && !COMPAT_USE_64BIT_TIME) { if (old_timeval && in_compat_syscall() && !COMPAT_USE_64BIT_TIME) {
struct old_timeval32 tv32; struct old_timeval32 tv32;
...@@ -393,8 +392,8 @@ static int sock_set_timeout(long *timeo_p, sockptr_t optval, int optlen, ...@@ -393,8 +392,8 @@ static int sock_set_timeout(long *timeo_p, sockptr_t optval, int optlen,
if (copy_from_sockptr(&tv32, optval, sizeof(tv32))) if (copy_from_sockptr(&tv32, optval, sizeof(tv32)))
return -EFAULT; return -EFAULT;
tv.tv_sec = tv32.tv_sec; tv->tv_sec = tv32.tv_sec;
tv.tv_usec = tv32.tv_usec; tv->tv_usec = tv32.tv_usec;
} else if (old_timeval) { } else if (old_timeval) {
struct __kernel_old_timeval old_tv; struct __kernel_old_timeval old_tv;
...@@ -402,14 +401,28 @@ static int sock_set_timeout(long *timeo_p, sockptr_t optval, int optlen, ...@@ -402,14 +401,28 @@ static int sock_set_timeout(long *timeo_p, sockptr_t optval, int optlen,
return -EINVAL; return -EINVAL;
if (copy_from_sockptr(&old_tv, optval, sizeof(old_tv))) if (copy_from_sockptr(&old_tv, optval, sizeof(old_tv)))
return -EFAULT; return -EFAULT;
tv.tv_sec = old_tv.tv_sec; tv->tv_sec = old_tv.tv_sec;
tv.tv_usec = old_tv.tv_usec; tv->tv_usec = old_tv.tv_usec;
} else { } else {
if (optlen < sizeof(tv)) if (optlen < sizeof(*tv))
return -EINVAL; return -EINVAL;
if (copy_from_sockptr(&tv, optval, sizeof(tv))) if (copy_from_sockptr(tv, optval, sizeof(*tv)))
return -EFAULT; return -EFAULT;
} }
return 0;
}
EXPORT_SYMBOL(sock_copy_user_timeval);
static int sock_set_timeout(long *timeo_p, sockptr_t optval, int optlen,
bool old_timeval)
{
struct __kernel_sock_timeval tv;
int err = sock_copy_user_timeval(&tv, optval, optlen, old_timeval);
if (err)
return err;
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;
......
...@@ -1614,13 +1614,18 @@ static int vsock_connectible_setsockopt(struct socket *sock, ...@@ -1614,13 +1614,18 @@ static int vsock_connectible_setsockopt(struct socket *sock,
vsock_update_buffer_size(vsk, transport, vsk->buffer_size); vsock_update_buffer_size(vsk, transport, vsk->buffer_size);
break; break;
case SO_VM_SOCKETS_CONNECT_TIMEOUT: { case SO_VM_SOCKETS_CONNECT_TIMEOUT_NEW:
struct __kernel_old_timeval tv; case SO_VM_SOCKETS_CONNECT_TIMEOUT_OLD: {
COPY_IN(tv); struct __kernel_sock_timeval tv;
err = sock_copy_user_timeval(&tv, optval, optlen,
optname == SO_VM_SOCKETS_CONNECT_TIMEOUT_OLD);
if (err)
break;
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)) {
vsk->connect_timeout = tv.tv_sec * HZ + vsk->connect_timeout = tv.tv_sec * HZ +
DIV_ROUND_UP(tv.tv_usec, (1000000 / HZ)); DIV_ROUND_UP((unsigned long)tv.tv_usec, (USEC_PER_SEC / HZ));
if (vsk->connect_timeout == 0) if (vsk->connect_timeout == 0)
vsk->connect_timeout = vsk->connect_timeout =
VSOCK_DEFAULT_CONNECT_TIMEOUT; VSOCK_DEFAULT_CONNECT_TIMEOUT;
...@@ -1653,7 +1658,9 @@ static int vsock_connectible_getsockopt(struct socket *sock, ...@@ -1653,7 +1658,9 @@ static int vsock_connectible_getsockopt(struct socket *sock,
union { union {
u64 val64; u64 val64;
struct old_timeval32 tm32;
struct __kernel_old_timeval tm; struct __kernel_old_timeval tm;
struct __kernel_sock_timeval stm;
} v; } v;
int lv = sizeof(v.val64); int lv = sizeof(v.val64);
...@@ -1680,12 +1687,10 @@ static int vsock_connectible_getsockopt(struct socket *sock, ...@@ -1680,12 +1687,10 @@ static int vsock_connectible_getsockopt(struct socket *sock,
v.val64 = vsk->buffer_min_size; v.val64 = vsk->buffer_min_size;
break; break;
case SO_VM_SOCKETS_CONNECT_TIMEOUT: case SO_VM_SOCKETS_CONNECT_TIMEOUT_NEW:
lv = sizeof(v.tm); case SO_VM_SOCKETS_CONNECT_TIMEOUT_OLD:
v.tm.tv_sec = vsk->connect_timeout / HZ; lv = sock_get_timeout(vsk->connect_timeout, &v,
v.tm.tv_usec = optname == SO_VM_SOCKETS_CONNECT_TIMEOUT_OLD);
(vsk->connect_timeout -
v.tm.tv_sec * HZ) * (1000000 / HZ);
break; break;
default: default:
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment