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