Blame src/libpfm4/lib/pfmlib_mips.c

Packit 577717
/*
Packit 577717
 * pfmlib_mips.c : support for MIPS chips
Packit 577717
 *
Packit 577717
 * Copyright (c) 2011 Samara Technology Group, Inc
Packit 577717
 * Contributed by Philip Mucci <phil.mucci@@samaratechnologygroup.com>
Packit 577717
 *
Packit 577717
 * Permission is hereby granted, free of charge, to any person obtaining a copy
Packit 577717
 * of this software and associated documentation files (the "Software"), to deal
Packit 577717
 * in the Software without restriction, including without limitation the rights
Packit 577717
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
Packit 577717
 * of the Software, and to permit persons to whom the Software is furnished to do so,
Packit 577717
 * subject to the following conditions:
Packit 577717
 *
Packit 577717
 * The above copyright notice and this permission notice shall be included in all
Packit 577717
 * copies or substantial portions of the Software.
Packit 577717
 *
Packit 577717
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED,
Packit 577717
 * INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A
Packit 577717
 * PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
Packit 577717
 * HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
Packit 577717
 * CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE
Packit 577717
 * OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
Packit 577717
 *
Packit 577717
 */
Packit 577717
Packit 577717
#include <sys/types.h>
Packit 577717
#include <string.h>
Packit 577717
#include <stdlib.h>
Packit 577717
#include <stdio.h>
Packit 577717
#include <stdarg.h>
Packit 577717
Packit 577717
/* private headers */
Packit 577717
#include "pfmlib_priv.h"			/* library private */
Packit 577717
#include "pfmlib_mips_priv.h"
Packit 577717
Packit 577717
pfm_mips_config_t pfm_mips_cfg;
Packit 577717
Packit 577717
static const pfmlib_attr_desc_t mips_mods[]={
Packit 577717
	PFM_ATTR_B("k", "monitor at system level"),
Packit 577717
	PFM_ATTR_B("u", "monitor at user level"),
Packit 577717
	PFM_ATTR_B("s", "monitor at supervisor level"),
Packit 577717
	PFM_ATTR_B("e", "monitor at exception level "),
Packit 577717
	PFM_ATTR_NULL /* end-marker to avoid exporting number of entries */
Packit 577717
};
Packit 577717
Packit 577717
#ifdef CONFIG_PFMLIB_OS_LINUX
Packit 577717
/*
Packit 577717
 * helper function to retrieve one value from /proc/cpuinfo
Packit 577717
 * for internal libpfm use only
Packit 577717
 * attr: the attribute (line) to look for
Packit 577717
 * ret_buf: a buffer to store the value of the attribute (as a string)
Packit 577717
 * maxlen : number of bytes of capacity in ret_buf
Packit 577717
 *
Packit 577717
 * ret_buf is null terminated.
Packit 577717
 *
Packit 577717
 * Return:
Packit 577717
 * 	0 : attribute found, ret_buf populated
Packit 577717
 * 	-1: attribute not found
Packit 577717
 */
