1 /* Copyright (C) 2003-2022 Free Software Foundation, Inc.
2    This file is part of the GNU C Library.
3 
4    The GNU C Library is free software; you can redistribute it and/or
5    modify it under the terms of the GNU Lesser General Public License as
6    published by the Free Software Foundation; either version 2.1 of the
7    License, or (at your option) any later version.
8 
9    The GNU C Library is distributed in the hope that it will be useful,
10    but WITHOUT ANY WARRANTY; without even the implied warranty of
11    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12    Lesser General Public License for more details.
13 
14    You should have received a copy of the GNU Lesser General Public
15    License along with the GNU C Library; see the file COPYING.LIB.  If
16    not, see <https://www.gnu.org/licenses/>.  */
17 
18 #include <errno.h>
19 #include <stdlib.h>
20 #include <time.h>
21 #include <sysdep.h>
22 #include <kernel-features.h>
23 #include "kernel-posix-timers.h"
24 #include <shlib-compat.h>
25 
26 #if !TIMER_T_WAS_INT_COMPAT
27 int
___timer_settime64(timer_t timerid,int flags,const struct __itimerspec64 * value,struct __itimerspec64 * ovalue)28 ___timer_settime64 (timer_t timerid, int flags,
29                    const struct __itimerspec64 *value,
30                    struct __itimerspec64 *ovalue)
31 {
32   kernel_timer_t ktimerid = timerid_to_kernel_timer (timerid);
33 
34 # ifdef __ASSUME_TIME64_SYSCALLS
35 #  ifndef __NR_timer_settime64
36 #   define __NR_timer_settime64 __NR_timer_settime
37 #  endif
38   return INLINE_SYSCALL_CALL (timer_settime64, ktimerid, flags, value,
39                               ovalue);
40 # else
41 #  ifdef __NR_timer_settime64
42   int ret = INLINE_SYSCALL_CALL (timer_settime64, ktimerid, flags, value,
43                                  ovalue);
44   if (ret == 0 || errno != ENOSYS)
45     return ret;
46 #  endif
47   struct itimerspec its32, oits32;
48 
49   if (! in_time_t_range ((value->it_value).tv_sec)
50       || ! in_time_t_range ((value->it_interval).tv_sec))
51     {
52       __set_errno (EOVERFLOW);
53       return -1;
54     }
55 
56   its32.it_interval = valid_timespec64_to_timespec (value->it_interval);
57   its32.it_value = valid_timespec64_to_timespec (value->it_value);
58 
59   int retval = INLINE_SYSCALL_CALL (timer_settime, ktimerid, flags,
60                                     &its32, ovalue ? &oits32 : NULL);
61   if (retval == 0 && ovalue)
62     {
63       ovalue->it_interval = valid_timespec_to_timespec64 (oits32.it_interval);
64       ovalue->it_value = valid_timespec_to_timespec64 (oits32.it_value);
65     }
66 
67   return retval;
68 # endif
69 }
70 
71 # if __TIMESIZE == 64
72 versioned_symbol (libc, ___timer_settime64, timer_settime, GLIBC_2_34);
73 #  if OTHER_SHLIB_COMPAT (librt, GLIBC_2_2, GLIBC_2_34)
74 compat_symbol (librt, ___timer_settime64, timer_settime, GLIBC_2_2);
75 #  endif
76 
77 #else /* __TIMESIZE != 64 */
78 libc_hidden_ver (___timer_settime64, __timer_settime64)
79 versioned_symbol (libc, ___timer_settime64, __timer_settime64, GLIBC_2_34);
80 
81 int
__timer_settime(timer_t timerid,int flags,const struct itimerspec * value,struct itimerspec * ovalue)82 __timer_settime (timer_t timerid, int flags, const struct itimerspec *value,
83                  struct itimerspec *ovalue)
84 {
85   struct __itimerspec64 its64, oits64;
86   int retval;
87 
88   its64.it_interval = valid_timespec_to_timespec64 (value->it_interval);
89   its64.it_value = valid_timespec_to_timespec64 (value->it_value);
90 
91   retval = __timer_settime64 (timerid, flags, &its64, ovalue ? &oits64 : NULL);
92   if (retval == 0 && ovalue)
93     {
94       ovalue->it_interval = valid_timespec64_to_timespec (oits64.it_interval);
95       ovalue->it_value = valid_timespec64_to_timespec (oits64.it_value);
96     }
97 
98   return retval;
99 }
100 versioned_symbol (libc, __timer_settime, timer_settime, GLIBC_2_34);
101 
102 #  if OTHER_SHLIB_COMPAT (librt, GLIBC_2_2, GLIBC_2_34)
103 compat_symbol (librt, __timer_settime, timer_settime, GLIBC_2_2);
104 #  endif
105 # endif /* __TIMESIZE != 64 */
106 
107 #else /* TIMER_T_WAS_INT_COMPAT */
108 
109 extern __typeof (timer_settime) __timer_settime_new;
libc_hidden_proto(__timer_settime_new)110 libc_hidden_proto (__timer_settime_new)
111 
112 int
113 ___timer_settime_new (timer_t timerid, int flags,
114                       const struct itimerspec *value,
115                       struct itimerspec *ovalue)
116 {
117   kernel_timer_t ktimerid = timerid_to_kernel_timer (timerid);
118 
119   return INLINE_SYSCALL_CALL (timer_settime, ktimerid, flags, value, ovalue);
120 }
121 versioned_symbol (libc, ___timer_settime_new, timer_settime, GLIBC_2_34);
122 libc_hidden_ver (___timer_settime_new, __timer_settime_new)
123 
124 # if OTHER_SHLIB_COMPAT (librt, GLIBC_2_3_3, GLIBC_2_34)
125 compat_symbol (librt, ___timer_settime_new, timer_settime, GLIBC_2_3_3);
126 # endif
127 
128 # if OTHER_SHLIB_COMPAT (librt, GLIBC_2_2, GLIBC_2_3_3)
129 int
__timer_settime_old(int timerid,int flags,const struct itimerspec * value,struct itimerspec * ovalue)130 __timer_settime_old (int timerid, int flags, const struct itimerspec *value,
131                      struct itimerspec *ovalue)
132 {
133   return __timer_settime_new (__timer_compat_list[timerid], flags,
134                               value, ovalue);
135 }
136 compat_symbol (librt, __timer_settime_old, timer_settime, GLIBC_2_2);
137 # endif
138 
139 #endif /* TIMER_T_WAS_INT_COMPAT */
140