Blame src/ctests/locks_pthreads.c

Packit 577717
/* This file checks to make sure the locking mechanisms work correctly	*/
Packit 577717
/* on the platform.							*/
Packit 577717
/* Platforms where the locking mechanisms are not implemented or are	*/
Packit 577717
/* incorrectly implemented will fail.  -KSL				*/
Packit 577717
Packit 577717
#define MAX_THREADS 256
Packit 577717
#define APPR_TOTAL_ITER 1000000
Packit 577717
Packit 577717
#include <stdio.h>
Packit 577717
#include <stdlib.h>
Packit 577717
#include <unistd.h>
Packit 577717
#include <pthread.h>
Packit 577717
#include <math.h>
Packit 577717
Packit 577717
#include "papi.h"
Packit 577717
#include "papi_test.h"
Packit 577717
Packit 577717
volatile long long count = 0;
Packit 577717
volatile long long tmpcount = 0;
Packit 577717
volatile long long thread_iter = 0;
Packit 577717
Packit 577717
static int quiet=0;
Packit 577717
Packit 577717
void
Packit 577717
lockloop( int iters, volatile long long *mycount )
Packit 577717
{
Packit 577717
	int i;
Packit 577717
	for ( i = 0; i < iters; i++ ) {
Packit 577717
		PAPI_lock( PAPI_USR1_LOCK );
Packit 577717
		*mycount = *mycount + 1;
Packit 577717
		PAPI_unlock( PAPI_USR1_LOCK );
Packit 577717
	}
Packit 577717
}
Packit 577717
Packit 577717
void *
Packit 577717
Slave( void *arg )
Packit 577717
{
Packit 577717
	long long duration;
Packit 577717
Packit 577717
	duration = PAPI_get_real_usec(  );
Packit 577717
	lockloop( thread_iter, &count );
Packit 577717
	duration = PAPI_get_real_usec(  ) - duration;
Packit 577717
Packit 577717
	if (!quiet) {
Packit 577717
		printf("%f lock/unlocks per us\n",
Packit 577717
			(float)thread_iter/(float)duration);
Packit 577717
	}
Packit 577717
	pthread_exit( arg );
Packit 577717
}
Packit 577717
Packit 577717
Packit 577717
int
Packit 577717
main( int argc, char **argv )
Packit 577717
{
Packit 577717
	pthread_t slaves[MAX_THREADS];
Packit 577717
	int rc, i, nthr;
Packit 577717
	int retval;
Packit 577717
	const PAPI_hw_info_t *hwinfo = NULL;
Packit 577717
Packit 577717
	/* Set TESTS_QUIET variable */
Packit 577717
	quiet = tests_quiet( argc, argv );
Packit 577717
Packit 577717
	retval = PAPI_library_init( PAPI_VER_CURRENT );
Packit 577717
	if (retval != PAPI_VER_CURRENT ) {
Packit 577717
		test_fail( __FILE__, __LINE__, "PAPI_library_init", retval );
Packit 577717
	}
Packit 577717
Packit 577717
	hwinfo = PAPI_get_hardware_info(  );
Packit 577717
	if (hwinfo == NULL ) {
Packit 577717
		test_fail( __FILE__, __LINE__, "PAPI_get_hardware_info", 2 );
Packit 577717
	}
Packit 577717
Packit 577717
	retval = PAPI_thread_init((unsigned long (*)(void)) ( pthread_self ) );
Packit 577717
	if ( retval != PAPI_OK ) {
Packit 577717
		if ( retval == PAPI_ECMP ) {
Packit 577717
			test_skip( __FILE__, __LINE__,
Packit 577717
				"PAPI_thread_init", retval );
Packit 577717
		}
Packit 577717
		else {
Packit 577717
			test_fail( __FILE__, __LINE__,
Packit 577717
				"PAPI_thread_init", retval );
Packit 577717
		}
Packit 577717
	}
Packit 577717
Packit 577717
	if ( hwinfo->ncpu > MAX_THREADS ) {
Packit 577717
		nthr = MAX_THREADS;
Packit 577717
	}
Packit 577717
	else {
Packit 577717
		nthr = hwinfo->ncpu;
Packit 577717
	}
Packit 577717
Packit 577717
	/* Scale the per thread work to keep the serial runtime about the same. */
Packit 577717
	thread_iter = APPR_TOTAL_ITER/sqrt(nthr);
Packit 577717
Packit 577717
	if (!quiet) {
Packit 577717
		printf( "Creating %d threads, %lld lock/unlock\n",
Packit 577717
			nthr , thread_iter);
Packit 577717
	}
Packit 577717
Packit 577717
	for ( i = 0; i < nthr; i++ ) {
Packit 577717
		rc = pthread_create( &slaves[i], NULL, Slave, NULL );
Packit 577717
		if ( rc ) {
Packit 577717
			retval = PAPI_ESYS;
Packit 577717
			test_fail( __FILE__, __LINE__,
Packit 577717
				"pthread_create", retval );
Packit 577717
		}
Packit 577717
	}
Packit 577717
Packit 577717
	for ( i = 0; i < nthr; i++ ) {
Packit 577717
		pthread_join( slaves[i], NULL );
Packit 577717
	}
Packit 577717
Packit 577717
	if (!quiet) {
Packit 577717
		printf( "Expected: %lld Received: %lld\n",
Packit 577717
			( long long ) nthr * thread_iter,
Packit 577717
			count );
Packit 577717
	}
Packit 577717
Packit 577717
	if ( nthr * thread_iter != count ) {
Packit 577717
		test_fail( __FILE__, __LINE__, "Thread Locks", 1 );
Packit 577717
	}
Packit 577717
Packit 577717
	test_pass( __FILE__ );
Packit 577717
Packit 577717
	return 0;
Packit 577717
Packit 577717
}