1 /* 2 * Copyright (c) 2005 David Xu <davidxu@freebsd.org> 3 * All rights reserved. 4 * 5 * Redistribution and use in source and binary forms, with or without 6 * modification, are permitted provided that the following conditions 7 * are met: 8 * 1. Redistributions of source code must retain the above copyright 9 * notice unmodified, this list of conditions, and the following 10 * disclaimer. 11 * 2. Redistributions in binary form must reproduce the above copyright 12 * notice, this list of conditions and the following disclaimer in the 13 * documentation and/or other materials provided with the distribution. 14 * 15 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR 16 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES 17 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. 18 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, 19 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT 20 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 21 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 22 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 23 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF 24 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 25 * 26 * $FreeBSD$ 27 * 28 */ 29 30 #include <sys/types.h> 31 #include <machine/atomic.h> 32 33 #include "namespace.h" 34 #include <err.h> 35 #include <errno.h> 36 #include <ucontext.h> 37 #include <sys/thr.h> 38 #include <stdio.h> 39 #include <stdlib.h> 40 #include <string.h> 41 #include <signal.h> 42 #include <pthread.h> 43 #include "un-namespace.h" 44 45 #include "sigev_thread.h" 46 47 LIST_HEAD(sigev_list_head, sigev_node); 48 #define HASH_QUEUES 17 49 #define HASH(t, id) ((((id) << 3) + (t)) % HASH_QUEUES) 50 51 static struct sigev_list_head sigev_hash[HASH_QUEUES]; 52 static struct sigev_list_head sigev_all; 53 static LIST_HEAD(,sigev_thread) sigev_threads; 54 static unsigned int sigev_generation; 55 static pthread_mutex_t *sigev_list_mtx; 56 static pthread_once_t sigev_once = PTHREAD_ONCE_INIT; 57 static pthread_once_t sigev_once_default = PTHREAD_ONCE_INIT; 58 static struct sigev_thread *sigev_default_thread; 59 static pthread_attr_t sigev_default_attr; 60 static int atfork_registered; 61 62 static void __sigev_fork_prepare(void); 63 static void __sigev_fork_parent(void); 64 static void __sigev_fork_child(void); 65 static struct sigev_thread *sigev_thread_create(int); 66 static void *sigev_service_loop(void *); 67 static void *worker_routine(void *); 68 static void worker_cleanup(void *); 69 70 #pragma weak _pthread_create 71 72 static void 73 attrcopy(pthread_attr_t *src, pthread_attr_t *dst) 74 { 75 struct sched_param sched; 76 void *a; 77 size_t u; 78 int v; 79 80 _pthread_attr_getschedpolicy(src, &v); 81 _pthread_attr_setschedpolicy(dst, v); 82 83 _pthread_attr_getinheritsched(src, &v); 84 _pthread_attr_setinheritsched(dst, v); 85 86 _pthread_attr_getschedparam(src, &sched); 87 _pthread_attr_setschedparam(dst, &sched); 88 89 _pthread_attr_getscope(src, &v); 90 _pthread_attr_setscope(dst, v); 91 92 _pthread_attr_getstacksize(src, &u); 93 _pthread_attr_setstacksize(dst, u); 94 95 _pthread_attr_getstackaddr(src, &a); 96 _pthread_attr_setstackaddr(src, a); 97 98 _pthread_attr_getguardsize(src, &u); 99 _pthread_attr_setguardsize(dst, u); 100 } 101 102 static __inline int 103 have_threads(void) 104 { 105 return (&_pthread_create != NULL); 106 } 107 108 void 109 __sigev_thread_init(void) 110 { 111 static int inited = 0; 112 pthread_mutexattr_t mattr; 113 int i; 114 115 _pthread_mutexattr_init(&mattr); 116 _pthread_mutexattr_settype(&mattr, PTHREAD_MUTEX_NORMAL); 117 sigev_list_mtx = malloc(sizeof(pthread_mutex_t)); 118 _pthread_mutex_init(sigev_list_mtx, &mattr); 119 _pthread_mutexattr_destroy(&mattr); 120 121 for (i = 0; i < HASH_QUEUES; ++i) 122 LIST_INIT(&sigev_hash[i]); 123 LIST_INIT(&sigev_all); 124 LIST_INIT(&sigev_threads); 125 sigev_default_thread = NULL; 126 if (atfork_registered == 0) { 127 _pthread_atfork( 128 __sigev_fork_prepare, 129 __sigev_fork_parent, 130 __sigev_fork_child); 131 atfork_registered = 1; 132 } 133 if (!inited) { 134 _pthread_attr_init(&sigev_default_attr); 135 _pthread_attr_setscope(&sigev_default_attr, 136 PTHREAD_SCOPE_SYSTEM); 137 _pthread_attr_setdetachstate(&sigev_default_attr, 138 PTHREAD_CREATE_DETACHED); 139 inited = 1; 140 } 141 sigev_default_thread = sigev_thread_create(0); 142 } 143 144 int 145 __sigev_check_init(void) 146 { 147 if (!have_threads()) 148 return (-1); 149 150 _pthread_once(&sigev_once, __sigev_thread_init); 151 return (sigev_default_thread != NULL) ? 0 : -1; 152 } 153 154 static void 155 __sigev_fork_prepare(void) 156 { 157 } 158 159 static void 160 __sigev_fork_parent(void) 161 { 162 } 163 164 static void 165 __sigev_fork_child(void) 166 { 167 /* 168 * This is a hack, the thread libraries really should 169 * check if the handlers were already registered in 170 * pthread_atfork(). 171 */ 172 atfork_registered = 1; 173 memcpy(&sigev_once, &sigev_once_default, sizeof(sigev_once)); 174 __sigev_thread_init(); 175 } 176 177 void 178 __sigev_list_lock(void) 179 { 180 _pthread_mutex_lock(sigev_list_mtx); 181 } 182 183 void 184 __sigev_list_unlock(void) 185 { 186 _pthread_mutex_unlock(sigev_list_mtx); 187 } 188 189 struct sigev_node * 190 __sigev_alloc(int type, const struct sigevent *evp, struct sigev_node *prev, 191 int usedefault) 192 { 193 struct sigev_node *sn; 194 195 sn = calloc(1, sizeof(*sn)); 196 if (sn != NULL) { 197 sn->sn_value = evp->sigev_value; 198 sn->sn_func = evp->sigev_notify_function; 199 sn->sn_gen = atomic_fetchadd_int(&sigev_generation, 1); 200 sn->sn_type = type; 201 _pthread_attr_init(&sn->sn_attr); 202 _pthread_attr_setdetachstate(&sn->sn_attr, PTHREAD_CREATE_DETACHED); 203 if (evp->sigev_notify_attributes) 204 attrcopy(evp->sigev_notify_attributes, &sn->sn_attr); 205 if (prev) { 206 __sigev_list_lock(); 207 prev->sn_tn->tn_refcount++; 208 __sigev_list_unlock(); 209 sn->sn_tn = prev->sn_tn; 210 } else { 211 sn->sn_tn = sigev_thread_create(usedefault); 212 if (sn->sn_tn == NULL) { 213 _pthread_attr_destroy(&sn->sn_attr); 214 free(sn); 215 sn = NULL; 216 } 217 } 218 } 219 return (sn); 220 } 221 222 void 223 __sigev_get_sigevent(struct sigev_node *sn, struct sigevent *newevp, 224 sigev_id_t id) 225 { 226 /* 227 * Build a new sigevent, and tell kernel to deliver SIGSERVICE 228 * signal to the new thread. 229 */ 230 newevp->sigev_notify = SIGEV_THREAD_ID; 231 newevp->sigev_signo = SIGSERVICE; 232 newevp->sigev_notify_thread_id = (lwpid_t)sn->sn_tn->tn_lwpid; 233 newevp->sigev_value.sival_ptr = (void *)id; 234 } 235 236 void 237 __sigev_free(struct sigev_node *sn) 238 { 239 _pthread_attr_destroy(&sn->sn_attr); 240 free(sn); 241 } 242 243 struct sigev_node * 244 __sigev_find(int type, sigev_id_t id) 245 { 246 struct sigev_node *sn; 247 int chain = HASH(type, id); 248 249 LIST_FOREACH(sn, &sigev_hash[chain], sn_link) { 250 if (sn->sn_type == type && sn->sn_id == id) 251 break; 252 } 253 return (sn); 254 } 255 256 int 257 __sigev_register(struct sigev_node *sn) 258 { 259 int chain = HASH(sn->sn_type, sn->sn_id); 260 261 LIST_INSERT_HEAD(&sigev_hash[chain], sn, sn_link); 262 return (0); 263 } 264 265 int 266 __sigev_delete(int type, sigev_id_t id) 267 { 268 struct sigev_node *sn; 269 270 sn = __sigev_find(type, id); 271 if (sn != NULL) 272 return (__sigev_delete_node(sn)); 273 return (0); 274 } 275 276 int 277 __sigev_delete_node(struct sigev_node *sn) 278 { 279 LIST_REMOVE(sn, sn_link); 280 281 if (--sn->sn_tn->tn_refcount == 0) 282 _pthread_kill(sn->sn_tn->tn_thread, SIGSERVICE); 283 if (sn->sn_flags & SNF_WORKING) 284 sn->sn_flags |= SNF_REMOVED; 285 else 286 __sigev_free(sn); 287 return (0); 288 } 289 290 static sigev_id_t 291 sigev_get_id(siginfo_t *si) 292 { 293 switch(si->si_code) { 294 case SI_TIMER: 295 return (si->si_timerid); 296 case SI_MESGQ: 297 return (si->si_mqd); 298 case SI_ASYNCIO: 299 return (sigev_id_t)si->si_value.sival_ptr; 300 } 301 return (-1); 302 } 303 304 static struct sigev_thread * 305 sigev_thread_create(int usedefault) 306 { 307 struct sigev_thread *tn; 308 sigset_t set, oset; 309 int ret; 310 311 if (usedefault && sigev_default_thread) { 312 __sigev_list_lock(); 313 sigev_default_thread->tn_refcount++; 314 __sigev_list_unlock(); 315 return (sigev_default_thread); 316 } 317 318 tn = malloc(sizeof(*tn)); 319 tn->tn_cur = NULL; 320 tn->tn_lwpid = -1; 321 tn->tn_refcount = 1; 322 _pthread_cond_init(&tn->tn_cv, NULL); 323 324 /* for debug */ 325 __sigev_list_lock(); 326 LIST_INSERT_HEAD(&sigev_threads, tn, tn_link); 327 __sigev_list_unlock(); 328 329 sigfillset(&set); /* SIGSERVICE is masked. */ 330 sigdelset(&set, SIGBUS); 331 sigdelset(&set, SIGILL); 332 sigdelset(&set, SIGFPE); 333 sigdelset(&set, SIGSEGV); 334 sigdelset(&set, SIGTRAP); 335 _sigprocmask(SIG_SETMASK, &set, &oset); 336 ret = _pthread_create(&tn->tn_thread, &sigev_default_attr, 337 sigev_service_loop, tn); 338 _sigprocmask(SIG_SETMASK, &oset, NULL); 339 340 if (ret != 0) { 341 __sigev_list_lock(); 342 LIST_REMOVE(tn, tn_link); 343 __sigev_list_unlock(); 344 free(tn); 345 tn = NULL; 346 } else { 347 /* wait the thread to get its lwpid */ 348 349 __sigev_list_lock(); 350 while (tn->tn_lwpid == -1) 351 _pthread_cond_wait(&tn->tn_cv, sigev_list_mtx); 352 __sigev_list_unlock(); 353 } 354 return (tn); 355 } 356 357 /* 358 * The thread receives notification from kernel and creates 359 * a thread to call user callback function. 360 */ 361 static void * 362 sigev_service_loop(void *arg) 363 { 364 static int failure; 365 366 siginfo_t si; 367 sigset_t set; 368 struct sigev_thread *tn; 369 struct sigev_node *sn; 370 sigev_id_t id; 371 pthread_t td; 372 int ret; 373 374 tn = arg; 375 thr_self(&tn->tn_lwpid); 376 __sigev_list_lock(); 377 _pthread_cond_broadcast(&tn->tn_cv); 378 __sigev_list_unlock(); 379 380 sigemptyset(&set); 381 sigaddset(&set, SIGSERVICE); 382 for (;;) { 383 ret = sigwaitinfo(&set, &si); 384 385 __sigev_list_lock(); 386 if (tn->tn_refcount == 0) { 387 LIST_REMOVE(tn, tn_link); 388 __sigev_list_unlock(); 389 free(tn); 390 break; 391 } 392 393 if (ret == -1) { 394 __sigev_list_unlock(); 395 continue; 396 } 397 398 id = sigev_get_id(&si); 399 sn = __sigev_find(si.si_code, id); 400 if (sn == NULL) { 401 __sigev_list_unlock(); 402 continue; 403 } 404 405 sn->sn_info = si; 406 if (sn->sn_flags & SNF_SYNC) 407 tn->tn_cur = sn; 408 else 409 tn->tn_cur = NULL; 410 sn->sn_flags |= SNF_WORKING; 411 __sigev_list_unlock(); 412 413 ret = _pthread_create(&td, &sn->sn_attr, worker_routine, sn); 414 if (ret != 0) { 415 if (failure++ < 5) 416 warnc(ret, "%s:%s failed to create thread.\n", 417 __FILE__, __func__); 418 419 __sigev_list_lock(); 420 sn->sn_flags &= ~SNF_WORKING; 421 if (sn->sn_flags & SNF_REMOVED) 422 __sigev_free(sn); 423 __sigev_list_unlock(); 424 } else if (tn->tn_cur) { 425 __sigev_list_lock(); 426 while (tn->tn_cur) 427 _pthread_cond_wait(&tn->tn_cv, sigev_list_mtx); 428 __sigev_list_unlock(); 429 } 430 } 431 return (0); 432 } 433 434 /* 435 * newly created worker thread to call user callback function. 436 */ 437 static void * 438 worker_routine(void *arg) 439 { 440 struct sigev_node *sn = arg; 441 442 _pthread_cleanup_push(worker_cleanup, sn); 443 sn->sn_dispatch(sn); 444 _pthread_cleanup_pop(1); 445 446 return (0); 447 } 448 449 /* clean up a notification after dispatch. */ 450 static void 451 worker_cleanup(void *arg) 452 { 453 struct sigev_node *sn = arg; 454 455 __sigev_list_lock(); 456 if (sn->sn_flags & SNF_SYNC) { 457 sn->sn_tn->tn_cur = NULL; 458 _pthread_cond_broadcast(&sn->sn_tn->tn_cv); 459 } 460 if (sn->sn_flags & SNF_REMOVED) 461 __sigev_free(sn); 462 else 463 sn->sn_flags &= ~SNF_WORKING; 464 __sigev_list_unlock(); 465 } 466