blob: fcb3b21d8bdcd444f32165f98dd9fe078b3d2446 [file] [log] [blame]
Thomas Gleixner35728b82018-10-31 19:21:09 +01001// SPDX-License-Identifier: GPL-2.0
Nicolas Pitrebaa73d92016-11-11 00:10:10 -05002/*
3 * Dummy stubs used when CONFIG_POSIX_TIMERS=n
4 *
5 * Created by: Nicolas Pitre, July 2016
6 * Copyright: (C) 2016 Linaro Limited
Nicolas Pitrebaa73d92016-11-11 00:10:10 -05007 */
8
9#include <linux/linkage.h>
10#include <linux/kernel.h>
11#include <linux/sched.h>
12#include <linux/errno.h>
13#include <linux/syscalls.h>
14#include <linux/ktime.h>
15#include <linux/timekeeping.h>
16#include <linux/posix-timers.h>
Andrei Vagin5a590f32019-11-12 01:27:00 +000017#include <linux/time_namespace.h>
Al Viroedbeda42017-06-07 09:42:31 +010018#include <linux/compat.h>
Nicolas Pitrebaa73d92016-11-11 00:10:10 -050019
Dominik Brodowski7303e302018-04-05 11:53:03 +020020#ifdef CONFIG_ARCH_HAS_SYSCALL_WRAPPER
21/* Architectures may override SYS_NI and COMPAT_SYS_NI */
22#include <asm/syscall_wrapper.h>
23#endif
24
Nicolas Pitrebaa73d92016-11-11 00:10:10 -050025asmlinkage long sys_ni_posix_timers(void)
26{
27 pr_err_once("process %d (%s) attempted a POSIX timer syscall "
28 "while CONFIG_POSIX_TIMERS is not set\n",
29 current->pid, current->comm);
30 return -ENOSYS;
31}
32
Dominik Brodowski7303e302018-04-05 11:53:03 +020033#ifndef SYS_NI
Nicolas Pitrebaa73d92016-11-11 00:10:10 -050034#define SYS_NI(name) SYSCALL_ALIAS(sys_##name, sys_ni_posix_timers)
Dominik Brodowski7303e302018-04-05 11:53:03 +020035#endif
36
37#ifndef COMPAT_SYS_NI
Al Viro3a4d44b2017-06-07 09:42:34 +010038#define COMPAT_SYS_NI(name) SYSCALL_ALIAS(compat_sys_##name, sys_ni_posix_timers)
Dominik Brodowski7303e302018-04-05 11:53:03 +020039#endif
Nicolas Pitrebaa73d92016-11-11 00:10:10 -050040
41SYS_NI(timer_create);
42SYS_NI(timer_gettime);
43SYS_NI(timer_getoverrun);
44SYS_NI(timer_settime);
45SYS_NI(timer_delete);
46SYS_NI(clock_adjtime);
47SYS_NI(getitimer);
48SYS_NI(setitimer);
Arnd Bergmann8dabe722019-01-07 00:33:08 +010049SYS_NI(clock_adjtime32);
Nicolas Pitrebaa73d92016-11-11 00:10:10 -050050#ifdef __ARCH_WANT_SYS_ALARM
51SYS_NI(alarm);
52#endif
53
54/*
55 * We preserve minimal support for CLOCK_REALTIME and CLOCK_MONOTONIC
56 * as it is easy to remain compatible with little code. CLOCK_BOOTTIME
57 * is also included for convenience as at least systemd uses it.
58 */
59
60SYSCALL_DEFINE2(clock_settime, const clockid_t, which_clock,
Deepa Dinamani6d5b8412018-03-13 21:03:32 -070061 const struct __kernel_timespec __user *, tp)
Nicolas Pitrebaa73d92016-11-11 00:10:10 -050062{
Deepa Dinamani5c499412017-06-24 11:45:05 -070063 struct timespec64 new_tp;
Nicolas Pitrebaa73d92016-11-11 00:10:10 -050064
65 if (which_clock != CLOCK_REALTIME)
66 return -EINVAL;
Deepa Dinamani5c499412017-06-24 11:45:05 -070067 if (get_timespec64(&new_tp, tp))
Nicolas Pitrebaa73d92016-11-11 00:10:10 -050068 return -EFAULT;
Deepa Dinamani2ac00f12017-03-26 12:04:12 -070069
Deepa Dinamani5c499412017-06-24 11:45:05 -070070 return do_sys_settimeofday64(&new_tp, NULL);
Nicolas Pitrebaa73d92016-11-11 00:10:10 -050071}
72
Deepa Dinamani5c499412017-06-24 11:45:05 -070073int do_clock_gettime(clockid_t which_clock, struct timespec64 *tp)
Nicolas Pitrebaa73d92016-11-11 00:10:10 -050074{
Nicolas Pitrebaa73d92016-11-11 00:10:10 -050075 switch (which_clock) {
Deepa Dinamani5c499412017-06-24 11:45:05 -070076 case CLOCK_REALTIME:
77 ktime_get_real_ts64(tp);
78 break;
79 case CLOCK_MONOTONIC:
80 ktime_get_ts64(tp);
Andrei Vagin5a590f32019-11-12 01:27:00 +000081 timens_add_monotonic(tp);
Deepa Dinamani5c499412017-06-24 11:45:05 -070082 break;
83 case CLOCK_BOOTTIME:
Arnd Bergmann58a10452018-06-18 16:32:24 +020084 ktime_get_boottime_ts64(tp);
Andrei Vagin5a590f32019-11-12 01:27:00 +000085 timens_add_boottime(tp);
Deepa Dinamani5c499412017-06-24 11:45:05 -070086 break;
87 default:
88 return -EINVAL;
Nicolas Pitrebaa73d92016-11-11 00:10:10 -050089 }
Deepa Dinamani3c9c12f2017-03-26 12:04:14 -070090
Deepa Dinamani5c499412017-06-24 11:45:05 -070091 return 0;
92}
93SYSCALL_DEFINE2(clock_gettime, const clockid_t, which_clock,
Deepa Dinamani6d5b8412018-03-13 21:03:32 -070094 struct __kernel_timespec __user *, tp)
Deepa Dinamani5c499412017-06-24 11:45:05 -070095{
96 int ret;
97 struct timespec64 kernel_tp;
98
99 ret = do_clock_gettime(which_clock, &kernel_tp);
100 if (ret)
101 return ret;
102
103 if (put_timespec64(&kernel_tp, tp))
Nicolas Pitrebaa73d92016-11-11 00:10:10 -0500104 return -EFAULT;
105 return 0;
106}
107
Deepa Dinamani6d5b8412018-03-13 21:03:32 -0700108SYSCALL_DEFINE2(clock_getres, const clockid_t, which_clock, struct __kernel_timespec __user *, tp)
Nicolas Pitrebaa73d92016-11-11 00:10:10 -0500109{
Deepa Dinamani5c499412017-06-24 11:45:05 -0700110 struct timespec64 rtn_tp = {
Nicolas Pitrebaa73d92016-11-11 00:10:10 -0500111 .tv_sec = 0,
112 .tv_nsec = hrtimer_resolution,
113 };
114
115 switch (which_clock) {
116 case CLOCK_REALTIME:
117 case CLOCK_MONOTONIC:
118 case CLOCK_BOOTTIME:
Deepa Dinamani5c499412017-06-24 11:45:05 -0700119 if (put_timespec64(&rtn_tp, tp))
Nicolas Pitrebaa73d92016-11-11 00:10:10 -0500120 return -EFAULT;
121 return 0;
122 default:
123 return -EINVAL;
124 }
125}
126
127SYSCALL_DEFINE4(clock_nanosleep, const clockid_t, which_clock, int, flags,
Deepa Dinamani01909972018-03-13 21:03:33 -0700128 const struct __kernel_timespec __user *, rqtp,
129 struct __kernel_timespec __user *, rmtp)
Nicolas Pitrebaa73d92016-11-11 00:10:10 -0500130{
Arnd Bergmannfe460422017-10-13 20:29:38 +0200131 struct timespec64 t;
Andrei Vagin1f9b37b2019-11-12 01:27:06 +0000132 ktime_t texp;
Nicolas Pitrebaa73d92016-11-11 00:10:10 -0500133
134 switch (which_clock) {
135 case CLOCK_REALTIME:
136 case CLOCK_MONOTONIC:
137 case CLOCK_BOOTTIME:
Al Viroedbeda42017-06-07 09:42:31 +0100138 break;
Nicolas Pitrebaa73d92016-11-11 00:10:10 -0500139 default:
140 return -EINVAL;
141 }
Al Viroedbeda42017-06-07 09:42:31 +0100142
Arnd Bergmannfe460422017-10-13 20:29:38 +0200143 if (get_timespec64(&t, rqtp))
Al Viroedbeda42017-06-07 09:42:31 +0100144 return -EFAULT;
Arnd Bergmannfe460422017-10-13 20:29:38 +0200145 if (!timespec64_valid(&t))
Al Viroedbeda42017-06-07 09:42:31 +0100146 return -EINVAL;
147 if (flags & TIMER_ABSTIME)
148 rmtp = NULL;
149 current->restart_block.nanosleep.type = rmtp ? TT_NATIVE : TT_NONE;
150 current->restart_block.nanosleep.rmtp = rmtp;
Andrei Vagin1f9b37b2019-11-12 01:27:06 +0000151 texp = timespec64_to_ktime(t);
152 if (flags & TIMER_ABSTIME)
153 texp = timens_ktime_to_host(which_clock, texp);
154 return hrtimer_nanosleep(texp, flags & TIMER_ABSTIME ?
Al Viroedbeda42017-06-07 09:42:31 +0100155 HRTIMER_MODE_ABS : HRTIMER_MODE_REL,
156 which_clock);
Nicolas Pitrebaa73d92016-11-11 00:10:10 -0500157}
158
159#ifdef CONFIG_COMPAT
Deepa Dinamani63a766a2017-06-24 11:45:04 -0700160COMPAT_SYS_NI(timer_create);
Arnd Bergmannf35deaf2019-12-07 20:10:26 +0100161#endif
162
163#if defined(CONFIG_COMPAT) || defined(CONFIG_ALPHA)
Deepa Dinamani63a766a2017-06-24 11:45:04 -0700164COMPAT_SYS_NI(getitimer);
165COMPAT_SYS_NI(setitimer);
Deepa Dinamanib5793b02018-03-13 21:03:29 -0700166#endif
Deepa Dinamani63a766a2017-06-24 11:45:04 -0700167
Deepa Dinamanib5793b02018-03-13 21:03:29 -0700168#ifdef CONFIG_COMPAT_32BIT_TIME
Arnd Bergmann8dabe722019-01-07 00:33:08 +0100169SYS_NI(timer_settime32);
170SYS_NI(timer_gettime32);
171
172SYSCALL_DEFINE2(clock_settime32, const clockid_t, which_clock,
173 struct old_timespec32 __user *, tp)
Al Virod822cdc2017-06-07 09:42:38 +0100174{
Deepa Dinamani5c499412017-06-24 11:45:05 -0700175 struct timespec64 new_tp;
Al Virod822cdc2017-06-07 09:42:38 +0100176
177 if (which_clock != CLOCK_REALTIME)
178 return -EINVAL;
Arnd Bergmann9afc5ee2018-07-13 12:52:28 +0200179 if (get_old_timespec32(&new_tp, tp))
Al Virod822cdc2017-06-07 09:42:38 +0100180 return -EFAULT;
181
Deepa Dinamani5c499412017-06-24 11:45:05 -0700182 return do_sys_settimeofday64(&new_tp, NULL);
Al Virod822cdc2017-06-07 09:42:38 +0100183}
184
Arnd Bergmann8dabe722019-01-07 00:33:08 +0100185SYSCALL_DEFINE2(clock_gettime32, clockid_t, which_clock,
186 struct old_timespec32 __user *, tp)
Al Virod822cdc2017-06-07 09:42:38 +0100187{
Deepa Dinamani5c499412017-06-24 11:45:05 -0700188 int ret;
189 struct timespec64 kernel_tp;
Al Virod822cdc2017-06-07 09:42:38 +0100190
Deepa Dinamani5c499412017-06-24 11:45:05 -0700191 ret = do_clock_gettime(which_clock, &kernel_tp);
192 if (ret)
193 return ret;
Al Virod822cdc2017-06-07 09:42:38 +0100194
Arnd Bergmann9afc5ee2018-07-13 12:52:28 +0200195 if (put_old_timespec32(&kernel_tp, tp))
Al Virod822cdc2017-06-07 09:42:38 +0100196 return -EFAULT;
197 return 0;
198}
199
Arnd Bergmann8dabe722019-01-07 00:33:08 +0100200SYSCALL_DEFINE2(clock_getres_time32, clockid_t, which_clock,
201 struct old_timespec32 __user *, tp)
Al Virod822cdc2017-06-07 09:42:38 +0100202{
Deepa Dinamani5c499412017-06-24 11:45:05 -0700203 struct timespec64 rtn_tp = {
Al Virod822cdc2017-06-07 09:42:38 +0100204 .tv_sec = 0,
205 .tv_nsec = hrtimer_resolution,
206 };
207
208 switch (which_clock) {
209 case CLOCK_REALTIME:
210 case CLOCK_MONOTONIC:
211 case CLOCK_BOOTTIME:
Arnd Bergmann9afc5ee2018-07-13 12:52:28 +0200212 if (put_old_timespec32(&rtn_tp, tp))
Al Virod822cdc2017-06-07 09:42:38 +0100213 return -EFAULT;
214 return 0;
215 default:
216 return -EINVAL;
217 }
218}
Deepa Dinamani5c499412017-06-24 11:45:05 -0700219
Arnd Bergmann8dabe722019-01-07 00:33:08 +0100220SYSCALL_DEFINE4(clock_nanosleep_time32, clockid_t, which_clock, int, flags,
221 struct old_timespec32 __user *, rqtp,
222 struct old_timespec32 __user *, rmtp)
Nicolas Pitrebaa73d92016-11-11 00:10:10 -0500223{
Arnd Bergmannfe460422017-10-13 20:29:38 +0200224 struct timespec64 t;
Andrei Vagin1f9b37b2019-11-12 01:27:06 +0000225 ktime_t texp;
Al Viroedbeda42017-06-07 09:42:31 +0100226
227 switch (which_clock) {
228 case CLOCK_REALTIME:
229 case CLOCK_MONOTONIC:
230 case CLOCK_BOOTTIME:
231 break;
232 default:
233 return -EINVAL;
234 }
235
Arnd Bergmann9afc5ee2018-07-13 12:52:28 +0200236 if (get_old_timespec32(&t, rqtp))
Al Viroedbeda42017-06-07 09:42:31 +0100237 return -EFAULT;
Arnd Bergmannfe460422017-10-13 20:29:38 +0200238 if (!timespec64_valid(&t))
Al Viroedbeda42017-06-07 09:42:31 +0100239 return -EINVAL;
240 if (flags & TIMER_ABSTIME)
241 rmtp = NULL;
242 current->restart_block.nanosleep.type = rmtp ? TT_COMPAT : TT_NONE;
243 current->restart_block.nanosleep.compat_rmtp = rmtp;
Andrei Vagin1f9b37b2019-11-12 01:27:06 +0000244 texp = timespec64_to_ktime(t);
245 if (flags & TIMER_ABSTIME)
246 texp = timens_ktime_to_host(which_clock, texp);
247 return hrtimer_nanosleep(texp, flags & TIMER_ABSTIME ?
Al Viroedbeda42017-06-07 09:42:31 +0100248 HRTIMER_MODE_ABS : HRTIMER_MODE_REL,
249 which_clock);
Nicolas Pitrebaa73d92016-11-11 00:10:10 -0500250}
251#endif