blob: 97e4fab465c102f5ee7165e7facce42e53f6b464 [file] [log] [blame]
/*
* Copyright (c) 2004, QUALCOMM Inc. All rights reserved.
* Created by: abisain REMOVE-THIS AT qualcomm 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_cancel()
*
* Any destructors for thread_specific data will be called
*
* Steps:
* 1. Create a new thread.
* 2. Create a thread specific object in the thread with a destructor
* 3. Call pthread_cancel on the thread.
* 4. Make sure that the destructor was called
*
*/
#include <pthread.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include "posixtest.h"
#define TEST "2-2"
#define FUNCTION "pthread_cancel"
#define ERROR_PREFIX "unexpected error: " FUNCTION " " TEST ": "
int cleanup_flag = 0;
int sem = 0; /* manual semaphore */
void destructor(void *tmp)
{
cleanup_flag = 1;
}
/* Thread's function. */
void *a_thread_func(void *tmp)
{
pthread_key_t key;
int value = 1;
int rc = 0;
/* To enable thread immediate cancelation, since the default
* is PTHREAD_CANCEL_DEFERRED. */
rc = pthread_setcanceltype(PTHREAD_CANCEL_ASYNCHRONOUS, NULL);
if (rc != 0) {
printf(ERROR_PREFIX "pthread_setcanceltype\n");
exit(PTS_UNRESOLVED);
}
rc = pthread_key_create(&key, destructor);
if (rc != 0) {
printf(ERROR_PREFIX "pthread_key_create\n");
exit(PTS_UNRESOLVED);
}
rc = pthread_setspecific(key, &value);
if (rc != 0) {
printf(ERROR_PREFIX "pthread_setspecific\n");
exit(PTS_UNRESOLVED);
}
/* Tell main that the key is created */
sem = 1;
/* Sleep forever */
while (1)
sleep(5);
return NULL;
}
int main()
{
pthread_t new_th;
int rc = 0;
sem = 0;
/* Create a new thread. */
rc = pthread_create(&new_th, NULL, a_thread_func, NULL);
if (rc != 0) {
printf(ERROR_PREFIX "pthread_create\n");
exit(PTS_UNRESOLVED);
}
/* Wait for the thread to be ready */
while (sem == 0)
sleep(1);
/* Cancel the thread. */
rc = pthread_cancel(new_th);
if (rc != 0) {
printf(ERROR_PREFIX "pthread_cancel\n");
exit(PTS_UNRESOLVED);
}
/* Delay enough so that the destructor must have been called */
sleep(5);
if (cleanup_flag != 1) {
printf(ERROR_PREFIX "Test FAIL: Destructor was not executed.\n");
exit(PTS_FAIL);
}
printf("Test PASSED\n");
exit(PTS_PASS);
}