Blame src/linux-common.c

Packit 577717
/*
Packit 577717
* File:    linux-common.c
Packit 577717
*/
Packit 577717
Packit 577717
#include <unistd.h>
Packit 577717
#include <string.h>
Packit 577717
#include <stdlib.h>
Packit 577717
#include <ctype.h>
Packit 577717
#include <err.h>
Packit 577717
#include <stdarg.h>
Packit 577717
#include <stdio.h>
Packit 577717
#include <errno.h>
Packit 577717
#include <syscall.h>
Packit 577717
#include <sys/utsname.h>
Packit 577717
#include <sys/time.h>
Packit 577717
Packit 577717
#include "papi.h"
Packit 577717
#include "papi_internal.h"
Packit 577717
#include "papi_vector.h"
Packit 577717
Packit 577717
#include "linux-memory.h"
Packit 577717
#include "linux-common.h"
Packit 577717
#include "linux-timer.h"
Packit 577717
Packit 577717
#include "x86_cpuid_info.h"
Packit 577717
Packit 577717
PAPI_os_info_t _papi_os_info;
Packit 577717
Packit 577717
/* The locks used by Linux */
Packit 577717
Packit 577717
#if defined(USE_PTHREAD_MUTEXES)
Packit 577717
pthread_mutex_t _papi_hwd_lock_data[PAPI_MAX_LOCK];
Packit 577717
#else
Packit 577717
volatile unsigned int _papi_hwd_lock_data[PAPI_MAX_LOCK];
Packit 577717
#endif
Packit 577717
Packit 577717
Packit 577717
static int _linux_init_locks(void) {
Packit 577717
Packit 577717
   int i;
Packit 577717
Packit 577717
   for ( i = 0; i < PAPI_MAX_LOCK; i++ ) {
Packit 577717
#if defined(USE_PTHREAD_MUTEXES)
Packit 577717
       pthread_mutex_init(&_papi_hwd_lock_data[i],NULL);
Packit 577717
#else
Packit 577717
       _papi_hwd_lock_data[i] = MUTEX_OPEN;
Packit 577717
#endif
Packit 577717
   }
Packit 577717
Packit 577717
   return PAPI_OK;
Packit 577717
}
Packit 577717
Packit 577717
Packit 577717
int
Packit 577717
_linux_detect_hypervisor(char *virtual_vendor_name) {
Packit 577717
Packit 577717
	int retval=0;
Packit 577717
Packit 577717
#if defined(__i386__)||defined(__x86_64__)
Packit 577717
	retval=_x86_detect_hypervisor(virtual_vendor_name);
Packit 577717
#else
Packit 577717
	(void) virtual_vendor_name;
Packit 577717
#endif
Packit 577717
Packit 577717
	return retval;
Packit 577717
}
Packit 577717
Packit 577717
Packit 577717
#define _PATH_SYS_SYSTEM "/sys/devices/system"
Packit 577717
#define _PATH_SYS_CPU0	 _PATH_SYS_SYSTEM "/cpu/cpu0"
Packit 577717
Packit 577717
static char pathbuf[PATH_MAX] = "/";
Packit 577717
Packit 577717
static char *
Packit 577717
search_cpu_info( FILE * f, char *search_str)
Packit 577717
{
Packit 577717
	static char line[PAPI_HUGE_STR_LEN] = "";
Packit 577717
	char *s, *start = NULL;
Packit 577717
Packit 577717
	rewind(f);
Packit 577717
Packit 577717
	while (fgets(line,PAPI_HUGE_STR_LEN,f)!=NULL) {
Packit 577717
		s=strstr(line,search_str);
Packit 577717
		if (s!=NULL) {
Packit 577717
			/* skip all characters in line up to the colon */
Packit 577717
			/* and then spaces */
Packit 577717
			s=strchr(s,':');
Packit 577717
			if (s==NULL) break;
Packit 577717
			s++;
Packit 577717
			while (isspace(*s)) {
Packit 577717
				s++;
Packit 577717
			}
Packit 577717
			start = s;
Packit 577717
			/* Find and clear newline */
Packit 577717
			s=strrchr(start,'\n');
Packit 577717
			if (s!=NULL) *s = 0;
Packit 577717
			break;
Packit 577717
		}
Packit 577717
	}
Packit 577717
	return start;
Packit 577717
}
Packit 577717
Packit 577717
static void
Packit 577717
decode_vendor_string( char *s, int *vendor )
Packit 577717
{
Packit 577717
	if ( strcasecmp( s, "GenuineIntel" ) == 0 )
Packit 577717
		*vendor = PAPI_VENDOR_INTEL;
Packit 577717
	else if ( ( strcasecmp( s, "AMD" ) == 0 ) ||
Packit 577717
			  ( strcasecmp( s, "AuthenticAMD" ) == 0 ) )
Packit 577717
		*vendor = PAPI_VENDOR_AMD;
Packit 577717
	else if ( strcasecmp( s, "IBM" ) == 0 )
Packit 577717
		*vendor = PAPI_VENDOR_IBM;
Packit 577717
	else if ( strcasecmp( s, "Cray" ) == 0 )
Packit 577717
		*vendor = PAPI_VENDOR_CRAY;
Packit 577717
	else if ( strcasecmp( s, "ARM" ) == 0 )
Packit 577717
		*vendor = PAPI_VENDOR_ARM;
Packit 577717
	else if ( strcasecmp( s, "MIPS" ) == 0 )
Packit 577717
		*vendor = PAPI_VENDOR_MIPS;
Packit 577717
	else if ( strcasecmp( s, "SiCortex" ) == 0 )
Packit 577717
		*vendor = PAPI_VENDOR_MIPS;
Packit 577717
	else
Packit 577717
		*vendor = PAPI_VENDOR_UNKNOWN;
Packit 577717
}
Packit 577717
Packit 577717
static FILE *
Packit 577717
xfopen( const char *path, const char *mode )
Packit 577717
{
Packit 577717
	FILE *fd = fopen( path, mode );
Packit 577717
	if ( !fd )
Packit 577717
		err( EXIT_FAILURE, "error: %s", path );
Packit 577717
	return fd;
Packit 577717
}
Packit 577717
Packit 577717
static FILE *
Packit 577717
path_vfopen( const char *mode, const char *path, va_list ap )
Packit 577717
{
Packit 577717
	vsnprintf( pathbuf, sizeof ( pathbuf ), path, ap );
Packit 577717
	return xfopen( pathbuf, mode );
Packit 577717
}
Packit 577717
Packit 577717
Packit 577717
static int
Packit 577717
path_sibling( const char *path, ... )
Packit 577717
{
Packit 577717
	int c;
Packit 577717
	long n;
Packit 577717
	int result = 0;
Packit 577717
	char s[2];
Packit 577717
	FILE *fp;
Packit 577717
	va_list ap;
Packit 577717
	va_start( ap, path );
Packit 577717
	fp = path_vfopen( "r", path, ap );
Packit 577717
	va_end( ap );
Packit 577717
Packit 577717
	while ( ( c = fgetc( fp ) ) != EOF ) {
Packit 577717
		if ( isxdigit( c ) ) {
Packit 577717
			s[0] = ( char ) c;
Packit 577717
			s[1] = '\0';
Packit 577717
			for ( n = strtol( s, NULL, 16 ); n > 0; n /= 2 ) {
Packit 577717
				if ( n % 2 )
Packit 577717
					result++;
Packit 577717
			}
Packit 577717
		}
Packit 577717
	}
Packit 577717
Packit 577717
	fclose( fp );
Packit 577717
	return result;
Packit 577717
}
Packit 577717
Packit 577717
static int
Packit 577717
path_exist( const char *path, ... )
Packit 577717
{
Packit 577717
	va_list ap;
Packit 577717
	va_start( ap, path );
Packit 577717
	vsnprintf( pathbuf, sizeof ( pathbuf ), path, ap );
Packit 577717
	va_end( ap );
Packit 577717
	return access( pathbuf, F_OK ) == 0;
Packit 577717
}
Packit 577717
Packit 577717
static int
Packit 577717
decode_cpuinfo_x86( FILE *f, PAPI_hw_info_t *hwinfo )
Packit 577717
{
Packit 577717
	int tmp;
Packit 577717
	unsigned int strSize;
Packit 577717
	char *s;
Packit 577717
Packit 577717
	/* Stepping */
Packit 577717
	s = search_cpu_info( f, "stepping");
Packit 577717
	if ( s ) {
Packit 577717
		if (sscanf( s, "%d", &tmp ) ==1 ) {
Packit 577717
			hwinfo->revision = ( float ) tmp;
Packit 577717
			hwinfo->cpuid_stepping = tmp;
Packit 577717
		}
Packit 577717
	}
Packit 577717
Packit 577717
	/* Model Name */
Packit 577717
	s = search_cpu_info( f, "model name");
Packit 577717
	strSize = sizeof(hwinfo->model_string);
Packit 577717
	if ( s ) {
Packit 577717
		strncpy( hwinfo->model_string, s, strSize);
Packit 577717
	}
Packit 577717
Packit 577717
	/* Family */
Packit 577717
	s = search_cpu_info( f, "cpu family");
Packit 577717
	if ( s ) {
Packit 577717
		sscanf( s, "%d", &tmp );
Packit 577717
		hwinfo->cpuid_family = tmp;
Packit 577717
	}
Packit 577717
Packit 577717
Packit 577717
	/* CPU Model */
Packit 577717
	s = search_cpu_info( f, "model");
Packit 577717
	if ( s ) {
Packit 577717
		sscanf( s , "%d", &tmp );
Packit 577717
		hwinfo->model = tmp;
Packit 577717
		hwinfo->cpuid_model = tmp;
Packit 577717
	}
Packit 577717
Packit 577717
	return PAPI_OK;
Packit 577717
}
Packit 577717
Packit 577717
static int
Packit 577717
decode_cpuinfo_power(FILE *f, PAPI_hw_info_t *hwinfo )
Packit 577717
{
Packit 577717
Packit 577717
	int tmp;
Packit 577717
	unsigned int strSize;
Packit 577717
	char *s;
Packit 577717
Packit 577717
	/* Revision */
Packit 577717
	s = search_cpu_info( f, "revision");
Packit 577717
	if ( s ) {
Packit 577717
		sscanf( s, "%d", &tmp );
Packit 577717
		hwinfo->revision = ( float ) tmp;
Packit 577717
		hwinfo->cpuid_stepping = tmp;
Packit 577717
	}
Packit 577717
Packit 577717
       /* Model Name */
Packit 577717
	s = search_cpu_info( f, "model");
Packit 577717
	strSize = sizeof(hwinfo->model_string);
Packit 577717
	if ( s ) {
Packit 577717
		strncpy( hwinfo->model_string, s, strSize);
Packit 577717
	}
Packit 577717
Packit 577717
	return PAPI_OK;
Packit 577717
}
Packit 577717
Packit 577717
Packit 577717
Packit 577717
static int
Packit 577717
decode_cpuinfo_arm(FILE *f, PAPI_hw_info_t *hwinfo )
Packit 577717
{
Packit 577717
Packit 577717
	int tmp;
Packit 577717
	unsigned int strSize;
Packit 577717
	char *s, *t;
Packit 577717
Packit 577717
	/* revision */
Packit 577717
	s = search_cpu_info( f, "CPU revision");
Packit 577717
	if ( s ) {
Packit 577717
		sscanf( s, "%d", &tmp );
Packit 577717
		hwinfo->revision = ( float ) tmp;
Packit 577717
		/* For compatability with old PAPI */
Packit 577717
		hwinfo->model = tmp;
Packit 577717
	}
Packit 577717
Packit 577717
	/* Model Name */
Packit 577717
	s = search_cpu_info( f, "model name");
Packit 577717
	strSize = sizeof(hwinfo->model_string);
Packit 577717
	if ( s ) {
Packit 577717
		strncpy( hwinfo->model_string, s, strSize );
Packit 577717
	}
Packit 577717
Packit 577717
	/* Architecture (ARMv6, ARMv7, ARMv8, etc.) */
Packit 577717
	/* Note the Raspberry Pi lies in the CPU architecture line */
Packit 577717
	/* (it's ARMv6 not ARMv7)                                  */
Packit 577717
	/* So we should actually get the value from the            */
Packit 577717
	/*	Processor/ model name line                         */
Packit 577717
Packit 577717
	s = search_cpu_info( f, "CPU architecture");
Packit 577717
	if ( s ) {
Packit 577717
Packit 577717
		if (strstr(s,"AArch64")) {
Packit 577717
			hwinfo->cpuid_family = 8;
Packit 577717
		}
Packit 577717
		else {
Packit 577717
			s = search_cpu_info( f, "Processor" );
Packit 577717
			if (s) {
Packit 577717
				t=strchr(s,'(');
Packit 577717
				tmp=*(t+2)-'0';
Packit 577717
				hwinfo->cpuid_family = tmp;
Packit 577717
			}
Packit 577717
			else {
Packit 577717
				s = search_cpu_info( f, "model name" );
Packit 577717
				if (s) {
Packit 577717
					t=strchr(s,'(');
Packit 577717
					tmp=*(t+2)-'0';
Packit 577717
					hwinfo->cpuid_family = tmp;
Packit 577717
				}
Packit 577717
			}
Packit 577717
		}
Packit 577717
	}
Packit 577717
Packit 577717
	/* CPU Model */
Packit 577717
	s = search_cpu_info( f, "CPU part" );
Packit 577717
	if ( s ) {
Packit 577717
		sscanf( s, "%x", &tmp );
Packit 577717
		hwinfo->cpuid_model = tmp;
Packit 577717
	}
Packit 577717
Packit 577717
	/* CPU Variant */
Packit 577717
	s = search_cpu_info( f, "CPU variant" );
Packit 577717
	if ( s ) {
Packit 577717
		sscanf( s, "%x", &tmp );
Packit 577717
		hwinfo->cpuid_stepping = tmp;
Packit 577717
	}
Packit 577717
Packit 577717
	return PAPI_OK;
Packit 577717
Packit 577717
}
Packit 577717
Packit 577717
Packit 577717
int
Packit 577717
_linux_get_cpu_info( PAPI_hw_info_t *hwinfo, int *cpuinfo_mhz )
Packit 577717
{
Packit 577717
	int retval = PAPI_OK;
Packit 577717
	char *s;
Packit 577717
	float mhz = 0.0;
Packit 577717
	FILE *f;
Packit 577717
	char cpuinfo_filename[]="/proc/cpuinfo";
Packit 577717
Packit 577717
	if ( ( f = fopen( cpuinfo_filename, "r" ) ) == NULL ) {
Packit 577717
		PAPIERROR( "fopen(/proc/cpuinfo) errno %d", errno );
Packit 577717
		return PAPI_ESYS;
Packit 577717
	}
Packit 577717
Packit 577717
	/* All of this information may be overwritten by the component */
Packit 577717
Packit 577717
	/***********************/
Packit 577717
	/* Attempt to find MHz */
Packit 577717
	/***********************/
Packit 577717
	s = search_cpu_info( f, "cpu MHz" );
Packit 577717
	if ( !s ) {
Packit 577717
		s = search_cpu_info( f, "clock" );
Packit 577717
	}
Packit 577717
	if ( s ) {
Packit 577717
		sscanf( s, "%f", &mhz );
Packit 577717
		*cpuinfo_mhz = mhz;
Packit 577717
	}
Packit 577717
	else {
Packit 577717
	//	PAPIWARN("Failed to find a clock speed in /proc/cpuinfo");
Packit 577717
	}
Packit 577717
Packit 577717
	/*******************************/
Packit 577717
	/* Vendor Name and Vendor Code */
Packit 577717
	/*******************************/
Packit 577717
Packit 577717
	/* First try to read "vendor_id" field */
Packit 577717
	/* Which is the most common field      */
Packit 577717
	s = search_cpu_info( f, "vendor_id");
Packit 577717
	if ( s ) {
Packit 577717
		strcpy( hwinfo->vendor_string, s );
Packit 577717
	}
Packit 577717
	else {
Packit 577717
		/* If not found, try "vendor" which seems to be Itanium specific */
Packit 577717
		s = search_cpu_info( f, "vendor" );
Packit 577717
		if ( s ) {
Packit 577717
			strcpy( hwinfo->vendor_string, s );
Packit 577717
		}
Packit 577717
		else {
Packit 577717
			/* "system type" seems to be MIPS and Alpha */
Packit 577717
			s = search_cpu_info( f, "system type");
Packit 577717
			if ( s ) {
Packit 577717
				strcpy( hwinfo->vendor_string, s );
Packit 577717
			}
Packit 577717
			else {
Packit 577717
				/* "platform" indicates Power */
Packit 577717
				s = search_cpu_info( f, "platform");
Packit 577717
				if ( s ) {
Packit 577717
					if ( ( strcasecmp( s, "pSeries" ) == 0 ) ||
Packit 577717
						( strcasecmp( s, "PowerNV" ) == 0 ) ||
Packit 577717
						( strcasecmp( s, "PowerMac" ) == 0 ) ) {
Packit 577717
						strcpy( hwinfo->vendor_string, "IBM" );
Packit 577717
					}
Packit 577717
				}
Packit 577717
				else {
Packit 577717
					/* "CPU implementer" indicates ARM */
Packit 577717
					s = search_cpu_info( f, "CPU implementer");
Packit 577717
					if ( s ) {
Packit 577717
						strcpy( hwinfo->vendor_string, "ARM" );
Packit 577717
					}
Packit 577717
				}
Packit 577717
			}
Packit 577717
		}
Packit 577717
	}
Packit 577717
Packit 577717
	/* Decode the string to a PAPI specific implementer value */
Packit 577717
	if ( strlen( hwinfo->vendor_string ) ) {
Packit 577717
		decode_vendor_string( hwinfo->vendor_string, &hwinfo->vendor );
Packit 577717
	}
Packit 577717
Packit 577717
	/**********************************************/
Packit 577717
	/* Provide more stepping/model/family numbers */
Packit 577717
	/**********************************************/
Packit 577717
Packit 577717
	if ((hwinfo->vendor==PAPI_VENDOR_INTEL) ||
Packit 577717
		(hwinfo->vendor==PAPI_VENDOR_AMD)) {
Packit 577717
Packit 577717
		decode_cpuinfo_x86(f,hwinfo);
Packit 577717
	}
Packit 577717
Packit 577717
	if (hwinfo->vendor==PAPI_VENDOR_IBM) {
Packit 577717
Packit 577717
		decode_cpuinfo_power(f,hwinfo);
Packit 577717
	}
Packit 577717
Packit 577717
	if (hwinfo->vendor==PAPI_VENDOR_ARM) {
Packit 577717
Packit 577717
		decode_cpuinfo_arm(f,hwinfo);
Packit 577717
	}
Packit 577717
Packit 577717
Packit 577717
Packit 577717
Packit 577717
	/* The following members are set using the same methodology */
Packit 577717
	/* used in lscpu.                                           */
Packit 577717
Packit 577717
	/* Total number of CPUs */
Packit 577717
	/* The following line assumes totalcpus was initialized to zero! */
Packit 577717
	while ( path_exist( _PATH_SYS_SYSTEM "/cpu/cpu%d", hwinfo->totalcpus ) )
Packit 577717
		hwinfo->totalcpus++;
Packit 577717
Packit 577717
	/* Number of threads per core */
Packit 577717
	if ( path_exist( _PATH_SYS_CPU0 "/topology/thread_siblings" ) )
Packit 577717
		hwinfo->threads =
Packit 577717
			path_sibling( _PATH_SYS_CPU0 "/topology/thread_siblings" );
Packit 577717
Packit 577717
	/* Number of cores per socket */
Packit 577717
	if ( path_exist( _PATH_SYS_CPU0 "/topology/core_siblings" ) &&
Packit 577717
		 hwinfo->threads > 0 )
Packit 577717
		hwinfo->cores =
Packit 577717
			path_sibling( _PATH_SYS_CPU0 "/topology/core_siblings" ) /
Packit 577717
			hwinfo->threads;
Packit 577717
Packit 577717
	/* Number of NUMA nodes */
Packit 577717
	/* The following line assumes nnodes was initialized to zero! */
Packit 577717
	while ( path_exist( _PATH_SYS_SYSTEM "/node/node%d", hwinfo->nnodes ) ) {
Packit 577717
		hwinfo->nnodes++;
Packit 577717
	}
Packit 577717
Packit 577717
	/* Number of CPUs per node */
Packit 577717
	hwinfo->ncpu = hwinfo->nnodes > 1 ?
Packit 577717
			hwinfo->totalcpus / hwinfo->nnodes : hwinfo->totalcpus;
Packit 577717
Packit 577717
	/* Number of sockets */
Packit 577717
	if ( hwinfo->threads > 0 && hwinfo->cores > 0 ) {
Packit 577717
		hwinfo->sockets = hwinfo->totalcpus / hwinfo->cores / hwinfo->threads;
Packit 577717
	}
Packit 577717
Packit 577717
#if 0
Packit 577717
	int *nodecpu;
Packit 577717
	/* cpumap data is not currently part of the _papi_hw_info struct */
Packit 577717
        nodecpu = malloc( (unsigned int) hwinfo->nnodes * sizeof(int) );
Packit 577717
	if ( nodecpu ) {
Packit 577717
	   int i;
Packit 577717
	   for ( i = 0; i < hwinfo->nnodes; ++i ) {
Packit 577717
	       nodecpu[i] = path_sibling(
Packit 577717
                             _PATH_SYS_SYSTEM "/node/node%d/cpumap", i );
Packit 577717
	   }
Packit 577717
	} else {
Packit 577717
		PAPIERROR( "malloc failed for variable not currently used" );
Packit 577717
	}
Packit 577717
#endif
Packit 577717
Packit 577717
Packit 577717
	/* Fixup missing Megahertz Value */
Packit 577717
	/* This is missing from cpuinfo on ARM and MIPS */
Packit 577717
	if (*cpuinfo_mhz < 1.0) {
Packit 577717
		s = search_cpu_info( f, "BogoMIPS" );
Packit 577717
		if ((!s) || (sscanf( s, "%f", &mhz ) != 1)) {
Packit 577717
			INTDBG("MHz detection failed. "
Packit 577717
				"Please edit file %s at line %d.\n",
Packit 577717
				__FILE__,__LINE__);
Packit 577717
		}
Packit 577717
Packit 577717
		if (hwinfo->vendor == PAPI_VENDOR_MIPS) {
Packit 577717
			/* MIPS has 2x clock multiplier */
Packit 577717
			*cpuinfo_mhz = 2*(((int)mhz)+1);
Packit 577717
Packit 577717
			/* Also update version info on MIPS */
Packit 577717
			s = search_cpu_info( f, "cpu model");
Packit 577717
			s = strstr(s," V")+2;
Packit 577717
			strtok(s," ");
Packit 577717
			sscanf(s, "%f ", &hwinfo->revision );
Packit 577717
		}
Packit 577717
		else {
Packit 577717
			/* In general bogomips is proportional to number of CPUs */
Packit 577717
			if (hwinfo->totalcpus) {
Packit 577717
				if (mhz!=0) *cpuinfo_mhz = mhz / hwinfo->totalcpus;
Packit 577717
			}
Packit 577717
		}
Packit 577717
	}
Packit 577717
Packit 577717
	fclose( f );
Packit 577717
Packit 577717
	return retval;
Packit 577717
}
Packit 577717
Packit 577717
int
Packit 577717
_linux_get_mhz( int *sys_min_mhz, int *sys_max_mhz ) {
Packit 577717
Packit 577717
  FILE *fff;
Packit 577717
  int result;
Packit 577717
Packit 577717
  /* Try checking for min MHz */
Packit 577717
  /* Assume cpu0 exists */
Packit 577717
  fff=fopen("/sys/devices/system/cpu/cpu0/cpufreq/cpuinfo_min_freq","r");
Packit 577717
  if (fff==NULL) return PAPI_EINVAL;
Packit 577717
  result=fscanf(fff,"%d",sys_min_mhz);
Packit 577717
  fclose(fff);
Packit 577717
  if (result!=1) return PAPI_EINVAL;
Packit 577717
Packit 577717
  fff=fopen("/sys/devices/system/cpu/cpu0/cpufreq/cpuinfo_max_freq","r");
Packit 577717
  if (fff==NULL) return PAPI_EINVAL;
Packit 577717
  result=fscanf(fff,"%d",sys_max_mhz);
Packit 577717
  fclose(fff);
Packit 577717
  if (result!=1) return PAPI_EINVAL;
Packit 577717
Packit 577717
  return PAPI_OK;
Packit 577717
Packit 577717
}
Packit 577717
Packit 577717
int
Packit 577717
_linux_get_system_info( papi_mdi_t *mdi ) {
Packit 577717
Packit 577717
	int retval;
Packit 577717
	char maxargs[PAPI_HUGE_STR_LEN];
Packit 577717
	pid_t pid;
Packit 577717
	int cpuinfo_mhz,sys_min_khz,sys_max_khz;
Packit 577717
Packit 577717
	/* Software info */
Packit 577717
Packit 577717
	/* Path and args */
Packit 577717
Packit 577717
	pid = getpid(  );
Packit 577717
	if ( pid < 0 ) {
Packit 577717
		PAPIERROR( "getpid() returned < 0" );
Packit 577717
		return PAPI_ESYS;
Packit 577717
	}
Packit 577717
	mdi->pid = pid;
Packit 577717
Packit 577717
	sprintf( maxargs, "/proc/%d/exe", ( int ) pid );
Packit 577717
	retval = readlink( maxargs, mdi->exe_info.fullname,
Packit 577717
			PAPI_HUGE_STR_LEN-1 );
Packit 577717
	if ( retval < 0 ) {
Packit 577717
		PAPIERROR( "readlink(%s) returned < 0", maxargs );
Packit 577717
		return PAPI_ESYS;
Packit 577717
	}
Packit 577717
Packit 577717
	if (retval > PAPI_HUGE_STR_LEN-1) {
Packit 577717
		retval=PAPI_HUGE_STR_LEN-1;
Packit 577717
	}
Packit 577717
	mdi->exe_info.fullname[retval] = '\0';
Packit 577717
Packit 577717
	/* Careful, basename can modify its argument */
Packit 577717
	strcpy( maxargs, mdi->exe_info.fullname );
Packit 577717
Packit 577717
	strncpy( mdi->exe_info.address_info.name, basename( maxargs ),
Packit 577717
		PAPI_HUGE_STR_LEN-1);
Packit 577717
	mdi->exe_info.address_info.name[PAPI_HUGE_STR_LEN-1] = '\0';
Packit 577717
Packit 577717
	SUBDBG( "Executable is %s\n", mdi->exe_info.address_info.name );
Packit 577717
	SUBDBG( "Full Executable is %s\n", mdi->exe_info.fullname );
Packit 577717
Packit 577717
	/* Executable regions, may require reading /proc/pid/maps file */
Packit 577717
Packit 577717
	retval = _linux_update_shlib_info( mdi );
Packit 577717
	SUBDBG( "Text: Start %p, End %p, length %d\n",
Packit 577717
			mdi->exe_info.address_info.text_start,
Packit 577717
			mdi->exe_info.address_info.text_end,
Packit 577717
			( int ) ( mdi->exe_info.address_info.text_end -
Packit 577717
					  mdi->exe_info.address_info.text_start ) );
Packit 577717
	SUBDBG( "Data: Start %p, End %p, length %d\n",
Packit 577717
			mdi->exe_info.address_info.data_start,
Packit 577717
			mdi->exe_info.address_info.data_end,
Packit 577717
			( int ) ( mdi->exe_info.address_info.data_end -
Packit 577717
					  mdi->exe_info.address_info.data_start ) );
Packit 577717
	SUBDBG( "Bss: Start %p, End %p, length %d\n",
Packit 577717
			mdi->exe_info.address_info.bss_start,
Packit 577717
			mdi->exe_info.address_info.bss_end,
Packit 577717
			( int ) ( mdi->exe_info.address_info.bss_end -
Packit 577717
					  mdi->exe_info.address_info.bss_start ) );
Packit 577717
Packit 577717
	/* PAPI_preload_option information */
Packit 577717
Packit 577717
	strcpy( mdi->preload_info.lib_preload_env, "LD_PRELOAD" );
Packit 577717
	mdi->preload_info.lib_preload_sep = ' ';
Packit 577717
	strcpy( mdi->preload_info.lib_dir_env, "LD_LIBRARY_PATH" );
Packit 577717
	mdi->preload_info.lib_dir_sep = ':';
Packit 577717
Packit 577717
	/* Hardware info */
Packit 577717
Packit 577717
	retval = _linux_get_cpu_info( &mdi->hw_info, &cpuinfo_mhz );
Packit 577717
	if ( retval )
Packit 577717
		return retval;
Packit 577717
Packit 577717
	/* Handle MHz */
Packit 577717
Packit 577717
	retval = _linux_get_mhz( &sys_min_khz, &sys_max_khz );
Packit 577717
	if ( retval ) {
Packit 577717
Packit 577717
		mdi->hw_info.cpu_max_mhz=cpuinfo_mhz;
Packit 577717
		mdi->hw_info.cpu_min_mhz=cpuinfo_mhz;
Packit 577717
Packit 577717
	   /*
Packit 577717
	   mdi->hw_info.mhz=cpuinfo_mhz;
Packit 577717
	   mdi->hw_info.clock_mhz=cpuinfo_mhz;
Packit 577717
	   */
Packit 577717
	}
Packit 577717
	else {
Packit 577717
		mdi->hw_info.cpu_max_mhz=sys_max_khz/1000;
Packit 577717
		mdi->hw_info.cpu_min_mhz=sys_min_khz/1000;
Packit 577717
Packit 577717
	   /*
Packit 577717
	   mdi->hw_info.mhz=sys_max_khz/1000;
Packit 577717
	   mdi->hw_info.clock_mhz=sys_max_khz/1000;
Packit 577717
	   */
Packit 577717
	}
Packit 577717
Packit 577717
	/* Set Up Memory */
Packit 577717
Packit 577717
	retval = _linux_get_memory_info( &mdi->hw_info, mdi->hw_info.model );
Packit 577717
	if ( retval )
Packit 577717
		return retval;
Packit 577717
Packit 577717
	SUBDBG( "Found %d %s(%d) %s(%d) CPUs at %d Mhz.\n",
Packit 577717
			mdi->hw_info.totalcpus,
Packit 577717
			mdi->hw_info.vendor_string,
Packit 577717
			mdi->hw_info.vendor,
Packit 577717
		        mdi->hw_info.model_string,
Packit 577717
		        mdi->hw_info.model,
Packit 577717
		        mdi->hw_info.cpu_max_mhz);
Packit 577717
Packit 577717
	/* Get virtualization info */
Packit 577717
	mdi->hw_info.virtualized=_linux_detect_hypervisor(mdi->hw_info.virtual_vendor_string);
Packit 577717
Packit 577717
	return PAPI_OK;
Packit 577717
}
Packit 577717
Packit 577717
int
Packit 577717
_papi_hwi_init_os(void) {
Packit 577717
Packit 577717
    int major=0,minor=0,sub=0;
Packit 577717
    char *ptr;
Packit 577717
    struct utsname uname_buffer;
Packit 577717
Packit 577717
    /* Initialize the locks */
Packit 577717
    _linux_init_locks();
Packit 577717
Packit 577717
    /* Get the kernel info */
Packit 577717
    uname(&uname_buffer);
Packit 577717
Packit 577717
    SUBDBG("Native kernel version %s\n",uname_buffer.release);
Packit 577717
Packit 577717
    strncpy(_papi_os_info.name,uname_buffer.sysname,PAPI_MAX_STR_LEN);
Packit 577717
Packit 577717
#ifdef ASSUME_KERNEL
Packit 577717
    strncpy(_papi_os_info.version,ASSUME_KERNEL,PAPI_MAX_STR_LEN);
Packit 577717
    SUBDBG("Assuming kernel version %s\n",_papi_os_info.name);
Packit 577717
#else
Packit 577717
    strncpy(_papi_os_info.version,uname_buffer.release,PAPI_MAX_STR_LEN);
Packit 577717
#endif
Packit 577717
Packit 577717
    ptr=strtok(_papi_os_info.version,".");
Packit 577717
    if (ptr!=NULL) major=atoi(ptr);
Packit 577717
Packit 577717
    ptr=strtok(NULL,".");
Packit 577717
    if (ptr!=NULL) minor=atoi(ptr);
Packit 577717
Packit 577717
    ptr=strtok(NULL,".");
Packit 577717
    if (ptr!=NULL) sub=atoi(ptr);
Packit 577717
Packit 577717
   _papi_os_info.os_version=LINUX_VERSION(major,minor,sub);
Packit 577717
Packit 577717
   _papi_os_info.itimer_sig = PAPI_INT_MPX_SIGNAL;
Packit 577717
   _papi_os_info.itimer_num = PAPI_INT_ITIMER;
Packit 577717
   _papi_os_info.itimer_ns = PAPI_INT_MPX_DEF_US * 1000;
Packit 577717
   _papi_os_info.itimer_res_ns = 1;
Packit 577717
   _papi_os_info.clock_ticks = sysconf( _SC_CLK_TCK );
Packit 577717
Packit 577717
   /* Get Linux-specific system info */
Packit 577717
   _linux_get_system_info( &_papi_hwi_system_info );
Packit 577717
Packit 577717
   return PAPI_OK;
Packit 577717
}
Packit 577717
Packit 577717
Packit 577717
Packit 577717
int _linux_detect_nmi_watchdog() {
Packit 577717
Packit 577717
  int watchdog_detected=0,watchdog_value=0;
Packit 577717
  FILE *fff;
Packit 577717
Packit 577717
  fff=fopen("/proc/sys/kernel/nmi_watchdog","r");
Packit 577717
  if (fff!=NULL) {
Packit 577717
     if (fscanf(fff,"%d",&watchdog_value)==1) {
Packit 577717
        if (watchdog_value>0) watchdog_detected=1;
Packit 577717
     }
Packit 577717
     fclose(fff);
Packit 577717
  }
Packit 577717
Packit 577717
  return watchdog_detected;
Packit 577717
}
Packit 577717
Packit 577717
papi_os_vector_t _papi_os_vector = {
Packit 577717
  .get_memory_info =   _linux_get_memory_info,
Packit 577717
  .get_dmem_info =     _linux_get_dmem_info,
Packit 577717
  .get_real_cycles =   _linux_get_real_cycles,
Packit 577717
  .update_shlib_info = _linux_update_shlib_info,
Packit 577717
  .get_system_info =   _linux_get_system_info,
Packit 577717
Packit 577717
Packit 577717
#if defined(HAVE_CLOCK_GETTIME)
Packit 577717
  .get_real_usec =  _linux_get_real_usec_gettime,
Packit 577717
#elif defined(HAVE_GETTIMEOFDAY)
Packit 577717
  .get_real_usec =  _linux_get_real_usec_gettimeofday,
Packit 577717
#else
Packit 577717
  .get_real_usec =  _linux_get_real_usec_cycles,
Packit 577717
#endif
Packit 577717
Packit 577717
Packit 577717
#if defined(USE_PROC_PTTIMER)
Packit 577717
  .get_virt_usec =   _linux_get_virt_usec_pttimer,
Packit 577717
#elif defined(HAVE_CLOCK_GETTIME_THREAD)
Packit 577717
  .get_virt_usec =   _linux_get_virt_usec_gettime,
Packit 577717
#elif defined(HAVE_PER_THREAD_TIMES)
Packit 577717
  .get_virt_usec =   _linux_get_virt_usec_times,
Packit 577717
#elif defined(HAVE_PER_THREAD_GETRUSAGE)
Packit 577717
  .get_virt_usec =   _linux_get_virt_usec_rusage,
Packit 577717
#endif
Packit 577717
Packit 577717
Packit 577717
#if defined(HAVE_CLOCK_GETTIME)
Packit 577717
  .get_real_nsec =  _linux_get_real_nsec_gettime,
Packit 577717
#endif
Packit 577717
Packit 577717
#if defined(HAVE_CLOCK_GETTIME_THREAD)
Packit 577717
  .get_virt_nsec =   _linux_get_virt_nsec_gettime,
Packit 577717
#endif
Packit 577717
Packit 577717
Packit 577717
};