mirror of
				https://github.com/torvalds/linux.git
				synced 2025-10-31 16:48:26 +02:00 
			
		
		
		
	 1f9b37bfbb
			
		
	
	
		1f9b37bfbb
		
	
	
	
	
		
			
			clock_nanosleep() accepts absolute values of expiration time, if the TIMER_ABSTIME flag is set. This value is in the tasks time namespace, which has to be converted to the host time namespace. Co-developed-by: Dmitry Safonov <dima@arista.com> Signed-off-by: Andrei Vagin <avagin@openvz.org> Signed-off-by: Dmitry Safonov <dima@arista.com> Signed-off-by: Thomas Gleixner <tglx@linutronix.de> Link: https://lore.kernel.org/r/20191112012724.250792-18-dima@arista.com
		
			
				
	
	
		
			251 lines
		
	
	
	
		
			5.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			251 lines
		
	
	
	
		
			5.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0
 | |
| /*
 | |
|  * Dummy stubs used when CONFIG_POSIX_TIMERS=n
 | |
|  *
 | |
|  * Created by:  Nicolas Pitre, July 2016
 | |
|  * Copyright:   (C) 2016 Linaro Limited
 | |
|  */
 | |
| 
 | |
| #include <linux/linkage.h>
 | |
| #include <linux/kernel.h>
 | |
| #include <linux/sched.h>
 | |
| #include <linux/errno.h>
 | |
| #include <linux/syscalls.h>
 | |
| #include <linux/ktime.h>
 | |
| #include <linux/timekeeping.h>
 | |
| #include <linux/posix-timers.h>
 | |
| #include <linux/time_namespace.h>
 | |
| #include <linux/compat.h>
 | |
| 
 | |
| #ifdef CONFIG_ARCH_HAS_SYSCALL_WRAPPER
 | |
| /* Architectures may override SYS_NI and COMPAT_SYS_NI */
 | |
| #include <asm/syscall_wrapper.h>
 | |
| #endif
 | |
| 
 | |
| asmlinkage long sys_ni_posix_timers(void)
 | |
| {
 | |
| 	pr_err_once("process %d (%s) attempted a POSIX timer syscall "
 | |
| 		    "while CONFIG_POSIX_TIMERS is not set\n",
 | |
| 		    current->pid, current->comm);
 | |
| 	return -ENOSYS;
 | |
| }
 | |
| 
 | |
| #ifndef SYS_NI
 | |
| #define SYS_NI(name)  SYSCALL_ALIAS(sys_##name, sys_ni_posix_timers)
 | |
| #endif
 | |
| 
 | |
| #ifndef COMPAT_SYS_NI
 | |
| #define COMPAT_SYS_NI(name)  SYSCALL_ALIAS(compat_sys_##name, sys_ni_posix_timers)
 | |
| #endif
 | |
| 
 | |
| SYS_NI(timer_create);
 | |
| SYS_NI(timer_gettime);
 | |
| SYS_NI(timer_getoverrun);
 | |
| SYS_NI(timer_settime);
 | |
| SYS_NI(timer_delete);
 | |
| SYS_NI(clock_adjtime);
 | |
| SYS_NI(getitimer);
 | |
| SYS_NI(setitimer);
 | |
| SYS_NI(clock_adjtime32);
 | |
| #ifdef __ARCH_WANT_SYS_ALARM
 | |
| SYS_NI(alarm);
 | |
| #endif
 | |
| 
 | |
| /*
 | |
|  * We preserve minimal support for CLOCK_REALTIME and CLOCK_MONOTONIC
 | |
|  * as it is easy to remain compatible with little code. CLOCK_BOOTTIME
 | |
|  * is also included for convenience as at least systemd uses it.
 | |
|  */
 | |
| 
 | |
| SYSCALL_DEFINE2(clock_settime, const clockid_t, which_clock,
 | |
| 		const struct __kernel_timespec __user *, tp)
 | |