Packit 577717
Packit 577717
static int
Packit 577717
pfmlib_getcpuinfo_attr(const char *attr, char *ret_buf, size_t maxlen)
Packit 577717
{
Packit 577717
	FILE *fp = NULL;
Packit 577717
	int ret = -1;
Packit 577717
	size_t attr_len, buf_len = 0;
Packit 577717
	char *p, *value = NULL;
Packit 577717
	char *buffer = NULL;
Packit 577717
Packit 577717
	if (attr == NULL || ret_buf == NULL || maxlen < 1)
Packit 577717
		return -1;
Packit 577717
Packit 577717
	attr_len = strlen(attr);
Packit 577717
Packit 577717
	fp = fopen("/proc/cpuinfo", "r");
Packit 577717
	if (fp == NULL)
Packit 577717
		return -1;
Packit 577717
Packit 577717
	while(pfmlib_getl(&buffer, &buf_len, fp) != -1){
Packit 577717
Packit 577717
		/* skip  blank lines */
Packit 577717
		if (*buffer == '\n')
Packit 577717
			continue;
Packit 577717
Packit 577717
		p = strchr(buffer, ':');
Packit 577717
		if (p == NULL)
Packit 577717
			goto error;
Packit 577717
Packit 577717
		/*
Packit 577717
		 * p+2: +1 = space, +2= firt character
Packit 577717
		 * strlen()-1 gets rid of \n
Packit 577717
		 */
Packit 577717
		*p = '\0';
Packit 577717
		value = p+2;
Packit 577717
Packit 577717
		value[strlen(value)-1] = '\0';
Packit 577717
Packit 577717
		if (!strncmp(attr, buffer, attr_len))
Packit 577717
			break;
Packit 577717
	}
Packit 577717
	strncpy(ret_buf, value, maxlen-1);
Packit 577717
	ret_buf[maxlen-1] = '\0';
Packit 577717
	ret = 0;
Packit 577717
error:
Packit 577717
	free(buffer);
Packit 577717
	fclose(fp);
Packit 577717
	return ret;
Packit 577717
}
Packit 577717
#else
Packit 577717
static int
Packit 577717
pfmlib_getcpuinfo_attr(const char *attr, char *ret_buf, size_t maxlen)
Packit 577717
{
Packit 577717
	DPRINT("/proc/cpuinfo ignored\n");
Packit 577717
}
Packit 577717
#endif
Packit 577717
Packit 577717
static void
Packit 577717
pfm_mips_display_reg(pfm_mips_sel_reg_t reg, uint64_t cntrs, char *fstr)
Packit 577717
{
Packit 577717
	__pfm_vbprintf("[0x%"PRIx64" mask=0x%x usr=%d sys=%d sup=%d int=%d cntrs=0x%"PRIx64"] %s\n",
Packit 577717
			reg.val,
Packit 577717
			reg.perfsel64.sel_event_mask,
Packit 577717
			reg.perfsel64.sel_usr,
Packit 577717
			reg.perfsel64.sel_os,
Packit 577717
			reg.perfsel64.sel_sup,
Packit 577717
			reg.perfsel64.sel_exl,
Packit 577717
			cntrs,
Packit 577717
			fstr);
Packit 577717
}
Packit 577717
Packit 577717
int
Packit 577717
pfm_mips_detect(void *this)
Packit 577717
{
Packit 577717
Packit 577717
	int ret;
Packit 577717
	char buffer[1024];
Packit 577717
Packit 577717
	DPRINT("mips_detect\n");
Packit 577717
Packit 577717
	ret = pfmlib_getcpuinfo_attr("cpu model", buffer, sizeof(buffer));
Packit 577717
	if (ret == -1)
Packit 577717
		return PFM_ERR_NOTSUPP;
Packit 577717
Packit 577717
	if (strstr(buffer,"MIPS") == NULL)
Packit 577717
	  return PFM_ERR_NOTSUPP;
Packit 577717
Packit 577717
	strncpy(pfm_mips_cfg.model,buffer,strlen(buffer));
Packit 577717
Packit 577717
/*	ret = pfmlib_getcpuinfo_attr("CPU implementer", buffer, sizeof(buffer));
Packit 577717
	if (ret == -1)
Packit 577717
		return PFM_ERR_NOTSUPP;
Packit 577717
Packit 577717
	pfm_mips_cfg.implementer = strtol(buffer, NULL, 16);
Packit 577717
Packit 577717
Packit 577717
	ret = pfmlib_getcpuinfo_attr("CPU part", buffer, sizeof(buffer));
Packit 577717
	if (ret == -1)
Packit 577717
		return PFM_ERR_NOTSUPP;
Packit 577717
Packit 577717
	pfm_mips_cfg.part = strtol(buffer, NULL, 16);
Packit 577717
Packit 577717
	ret = pfmlib_getcpuinfo_attr("CPU architecture", buffer, sizeof(buffer));
Packit 577717
	if (ret == -1)
Packit 577717
		return PFM_ERR_NOTSUPP;
Packit 577717
Packit 577717
	pfm_mips_cfg.architecture = strtol(buffer, NULL, 16); */
Packit 577717
Packit 577717
	return PFM_SUCCESS;
Packit 577717
}
Packit 577717
Packit 577717
int
Packit 577717
pfm_mips_get_encoding(void *this, pfmlib_event_desc_t *e)
Packit 577717
{
Packit 577717
Packit 577717
	pfmlib_pmu_t *pmu = this;
Packit 577717
	const mips_entry_t *pe = this_pe(this);
Packit 577717
	pfmlib_event_attr_info_t *a;
Packit 577717
	pfm_mips_sel_reg_t reg;
Packit 577717
	uint64_t ival, cntmask = 0;
Packit 577717
	int plmmsk = 0, code;
Packit 577717
	int k, id;
Packit 577717
Packit 577717
	reg.val = 0;
Packit 577717
	code = pe[e->event].code;
Packit 577717
Packit 577717
	/* truncates bit 7 (counter info) */
Packit 577717
	reg.perfsel64.sel_event_mask = code;
Packit 577717
Packit 577717
	for (k = 0; k < e->nattrs; k++) {
Packit 577717
		a = attr(e, k);
Packit 577717
Packit 577717
		if (a->ctrl != PFM_ATTR_CTRL_PMU)
Packit 577717
			continue;
Packit 577717
Packit 577717
		ival = e->attrs[k].ival;
Packit 577717
		switch(a->idx) {
Packit 577717
		case MIPS_ATTR_K: /* os */
Packit 577717
			reg.perfsel64.sel_os = !!ival;
Packit 577717
			plmmsk |= _MIPS_ATTR_K;
Packit 577717
			break;
Packit 577717
		case MIPS_ATTR_U: /* user */
Packit 577717
			reg.perfsel64.sel_usr = !!ival;
Packit 577717
			plmmsk |= _MIPS_ATTR_U;
Packit 577717
			break;
Packit 577717
		case MIPS_ATTR_S: /* supervisor */
Packit 577717
			reg.perfsel64.sel_sup = !!ival;
Packit 577717
			plmmsk |= _MIPS_ATTR_S;
Packit 577717
			break;
Packit 577717
		case MIPS_ATTR_E: /* int */
Packit 577717
			reg.perfsel64.sel_exl = !!ival;
Packit 577717
			plmmsk |= _MIPS_ATTR_E;
Packit 577717
		}
Packit 577717
	}
Packit 577717
Packit 577717
	/*
Packit 577717
	 * handle case where no priv level mask was passed.
Packit 577717
	 * then we use the dfl_plm
Packit 577717
	 */
Packit 577717
	if (!(plmmsk & MIPS_PLM_ALL)) {
Packit 577717
		if (e->dfl_plm & PFM_PLM0)
Packit 577717
			reg.perfsel64.sel_os = 1;
Packit 577717
		if (e->dfl_plm & PFM_PLM1)
Packit 577717
			reg.perfsel64.sel_sup = 1;
Packit 577717
		if (e->dfl_plm & PFM_PLM2)
Packit 577717
			reg.perfsel64.sel_exl = 1;
Packit 577717
		if (e->dfl_plm & PFM_PLM3)
Packit 577717
			reg.perfsel64.sel_usr = 1;
Packit 577717
	}
Packit 577717
Packit 577717
        evt_strcat(e->fstr, "%s", pe[e->event].name);
Packit 577717
Packit 577717
	for (k = 0; k < e->npattrs; k++) {
Packit 577717
Packit 577717
		if (e->pattrs[k].ctrl != PFM_ATTR_CTRL_PMU)
Packit 577717
			continue;
Packit 577717
Packit 577717
		id = e->pattrs[k].idx;
Packit 577717
		switch(id) {
Packit 577717
		case MIPS_ATTR_K:
Packit 577717
			evt_strcat(e->fstr, ":%s=%lu", mips_mods[id].name, reg.perfsel64.sel_os);
Packit 577717
			break;
Packit 577717
		case MIPS_ATTR_U:
Packit 577717
			evt_strcat(e->fstr, ":%s=%lu", mips_mods[id].name, reg.perfsel64.sel_usr);
Packit 577717
			break;
Packit 577717
		case MIPS_ATTR_S:
Packit 577717
			evt_strcat(e->fstr, ":%s=%lu", mips_mods[id].name, reg.perfsel64.sel_sup);
Packit 577717
			break;
Packit 577717
		case MIPS_ATTR_E:
Packit 577717
			evt_strcat(e->fstr, ":%s=%lu", mips_mods[id].name, reg.perfsel64.sel_exl);
Packit 577717
			break;
Packit 577717
		}
Packit 577717
	}
Packit 577717
	e->codes[0] = reg.val;
Packit 577717
Packit 577717
	/* cycles and instructions support all counters */
Packit 577717
	if (code == 0 || code == 1) {
Packit 577717
		cntmask = (1ULL << pmu->num_cntrs) -1;
Packit 577717
	} else {
Packit 577717
		/* event work on odd counters only */
Packit 577717
		for (k = !!(code & 0x80) ; k < pmu->num_cntrs; k+=2) {
Packit 577717
			cntmask |= 1ULL << k;
Packit 577717
		}
Packit 577717
	}
Packit 577717
	e->codes[1] = cntmask;
Packit 577717
	e->count    = 2;
Packit 577717
Packit 577717
	pfm_mips_display_reg(reg, cntmask, e->fstr);
Packit 577717
Packit 577717
	return PFM_SUCCESS;
Packit 577717
}
Packit 577717
Packit 577717
int
Packit 577717
pfm_mips_get_event_first(void *this)
Packit 577717
{
Packit 577717
	return 0;
Packit 577717
}
Packit 577717
Packit 577717
int
Packit 577717
pfm_mips_get_event_next(void *this, int idx)
Packit 577717
{
Packit 577717
	pfmlib_pmu_t *p = this;
Packit 577717
Packit 577717
	if (idx >= (p->pme_count-1))
Packit 577717
		return -1;
Packit 577717
Packit 577717
	return idx+1;
Packit 577717
}
Packit 577717
Packit 577717
int
Packit 577717
pfm_mips_event_is_valid(void *this, int pidx)
Packit 577717
{
Packit 577717
	pfmlib_pmu_t *p = this;
Packit 577717
	return pidx >= 0 && pidx < p->pme_count;
Packit 577717
}
Packit 577717
Packit 577717
int
Packit 577717
pfm_mips_validate_table(void *this, FILE *fp)
Packit 577717
{
Packit 577717
Packit 577717
	pfmlib_pmu_t *pmu = this;
Packit 577717
	const mips_entry_t *pe = this_pe(this);
Packit 577717
	int i, j, error = 0;
Packit 577717
Packit 577717
	for(i=0; i < pmu->pme_count; i++) {
Packit 577717
		if (!pe[i].name) {
Packit 577717
			fprintf(fp, "pmu: %s event%d: :: no name (prev event was %s)\n", pmu->name, i,
Packit 577717
			i > 1 ? pe[i-1].name : "??");
Packit 577717
			error++;
Packit 577717
		}
Packit 577717
		if (!pe[i].desc) {
Packit 577717
			fprintf(fp, "pmu: %s event%d: %s :: no description\n", pmu->name, i, pe[i].name);
Packit 577717
			error++;
Packit 577717
		}
Packit 577717
		for (j=i+1; j < pmu->pme_count; j++) {
Packit 577717
			if (pe[i].code == pe[j].code) {
Packit 577717
				fprintf(fp, "pmu: %s events %s and %s have the same code 0x%x\n", pmu->name, pe[i].name, pe[j].name, pe[i].code);
Packit 577717
				error++;
Packit 577717
			}
Packit 577717
		}
Packit 577717
	}
Packit 577717
	if (!pmu->supported_plm) {
Packit 577717
		fprintf(fp, "pmu: %s supported_plm=0, is that right?\n", pmu->name);
Packit 577717
		error++;
Packit 577717
	}
Packit 577717
	return error ? PFM_ERR_INVAL : PFM_SUCCESS;
Packit 577717
}
Packit 577717
Packit 577717
unsigned int
Packit 577717
pfm_mips_get_event_nattrs(void *this, int pidx)
Packit 577717
{
Packit 577717
	/* assume all pmus have the same number of attributes */
Packit 577717
	return MIPS_NUM_ATTRS;
Packit 577717
}
Packit 577717
Packit 577717
int
Packit 577717
pfm_mips_get_event_attr_info(void *this, int pidx, int attr_idx, pfmlib_event_attr_info_t *info)
Packit 577717
{
Packit 577717
	/* no umasks, so all attrs are modifiers */
Packit 577717
Packit 577717
	info->name = mips_mods[attr_idx].name;
Packit 577717
	info->desc = mips_mods[attr_idx].desc;
Packit 577717
	info->type = mips_mods[attr_idx].type;
Packit 577717
	info->type = mips_mods[attr_idx].type;
Packit 577717
	info->equiv= NULL;
Packit 577717
	info->idx   = attr_idx; /* private index */
Packit 577717
	info->code = attr_idx;
Packit 577717
	info->is_dfl = 0;
Packit 577717
	info->is_precise = 0;
Packit 577717
	info->ctrl = PFM_ATTR_CTRL_PMU;;
Packit 577717
Packit 577717
	return PFM_SUCCESS;
Packit 577717
}
Packit 577717
Packit 577717
int
Packit 577717
pfm_mips_get_event_info(void *this, int idx, pfm_event_info_t *info)
Packit 577717
{
Packit 577717
	pfmlib_pmu_t *pmu = this;
Packit 577717
	const mips_entry_t *pe = this_pe(this);
Packit 577717
Packit 577717
	info->name  = pe[idx].name;
Packit 577717
	info->desc  = pe[idx].desc;
Packit 577717
	info->code  = pe[idx].code;
Packit 577717
	info->equiv = NULL;
Packit 577717
	info->idx   = idx; /* private index */
Packit 577717
	info->pmu   = pmu->pmu;
Packit 577717
	info->is_precise = 0;
Packit 577717
Packit 577717
	/* no attributes defined for MIPS yet */
Packit 577717
	info->nattrs = pfm_mips_get_event_nattrs(this, idx);
Packit 577717
Packit 577717
	return PFM_SUCCESS;
Packit 577717
}