|
@@ -25,6 +25,9 @@
|
|
|
#include <rpc/rpc.h>
|
|
|
extern void __rpc_thread_destroy(void);
|
|
|
#endif
|
|
|
+#include <bits/stackinfo.h>
|
|
|
+
|
|
|
+#include <stdio.h>
|
|
|
|
|
|
#ifdef _STACK_GROWS_DOWN
|
|
|
# define FRAME_LEFT(frame, other) ((char *) frame >= (char *) other)
|
|
@@ -45,7 +48,7 @@ int pthread_setcancelstate(int state, int * oldstate)
|
|
|
if (THREAD_GETMEM(self, p_canceled) &&
|
|
|
THREAD_GETMEM(self, p_cancelstate) == PTHREAD_CANCEL_ENABLE &&
|
|
|
THREAD_GETMEM(self, p_canceltype) == PTHREAD_CANCEL_ASYNCHRONOUS)
|
|
|
- pthread_exit(PTHREAD_CANCELED);
|
|
|
+ __pthread_do_exit(PTHREAD_CANCELED, CURRENT_STACK_FRAME);
|
|
|
return 0;
|
|
|
}
|
|
|
|
|
@@ -59,7 +62,7 @@ int pthread_setcanceltype(int type, int * oldtype)
|
|
|
if (THREAD_GETMEM(self, p_canceled) &&
|
|
|
THREAD_GETMEM(self, p_cancelstate) == PTHREAD_CANCEL_ENABLE &&
|
|
|
THREAD_GETMEM(self, p_canceltype) == PTHREAD_CANCEL_ASYNCHRONOUS)
|
|
|
- pthread_exit(PTHREAD_CANCELED);
|
|
|
+ __pthread_do_exit(PTHREAD_CANCELED, CURRENT_STACK_FRAME);
|
|
|
return 0;
|
|
|
}
|
|
|
|
|
@@ -126,7 +129,7 @@ void pthread_testcancel(void)
|
|
|
pthread_descr self = thread_self();
|
|
|
if (THREAD_GETMEM(self, p_canceled)
|
|
|
&& THREAD_GETMEM(self, p_cancelstate) == PTHREAD_CANCEL_ENABLE)
|
|
|
- pthread_exit(PTHREAD_CANCELED);
|
|
|
+ __pthread_do_exit(PTHREAD_CANCELED, CURRENT_STACK_FRAME);
|
|
|
}
|
|
|
|
|
|
void _pthread_cleanup_push(struct _pthread_cleanup_buffer * buffer,
|
|
@@ -173,15 +176,27 @@ void _pthread_cleanup_pop_restore(struct _pthread_cleanup_buffer * buffer,
|
|
|
if (THREAD_GETMEM(self, p_canceled) &&
|
|
|
THREAD_GETMEM(self, p_cancelstate) == PTHREAD_CANCEL_ENABLE &&
|
|
|
THREAD_GETMEM(self, p_canceltype) == PTHREAD_CANCEL_ASYNCHRONOUS)
|
|
|
- pthread_exit(PTHREAD_CANCELED);
|
|
|
+ __pthread_do_exit(PTHREAD_CANCELED, CURRENT_STACK_FRAME);
|
|
|
}
|
|
|
|
|
|
-void __pthread_perform_cleanup(void)
|
|
|
+void __pthread_perform_cleanup(char *currentframe)
|
|
|
{
|
|
|
pthread_descr self = thread_self();
|
|
|
struct _pthread_cleanup_buffer * c;
|
|
|
+
|
|
|
for (c = THREAD_GETMEM(self, p_cleanup); c != NULL; c = c->__prev)
|
|
|
- c->__routine(c->__arg);
|
|
|
+ {
|
|
|
+#if _STACK_GROWS_DOWN
|
|
|
+ if ((char *) c <= currentframe)
|
|
|
+ break;
|
|
|
+#elif _STACK_GROWS_UP
|
|
|
+ if ((char *) c >= currentframe)
|
|
|
+ break;
|
|
|
+#else
|
|
|
+# error "Define either _STACK_GROWS_DOWN or _STACK_GROWS_UP"
|
|
|
+#endif
|
|
|
+ c->__routine(c->__arg);
|
|
|
+ }
|
|
|
|
|
|
#ifdef __UCLIBC_HAS_RPC__
|
|
|
|