/* * Copyright (C) 2008 The Android Open Source Project * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in * the documentation and/or other materials provided with the * distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. */ #include <pthread.h> #include <stdio.h> #include <stdlib.h> #define MAGIC1 0xcafebabeU #define MAGIC2 0x8badf00dU #define MAGIC3 0x12345667U static int g_ok1 = 0; static int g_ok2 = 0; static int g_ok3 = 0; static void cleanup1( void* arg ) { if ((unsigned)arg != MAGIC1) g_ok1 = -1; else g_ok1 = +1; } static void cleanup2( void* arg ) { if ((unsigned)arg != MAGIC2) { g_ok2 = -1; } else g_ok2 = +1; } static void cleanup3( void* arg ) { if ((unsigned)arg != MAGIC3) g_ok3 = -1; else g_ok3 = +1; } static void* thread1_func( void* arg ) { pthread_cleanup_push( cleanup1, (void*)MAGIC1 ); pthread_cleanup_push( cleanup2, (void*)MAGIC2 ); pthread_cleanup_push( cleanup3, (void*)MAGIC3 ); if (arg != NULL) pthread_exit(0); pthread_cleanup_pop(0); pthread_cleanup_pop(1); pthread_cleanup_pop(1); return NULL; } static int test( int do_exit ) { pthread_t t; pthread_create( &t, NULL, thread1_func, (void*)do_exit ); pthread_join( t, NULL ); if (g_ok1 != +1) { if (g_ok1 == 0) { fprintf(stderr, "cleanup1 not called !!\n"); } else { fprintf(stderr, "cleanup1 called with wrong argument\n" ); } exit(1); } else if (g_ok2 != +1) { if (g_ok2 == 0) fprintf(stderr, "cleanup2 not called !!\n"); else fprintf(stderr, "cleanup2 called with wrong argument\n"); exit(2); } else if (do_exit && g_ok3 != +1) { if (g_ok3 == 0) { fprintf(stderr, "cleanup3 not called !!\n"); } else { fprintf(stderr, "cleanup3 called with bad argument !!\n"); } exit(3); } else if (!do_exit && g_ok3 != 0) { if (g_ok3 == 1) { fprintf(stderr, "cleanup3 wrongly called !!\n"); } else { fprintf(stderr, "cleanup3 wrongly called with bad argument !!\n"); } exit(3); } return 0; } int main( void ) { test(0); test(1); printf("OK\n"); return 0; }