|
|
|
@ -22,6 +22,7 @@ |
|
|
|
|
#include <linux/anon_inodes.h> |
|
|
|
|
#include <linux/timerfd.h> |
|
|
|
|
#include <linux/syscalls.h> |
|
|
|
|
#include <linux/compat.h> |
|
|
|
|
#include <linux/rcupdate.h> |
|
|
|
|
|
|
|
|
|
struct timerfd_ctx { |
|
|
|
@ -278,21 +279,17 @@ SYSCALL_DEFINE2(timerfd_create, int, clockid, int, flags) |
|
|
|
|
return ufd; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
SYSCALL_DEFINE4(timerfd_settime, int, ufd, int, flags, |
|
|
|
|
const struct itimerspec __user *, utmr, |
|
|
|
|
struct itimerspec __user *, otmr) |
|
|
|
|
static int do_timerfd_settime(int ufd, int flags,
|
|
|
|
|
const struct itimerspec *new, |
|
|
|
|
struct itimerspec *old) |
|
|
|
|
{ |
|
|
|
|
struct fd f; |
|
|
|
|
struct timerfd_ctx *ctx; |
|
|
|
|
struct itimerspec ktmr, kotmr; |
|
|
|
|
int ret; |
|
|
|
|
|
|
|
|
|
if (copy_from_user(&ktmr, utmr, sizeof(ktmr))) |
|
|
|
|
return -EFAULT; |
|
|
|
|
|
|
|
|
|
if ((flags & ~TFD_SETTIME_FLAGS) || |
|
|
|
|
!timespec_valid(&ktmr.it_value) || |
|
|
|
|
!timespec_valid(&ktmr.it_interval)) |
|
|
|
|
!timespec_valid(&new->it_value) || |
|
|
|
|
!timespec_valid(&new->it_interval)) |
|
|
|
|
return -EINVAL; |
|
|
|
|
|
|
|
|
|
ret = timerfd_fget(ufd, &f); |
|
|
|
@ -323,27 +320,23 @@ SYSCALL_DEFINE4(timerfd_settime, int, ufd, int, flags, |
|
|
|
|
if (ctx->expired && ctx->tintv.tv64) |
|
|
|
|
hrtimer_forward_now(&ctx->tmr, ctx->tintv); |
|
|
|
|
|
|
|
|
|
kotmr.it_value = ktime_to_timespec(timerfd_get_remaining(ctx)); |
|
|
|
|
kotmr.it_interval = ktime_to_timespec(ctx->tintv); |
|
|
|
|
old->it_value = ktime_to_timespec(timerfd_get_remaining(ctx)); |
|
|
|
|
old->it_interval = ktime_to_timespec(ctx->tintv); |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Re-program the timer to the new value ... |
|
|
|
|
*/ |
|
|
|
|
ret = timerfd_setup(ctx, flags, &ktmr); |
|
|
|
|
ret = timerfd_setup(ctx, flags, new); |
|
|
|
|
|
|
|
|
|
spin_unlock_irq(&ctx->wqh.lock); |
|
|
|
|
fdput(f); |
|
|
|
|
if (otmr && copy_to_user(otmr, &kotmr, sizeof(kotmr))) |
|
|
|
|
return -EFAULT; |
|
|
|
|
|
|
|
|
|
return ret; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
SYSCALL_DEFINE2(timerfd_gettime, int, ufd, struct itimerspec __user *, otmr) |
|
|
|
|
static int do_timerfd_gettime(int ufd, struct itimerspec *t) |
|
|
|
|
{ |
|
|
|
|
struct fd f; |
|
|
|
|
struct timerfd_ctx *ctx; |
|
|
|
|
struct itimerspec kotmr; |
|
|
|
|
int ret = timerfd_fget(ufd, &f); |
|
|
|
|
if (ret) |
|
|
|
|
return ret; |
|
|
|
@ -356,11 +349,65 @@ SYSCALL_DEFINE2(timerfd_gettime, int, ufd, struct itimerspec __user *, otmr) |
|
|
|
|
hrtimer_forward_now(&ctx->tmr, ctx->tintv) - 1; |
|
|
|
|
hrtimer_restart(&ctx->tmr); |
|
|
|
|
} |
|
|
|
|
kotmr.it_value = ktime_to_timespec(timerfd_get_remaining(ctx)); |
|
|
|
|
kotmr.it_interval = ktime_to_timespec(ctx->tintv); |
|
|
|
|
t->it_value = ktime_to_timespec(timerfd_get_remaining(ctx)); |
|
|
|
|
t->it_interval = ktime_to_timespec(ctx->tintv); |
|
|
|
|
spin_unlock_irq(&ctx->wqh.lock); |
|
|
|
|
fdput(f); |
|
|
|
|
return 0; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
SYSCALL_DEFINE4(timerfd_settime, int, ufd, int, flags, |
|
|
|
|
const struct itimerspec __user *, utmr, |
|
|
|
|
struct itimerspec __user *, otmr) |
|
|
|
|
{ |
|
|
|
|
struct itimerspec new, old; |
|
|
|
|
int ret; |
|
|
|
|
|
|
|
|
|
if (copy_from_user(&new, utmr, sizeof(new))) |
|
|
|
|
return -EFAULT; |
|
|
|
|
ret = do_timerfd_settime(ufd, flags, &new, &old); |
|
|
|
|
if (ret) |
|
|
|
|
return ret; |
|
|
|
|
if (otmr && copy_to_user(otmr, &old, sizeof(old))) |
|
|
|
|
return -EFAULT; |
|
|
|
|
|
|
|
|
|
return ret; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
SYSCALL_DEFINE2(timerfd_gettime, int, ufd, struct itimerspec __user *, otmr) |
|
|
|
|
{ |
|
|
|
|
struct itimerspec kotmr; |
|
|
|
|
int ret = do_timerfd_gettime(ufd, &kotmr); |
|
|
|
|
if (ret) |
|
|
|
|
return ret; |
|
|
|
|
return copy_to_user(otmr, &kotmr, sizeof(kotmr)) ? -EFAULT: 0; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
#ifdef COMPAT |
|
|
|
|
COMPAT_SYSCALL_DEFINE4(timerfd_settime, int, ufd, int, flags, |
|
|
|
|
const struct itimerspec __user *, utmr, |
|
|
|
|
struct itimerspec __user *, otmr) |
|
|
|
|
{ |
|
|
|
|
struct itimerspec new, old; |
|
|
|
|
int ret; |
|
|
|
|
|
|
|
|
|
if (get_compat_itimerspec(&new, utmr)) |
|
|
|
|
return -EFAULT; |
|
|
|
|
ret = do_timerfd_settime(ufd, flags, &new, &old); |
|
|
|
|
if (ret) |
|
|
|
|
return ret; |
|
|
|
|
if (otmr && put_compat_itimerspec(otmr, &old)) |
|
|
|
|
return -EFAULT; |
|
|
|
|
return ret; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
COMPAT_SYSCALL_DEFINE2(timerfd_gettime, int, ufd, |
|
|
|
|
struct itimerspec __user *, otmr) |
|
|
|
|
{ |
|
|
|
|
struct itimerspec kotmr; |
|
|
|
|
int ret = do_timerfd_gettime(ufd, &kotmr); |
|
|
|
|
if (ret) |
|
|
|
|
return ret; |
|
|
|
|
return put_compat_itimerspec(otmr, &t) ? -EFAULT: 0; |
|
|
|
|
} |
|
|
|
|
#endif |
|
|
|
|