summaryrefslogtreecommitdiff
path: root/lib/libc_r/uthread/uthread_cancel.c
diff options
context:
space:
mode:
Diffstat (limited to 'lib/libc_r/uthread/uthread_cancel.c')
-rw-r--r--lib/libc_r/uthread/uthread_cancel.c164
1 files changed, 164 insertions, 0 deletions
diff --git a/lib/libc_r/uthread/uthread_cancel.c b/lib/libc_r/uthread/uthread_cancel.c
new file mode 100644
index 00000000000..f3a205d1e9e
--- /dev/null
+++ b/lib/libc_r/uthread/uthread_cancel.c
@@ -0,0 +1,164 @@
+/*
+ * David Leonard <d@openbsd.org>, 1999. Public domain.
+ */
+#include <sys/errno.h>
+#include <pthread.h>
+#include "pthread_private.h"
+
+int
+pthread_cancel(pthread)
+ pthread_t pthread;
+{
+ int ret;
+
+ if ((ret = _find_thread(pthread))) {
+ } else if (pthread->state == PS_DEAD)
+ ret = 0;
+ else {
+ /* Set the threads's I've-been-cancelled flag: */
+ pthread->flags |= PTHREAD_CANCELLING;
+ /* Check if we need to kick it back into the run queue: */
+ if ((pthread->cancelstate == PTHREAD_CANCEL_ENABLE) &&
+ ((pthread->canceltype == PTHREAD_CANCEL_ASYNCHRONOUS) ||
+ (pthread->flags & PTHREAD_AT_CANCEL_POINT)))
+ switch (pthread->state) {
+ case PS_RUNNING:
+ /* No need to resume: */
+ break;
+ case PS_WAIT_WAIT:
+ case PS_FDR_WAIT:
+ case PS_FDW_WAIT:
+ case PS_SLEEP_WAIT:
+ case PS_SELECT_WAIT:
+ case PS_SIGSUSPEND:
+ /* Interrupt and resume: */
+ pthread->interrupted = 1;
+ PTHREAD_NEW_STATE(pthread,PS_RUNNING);
+ break;
+ case PS_MUTEX_WAIT:
+ case PS_COND_WAIT:
+ case PS_FDLR_WAIT:
+ case PS_FDLW_WAIT:
+ case PS_FILE_WAIT:
+ case PS_SIGWAIT:
+ case PS_JOIN:
+ case PS_SUSPENDED:
+ case PS_SIGTHREAD:
+ /* Simply wake: */
+ /* XXX may be incorrect */
+ PTHREAD_NEW_STATE(pthread,PS_RUNNING);
+ break;
+ case PS_DEAD:
+ case PS_STATE_MAX:
+ /* Ignore */
+ break;
+ }
+ ret = 0;
+ }
+ return (ret);
+}
+
+int
+pthread_setcancelstate(state, oldstate)
+ int state;
+ int *oldstate;
+{
+ int ostate;
+ int ret;
+
+ ostate = _thread_run->cancelstate;
+
+ switch (state) {
+ case PTHREAD_CANCEL_ENABLE:
+ if (oldstate)
+ *oldstate = ostate;
+ _thread_run->cancelstate = PTHREAD_CANCEL_ENABLE;
+ if (_thread_run->canceltype == PTHREAD_CANCEL_ASYNCHRONOUS)
+ _thread_cancellation_point();
+ ret = 0;
+ break;
+ case PTHREAD_CANCEL_DISABLE:
+ if (oldstate)
+ *oldstate = ostate;
+ _thread_run->cancelstate = PTHREAD_CANCEL_DISABLE;
+ ret = 0;
+ break;
+ default:
+ ret = EINVAL;
+ }
+
+ return (ret);
+}
+
+
+int
+pthread_setcanceltype(type, oldtype)
+ int type;
+ int *oldtype;
+{
+ int otype;
+ int ret;
+
+ otype = _thread_run->canceltype;
+ switch (type) {
+ case PTHREAD_CANCEL_ASYNCHRONOUS:
+ if (oldtype)
+ *oldtype = otype;
+ _thread_run->canceltype = PTHREAD_CANCEL_ASYNCHRONOUS;
+ _thread_cancellation_point();
+ ret = 0;
+ break;
+ case PTHREAD_CANCEL_DEFERRED:
+ if (oldtype)
+ *oldtype = otype;
+ _thread_run->canceltype = PTHREAD_CANCEL_DEFERRED;
+ ret = 0;
+ break;
+ default:
+ ret = EINVAL;
+ }
+
+ return (ret);
+}
+
+void
+pthread_testcancel()
+{
+
+ _thread_cancellation_point();
+}
+
+void
+_thread_enter_cancellation_point()
+{
+
+ /* Look for a cancellation before we block: */
+ _thread_cancellation_point();
+ _thread_run->flags |= PTHREAD_AT_CANCEL_POINT;
+}
+
+void
+_thread_leave_cancellation_point()
+{
+
+ _thread_run->flags &=~ PTHREAD_AT_CANCEL_POINT;
+ /* Look for a cancellation after we unblock: */
+ _thread_cancellation_point();
+}
+
+/*
+ * Must only be called when in asynchronous cancel mode, or
+ * from pthread_testcancel().
+ */
+void
+_thread_cancellation_point()
+{
+
+ if ((_thread_run->cancelstate == PTHREAD_CANCEL_ENABLE) &&
+ ((_thread_run->flags & (PTHREAD_CANCELLING|PTHREAD_EXITING)) ==
+ PTHREAD_CANCELLING)) {
+ _thread_run->flags &=~ PTHREAD_CANCELLING;
+ pthread_exit(PTHREAD_CANCELED);
+ PANIC("cancel");
+ }
+}