raise.c 2.4 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273
  1. /* Copyright (C) 2002, 2003, 2004 Free Software Foundation, Inc.
  2. This file is part of the GNU C Library.
  3. Contributed by Ulrich Drepper <drepper@redhat.com>, 2002.
  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.
  8. The GNU C Library is distributed in the hope that it will be useful,
  9. but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  11. Lesser General Public License for more details.
  12. You should have received a copy of the GNU Lesser General Public
  13. License along with the GNU C Library; if not, see
  14. <http://www.gnu.org/licenses/>. */
  15. #include <errno.h>
  16. #include <limits.h>
  17. #include <signal.h>
  18. #include <sysdep.h>
  19. #include <pthreadP.h>
  20. #include <bits/kernel-features.h>
  21. int
  22. raise (
  23. int sig)
  24. {
  25. struct pthread *pd = THREAD_SELF;
  26. #if (defined(__ASSUME_TGKILL) && __ASSUME_TGKILL) || defined __NR_tgkill
  27. pid_t pid = THREAD_GETMEM (pd, pid);
  28. #endif
  29. pid_t selftid = THREAD_GETMEM (pd, tid);
  30. if (selftid == 0)
  31. {
  32. /* This system call is not supposed to fail. */
  33. #ifdef INTERNAL_SYSCALL
  34. INTERNAL_SYSCALL_DECL (err);
  35. selftid = INTERNAL_SYSCALL (gettid, err, 0);
  36. #else
  37. selftid = INLINE_SYSCALL (gettid, 0);
  38. #endif
  39. THREAD_SETMEM (pd, tid, selftid);
  40. #if (defined(__ASSUME_TGKILL) && __ASSUME_TGKILL) || defined __NR_tgkill
  41. /* We do not set the PID field in the TID here since we might be
  42. called from a signal handler while the thread executes fork. */
  43. pid = selftid;
  44. #endif
  45. }
  46. #if (defined(__ASSUME_TGKILL) && __ASSUME_TGKILL) || defined __NR_tgkill
  47. else
  48. /* raise is an async-safe function. It could be called while the
  49. fork/vfork function temporarily invalidated the PID field. Adjust for
  50. that. */
  51. if (__builtin_expect (pid <= 0, 0))
  52. pid = (pid & INT_MAX) == 0 ? selftid : -pid;
  53. #endif
  54. #if defined(__ASSUME_TGKILL) && __ASSUME_TGKILL
  55. return INLINE_SYSCALL (tgkill, 3, pid, selftid, sig);
  56. #else
  57. # ifdef __NR_tgkill
  58. int res = INLINE_SYSCALL (tgkill, 3, pid, selftid, sig);
  59. if (res != -1 || errno != ENOSYS)
  60. return res;
  61. # endif
  62. return INLINE_SYSCALL (tkill, 2, selftid, sig);
  63. #endif
  64. }
  65. libc_hidden_def (raise)