/* Check that an error is reported for various kinds of bogus pthread_mutex_unlock calls. */ #include #include #include void* child_fn ( void* arg ) { pthread_mutex_unlock( (pthread_mutex_t*)arg ); /* ERROR */ return NULL; } void nearly_main ( void ) { pthread_t child; pthread_mutex_t mx1, mx2; int bogus[100], i; /* fill bogus with values which will cause glibc's pth_mx_unlock to fail */ for (i = 0; i < 100; i++) bogus[i] = 0xFFFFFFFF; /* Unlocking a lock that is already unlocked */ pthread_mutex_init( &mx1, NULL ); pthread_mutex_lock( &mx1 ); pthread_mutex_unlock( &mx1 ); pthread_mutex_unlock( &mx1 ); /* ERROR */ /* Unlocking a lock that is held by a different thread */ pthread_mutex_init( &mx2, NULL ); pthread_mutex_lock( &mx2 ); // start child and get it to unlock this lock pthread_create( &child, NULL, child_fn, (void*)&mx2 ); /* child runs and attempts to unlock our lock. Error is reported in child_fn. */ pthread_join(child, NULL ); /* Unlocking a totally bogus lock. */ pthread_mutex_unlock( (pthread_mutex_t*) &bogus[50] ); /* ERROR */ /* Now we get a freeing-locked-lock error, since the stack frame is removed whilst mx2 is still locked. */ } int main ( void ) { nearly_main(); fprintf(stderr, "---------------------\n" ); nearly_main(); return 0; }