blob: 39263852a97955893363883b2c4737f4d258025b [file] [log] [blame]
/*
* 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_setcanceltype
* 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 <pthread.h>
#include <stdio.h>
#include <errno.h>
#include <unistd.h>
#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)
{
printf("Test FAILED: Cancel request was not deferred.\n");
return PTS_FAIL;
}
if (cleanup_flag == -2)
{
printf("Test FAILED: (1) Cancel request not honored at cancelation point pthread_testcancel() OR (2) pthread_testcancel() not treated as a cancelation point.\n");
return PTS_FAIL;
}
printf("Test PASSED\n");
return PTS_PASS;
}