1 /* Copyright (C) 2002-2024 Free Software Foundation, Inc.
2 This file is part of the GNU C Library.
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
6 License as published by the Free Software Foundation; either
7 version 2.1 of the License, or (at your option) any later version.
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.
14 You should have received a copy of the GNU Lesser General Public
15 License along with the GNU C Library; if not, see
16 <https://www.gnu.org/licenses/>. */
18 #include <futex-internal.h>
21 #include <lowlevellock.h>
25 /* Check for consistency across set*id system call results. The abort
26 should not happen as long as all privileges changes happen through
27 the glibc wrappers. ERROR must be 0 (no error) or an errno
30 setxid_error (struct xid_command
*cmdp
, int error
)
34 int olderror
= cmdp
->error
;
35 if (olderror
== error
)
39 /* Mismatch between current and previous results. Save the
40 error value to memory so that is not clobbered by the
41 abort function and preserved in coredumps. */
42 volatile int xid_err
__attribute__ ((unused
)) = error
;
46 while (atomic_compare_and_exchange_bool_acq (&cmdp
->error
, error
, -1));
49 /* Set by __nptl_setxid and used by __nptl_setxid_sighandler. */
50 static struct xid_command
*xidcmd
;
52 /* We use the SIGSETXID signal in the setuid, setgid, etc. implementations to
53 tell each thread to call the respective setxid syscall on itself. This is
56 __nptl_setxid_sighandler (int sig
, siginfo_t
*si
, void *ctx
)
60 /* Safety check. It would be possible to call this function for
61 other signals and send a signal from another process. This is not
62 correct and might even be a security problem. Try to catch as
63 many incorrect invocations as possible. */
65 || si
->si_pid
!= __getpid ()
66 || si
->si_code
!= SI_TKILL
)
69 result
= INTERNAL_SYSCALL_NCS (xidcmd
->syscall_no
, 3, xidcmd
->id
[0],
70 xidcmd
->id
[1], xidcmd
->id
[2]);
72 if (__glibc_unlikely (INTERNAL_SYSCALL_ERROR_P (result
)))
73 error
= INTERNAL_SYSCALL_ERRNO (result
);
74 setxid_error (xidcmd
, error
);
76 /* Reset the SETXID flag. */
77 struct pthread
*self
= THREAD_SELF
;
81 flags
= THREAD_GETMEM (self
, cancelhandling
);
82 newval
= THREAD_ATOMIC_CMPXCHG_VAL (self
, cancelhandling
,
83 flags
& ~SETXID_BITMASK
, flags
);
85 while (flags
!= newval
);
87 /* And release the futex. */
88 self
->setxid_futex
= 1;
89 futex_wake (&self
->setxid_futex
, 1, FUTEX_PRIVATE
);
91 if (atomic_fetch_add_relaxed (&xidcmd
->cntr
, -1) == 1)
92 futex_wake ((unsigned int *) &xidcmd
->cntr
, 1, FUTEX_PRIVATE
);
94 libc_hidden_def (__nptl_setxid_sighandler
)
97 setxid_mark_thread (struct xid_command
*cmdp
, struct pthread
*t
)
101 /* Wait until this thread is cloned. */
102 if (t
->setxid_futex
== -1
103 && ! atomic_compare_and_exchange_bool_acq (&t
->setxid_futex
, -2, -1))
105 futex_wait_simple (&t
->setxid_futex
, -2, FUTEX_PRIVATE
);
106 while (t
->setxid_futex
== -2);
108 /* Don't let the thread exit before the setxid handler runs. */
113 ch
= t
->cancelhandling
;
115 /* If the thread is exiting right now, ignore it. */
116 if ((ch
& EXITING_BITMASK
) != 0)
118 /* Release the futex if there is no other setxid in
120 if ((ch
& SETXID_BITMASK
) == 0)
123 futex_wake (&t
->setxid_futex
, 1, FUTEX_PRIVATE
);
128 while (atomic_compare_and_exchange_bool_acq (&t
->cancelhandling
,
129 ch
| SETXID_BITMASK
, ch
));
134 setxid_unmark_thread (struct xid_command
*cmdp
, struct pthread
*t
)
140 ch
= t
->cancelhandling
;
141 if ((ch
& SETXID_BITMASK
) == 0)
144 while (atomic_compare_and_exchange_bool_acq (&t
->cancelhandling
,
145 ch
& ~SETXID_BITMASK
, ch
));
147 /* Release the futex just in case. */
149 futex_wake (&t
->setxid_futex
, 1, FUTEX_PRIVATE
);
154 setxid_signal_thread (struct xid_command
*cmdp
, struct pthread
*t
)
156 if ((t
->cancelhandling
& SETXID_BITMASK
) == 0)
160 pid_t pid
= __getpid ();
161 val
= INTERNAL_SYSCALL_CALL (tgkill
, pid
, t
->tid
, SIGSETXID
);
163 /* If this failed, it must have had not started yet or else exited. */
164 if (!INTERNAL_SYSCALL_ERROR_P (val
))
166 atomic_fetch_add_relaxed (&cmdp
->cntr
, 1);
175 __nptl_setxid (struct xid_command
*cmdp
)
179 lll_lock (GL (dl_stack_cache_lock
), LLL_PRIVATE
);
185 struct pthread
*self
= THREAD_SELF
;
187 /* Iterate over the list with system-allocated threads first. */
189 list_for_each (runp
, &GL (dl_stack_used
))
191 struct pthread
*t
= list_entry (runp
, struct pthread
, list
);
195 setxid_mark_thread (cmdp
, t
);
198 /* Now the list with threads using user-allocated stacks. */
199 list_for_each (runp
, &GL (dl_stack_user
))
201 struct pthread
*t
= list_entry (runp
, struct pthread
, list
);
205 setxid_mark_thread (cmdp
, t
);
208 /* Iterate until we don't succeed in signalling anyone. That means
209 we have gotten all running threads, and their children will be
210 automatically correct once started. */
215 list_for_each (runp
, &GL (dl_stack_used
))
217 struct pthread
*t
= list_entry (runp
, struct pthread
, list
);
221 signalled
+= setxid_signal_thread (cmdp
, t
);
224 list_for_each (runp
, &GL (dl_stack_user
))
226 struct pthread
*t
= list_entry (runp
, struct pthread
, list
);
230 signalled
+= setxid_signal_thread (cmdp
, t
);
233 int cur
= cmdp
->cntr
;
236 futex_wait_simple ((unsigned int *) &cmdp
->cntr
, cur
,
241 while (signalled
!= 0);
243 /* Clean up flags, so that no thread blocks during exit waiting
244 for a signal which will never come. */
245 list_for_each (runp
, &GL (dl_stack_used
))
247 struct pthread
*t
= list_entry (runp
, struct pthread
, list
);
251 setxid_unmark_thread (cmdp
, t
);
254 list_for_each (runp
, &GL (dl_stack_user
))
256 struct pthread
*t
= list_entry (runp
, struct pthread
, list
);
260 setxid_unmark_thread (cmdp
, t
);
263 /* This must be last, otherwise the current thread might not have
264 permissions to send SIGSETXID syscall to the other threads. */
265 result
= INTERNAL_SYSCALL_NCS (cmdp
->syscall_no
, 3,
266 cmdp
->id
[0], cmdp
->id
[1], cmdp
->id
[2]);
268 if (__glibc_unlikely (INTERNAL_SYSCALL_ERROR_P (result
)))
270 error
= INTERNAL_SYSCALL_ERRNO (result
);
274 setxid_error (cmdp
, error
);
276 lll_unlock (GL (dl_stack_cache_lock
), LLL_PRIVATE
);