summaryrefslogtreecommitdiffstats
path: root/net/compat.c
diff options
context:
space:
mode:
authorArnd Bergmann <arnd@arndb.de>2019-04-17 16:51:48 -0400
committerDavid S. Miller <davem@davemloft.net>2019-04-19 17:07:40 -0400
commitc7cbdbf29f488a19982cd9f4a109887f18028bbb (patch)
tree74b39321c14b93411eda0939683dc9b8b713717b /net/compat.c
parent1ab839281cf72476988901a2606378d76530f99c (diff)
net: rework SIOCGSTAMP ioctl handling
The SIOCGSTAMP/SIOCGSTAMPNS ioctl commands are implemented by many socket protocol handlers, and all of those end up calling the same sock_get_timestamp()/sock_get_timestampns() helper functions, which results in a lot of duplicate code. With the introduction of 64-bit time_t on 32-bit architectures, this gets worse, as we then need four different ioctl commands in each socket protocol implementation. To simplify that, let's add a new .gettstamp() operation in struct proto_ops, and move ioctl implementation into the common sock_ioctl()/compat_sock_ioctl_trans() functions that these all go through. We can reuse the sock_get_timestamp() implementation, but generalize it so it can deal with both native and compat mode, as well as timeval and timespec structures. Acked-by: Stefan Schmidt <stefan@datenfreihafen.org> Acked-by: Neil Horman <nhorman@tuxdriver.com> Acked-by: Marc Kleine-Budde <mkl@pengutronix.de> Link: https://lore.kernel.org/lkml/CAK8P3a038aDQQotzua_QtKGhq8O9n+rdiz2=WDCp82ys8eUT+A@mail.gmail.com/ Signed-off-by: Arnd Bergmann <arnd@arndb.de> Acked-by: Willem de Bruijn <willemb@google.com> Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/compat.c')
-rw-r--r--net/compat.c57
1 files changed, 0 insertions, 57 deletions
diff --git a/net/compat.c b/net/compat.c
index eeea5eb71639..a031bd333092 100644
--- a/net/compat.c
+++ b/net/compat.c
@@ -395,63 +395,6 @@ COMPAT_SYSCALL_DEFINE5(setsockopt, int, fd, int, level, int, optname,
395 return __compat_sys_setsockopt(fd, level, optname, optval, optlen); 395 return __compat_sys_setsockopt(fd, level, optname, optval, optlen);
396} 396}
397 397
398int compat_sock_get_timestamp(struct sock *sk, struct timeval __user *userstamp)
399{
400 struct compat_timeval __user *ctv;
401 int err;
402 struct timeval tv;
403
404 if (COMPAT_USE_64BIT_TIME)
405 return sock_get_timestamp(sk, userstamp);
406
407 ctv = (struct compat_timeval __user *) userstamp;
408 err = -ENOENT;
409 sock_enable_timestamp(sk, SOCK_TIMESTAMP);
410 tv = ktime_to_timeval(sock_read_timestamp(sk));
411
412 if (tv.tv_sec == -1)
413 return err;
414 if (tv.tv_sec == 0) {
415 ktime_t kt = ktime_get_real();
416 sock_write_timestamp(sk, kt);
417 tv = ktime_to_timeval(kt);
418 }
419 err = 0;
420 if (put_user(tv.tv_sec, &ctv->tv_sec) ||
421 put_user(tv.tv_usec, &ctv->tv_usec))
422 err = -EFAULT;
423 return err;
424}
425EXPORT_SYMBOL(compat_sock_get_timestamp);
426
427int compat_sock_get_timestampns(struct sock *sk, struct timespec __user *userstamp)
428{
429 struct compat_timespec __user *ctv;
430 int err;
431 struct timespec ts;
432
433 if (COMPAT_USE_64BIT_TIME)
434 return sock_get_timestampns (sk, userstamp);
435
436 ctv = (struct compat_timespec __user *) userstamp;
437 err = -ENOENT;
438 sock_enable_timestamp(sk, SOCK_TIMESTAMP);
439 ts = ktime_to_timespec(sock_read_timestamp(sk));
440 if (ts.tv_sec == -1)
441 return err;
442 if (ts.tv_sec == 0) {
443 ktime_t kt = ktime_get_real();
444 sock_write_timestamp(sk, kt);
445 ts = ktime_to_timespec(kt);
446 }
447 err = 0;
448 if (put_user(ts.tv_sec, &ctv->tv_sec) ||
449 put_user(ts.tv_nsec, &ctv->tv_nsec))
450 err = -EFAULT;
451 return err;
452}
453EXPORT_SYMBOL(compat_sock_get_timestampns);
454
455static int __compat_sys_getsockopt(int fd, int level, int optname, 398static int __compat_sys_getsockopt(int fd, int level, int optname,
456 char __user *optval, 399 char __user *optval,
457 int __user *optlen) 400 int __user *optlen)