/* Check that an error is reported for various kinds of bogus
   pthread_mutex_unlock calls. */

#include <pthread.h>
#include <stdio.h>
#include <stdlib.h>

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;
}