/* * Copyright (c) 2002, Intel Corporation. All rights reserved. * Created by: rolla.n.selbak REMOVE-THIS AT intel DOT com * This file is licensed under the GPL license. For the full content * of this license, see the COPYING file at the top level of this * source tree. * Test pthread_testcancel * Atomically sets the cancelability type to 'type' and returns the old * cancelability type in the location referenced by 'oldtype'. * 'state' can either be PTHREAD_CANCEL_DEFERRED, or PTHREAD_CANCEL_ASYNCHRONOUS. * * Test when a thread is PTHREAD_CANCEL_DEFERRED * * STEPS: * 1. Setup a mutex and lock it in main() * 2. Create a thread. * 3. In the thread function, set the type to PTHREAD_CANCEL_DEFERRED * 4. Setup a cleanup handler for the thread. * 5. Make the thread block on the locked mutex * 6. Send out a thread cancel request to the new thread, and unlock the mutex allowing the * thread to continue execution. * 7. If the cancel request was honored immediately, the * cleanup handler would have been executed, setting the cleanup_flag to -1, making the * test fail. * 8. If not, the thread will continue execution, pop the cleanup handler, set the cleanup * flag to 1, and call the cancelation point pthread_testcancel(). The test will pass. * 9. If the thread did not cancel at the cancelation point like it was supposed to, the thread * will continue execution and set the cleanup_flag to -2, failing the test. */ #include #include #include #include #include "posixtest.h" # define INTHREAD 0 /* Control going to or is already for Thread */ # define INMAIN 1 /* Control going to or is already for Main */ int sem1; /* Manual semaphore */ int cleanup_flag; /* Flag to indicate the thread's cleanup handler was called */ pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER; /* Mutex */ /* Cleanup function that the thread executes when it is canceled. So if * cleanup_flag is 1, it means that the thread was canceled. */ void a_cleanup_func() { cleanup_flag=-1; return; } /* Function that the thread executes upon its creation */ void *a_thread_func() { pthread_setcancelstate(PTHREAD_CANCEL_ENABLE, NULL); pthread_setcanceltype(PTHREAD_CANCEL_DEFERRED, NULL); pthread_cleanup_push(a_cleanup_func,NULL); /* Indicate to main() that the thread has been created. */ sem1=INMAIN; /* Lock the mutex. It should have already been locked in main, so the thread * should block. */ if(pthread_mutex_lock(&mutex) != 0) { perror("Error in pthread_mutex_lock()\n"); pthread_exit((void*)PTS_UNRESOLVED); return (void*)PTS_UNRESOLVED; } /* Should get here if the cancel request was deffered. */ pthread_cleanup_pop(0); cleanup_flag=1; /* Cancelation point. Cancel request should not be honored here. */ pthread_testcancel(); /* Should not get here if the cancel request was honored at the cancelation point * pthread_testcancel(). */ cleanup_flag=-2; pthread_exit(0); return NULL; } int main() { pthread_t new_th; /* Initializing values */ sem1=INTHREAD; cleanup_flag=0; /* Lock the mutex */ if(pthread_mutex_lock(&mutex) != 0) { perror("Error in pthread_mutex_lock()\n"); return PTS_UNRESOLVED; } /* Create a new thread. */ if(pthread_create(&new_th, NULL, a_thread_func, NULL) != 0) { perror("Error creating thread\n"); return PTS_UNRESOLVED; } /* Make sure thread is created before we cancel it. (wait for * a_thread_func() to set sem1=INMAIN.) */ while(sem1==INTHREAD) sleep(1); /* Send cancel request to the thread. */ if(pthread_cancel(new_th) != 0) { printf("Test FAILED: Couldn't cancel thread\n"); return PTS_FAIL; } /* Cancel request has been sent, unlock the mutex */ if(pthread_mutex_unlock(&mutex) != 0) { perror("Error in pthread_mutex_unlock()\n"); return PTS_UNRESOLVED; } /* Wait 'till the thread has been canceled or has ended execution. */ if(pthread_join(new_th, NULL) != 0) { perror("Error in pthread_join()\n"); return PTS_UNRESOLVED; } /* This means that the cleanup function wasn't called, so the cancel * request was not honord immediately like it should have been. */ if(cleanup_flag == -1) { perror("Cancel request was not deferred.\n"); return PTS_UNRESOLVED; } if(cleanup_flag == -2) { printf("Test FAILED:pthread_testcancel() not treated as a cancelation point.\n"); return PTS_FAIL; } printf("Test PASSED\n"); return PTS_PASS; }