ptlongjmp.c 2.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566
  1. /* Linuxthreads - a simple clone()-based implementation of Posix */
  2. /* threads for Linux. */
  3. /* Copyright (C) 1998 Xavier Leroy (Xavier.Leroy@inria.fr) */
  4. /* */
  5. /* This program is free software; you can redistribute it and/or */
  6. /* modify it under the terms of the GNU Library General Public License */
  7. /* as published by the Free Software Foundation; either version 2 */
  8. /* of the License, or (at your option) any later version. */
  9. /* */
  10. /* This program 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 */
  13. /* GNU Library General Public License for more details. */
  14. /* Redefine siglongjmp and longjmp so that they interact correctly
  15. with cleanup handlers */
  16. #include <setjmp.h>
  17. #include "pthread.h"
  18. #include "internals.h"
  19. #include <bits/stackinfo.h>
  20. static void pthread_cleanup_upto(__jmp_buf target)
  21. {
  22. pthread_descr self = thread_self();
  23. struct _pthread_cleanup_buffer * c;
  24. char *currentframe = CURRENT_STACK_FRAME;
  25. for (c = THREAD_GETMEM(self, p_cleanup);
  26. c != NULL && _JMPBUF_UNWINDS(target, c);
  27. c = c->__prev)
  28. {
  29. #ifdef _STACK_GROWS_DOWN
  30. if ((char *) c <= currentframe)
  31. {
  32. c = NULL;
  33. break;
  34. }
  35. #elif defined _STACK_GROWS_UP
  36. if ((char *) c >= currentframe)
  37. {
  38. c = NULL;
  39. break;
  40. }
  41. #else
  42. # error "Define either _STACK_GROWS_DOWN or _STACK_GROWS_UP"
  43. #endif
  44. c->__routine(c->__arg);
  45. }
  46. THREAD_SETMEM(self, p_cleanup, c);
  47. if (THREAD_GETMEM(self, p_in_sighandler)
  48. && _JMPBUF_UNWINDS(target, THREAD_GETMEM(self, p_in_sighandler)))
  49. THREAD_SETMEM(self, p_in_sighandler, NULL);
  50. }
  51. void siglongjmp(sigjmp_buf env, int val)
  52. {
  53. pthread_cleanup_upto(env->__jmpbuf);
  54. __libc_siglongjmp(env, val);
  55. }
  56. void longjmp(jmp_buf env, int val)
  57. {
  58. pthread_cleanup_upto(env->__jmpbuf);
  59. __libc_longjmp(env, val);
  60. }