1 /* clock_getcpuclockid -- Get a clockid_t for process CPU time. Linux version.
2 Copyright (C) 2004, 2005 Free Software Foundation, Inc.
3 This file is part of the GNU C Library.
4
5 The GNU C Library is free software; you can redistribute it and/or
6 modify it under the terms of the GNU Lesser General Public
7 License as published by the Free Software Foundation; either
8 version 2.1 of the License, or (at your option) any later version.
9
10 The GNU C Library is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 Lesser General Public License for more details.
14
15 You should have received a copy of the GNU Lesser General Public
16 License along with the GNU C Library; if not, see
17 <http://www.gnu.org/licenses/>. */
18
19 #include <errno.h>
20 #include <time.h>
21 #include <sysdep.h>
22 #include <unistd.h>
23 #include <bits/kernel-features.h>
24 #include "kernel-posix-cpu-timers.h"
25
26 #ifndef HAS_CPUCLOCK
27 # define HAS_CPUCLOCK 1
28 #endif
29
30 int
clock_getcpuclockid(pid_t pid,clockid_t * clock_id)31 clock_getcpuclockid (pid_t pid, clockid_t *clock_id)
32 {
33 #if defined(__NR_clock_getres) || defined(__NR_clock_getres_time64)
34 /* The clockid_t value is a simple computation from the PID.
35 But we do a clock_getres call to validate it. */
36
37 const clockid_t pidclock = MAKE_PROCESS_CPUCLOCK (pid, CPUCLOCK_SCHED);
38
39 # if !(__ASSUME_POSIX_CPU_TIMERS > 0)
40 extern int __libc_missing_posix_cpu_timers attribute_hidden;
41 # if !(__ASSUME_POSIX_TIMERS > 0)
42 extern int __libc_missing_posix_timers attribute_hidden;
43 if (__libc_missing_posix_timers && !__libc_missing_posix_cpu_timers)
44 __libc_missing_posix_cpu_timers = 1;
45 # endif
46 if (!__libc_missing_posix_cpu_timers)
47 # endif
48 {
49 INTERNAL_SYSCALL_DECL (err);
50 # if defined(__UCLIBC_USE_TIME64__) && defined(__NR_clock_getres_time64)
51 int r = INTERNAL_SYSCALL (clock_getres_time64, err, 2, pidclock, NULL);
52 # else
53 int r = INTERNAL_SYSCALL (clock_getres, err, 2, pidclock, NULL);
54 # endif
55 if (!INTERNAL_SYSCALL_ERROR_P (r, err))
56 {
57 *clock_id = pidclock;
58 return 0;
59 }
60
61 # if !(__ASSUME_POSIX_TIMERS > 0)
62 if (INTERNAL_SYSCALL_ERRNO (r, err) == ENOSYS)
63 {
64 /* The kernel doesn't support these calls at all. */
65 __libc_missing_posix_timers = 1;
66 __libc_missing_posix_cpu_timers = 1;
67 }
68 else
69 # endif
70 if (INTERNAL_SYSCALL_ERRNO (r, err) == EINVAL)
71 {
72 # if !(__ASSUME_POSIX_CPU_TIMERS > 0)
73 # if defined(__UCLIBC_USE_TIME64__) && defined(__NR_clock_getres_time64)
74 if (pidclock == MAKE_PROCESS_CPUCLOCK (0, CPUCLOCK_SCHED)
75 || INTERNAL_SYSCALL_ERROR_P (INTERNAL_SYSCALL
76 (clock_getres_time64, err, 2,
77 MAKE_PROCESS_CPUCLOCK
78 (0, CPUCLOCK_SCHED), NULL),
79 err))
80 # else
81 if (pidclock == MAKE_PROCESS_CPUCLOCK (0, CPUCLOCK_SCHED)
82 || INTERNAL_SYSCALL_ERROR_P (INTERNAL_SYSCALL
83 (clock_getres, err, 2,
84 MAKE_PROCESS_CPUCLOCK
85 (0, CPUCLOCK_SCHED), NULL),
86 err))
87 # endif
88 /* The kernel doesn't support these clocks at all. */
89 __libc_missing_posix_cpu_timers = 1;
90 else
91 # endif
92 /* The clock_getres system call checked the PID for us. */
93 return ESRCH;
94 }
95 else
96 return INTERNAL_SYSCALL_ERRNO (r, err);
97 }
98 #endif
99
100 /* We don't allow any process ID but our own. */
101 if (pid != 0 && pid != getpid ())
102 return EPERM;
103
104 #ifdef CLOCK_PROCESS_CPUTIME_ID
105 if (HAS_CPUCLOCK)
106 {
107 /* Store the number. */
108 *clock_id = CLOCK_PROCESS_CPUTIME_ID;
109
110 return 0;
111 }
112 #endif
113
114 /* We don't have a timer for that. */
115 return ENOENT;
116 }
117