|
Packit |
577717 |
/*
|
|
Packit |
577717 |
* pfmlib_arm.c : support for ARM chips
|
|
Packit |
577717 |
*
|
|
Packit |
577717 |
* Copyright (c) 2010 University of Tennessee
|
|
Packit |
577717 |
* Contributed by Vince Weaver <vweaver1@utk.edu>
|
|
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_arm_priv.h"
|
|
Packit |
577717 |
|
|
Packit |
577717 |
const pfmlib_attr_desc_t arm_mods[]={
|
|
Packit |
577717 |
PFM_ATTR_B("k", "monitor at kernel level"),
|
|
Packit |
577717 |
PFM_ATTR_B("u", "monitor at user level"),
|
|
Packit |
577717 |
PFM_ATTR_B("hv", "monitor in hypervisor"),
|
|
Packit |
577717 |
PFM_ATTR_NULL /* end-marker to avoid exporting number of entries */
|
|
Packit |
577717 |
};
|
|
Packit |
577717 |
|
|
Packit |
577717 |
pfm_arm_config_t pfm_arm_cfg;
|
|
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 |
return -1;
|
|
Packit |
577717 |
}
|
|
Packit |
577717 |
#endif
|
|
Packit |
577717 |
|
|
Packit |
577717 |
static int
|
|
Packit |
577717 |
arm_num_mods(void *this, int idx)
|
|
Packit |
577717 |
{
|
|
Packit |
577717 |
const arm_entry_t *pe = this_pe(this);
|
|
Packit |
577717 |
unsigned int mask;
|
|
Packit |
577717 |
|
|
Packit |
577717 |
mask = pe[idx].modmsk;
|
|
Packit |
577717 |
return pfmlib_popcnt(mask);
|
|
Packit |
577717 |
}
|
|
Packit |
577717 |
|
|
Packit |
577717 |
static inline int
|
|
Packit |
577717 |
arm_attr2mod(void *this, int pidx, int attr_idx)
|
|
Packit |
577717 |
{
|
|
Packit |
577717 |
const arm_entry_t *pe = this_pe(this);
|
|
Packit |
577717 |
size_t x;
|
|
Packit |
577717 |
int n;
|
|
Packit |
577717 |
|
|
Packit |
577717 |
n = attr_idx;
|
|
Packit |
577717 |
|
|
Packit |
577717 |
pfmlib_for_each_bit(x, pe[pidx].modmsk) {
|
|
Packit |
577717 |
if (n == 0)
|
|
Packit |
577717 |
break;
|
|
Packit |
577717 |
n--;
|
|
Packit |
577717 |
}
|
|
Packit |
577717 |
return x;
|
|
Packit |
577717 |
}
|
|
Packit |
577717 |
|
|
Packit |
577717 |
static void
|
|
Packit |
577717 |
pfm_arm_display_reg(void *this, pfmlib_event_desc_t *e, pfm_arm_reg_t reg)
|
|
Packit |
577717 |
{
|
|
Packit |
577717 |
__pfm_vbprintf("[0x%x] %s\n", reg.val, e->fstr);
|
|
Packit |
577717 |
}
|
|
Packit |
577717 |
|
|
Packit |
577717 |
int
|
|
Packit |
577717 |
pfm_arm_detect(void *this)
|
|
Packit |
577717 |
{
|
|
Packit |
577717 |
|
|
Packit |
577717 |
int ret;
|
|
Packit |
577717 |
char buffer[128];
|
|
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_arm_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_arm_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_arm_cfg.architecture = strtol(buffer, NULL, 16);
|
|
Packit |
577717 |
|
|
Packit |
577717 |
return PFM_SUCCESS;
|
|
Packit |
577717 |
}
|
|
Packit |
577717 |
|
|
Packit |
577717 |
int
|
|
Packit |
577717 |
pfm_arm_get_encoding(void *this, pfmlib_event_desc_t *e)
|
|
Packit |
577717 |
{
|
|
Packit |
577717 |
|
|
Packit |
577717 |
const arm_entry_t *pe = this_pe(this);
|
|
Packit |
577717 |
pfmlib_event_attr_info_t *a;
|
|
Packit |
577717 |
pfm_arm_reg_t reg;
|
|
Packit |
577717 |
unsigned int plm = 0;
|
|
Packit |
577717 |
int i, idx, has_plm = 0;
|
|
Packit |
577717 |
|
|
Packit |
577717 |
reg.val = pe[e->event].code;
|
|
Packit |
577717 |
|
|
Packit |
577717 |
|
|
Packit |
577717 |
for (i = 0; i < e->nattrs; i++) {
|
|
Packit |
577717 |
a = attr(e, i);
|
|
Packit |
577717 |
|
|
Packit |
577717 |
if (a->ctrl != PFM_ATTR_CTRL_PMU)
|
|
Packit |
577717 |
continue;
|
|
Packit |
577717 |
|
|
Packit |
577717 |
if (a->type > PFM_ATTR_UMASK) {
|
|
Packit |
577717 |
uint64_t ival = e->attrs[i].ival;
|
|
Packit |
577717 |
|
|
Packit |
577717 |
switch(a->idx) {
|
|
Packit |
577717 |
case ARM_ATTR_U: /* USR */
|
|
Packit |
577717 |
if (ival)
|
|
Packit |
577717 |
plm |= PFM_PLM3;
|
|
Packit |
577717 |
has_plm = 1;
|
|
Packit |
577717 |
break;
|
|
Packit |
577717 |
case ARM_ATTR_K: /* OS */
|
|
Packit |
577717 |
if (ival)
|
|
Packit |
577717 |
plm |= PFM_PLM0;
|
|
Packit |
577717 |
has_plm = 1;
|
|
Packit |
577717 |
break;
|
|
Packit |
577717 |
case ARM_ATTR_HV: /* HYPERVISOR */
|
|
Packit |
577717 |
if (ival)
|
|
Packit |
577717 |
plm |= PFM_PLMH;
|
|
Packit |
577717 |
has_plm = 1;
|
|
Packit |
577717 |
break;
|
|
Packit |
577717 |
default:
|
|
Packit |
577717 |
return PFM_ERR_ATTR;
|
|
Packit |
577717 |
}
|
|
Packit |
577717 |
}
|
|
Packit |
577717 |
}
|
|
Packit |
577717 |
|
|
Packit |
577717 |
if (arm_has_plm(this, e)) {
|
|
Packit |
577717 |
if (!has_plm)
|
|
Packit |
577717 |
plm = e->dfl_plm;
|
|
Packit |
577717 |
reg.evtsel.excl_pl1 = !(plm & PFM_PLM0);
|
|
Packit |
577717 |
reg.evtsel.excl_usr = !(plm & PFM_PLM3);
|
|
Packit |
577717 |
reg.evtsel.excl_hyp = !(plm & PFM_PLMH);
|
|
Packit |
577717 |
}
|
|
Packit |
577717 |
|
|
Packit |
577717 |
evt_strcat(e->fstr, "%s", pe[e->event].name);
|
|
Packit |
577717 |
|
|
Packit |
577717 |
e->codes[0] = reg.val;
|
|
Packit |
577717 |
e->count = 1;
|
|
Packit |
577717 |
|
|
Packit |
577717 |
for (i = 0; i < e->npattrs; i++) {
|
|
Packit |
577717 |
if (e->pattrs[i].ctrl != PFM_ATTR_CTRL_PMU)
|
|
Packit |
577717 |
continue;
|
|
Packit |
577717 |
|
|
Packit |
577717 |
if (e->pattrs[i].type == PFM_ATTR_UMASK)
|
|
Packit |
577717 |
continue;
|
|
Packit |
577717 |
|
|
Packit |
577717 |
idx = e->pattrs[i].idx;
|
|
Packit |
577717 |
switch(idx) {
|
|
Packit |
577717 |
case ARM_ATTR_K:
|
|
Packit |
577717 |
evt_strcat(e->fstr, ":%s=%lu", arm_mods[idx].name, !reg.evtsel.excl_pl1);
|
|
Packit |
577717 |
break;
|
|
Packit |
577717 |
case ARM_ATTR_U:
|
|
Packit |
577717 |
evt_strcat(e->fstr, ":%s=%lu", arm_mods[idx].name, !reg.evtsel.excl_usr);
|
|
Packit |
577717 |
break;
|
|
Packit |
577717 |
case ARM_ATTR_HV:
|
|
Packit |
577717 |
evt_strcat(e->fstr, ":%s=%lu", arm_mods[idx].name, !reg.evtsel.excl_hyp);
|
|
Packit |
577717 |
break;
|
|
Packit |
577717 |
}
|
|
Packit |
577717 |
}
|
|
Packit |
577717 |
|
|
Packit |
577717 |
pfm_arm_display_reg(this, e, reg);
|
|
Packit |
577717 |
|
|
Packit |
577717 |
return PFM_SUCCESS;
|
|
Packit |
577717 |
}
|
|
Packit |
577717 |
|
|
Packit |
577717 |
int
|
|
Packit |
577717 |
pfm_arm_get_event_first(void *this)
|
|
Packit |
577717 |
{
|
|
Packit |
577717 |
return 0;
|
|
Packit |
577717 |
}
|
|
Packit |
577717 |
|
|
Packit |
577717 |
int
|
|
Packit |
577717 |
pfm_arm_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_arm_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_arm_validate_table(void *this, FILE *fp)
|
|
Packit |
577717 |
{
|
|
Packit |
577717 |
|
|
Packit |
577717 |
pfmlib_pmu_t *pmu = this;
|
|
Packit |
577717 |
const arm_entry_t *pe = this_pe(this);
|
|
Packit |
577717 |
int i, 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 |
}
|
|
Packit |
577717 |
return error ? PFM_ERR_INVAL : PFM_SUCCESS;
|
|
Packit |
577717 |
}
|
|
Packit |
577717 |
|
|
Packit |
577717 |
int
|
|
Packit |
577717 |
pfm_arm_get_event_attr_info(void *this, int pidx, int attr_idx, pfmlib_event_attr_info_t *info)
|
|
Packit |
577717 |
{
|
|
Packit |
577717 |
int idx;
|
|
Packit |
577717 |
|
|
Packit |
577717 |
idx = arm_attr2mod(this, pidx, attr_idx);
|
|
Packit |
577717 |
info->name = arm_mods[idx].name;
|
|
Packit |
577717 |
info->desc = arm_mods[idx].desc;
|
|
Packit |
577717 |
info->type = arm_mods[idx].type;
|
|
Packit |
577717 |
info->code = idx;
|
|
Packit |
577717 |
|
|
Packit |
577717 |
info->is_dfl = 0;
|
|
Packit |
577717 |
info->equiv = NULL;
|
|
Packit |
577717 |
info->ctrl = PFM_ATTR_CTRL_PMU;
|
|
Packit |
577717 |
info->idx = idx; /* namespace specific index */
|
|
Packit |
577717 |
|
|
Packit |
577717 |
info->dfl_val64 = 0;
|
|
Packit |
577717 |
info->is_precise = 0;
|
|
Packit |
577717 |
|
|
Packit |
577717 |
return PFM_SUCCESS;
|
|
Packit |
577717 |
}
|
|
Packit |
577717 |
|
|
Packit |
577717 |
unsigned int
|
|
Packit |
577717 |
pfm_arm_get_event_nattrs(void *this, int pidx)
|
|
Packit |
577717 |
{
|
|
Packit |
577717 |
return arm_num_mods(this, pidx);
|
|
Packit |
577717 |
}
|
|
Packit |
577717 |
|
|
Packit |
577717 |
int
|
|
Packit |
577717 |
pfm_arm_get_event_info(void *this, int idx, pfm_event_info_t *info)
|
|
Packit |
577717 |
{
|
|
Packit |
577717 |
pfmlib_pmu_t *pmu = this;
|
|
Packit |
577717 |
const arm_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 ARM yet */
|
|
Packit |
577717 |
info->nattrs = 0;
|
|
Packit |
577717 |
|
|
Packit |
577717 |
return PFM_SUCCESS;
|
|
Packit |
577717 |
}
|