1 /* Copyright (C) 2004-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/>. */
28 #include <sys/socket.h>
29 #include <not-cancel.h>
31 #include <shlib-compat.h>
33 /* Defined in the kernel headers: */
34 #define NOTIFY_COOKIE_LEN 32 /* Length of the cookie used. */
35 #define NOTIFY_WOKENUP 1 /* Code for notification. */
36 #define NOTIFY_REMOVED 2 /* Code for closed message queue
37 of de-notification. */
40 /* Data structure for the queued notification requests. */
45 void (*fct
) (union sigval
); /* The function to run. */
46 union sigval param
; /* The parameter to pass. */
47 pthread_attr_t
*attr
; /* Attributes to create the thread with. */
48 /* NB: on 64-bit machines the struct as a size of 24 bytes. Which means
49 byte 31 can still be used for returning the status. */
51 char raw
[NOTIFY_COOKIE_LEN
];
55 /* Keep track of the initialization. */
56 static pthread_once_t once
= PTHREAD_ONCE_INIT
;
59 /* The netlink socket. */
60 static int netlink_socket
= -1;
63 /* Barrier used to make sure data passed to the new thread is not
64 reused by the parent. */
65 static pthread_barrier_t notify_barrier
;
68 /* Modify the signal mask. We move this into a separate function so
69 that the stack space needed for sigset_t is not deducted from what
70 the thread can use. */
72 __attribute__ ((noinline
))
73 change_sigmask (int how
, sigset_t
*oss
)
77 return __pthread_sigmask (how
, &ss
, oss
);
81 /* The function used for the notification. */
83 notification_function (void *arg
)
85 /* Copy the function and parameter so that the parent thread can go
87 volatile union notify_data
*data
= (volatile union notify_data
*) arg
;
88 void (*fct
) (union sigval
) = data
->fct
;
89 union sigval param
= data
->param
;
91 /* Let the parent go. */
92 (void) __pthread_barrier_wait (¬ify_barrier
);
94 /* Make the thread detached. */
95 __pthread_detach (__pthread_self ());
97 /* The parent thread has all signals blocked. This is probably a
98 bit surprising for this thread. So we unblock all of them. */
99 (void) change_sigmask (SIG_UNBLOCK
, NULL
);
101 /* Now run the user code. */
104 /* And we are done. */
111 helper_thread (void *arg
)
115 union notify_data data
;
117 ssize_t n
= __recv (netlink_socket
, &data
, sizeof (data
),
118 MSG_NOSIGNAL
| MSG_WAITALL
);
119 if (n
< NOTIFY_COOKIE_LEN
)
122 if (data
.raw
[NOTIFY_COOKIE_LEN
- 1] == NOTIFY_WOKENUP
)
124 /* Just create the thread as instructed. There is no way to
125 report a problem with creating a thread. */
127 if (__pthread_create (&th
, data
.attr
, notification_function
, &data
)
129 /* Since we passed a pointer to DATA to the new thread we have
130 to wait until it is done with it. */
131 (void) __pthread_barrier_wait (¬ify_barrier
);
133 else if (data
.raw
[NOTIFY_COOKIE_LEN
- 1] == NOTIFY_REMOVED
&& data
.attr
!= NULL
)
135 /* The only state we keep is the copy of the thread attributes. */
136 __pthread_attr_destroy (data
.attr
);
145 __mq_notify_fork_subprocess (void)
147 once
= PTHREAD_ONCE_INIT
;
152 init_mq_netlink (void)
154 /* This code might be called a second time after fork(). The file
155 descriptor is inherited from the parent. */
156 if (netlink_socket
== -1)
158 /* Just a normal netlink socket, not bound. */
159 netlink_socket
= __socket (AF_NETLINK
, SOCK_RAW
| SOCK_CLOEXEC
, 0);
160 /* No need to do more if we have no socket. */
161 if (netlink_socket
== -1)
167 /* Initialize the barrier. */
168 if (__pthread_barrier_init (¬ify_barrier
, NULL
, 2) == 0)
170 /* Create the helper thread. */
172 __pthread_attr_init (&attr
);
173 __pthread_attr_setdetachstate (&attr
, PTHREAD_CREATE_DETACHED
);
174 /* We do not need much stack space, the bare minimum will be enough. */
175 __pthread_attr_setstacksize (&attr
, __pthread_get_minstack (&attr
));
177 /* Temporarily block all signals so that the newly created
178 thread inherits the mask. */
180 int have_no_oss
= change_sigmask (SIG_BLOCK
, &oss
);
183 err
= __pthread_create (&th
, &attr
, helper_thread
, NULL
);
185 /* Reset the signal mask. */
187 __pthread_sigmask (SIG_SETMASK
, &oss
, NULL
);
189 __pthread_attr_destroy (&attr
);
194 __close_nocancel_nostatus (netlink_socket
);
200 /* Register notification upon message arrival to an empty message queue
203 __mq_notify (mqd_t mqdes
, const struct sigevent
*notification
)
205 /* Make sure the type is correctly defined. */
206 assert (sizeof (union notify_data
) == NOTIFY_COOKIE_LEN
);
208 /* Special treatment needed for SIGEV_THREAD. */
209 if (notification
== NULL
|| notification
->sigev_notify
!= SIGEV_THREAD
)
210 return INLINE_SYSCALL (mq_notify
, 2, mqdes
, notification
);
212 /* The kernel cannot directly start threads. This will have to be
213 done at userlevel. Since we cannot start threads from signal
214 handlers we have to create a dedicated thread which waits for
215 notifications for arriving messages and creates threads in
218 /* Initialize only once. */
219 __pthread_once (&once
, init_mq_netlink
);
221 /* If we cannot create the netlink socket we cannot provide
222 SIGEV_THREAD support. */
223 if (__glibc_unlikely (netlink_socket
== -1))
225 __set_errno (ENOSYS
);
229 /* Create the cookie. It will hold almost all the state. */
230 union notify_data data
;
231 memset (&data
, '\0', sizeof (data
));
232 data
.fct
= notification
->sigev_notify_function
;
233 data
.param
= notification
->sigev_value
;
235 if (notification
->sigev_notify_attributes
!= NULL
)
237 /* The thread attribute has to be allocated separately. */
238 data
.attr
= (pthread_attr_t
*) malloc (sizeof (pthread_attr_t
));
239 if (data
.attr
== NULL
)
242 int ret
= __pthread_attr_copy (data
.attr
,
243 notification
->sigev_notify_attributes
);
252 /* Construct the new request. */
254 se
.sigev_notify
= SIGEV_THREAD
;
255 se
.sigev_signo
= netlink_socket
;
256 se
.sigev_value
.sival_ptr
= &data
;
258 /* Tell the kernel. */
259 int retval
= INLINE_SYSCALL (mq_notify
, 2, mqdes
, &se
);
261 /* If it failed, free the allocated memory. */
262 if (retval
!= 0 && data
.attr
!= NULL
)
264 __pthread_attr_destroy (data
.attr
);
270 versioned_symbol (libc
, __mq_notify
, mq_notify
, GLIBC_2_34
);
271 libc_hidden_ver (__mq_notify
, mq_notify
)
272 #if OTHER_SHLIB_COMPAT (librt, GLIBC_2_3_4, GLIBC_2_34)
273 compat_symbol (librt
, __mq_notify
, mq_notify
, GLIBC_2_3_4
);