| {
 | |
| 	struct timespec64 new_tp;
 | |
| 
 | |
| 	if (which_clock != CLOCK_REALTIME)
 | |
| 		return -EINVAL;
 | |
| 	if (get_timespec64(&new_tp, tp))
 | |
| 		return -EFAULT;
 | |
| 
 | |
| 	return do_sys_settimeofday64(&new_tp, NULL);
 | |
| }
 | |
| 
 | |
| int do_clock_gettime(clockid_t which_clock, struct timespec64 *tp)
 | |
| {
 | |
| 	switch (which_clock) {
 | |
| 	case CLOCK_REALTIME:
 | |
| 		ktime_get_real_ts64(tp);
 | |
| 		break;
 | |
| 	case CLOCK_MONOTONIC:
 | |
| 		ktime_get_ts64(tp);
 | |
| 		timens_add_monotonic(tp);
 | |
| 		break;
 | |
| 	case CLOCK_BOOTTIME:
 | |
| 		ktime_get_boottime_ts64(tp);
 | |
| 		timens_add_boottime(tp);
 | |
| 		break;
 | |
| 	default:
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| SYSCALL_DEFINE2(clock_gettime, const clockid_t, which_clock,
 | |
| 		struct __kernel_timespec __user *, tp)
 | |
| {
 | |
| 	int ret;
 | |
| 	struct timespec64 kernel_tp;
 | |
| 
 | |
| 	ret = do_clock_gettime(which_clock, &kernel_tp);
 | |
| 	if (ret)
 | |
| 		return ret;
 | |
| 
 | |
| 	if (put_timespec64(&kernel_tp, tp))
 | |
| 		return -EFAULT;
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| SYSCALL_DEFINE2(clock_getres, const clockid_t, which_clock, struct __kernel_timespec __user *, tp)
 | |
| {
 | |
| 	struct timespec64 rtn_tp = {
 | |
| 		.tv_sec = 0,
 | |
| 		.tv_nsec = hrtimer_resolution,
 | |
| 	};
 | |
| 
 | |
| 	switch (which_clock) {
 | |
| 	case CLOCK_REALTIME:
 | |
| 	case CLOCK_MONOTONIC:
 | |
| 	case CLOCK_BOOTTIME:
 | |
| 		if (put_timespec64(&rtn_tp, tp))
 | |
| 			return -EFAULT;
 | |
| 		return 0;
 | |
| 	default:
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| SYSCALL_DEFINE4(clock_nanosleep, const clockid_t, which_clock, int, flags,
 | |
| 		const struct __kernel_timespec __user *, rqtp,
 | |
| 		struct __kernel_timespec __user *, rmtp)
 | |
| {
 | |
| 	struct timespec64 t;
 | |
| 	ktime_t texp;
 | |
| 
 | |
| 	switch (which_clock) {
 | |
| 	case CLOCK_REALTIME:
 | |
| 	case CLOCK_MONOTONIC:
 | |
| 	case CLOCK_BOOTTIME:
 | |
| 		break;
 | |
| 	default:
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 
 | |
| 	if (get_timespec64(&t, rqtp))
 | |
| 		return -EFAULT;
 | |
| 	if (!timespec64_valid(&t))
 | |
| 		return -EINVAL;
 | |
| 	if (flags & TIMER_ABSTIME)
 | |
| 		rmtp = NULL;
 | |
| 	current->restart_block.nanosleep.type = rmtp ? TT_NATIVE : TT_NONE;
 | |
| 	current->restart_block.nanosleep.rmtp = rmtp;
 | |
| 	texp = timespec64_to_ktime(t);
 | |
| 	if (flags & TIMER_ABSTIME)
 | |
| 		texp = timens_ktime_to_host(which_clock, texp);
 | |
| 	return hrtimer_nanosleep(texp, flags & TIMER_ABSTIME ?
 | |
| 				 HRTIMER_MODE_ABS : HRTIMER_MODE_REL,
 | |
| 				 which_clock);
 | |
| }
 | |
| 
 | |
| #ifdef CONFIG_COMPAT
 | |
| COMPAT_SYS_NI(timer_create);
 | |
| #endif
 | |
| 
 | |
| #if defined(CONFIG_COMPAT) || defined(CONFIG_ALPHA)
 | |
| COMPAT_SYS_NI(getitimer);
 | |
| COMPAT_SYS_NI(setitimer);
 | |
| #endif
 | |
| 
 | |
| #ifdef CONFIG_COMPAT_32BIT_TIME
 | |
| SYS_NI(timer_settime32);
 | |
| SYS_NI(timer_gettime32);
 | |
| 
 | |
| SYSCALL_DEFINE2(clock_settime32, const clockid_t, which_clock,
 | |
| 		struct old_timespec32 __user *, tp)
 | |
| {
 | |
| 	struct timespec64 new_tp;
 | |
| 
 | |
| 	if (which_clock != CLOCK_REALTIME)
 | |
| 		return -EINVAL;
 | |
| 	if (get_old_timespec32(&new_tp, tp))
 | |
| 		return -EFAULT;
 | |
| 
 | |
| 	return do_sys_settimeofday64(&new_tp, NULL);
 | |
| }
 | |
| 
 | |
| SYSCALL_DEFINE2(clock_gettime32, clockid_t, which_clock,
 | |
| 		struct old_timespec32 __user *, tp)
 | |
| {
 | |
| 	int ret;
 | |
| 	struct timespec64 kernel_tp;
 | |
| 
 | |
| 	ret = do_clock_gettime(which_clock, &kernel_tp);
 | |
| 	if (ret)
 | |
| 		return ret;
 | |
| 
 | |
| 	if (put_old_timespec32(&kernel_tp, tp))
 | |
| 		return -EFAULT;
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| SYSCALL_DEFINE2(clock_getres_time32, clockid_t, which_clock,
 | |
| 		struct old_timespec32 __user *, tp)
 | |
| {
 | |
| 	struct timespec64 rtn_tp = {
 | |
| 		.tv_sec = 0,
 | |
| 		.tv_nsec = hrtimer_resolution,
 | |
| 	};
 | |
| 
 | |
| 	switch (which_clock) {
 | |
| 	case CLOCK_REALTIME:
 | |
| 	case CLOCK_MONOTONIC:
 | |
| 	case CLOCK_BOOTTIME:
 | |
| 		if (put_old_timespec32(&rtn_tp, tp))
 | |
| 			return -EFAULT;
 | |
| 		return 0;
 | |
| 	default:
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| SYSCALL_DEFINE4(clock_nanosleep_time32, clockid_t, which_clock, int, flags,
 | |
| 		struct old_timespec32 __user *, rqtp,
 | |
| 		struct old_timespec32 __user *, rmtp)
 | |
| {
 | |
| 	struct timespec64 t;
 | |
| 	ktime_t texp;
 | |
| 
 | |
| 	switch (which_clock) {
 | |
| 	case CLOCK_REALTIME:
 | |
| 	case CLOCK_MONOTONIC:
 | |
| 	case CLOCK_BOOTTIME:
 | |
| 		break;
 | |
| 	default:
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 
 | |
| 	if (get_old_timespec32(&t, rqtp))
 | |
| 		return -EFAULT;
 | |
| 	if (!timespec64_valid(&t))
 | |
| 		return -EINVAL;
 | |
| 	if (flags & TIMER_ABSTIME)
 | |
| 		rmtp = NULL;
 | |
| 	current->restart_block.nanosleep.type = rmtp ? TT_COMPAT : TT_NONE;
 | |
| 	current->restart_block.nanosleep.compat_rmtp = rmtp;
 | |
| 	texp = timespec64_to_ktime(t);
 | |
| 	if (flags & TIMER_ABSTIME)
 | |
| 		texp = timens_ktime_to_host(which_clock, texp);
 | |
| 	return hrtimer_nanosleep(texp, flags & TIMER_ABSTIME ?
 | |
| 				 HRTIMER_MODE_ABS : HRTIMER_MODE_REL,
 | |
| 				 which_clock);
 | |
| }
 | |
| #endif
 |