Skip site navigation (1)Skip section navigation (2)
Date:      Sat, 20 Nov 2004 04:15:30 GMT
From:      David Xu <davidxu@FreeBSD.org>
To:        Perforce Change Reviews <perforce@freebsd.org>
Subject:   PERFORCE change 65508 for review
Message-ID:  <200411200415.iAK4FUEQ005435@repoman.freebsd.org>

next in thread | raw e-mail | index | archive | help
http://perforce.freebsd.org/chv.cgi?CH=65508

Change 65508 by davidxu@davidxu_alona on 2004/11/20 04:15:10

	1. If thread is exiting, don't return true in checkcancel().
	2. Use thread lock.

Affected files ...

.. //depot/projects/davidxu_thread/src/lib/libthread/thread/thr_cancel.c#2 edit

Differences ...

==== //depot/projects/davidxu_thread/src/lib/libthread/thread/thr_cancel.c#2 (text+ko) ====

@@ -14,15 +14,24 @@
 static inline int
 checkcancel(struct pthread *curthread)
 {
-	if (((curthread->cancelflags & PTHREAD_CANCEL_DISABLE) == 0) &&
-	    ((curthread->cancelflags & THR_CANCELLING) != 0)) {
+	if ((curthread->cancelflags & THR_CANCELLING) != 0) {
 		/*
 		 * It is possible for this thread to be swapped out
 		 * while performing cancellation; do not allow it
 		 * to be cancelled again.
 		 */
-		curthread->cancelflags &= ~THR_CANCELLING;
-		return (1);
+		if ((curthread->flags & THR_FLAGS_EXITING) != 0) {
+			/*
+			 * this may happen once, but after this, it
+			 * shouldn't happen again.
+			 */
+			curthread->cancelflags &= ~THR_CANCELLING;
+			return (0);
+		}
+		if ((curthread->cancelflags & PTHREAD_CANCEL_DISABLE) == 0) {
+			curthread->cancelflags &= ~THR_CANCELLING;
+			return (1);
+		}
 	}
 	else
 		return (0);
@@ -33,7 +42,7 @@
 {
 	if (checkcancel(curthread) != 0) {
 		/* Unlock before exiting: */
-		THR_THREAD_UNLOCK(curthread, curthread);
+		THR_UNLOCK(curthread);
 
 		_thr_exit_cleanup();
 		pthread_exit(PTHREAD_CANCELED);
@@ -46,7 +55,7 @@
 {
 	struct pthread *curthread = _get_curthread();
 	struct pthread *joinee = NULL;
-	struct kse_mailbox *kmbx = NULL;
+	long tid = -1;
 	int ret;
 
 	if ((ret = _thr_ref_add(curthread, pthread, /*include dead*/0)) == 0) {
@@ -54,9 +63,7 @@
 		 * Take the thread's lock while we change the cancel flags.
 		 */
 		THR_THREAD_LOCK(curthread, pthread);
-		THR_SCHED_LOCK(curthread, pthread);
 		if (pthread->flags & THR_FLAGS_EXITING) {
-			THR_SCHED_UNLOCK(curthread, pthread);
 			THR_THREAD_UNLOCK(curthread, pthread);
 			_thr_ref_delete(curthread, pthread);
 			return (ESRCH);
@@ -75,40 +82,15 @@
 			case PS_RUNNING:
 				/* No need to resume: */
 				pthread->cancelflags |= THR_CANCELLING;
+				tid = pthread->tid;
 				break;
 
-			case PS_LOCKWAIT:
-				/*
-				 * These can't be removed from the queue.
-				 * Just mark it as cancelling and tell it
-				 * to yield once it leaves the critical
-				 * region.
-				 */
-				pthread->cancelflags |= THR_CANCELLING;
-				pthread->critical_yield = 1;
-				break;
-
-			case PS_SLEEP_WAIT:
-			case PS_SIGSUSPEND:
-			case PS_SIGWAIT:
-				/* Interrupt and resume: */
-				pthread->interrupted = 1;
-				pthread->cancelflags |= THR_CANCELLING;
-				kmbx = _thr_setrunnable_unlocked(pthread);
-				break;
-
 			case PS_JOIN:
 				/* Disconnect the thread from the joinee: */
 				joinee = pthread->join_status.thread;
 				pthread->join_status.thread = NULL;
 				pthread->cancelflags |= THR_CANCELLING;
-				kmbx = _thr_setrunnable_unlocked(pthread);
-				if ((joinee != NULL) &&
-				    (pthread->kseg == joinee->kseg)) {
-					/* Remove the joiner from the joinee. */
-					joinee->joiner = NULL;
-					joinee = NULL;
-				}
+				tid = _thr_setrunnable_unlocked(pthread);
 				break;
 
 			case PS_SUSPENDED:
@@ -126,7 +108,7 @@
 				 */
 				pthread->interrupted = 1;
 				pthread->cancelflags |= THR_CANCEL_NEEDED;
-				kmbx = _thr_setrunnable_unlocked(pthread);
+				tid = _thr_setrunnable_unlocked(pthread);
 				pthread->continuation =
 					_thr_finish_cancellation;
 				break;
@@ -137,29 +119,27 @@
 				/* Ignore - only here to silence -Wall: */
 				break;
 			}
-			if ((pthread->cancelflags & THR_AT_CANCEL_POINT) &&
-			    (pthread->blocked != 0 ||
-			     pthread->attr.flags & PTHREAD_SCOPE_SYSTEM))
-				kse_thr_interrupt(&pthread->tcb->tcb_tmbx,
-					KSE_INTR_INTERRUPT, 0);
+
+#if 0
+			thr_interrupt(pthread->tid, 1);
+#endif
 		}
 
 		/*
 		 * Release the thread's lock and remove the
 		 * reference:
 		 */
-		THR_SCHED_UNLOCK(curthread, pthread);
 		THR_THREAD_UNLOCK(curthread, pthread);
 		_thr_ref_delete(curthread, pthread);
-		if (kmbx != NULL)
-			kse_wakeup(kmbx);
+		if (tid != -1)
+			thr_wake(tid);
 
 		if ((joinee != NULL) &&
 		    (_thr_ref_add(curthread, joinee, /* include dead */1) == 0)) {
 			/* Remove the joiner from the joinee. */
-			THR_SCHED_LOCK(curthread, joinee);
+			THR_THREAD_LOCK(curthread, joinee);
 			joinee->joiner = NULL;
-			THR_SCHED_UNLOCK(curthread, joinee);
+			THR_THREAD_UNLOCK(curthread, joinee);
 			_thr_ref_delete(curthread, joinee);
 		}
 	}
@@ -175,7 +155,7 @@
 	int need_exit = 0;
 
 	/* Take the thread's lock while fiddling with the state: */
-	THR_THREAD_LOCK(curthread, curthread);
+	THR_LOCK(curthread);
 
 	ostate = curthread->cancelflags & PTHREAD_CANCEL_DISABLE;
 
@@ -194,7 +174,7 @@
 		ret = EINVAL;
 	}
 
-	THR_THREAD_UNLOCK(curthread, curthread);
+	THR_UNLOCK(curthread);
 	if (need_exit != 0) {
 		_thr_exit_cleanup();
 		pthread_exit(PTHREAD_CANCELED);
@@ -215,7 +195,7 @@
 	int need_exit = 0;
 
 	/* Take the thread's lock while fiddling with the state: */
-	THR_THREAD_LOCK(curthread, curthread);
+	THR_LOCK(curthread);
 
 	otype = curthread->cancelflags & PTHREAD_CANCEL_ASYNCHRONOUS;
 	switch (type) {
@@ -232,7 +212,7 @@
 		ret = EINVAL;
 	}
 
-	THR_THREAD_UNLOCK(curthread, curthread);
+	THR_UNLOCK(curthread);
 	if (need_exit != 0) {
 		_thr_exit_cleanup();
 		pthread_exit(PTHREAD_CANCELED);
@@ -249,30 +229,30 @@
 {
 	struct pthread	*curthread = _get_curthread();
 
-	THR_THREAD_LOCK(curthread, curthread);
+	THR_LOCK(curthread);
 	testcancel(curthread);
-	THR_THREAD_UNLOCK(curthread, curthread);
+	THR_UNLOCK(curthread);
 }
 
 void
-_thr_cancel_enter(struct pthread *thread)
+_thr_cancel_enter(struct pthread *curthread)
 {
 	/* Look for a cancellation before we block: */
-	THR_THREAD_LOCK(thread, thread);
-	testcancel(thread);
-	thread->cancelflags |= THR_AT_CANCEL_POINT;
-	THR_THREAD_UNLOCK(thread, thread);
+	THR_LOCK(curthread);
+	testcancel(curthread);
+	curthread->cancelflags |= THR_AT_CANCEL_POINT;
+	THR_UNLOCK(curthread);
 }
 
 void
-_thr_cancel_leave(struct pthread *thread, int check)
+_thr_cancel_leave(struct pthread *curthread, int check)
 {
-	THR_THREAD_LOCK(thread, thread);
-	thread->cancelflags &= ~THR_AT_CANCEL_POINT;
+	THR_LOCK(curthread);
+	curthread->cancelflags &= ~THR_AT_CANCEL_POINT;
 	/* Look for a cancellation after we unblock: */
 	if (check)
-		testcancel(thread);
-	THR_THREAD_UNLOCK(thread, thread);
+		testcancel(curthread);
+	THR_UNLOCK(curthread);
 }
 
 void
@@ -283,12 +263,12 @@
 	curthread->continuation = NULL;
 	curthread->interrupted = 0;
 
-	THR_THREAD_LOCK(curthread, curthread);
+	THR_LOCK(curthread);
 	if ((curthread->cancelflags & THR_CANCEL_NEEDED) != 0) {
 		curthread->cancelflags &= ~THR_CANCEL_NEEDED;
-		THR_THREAD_UNLOCK(curthread, curthread);
+		THR_UNLOCK(curthread);
 		_thr_exit_cleanup();
 		pthread_exit(PTHREAD_CANCELED);
 	}
-	THR_THREAD_UNLOCK(curthread, curthread);
+	THR_UNLOCK(curthread);
 }



Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?200411200415.iAK4FUEQ005435>